From bd548a3e296c3ae4d0365ba70971c16d5119a11e Mon Sep 17 00:00:00 2001 From: Lucas Roesler Date: Sun, 30 Jan 2022 13:08:49 +0100 Subject: [PATCH 1/5] feat: OPA based inline auth middleware Create a middleware that allows executing OPA policies prior to invoking the function. The policy can be configurd via the `OPA_POLICY` environment variable. The query is defined via the `OPA_QUERY` variable. The OPA_POLICY can be a comma separated list of paths. The default query input will contain `path`, `method`, and the `authorization` header value. You can optional include the parsed `body` `rawBody` and the raw `headers` via the `OPA_INCLUDE_JSON_BODY`, `OPA_INCLUDE_RAW_BODY` and `OPA_INCLUDE_HEADERS` environment variables. Signed-off-by: Lucas Roesler --- .vscode/settings.json | 3 + README.md | 182 +- auth/local.go | 99 + auth/local_test.go | 164 + auth/middleware.go | 206 + auth/testdata/allow_all.rego | 3 + auth/testdata/basic_auth.rego | 26 + auth/testdata/policy.rego | 24 + go.mod | 16 + go.sum | 441 +- main.go | 21 + vendor/github.com/OneOfOne/xxhash/.gitignore | 4 + vendor/github.com/OneOfOne/xxhash/.travis.yml | 13 + vendor/github.com/OneOfOne/xxhash/LICENSE | 187 + vendor/github.com/OneOfOne/xxhash/README.md | 74 + vendor/github.com/OneOfOne/xxhash/xxhash.go | 294 ++ .../github.com/OneOfOne/xxhash/xxhash_go17.go | 161 + .../github.com/OneOfOne/xxhash/xxhash_safe.go | 183 + .../OneOfOne/xxhash/xxhash_unsafe.go | 240 + vendor/github.com/ghodss/yaml/.gitignore | 20 + vendor/github.com/ghodss/yaml/.travis.yml | 7 + vendor/github.com/ghodss/yaml/LICENSE | 50 + vendor/github.com/ghodss/yaml/README.md | 121 + vendor/github.com/ghodss/yaml/fields.go | 501 ++ vendor/github.com/ghodss/yaml/yaml.go | 277 ++ vendor/github.com/gobwas/glob/.gitignore | 8 + vendor/github.com/gobwas/glob/.travis.yml | 9 + vendor/github.com/gobwas/glob/LICENSE | 21 + vendor/github.com/gobwas/glob/bench.sh | 26 + .../gobwas/glob/compiler/compiler.go | 525 ++ vendor/github.com/gobwas/glob/glob.go | 80 + vendor/github.com/gobwas/glob/match/any.go | 45 + vendor/github.com/gobwas/glob/match/any_of.go | 82 + vendor/github.com/gobwas/glob/match/btree.go | 146 + .../github.com/gobwas/glob/match/contains.go | 58 + .../github.com/gobwas/glob/match/every_of.go | 99 + vendor/github.com/gobwas/glob/match/list.go | 49 + vendor/github.com/gobwas/glob/match/match.go | 81 + vendor/github.com/gobwas/glob/match/max.go | 49 + vendor/github.com/gobwas/glob/match/min.go | 57 + .../github.com/gobwas/glob/match/nothing.go | 27 + vendor/github.com/gobwas/glob/match/prefix.go | 50 + .../gobwas/glob/match/prefix_any.go | 55 + .../gobwas/glob/match/prefix_suffix.go | 62 + vendor/github.com/gobwas/glob/match/range.go | 48 + vendor/github.com/gobwas/glob/match/row.go | 77 + .../github.com/gobwas/glob/match/segments.go | 91 + vendor/github.com/gobwas/glob/match/single.go | 43 + vendor/github.com/gobwas/glob/match/suffix.go | 35 + .../gobwas/glob/match/suffix_any.go | 43 + vendor/github.com/gobwas/glob/match/super.go | 33 + vendor/github.com/gobwas/glob/match/text.go | 45 + vendor/github.com/gobwas/glob/readme.md | 148 + .../github.com/gobwas/glob/syntax/ast/ast.go | 122 + .../gobwas/glob/syntax/ast/parser.go | 157 + .../gobwas/glob/syntax/lexer/lexer.go | 273 + .../gobwas/glob/syntax/lexer/token.go | 88 + .../github.com/gobwas/glob/syntax/syntax.go | 14 + .../gobwas/glob/util/runes/runes.go | 154 + .../gobwas/glob/util/strings/strings.go | 39 + .../github.com/open-policy-agent/opa/LICENSE | 202 + .../open-policy-agent/opa/ast/builtins.go | 2523 ++++++++++ .../open-policy-agent/opa/ast/capabilities.go | 65 + .../open-policy-agent/opa/ast/check.go | 1372 +++++ .../open-policy-agent/opa/ast/compare.go | 386 ++ .../open-policy-agent/opa/ast/compile.go | 4411 +++++++++++++++++ .../opa/ast/compilehelper.go | 59 + .../opa/ast/compilemetrics.go | 9 + .../open-policy-agent/opa/ast/conflicts.go | 48 + .../open-policy-agent/opa/ast/doc.go | 36 + .../open-policy-agent/opa/ast/env.go | 327 ++ .../open-policy-agent/opa/ast/errors.go | 132 + .../open-policy-agent/opa/ast/fuzz.go | 16 + .../open-policy-agent/opa/ast/index.go | 884 ++++ .../opa/ast/internal/scanner/scanner.go | 411 ++ .../opa/ast/internal/tokens/tokens.go | 138 + .../opa/ast/location/location.go | 90 + .../open-policy-agent/opa/ast/map.go | 133 + .../open-policy-agent/opa/ast/parser.go | 2040 ++++++++ .../open-policy-agent/opa/ast/parser_ext.go | 768 +++ .../open-policy-agent/opa/ast/policy.go | 1622 ++++++ .../open-policy-agent/opa/ast/pretty.go | 82 + .../open-policy-agent/opa/ast/schema.go | 63 + .../open-policy-agent/opa/ast/strings.go | 15 + .../open-policy-agent/opa/ast/term.go | 2875 +++++++++++ .../open-policy-agent/opa/ast/transform.go | 396 ++ .../open-policy-agent/opa/ast/unify.go | 235 + .../open-policy-agent/opa/ast/varset.go | 100 + .../open-policy-agent/opa/ast/visit.go | 695 +++ .../open-policy-agent/opa/bundle/bundle.go | 1163 +++++ .../open-policy-agent/opa/bundle/file.go | 268 + .../open-policy-agent/opa/bundle/filefs.go | 74 + .../open-policy-agent/opa/bundle/hash.go | 141 + .../open-policy-agent/opa/bundle/keys.go | 145 + .../open-policy-agent/opa/bundle/sign.go | 135 + .../open-policy-agent/opa/bundle/store.go | 644 +++ .../open-policy-agent/opa/bundle/verify.go | 233 + .../open-policy-agent/opa/format/format.go | 1201 +++++ .../opa/internal/bundle/utils.go | 85 + .../opa/internal/cidr/merge/merge.go | 367 ++ .../internal/compiler/wasm/opa/callgraph.csv | 2401 +++++++++ .../opa/internal/compiler/wasm/opa/opa.go | 51 + .../opa/internal/compiler/wasm/opa/opa.wasm | Bin 0 -> 421950 bytes .../internal/compiler/wasm/optimizations.go | 274 + .../opa/internal/compiler/wasm/wasm.go | 1783 +++++++ .../opa/internal/debug/debug.go | 36 + .../opa/internal/deepcopy/deepcopy.go | 27 + .../opa/internal/file/archive/tarball.go | 42 + .../opa/internal/file/url/url.go | 42 + .../opa/internal/future/filter_imports.go | 20 + .../opa/internal/future/parser_opts.go | 42 + .../gojsonschema/LICENSE-APACHE-2.0.txt | 202 + .../opa/internal/gojsonschema/README.md | 481 ++ .../opa/internal/gojsonschema/draft.go | 122 + .../opa/internal/gojsonschema/errors.go | 366 ++ .../internal/gojsonschema/format_checkers.go | 368 ++ .../opa/internal/gojsonschema/internalLog.go | 37 + .../opa/internal/gojsonschema/jsonContext.go | 73 + .../opa/internal/gojsonschema/jsonLoader.go | 405 ++ .../opa/internal/gojsonschema/locales.go | 472 ++ .../opa/internal/gojsonschema/result.go | 220 + .../opa/internal/gojsonschema/schema.go | 963 ++++ .../opa/internal/gojsonschema/schemaLoader.go | 206 + .../opa/internal/gojsonschema/schemaPool.go | 230 + .../gojsonschema/schemaReferencePool.go | 68 + .../opa/internal/gojsonschema/schemaType.go | 83 + .../opa/internal/gojsonschema/subSchema.go | 151 + .../opa/internal/gojsonschema/types.go | 62 + .../opa/internal/gojsonschema/utils.go | 165 + .../opa/internal/gojsonschema/validation.go | 837 ++++ .../open-policy-agent/opa/internal/ir/ir.go | 497 ++ .../opa/internal/ir/pretty.go | 44 + .../open-policy-agent/opa/internal/ir/walk.go | 93 + .../opa/internal/jwx/LICENSE | 21 + .../opa/internal/jwx/buffer/buffer.go | 113 + .../opa/internal/jwx/jwa/elliptic.go | 11 + .../opa/internal/jwx/jwa/key_type.go | 67 + .../opa/internal/jwx/jwa/parameters.go | 29 + .../opa/internal/jwx/jwa/signature.go | 76 + .../opa/internal/jwx/jwk/ecdsa.go | 120 + .../opa/internal/jwx/jwk/headers.go | 178 + .../opa/internal/jwx/jwk/interface.go | 71 + .../opa/internal/jwx/jwk/jwk.go | 150 + .../opa/internal/jwx/jwk/key_ops.go | 68 + .../opa/internal/jwx/jwk/rsa.go | 133 + .../opa/internal/jwx/jwk/symmetric.go | 41 + .../opa/internal/jwx/jws/headers.go | 154 + .../opa/internal/jwx/jws/interface.go | 22 + .../opa/internal/jwx/jws/jws.go | 221 + .../opa/internal/jwx/jws/message.go | 26 + .../opa/internal/jwx/jws/sign/ecdsa.go | 90 + .../opa/internal/jwx/jws/sign/hmac.go | 66 + .../opa/internal/jwx/jws/sign/interface.go | 46 + .../opa/internal/jwx/jws/sign/rsa.go | 97 + .../opa/internal/jwx/jws/sign/sign.go | 67 + .../opa/internal/jwx/jws/verify/ecdsa.go | 67 + .../opa/internal/jwx/jws/verify/hmac.go | 33 + .../opa/internal/jwx/jws/verify/interface.go | 39 + .../opa/internal/jwx/jws/verify/rsa.go | 88 + .../opa/internal/jwx/jws/verify/verify.go | 57 + .../opa/internal/lcss/README.md | 3 + .../opa/internal/lcss/lcss.go | 197 + .../opa/internal/lcss/qsufsort.go | 169 + .../opa/internal/leb128/leb128.go | 170 + .../opa/internal/merge/merge.go | 64 + .../opa/internal/planner/planner.go | 2117 ++++++++ .../opa/internal/planner/rules.go | 168 + .../opa/internal/planner/varstack.go | 58 + .../opa/internal/rego/opa/engine.go | 65 + .../opa/internal/rego/opa/options.go | 28 + .../opa/internal/semver/LICENSE | 202 + .../opa/internal/semver/semver.go | 235 + .../opa/internal/strings/strings.go | 74 + .../opa/internal/uuid/uuid.go | 22 + .../opa/internal/version/version.go | 36 + .../opa/internal/wasm/constant/constant.go | 77 + .../opa/internal/wasm/encoding/doc.go | 6 + .../opa/internal/wasm/encoding/reader.go | 967 ++++ .../opa/internal/wasm/encoding/writer.go | 777 +++ .../opa/internal/wasm/instruction/control.go | 183 + .../internal/wasm/instruction/instruction.go | 33 + .../opa/internal/wasm/instruction/memory.go | 39 + .../opa/internal/wasm/instruction/numeric.go | 199 + .../internal/wasm/instruction/parametric.go | 29 + .../opa/internal/wasm/instruction/variable.go | 54 + .../opa/internal/wasm/module/module.go | 385 ++ .../opa/internal/wasm/module/pretty.go | 84 + .../opa/internal/wasm/opcode/opcode.go | 218 + .../wasm/sdk/opa/capabilities/capabilities.go | 12 + .../opa/capabilities/capabilities_nowasm.go | 14 + .../opa/internal/wasm/types/types.go | 36 + .../opa/internal/wasm/util/util.go | 18 + .../open-policy-agent/opa/keys/keys.go | 100 + .../open-policy-agent/opa/loader/errors.go | 62 + .../open-policy-agent/opa/loader/loader.go | 670 +++ .../open-policy-agent/opa/metrics/metrics.go | 312 ++ .../open-policy-agent/opa/rego/errors.go | 24 + .../open-policy-agent/opa/rego/rego.go | 2534 ++++++++++ .../open-policy-agent/opa/rego/resultset.go | 90 + .../opa/resolver/interface.go | 29 + .../opa/resolver/wasm/wasm.go | 173 + .../open-policy-agent/opa/storage/doc.go | 6 + .../open-policy-agent/opa/storage/errors.go | 122 + .../opa/storage/inmem/inmem.go | 250 + .../opa/storage/inmem/txn.go | 392 ++ .../opa/storage/interface.go | 194 + .../opa/storage/internal/errors/errors.go | 32 + .../opa/storage/internal/ptr/ptr.go | 48 + .../open-policy-agent/opa/storage/path.go | 154 + .../open-policy-agent/opa/storage/storage.go | 126 + .../opa/topdown/aggregates.go | 260 + .../opa/topdown/arithmetic.go | 184 + .../open-policy-agent/opa/topdown/array.go | 94 + .../open-policy-agent/opa/topdown/binary.go | 45 + .../open-policy-agent/opa/topdown/bindings.go | 400 ++ .../open-policy-agent/opa/topdown/bits.go | 88 + .../open-policy-agent/opa/topdown/builtins.go | 205 + .../opa/topdown/builtins/builtins.go | 237 + .../open-policy-agent/opa/topdown/cache.go | 237 + .../opa/topdown/cache/cache.go | 167 + .../open-policy-agent/opa/topdown/cancel.go | 33 + .../open-policy-agent/opa/topdown/casts.go | 117 + .../open-policy-agent/opa/topdown/cidr.go | 405 ++ .../opa/topdown/comparison.go | 48 + .../copypropagation/copypropagation.go | 439 ++ .../opa/topdown/copypropagation/unionfind.go | 135 + .../open-policy-agent/opa/topdown/crypto.go | 419 ++ .../open-policy-agent/opa/topdown/doc.go | 10 + .../open-policy-agent/opa/topdown/encoding.go | 309 ++ .../open-policy-agent/opa/topdown/errors.go | 139 + .../open-policy-agent/opa/topdown/eval.go | 3163 ++++++++++++ .../open-policy-agent/opa/topdown/glob.go | 65 + .../open-policy-agent/opa/topdown/http.go | 1367 +++++ .../open-policy-agent/opa/topdown/input.go | 100 + .../opa/topdown/instrumentation.go | 63 + .../open-policy-agent/opa/topdown/json.go | 623 +++ .../open-policy-agent/opa/topdown/net.go | 63 + .../open-policy-agent/opa/topdown/numbers.go | 99 + .../open-policy-agent/opa/topdown/object.go | 140 + .../open-policy-agent/opa/topdown/parse.go | 47 + .../opa/topdown/parse_bytes.go | 143 + .../open-policy-agent/opa/topdown/print.go | 86 + .../opa/topdown/print/print.go | 21 + .../open-policy-agent/opa/topdown/query.go | 482 ++ .../opa/topdown/reachable.go | 61 + .../open-policy-agent/opa/topdown/regex.go | 218 + .../opa/topdown/regex_template.go | 122 + .../open-policy-agent/opa/topdown/resolver.go | 107 + .../open-policy-agent/opa/topdown/runtime.go | 20 + .../open-policy-agent/opa/topdown/save.go | 427 ++ .../open-policy-agent/opa/topdown/semver.go | 60 + .../open-policy-agent/opa/topdown/sets.go | 84 + .../open-policy-agent/opa/topdown/strings.go | 455 ++ .../open-policy-agent/opa/topdown/time.go | 297 ++ .../open-policy-agent/opa/topdown/tokens.go | 1113 +++++ .../open-policy-agent/opa/topdown/trace.go | 423 ++ .../open-policy-agent/opa/topdown/type.go | 82 + .../opa/topdown/type_name.go | 36 + .../open-policy-agent/opa/topdown/uuid.go | 36 + .../open-policy-agent/opa/topdown/walk.go | 96 + .../open-policy-agent/opa/tracing/tracing.go | 55 + .../open-policy-agent/opa/types/decode.go | 191 + .../open-policy-agent/opa/types/types.go | 995 ++++ .../open-policy-agent/opa/util/backoff.go | 42 + .../open-policy-agent/opa/util/close.go | 23 + .../open-policy-agent/opa/util/compare.go | 161 + .../open-policy-agent/opa/util/doc.go | 6 + .../open-policy-agent/opa/util/enumflag.go | 59 + .../open-policy-agent/opa/util/graph.go | 90 + .../open-policy-agent/opa/util/hashmap.go | 157 + .../open-policy-agent/opa/util/json.go | 113 + .../open-policy-agent/opa/util/queue.go | 113 + .../open-policy-agent/opa/util/wait.go | 34 + .../open-policy-agent/opa/version/version.go | 24 + .../open-policy-agent/opa/version/wasm.go | 13 + vendor/github.com/pkg/errors/.gitignore | 24 + vendor/github.com/pkg/errors/.travis.yml | 10 + vendor/github.com/pkg/errors/LICENSE | 23 + vendor/github.com/pkg/errors/Makefile | 44 + vendor/github.com/pkg/errors/README.md | 59 + vendor/github.com/pkg/errors/appveyor.yml | 32 + vendor/github.com/pkg/errors/errors.go | 288 ++ vendor/github.com/pkg/errors/go113.go | 38 + vendor/github.com/pkg/errors/stack.go | 177 + vendor/github.com/pmezard/go-difflib/LICENSE | 27 + .../pmezard/go-difflib/difflib/difflib.go | 772 +++ .../github.com/rcrowley/go-metrics/.gitignore | 9 + .../rcrowley/go-metrics/.travis.yml | 22 + vendor/github.com/rcrowley/go-metrics/LICENSE | 29 + .../github.com/rcrowley/go-metrics/README.md | 171 + .../github.com/rcrowley/go-metrics/counter.go | 112 + .../github.com/rcrowley/go-metrics/debug.go | 80 + vendor/github.com/rcrowley/go-metrics/ewma.go | 138 + .../github.com/rcrowley/go-metrics/gauge.go | 120 + .../rcrowley/go-metrics/gauge_float64.go | 125 + .../rcrowley/go-metrics/graphite.go | 113 + .../rcrowley/go-metrics/healthcheck.go | 61 + .../rcrowley/go-metrics/histogram.go | 202 + vendor/github.com/rcrowley/go-metrics/json.go | 31 + vendor/github.com/rcrowley/go-metrics/log.go | 100 + .../github.com/rcrowley/go-metrics/memory.md | 285 ++ .../github.com/rcrowley/go-metrics/meter.go | 251 + .../github.com/rcrowley/go-metrics/metrics.go | 13 + .../rcrowley/go-metrics/opentsdb.go | 119 + .../rcrowley/go-metrics/registry.go | 373 ++ .../github.com/rcrowley/go-metrics/runtime.go | 216 + .../rcrowley/go-metrics/runtime_cgo.go | 10 + .../go-metrics/runtime_gccpufraction.go | 9 + .../rcrowley/go-metrics/runtime_no_cgo.go | 7 + .../go-metrics/runtime_no_gccpufraction.go | 9 + .../github.com/rcrowley/go-metrics/sample.go | 616 +++ .../github.com/rcrowley/go-metrics/syslog.go | 78 + .../github.com/rcrowley/go-metrics/timer.go | 329 ++ .../rcrowley/go-metrics/validate.sh | 10 + .../github.com/rcrowley/go-metrics/writer.go | 100 + vendor/github.com/stretchr/testify/LICENSE | 21 + .../testify/assert/assertion_compare.go | 394 ++ .../testify/assert/assertion_format.go | 741 +++ .../testify/assert/assertion_format.go.tmpl | 5 + .../testify/assert/assertion_forward.go | 1470 ++++++ .../testify/assert/assertion_forward.go.tmpl | 5 + .../testify/assert/assertion_order.go | 81 + .../stretchr/testify/assert/assertions.go | 1774 +++++++ .../github.com/stretchr/testify/assert/doc.go | 45 + .../stretchr/testify/assert/errors.go | 10 + .../testify/assert/forward_assertions.go | 16 + .../testify/assert/http_assertions.go | 162 + .../stretchr/testify/require/doc.go | 28 + .../testify/require/forward_requirements.go | 16 + .../stretchr/testify/require/require.go | 1879 +++++++ .../stretchr/testify/require/require.go.tmpl | 6 + .../testify/require/require_forward.go | 1471 ++++++ .../testify/require/require_forward.go.tmpl | 5 + .../stretchr/testify/require/requirements.go | 29 + .../gojsonpointer/LICENSE-APACHE-2.0.txt | 202 + .../xeipuuv/gojsonpointer/README.md | 41 + .../xeipuuv/gojsonpointer/pointer.go | 211 + .../gojsonreference/LICENSE-APACHE-2.0.txt | 202 + .../xeipuuv/gojsonreference/README.md | 10 + .../xeipuuv/gojsonreference/reference.go | 147 + .../yashtewari/glob-intersection/LICENSE | 201 + .../yashtewari/glob-intersection/README.md | 26 + .../yashtewari/glob-intersection/glob.go | 182 + .../yashtewari/glob-intersection/match.go | 91 + .../yashtewari/glob-intersection/non_empty.go | 154 + .../yashtewari/glob-intersection/simplify.go | 43 + .../glob-intersection/test_samples.go | 84 + .../yashtewari/glob-intersection/tokenize.go | 251 + vendor/gopkg.in/yaml.v2/.travis.yml | 17 + vendor/gopkg.in/yaml.v2/LICENSE | 201 + vendor/gopkg.in/yaml.v2/LICENSE.libyaml | 31 + vendor/gopkg.in/yaml.v2/NOTICE | 13 + vendor/gopkg.in/yaml.v2/README.md | 133 + vendor/gopkg.in/yaml.v2/apic.go | 744 +++ vendor/gopkg.in/yaml.v2/decode.go | 815 +++ vendor/gopkg.in/yaml.v2/emitterc.go | 1685 +++++++ vendor/gopkg.in/yaml.v2/encode.go | 390 ++ vendor/gopkg.in/yaml.v2/parserc.go | 1095 ++++ vendor/gopkg.in/yaml.v2/readerc.go | 412 ++ vendor/gopkg.in/yaml.v2/resolve.go | 258 + vendor/gopkg.in/yaml.v2/scannerc.go | 2711 ++++++++++ vendor/gopkg.in/yaml.v2/sorter.go | 113 + vendor/gopkg.in/yaml.v2/writerc.go | 26 + vendor/gopkg.in/yaml.v2/yaml.go | 478 ++ vendor/gopkg.in/yaml.v2/yamlh.go | 739 +++ vendor/gopkg.in/yaml.v2/yamlprivateh.go | 173 + vendor/gopkg.in/yaml.v3/LICENSE | 50 + vendor/gopkg.in/yaml.v3/NOTICE | 13 + vendor/gopkg.in/yaml.v3/README.md | 150 + vendor/gopkg.in/yaml.v3/apic.go | 747 +++ vendor/gopkg.in/yaml.v3/decode.go | 950 ++++ vendor/gopkg.in/yaml.v3/emitterc.go | 2020 ++++++++ vendor/gopkg.in/yaml.v3/encode.go | 577 +++ vendor/gopkg.in/yaml.v3/parserc.go | 1249 +++++ vendor/gopkg.in/yaml.v3/readerc.go | 434 ++ vendor/gopkg.in/yaml.v3/resolve.go | 326 ++ vendor/gopkg.in/yaml.v3/scannerc.go | 3038 ++++++++++++ vendor/gopkg.in/yaml.v3/sorter.go | 134 + vendor/gopkg.in/yaml.v3/writerc.go | 48 + vendor/gopkg.in/yaml.v3/yaml.go | 698 +++ vendor/gopkg.in/yaml.v3/yamlh.go | 807 +++ vendor/gopkg.in/yaml.v3/yamlprivateh.go | 198 + vendor/modules.txt | 105 + 383 files changed, 108812 insertions(+), 63 deletions(-) create mode 100644 .vscode/settings.json create mode 100644 auth/local.go create mode 100644 auth/local_test.go create mode 100644 auth/middleware.go create mode 100644 auth/testdata/allow_all.rego create mode 100644 auth/testdata/basic_auth.rego create mode 100644 auth/testdata/policy.rego create mode 100644 vendor/github.com/OneOfOne/xxhash/.gitignore create mode 100644 vendor/github.com/OneOfOne/xxhash/.travis.yml create mode 100644 vendor/github.com/OneOfOne/xxhash/LICENSE create mode 100644 vendor/github.com/OneOfOne/xxhash/README.md create mode 100644 vendor/github.com/OneOfOne/xxhash/xxhash.go create mode 100644 vendor/github.com/OneOfOne/xxhash/xxhash_go17.go create mode 100644 vendor/github.com/OneOfOne/xxhash/xxhash_safe.go create mode 100644 vendor/github.com/OneOfOne/xxhash/xxhash_unsafe.go create mode 100644 vendor/github.com/ghodss/yaml/.gitignore create mode 100644 vendor/github.com/ghodss/yaml/.travis.yml create mode 100644 vendor/github.com/ghodss/yaml/LICENSE create mode 100644 vendor/github.com/ghodss/yaml/README.md create mode 100644 vendor/github.com/ghodss/yaml/fields.go create mode 100644 vendor/github.com/ghodss/yaml/yaml.go create mode 100644 vendor/github.com/gobwas/glob/.gitignore create mode 100644 vendor/github.com/gobwas/glob/.travis.yml create mode 100644 vendor/github.com/gobwas/glob/LICENSE create mode 100644 vendor/github.com/gobwas/glob/bench.sh create mode 100644 vendor/github.com/gobwas/glob/compiler/compiler.go create mode 100644 vendor/github.com/gobwas/glob/glob.go create mode 100644 vendor/github.com/gobwas/glob/match/any.go create mode 100644 vendor/github.com/gobwas/glob/match/any_of.go create mode 100644 vendor/github.com/gobwas/glob/match/btree.go create mode 100644 vendor/github.com/gobwas/glob/match/contains.go create mode 100644 vendor/github.com/gobwas/glob/match/every_of.go create mode 100644 vendor/github.com/gobwas/glob/match/list.go create mode 100644 vendor/github.com/gobwas/glob/match/match.go create mode 100644 vendor/github.com/gobwas/glob/match/max.go create mode 100644 vendor/github.com/gobwas/glob/match/min.go create mode 100644 vendor/github.com/gobwas/glob/match/nothing.go create mode 100644 vendor/github.com/gobwas/glob/match/prefix.go create mode 100644 vendor/github.com/gobwas/glob/match/prefix_any.go create mode 100644 vendor/github.com/gobwas/glob/match/prefix_suffix.go create mode 100644 vendor/github.com/gobwas/glob/match/range.go create mode 100644 vendor/github.com/gobwas/glob/match/row.go create mode 100644 vendor/github.com/gobwas/glob/match/segments.go create mode 100644 vendor/github.com/gobwas/glob/match/single.go create mode 100644 vendor/github.com/gobwas/glob/match/suffix.go create mode 100644 vendor/github.com/gobwas/glob/match/suffix_any.go create mode 100644 vendor/github.com/gobwas/glob/match/super.go create mode 100644 vendor/github.com/gobwas/glob/match/text.go create mode 100644 vendor/github.com/gobwas/glob/readme.md create mode 100644 vendor/github.com/gobwas/glob/syntax/ast/ast.go create mode 100644 vendor/github.com/gobwas/glob/syntax/ast/parser.go create mode 100644 vendor/github.com/gobwas/glob/syntax/lexer/lexer.go create mode 100644 vendor/github.com/gobwas/glob/syntax/lexer/token.go create mode 100644 vendor/github.com/gobwas/glob/syntax/syntax.go create mode 100644 vendor/github.com/gobwas/glob/util/runes/runes.go create mode 100644 vendor/github.com/gobwas/glob/util/strings/strings.go create mode 100644 vendor/github.com/open-policy-agent/opa/LICENSE create mode 100644 vendor/github.com/open-policy-agent/opa/ast/builtins.go create mode 100644 vendor/github.com/open-policy-agent/opa/ast/capabilities.go create mode 100644 vendor/github.com/open-policy-agent/opa/ast/check.go create mode 100644 vendor/github.com/open-policy-agent/opa/ast/compare.go create mode 100644 vendor/github.com/open-policy-agent/opa/ast/compile.go create mode 100644 vendor/github.com/open-policy-agent/opa/ast/compilehelper.go create mode 100644 vendor/github.com/open-policy-agent/opa/ast/compilemetrics.go create mode 100644 vendor/github.com/open-policy-agent/opa/ast/conflicts.go create mode 100644 vendor/github.com/open-policy-agent/opa/ast/doc.go create mode 100644 vendor/github.com/open-policy-agent/opa/ast/env.go create mode 100644 vendor/github.com/open-policy-agent/opa/ast/errors.go create mode 100644 vendor/github.com/open-policy-agent/opa/ast/fuzz.go create mode 100644 vendor/github.com/open-policy-agent/opa/ast/index.go create mode 100644 vendor/github.com/open-policy-agent/opa/ast/internal/scanner/scanner.go create mode 100644 vendor/github.com/open-policy-agent/opa/ast/internal/tokens/tokens.go create mode 100644 vendor/github.com/open-policy-agent/opa/ast/location/location.go create mode 100644 vendor/github.com/open-policy-agent/opa/ast/map.go create mode 100644 vendor/github.com/open-policy-agent/opa/ast/parser.go create mode 100644 vendor/github.com/open-policy-agent/opa/ast/parser_ext.go create mode 100644 vendor/github.com/open-policy-agent/opa/ast/policy.go create mode 100644 vendor/github.com/open-policy-agent/opa/ast/pretty.go create mode 100644 vendor/github.com/open-policy-agent/opa/ast/schema.go create mode 100644 vendor/github.com/open-policy-agent/opa/ast/strings.go create mode 100644 vendor/github.com/open-policy-agent/opa/ast/term.go create mode 100644 vendor/github.com/open-policy-agent/opa/ast/transform.go create mode 100644 vendor/github.com/open-policy-agent/opa/ast/unify.go create mode 100644 vendor/github.com/open-policy-agent/opa/ast/varset.go create mode 100644 vendor/github.com/open-policy-agent/opa/ast/visit.go create mode 100644 vendor/github.com/open-policy-agent/opa/bundle/bundle.go create mode 100644 vendor/github.com/open-policy-agent/opa/bundle/file.go create mode 100644 vendor/github.com/open-policy-agent/opa/bundle/filefs.go create mode 100644 vendor/github.com/open-policy-agent/opa/bundle/hash.go create mode 100644 vendor/github.com/open-policy-agent/opa/bundle/keys.go create mode 100644 vendor/github.com/open-policy-agent/opa/bundle/sign.go create mode 100644 vendor/github.com/open-policy-agent/opa/bundle/store.go create mode 100644 vendor/github.com/open-policy-agent/opa/bundle/verify.go create mode 100644 vendor/github.com/open-policy-agent/opa/format/format.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/bundle/utils.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/cidr/merge/merge.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/compiler/wasm/opa/callgraph.csv create mode 100644 vendor/github.com/open-policy-agent/opa/internal/compiler/wasm/opa/opa.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/compiler/wasm/opa/opa.wasm create mode 100644 vendor/github.com/open-policy-agent/opa/internal/compiler/wasm/optimizations.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/compiler/wasm/wasm.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/debug/debug.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/deepcopy/deepcopy.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/file/archive/tarball.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/file/url/url.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/future/filter_imports.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/future/parser_opts.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/gojsonschema/LICENSE-APACHE-2.0.txt create mode 100644 vendor/github.com/open-policy-agent/opa/internal/gojsonschema/README.md create mode 100644 vendor/github.com/open-policy-agent/opa/internal/gojsonschema/draft.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/gojsonschema/errors.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/gojsonschema/format_checkers.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/gojsonschema/internalLog.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/gojsonschema/jsonContext.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/gojsonschema/jsonLoader.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/gojsonschema/locales.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/gojsonschema/result.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/gojsonschema/schema.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/gojsonschema/schemaLoader.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/gojsonschema/schemaPool.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/gojsonschema/schemaReferencePool.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/gojsonschema/schemaType.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/gojsonschema/subSchema.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/gojsonschema/types.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/gojsonschema/utils.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/gojsonschema/validation.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/ir/ir.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/ir/pretty.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/ir/walk.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/jwx/LICENSE create mode 100644 vendor/github.com/open-policy-agent/opa/internal/jwx/buffer/buffer.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/jwx/jwa/elliptic.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/jwx/jwa/key_type.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/jwx/jwa/parameters.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/jwx/jwa/signature.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/jwx/jwk/ecdsa.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/jwx/jwk/headers.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/jwx/jwk/interface.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/jwx/jwk/jwk.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/jwx/jwk/key_ops.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/jwx/jwk/rsa.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/jwx/jwk/symmetric.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/jwx/jws/headers.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/jwx/jws/interface.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/jwx/jws/jws.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/jwx/jws/message.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/jwx/jws/sign/ecdsa.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/jwx/jws/sign/hmac.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/jwx/jws/sign/interface.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/jwx/jws/sign/rsa.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/jwx/jws/sign/sign.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/jwx/jws/verify/ecdsa.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/jwx/jws/verify/hmac.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/jwx/jws/verify/interface.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/jwx/jws/verify/rsa.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/jwx/jws/verify/verify.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/lcss/README.md create mode 100644 vendor/github.com/open-policy-agent/opa/internal/lcss/lcss.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/lcss/qsufsort.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/leb128/leb128.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/merge/merge.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/planner/planner.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/planner/rules.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/planner/varstack.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/rego/opa/engine.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/rego/opa/options.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/semver/LICENSE create mode 100644 vendor/github.com/open-policy-agent/opa/internal/semver/semver.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/strings/strings.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/uuid/uuid.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/version/version.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/wasm/constant/constant.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/wasm/encoding/doc.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/wasm/encoding/reader.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/wasm/encoding/writer.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/wasm/instruction/control.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/wasm/instruction/instruction.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/wasm/instruction/memory.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/wasm/instruction/numeric.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/wasm/instruction/parametric.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/wasm/instruction/variable.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/wasm/module/module.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/wasm/module/pretty.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/wasm/opcode/opcode.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/wasm/sdk/opa/capabilities/capabilities.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/wasm/sdk/opa/capabilities/capabilities_nowasm.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/wasm/types/types.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/wasm/util/util.go create mode 100644 vendor/github.com/open-policy-agent/opa/keys/keys.go create mode 100644 vendor/github.com/open-policy-agent/opa/loader/errors.go create mode 100644 vendor/github.com/open-policy-agent/opa/loader/loader.go create mode 100644 vendor/github.com/open-policy-agent/opa/metrics/metrics.go create mode 100644 vendor/github.com/open-policy-agent/opa/rego/errors.go create mode 100644 vendor/github.com/open-policy-agent/opa/rego/rego.go create mode 100644 vendor/github.com/open-policy-agent/opa/rego/resultset.go create mode 100644 vendor/github.com/open-policy-agent/opa/resolver/interface.go create mode 100644 vendor/github.com/open-policy-agent/opa/resolver/wasm/wasm.go create mode 100644 vendor/github.com/open-policy-agent/opa/storage/doc.go create mode 100644 vendor/github.com/open-policy-agent/opa/storage/errors.go create mode 100644 vendor/github.com/open-policy-agent/opa/storage/inmem/inmem.go create mode 100644 vendor/github.com/open-policy-agent/opa/storage/inmem/txn.go create mode 100644 vendor/github.com/open-policy-agent/opa/storage/interface.go create mode 100644 vendor/github.com/open-policy-agent/opa/storage/internal/errors/errors.go create mode 100644 vendor/github.com/open-policy-agent/opa/storage/internal/ptr/ptr.go create mode 100644 vendor/github.com/open-policy-agent/opa/storage/path.go create mode 100644 vendor/github.com/open-policy-agent/opa/storage/storage.go create mode 100644 vendor/github.com/open-policy-agent/opa/topdown/aggregates.go create mode 100644 vendor/github.com/open-policy-agent/opa/topdown/arithmetic.go create mode 100644 vendor/github.com/open-policy-agent/opa/topdown/array.go create mode 100644 vendor/github.com/open-policy-agent/opa/topdown/binary.go create mode 100644 vendor/github.com/open-policy-agent/opa/topdown/bindings.go create mode 100644 vendor/github.com/open-policy-agent/opa/topdown/bits.go create mode 100644 vendor/github.com/open-policy-agent/opa/topdown/builtins.go create mode 100644 vendor/github.com/open-policy-agent/opa/topdown/builtins/builtins.go create mode 100644 vendor/github.com/open-policy-agent/opa/topdown/cache.go create mode 100644 vendor/github.com/open-policy-agent/opa/topdown/cache/cache.go create mode 100644 vendor/github.com/open-policy-agent/opa/topdown/cancel.go create mode 100644 vendor/github.com/open-policy-agent/opa/topdown/casts.go create mode 100644 vendor/github.com/open-policy-agent/opa/topdown/cidr.go create mode 100644 vendor/github.com/open-policy-agent/opa/topdown/comparison.go create mode 100644 vendor/github.com/open-policy-agent/opa/topdown/copypropagation/copypropagation.go create mode 100644 vendor/github.com/open-policy-agent/opa/topdown/copypropagation/unionfind.go create mode 100644 vendor/github.com/open-policy-agent/opa/topdown/crypto.go create mode 100644 vendor/github.com/open-policy-agent/opa/topdown/doc.go create mode 100644 vendor/github.com/open-policy-agent/opa/topdown/encoding.go create mode 100644 vendor/github.com/open-policy-agent/opa/topdown/errors.go create mode 100644 vendor/github.com/open-policy-agent/opa/topdown/eval.go create mode 100644 vendor/github.com/open-policy-agent/opa/topdown/glob.go create mode 100644 vendor/github.com/open-policy-agent/opa/topdown/http.go create mode 100644 vendor/github.com/open-policy-agent/opa/topdown/input.go create mode 100644 vendor/github.com/open-policy-agent/opa/topdown/instrumentation.go create mode 100644 vendor/github.com/open-policy-agent/opa/topdown/json.go create mode 100644 vendor/github.com/open-policy-agent/opa/topdown/net.go create mode 100644 vendor/github.com/open-policy-agent/opa/topdown/numbers.go create mode 100644 vendor/github.com/open-policy-agent/opa/topdown/object.go create mode 100644 vendor/github.com/open-policy-agent/opa/topdown/parse.go create mode 100644 vendor/github.com/open-policy-agent/opa/topdown/parse_bytes.go create mode 100644 vendor/github.com/open-policy-agent/opa/topdown/print.go create mode 100644 vendor/github.com/open-policy-agent/opa/topdown/print/print.go create mode 100644 vendor/github.com/open-policy-agent/opa/topdown/query.go create mode 100644 vendor/github.com/open-policy-agent/opa/topdown/reachable.go create mode 100644 vendor/github.com/open-policy-agent/opa/topdown/regex.go create mode 100644 vendor/github.com/open-policy-agent/opa/topdown/regex_template.go create mode 100644 vendor/github.com/open-policy-agent/opa/topdown/resolver.go create mode 100644 vendor/github.com/open-policy-agent/opa/topdown/runtime.go create mode 100644 vendor/github.com/open-policy-agent/opa/topdown/save.go create mode 100644 vendor/github.com/open-policy-agent/opa/topdown/semver.go create mode 100644 vendor/github.com/open-policy-agent/opa/topdown/sets.go create mode 100644 vendor/github.com/open-policy-agent/opa/topdown/strings.go create mode 100644 vendor/github.com/open-policy-agent/opa/topdown/time.go create mode 100644 vendor/github.com/open-policy-agent/opa/topdown/tokens.go create mode 100644 vendor/github.com/open-policy-agent/opa/topdown/trace.go create mode 100644 vendor/github.com/open-policy-agent/opa/topdown/type.go create mode 100644 vendor/github.com/open-policy-agent/opa/topdown/type_name.go create mode 100644 vendor/github.com/open-policy-agent/opa/topdown/uuid.go create mode 100644 vendor/github.com/open-policy-agent/opa/topdown/walk.go create mode 100644 vendor/github.com/open-policy-agent/opa/tracing/tracing.go create mode 100644 vendor/github.com/open-policy-agent/opa/types/decode.go create mode 100644 vendor/github.com/open-policy-agent/opa/types/types.go create mode 100644 vendor/github.com/open-policy-agent/opa/util/backoff.go create mode 100644 vendor/github.com/open-policy-agent/opa/util/close.go create mode 100644 vendor/github.com/open-policy-agent/opa/util/compare.go create mode 100644 vendor/github.com/open-policy-agent/opa/util/doc.go create mode 100644 vendor/github.com/open-policy-agent/opa/util/enumflag.go create mode 100644 vendor/github.com/open-policy-agent/opa/util/graph.go create mode 100644 vendor/github.com/open-policy-agent/opa/util/hashmap.go create mode 100644 vendor/github.com/open-policy-agent/opa/util/json.go create mode 100644 vendor/github.com/open-policy-agent/opa/util/queue.go create mode 100644 vendor/github.com/open-policy-agent/opa/util/wait.go create mode 100644 vendor/github.com/open-policy-agent/opa/version/version.go create mode 100644 vendor/github.com/open-policy-agent/opa/version/wasm.go create mode 100644 vendor/github.com/pkg/errors/.gitignore create mode 100644 vendor/github.com/pkg/errors/.travis.yml create mode 100644 vendor/github.com/pkg/errors/LICENSE create mode 100644 vendor/github.com/pkg/errors/Makefile create mode 100644 vendor/github.com/pkg/errors/README.md create mode 100644 vendor/github.com/pkg/errors/appveyor.yml create mode 100644 vendor/github.com/pkg/errors/errors.go create mode 100644 vendor/github.com/pkg/errors/go113.go create mode 100644 vendor/github.com/pkg/errors/stack.go create mode 100644 vendor/github.com/pmezard/go-difflib/LICENSE create mode 100644 vendor/github.com/pmezard/go-difflib/difflib/difflib.go create mode 100644 vendor/github.com/rcrowley/go-metrics/.gitignore create mode 100644 vendor/github.com/rcrowley/go-metrics/.travis.yml create mode 100644 vendor/github.com/rcrowley/go-metrics/LICENSE create mode 100644 vendor/github.com/rcrowley/go-metrics/README.md create mode 100644 vendor/github.com/rcrowley/go-metrics/counter.go create mode 100644 vendor/github.com/rcrowley/go-metrics/debug.go create mode 100644 vendor/github.com/rcrowley/go-metrics/ewma.go create mode 100644 vendor/github.com/rcrowley/go-metrics/gauge.go create mode 100644 vendor/github.com/rcrowley/go-metrics/gauge_float64.go create mode 100644 vendor/github.com/rcrowley/go-metrics/graphite.go create mode 100644 vendor/github.com/rcrowley/go-metrics/healthcheck.go create mode 100644 vendor/github.com/rcrowley/go-metrics/histogram.go create mode 100644 vendor/github.com/rcrowley/go-metrics/json.go create mode 100644 vendor/github.com/rcrowley/go-metrics/log.go create mode 100644 vendor/github.com/rcrowley/go-metrics/memory.md create mode 100644 vendor/github.com/rcrowley/go-metrics/meter.go create mode 100644 vendor/github.com/rcrowley/go-metrics/metrics.go create mode 100644 vendor/github.com/rcrowley/go-metrics/opentsdb.go create mode 100644 vendor/github.com/rcrowley/go-metrics/registry.go create mode 100644 vendor/github.com/rcrowley/go-metrics/runtime.go create mode 100644 vendor/github.com/rcrowley/go-metrics/runtime_cgo.go create mode 100644 vendor/github.com/rcrowley/go-metrics/runtime_gccpufraction.go create mode 100644 vendor/github.com/rcrowley/go-metrics/runtime_no_cgo.go create mode 100644 vendor/github.com/rcrowley/go-metrics/runtime_no_gccpufraction.go create mode 100644 vendor/github.com/rcrowley/go-metrics/sample.go create mode 100644 vendor/github.com/rcrowley/go-metrics/syslog.go create mode 100644 vendor/github.com/rcrowley/go-metrics/timer.go create mode 100644 vendor/github.com/rcrowley/go-metrics/validate.sh create mode 100644 vendor/github.com/rcrowley/go-metrics/writer.go create mode 100644 vendor/github.com/stretchr/testify/LICENSE create mode 100644 vendor/github.com/stretchr/testify/assert/assertion_compare.go create mode 100644 vendor/github.com/stretchr/testify/assert/assertion_format.go create mode 100644 vendor/github.com/stretchr/testify/assert/assertion_format.go.tmpl create mode 100644 vendor/github.com/stretchr/testify/assert/assertion_forward.go create mode 100644 vendor/github.com/stretchr/testify/assert/assertion_forward.go.tmpl create mode 100644 vendor/github.com/stretchr/testify/assert/assertion_order.go create mode 100644 vendor/github.com/stretchr/testify/assert/assertions.go create mode 100644 vendor/github.com/stretchr/testify/assert/doc.go create mode 100644 vendor/github.com/stretchr/testify/assert/errors.go create mode 100644 vendor/github.com/stretchr/testify/assert/forward_assertions.go create mode 100644 vendor/github.com/stretchr/testify/assert/http_assertions.go create mode 100644 vendor/github.com/stretchr/testify/require/doc.go create mode 100644 vendor/github.com/stretchr/testify/require/forward_requirements.go create mode 100644 vendor/github.com/stretchr/testify/require/require.go create mode 100644 vendor/github.com/stretchr/testify/require/require.go.tmpl create mode 100644 vendor/github.com/stretchr/testify/require/require_forward.go create mode 100644 vendor/github.com/stretchr/testify/require/require_forward.go.tmpl create mode 100644 vendor/github.com/stretchr/testify/require/requirements.go create mode 100644 vendor/github.com/xeipuuv/gojsonpointer/LICENSE-APACHE-2.0.txt create mode 100644 vendor/github.com/xeipuuv/gojsonpointer/README.md create mode 100644 vendor/github.com/xeipuuv/gojsonpointer/pointer.go create mode 100644 vendor/github.com/xeipuuv/gojsonreference/LICENSE-APACHE-2.0.txt create mode 100644 vendor/github.com/xeipuuv/gojsonreference/README.md create mode 100644 vendor/github.com/xeipuuv/gojsonreference/reference.go create mode 100644 vendor/github.com/yashtewari/glob-intersection/LICENSE create mode 100644 vendor/github.com/yashtewari/glob-intersection/README.md create mode 100644 vendor/github.com/yashtewari/glob-intersection/glob.go create mode 100644 vendor/github.com/yashtewari/glob-intersection/match.go create mode 100644 vendor/github.com/yashtewari/glob-intersection/non_empty.go create mode 100644 vendor/github.com/yashtewari/glob-intersection/simplify.go create mode 100644 vendor/github.com/yashtewari/glob-intersection/test_samples.go create mode 100644 vendor/github.com/yashtewari/glob-intersection/tokenize.go create mode 100644 vendor/gopkg.in/yaml.v2/.travis.yml create mode 100644 vendor/gopkg.in/yaml.v2/LICENSE create mode 100644 vendor/gopkg.in/yaml.v2/LICENSE.libyaml create mode 100644 vendor/gopkg.in/yaml.v2/NOTICE create mode 100644 vendor/gopkg.in/yaml.v2/README.md create mode 100644 vendor/gopkg.in/yaml.v2/apic.go create mode 100644 vendor/gopkg.in/yaml.v2/decode.go create mode 100644 vendor/gopkg.in/yaml.v2/emitterc.go create mode 100644 vendor/gopkg.in/yaml.v2/encode.go create mode 100644 vendor/gopkg.in/yaml.v2/parserc.go create mode 100644 vendor/gopkg.in/yaml.v2/readerc.go create mode 100644 vendor/gopkg.in/yaml.v2/resolve.go create mode 100644 vendor/gopkg.in/yaml.v2/scannerc.go create mode 100644 vendor/gopkg.in/yaml.v2/sorter.go create mode 100644 vendor/gopkg.in/yaml.v2/writerc.go create mode 100644 vendor/gopkg.in/yaml.v2/yaml.go create mode 100644 vendor/gopkg.in/yaml.v2/yamlh.go create mode 100644 vendor/gopkg.in/yaml.v2/yamlprivateh.go create mode 100644 vendor/gopkg.in/yaml.v3/LICENSE create mode 100644 vendor/gopkg.in/yaml.v3/NOTICE create mode 100644 vendor/gopkg.in/yaml.v3/README.md create mode 100644 vendor/gopkg.in/yaml.v3/apic.go create mode 100644 vendor/gopkg.in/yaml.v3/decode.go create mode 100644 vendor/gopkg.in/yaml.v3/emitterc.go create mode 100644 vendor/gopkg.in/yaml.v3/encode.go create mode 100644 vendor/gopkg.in/yaml.v3/parserc.go create mode 100644 vendor/gopkg.in/yaml.v3/readerc.go create mode 100644 vendor/gopkg.in/yaml.v3/resolve.go create mode 100644 vendor/gopkg.in/yaml.v3/scannerc.go create mode 100644 vendor/gopkg.in/yaml.v3/sorter.go create mode 100644 vendor/gopkg.in/yaml.v3/writerc.go create mode 100644 vendor/gopkg.in/yaml.v3/yaml.go create mode 100644 vendor/gopkg.in/yaml.v3/yamlh.go create mode 100644 vendor/gopkg.in/yaml.v3/yamlprivateh.go diff --git a/.vscode/settings.json b/.vscode/settings.json new file mode 100644 index 00000000..a4606455 --- /dev/null +++ b/.vscode/settings.json @@ -0,0 +1,3 @@ +{ + "go.inferGopath": false +} \ No newline at end of file diff --git a/README.md b/README.md index 65741881..56a19f0f 100644 --- a/README.md +++ b/README.md @@ -10,11 +10,11 @@ The `of-watchdog` implements an HTTP server listening on port 8080, and acts as This version of the OpenFaaS watchdog adds support for HTTP proxying as well as STDIO, which enables reuse of memory and very fast serving of requests. It does not aim to replace the [Classic Watchdog](https://github.com/openfaas/classic-watchdog), but offers another option for those who need these features. -### Goals +### Goals: -* Keep function process warm for lower latency / caching / persistent connections through using HTTP -* Enable streaming of large responses from functions, beyond the RAM or disk capacity of the container -* Cleaner abstractions for each "mode" +- Keep function process warm for lower latency / caching / persistent connections through using HTTP +- Enable streaming of large responses from functions, beyond the RAM or disk capacity of the container +- Cleaner abstractions for each "mode" ## Modes @@ -24,20 +24,20 @@ There are several modes available for the of-watchdog which changes how it inter > A comparison of three watchdog modes. Top left - Classic Watchdog, top right: afterburn (deprecated), bottom left HTTP mode from of-watchdog. -1) HTTP mode - the default and most efficient option all template authors should consider this option if the target language has a HTTP server implementation. -2) Serializing mode - for when a HTTP server implementation doesn't exist, STDIO is read into memory then sent into a forked process. -3) Streaming mode - as per serializing mode, however the request and response are both streamed instead of being buffered completely into memory before the function starts running. +1. HTTP mode - the default and most efficient option all template authors should consider this option if the target language has a HTTP server implementation. +2. Serializing mode - for when a HTTP server implementation doesn't exist, STDIO is read into memory then sent into a forked process. +3. Streaming mode - as per serializing mode, however the request and response are both streamed instead of being buffered completely into memory before the function starts running. ### API Private endpoints, served by watchdog: -* `/_/health` - returns true when the process is started, or if a lock file is in use, when that file exists. -* `/_/ready` - as per `/_/health`, but if `max_inflight` is configured to a non-zero value, and the maximum number of connections is met, it will return a 429 status +- `/_/health` - returns true when the process is started, or if a lock file is in use, when that file exists. +- `/_/ready` - as per `/_/health`, but if `max_inflight` is configured to a non-zero value, and the maximum number of connections is met, it will return a 429 status Any other HTTP requests: -* `/*` any other Path and HTTP verbs are sent to the function +- `/*` any other Path and HTTP verbs are sent to the function ### 1. HTTP (mode=http) @@ -47,13 +47,13 @@ HTTP mode is recommend for all templates where the target language has a HTTP se See a few different examples of templates, more are available via `faas-cli template store list` -| Template | HTTP framework | Repo | -| ---------------- | ------------------ | ----------------------------------------------------------- | +| Template | HTTP framework | Repo | +| ---------------- | ------------------ | ------------------------------------------------- | | Node.js 12 (LTS) | Express.js | https://github.com/openfaas/templates/ | -| Python 3 & 2.7 | Flask | https://github.com/openfaas-incubator/python-flask-template | -| Golang | Go HTTP (stdlib) | https://github.com/openfaas-incubator/golang-http-template | -| Golang | (http.HandlerFunc) | https://github.com/openfaas-incubator/golang-http-template | -| Ruby | Sinatra | https://github.com/openfaas-incubator/ruby-http | +| Python 3 & 2.7 | Flask | https://github.com/openfaas/python-flask-template | +| Golang | Go HTTP (stdlib) | https://github.com/openfaas/golang-http-template | +| Golang | (http.HandlerFunc) | https://github.com/openfaas/golang-http-template | +| Ruby | Sinatra | https://github.com/openfaas/ruby-http | | Java 11 | Sun HTTP / Gradle | https://github.com/openfaas/templates/ | Unofficial: [.NET Core / C# and Kestrel](https://github.com/burtonr/csharp-kestrel-template) @@ -65,21 +65,21 @@ the container. Pros: -* Fastest option for high concurrency and throughput -* More efficient concurrency and RAM usage vs. forking model -* Database connections can be persisted for the lifetime of the container -* Files or models can be fetched and stored in `/tmp/` as a one-off initialization task and used for all requests after that -* Does not require new/custom client libraries like afterburn but makes use of a long-running daemon such as Express.js for Node or Flask for Python +- Fastest option for high concurrency and throughput +- More efficient concurrency and RAM usage vs. forking model +- Database connections can be persisted for the lifetime of the container +- Files or models can be fetched and stored in `/tmp/` as a one-off initialization task and used for all requests after that +- Does not require new/custom client libraries like afterburn but makes use of a long-running daemon such as Express.js for Node or Flask for Python Example usage for testing: -* Forward to an Nginx container: +- Forward to an Nginx container: ``` $ go build ; mode=http port=8081 fprocess="docker run -p 80:80 --name nginx -t nginx" upstream_url=http://127.0.0.1:80 ./of-watchdog ``` -* Forward to a Node.js / Express.js hello-world app: +- Forward to a Node.js / Express.js hello-world app: ``` $ go build ; mode=http port=8081 fprocess="node expressjs-hello-world.js" upstream_url=http://127.0.0.1:3000 ./of-watchdog @@ -87,9 +87,9 @@ $ go build ; mode=http port=8081 fprocess="node expressjs-hello-world.js" upstre Cons: -* One more HTTP hop in the chain between the client and the function -* Daemons such as express/flask/sinatra can be unpredictable when used in this way so many need additional configuration -* Additional memory may be occupied between invocations vs. forking model +- One more HTTP hop in the chain between the client and the function +- Daemons such as express/flask/sinatra can be unpredictable when used in this way so many need additional configuration +- Additional memory may be occupied between invocations vs. forking model ### 2. Serializing fork (mode=serializing) @@ -107,10 +107,10 @@ Limited to processing files sized as per available memory. Reads entire request into memory from the HTTP request. At this point we serialize or modify if required. That is then written into the stdin pipe. -* Stdout pipe is read into memory and then serialized or modified if necessary before being written back to the HTTP response. -* A static Content-type can be set ahead of time. -* HTTP headers can be set even after executing the function (not implemented). -* Exec timeout: supported. +- Stdout pipe is read into memory and then serialized or modified if necessary before being written back to the HTTP response. +- A static Content-type can be set ahead of time. +- HTTP headers can be set even after executing the function (not implemented). +- Exec timeout: supported. ### 3. Streaming fork (mode=streaming) - default. @@ -120,9 +120,9 @@ HTTP headers cannot be sent after function starts executing due to input/output streaming efficiencies. Response code is always 200 unless there is an issue forking the process. An error mid-flight will have to be picked up on the client. Multi-threaded. -* Input is sent back to client as soon as it's printed to stdout by the executing process. -* A static Content-type can be set ahead of time. -* Exec timeout: supported. +- Input is sent back to client as soon as it's printed to stdout by the executing process. +- A static Content-type can be set ahead of time. +- Exec timeout: supported. ### 4. Static (mode=static) @@ -130,9 +130,75 @@ This mode starts an HTTP file server for serving static content found at the dir See an example in the [Hugo blog post](https://www.openfaas.com/blog/serverless-static-sites/). +## Auth + +The watchdog has an _OPTIONAL_ auth middleware that leverages [Open Policy Agent](https://www.openpolicyagent.org/) to allow maximum flexibility. + +This allows writing authentication and authorization policies in Rego and applying these before the invocation is sent to your function implementation. This decouples the auth layer from your function layer, allowing your to write the authentication logic once and then easily reuse it across your functions. + +The auth policy can be loaded from a secret _or_ can be built directly into your function. + +If you have created an auth policy as a secret, then you simply need to add the secret to your `secrets` and then set the `OPA_POLICY` environment variable to the secret name during deployment. You must also set the `OPA_QUERY` environment variable. + +### Example basic auth + +You can implement a simplified basic auth with the following policy as the secret `basic_auth_policy` + +```rego +package api.auth.basic + +default allow = false + +allow { + valid_basic_auth +} + +valid_basic_auth { + value := trim_prefix(input.authorization, "Basic ") + decoded := base64.decode(value) + + idx := indexof(decoded, ":") + username := substring(decoded, 0, idx) + password := substring(decoded, idx + 1, -1) + + credentials[username] == password +} + +credentials := {"bob": "secretvalue"} +``` + +Then your function would need the environment variables + +```env +OPA_POLICY=basic_auth_policy +OPA_QUERY=data.api.auth.basic +``` + +Note: the OPA query will always be `data.` + +### Policy Input + +The policy input will receive the following input data: + +| key | description | +| --------------------- | ------------------------------------------------------------------------------------------------------ | +| `input.method` | the request method, e.g. `POST` or `GET` | +| `input.path` | the request path, e.g. `/api/profile` | +| `input.authorization` | the `Authorization` header value | +| `input.headers` | the full request headers, note this is a map from string to a list of string, ie `map[string][]string` | +| `input.rawBody` | the request body as a string | +| `input.body` | the JSON parsed request body | +| `input.data` | additional input data as configured via the `OPA_INPUT_*` environnement variables | + +The `headers`, `rawBody`, and `body` are not included by default. Each is controlled the value of the `OPA_INCLUDE_HEADERS`, `OPA_INCLUDE_RAW_BODY`, and `OPA_INCLUDE_BODY` environment variables respectively. + +The `data` field contains additional data from the environment variables that have the prefix `OPA_INPUT`. For example, `OPA_INPUT_VALID_ISSUERS=http://google.com` will then be available to the policy as `input.data.valid_issuers`. + +Secret values can also be loaded into the `data` field, this is controlled via the `OPA_INPUT_SECRETS`. This can be a comma separated list of secrets to be loaded. For example, if you have a secret name `slack_hash_key` and you set `OPA_INPUT_SECRETS=slack_hash_key`, you can access the value from `input.data.slack_hash_key` + ## Metrics -| Name | Description | Type | +| Name | Description | Type | | ----------------------------- | ---------------------------- | --------- | | http_requests_total | Total number of requests | Counter | | http_request_duration_seconds | Duration of requests | Histogram | @@ -144,31 +210,31 @@ Environmental variables: > Note: timeouts should be specified as Golang durations i.e. `1m` or `20s`. -| Option |Usage| -| ---------------------- |-----| -| `fprocess` / `function_process` | Process to execute a server in `http` mode or to be executed for each request in the other modes. For non `http` mode the process must accept input via STDIN and print output via STDOUT. Also known as "function process". | -| `mode` | The mode which of-watchdog operates in, Default `streaming` [see doc](#3-streaming-fork-modestreaming---default). Options are [http](#1-http-modehttp), [serialising fork](#2-serializing-fork-modeserializing), [streaming fork](#3-streaming-fork-modestreaming---default), [static](#4-static-modestatic) | -| `read_timeout` | HTTP timeout for reading the payload from the client caller (in seconds) | -| `write_timeout` | HTTP timeout for writing a response body from your function (in seconds) | -| `exec_timeout` | Exec timeout for process exec'd for each incoming request (in seconds). Disabled if set to 0. | -| `max_inflight` | Limit the maximum number of requests in flight, and return a HTTP status 429 when exceeded | -| `prefix_logs` | When set to `true` the watchdog will add a prefix of "Date Time" + "stderr/stdout" to every line read from the function process. Default `true` | -| `log_buffer_size` | The amount of bytes to read from stderr/stdout for log lines. When exceeded, the user will see an "bufio.Scanner: token too long" error. The default value is `bufio.MaxScanTokenSize` | -| `healthcheck_interval` | Interval (in seconds) for HTTP healthcheck by container orchestrator i.e. kubelet. Used for graceful shutdowns. | -| `port` | Specify an alternative TCP port for testing. Default: `8080` | -| `content_type` | Force a specific Content-Type response for all responses - only in forking/serializing modes. | -| `suppress_lock` | When set to `false` the watchdog will attempt to write a lockfile to `/tmp/.lock` for healthchecks. Default `false` | -| `http_upstream_url` | `http` mode only - where to forward requests i.e. `127.0.0.1:5000` | -| `upstream_url` | alias for `http_upstream_url` | -| `http_buffer_req_body` | `http` mode only - buffers request body in memory before forwarding upstream to your template's `upstream_url`. Use if your upstream HTTP server does not accept `Transfer-Encoding: chunked`, for example WSGI tends to require this setting. Default: `false` | -| `buffer_http` | deprecated alias for `http_buffer_req_body`, will be removed in future version | -| `static_path` | Absolute or relative path to the directory that will be served if `mode="static"` | -| `ready_path` | When non-empty, requests to `/_/ready` will invoke the function handler with this path. This can be used to provide custom readiness logic. When `max_inflight` is set, the concurrency limit is checked first before proxying the request to the function. | +| Option | Usage | +| ------------------------------- | ------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------------ | +| `fprocess` / `function_process` | Process to execute a server in `http` mode or to be executed for each request in the other modes. For non `http` mode the process must accept input via STDIN and print output via STDOUT. Also known as "function process". | +| `mode` | The mode which of-watchdog operates in, Default `streaming` [see doc](#3-streaming-fork-modestreaming---default). Options are [http](#1-http-modehttp), [serialising fork](#2-serializing-fork-modeserializing), [streaming fork](#3-streaming-fork-modestreaming---default), [static](#4-static-modestatic) | +| `read_timeout` | HTTP timeout for reading the payload from the client caller (in seconds) | +| `write_timeout` | HTTP timeout for writing a response body from your function (in seconds) | +| `exec_timeout` | Exec timeout for process exec'd for each incoming request (in seconds). Disabled if set to 0. | +| `max_inflight` | Limit the maximum number of requests in flight, and return a HTTP status 429 when exceeded | +| `prefix_logs` | When set to `true` the watchdog will add a prefix of "Date Time" + "stderr/stdout" to every line read from the function process. Default `true` | +| `log_buffer_size` | The amount of bytes to read from stderr/stdout for log lines. When exceeded, the user will see an "bufio.Scanner: token too long" error. The default value is `bufio.MaxScanTokenSize` | +| `healthcheck_interval` | Interval (in seconds) for HTTP healthcheck by container orchestrator i.e. kubelet. Used for graceful shutdowns. | +| `port` | Specify an alternative TCP port for testing. Default: `8080` | +| `content_type` | Force a specific Content-Type response for all responses - only in forking/serializing modes. | +| `suppress_lock` | When set to `false` the watchdog will attempt to write a lockfile to `/tmp/.lock` for healthchecks. Default `false` | +| `http_upstream_url` | `http` mode only - where to forward requests i.e. `127.0.0.1:5000` | +| `upstream_url` | alias for `http_upstream_url` | +| `http_buffer_req_body` | `http` mode only - buffers request body in memory before forwarding upstream to your template's `upstream_url`. Use if your upstream HTTP server does not accept `Transfer-Encoding: chunked`, for example WSGI tends to require this setting. Default: `false` | +| `buffer_http` | deprecated alias for `http_buffer_req_body`, will be removed in future version | +| `static_path` | Absolute or relative path to the directory that will be served if `mode="static"` | +| `ready_path` | When non-empty, requests to `/_/ready` will invoke the function handler with this path. This can be used to provide custom readiness logic. When `max_inflight` is set, the concurrency limit is checked first before proxying the request to the function. | Unsupported options from the [Classic Watchdog](https://github.com/openfaas/classic-watchdog): -| Option | Usage | -| ------------- | --------------------------------------------------------------------------------------------- | -| `write_debug` | In the classic watchdog, this prints the response body out to the console | -| `read_debug` | In the classic watchdog, this prints the request body out to the console | +| Option | Usage | +| ----------------- | ---------------------------------------------------------------------------------------------------------------------------------------------------------------------- | +| `write_debug` | In the classic watchdog, this prints the response body out to the console | +| `read_debug` | In the classic watchdog, this prints the request body out to the console | | `combined_output` | In the classic watchdog, this returns STDOUT and STDERR in the function's HTTP response, when off it only returns STDOUT and prints STDERR to the logs of the watchdog | diff --git a/auth/local.go b/auth/local.go new file mode 100644 index 00000000..6dd2ca0a --- /dev/null +++ b/auth/local.go @@ -0,0 +1,99 @@ +package auth + +import ( + "context" + "encoding/json" + "fmt" + "log" + "os" + "path/filepath" + "strings" + + "github.com/open-policy-agent/opa/rego" + "github.com/open-policy-agent/opa/topdown" +) + +// Policy is the OPA policy configuration method, which is returned from +// rego.Module, rego.LoadBundle, or rego.Load +type Policy func(r *rego.Rego) + +// LoadPolicy creates an OPA Policy loader for the given path +func LoadPolicy(name string) Policy { + if strings.HasSuffix(name, "tar.gz") { + // optionally use rego.LoadBundle if name ends with tar.gz, + // this method can read a compressed bundle.tar.gz + return rego.LoadBundle(name) + } + + paths := strings.Split(name, ",") + for idx, value := range paths { + if strings.Contains(value, "/") { + // if it contains a slash, then it is already a path + continue + } + secretPath := filepath.Join(secretDir, value) + log.Printf("auth policy looks like secret name, loading from %q", secretPath) + paths[idx] = secretPath + } + return rego.Load(paths, nil) +} + +// OPAConfig controls the OPA authorizer options. +type OPAConfig struct { + // Debug enables debug logging of the query result. + Debug bool + // Query is the OPA query to evaluate. + Query string +} + +func OPAConfigFromEnv() (cfg OPAConfig) { + cfg.Debug = truthy("OPA_DEBUG", "false") + cfg.Query = os.Getenv("OPA_QUERY") + return cfg +} + +func NewLocalAuthorizer(policy Policy, cfg OPAConfig) (_ Authorizer, err error) { + auth := opa{ + cfg: cfg, + } + r := rego.New( + rego.Query(cfg.Query), + policy, + rego.EnablePrintStatements(cfg.Debug), + rego.PrintHook(topdown.NewPrintHook(log.Writer())), + ) + auth.query, err = r.PrepareForEval(context.Background()) + return auth, err +} + +type opa struct { + query rego.PreparedEvalQuery + cfg OPAConfig +} + +func (a opa) Allowed(ctx context.Context, input Input) (_ bool, err error) { + result, err := a.query.Eval(ctx, rego.EvalInput(input)) + if err != nil { + return false, fmt.Errorf("can not evaluate OPA query: %w", err) + } + + if a.cfg.Debug { + data, _ := json.Marshal(result) + log.Printf("OPA query result: %s", string(data)) + } + + return result.Allowed(), nil +} + +// truthy converts the given env variable to a boolean. +func truthy(name string, fallback string) bool { + value, ok := os.LookupEnv(name) + if !ok { + value = fallback + } + switch strings.ToLower(value) { + case "true", "yes", "on", "1": + return true + } + return false +} diff --git a/auth/local_test.go b/auth/local_test.go new file mode 100644 index 00000000..8ed7b636 --- /dev/null +++ b/auth/local_test.go @@ -0,0 +1,164 @@ +package auth + +import ( + "context" + "encoding/base64" + "encoding/json" + "net/http" + "testing" + "time" + + "github.com/stretchr/testify/require" + + _ "embed" +) + +func TestOPAAuthorizer(t *testing.T) { + ctx, cancel := context.WithTimeout(context.Background(), 2*time.Second) + defer cancel() + + cases := []struct { + name string + cfg OPAConfig + policy string + input Input + expected bool + }{ + { + name: "correctly loads and permits request when the default is true", + cfg: OPAConfig{ + Debug: false, + Query: "data.api.authz.allow", + }, + policy: "testdata/allow_all.rego", + input: Input{ + Method: http.MethodGet, + Path: "/api/endpoint", + }, + expected: true, + }, + { + name: "allow GET request on the public path", + cfg: OPAConfig{ + Debug: false, + Query: "data.api.authz.allow", + }, + policy: "testdata/policy.rego", + input: Input{ + Method: http.MethodGet, + Path: "/api/public", + }, + expected: true, + }, + { + name: "block unauthenticated POST request on the public path", + cfg: OPAConfig{ + Debug: false, + Query: "data.api.authz.allow", + }, + policy: "testdata/policy.rego", + input: Input{ + Method: http.MethodPost, + Path: "/api/public", + }, + expected: false, + }, + { + name: "allow POST request on the public path authenticated as admin", + cfg: OPAConfig{ + Debug: false, + Query: "data.api.authz.allow", + }, + policy: "testdata/policy.rego", + input: Input{ + Method: http.MethodPost, + Path: "/api/public", + Authorization: "admin", + }, + expected: true, + }, + { + name: "policy can inspect the rawBody value correctly and returns true", + cfg: OPAConfig{ + Debug: false, + Query: "data.api.authz.allow", + }, + policy: "testdata/policy.rego", + input: Input{ + Method: http.MethodPost, + Path: "/api/public", + RawBody: "permitted", + }, + expected: true, + }, + { + name: "policy can inspect the json body value correctly and returns true", + cfg: OPAConfig{ + Debug: false, + Query: "data.api.authz.allow", + }, + policy: "testdata/policy.rego", + input: Input{ + Method: http.MethodPost, + Path: "/api/public", + Body: json.RawMessage(`{"override": "permitted"}`), + }, + expected: true, + }, + { + name: "policy can implement basic auth", + cfg: OPAConfig{ + Debug: false, + Query: "data.api.authz.basic.allow", + }, + policy: "testdata/basic_auth.rego", + input: Input{ + Method: http.MethodPost, + Path: "/api/private", + Authorization: "Basic " + base64.StdEncoding.EncodeToString([]byte("bob:secretvalue")), + }, + expected: true, + }, + { + name: "can load and merge multiple policies, can evaluate basic auth", + cfg: OPAConfig{ + Debug: false, + Query: "data.api.authz.basic.allow", + }, + policy: "testdata/basic_auth.rego,testdata/policy.rego", + input: Input{ + Method: http.MethodPost, + Path: "/api/private", + Authorization: "Basic " + base64.StdEncoding.EncodeToString([]byte("bob:secretvalue")), + }, + expected: true, + }, + { + name: "can load and merge multiple policies, can evaluate token auth", + cfg: OPAConfig{ + Debug: false, + Query: "data.api.authz.allow", + }, + policy: "testdata/basic_auth.rego,testdata/policy.rego", + input: Input{ + Method: http.MethodPost, + Path: "/api/private", + Authorization: "admin", + }, + expected: true, + }, + } + + for _, tc := range cases { + t.Run(tc.name, func(t *testing.T) { + policy := LoadPolicy(tc.policy) + + opa, err := NewLocalAuthorizer(policy, tc.cfg) + require.NoError(t, err) + + result, err := opa.Allowed(ctx, tc.input) + require.NoError(t, err) + require.Equal(t, tc.expected, result) + }) + } +} diff --git a/auth/middleware.go b/auth/middleware.go new file mode 100644 index 00000000..548798e7 --- /dev/null +++ b/auth/middleware.go @@ -0,0 +1,206 @@ +package auth + +import ( + "bytes" + "context" + "encoding/json" + "fmt" + "io" + "io/ioutil" + "net/http" + "os" + "path/filepath" + "strings" +) + +const secretDir = "/var/openfaas/secrets" + +// Authorizer is the generic request authorizer interface. +type Authorizer interface { + Allowed(context.Context, Input) (bool, error) +} + +type InputConfig struct { + // IncludeJSONBody controls whether the request body is included in the OPA query, + // when true the inpug contains the `body` key as parsed JSON. + IncludeJSONBody bool + // IncludeRawBody controls whether the request body is included in the OPA query, + // when true the input contains the `rawBody` key as the raw request body. + IncludeRawBody bool + // IncludeHeaders controls if the raw request headers are included in the OPA query, + // when true the input contains the `header` key which is a map[string][]string + IncludeHeaders bool + // ErrorContentType is the content type used for the error message when the + // authorizer rejects the request. + ErrorContentType string + AdditionalData map[string]string +} + +type Input struct { + Method string `json:"method,omitempty"` + Path string `json:"path,omitempty"` + Headers http.Header `json:"headers,omitempty"` + RawBody string `json:"rawBody,omitempty"` + Authorization string `json:"authorization,omitempty"` + Body json.RawMessage `json:"body,omitempty"` + Data map[string]string `json:"data,omitempty"` +} + +func InputConfigFromEnv() (cfg InputConfig, err error) { + cfg.ErrorContentType = os.Getenv("OPA_CONTENT_TYPE") + if cfg.ErrorContentType == "" { + cfg.ErrorContentType = "text/plain" + } + + cfg.IncludeJSONBody = truthy("OPA_INCLUDE_BODY", "false") + cfg.IncludeRawBody = truthy("OPA_INCLUDE_RAW_BODY", "false") + cfg.IncludeHeaders = truthy("OPA_INCLUDE_HEADERS", "false") + + env := authEnviron() + cfg.AdditionalData, err = loadAdditionalData(env) + + return cfg, err +} + +type Middleware func(next http.Handler) http.Handler + +func New(impl Authorizer, cfg InputConfig) Middleware { + var errorWriter func(w http.ResponseWriter, msg string, status int) = http.Error + if strings.Contains(cfg.ErrorContentType, "json") { + errorWriter = jsonError + } + + return func(next http.Handler) http.Handler { + return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { + input := Input{ + Method: r.Method, + Path: r.URL.Path, + Authorization: r.Header.Get("Authorization"), + Data: cfg.AdditionalData, + } + + if cfg.IncludeHeaders { + input.Headers = r.Header + } + + var err error + var body []byte + if cfg.IncludeRawBody || cfg.IncludeJSONBody { + body, err = safeReadBody(r) + if err != nil { + errorWriter(w, "can not read request body", http.StatusInternalServerError) + return + } + } + + if cfg.IncludeRawBody { + input.RawBody = string(body) + } + + if cfg.IncludeJSONBody { + input.Body = json.RawMessage(body) + } + + allowed, err := impl.Allowed(r.Context(), input) + if err != nil { + errorWriter(w, "can not process authentication", http.StatusInternalServerError) + return + } + + if allowed { + next.ServeHTTP(w, r) + return + } + + errorWriter(w, "unauthorized", http.StatusUnauthorized) + return + }) + } +} + +// jsonError writes the error msg as a JSON object to the response writer +func jsonError(w http.ResponseWriter, msg string, status int) { + w.Header().Set("Content-Type", "application/json") + http.Error(w, fmt.Sprintf(`{"error": %q}`, msg), http.StatusInternalServerError) +} + +// return a copy of the request body and then reset the request body +func safeReadBody(r *http.Request) ([]byte, error) { + body, err := ioutil.ReadAll(r.Body) + if err != nil { + return nil, err + } + defer r.Body.Close() + + r.Body = io.NopCloser(bytes.NewReader(body)) + + return body, nil +} + +// NewAuthorizer constructs an authorizer from the environment +func NewAuthorizer(path string) (Authorizer, error) { + cfg := OPAConfigFromEnv() + + switch { + case !strings.Contains(path, "://"): + policy := LoadPolicy(path) + return NewLocalAuthorizer(policy, cfg) + case strings.HasPrefix(path, "http://"), strings.HasPrefix(path, "https://"): + return nil, fmt.Errorf("remote auth is not implemented") + default: + return nil, fmt.Errorf("unsupported auth path") + } +} + +func loadAdditionalData(options map[string]string) (map[string]string, error) { + out := map[string]string{} + for name, value := range options { + if !strings.HasPrefix(name, "OPA_INPUT") { + continue + } + if name == "OPA_INPUT_SECRETS" { + continue + } + + out[strings.TrimPrefix(name, "OPA_INPUT")] = value + } + + names := options["OPA_INPUT_SECRETS"] + if names == "" { + return out, nil + } + + requiredSecrets := strings.Split(names, ",") + for _, name := range requiredSecrets { + secretPath := filepath.Join(secretDir, name) + data, err := os.ReadFile(secretPath) + if err != nil { + return nil, err + } + out[name] = string(data) + } + + return out, nil +} + +func authEnviron() map[string]string { + out := map[string]string{} + for _, env := range os.Environ() { + if strings.HasPrefix(env, "OPA_INPUT") { + name, value, ok := cut(env, "=") + if !ok { + continue + } + out[name] = value + } + } + + return out +} + +func cut(s, sep string) (before, after string, found bool) { + if i := strings.Index(s, sep); i >= 0 { + return s[:i], s[i+len(sep):], true + } + return s, "", false +} diff --git a/auth/testdata/allow_all.rego b/auth/testdata/allow_all.rego new file mode 100644 index 00000000..33e440a8 --- /dev/null +++ b/auth/testdata/allow_all.rego @@ -0,0 +1,3 @@ +package api.authz + +allow = true diff --git a/auth/testdata/basic_auth.rego b/auth/testdata/basic_auth.rego new file mode 100644 index 00000000..fc607c4e --- /dev/null +++ b/auth/testdata/basic_auth.rego @@ -0,0 +1,26 @@ +package api.authz.basic + +default allow = false + +allow { + input.path = "/api/private" + valid_basic_auth +} + +valid_basic_auth { + value := trim_prefix(input.authorization, "Basic ") + decoded := base64.decode(value) + + print(decoded) + + colon := indexof(decoded, ":") + username := substring(decoded, 0, colon) + password := substring(decoded, colon + 1, -1) + + print(username) + print(password) + + credentials[username] == password +} + +credentials := {"bob": "secretvalue"} diff --git a/auth/testdata/policy.rego b/auth/testdata/policy.rego new file mode 100644 index 00000000..dab7927d --- /dev/null +++ b/auth/testdata/policy.rego @@ -0,0 +1,24 @@ +package api.authz + +default allow = false + +allow { + input.method == "GET" + input.path == "/api/public" +} + +allow { + is_admin +} + +is_admin { + input.authorization == "admin" +} + +allow { + input.rawBody == "permitted" +} + +allow { + input.body.override == "permitted" +} diff --git a/go.mod b/go.mod index 27a3bbc7..7f8f5daf 100644 --- a/go.mod +++ b/go.mod @@ -8,15 +8,31 @@ require ( github.com/prometheus/client_golang v1.13.0 ) +require ( + github.com/OneOfOne/xxhash v1.2.8 // indirect + github.com/ghodss/yaml v1.0.0 // indirect + github.com/gobwas/glob v0.2.3 // indirect + github.com/pkg/errors v0.9.1 // indirect + github.com/pmezard/go-difflib v1.0.0 // indirect + github.com/rcrowley/go-metrics v0.0.0-20200313005456-10cdbea86bc0 // indirect + github.com/xeipuuv/gojsonpointer v0.0.0-20190905194746-02993c407bfb // indirect + github.com/xeipuuv/gojsonreference v0.0.0-20180127040603-bd5ef7bd5415 // indirect + github.com/yashtewari/glob-intersection v0.0.0-20180916065949-5c77d914dd0b // indirect + gopkg.in/yaml.v2 v2.4.0 // indirect + gopkg.in/yaml.v3 v3.0.0-20210107192922-496545a6307b // indirect +) + require ( github.com/beorn7/perks v1.0.1 // indirect github.com/cespare/xxhash/v2 v2.1.2 // indirect github.com/davecgh/go-spew v1.1.1 // indirect github.com/golang/protobuf v1.5.2 // indirect github.com/matttproud/golang_protobuf_extensions v1.0.2 // indirect + github.com/open-policy-agent/opa v0.36.1 github.com/prometheus/client_model v0.2.0 // indirect github.com/prometheus/common v0.37.0 // indirect github.com/prometheus/procfs v0.8.0 // indirect + github.com/stretchr/testify v1.7.0 golang.org/x/sys v0.0.0-20221013171732-95e765b1cc43 // indirect google.golang.org/protobuf v1.28.1 // indirect ) diff --git a/go.sum b/go.sum index b06518b1..ed61694e 100644 --- a/go.sum +++ b/go.sum @@ -13,6 +13,20 @@ cloud.google.com/go v0.56.0/go.mod h1:jr7tqZxxKOVYizybht9+26Z/gUq7tiRzu+ACVAMbKV cloud.google.com/go v0.57.0/go.mod h1:oXiQ6Rzq3RAkkY7N6t3TcE6jE+CIBBbA36lwQ1JyzZs= cloud.google.com/go v0.62.0/go.mod h1:jmCYTdRCQuc1PHIIJ/maLInMho30T/Y0M4hTdTShOYc= cloud.google.com/go v0.65.0/go.mod h1:O5N8zS7uWy9vkA9vayVHs65eM1ubvY4h553ofrNHObY= +cloud.google.com/go v0.72.0/go.mod h1:M+5Vjvlc2wnp6tjzE102Dw08nGShTscUx2nZMufOKPI= +cloud.google.com/go v0.74.0/go.mod h1:VV1xSbzvo+9QJOxLDaJfTjx5e+MePCpCWwvftOeQmWk= +cloud.google.com/go v0.78.0/go.mod h1:QjdrLG0uq+YwhjoVOLsS1t7TW8fs36kLs4XO5R5ECHg= +cloud.google.com/go v0.79.0/go.mod h1:3bzgcEeQlzbuEAYu4mrWhKqWjmpprinYgKJLgKHnbb8= +cloud.google.com/go v0.81.0/go.mod h1:mk/AM35KwGk/Nm2YSeZbxXdrNK3KZOYHmLkOqC2V6E0= +cloud.google.com/go v0.83.0/go.mod h1:Z7MJUsANfY0pYPdw0lbnivPx4/vhy/e2FEkSkF7vAVY= +cloud.google.com/go v0.84.0/go.mod h1:RazrYuxIK6Kb7YrzzhPoLmCVzl7Sup4NrbKPg8KHSUM= +cloud.google.com/go v0.87.0/go.mod h1:TpDYlFy7vuLzZMMZ+B6iRiELaY7z/gJPaqbMx6mlWcY= +cloud.google.com/go v0.90.0/go.mod h1:kRX0mNRHe0e2rC6oNakvwQqzyDmg57xJ+SZU1eT2aDQ= +cloud.google.com/go v0.93.3/go.mod h1:8utlLll2EF5XMAV15woO4lSbWQlk8rer9aLOfLh7+YI= +cloud.google.com/go v0.94.1/go.mod h1:qAlAugsXlC+JWO+Bke5vCtc9ONxjQT3drlTTnAplMW4= +cloud.google.com/go v0.97.0/go.mod h1:GF7l59pYBVlXQIBLx3a761cZ41F9bBH3JUlihCt2Udc= +cloud.google.com/go v0.98.0/go.mod h1:ua6Ush4NALrHk5QXDWnjvZHN93OuF0HfuEPq9I1X0cM= +cloud.google.com/go v0.99.0/go.mod h1:w0Xx2nLzqWJPuozYQX+hFfCSI8WioryfRDzkoI/Y2ZA= cloud.google.com/go/bigquery v1.0.1/go.mod h1:i/xbL2UlR5RvWAURpBYZTtm/cXjCha9lbfbpx4poX+o= cloud.google.com/go/bigquery v1.3.0/go.mod h1:PjpwJnslEMmckchkHFfq+HTD2DmtT67aNFKH1/VBDHE= cloud.google.com/go/bigquery v1.4.0/go.mod h1:S8dzgnTigyfTmLBfrtrhyYhwRxG72rYxvftPBK2Dvzc= @@ -21,6 +35,7 @@ cloud.google.com/go/bigquery v1.7.0/go.mod h1://okPTzCYNXSlb24MZs83e2Do+h+VXtc4g cloud.google.com/go/bigquery v1.8.0/go.mod h1:J5hqkt3O0uAFnINi6JXValWIb1v0goeZM77hZzJN/fQ= cloud.google.com/go/datastore v1.0.0/go.mod h1:LXYbyblFSglQ5pkeyhO+Qmw7ukd3C+pD7TKLgZqpHYE= cloud.google.com/go/datastore v1.1.0/go.mod h1:umbIZjpQpHh4hmRpGhH4tLFup+FVzqBi1b3c64qFpCk= +cloud.google.com/go/firestore v1.6.1/go.mod h1:asNXNOzBdyVQmEU+ggO8UPodTkEVFW5Qx+rwHnAz+EY= cloud.google.com/go/pubsub v1.0.1/go.mod h1:R0Gpsv3s54REJCy4fxDixWD93lHJMoZTyQ2kNxGRt3I= cloud.google.com/go/pubsub v1.1.0/go.mod h1:EwwdRX2sKPjnvnqCa270oGRyludottCI76h+R3AArQw= cloud.google.com/go/pubsub v1.2.0/go.mod h1:jhfEVHT8odbXTkndysNHCcx0awwzvfOlguIAii9o8iA= @@ -33,36 +48,101 @@ cloud.google.com/go/storage v1.10.0/go.mod h1:FLPqc6j+Ki4BU591ie1oL6qBQGu2Bl/tZ9 dmitri.shuralyov.com/gpu/mtl v0.0.0-20190408044501-666a987793e9/go.mod h1:H6x//7gZCb22OMCxBHrMx7a5I7Hp++hsVxbQ4BYO7hU= github.com/BurntSushi/toml v0.3.1/go.mod h1:xHWCNGjB5oqiDr8zfno3MHue2Ht5sIBksp03qcyfWMU= github.com/BurntSushi/xgb v0.0.0-20160522181843-27f122750802/go.mod h1:IVnqGOEym/WlBOVXweHU+Q+/VP0lqqI8lqeDx9IjBqo= +github.com/DataDog/datadog-go v3.2.0+incompatible/go.mod h1:LButxg5PwREeZtORoXG3tL4fMGNddJ+vMq1mwgfaqoQ= +github.com/OneOfOne/xxhash v1.2.2/go.mod h1:HSdplMjZKSmBqAxg5vPj2TmRDmfkzw+cTzAElWljhcU= +github.com/OneOfOne/xxhash v1.2.8 h1:31czK/TI9sNkxIKfaUfGlU47BAxQ0ztGgd9vPyqimf8= +github.com/OneOfOne/xxhash v1.2.8/go.mod h1:eZbhyaAYD41SGSSsnmcpxVoRiQ/MPUTjUdIIOT9Um7Q= github.com/alecthomas/template v0.0.0-20160405071501-a0175ee3bccc/go.mod h1:LOuyumcjzFXgccqObfd/Ljyb9UuFJ6TxHnclSeseNhc= github.com/alecthomas/template v0.0.0-20190718012654-fb15b899a751/go.mod h1:LOuyumcjzFXgccqObfd/Ljyb9UuFJ6TxHnclSeseNhc= github.com/alecthomas/units v0.0.0-20151022065526-2efee857e7cf/go.mod h1:ybxpYRFXyAe+OPACYpWeL0wqObRcbAqCMya13uyzqw0= github.com/alecthomas/units v0.0.0-20190717042225-c3de453c63f4/go.mod h1:ybxpYRFXyAe+OPACYpWeL0wqObRcbAqCMya13uyzqw0= github.com/alecthomas/units v0.0.0-20190924025748-f65c72e2690d/go.mod h1:rBZYJk541a8SKzHPHnH3zbiI+7dagKZ0cgpgrD7Fyho= +github.com/antihax/optional v1.0.0/go.mod h1:uupD/76wgC+ih3iEmQUL+0Ugr19nfwCT1kdvxnR2qWY= +github.com/armon/circbuf v0.0.0-20150827004946-bbbad097214e/go.mod h1:3U/XgcO3hCbHZ8TKRvWD2dDTCfh9M9ya+I9JpbB7O8o= +github.com/armon/consul-api v0.0.0-20180202201655-eb2c6b5be1b6/go.mod h1:grANhF5doyWs3UAsr3K4I6qtAmlQcZDesFNEHPZAzj8= +github.com/armon/go-metrics v0.0.0-20180917152333-f0300d1749da/go.mod h1:Q73ZrmVTwzkszR9V5SSuryQ31EELlFMUz1kKyl939pY= +github.com/armon/go-metrics v0.3.10/go.mod h1:4O98XIr/9W0sxpJ8UaYkvjk10Iff7SnFrb4QAOwNTFc= +github.com/armon/go-radix v0.0.0-20180808171621-7fddfc383310/go.mod h1:ufUuZ+zHj4x4TnLV4JWEpy2hxWSpsRywHrMgIH9cCH8= +github.com/armon/go-radix v1.0.0/go.mod h1:ufUuZ+zHj4x4TnLV4JWEpy2hxWSpsRywHrMgIH9cCH8= github.com/beorn7/perks v0.0.0-20180321164747-3a771d992973/go.mod h1:Dwedo/Wpr24TaqPxmxbtue+5NUziq4I4S80YR8gNf3Q= github.com/beorn7/perks v1.0.0/go.mod h1:KWe93zE9D1o94FZ5RNwFwVgaQK1VOXiVxmqh+CedLV8= github.com/beorn7/perks v1.0.1 h1:VlbKKnNfV8bJzeqoa4cOKqO6bYr3WgKZxO8Z16+hsOM= github.com/beorn7/perks v1.0.1/go.mod h1:G2ZrVWU2WbWT9wwq4/hrbKbnv/1ERSJQ0ibhJ6rlkpw= +github.com/bgentry/speakeasy v0.1.0/go.mod h1:+zsyZBPWlz7T6j88CTgSN5bM796AkVf0kBD4zp0CCIs= +github.com/bytecodealliance/wasmtime-go v0.32.0 h1:/GsrnJz2bfULAIZygN4vUElLYliQrx/o/1opP9X7Gck= +github.com/bytecodealliance/wasmtime-go v0.32.0/go.mod h1:q320gUxqyI8yB+ZqRuaJOEnGkAnHh6WtJjMaT2CW4wI= +github.com/cenkalti/backoff/v4 v4.1.2/go.mod h1:scbssz8iZGpm3xbr14ovlUdkxfGXNInqkPWOWmG2CLw= github.com/census-instrumentation/opencensus-proto v0.2.1/go.mod h1:f6KPmirojxKA12rnyqOA5BBL4O983OfeGPqjHWSTneU= +github.com/census-instrumentation/opencensus-proto v0.3.0/go.mod h1:f6KPmirojxKA12rnyqOA5BBL4O983OfeGPqjHWSTneU= +github.com/cespare/xxhash v1.1.0 h1:a6HrQnmkObjyL+Gs60czilIUGqrzKutQD6XZog3p+ko= +github.com/cespare/xxhash v1.1.0/go.mod h1:XrSqR1VqqWfGrhpAt58auRo0WTKS1nRRg3ghfAqPWnc= github.com/cespare/xxhash/v2 v2.1.1/go.mod h1:VGX0DQ3Q6kWi7AoAeZDth3/j3BFtOZR5XLFGgcrjCOs= github.com/cespare/xxhash/v2 v2.1.2 h1:YRXhKfTDauu4ajMg1TPgFO5jnlC2HCbmLXMcTG5cbYE= github.com/cespare/xxhash/v2 v2.1.2/go.mod h1:VGX0DQ3Q6kWi7AoAeZDth3/j3BFtOZR5XLFGgcrjCOs= github.com/chzyer/logex v1.1.10/go.mod h1:+Ywpsq7O8HXn0nuIou7OrIPyXbp3wmkHB+jjWRnGsAI= github.com/chzyer/readline v0.0.0-20180603132655-2972be24d48e/go.mod h1:nSuG5e5PlCu98SY8svDHJxuZscDgtXS6KTTbou5AhLI= github.com/chzyer/test v0.0.0-20180213035817-a1ea475d72b1/go.mod h1:Q3SI9o4m/ZMnBNeIyt5eFwwo7qiLfzFZmjNmxjkiQlU= +github.com/circonus-labs/circonus-gometrics v2.3.1+incompatible/go.mod h1:nmEj6Dob7S7YxXgwXpfOuvO54S+tGdZdw9fuRZt25Ag= +github.com/circonus-labs/circonusllhist v0.1.3/go.mod h1:kMXHVDlOchFAehlya5ePtbp5jckzBHf4XRpQvBOLI+I= github.com/client9/misspell v0.3.4/go.mod h1:qj6jICC3Q7zFZvVWo7KLAzC3yx5G7kyvSDkc90ppPyw= github.com/cncf/udpa/go v0.0.0-20191209042840-269d4d468f6f/go.mod h1:M8M6+tZqaGXZJjfX53e64911xZQV5JYwmTeXPW+k8Sc= +github.com/cncf/udpa/go v0.0.0-20200629203442-efcf912fb354/go.mod h1:WmhPx2Nbnhtbo57+VJT5O0JRkEi1Wbu0z5j0R8u5Hbk= +github.com/cncf/udpa/go v0.0.0-20201120205902-5459f2c99403/go.mod h1:WmhPx2Nbnhtbo57+VJT5O0JRkEi1Wbu0z5j0R8u5Hbk= +github.com/cncf/udpa/go v0.0.0-20210930031921-04548b0d99d4/go.mod h1:6pvJx4me5XPnfI9Z40ddWsdw2W/uZgQLFXToKeRcDiI= +github.com/cncf/xds/go v0.0.0-20210312221358-fbca930ec8ed/go.mod h1:eXthEFrGJvWHgFFCl3hGmgk+/aYT6PnTQLykKQRLhEs= +github.com/cncf/xds/go v0.0.0-20210805033703-aa0b78936158/go.mod h1:eXthEFrGJvWHgFFCl3hGmgk+/aYT6PnTQLykKQRLhEs= +github.com/cncf/xds/go v0.0.0-20210922020428-25de7278fc84/go.mod h1:eXthEFrGJvWHgFFCl3hGmgk+/aYT6PnTQLykKQRLhEs= +github.com/cncf/xds/go v0.0.0-20211001041855-01bcc9b48dfe/go.mod h1:eXthEFrGJvWHgFFCl3hGmgk+/aYT6PnTQLykKQRLhEs= +github.com/cncf/xds/go v0.0.0-20211011173535-cb28da3451f1/go.mod h1:eXthEFrGJvWHgFFCl3hGmgk+/aYT6PnTQLykKQRLhEs= +github.com/cncf/xds/go v0.0.0-20211130200136-a8f946100490/go.mod h1:eXthEFrGJvWHgFFCl3hGmgk+/aYT6PnTQLykKQRLhEs= +github.com/coreos/etcd v3.3.10+incompatible/go.mod h1:uF7uidLiAD3TWHmW31ZFd/JWoc32PjwdhPthX9715RE= +github.com/coreos/go-etcd v2.0.0+incompatible/go.mod h1:Jez6KQU2B/sWsbdaef3ED8NzMklzPG4d5KIOhIy30Tk= +github.com/coreos/go-semver v0.2.0/go.mod h1:nnelYz7RCh+5ahJtPPxZlU+153eP4D4r3EedlOD2RNk= +github.com/coreos/go-semver v0.3.0/go.mod h1:nnelYz7RCh+5ahJtPPxZlU+153eP4D4r3EedlOD2RNk= +github.com/coreos/go-systemd/v22 v22.3.2/go.mod h1:Y58oyj3AT4RCenI/lSvhwexgC+NSVTIJ3seZv2GcEnc= +github.com/cpuguy83/go-md2man v1.0.10/go.mod h1:SmD6nW6nTyfqj6ABTjUi3V3JVMnlJmwcJI5acqYI6dE= +github.com/cpuguy83/go-md2man/v2 v2.0.1/go.mod h1:tgQtvFlXSQOSOSIRvRPT7W67SCa46tRHOmNcaadrF8o= +github.com/creack/pty v1.1.9/go.mod h1:oKZEueFk5CKHvIhNR5MUki03XCEU+Q6VDXinZuGJ33E= github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c= github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= +github.com/dgraph-io/badger/v3 v3.2103.2 h1:dpyM5eCJAtQCBcMCZcT4UBZchuTJgCywerHHgmxfxM8= +github.com/dgraph-io/badger/v3 v3.2103.2/go.mod h1:RHo4/GmYcKKh5Lxu63wLEMHJ70Pac2JqZRYGhlyAo2M= +github.com/dgraph-io/ristretto v0.1.0 h1:Jv3CGQHp9OjuMBSne1485aDpUkTKEcUqF+jm/LuerPI= +github.com/dgraph-io/ristretto v0.1.0/go.mod h1:fux0lOrBhrVCJd3lcTHsIJhq1T2rokOu6v9Vcb3Q9ug= +github.com/dgryski/go-farm v0.0.0-20190423205320-6a90982ecee2/go.mod h1:SqUrOPUnsFjfmXRMNPybcSiG0BgUW2AuFH8PAnS2iTw= +github.com/dgryski/go-farm v0.0.0-20200201041132-a6ae2369ad13/go.mod h1:SqUrOPUnsFjfmXRMNPybcSiG0BgUW2AuFH8PAnS2iTw= github.com/docker/go-units v0.5.0 h1:69rxXcBk27SvSaaxTtLh/8llcHD8vYHT7WSdRZ/jvr4= github.com/docker/go-units v0.5.0/go.mod h1:fgPhTUdO+D/Jk86RDLlptpiXQzgHJF7gydDDbaIK4Dk= +github.com/dustin/go-humanize v1.0.0 h1:VSnTsYCnlFHaM2/igO1h6X3HA71jcobQuxemgkq4zYo= +github.com/dustin/go-humanize v1.0.0/go.mod h1:HtrtbFcZ19U5GC7JDqmcUSB87Iq5E25KnS6fMYU6eOk= github.com/envoyproxy/go-control-plane v0.9.0/go.mod h1:YTl/9mNaCwkRvm6d1a2C3ymFceY/DCBVvsKhRF0iEA4= github.com/envoyproxy/go-control-plane v0.9.1-0.20191026205805-5f8ba28d4473/go.mod h1:YTl/9mNaCwkRvm6d1a2C3ymFceY/DCBVvsKhRF0iEA4= github.com/envoyproxy/go-control-plane v0.9.4/go.mod h1:6rpuAdCZL397s3pYoYcLgu1mIlRU8Am5FuJP05cCM98= +github.com/envoyproxy/go-control-plane v0.9.7/go.mod h1:cwu0lG7PUMfa9snN8LXBig5ynNVH9qI8YYLbd1fK2po= +github.com/envoyproxy/go-control-plane v0.9.9-0.20201210154907-fd9021fe5dad/go.mod h1:cXg6YxExXjJnVBQHBLXeUAgxn2UodCpnH306RInaBQk= +github.com/envoyproxy/go-control-plane v0.9.9-0.20210217033140-668b12f5399d/go.mod h1:cXg6YxExXjJnVBQHBLXeUAgxn2UodCpnH306RInaBQk= +github.com/envoyproxy/go-control-plane v0.9.9-0.20210512163311-63b5d3c536b0/go.mod h1:hliV/p42l8fGbc6Y9bQ70uLwIvmJyVE5k4iMKlh8wCQ= +github.com/envoyproxy/go-control-plane v0.9.10-0.20210907150352-cf90f659a021/go.mod h1:AFq3mo9L8Lqqiid3OhADV3RfLJnjiw63cSpi+fDTRC0= +github.com/envoyproxy/go-control-plane v0.10.1/go.mod h1:AY7fTTXNdv/aJ2O5jwpxAPOWUZ7hQAEvzN5Pf27BkQQ= github.com/envoyproxy/protoc-gen-validate v0.1.0/go.mod h1:iSmxcyjqTsJpI2R4NaDN7+kN2VEUnK/pcBlmesArF7c= +github.com/envoyproxy/protoc-gen-validate v0.6.2/go.mod h1:2t7qjJNvHPx8IjnBOzl9E9/baC+qXE/TeeyBRzgJDws= +github.com/fatih/color v1.7.0/go.mod h1:Zm6kSWBoL9eyXnKyktHP6abPY2pDugNf5KwzbycvMj4= +github.com/fatih/color v1.9.0/go.mod h1:eQcE1qtQxscV5RaZvpXrrb8Drkc3/DdQ+uUYCNjL+zU= +github.com/fatih/color v1.13.0/go.mod h1:kLAiJbzzSOZDVNGyDpeOxJ47H46qBXwg5ILebYFFOfk= +github.com/felixge/httpsnoop v1.0.2/go.mod h1:m8KPJKqk1gH5J9DgRY2ASl2lWCfGKXixSwevea8zH2U= +github.com/fortytw2/leaktest v1.3.0 h1:u8491cBMTQ8ft8aeV+adlcytMZylmA5nnwwkRZjI8vw= +github.com/fortytw2/leaktest v1.3.0/go.mod h1:jDsjWgpAGjm2CA7WthBh/CdZYEPF31XHquHwclZch5g= +github.com/foxcpp/go-mockdns v0.0.0-20210729171921-fb145fc6f897 h1:E52jfcE64UG42SwLmrW0QByONfGynWuzBvm86BoB9z8= +github.com/foxcpp/go-mockdns v0.0.0-20210729171921-fb145fc6f897/go.mod h1:lgRN6+KxQBawyIghpnl5CezHFGS9VLzvtVlwxvzXTQ4= +github.com/fsnotify/fsnotify v1.4.7/go.mod h1:jwhsz4b93w/PPRr/qN1Yymfu8t87LnFCMoQvtojpjFo= +github.com/fsnotify/fsnotify v1.5.1/go.mod h1:T3375wBYaZdLLcVNkcVbzGHY7f1l/uK5T5Ai1i3InKU= +github.com/ghodss/yaml v1.0.0 h1:wQHKEahhL6wmXdzwWG11gIVCkOv05bNOh+Rxn0yngAk= +github.com/ghodss/yaml v1.0.0/go.mod h1:4dBDuWmgqj2HViK6kFavaiC9ZROes6MMH2rRYeMEF04= github.com/go-gl/glfw v0.0.0-20190409004039-e6da0acd62b1/go.mod h1:vR7hzQXu2zJy9AVAgeJqvqgH9Q5CA+iKCZ2gyEVpxRU= github.com/go-gl/glfw/v3.3/glfw v0.0.0-20191125211704-12ad95a8df72/go.mod h1:tQ2UAYgL5IevRw8kRxooKSPJfGvJ9fJQFa0TUsXzTg8= github.com/go-gl/glfw/v3.3/glfw v0.0.0-20200222043503-6f7a984d4dc4/go.mod h1:tQ2UAYgL5IevRw8kRxooKSPJfGvJ9fJQFa0TUsXzTg8= +github.com/go-ini/ini v1.66.2/go.mod h1:ByCAeIL28uOIIG0E3PJtZPDL8WnHpFKFOtgjp+3Ies8= github.com/go-kit/kit v0.8.0/go.mod h1:xBxKIO96dXMWWy0MnWVtmwkA9/13aqxPnvrjFYMA2as= github.com/go-kit/kit v0.9.0/go.mod h1:xBxKIO96dXMWWy0MnWVtmwkA9/13aqxPnvrjFYMA2as= github.com/go-kit/log v0.1.0/go.mod h1:zbhenjAZHb184qTLMA9ZjW7ThYL0H2mk7Q6pNt4vbaY= @@ -71,12 +151,25 @@ github.com/go-logfmt/logfmt v0.3.0/go.mod h1:Qt1PoO58o5twSAckw1HlFXLmHsOX5/0LbT9 github.com/go-logfmt/logfmt v0.4.0/go.mod h1:3RMwSq7FuexP4Kalkev3ejPJsZTpXXBr9+V4qmtdjCk= github.com/go-logfmt/logfmt v0.5.0/go.mod h1:wCYkCAKZfumFQihp8CzCvQ3paCTfi41vtzG1KdI/P7A= github.com/go-logfmt/logfmt v0.5.1/go.mod h1:WYhtIu8zTZfxdn5+rREduYbwxfcBr/Vr6KEVveWlfTs= +github.com/go-logr/logr v1.2.0/go.mod h1:jdQByPbusPIv2/zmleS9BjJVeZ6kBagPoEUsqbVz/1A= +github.com/go-logr/logr v1.2.1/go.mod h1:jdQByPbusPIv2/zmleS9BjJVeZ6kBagPoEUsqbVz/1A= +github.com/go-logr/logr v1.2.2/go.mod h1:jdQByPbusPIv2/zmleS9BjJVeZ6kBagPoEUsqbVz/1A= +github.com/go-logr/stdr v1.2.0/go.mod h1:YkVgnZu1ZjjL7xTxrfm/LLZBfkhTqSR1ydtm6jTKKwI= github.com/go-stack/stack v1.8.0/go.mod h1:v0f6uXyyMGvRgIKkXu+yp6POWl0qKG85gN/melR3HDY= +github.com/gobwas/glob v0.2.3 h1:A4xDbljILXROh+kObIiy5kIaPYD8e96x1tgBhUI5J+Y= +github.com/gobwas/glob v0.2.3/go.mod h1:d3Ez4x06l9bZtSvzIay5+Yzi0fmZzPgnTbPcKjJAkT8= +github.com/godbus/dbus/v5 v5.0.4/go.mod h1:xhWf0FNVPg57R7Z0UbKHbJfkEywrmjJnf7w5xrFpKfA= github.com/gogo/protobuf v1.1.1/go.mod h1:r8qH/GZQm5c6nD/R0oafs1akxWv10x8SbQlK7atdtwQ= +github.com/gogo/protobuf v1.3.2 h1:Ov1cvc58UF3b5XjBnZv7+opcTcQFZebYjWzi34vdm4Q= +github.com/gogo/protobuf v1.3.2/go.mod h1:P1XiOD3dCwIKUDQYPy72D8LYyHL2YPYrpS2s69NZV8Q= github.com/golang/glog v0.0.0-20160126235308-23def4e6c14b/go.mod h1:SBH7ygxi8pfUlaOkMMuAQtPIUF8ecWP5IEl/CR7VP2Q= +github.com/golang/glog v1.0.0 h1:nfP3RFugxnNRyKgeWd4oI1nYvXpxrx8ck8ZrcizshdQ= +github.com/golang/glog v1.0.0/go.mod h1:EWib/APOK0SL3dFbYqvxE3UYd8E6s1ouQ7iEp/0LWV4= github.com/golang/groupcache v0.0.0-20190702054246-869f871628b6/go.mod h1:cIg4eruTrX1D+g88fzRXU5OdNfaM+9IcxsU14FzY7Hc= github.com/golang/groupcache v0.0.0-20191227052852-215e87163ea7/go.mod h1:cIg4eruTrX1D+g88fzRXU5OdNfaM+9IcxsU14FzY7Hc= github.com/golang/groupcache v0.0.0-20200121045136-8c9f03a8e57e/go.mod h1:cIg4eruTrX1D+g88fzRXU5OdNfaM+9IcxsU14FzY7Hc= +github.com/golang/groupcache v0.0.0-20210331224755-41bb18bfe9da h1:oI5xCqsCo564l8iNU+DwB5epxmsaqB+rhGL0m5jtYqE= +github.com/golang/groupcache v0.0.0-20210331224755-41bb18bfe9da/go.mod h1:cIg4eruTrX1D+g88fzRXU5OdNfaM+9IcxsU14FzY7Hc= github.com/golang/mock v1.1.1/go.mod h1:oTYuIxOrZwtPieC+H1uAHpcLFnEyAGVDL/k47Jfbm0A= github.com/golang/mock v1.2.0/go.mod h1:oTYuIxOrZwtPieC+H1uAHpcLFnEyAGVDL/k47Jfbm0A= github.com/golang/mock v1.3.1/go.mod h1:sBzyDLLjw3U8JLTeZvSv8jJB+tU5PVekmnlKIyFUx0Y= @@ -84,6 +177,8 @@ github.com/golang/mock v1.4.0/go.mod h1:UOMv5ysSaYNkG+OFQykRIcU/QvvxJf3p21QfJ2Bt github.com/golang/mock v1.4.1/go.mod h1:UOMv5ysSaYNkG+OFQykRIcU/QvvxJf3p21QfJ2Bt3cw= github.com/golang/mock v1.4.3/go.mod h1:UOMv5ysSaYNkG+OFQykRIcU/QvvxJf3p21QfJ2Bt3cw= github.com/golang/mock v1.4.4/go.mod h1:l3mdAwkq5BuhzHwde/uurv3sEJeZMXNpwsxVWU71h+4= +github.com/golang/mock v1.5.0/go.mod h1:CWnOUgYIOo4TcNZ0wHX3YZCqsaM1I1Jvs6v3mP3KVu8= +github.com/golang/mock v1.6.0/go.mod h1:p6yTPP+5HYm5mzsMV8JkE6ZKdX+/wYM6Hr+LicevLPs= github.com/golang/protobuf v1.2.0/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U= github.com/golang/protobuf v1.3.1/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U= github.com/golang/protobuf v1.3.2/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U= @@ -99,10 +194,16 @@ github.com/golang/protobuf v1.4.1/go.mod h1:U8fpvMrcmy5pZrNK1lt4xCsGvpyWQ/VVv6QD github.com/golang/protobuf v1.4.2/go.mod h1:oDoupMAO8OvCJWAcko0GGGIgR6R6ocIYbsSw735rRwI= github.com/golang/protobuf v1.4.3/go.mod h1:oDoupMAO8OvCJWAcko0GGGIgR6R6ocIYbsSw735rRwI= github.com/golang/protobuf v1.5.0/go.mod h1:FsONVRAS9T7sI+LIUmWTfcYkHO4aIWwzhcaSAoJOfIk= +github.com/golang/protobuf v1.5.1/go.mod h1:DopwsBzvsk0Fs44TXzsVbJyPhcCPeIwnvohx4u74HPM= github.com/golang/protobuf v1.5.2 h1:ROPKBNFfQgOUMifHyP+KYbvpjbdoFNs+aK7DXlji0Tw= github.com/golang/protobuf v1.5.2/go.mod h1:XVQd3VNwM+JqD3oG2Ue2ip4fOMUkwXdXDdiuN0vRsmY= +github.com/golang/snappy v0.0.3/go.mod h1:/XxbfmMg8lxefKM7IXC3fBNl/7bRcc72aCRzEWrmP2Q= +github.com/golang/snappy v0.0.4 h1:yAGX7huGHXlcLOEtBnF4w7FQwA26wojNCwOYAEhLjQM= +github.com/golang/snappy v0.0.4/go.mod h1:/XxbfmMg8lxefKM7IXC3fBNl/7bRcc72aCRzEWrmP2Q= github.com/google/btree v0.0.0-20180813153112-4030bb1f1f0c/go.mod h1:lNA+9X1NB3Zf8V7Ke586lFgjr2dZNuvo3lPJSGZ5JPQ= github.com/google/btree v1.0.0/go.mod h1:lNA+9X1NB3Zf8V7Ke586lFgjr2dZNuvo3lPJSGZ5JPQ= +github.com/google/flatbuffers v1.12.1 h1:MVlul7pQNoDzWRLTw5imwYsl+usrS1TXG2H4jg6ImGw= +github.com/google/flatbuffers v1.12.1/go.mod h1:1AeVuKshWv4vARoZatz6mlQ0JxURH0Kv5+zNeJKJCa8= github.com/google/go-cmp v0.2.0/go.mod h1:oXzfMopK8JAjlY9xF4vHSVASa0yLyX7SntLO5aqRK0M= github.com/google/go-cmp v0.3.0/go.mod h1:8QqcDgzrUqlUb/G2PQTWiueGozuR1884gddMywk6iLU= github.com/google/go-cmp v0.3.1/go.mod h1:8QqcDgzrUqlUb/G2PQTWiueGozuR1884gddMywk6iLU= @@ -110,12 +211,17 @@ github.com/google/go-cmp v0.4.0/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/ github.com/google/go-cmp v0.4.1/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= github.com/google/go-cmp v0.5.0/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= github.com/google/go-cmp v0.5.1/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= +github.com/google/go-cmp v0.5.2/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= +github.com/google/go-cmp v0.5.3/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= github.com/google/go-cmp v0.5.4/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= github.com/google/go-cmp v0.5.5/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= +github.com/google/go-cmp v0.5.6/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= github.com/google/go-cmp v0.5.8 h1:e6P7q2lk1O+qJJb4BtCQXlK8vWEO8V1ZeuEdJNOqZyg= github.com/google/gofuzz v1.0.0/go.mod h1:dBl0BpW6vV/+mYPU4Po3pmUjxk6FQPldtuIdl/M65Eg= github.com/google/martian v2.1.0+incompatible/go.mod h1:9I4somxYTbIHy5NJKHRl3wXiIaQGbYVAs8BPL6v8lEs= github.com/google/martian/v3 v3.0.0/go.mod h1:y5Zk1BBys9G+gd6Jrk0W3cC1+ELVxBWuIGO+w/tUAp0= +github.com/google/martian/v3 v3.1.0/go.mod h1:y5Zk1BBys9G+gd6Jrk0W3cC1+ELVxBWuIGO+w/tUAp0= +github.com/google/martian/v3 v3.2.1/go.mod h1:oBOf6HBosgwRXnUGWUB05QECsc6uvmMiJ3+6W4l/CUk= github.com/google/pprof v0.0.0-20181206194817-3ea8567a2e57/go.mod h1:zfwlbNMJ+OItoe0UupaVj+oy1omPYYDuagoSzA8v9mc= github.com/google/pprof v0.0.0-20190515194954-54271f7e092f/go.mod h1:zfwlbNMJ+OItoe0UupaVj+oy1omPYYDuagoSzA8v9mc= github.com/google/pprof v0.0.0-20191218002539-d4f498aebedc/go.mod h1:ZgVRPoUq/hfqzAqh7sHMqb3I9Rq5C59dIz2SbBwJ4eM= @@ -123,14 +229,58 @@ github.com/google/pprof v0.0.0-20200212024743-f11f1df84d12/go.mod h1:ZgVRPoUq/hf github.com/google/pprof v0.0.0-20200229191704-1ebb73c60ed3/go.mod h1:ZgVRPoUq/hfqzAqh7sHMqb3I9Rq5C59dIz2SbBwJ4eM= github.com/google/pprof v0.0.0-20200430221834-fc25d7d30c6d/go.mod h1:ZgVRPoUq/hfqzAqh7sHMqb3I9Rq5C59dIz2SbBwJ4eM= github.com/google/pprof v0.0.0-20200708004538-1a94d8640e99/go.mod h1:ZgVRPoUq/hfqzAqh7sHMqb3I9Rq5C59dIz2SbBwJ4eM= +github.com/google/pprof v0.0.0-20201023163331-3e6fc7fc9c4c/go.mod h1:kpwsk12EmLew5upagYY7GY0pfYCcupk39gWOCRROcvE= +github.com/google/pprof v0.0.0-20201203190320-1bf35d6f28c2/go.mod h1:kpwsk12EmLew5upagYY7GY0pfYCcupk39gWOCRROcvE= +github.com/google/pprof v0.0.0-20210122040257-d980be63207e/go.mod h1:kpwsk12EmLew5upagYY7GY0pfYCcupk39gWOCRROcvE= +github.com/google/pprof v0.0.0-20210226084205-cbba55b83ad5/go.mod h1:kpwsk12EmLew5upagYY7GY0pfYCcupk39gWOCRROcvE= +github.com/google/pprof v0.0.0-20210601050228-01bbb1931b22/go.mod h1:kpwsk12EmLew5upagYY7GY0pfYCcupk39gWOCRROcvE= +github.com/google/pprof v0.0.0-20210609004039-a478d1d731e9/go.mod h1:kpwsk12EmLew5upagYY7GY0pfYCcupk39gWOCRROcvE= +github.com/google/pprof v0.0.0-20210720184732-4bb14d4b1be1/go.mod h1:kpwsk12EmLew5upagYY7GY0pfYCcupk39gWOCRROcvE= github.com/google/renameio v0.1.0/go.mod h1:KWCgfxg9yswjAJkECMjeO8J8rahYeXnNhOm40UhjYkI= +github.com/google/uuid v1.1.2/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= github.com/googleapis/gax-go/v2 v2.0.4/go.mod h1:0Wqv26UfaUD9n4G6kQubkQ+KchISgw+vpHVxEJEs9eg= github.com/googleapis/gax-go/v2 v2.0.5/go.mod h1:DWXyrwAJ9X0FpwwEdw+IPEYBICEFu5mhpdKc/us6bOk= +github.com/googleapis/gax-go/v2 v2.1.0/go.mod h1:Q3nei7sK6ybPYH7twZdmQpAd1MKb7pfu6SK+H1/DsU0= +github.com/googleapis/gax-go/v2 v2.1.1/go.mod h1:hddJymUZASv3XPyGkUpKj8pPO47Rmb0eJc8R6ouapiM= +github.com/gorilla/mux v1.8.0/go.mod h1:DVbg23sWSpFRCP0SfiEN6jmj59UnW/n46BH5rLB71So= +github.com/grpc-ecosystem/grpc-gateway v1.16.0/go.mod h1:BDjrQk3hbvj6Nolgz8mAMFbcEtjT1g+wF4CSlocrBnw= +github.com/hashicorp/consul/api v1.11.0/go.mod h1:XjsvQN+RJGWI2TWy1/kqaE16HrR2J/FWgkYjdZQsX9M= +github.com/hashicorp/consul/sdk v0.8.0/go.mod h1:GBvyrGALthsZObzUGsfgHZQDXjg4lOjagTIwIR1vPms= +github.com/hashicorp/errwrap v1.0.0/go.mod h1:YH+1FKiLXxHSkmPseP+kNlulaMuP3n2brvKWEqk/Jc4= +github.com/hashicorp/go-cleanhttp v0.5.0/go.mod h1:JpRdi6/HCYpAwUzNwuwqhbovhLtngrth3wmdIIUrZ80= +github.com/hashicorp/go-cleanhttp v0.5.1/go.mod h1:JpRdi6/HCYpAwUzNwuwqhbovhLtngrth3wmdIIUrZ80= +github.com/hashicorp/go-cleanhttp v0.5.2/go.mod h1:kO/YDlP8L1346E6Sodw+PrpBSV4/SoxCXGY6BqNFT48= +github.com/hashicorp/go-hclog v0.12.0/go.mod h1:whpDNt7SSdeAju8AWKIWsul05p54N/39EeqMAyrmvFQ= +github.com/hashicorp/go-hclog v1.0.0/go.mod h1:whpDNt7SSdeAju8AWKIWsul05p54N/39EeqMAyrmvFQ= +github.com/hashicorp/go-immutable-radix v1.0.0/go.mod h1:0y9vanUI8NX6FsYoO3zeMjhV/C5i9g4Q3DwcSNZ4P60= +github.com/hashicorp/go-immutable-radix v1.3.1/go.mod h1:0y9vanUI8NX6FsYoO3zeMjhV/C5i9g4Q3DwcSNZ4P60= +github.com/hashicorp/go-msgpack v0.5.3/go.mod h1:ahLV/dePpqEmjfWmKiqvPkv/twdG7iPBM1vqhUKIvfM= +github.com/hashicorp/go-multierror v1.0.0/go.mod h1:dHtQlpGsu+cZNNAkkCN/P3hoUDHhCYQXV3UM06sGGrk= +github.com/hashicorp/go-multierror v1.1.0/go.mod h1:spPvp8C1qA32ftKqdAHm4hHTbPw+vmowP0z+KUhOZdA= +github.com/hashicorp/go-retryablehttp v0.5.3/go.mod h1:9B5zBasrRhHXnJnui7y6sL7es7NDiJgTc6Er0maI1Xs= +github.com/hashicorp/go-rootcerts v1.0.2/go.mod h1:pqUvnprVnM5bf7AOirdbb01K4ccR319Vf4pU3K5EGc8= +github.com/hashicorp/go-sockaddr v1.0.0/go.mod h1:7Xibr9yA9JjQq1JpNB2Vw7kxv8xerXegt+ozgdvDeDU= +github.com/hashicorp/go-syslog v1.0.0/go.mod h1:qPfqrKkXGihmCqbJM2mZgkZGvKG1dFdvsLplgctolz4= +github.com/hashicorp/go-uuid v1.0.0/go.mod h1:6SBZvOh/SIDV7/2o3Jml5SYk/TvGqwFJ/bN7x4byOro= +github.com/hashicorp/go-uuid v1.0.1/go.mod h1:6SBZvOh/SIDV7/2o3Jml5SYk/TvGqwFJ/bN7x4byOro= github.com/hashicorp/golang-lru v0.5.0/go.mod h1:/m3WP610KZHVQ1SGc6re/UDhFvYD7pJ4Ao+sR/qLZy8= github.com/hashicorp/golang-lru v0.5.1/go.mod h1:/m3WP610KZHVQ1SGc6re/UDhFvYD7pJ4Ao+sR/qLZy8= +github.com/hashicorp/golang-lru v0.5.4/go.mod h1:iADmTwqILo4mZ8BN3D2Q6+9jd8WM5uGBxy+E8yxSoD4= +github.com/hashicorp/hcl v1.0.0/go.mod h1:E5yfLk+7swimpb2L/Alb/PJmXilQ/rhwaUYs4T20WEQ= +github.com/hashicorp/logutils v1.0.0/go.mod h1:QIAnNjmIWmVIIkWDTG1z5v++HQmx9WQRO+LraFDTW64= +github.com/hashicorp/mdns v1.0.1/go.mod h1:4gW7WsVCke5TE7EPeYliwHlRUyBtfCwuFwuMg2DmyNY= +github.com/hashicorp/mdns v1.0.4/go.mod h1:mtBihi+LeNXGtG8L9dX59gAEa12BDtBQSp4v/YAJqrc= +github.com/hashicorp/memberlist v0.2.2/go.mod h1:MS2lj3INKhZjWNqd3N0m3J+Jxf3DAOnAH9VT3Sh9MUE= +github.com/hashicorp/memberlist v0.3.0/go.mod h1:MS2lj3INKhZjWNqd3N0m3J+Jxf3DAOnAH9VT3Sh9MUE= +github.com/hashicorp/serf v0.9.5/go.mod h1:UWDWwZeL5cuWDJdl0C6wrvrUwEqtQ4ZKBKKENpqIUyk= +github.com/hashicorp/serf v0.9.6/go.mod h1:TXZNMjZQijwlDvp+r0b63xZ45H7JmCmgg4gpTwn9UV4= +github.com/iancoleman/strcase v0.2.0/go.mod h1:iwCmte+B7n89clKwxIoIXy/HfoL7AsD47ZCWhYzw7ho= github.com/ianlancetaylor/demangle v0.0.0-20181102032728-5e5cf60278f6/go.mod h1:aSSvb/t6k1mPoxDqO4vJh6VOCGPwU4O0C2/Eqndh1Sc= +github.com/ianlancetaylor/demangle v0.0.0-20200824232613-28f6c0f3b639/go.mod h1:aSSvb/t6k1mPoxDqO4vJh6VOCGPwU4O0C2/Eqndh1Sc= +github.com/inconshreveable/mousetrap v1.0.0/go.mod h1:PxqpIevigyE2G7u3NXJIT2ANytuPF1OarO4DADm73n8= github.com/jpillora/backoff v1.0.0/go.mod h1:J/6gKK9jxlEcS3zixgDgUAsiuZ7yrSoa/FX5e0EB2j4= github.com/json-iterator/go v1.1.6/go.mod h1:+SdeFBvtyEkXs7REEP0seUULqWtbJapLOCVDaaPEHmU= +github.com/json-iterator/go v1.1.9/go.mod h1:KdQUCv79m/52Kvf8AW2vK1V8akMuk1QjK/uOdHXbAo4= github.com/json-iterator/go v1.1.10/go.mod h1:KdQUCv79m/52Kvf8AW2vK1V8akMuk1QjK/uOdHXbAo4= github.com/json-iterator/go v1.1.11/go.mod h1:KdQUCv79m/52Kvf8AW2vK1V8akMuk1QjK/uOdHXbAo4= github.com/json-iterator/go v1.1.12/go.mod h1:e30LSqwooZae/UwlEbR2852Gd8hjQvJoHmT4TnhNGBo= @@ -138,17 +288,51 @@ github.com/jstemmer/go-junit-report v0.0.0-20190106144839-af01ea7f8024/go.mod h1 github.com/jstemmer/go-junit-report v0.9.1/go.mod h1:Brl9GWCQeLvo8nXZwPNNblvFj/XSXhF0NWZEnDohbsk= github.com/julienschmidt/httprouter v1.2.0/go.mod h1:SYymIcj16QtmaHHD7aYtjjsJG7VTCxuUUipMqKk8s4w= github.com/julienschmidt/httprouter v1.3.0/go.mod h1:JR6WtHb+2LUe8TCKY3cZOxFyyO8IZAc4RVcycCCAKdM= +github.com/kisielk/errcheck v1.5.0/go.mod h1:pFxgyoBC7bSaBwPgfKdkLd5X25qrDl4LWUI2bnpBCr8= github.com/kisielk/gotool v1.0.0/go.mod h1:XhKaO+MFFWcvkIS/tQcRk01m1F5IRFswLeQ+oQHNcck= +github.com/klauspost/compress v1.12.3/go.mod h1:8dP1Hq4DHOhN9w426knH3Rhby4rFm6D8eO+e+Dq5Gzg= +github.com/klauspost/compress v1.13.5 h1:9O69jUPDcsT9fEm74W92rZL9FQY7rCdaXVneq+yyzl4= +github.com/klauspost/compress v1.13.5/go.mod h1:/3/Vjq9QcHkK5uEr5lBEmyoZ1iFhe47etQ6QUkpK6sk= github.com/konsorten/go-windows-terminal-sequences v1.0.1/go.mod h1:T0+1ngSBFLxvqU3pZ+m/2kptfBszLMUkC4ZK/EgS/cQ= github.com/konsorten/go-windows-terminal-sequences v1.0.3/go.mod h1:T0+1ngSBFLxvqU3pZ+m/2kptfBszLMUkC4ZK/EgS/cQ= +github.com/kr/fs v0.1.0/go.mod h1:FFnZGqtBN9Gxj7eW1uZ42v5BccTP0vu6NEaFoC2HwRg= github.com/kr/logfmt v0.0.0-20140226030751-b84e30acd515/go.mod h1:+0opPa2QZZtGFBFZlji/RkVcI2GknAs/DXo4wKdlNEc= github.com/kr/pretty v0.1.0/go.mod h1:dAy3ld7l9f0ibDNOQOHHMYYIIbhfbHSm3C4ZsoJORNo= +github.com/kr/pretty v0.2.0/go.mod h1:ipq/a2n7PKx3OHsz4KJII5eveXtPO4qwEXGdVfWzfnI= github.com/kr/pty v1.1.1/go.mod h1:pFQYn66WHrOpPYNljwOMqo10TkYh1fy3cYio2l3bCsQ= github.com/kr/text v0.1.0/go.mod h1:4Jbv+DJW3UT/LiOwJeYQe1efqtUx/iVham/4vfdArNI= -github.com/matttproud/golang_protobuf_extensions v1.0.1 h1:4hp9jkHxhMHkqkrB3Ix0jegS5sx/RkqARlsWZ6pIwiU= +github.com/kr/text v0.2.0 h1:5Nx0Ya0ZqY2ygV366QzturHI13Jq95ApcVaJBhpS+AY= +github.com/kr/text v0.2.0/go.mod h1:eLer722TekiGuMkidMxC/pM04lWEeraHUUmBw8l2grE= +github.com/lyft/protoc-gen-star v0.5.3/go.mod h1:V0xaHgaf5oCCqmcxYcWiDfTiKsZsRc87/1qhoTACD8w= +github.com/magiconair/properties v1.8.0/go.mod h1:PppfXfuXeibc/6YijjN8zIbojt8czPbwD3XqdrwzmxQ= +github.com/magiconair/properties v1.8.5/go.mod h1:y3VJvCyxH9uVvJTWEGAELF3aiYNyPKd5NZ3oSwXrF60= +github.com/mattn/go-colorable v0.0.9/go.mod h1:9vuHe8Xs5qXnSaW/c/ABM9alt+Vo+STaOChaDxuIBZU= +github.com/mattn/go-colorable v0.1.4/go.mod h1:U0ppj6V5qS13XJ6of8GYAs25YV2eR4EVcfRqFIhoBtE= +github.com/mattn/go-colorable v0.1.6/go.mod h1:u6P/XSegPjTcexA+o6vUJrdnUu04hMope9wVRipJSqc= +github.com/mattn/go-colorable v0.1.9/go.mod h1:u6P/XSegPjTcexA+o6vUJrdnUu04hMope9wVRipJSqc= +github.com/mattn/go-colorable v0.1.12/go.mod h1:u5H1YNBxpqRaxsYJYSkiCWKzEfiAb1Gb520KVy5xxl4= +github.com/mattn/go-isatty v0.0.3/go.mod h1:M+lRXTBqGeGNdLjl/ufCoiOlB5xdOkqRJdNxMWT7Zi4= +github.com/mattn/go-isatty v0.0.8/go.mod h1:Iq45c/XA43vh69/j3iqttzPXn0bhXyGjM0Hdxcsrc5s= +github.com/mattn/go-isatty v0.0.10/go.mod h1:qgIWMr58cqv1PHHyhnkY9lrL7etaEgOFcMEpPG5Rm84= +github.com/mattn/go-isatty v0.0.11/go.mod h1:PhnuNfih5lzO57/f3n+odYbM4JtupLOxQOAqxQCu2WE= +github.com/mattn/go-isatty v0.0.12/go.mod h1:cbi8OIDigv2wuxKPP5vlRcQ1OAZbq2CE4Kysco4FUpU= +github.com/mattn/go-isatty v0.0.14/go.mod h1:7GGIvUiUoEMVVmxf/4nioHXj79iQHKdU27kJ6hsGG94= +github.com/mattn/go-runewidth v0.0.9/go.mod h1:H031xJmbD/WCDINGzjvQ9THkh0rPKHF+m2gUSrubnMI= github.com/matttproud/golang_protobuf_extensions v1.0.1/go.mod h1:D8He9yQNgCq6Z5Ld7szi9bcBfOoFv/3dc6xSMkL2PC0= github.com/matttproud/golang_protobuf_extensions v1.0.2 h1:hAHbPm5IJGijwng3PWk09JkG9WeqChjprR5s9bBZ+OM= github.com/matttproud/golang_protobuf_extensions v1.0.2/go.mod h1:BSXmuO+STAnVfrANrmjBb36TMTDstsz7MSK+HVaYKv4= +github.com/miekg/dns v1.0.14/go.mod h1:W1PPwlIAgtquWBMBEV9nkV9Cazfe8ScdGz/Lj7v3Nrg= +github.com/miekg/dns v1.1.25/go.mod h1:bPDLeHnStXmXAq1m/Ch/hvfNHr14JKNPMBo3VZKjuso= +github.com/miekg/dns v1.1.26/go.mod h1:bPDLeHnStXmXAq1m/Ch/hvfNHr14JKNPMBo3VZKjuso= +github.com/miekg/dns v1.1.41/go.mod h1:p6aan82bvRIyn+zDIv9xYNUpwa73JcSh9BKwknJysuI= +github.com/miekg/dns v1.1.43 h1:JKfpVSCB84vrAmHzyrsxB5NAr5kLoMXZArPSw7Qlgyg= +github.com/miekg/dns v1.1.43/go.mod h1:+evo5L0630/F6ca/Z9+GAqzhjGyn8/c+TBaOyfEl0V4= +github.com/mitchellh/cli v1.1.0/go.mod h1:xcISNoH86gajksDmfB23e/pu+B+GeFRMYmoHXxx3xhI= +github.com/mitchellh/go-homedir v1.1.0/go.mod h1:SfyaCUpYCn1Vlf4IUYiD9fPX4A5wJrkLzIz1N1q0pr0= +github.com/mitchellh/go-testing-interface v1.0.0/go.mod h1:kRemZodwjscx+RGhAo8eIhFbs2+BFgRtFPeD/KE+zxI= +github.com/mitchellh/mapstructure v0.0.0-20160808181253-ca63d7c062ee/go.mod h1:FVVH3fgwuzCH5S8UJGiWEs2h04kUh9fWfEaFds41c1Y= +github.com/mitchellh/mapstructure v1.1.2/go.mod h1:FVVH3fgwuzCH5S8UJGiWEs2h04kUh9fWfEaFds41c1Y= +github.com/mitchellh/mapstructure v1.4.3/go.mod h1:bFUtVrKA4DC2yAKiSyO/QUcy7e+RRV2QTWOzhPopBRo= github.com/modern-go/concurrent v0.0.0-20180228061459-e0a39a4cb421/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q= github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q= github.com/modern-go/reflect2 v0.0.0-20180701023420-4b7aa43c6742/go.mod h1:bx2lNnkwVCuqBIxFjflWJWanXIb3RllmbCylyMrvgv0= @@ -156,16 +340,30 @@ github.com/modern-go/reflect2 v1.0.1/go.mod h1:bx2lNnkwVCuqBIxFjflWJWanXIb3Rllmb github.com/modern-go/reflect2 v1.0.2/go.mod h1:yWuevngMOJpCy52FWWMvUC8ws7m/LJsjYzDa0/r8luk= github.com/mwitkow/go-conntrack v0.0.0-20161129095857-cc309e4a2223/go.mod h1:qRWi+5nqEBWmkhHvq77mSJWrCKwh8bxhgT7d/eI7P4U= github.com/mwitkow/go-conntrack v0.0.0-20190716064945-2f068394615f/go.mod h1:qRWi+5nqEBWmkhHvq77mSJWrCKwh8bxhgT7d/eI7P4U= -github.com/openfaas/faas-middleware v1.2.1 h1:uHDJbaajvD2N011v8Ncin0mTpdkrIAcE80u2vHTBRWQ= -github.com/openfaas/faas-middleware v1.2.1/go.mod h1:RgkVC/llBh+Eqb4bKxcFneB4OMnYsjUrEs7TWrRf51s= +github.com/niemeyer/pretty v0.0.0-20200227124842-a10e7caefd8e h1:fD57ERR4JtEqsWbfPhv4DMiApHyliiK5xCTNVSPiaAs= +github.com/niemeyer/pretty v0.0.0-20200227124842-a10e7caefd8e/go.mod h1:zD1mROLANZcx1PVRCS0qkT7pwLkGfwJo4zjcN/Tysno= +github.com/olekukonko/tablewriter v0.0.5/go.mod h1:hPp6KlRPjbx+hW8ykQs1w3UBbZlj6HuIJcUGPhkA7kY= +github.com/open-policy-agent/opa v0.36.1 h1:FkopbI3Rlor1nAvu78hgbdOLZyRd4vPepeMY4Tcm1Hc= +github.com/open-policy-agent/opa v0.36.1/go.mod h1:/ZcbCeVlsRFp+n7aAvcTfElCL24b53p9v9QGo2y+0RM= github.com/openfaas/faas-middleware v1.2.2 h1:0aFr55OSODCR1t6FQ5LoOetgL3RzCZTz5doHI1mAdwo= github.com/openfaas/faas-middleware v1.2.2/go.mod h1:RgkVC/llBh+Eqb4bKxcFneB4OMnYsjUrEs7TWrRf51s= +github.com/pascaldekloe/goe v0.0.0-20180627143212-57f6aae5913c/go.mod h1:lzWF7FIEvWOWxwDKqyGYQf6ZUaNfKdP144TG7ZOy1lc= +github.com/pascaldekloe/goe v0.1.0/go.mod h1:lzWF7FIEvWOWxwDKqyGYQf6ZUaNfKdP144TG7ZOy1lc= +github.com/pelletier/go-toml v1.2.0/go.mod h1:5z9KED0ma1S8pY6P1sdut58dfprrGBbd/94hg7ilaic= +github.com/pelletier/go-toml v1.9.4/go.mod h1:u1nR/EPcESfeI/szUZKdtJ0xRNbUoANCkoOuaOx1Y+c= +github.com/peterh/liner v0.0.0-20170211195444-bf27d3ba8e1d/go.mod h1:xIteQHvHuaLYG9IFj6mSxM0fCKrs34IrEQUhOYuGPHc= github.com/pkg/errors v0.8.0/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0= github.com/pkg/errors v0.8.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0= +github.com/pkg/errors v0.9.1 h1:FEBLx1zS214owpjy7qsBeixbURkuhQAwrK5UwLGTwt4= github.com/pkg/errors v0.9.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0= +github.com/pkg/sftp v1.10.1/go.mod h1:lYOWFsE0bwd1+KfKJaKeuokY15vzFx25BLbzYYoAxZI= +github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM= github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4= +github.com/posener/complete v1.1.1/go.mod h1:em0nMJCgc9GFtwrmVmEMR/ZL6WyhyjMBndrE9hABlRI= +github.com/posener/complete v1.2.3/go.mod h1:WZIdtGGp+qx0sLrYKtIRAruyNpv6hFCicSgv7Sy7s/s= github.com/prometheus/client_golang v0.9.1/go.mod h1:7SWBe2y4D6OKWSNQJUaRYU/AaXPKyh/dDVn+NZz0KFw= github.com/prometheus/client_golang v1.0.0/go.mod h1:db9x61etRT2tGnBNRi70OPL5FsnadC4Ky3P0J6CfImo= +github.com/prometheus/client_golang v1.4.0/go.mod h1:e9GMxYsXl05ICDXkRhurwBS4Q3OK1iX/F2sw+iXX5zU= github.com/prometheus/client_golang v1.7.1/go.mod h1:PY5Wy2awLA44sXw4AOSfFBetzPP4j5+D6mVACh+pe2M= github.com/prometheus/client_golang v1.11.0/go.mod h1:Z6t4BnS23TR94PD6BsDNk8yVqroYurpAkEiz0P2BEV0= github.com/prometheus/client_golang v1.12.1/go.mod h1:3Z9XVyYiZYEO+YQWt3RD2R3jrbd179Rt297l4aS6nDY= @@ -177,41 +375,111 @@ github.com/prometheus/client_model v0.0.0-20190812154241-14fe0d1b01d4/go.mod h1: github.com/prometheus/client_model v0.2.0 h1:uq5h0d+GuxiXLJLNABMgp2qUWDPiLvgCzz2dUR+/W/M= github.com/prometheus/client_model v0.2.0/go.mod h1:xMI15A0UPsDsEKsMN9yxemIoYk6Tm2C1GtYGdfGttqA= github.com/prometheus/common v0.4.1/go.mod h1:TNfzLD0ON7rHzMJeJkieUDPYmFC7Snx/y86RQel1bk4= +github.com/prometheus/common v0.9.1/go.mod h1:yhUN8i9wzaXS3w1O07YhxHEBxD+W35wd8bs7vj7HSQ4= github.com/prometheus/common v0.10.0/go.mod h1:Tlit/dnDKsSWFlCLTWaA1cyBgKHSMdTB80sz/V91rCo= github.com/prometheus/common v0.26.0/go.mod h1:M7rCNAaPfAosfx8veZJCuw84e35h3Cfd9VFqTh1DIvc= +github.com/prometheus/common v0.29.0/go.mod h1:vu+V0TpY+O6vW9J44gczi3Ap/oXXR10b+M/gUGO4Hls= github.com/prometheus/common v0.32.1/go.mod h1:vu+V0TpY+O6vW9J44gczi3Ap/oXXR10b+M/gUGO4Hls= github.com/prometheus/common v0.37.0 h1:ccBbHCgIiT9uSoFY0vX8H3zsNR5eLt17/RQLUvn8pXE= github.com/prometheus/common v0.37.0/go.mod h1:phzohg0JFMnBEFGxTDbfu3QyL5GI8gTQJFhYO5B3mfA= github.com/prometheus/procfs v0.0.0-20181005140218-185b4288413d/go.mod h1:c3At6R/oaqEKCNdg8wHV1ftS6bRYblBhIjjI8uT2IGk= github.com/prometheus/procfs v0.0.2/go.mod h1:TjEm7ze935MbeOT/UhFTIMYKhuLP4wbCsTZCD3I8kEA= +github.com/prometheus/procfs v0.0.8/go.mod h1:7Qr8sr6344vo1JqZ6HhLceV9o3AJ1Ff+GxbHq6oeK9A= github.com/prometheus/procfs v0.1.3/go.mod h1:lV6e/gmhEcM9IjHGsFOCxxuZ+z1YqCvr4OA4YeYWdaU= github.com/prometheus/procfs v0.6.0/go.mod h1:cz+aTbrPOrUb4q7XlbU9ygM+/jj0fzG6c1xBZuNvfVA= github.com/prometheus/procfs v0.7.3/go.mod h1:cz+aTbrPOrUb4q7XlbU9ygM+/jj0fzG6c1xBZuNvfVA= github.com/prometheus/procfs v0.8.0 h1:ODq8ZFEaYeCaZOJlZZdJA2AbQR98dSHSM1KW/You5mo= github.com/prometheus/procfs v0.8.0/go.mod h1:z7EfXMXOkbkqb9IINtpCn86r/to3BnA0uaxHdg830/4= +github.com/rcrowley/go-metrics v0.0.0-20200313005456-10cdbea86bc0 h1:MkV+77GLUNo5oJ0jf870itWm3D0Sjh7+Za9gazKc5LQ= +github.com/rcrowley/go-metrics v0.0.0-20200313005456-10cdbea86bc0/go.mod h1:bCqnVzQkZxMG4s8nGwiZ5l3QUCyqpo9Y+/ZMZ9VjZe4= +github.com/rogpeppe/fastuuid v1.2.0/go.mod h1:jVj6XXZzXRy/MSR5jhDC/2q6DgLz+nrA6LYCDYWNEvQ= github.com/rogpeppe/go-internal v1.3.0/go.mod h1:M8bDsm7K2OlrFYOpmOWEs/qY81heoFRclV5y23lUDJ4= +github.com/russross/blackfriday v1.5.2/go.mod h1:JO/DiYxRf+HjHt06OyowR9PTA263kcR/rfWxYHBV53g= +github.com/russross/blackfriday/v2 v2.1.0/go.mod h1:+Rmxgy9KzJVeS9/2gXHxylqXiyQDYRxCVz55jmeOWTM= +github.com/ryanuber/columnize v0.0.0-20160712163229-9b3edd62028f/go.mod h1:sm1tb6uqfes/u+d4ooFouqFdy9/2g9QGwK3SQygK0Ts= +github.com/sagikazarmark/crypt v0.3.0/go.mod h1:uD/D+6UF4SrIR1uGEv7bBNkNqLGqUr43MRiaGWX1Nig= +github.com/sean-/seed v0.0.0-20170313163322-e2103e2c3529/go.mod h1:DxrIzT+xaE7yg65j358z/aeFdxmN0P9QXhEzd20vsDc= github.com/sirupsen/logrus v1.2.0/go.mod h1:LxeOpSwHxABJmUn/MG1IvRgCAasNZTLOkJPxbbu5VWo= github.com/sirupsen/logrus v1.4.2/go.mod h1:tLMulIdttU9McNUspp0xgXVQah82FyeX6MwdIuYE2rE= github.com/sirupsen/logrus v1.6.0/go.mod h1:7uNnSEd1DgxDLC74fIahvMZmmYsHGZGEOFrfsX/uA88= +github.com/sirupsen/logrus v1.8.1/go.mod h1:yWOB1SBYBC5VeMP7gHvWumXLIWorT60ONWic61uBYv0= +github.com/spaolacci/murmur3 v0.0.0-20180118202830-f09979ecbc72/go.mod h1:JwIasOWyU6f++ZhiEuf87xNszmSA2myDM2Kzu9HwQUA= +github.com/spaolacci/murmur3 v1.1.0/go.mod h1:JwIasOWyU6f++ZhiEuf87xNszmSA2myDM2Kzu9HwQUA= +github.com/spf13/afero v1.1.2/go.mod h1:j4pytiNVoe2o6bmDsKpLACNPDBIoEAkihy7loJ1B0CQ= +github.com/spf13/afero v1.3.3/go.mod h1:5KUK8ByomD5Ti5Artl0RtHeI5pTF7MIDuXL3yY520V4= +github.com/spf13/afero v1.6.0/go.mod h1:Ai8FlHk4v/PARR026UzYexafAt9roJ7LcLMAmO6Z93I= +github.com/spf13/cast v1.3.0/go.mod h1:Qx5cxh0v+4UWYiBimWS+eyWzqEqokIECu5etghLkUJE= +github.com/spf13/cast v1.4.1/go.mod h1:Qx5cxh0v+4UWYiBimWS+eyWzqEqokIECu5etghLkUJE= +github.com/spf13/cobra v0.0.5/go.mod h1:3K3wKZymM7VvHMDS9+Akkh4K60UwM26emMESw8tLCHU= +github.com/spf13/cobra v1.3.0/go.mod h1:BrRVncBjOJa/eUcVVm9CE+oC6as8k+VYr4NY7WCi9V4= +github.com/spf13/jwalterweatherman v1.0.0/go.mod h1:cQK4TGJAtQXfYWX+Ddv3mKDzgVb68N+wFjFa4jdeBTo= +github.com/spf13/jwalterweatherman v1.1.0/go.mod h1:aNWZUN0dPAAO/Ljvb5BEdw96iTZ0EXowPYD95IqWIGo= +github.com/spf13/pflag v1.0.3/go.mod h1:DYY7MBk1bdzusC3SYhjObp+wFpr4gzcvqqNjLnInEg4= +github.com/spf13/pflag v1.0.5/go.mod h1:McXfInJRrz4CZXVZOBLb0bTZqETkiAhM9Iw0y3An2Bg= +github.com/spf13/viper v1.3.2/go.mod h1:ZiWeW+zYFKm7srdB9IoDzzZXaJaI5eL9QjNiN/DMA2s= +github.com/spf13/viper v1.10.0/go.mod h1:SoyBPwAtKDzypXNDFKN5kzH7ppppbGZtls1UpIy5AsM= github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME= github.com/stretchr/objx v0.1.1/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME= github.com/stretchr/testify v1.2.2/go.mod h1:a8OnRcib4nhh0OaRAV+Yts87kKdq0PP7pXfy6kDkUVs= github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI= github.com/stretchr/testify v1.4.0/go.mod h1:j7eGeouHqKxXV5pUuKE4zz7dFj8WfuZ+81PSLYec5m4= +github.com/stretchr/testify v1.5.1/go.mod h1:5W2xD1RspED5o8YsWQXVCued0rvSQ+mT+I5cxcmMvtA= +github.com/stretchr/testify v1.6.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= +github.com/stretchr/testify v1.7.0 h1:nwc3DEeHmmLAfoZucVR881uASk0Mfjw8xYJ99tb5CcY= +github.com/stretchr/testify v1.7.0/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= +github.com/subosito/gotenv v1.2.0/go.mod h1:N0PQaV/YGNqwC0u51sEeR/aUtSLEXKX9iv69rRypqCw= +github.com/tv42/httpunix v0.0.0-20150427012821-b75d8614f926/go.mod h1:9ESjWnEqriFuLhtthL60Sar/7RFoluCcXsuvEwTV5KM= +github.com/ugorji/go/codec v0.0.0-20181204163529-d75b2dcb6bc8/go.mod h1:VFNgLljTbGfSG7qAOspJ7OScBnGdDN/yBr0sguwnwf0= +github.com/xeipuuv/gojsonpointer v0.0.0-20190905194746-02993c407bfb h1:zGWFAtiMcyryUHoUjUJX0/lt1H2+i2Ka2n+D3DImSNo= +github.com/xeipuuv/gojsonpointer v0.0.0-20190905194746-02993c407bfb/go.mod h1:N2zxlSyiKSe5eX1tZViRH5QA0qijqEDrYZiPEAiq3wU= +github.com/xeipuuv/gojsonreference v0.0.0-20180127040603-bd5ef7bd5415 h1:EzJWgHovont7NscjpAxXsDA8S8BMYve8Y5+7cuRE7R0= +github.com/xeipuuv/gojsonreference v0.0.0-20180127040603-bd5ef7bd5415/go.mod h1:GwrjFmJcFw6At/Gs6z4yjiIwzuJ1/+UwLxMQDVQXShQ= +github.com/xordataexchange/crypt v0.0.3-0.20170626215501-b2862e3d0a77/go.mod h1:aYKd//L2LvnjZzWKhF00oedf4jCCReLcmhLdhm1A27Q= +github.com/yashtewari/glob-intersection v0.0.0-20180916065949-5c77d914dd0b h1:vVRagRXf67ESqAb72hG2C/ZwI8NtJF2u2V76EsuOHGY= +github.com/yashtewari/glob-intersection v0.0.0-20180916065949-5c77d914dd0b/go.mod h1:HptNXiXVDcJjXe9SqMd0v2FsL9f8dz4GnXgltU6q/co= github.com/yuin/goldmark v1.1.25/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74= github.com/yuin/goldmark v1.1.27/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74= github.com/yuin/goldmark v1.1.32/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74= +github.com/yuin/goldmark v1.2.1/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74= +github.com/yuin/goldmark v1.3.5/go.mod h1:mwnBkeHKe2W/ZEtQ+71ViKU8L12m81fl3OWwC1Zlc8k= +go.etcd.io/etcd/api/v3 v3.5.1/go.mod h1:cbVKeC6lCfl7j/8jBhAK6aIYO9XOjdptoxU/nLQcPvs= +go.etcd.io/etcd/client/pkg/v3 v3.5.1/go.mod h1:IJHfcCEKxYu1Os13ZdwCwIUTUVGYTSAM3YSwc9/Ac1g= +go.etcd.io/etcd/client/v2 v2.305.1/go.mod h1:pMEacxZW7o8pg4CrFE7pquyCJJzZvkvdD2RibOCCCGs= go.opencensus.io v0.21.0/go.mod h1:mSImk1erAIZhrmZN+AvHh14ztQfjbGwt4TtuofqLduU= go.opencensus.io v0.22.0/go.mod h1:+kGneAE2xo2IficOXnaByMWTGM9T73dGwxeWcUqIpI8= go.opencensus.io v0.22.2/go.mod h1:yxeiOL68Rb0Xd1ddK5vPZ/oVn4vY4Ynel7k9FzqtOIw= go.opencensus.io v0.22.3/go.mod h1:yxeiOL68Rb0Xd1ddK5vPZ/oVn4vY4Ynel7k9FzqtOIw= go.opencensus.io v0.22.4/go.mod h1:yxeiOL68Rb0Xd1ddK5vPZ/oVn4vY4Ynel7k9FzqtOIw= +go.opencensus.io v0.22.5/go.mod h1:5pWMHQbX5EPX2/62yrJeAkowc+lfs/XD7Uxpq3pI6kk= +go.opencensus.io v0.23.0 h1:gqCw0LfLxScz8irSi8exQc7fyQ0fKQU/qnC/X8+V/1M= +go.opencensus.io v0.23.0/go.mod h1:XItmlyltB5F7CS4xOC1DcqMoFqwtC6OG2xF7mCv7P7E= +go.opentelemetry.io/contrib/instrumentation/net/http/otelhttp v0.28.0/go.mod h1:Ihno+mNBfZlT0Qot3XyRTdZ/9U/Cg2Pfgj75DTdIfq4= +go.opentelemetry.io/otel v1.3.0/go.mod h1:PWIKzi6JCp7sM0k9yZ43VX+T345uNbAkDKwHVjb2PTs= +go.opentelemetry.io/otel/exporters/otlp/internal/retry v1.3.0/go.mod h1:VpP4/RMn8bv8gNo9uK7/IMY4mtWLELsS+JIP0inH0h4= +go.opentelemetry.io/otel/exporters/otlp/otlptrace v1.3.0/go.mod h1:hO1KLR7jcKaDDKDkvI9dP/FIhpmna5lkqPUQdEjFAM8= +go.opentelemetry.io/otel/exporters/otlp/otlptrace/otlptracegrpc v1.3.0/go.mod h1:keUU7UfnwWTWpJ+FWnyqmogPa82nuU5VUANFq49hlMY= +go.opentelemetry.io/otel/internal/metric v0.26.0/go.mod h1:CbBP6AxKynRs3QCbhklyLUtpfzbqCLiafV9oY2Zj1Jk= +go.opentelemetry.io/otel/metric v0.26.0/go.mod h1:c6YL0fhRo4YVoNs6GoByzUgBp36hBL523rECoZA5UWg= +go.opentelemetry.io/otel/sdk v1.3.0/go.mod h1:rIo4suHNhQwBIPg9axF8V9CA72Wz2mKF1teNrup8yzs= +go.opentelemetry.io/otel/trace v1.3.0/go.mod h1:c/VDhno8888bvQYmbYLqe41/Ldmr/KKunbvWM4/fEjk= +go.opentelemetry.io/proto/otlp v0.7.0/go.mod h1:PqfVotwruBrMGOCsRd/89rSnXhoiJIqeYNgFYFoEGnI= +go.opentelemetry.io/proto/otlp v0.11.0/go.mod h1:QpEjXPrNQzrFDZgoTo49dgHR9RYRSrg3NAKnUGl9YpQ= +go.uber.org/atomic v1.7.0/go.mod h1:fEN4uk6kAWBTFdckzkM89CLk9XfWZrxpCo0nPH17wJc= +go.uber.org/automaxprocs v1.4.0/go.mod h1:/mTEdr7LvHhs0v7mjdxDreTz1OG5zdZGqgOnhWiR/+Q= +go.uber.org/goleak v1.1.12/go.mod h1:cwTWslyiVhfpKIDGSZEM2HlOvcqm+tG4zioyIeLoqMQ= +go.uber.org/multierr v1.6.0/go.mod h1:cdWPpRnG4AhwMwsgIHip0KRBQjJy5kYEpYjJxpXp9iU= +go.uber.org/zap v1.17.0/go.mod h1:MXVU+bhUf/A7Xi2HNOnopQOrmycQ5Ih87HtOu4q5SSo= golang.org/x/crypto v0.0.0-20180904163835-0709b304e793/go.mod h1:6SG95UA2DQfeDnfUPMdvaQW0Q7yPrPDi9nlGo2tz2b4= +golang.org/x/crypto v0.0.0-20181029021203-45a5f77698d3/go.mod h1:6SG95UA2DQfeDnfUPMdvaQW0Q7yPrPDi9nlGo2tz2b4= +golang.org/x/crypto v0.0.0-20181203042331-505ab145d0a9/go.mod h1:6SG95UA2DQfeDnfUPMdvaQW0Q7yPrPDi9nlGo2tz2b4= golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w= golang.org/x/crypto v0.0.0-20190510104115-cbcb75029529/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= golang.org/x/crypto v0.0.0-20190605123033-f99c8df09eb5/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= +golang.org/x/crypto v0.0.0-20190820162420-60c769a6c586/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= +golang.org/x/crypto v0.0.0-20190923035154-9ee001bba392/go.mod h1:/lpIB1dKB+9EgE3H3cr1v9wB50oz8l4C4h62xy7jSTY= golang.org/x/crypto v0.0.0-20191011191535-87dc89f01550/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= golang.org/x/crypto v0.0.0-20200622213623-75b288015ac9/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto= +golang.org/x/crypto v0.0.0-20210817164053-32db794688a5/go.mod h1:GvvjBRRGRdwPK5ydBHafDWAxML/pGHZbMvKqRZ5+Abc= golang.org/x/exp v0.0.0-20190121172915-509febef88a4/go.mod h1:CJ0aWSM057203Lf6IL+f9T1iT9GByDxfZKAQTCR3kQA= golang.org/x/exp v0.0.0-20190306152737-a1d7652674e8/go.mod h1:CJ0aWSM057203Lf6IL+f9T1iT9GByDxfZKAQTCR3kQA= golang.org/x/exp v0.0.0-20190510132918-efd6b22b2522/go.mod h1:ZjyILWgesfNpC6sMxTJOJm9Kp84zZh5NQWvqDGG3Qr8= @@ -234,6 +502,8 @@ golang.org/x/lint v0.0.0-20190930215403-16217165b5de/go.mod h1:6SW0HCj/g11FgYtHl golang.org/x/lint v0.0.0-20191125180803-fdd1cda4f05f/go.mod h1:5qLYkcX4OjUUV8bRuDixDT3tpyyb+LUpUlRWLxfhWrs= golang.org/x/lint v0.0.0-20200130185559-910be7a94367/go.mod h1:3xt1FjdF8hUf6vQPIChWIBhFzV8gjjsPE/fR3IyQdNY= golang.org/x/lint v0.0.0-20200302205851-738671d3881b/go.mod h1:3xt1FjdF8hUf6vQPIChWIBhFzV8gjjsPE/fR3IyQdNY= +golang.org/x/lint v0.0.0-20201208152925-83fdc39ff7b5/go.mod h1:3xt1FjdF8hUf6vQPIChWIBhFzV8gjjsPE/fR3IyQdNY= +golang.org/x/lint v0.0.0-20210508222113-6edffad5e616/go.mod h1:3xt1FjdF8hUf6vQPIChWIBhFzV8gjjsPE/fR3IyQdNY= golang.org/x/mobile v0.0.0-20190312151609-d3739f865fa6/go.mod h1:z+o9i4GpDbdi3rU15maQ/Ox0txvL9dWGYEHz965HBQE= golang.org/x/mobile v0.0.0-20190719004257-d2bd2a29d028/go.mod h1:E/iHnbuqvinMTCcRqshq8CkpyQDoeVncDDYHnLhea+o= golang.org/x/mod v0.0.0-20190513183733-4bf6d317e70e/go.mod h1:mXi4GBBbnImb6dmsKGUJ2LatrhH/nqhxcFungHvyanc= @@ -242,8 +512,13 @@ golang.org/x/mod v0.1.1-0.20191105210325-c90efee705ee/go.mod h1:QqPTAvyqsEbceGzB golang.org/x/mod v0.1.1-0.20191107180719-034126e5016b/go.mod h1:QqPTAvyqsEbceGzBzNggFXnrqF1CaUcvgkdR5Ot7KZg= golang.org/x/mod v0.2.0/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA= golang.org/x/mod v0.3.0/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA= +golang.org/x/mod v0.4.0/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA= +golang.org/x/mod v0.4.1/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA= +golang.org/x/mod v0.4.2/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA= +golang.org/x/mod v0.5.0/go.mod h1:5OXOZSfqPIIbmVBIIKWRFfZjPR0E5r58TLhUjH0a2Ro= golang.org/x/net v0.0.0-20180724234803-3673e40ba225/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= golang.org/x/net v0.0.0-20180826012351-8a410e7b638d/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= +golang.org/x/net v0.0.0-20181023162649-9b4f9f5ad519/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= golang.org/x/net v0.0.0-20181114220301-adae6a3d119a/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= golang.org/x/net v0.0.0-20190108225652-1e06a53dbb7e/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= golang.org/x/net v0.0.0-20190213061140-3a22650c66bd/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= @@ -256,6 +531,7 @@ golang.org/x/net v0.0.0-20190613194153-d28f0bde5980/go.mod h1:z5CRVTTTmAJ677TzLL golang.org/x/net v0.0.0-20190620200207-3b0461eec859/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= golang.org/x/net v0.0.0-20190628185345-da137c7871d7/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= golang.org/x/net v0.0.0-20190724013045-ca1201d0de80/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= +golang.org/x/net v0.0.0-20190923162816-aa69164e4478/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= golang.org/x/net v0.0.0-20191209160850-c0dbc17a3553/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= golang.org/x/net v0.0.0-20200114155413-6afb5195e5aa/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= golang.org/x/net v0.0.0-20200202094626-16171245cfb2/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= @@ -270,15 +546,39 @@ golang.org/x/net v0.0.0-20200520182314-0ba52f642ac2/go.mod h1:qpuaurCH72eLCgpAm/ golang.org/x/net v0.0.0-20200625001655-4c5254603344/go.mod h1:/O7V0waA8r7cgGh81Ro3o1hOxt32SMVPicZroKQ2sZA= golang.org/x/net v0.0.0-20200707034311-ab3426394381/go.mod h1:/O7V0waA8r7cgGh81Ro3o1hOxt32SMVPicZroKQ2sZA= golang.org/x/net v0.0.0-20200822124328-c89045814202/go.mod h1:/O7V0waA8r7cgGh81Ro3o1hOxt32SMVPicZroKQ2sZA= +golang.org/x/net v0.0.0-20201021035429-f5854403a974/go.mod h1:sp8m0HH+o8qH0wwXwYZr8TS3Oi6o0r6Gce1SSxlDquU= +golang.org/x/net v0.0.0-20201031054903-ff519b6c9102/go.mod h1:sp8m0HH+o8qH0wwXwYZr8TS3Oi6o0r6Gce1SSxlDquU= +golang.org/x/net v0.0.0-20201110031124-69a78807bb2b/go.mod h1:sp8m0HH+o8qH0wwXwYZr8TS3Oi6o0r6Gce1SSxlDquU= +golang.org/x/net v0.0.0-20201209123823-ac852fbbde11/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg= +golang.org/x/net v0.0.0-20210119194325-5f4716e94777/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg= +golang.org/x/net v0.0.0-20210226172049-e18ecbb05110/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg= +golang.org/x/net v0.0.0-20210316092652-d523dce5a7f4/go.mod h1:RBQZq4jEuRlivfhVLdyRGr576XBO4/greRjx4P4O3yc= +golang.org/x/net v0.0.0-20210405180319-a5a99cb37ef4/go.mod h1:p54w0d4576C0XHj96bSt6lcn1PtDYWL6XObtHCRCNQM= +golang.org/x/net v0.0.0-20210410081132-afb366fc7cd1/go.mod h1:9tjilg8BloeKEkVJvy7fQ90B1CfIiPueXVOjqfkSzI8= +golang.org/x/net v0.0.0-20210503060351-7fd8e65b6420/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y= golang.org/x/net v0.0.0-20210525063256-abc453219eb5/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y= +golang.org/x/net v0.0.0-20210813160813-60bc85c4be6d/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y= +golang.org/x/net v0.0.0-20211111083644-e5c967477495/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y= golang.org/x/net v0.0.0-20220127200216-cd36cc0744dd/go.mod h1:CfG3xpIq0wQ8r1q4Su4UZFWDARRcnwPjda9FqA0JpMk= +golang.org/x/net v0.0.0-20220225172249-27dd8689420f h1:oA4XRj0qtSt8Yo1Zms0CUlsT3KG69V2UGQWPBxujDmc= golang.org/x/net v0.0.0-20220225172249-27dd8689420f/go.mod h1:CfG3xpIq0wQ8r1q4Su4UZFWDARRcnwPjda9FqA0JpMk= golang.org/x/oauth2 v0.0.0-20180821212333-d2e6202438be/go.mod h1:N/0e6XlmueqKjAGxoOufVs8QHGRruUQn6yWY3a++T0U= golang.org/x/oauth2 v0.0.0-20190226205417-e64efc72b421/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw= golang.org/x/oauth2 v0.0.0-20190604053449-0f29369cfe45/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw= golang.org/x/oauth2 v0.0.0-20191202225959-858c2ad4c8b6/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw= golang.org/x/oauth2 v0.0.0-20200107190931-bf48bf16ab8d/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw= +golang.org/x/oauth2 v0.0.0-20200902213428-5d25da1a8d43/go.mod h1:KelEdhl1UZF7XfJ4dDtk6s++YSgaE7mD/BuKKDLBl4A= +golang.org/x/oauth2 v0.0.0-20201109201403-9fd604954f58/go.mod h1:KelEdhl1UZF7XfJ4dDtk6s++YSgaE7mD/BuKKDLBl4A= +golang.org/x/oauth2 v0.0.0-20201208152858-08078c50e5b5/go.mod h1:KelEdhl1UZF7XfJ4dDtk6s++YSgaE7mD/BuKKDLBl4A= +golang.org/x/oauth2 v0.0.0-20210218202405-ba52d332ba99/go.mod h1:KelEdhl1UZF7XfJ4dDtk6s++YSgaE7mD/BuKKDLBl4A= +golang.org/x/oauth2 v0.0.0-20210220000619-9bb904979d93/go.mod h1:KelEdhl1UZF7XfJ4dDtk6s++YSgaE7mD/BuKKDLBl4A= +golang.org/x/oauth2 v0.0.0-20210313182246-cd4f82c27b84/go.mod h1:KelEdhl1UZF7XfJ4dDtk6s++YSgaE7mD/BuKKDLBl4A= golang.org/x/oauth2 v0.0.0-20210514164344-f6687ab2804c/go.mod h1:KelEdhl1UZF7XfJ4dDtk6s++YSgaE7mD/BuKKDLBl4A= +golang.org/x/oauth2 v0.0.0-20210628180205-a41e5a781914/go.mod h1:KelEdhl1UZF7XfJ4dDtk6s++YSgaE7mD/BuKKDLBl4A= +golang.org/x/oauth2 v0.0.0-20210805134026-6f1e6394065a/go.mod h1:KelEdhl1UZF7XfJ4dDtk6s++YSgaE7mD/BuKKDLBl4A= +golang.org/x/oauth2 v0.0.0-20210819190943-2bc19b11175f/go.mod h1:KelEdhl1UZF7XfJ4dDtk6s++YSgaE7mD/BuKKDLBl4A= +golang.org/x/oauth2 v0.0.0-20211005180243-6b3c2da341f1/go.mod h1:KelEdhl1UZF7XfJ4dDtk6s++YSgaE7mD/BuKKDLBl4A= +golang.org/x/oauth2 v0.0.0-20211104180415-d3ed0bb246c8/go.mod h1:KelEdhl1UZF7XfJ4dDtk6s++YSgaE7mD/BuKKDLBl4A= golang.org/x/oauth2 v0.0.0-20220223155221-ee480838109b/go.mod h1:DAh4E804XQdzx2j+YRIaUnCqCV2RuMz24cGBJ5QYIrc= golang.org/x/sync v0.0.0-20180314180146-1d60e4601c6f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= golang.org/x/sync v0.0.0-20181108010431-42b317875d0f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= @@ -288,11 +588,17 @@ golang.org/x/sync v0.0.0-20190423024810-112230192c58/go.mod h1:RxMgew5VJxzue5/jJ golang.org/x/sync v0.0.0-20190911185100-cd5d95a43a6e/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= golang.org/x/sync v0.0.0-20200317015054-43a5402ce75a/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= golang.org/x/sync v0.0.0-20200625203802-6e8e738ad208/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= +golang.org/x/sync v0.0.0-20201020160332-67f06af15bc9/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= golang.org/x/sync v0.0.0-20201207232520-09787c993a3a/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= +golang.org/x/sync v0.0.0-20210220032951-036812b2e83c/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= +golang.org/x/sys v0.0.0-20180823144017-11551d06cbcc/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= golang.org/x/sys v0.0.0-20180830151530-49385e6e1522/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= golang.org/x/sys v0.0.0-20180905080454-ebe1bf3edb33/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= +golang.org/x/sys v0.0.0-20181026203630-95b1ffbd15a5/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= golang.org/x/sys v0.0.0-20181116152217-5ac8a444bdc5/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= +golang.org/x/sys v0.0.0-20181205085412-a5c9d58dba9a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= +golang.org/x/sys v0.0.0-20190222072716-a9d3bda3a223/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= golang.org/x/sys v0.0.0-20190312061237-fead79001313/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20190412213103-97732733099d/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20190422165155-953cdadca894/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= @@ -301,12 +607,18 @@ golang.org/x/sys v0.0.0-20190507160741-ecd444e8653b/go.mod h1:h1NjWce9XRLGQEsW7w golang.org/x/sys v0.0.0-20190606165138-5da285871e9c/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20190624142023-c5567b49c5d0/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20190726091711-fc99dfbffb4e/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20190922100055-0a153f010e69/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20190924154521-2837fb4f24fe/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20191001151750-bb3f8db39f24/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20191008105621-543471e840be/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20191026070338-33540a1f6037/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20191204072324-ce4227a45e2e/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20191228213918-04cbcbbfeed8/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20200106162015-b016eb3dc98e/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20200113162924-86b910548bc1/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20200116001909-b77594299b42/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20200122134326-e047566fdf82/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20200124204421-9fbb57f87de9/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20200202164722-d101bd2416d5/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20200212091648-12a6c2dcc1e4/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20200223170610-d5e6a3e2c0ae/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= @@ -320,15 +632,39 @@ golang.org/x/sys v0.0.0-20200523222454-059865788121/go.mod h1:h1NjWce9XRLGQEsW7w golang.org/x/sys v0.0.0-20200615200032-f1bc736245b1/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20200625212154-ddb9806d33ae/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20200803210538-64077c9b5642/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20200905004654-be1d3432aa8f/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20200930185726-fdedc70b468f/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20201119102817-f84b799fce68/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20201201145000-ef89a241ccb3/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20210104204734-6f8348627aad/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20210119212857-b64e53b001e4/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20210124154548-22da62e12c0c/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20210220050731-9a76102bfb43/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20210303074136-134d130e1a04/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20210305230114-8fe3ee5dd75b/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20210315160823-c6e025ad8005/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20210320140829-1e4c9ba3b0c4/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20210330210617-4fbd30eecc44/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20210403161142-5e06dd20ab57/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20210423082822-04245dca01da/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20210423185535-09eb48e85fd7/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20210510120138-977fb7262007/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.0.0-20210514084401-e8d321eab015/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20210603081109-ebe580a85c40/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.0.0-20210603125802-9665404d3644/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20210615035016-665e8c7367d1/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.0.0-20210616094352-59db8d763f22/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.0.0-20210630005230-0f9fa26af87c/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.0.0-20210806184541-e5e7981a1069/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.0.0-20210816183151-1e6c022a8912/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.0.0-20210823070655-63515b42dcdf/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.0.0-20210908233432-aa78b53d3365/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.0.0-20210927094055-39ccf1dd6fa6/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.0.0-20211007075335-d3039528d8ac/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.0.0-20211124211545-fe61309f8881/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.0.0-20211205182925-97ca703d548d/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20211216021012-1d35b9e2eb4e/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20220114195835-da31bd327af9/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= -golang.org/x/sys v0.0.0-20220520151302-bc2c85ada10a h1:dGzPydgVsqGcTRVwiLJ1jVbufYwmzD3LfVPLKsKg+0k= -golang.org/x/sys v0.0.0-20220520151302-bc2c85ada10a/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20221013171732-95e765b1cc43 h1:OK7RB6t2WQX54srQQYSXMW8dF5C6/8+oA/s5QBmmto4= golang.org/x/sys v0.0.0-20221013171732-95e765b1cc43/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo= @@ -338,11 +674,14 @@ golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= golang.org/x/text v0.3.1-0.20180807135948-17ff2d5776d2/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= golang.org/x/text v0.3.2/go.mod h1:bEr9sfX3Q8Zfm5fL9x+3itogRgK3+ptLWKqgva+5dAk= golang.org/x/text v0.3.3/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= +golang.org/x/text v0.3.4/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= +golang.org/x/text v0.3.5/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= golang.org/x/text v0.3.6/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= golang.org/x/text v0.3.7/go.mod h1:u+2+/6zg+i71rQMx5EYifcz6MCKuco9NR6JIITiCfzQ= golang.org/x/time v0.0.0-20181108054448-85acf8d2951c/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ= golang.org/x/time v0.0.0-20190308202827-9d24e82272b4/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ= golang.org/x/time v0.0.0-20191024005414-555d28b269f0/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ= +golang.org/x/time v0.0.0-20210220033141-f8bda1e9f3ba/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ= golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= golang.org/x/tools v0.0.0-20190114222345-bf090417da8b/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= golang.org/x/tools v0.0.0-20190226205152-f727befe758c/go.mod h1:9Yl7xja0Znq3iFh3HoIrodX9oNMXvdceNzlUR8zjMvY= @@ -356,6 +695,7 @@ golang.org/x/tools v0.0.0-20190606124116-d0a3d012864b/go.mod h1:/rFqwRUd4F7ZHNgw golang.org/x/tools v0.0.0-20190621195816-6e04913cbbac/go.mod h1:/rFqwRUd4F7ZHNgwSSTFct+R/Kf4OFW1sUzUTQQTgfc= golang.org/x/tools v0.0.0-20190628153133-6cdbf07be9d0/go.mod h1:/rFqwRUd4F7ZHNgwSSTFct+R/Kf4OFW1sUzUTQQTgfc= golang.org/x/tools v0.0.0-20190816200558-6889da9d5479/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= +golang.org/x/tools v0.0.0-20190907020128-2ca718005c18/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= golang.org/x/tools v0.0.0-20190911174233-4f2ddba30aff/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= golang.org/x/tools v0.0.0-20191012152004-8de300cfc20a/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= golang.org/x/tools v0.0.0-20191113191852-77e3bb0ad9e7/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= @@ -380,9 +720,22 @@ golang.org/x/tools v0.0.0-20200501065659-ab2804fb9c9d/go.mod h1:EkVYQZoAsY45+roY golang.org/x/tools v0.0.0-20200512131952-2bc93b1c0c88/go.mod h1:EkVYQZoAsY45+roYkvgYkIh4xh/qjgUK9TdY2XT94GE= golang.org/x/tools v0.0.0-20200515010526-7d3b6ebf133d/go.mod h1:EkVYQZoAsY45+roYkvgYkIh4xh/qjgUK9TdY2XT94GE= golang.org/x/tools v0.0.0-20200618134242-20370b0cb4b2/go.mod h1:EkVYQZoAsY45+roYkvgYkIh4xh/qjgUK9TdY2XT94GE= +golang.org/x/tools v0.0.0-20200619180055-7c47624df98f/go.mod h1:EkVYQZoAsY45+roYkvgYkIh4xh/qjgUK9TdY2XT94GE= golang.org/x/tools v0.0.0-20200729194436-6467de6f59a7/go.mod h1:njjCfa9FT2d7l9Bc6FUM5FLjQPp3cFF28FI3qnDFljA= golang.org/x/tools v0.0.0-20200804011535-6c149bb5ef0d/go.mod h1:njjCfa9FT2d7l9Bc6FUM5FLjQPp3cFF28FI3qnDFljA= golang.org/x/tools v0.0.0-20200825202427-b303f430e36d/go.mod h1:njjCfa9FT2d7l9Bc6FUM5FLjQPp3cFF28FI3qnDFljA= +golang.org/x/tools v0.0.0-20200904185747-39188db58858/go.mod h1:Cj7w3i3Rnn0Xh82ur9kSqwfTHTeVxaDqrfMjpcNT6bE= +golang.org/x/tools v0.0.0-20201110124207-079ba7bd75cd/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA= +golang.org/x/tools v0.0.0-20201201161351-ac6f37ff4c2a/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA= +golang.org/x/tools v0.0.0-20201208233053-a543418bbed2/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA= +golang.org/x/tools v0.0.0-20210105154028-b0ab187a4818/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA= +golang.org/x/tools v0.0.0-20210106214847-113979e3529a/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA= +golang.org/x/tools v0.1.0/go.mod h1:xkSsbof2nBLbhDlRMhhhyNLN/zl3eTqcnHD5viDpcZ0= +golang.org/x/tools v0.1.1/go.mod h1:o0xws9oXOQQZyjljx8fwUC0k7L1pTE6eaCbjGeHmOkk= +golang.org/x/tools v0.1.2/go.mod h1:o0xws9oXOQQZyjljx8fwUC0k7L1pTE6eaCbjGeHmOkk= +golang.org/x/tools v0.1.3/go.mod h1:o0xws9oXOQQZyjljx8fwUC0k7L1pTE6eaCbjGeHmOkk= +golang.org/x/tools v0.1.4/go.mod h1:o0xws9oXOQQZyjljx8fwUC0k7L1pTE6eaCbjGeHmOkk= +golang.org/x/tools v0.1.5/go.mod h1:o0xws9oXOQQZyjljx8fwUC0k7L1pTE6eaCbjGeHmOkk= golang.org/x/xerrors v0.0.0-20190717185122-a985d3407aa7/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= golang.org/x/xerrors v0.0.0-20191011141410-1b5146add898/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= golang.org/x/xerrors v0.0.0-20191204190536-9bdfabe68543/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= @@ -403,12 +756,29 @@ google.golang.org/api v0.24.0/go.mod h1:lIXQywCXRcnZPGlsd8NbLnOjtAoL6em04bJ9+z0M google.golang.org/api v0.28.0/go.mod h1:lIXQywCXRcnZPGlsd8NbLnOjtAoL6em04bJ9+z0MncE= google.golang.org/api v0.29.0/go.mod h1:Lcubydp8VUV7KeIHD9z2Bys/sm/vGKnG1UHuDBSrHWM= google.golang.org/api v0.30.0/go.mod h1:QGmEvQ87FHZNiUVJkT14jQNYJ4ZJjdRF23ZXz5138Fc= +google.golang.org/api v0.35.0/go.mod h1:/XrVsuzM0rZmrsbjJutiuftIzeuTQcEeaYcSk/mQ1dg= +google.golang.org/api v0.36.0/go.mod h1:+z5ficQTmoYpPn8LCUNVpK5I7hwkpjbcgqA7I34qYtE= +google.golang.org/api v0.40.0/go.mod h1:fYKFpnQN0DsDSKRVRcQSDQNtqWPfM9i+zNPxepjRCQ8= +google.golang.org/api v0.41.0/go.mod h1:RkxM5lITDfTzmyKFPt+wGrCJbVfniCr2ool8kTBzRTU= +google.golang.org/api v0.43.0/go.mod h1:nQsDGjRXMo4lvh5hP0TKqF244gqhGcr/YSIykhUk/94= +google.golang.org/api v0.47.0/go.mod h1:Wbvgpq1HddcWVtzsVLyfLp8lDg6AA241LmgIL59tHXo= +google.golang.org/api v0.48.0/go.mod h1:71Pr1vy+TAZRPkPs/xlCf5SsU8WjuAWv1Pfjbtukyy4= +google.golang.org/api v0.50.0/go.mod h1:4bNT5pAuq5ji4SRZm+5QIkjny9JAyVD/3gaSihNefaw= +google.golang.org/api v0.51.0/go.mod h1:t4HdrdoNgyN5cbEfm7Lum0lcLDLiise1F8qDKX00sOU= +google.golang.org/api v0.54.0/go.mod h1:7C4bFFOvVDGXjfDTAsgGwDgAxRDeQ4X8NvUedIt6z3k= +google.golang.org/api v0.55.0/go.mod h1:38yMfeP1kfjsl8isn0tliTjIb1rJXcQi4UXlbqivdVE= +google.golang.org/api v0.56.0/go.mod h1:38yMfeP1kfjsl8isn0tliTjIb1rJXcQi4UXlbqivdVE= +google.golang.org/api v0.57.0/go.mod h1:dVPlbZyBo2/OjBpmvNdpn2GRm6rPy75jyU7bmhdrMgI= +google.golang.org/api v0.59.0/go.mod h1:sT2boj7M9YJxZzgeZqXogmhfmRWDtPzT31xkieUbuZU= +google.golang.org/api v0.61.0/go.mod h1:xQRti5UdCmoCEqFxcz93fTl338AVqDgyaDRuOZ3hg9I= +google.golang.org/api v0.62.0/go.mod h1:dKmwPCydfsad4qCH08MSdgWjfHOyfpd4VtDGgRFdavw= google.golang.org/appengine v1.1.0/go.mod h1:EbEs0AVv82hx2wNQdGPgUI5lhzA/G0D9YwlJXL52JkM= google.golang.org/appengine v1.4.0/go.mod h1:xpcJRLb0r/rnEns0DIKYYv+WjYCduHsrkT7/EB5XEv4= google.golang.org/appengine v1.5.0/go.mod h1:xpcJRLb0r/rnEns0DIKYYv+WjYCduHsrkT7/EB5XEv4= google.golang.org/appengine v1.6.1/go.mod h1:i06prIuMbXzDqacNJfV5OdTW448YApPu5ww/cMBSeb0= google.golang.org/appengine v1.6.5/go.mod h1:8WjMMxjGQR8xUklV/ARdw2HLXBOI7O7uCIDZVag1xfc= google.golang.org/appengine v1.6.6/go.mod h1:8WjMMxjGQR8xUklV/ARdw2HLXBOI7O7uCIDZVag1xfc= +google.golang.org/appengine v1.6.7/go.mod h1:8WjMMxjGQR8xUklV/ARdw2HLXBOI7O7uCIDZVag1xfc= google.golang.org/genproto v0.0.0-20180817151627-c66870c02cf8/go.mod h1:JiN7NxoALGmiZfu7CAH4rXhgtRTLTxftemlI0sWmxmc= google.golang.org/genproto v0.0.0-20190307195333-5fe7a883aa19/go.mod h1:VzzqZJRnGkLBvHegQrXjBqPurQTc5/KpmUdxsrq26oE= google.golang.org/genproto v0.0.0-20190418145605-e7d98fc518a7/go.mod h1:VzzqZJRnGkLBvHegQrXjBqPurQTc5/KpmUdxsrq26oE= @@ -432,12 +802,46 @@ google.golang.org/genproto v0.0.0-20200312145019-da6875a35672/go.mod h1:55QSHmfG google.golang.org/genproto v0.0.0-20200331122359-1ee6d9798940/go.mod h1:55QSHmfGQM9UVYDPBsyGGes0y52j32PQ3BqQfXhyH3c= google.golang.org/genproto v0.0.0-20200430143042-b979b6f78d84/go.mod h1:55QSHmfGQM9UVYDPBsyGGes0y52j32PQ3BqQfXhyH3c= google.golang.org/genproto v0.0.0-20200511104702-f5ebc3bea380/go.mod h1:55QSHmfGQM9UVYDPBsyGGes0y52j32PQ3BqQfXhyH3c= +google.golang.org/genproto v0.0.0-20200513103714-09dca8ec2884/go.mod h1:55QSHmfGQM9UVYDPBsyGGes0y52j32PQ3BqQfXhyH3c= google.golang.org/genproto v0.0.0-20200515170657-fc4c6c6a6587/go.mod h1:YsZOwe1myG/8QRHRsmBRE1LrgQY60beZKjly0O1fX9U= google.golang.org/genproto v0.0.0-20200526211855-cb27e3aa2013/go.mod h1:NbSheEEYHJ7i3ixzK3sjbqSGDJWnxyFXZblF3eUsNvo= google.golang.org/genproto v0.0.0-20200618031413-b414f8b61790/go.mod h1:jDfRM7FcilCzHH/e9qn6dsT145K34l5v+OpcnNgKAAA= google.golang.org/genproto v0.0.0-20200729003335-053ba62fc06f/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no= google.golang.org/genproto v0.0.0-20200804131852-c06518451d9c/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no= google.golang.org/genproto v0.0.0-20200825200019-8632dd797987/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no= +google.golang.org/genproto v0.0.0-20200904004341-0bd0a958aa1d/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no= +google.golang.org/genproto v0.0.0-20201109203340-2640f1f9cdfb/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no= +google.golang.org/genproto v0.0.0-20201201144952-b05cb90ed32e/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no= +google.golang.org/genproto v0.0.0-20201210142538-e3217bee35cc/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no= +google.golang.org/genproto v0.0.0-20201214200347-8c77b98c765d/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no= +google.golang.org/genproto v0.0.0-20210222152913-aa3ee6e6a81c/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no= +google.golang.org/genproto v0.0.0-20210303154014-9728d6b83eeb/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no= +google.golang.org/genproto v0.0.0-20210310155132-4ce2db91004e/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no= +google.golang.org/genproto v0.0.0-20210319143718-93e7006c17a6/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no= +google.golang.org/genproto v0.0.0-20210402141018-6c239bbf2bb1/go.mod h1:9lPAdzaEmUacj36I+k7YKbEc5CXzPIeORRgDAUOu28A= +google.golang.org/genproto v0.0.0-20210513213006-bf773b8c8384/go.mod h1:P3QM42oQyzQSnHPnZ/vqoCdDmzH28fzWByN9asMeM8A= +google.golang.org/genproto v0.0.0-20210602131652-f16073e35f0c/go.mod h1:UODoCrxHCcBojKKwX1terBiRUaqAsFqJiF615XL43r0= +google.golang.org/genproto v0.0.0-20210604141403-392c879c8b08/go.mod h1:UODoCrxHCcBojKKwX1terBiRUaqAsFqJiF615XL43r0= +google.golang.org/genproto v0.0.0-20210608205507-b6d2f5bf0d7d/go.mod h1:UODoCrxHCcBojKKwX1terBiRUaqAsFqJiF615XL43r0= +google.golang.org/genproto v0.0.0-20210624195500-8bfb893ecb84/go.mod h1:SzzZ/N+nwJDaO1kznhnlzqS8ocJICar6hYhVyhi++24= +google.golang.org/genproto v0.0.0-20210713002101-d411969a0d9a/go.mod h1:AxrInvYm1dci+enl5hChSFPOmmUF1+uAa/UsgNRWd7k= +google.golang.org/genproto v0.0.0-20210716133855-ce7ef5c701ea/go.mod h1:AxrInvYm1dci+enl5hChSFPOmmUF1+uAa/UsgNRWd7k= +google.golang.org/genproto v0.0.0-20210728212813-7823e685a01f/go.mod h1:ob2IJxKrgPT52GcgX759i1sleT07tiKowYBGbczaW48= +google.golang.org/genproto v0.0.0-20210805201207-89edb61ffb67/go.mod h1:ob2IJxKrgPT52GcgX759i1sleT07tiKowYBGbczaW48= +google.golang.org/genproto v0.0.0-20210813162853-db860fec028c/go.mod h1:cFeNkxwySK631ADgubI+/XFU/xp8FD5KIVV4rj8UC5w= +google.golang.org/genproto v0.0.0-20210821163610-241b8fcbd6c8/go.mod h1:eFjDcFEctNawg4eG61bRv87N7iHBWyVhJu7u1kqDUXY= +google.golang.org/genproto v0.0.0-20210828152312-66f60bf46e71/go.mod h1:eFjDcFEctNawg4eG61bRv87N7iHBWyVhJu7u1kqDUXY= +google.golang.org/genproto v0.0.0-20210831024726-fe130286e0e2/go.mod h1:eFjDcFEctNawg4eG61bRv87N7iHBWyVhJu7u1kqDUXY= +google.golang.org/genproto v0.0.0-20210903162649-d08c68adba83/go.mod h1:eFjDcFEctNawg4eG61bRv87N7iHBWyVhJu7u1kqDUXY= +google.golang.org/genproto v0.0.0-20210909211513-a8c4777a87af/go.mod h1:eFjDcFEctNawg4eG61bRv87N7iHBWyVhJu7u1kqDUXY= +google.golang.org/genproto v0.0.0-20210924002016-3dee208752a0/go.mod h1:5CzLGKJ67TSI2B9POpiiyGha0AjJvZIUgRMt1dSmuhc= +google.golang.org/genproto v0.0.0-20211008145708-270636b82663/go.mod h1:5CzLGKJ67TSI2B9POpiiyGha0AjJvZIUgRMt1dSmuhc= +google.golang.org/genproto v0.0.0-20211028162531-8db9c33dc351/go.mod h1:5CzLGKJ67TSI2B9POpiiyGha0AjJvZIUgRMt1dSmuhc= +google.golang.org/genproto v0.0.0-20211118181313-81c1377c94b1/go.mod h1:5CzLGKJ67TSI2B9POpiiyGha0AjJvZIUgRMt1dSmuhc= +google.golang.org/genproto v0.0.0-20211129164237-f09f9a12af12/go.mod h1:5CzLGKJ67TSI2B9POpiiyGha0AjJvZIUgRMt1dSmuhc= +google.golang.org/genproto v0.0.0-20211203200212-54befc351ae9/go.mod h1:5CzLGKJ67TSI2B9POpiiyGha0AjJvZIUgRMt1dSmuhc= +google.golang.org/genproto v0.0.0-20211206160659-862468c7d6e0/go.mod h1:5CzLGKJ67TSI2B9POpiiyGha0AjJvZIUgRMt1dSmuhc= +google.golang.org/genproto v0.0.0-20211208223120-3a66f561d7aa/go.mod h1:5CzLGKJ67TSI2B9POpiiyGha0AjJvZIUgRMt1dSmuhc= google.golang.org/grpc v1.19.0/go.mod h1:mqu4LbDTu4XGKhr4mRzUsmM4RtVoemTSY81AxZiDr8c= google.golang.org/grpc v1.20.1/go.mod h1:10oTOabMzJvdu6/UiuZezV6QK5dSlG84ov/aaiqXj38= google.golang.org/grpc v1.21.1/go.mod h1:oYelfM1adQP15Ek0mdvEgi9Df8B9CZIaU1084ijfRaM= @@ -450,6 +854,23 @@ google.golang.org/grpc v1.28.0/go.mod h1:rpkK4SK4GF4Ach/+MFLZUBavHOvF2JJB5uozKKa google.golang.org/grpc v1.29.1/go.mod h1:itym6AZVZYACWQqET3MqgPpjcuV5QH3BxFS3IjizoKk= google.golang.org/grpc v1.30.0/go.mod h1:N36X2cJ7JwdamYAgDz+s+rVMFjt3numwzf/HckM8pak= google.golang.org/grpc v1.31.0/go.mod h1:N36X2cJ7JwdamYAgDz+s+rVMFjt3numwzf/HckM8pak= +google.golang.org/grpc v1.31.1/go.mod h1:N36X2cJ7JwdamYAgDz+s+rVMFjt3numwzf/HckM8pak= +google.golang.org/grpc v1.33.1/go.mod h1:fr5YgcSWrqhRRxogOsw7RzIpsmvOZ6IcH4kBYTpR3n0= +google.golang.org/grpc v1.33.2/go.mod h1:JMHMWHQWaTccqQQlmk3MJZS+GWXOdAesneDmEnv2fbc= +google.golang.org/grpc v1.34.0/go.mod h1:WotjhfgOW/POjDeRt8vscBtXq+2VjORFy659qA51WJ8= +google.golang.org/grpc v1.35.0/go.mod h1:qjiiYl8FncCW8feJPdyg3v6XW24KsRHe+dy9BAGRRjU= +google.golang.org/grpc v1.36.0/go.mod h1:qjiiYl8FncCW8feJPdyg3v6XW24KsRHe+dy9BAGRRjU= +google.golang.org/grpc v1.36.1/go.mod h1:qjiiYl8FncCW8feJPdyg3v6XW24KsRHe+dy9BAGRRjU= +google.golang.org/grpc v1.37.0/go.mod h1:NREThFqKR1f3iQ6oBuvc5LadQuXVGo9rkm5ZGrQdJfM= +google.golang.org/grpc v1.37.1/go.mod h1:NREThFqKR1f3iQ6oBuvc5LadQuXVGo9rkm5ZGrQdJfM= +google.golang.org/grpc v1.38.0/go.mod h1:NREThFqKR1f3iQ6oBuvc5LadQuXVGo9rkm5ZGrQdJfM= +google.golang.org/grpc v1.39.0/go.mod h1:PImNr+rS9TWYb2O4/emRugxiyHZ5JyHW5F+RPnDzfrE= +google.golang.org/grpc v1.39.1/go.mod h1:PImNr+rS9TWYb2O4/emRugxiyHZ5JyHW5F+RPnDzfrE= +google.golang.org/grpc v1.40.0/go.mod h1:ogyxbiOoUXAkP+4+xa6PZSE9DZgIHtSpzjDTB9KAK34= +google.golang.org/grpc v1.40.1/go.mod h1:ogyxbiOoUXAkP+4+xa6PZSE9DZgIHtSpzjDTB9KAK34= +google.golang.org/grpc v1.42.0/go.mod h1:k+4IHHFw41K8+bbowsex27ge2rCb65oeWqe4jJ590SU= +google.golang.org/grpc v1.43.0/go.mod h1:k+4IHHFw41K8+bbowsex27ge2rCb65oeWqe4jJ590SU= +google.golang.org/grpc/cmd/protoc-gen-go-grpc v1.1.0/go.mod h1:6Kw0yEErY5E/yWrBtf03jp27GLLJujG4z/JK95pnjjw= google.golang.org/protobuf v0.0.0-20200109180630-ec00e32a8dfd/go.mod h1:DFci5gLYBciE7Vtevhsrf46CRTquxDuWsQurQQe4oz8= google.golang.org/protobuf v0.0.0-20200221191635-4d8936d0db64/go.mod h1:kwYJMbMJ01Woi6D6+Kah6886xMZcty6N08ah7+eCXa0= google.golang.org/protobuf v0.0.0-20200228230310-ab0ca4ff8a60/go.mod h1:cfTl7dwQJ+fmap5saPgwCLgHXTUD7jkjRqWcaiX5VyM= @@ -462,19 +883,29 @@ google.golang.org/protobuf v1.24.0/go.mod h1:r/3tXBNzIEhYS9I1OUVjXDlt8tc493IdKGj google.golang.org/protobuf v1.25.0/go.mod h1:9JNX74DMeImyA3h4bdi1ymwjUzf21/xIlbajtzgsN7c= google.golang.org/protobuf v1.26.0-rc.1/go.mod h1:jlhhOSvTdKEhbULTjvd4ARK9grFBp09yW+WbY/TyQbw= google.golang.org/protobuf v1.26.0/go.mod h1:9q0QmTI4eRPtz6boOQmLYwt+qCgq0jsYwAQnmE0givc= +google.golang.org/protobuf v1.27.1/go.mod h1:9q0QmTI4eRPtz6boOQmLYwt+qCgq0jsYwAQnmE0givc= google.golang.org/protobuf v1.28.1 h1:d0NfwRgPtno5B1Wa6L2DAG+KivqkdutMf1UhdNx175w= google.golang.org/protobuf v1.28.1/go.mod h1:HV8QOd/L58Z+nl8r43ehVNZIU/HEI6OcFqwMG9pJV4I= gopkg.in/alecthomas/kingpin.v2 v2.2.6/go.mod h1:FMv+mEhP44yOT+4EoQTLFTRgOQ1FBLkstjWtayDeSgw= gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= gopkg.in/check.v1 v1.0.0-20180628173108-788fd7840127/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= gopkg.in/check.v1 v1.0.0-20190902080502-41f04d3bba15/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= +gopkg.in/check.v1 v1.0.0-20200227125254-8fa46927fb4f h1:BLraFXnmrev5lT+xlilqcH8XK9/i0At2xKjWk4p6zsU= +gopkg.in/check.v1 v1.0.0-20200227125254-8fa46927fb4f/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= gopkg.in/errgo.v2 v2.1.0/go.mod h1:hNsd1EY+bozCKY1Ytp96fpM3vjJbqLJn88ws8XvfDNI= +gopkg.in/ini.v1 v1.66.2/go.mod h1:pNLf8WUiyNEtQjuu5G5vTm06TEv9tsIgeAvK8hOrP4k= gopkg.in/yaml.v2 v2.2.1/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= gopkg.in/yaml.v2 v2.2.2/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= +gopkg.in/yaml.v2 v2.2.3/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= gopkg.in/yaml.v2 v2.2.4/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= gopkg.in/yaml.v2 v2.2.5/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= +gopkg.in/yaml.v2 v2.2.8/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= gopkg.in/yaml.v2 v2.3.0/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= +gopkg.in/yaml.v2 v2.4.0 h1:D8xgwECY7CYvx+Y2n4sBz93Jn9JRvxdiyyo8CTfuKaY= gopkg.in/yaml.v2 v2.4.0/go.mod h1:RDklbk79AGWmwhnvt/jBztapEOGDOx6ZbXqjP6csGnQ= +gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= +gopkg.in/yaml.v3 v3.0.0-20210107192922-496545a6307b h1:h8qDotaEPuJATrMmW04NCwg7v22aHH28wwpauUhK9Oo= +gopkg.in/yaml.v3 v3.0.0-20210107192922-496545a6307b/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= honnef.co/go/tools v0.0.0-20190102054323-c2f93a96b099/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4= honnef.co/go/tools v0.0.0-20190106161140-3f1c8253044a/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4= honnef.co/go/tools v0.0.0-20190418001031-e561f6794a2a/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4= diff --git a/main.go b/main.go index 0e8602e8..db1b67ea 100644 --- a/main.go +++ b/main.go @@ -23,6 +23,7 @@ import ( units "github.com/docker/go-units" limiter "github.com/openfaas/faas-middleware/concurrency-limiter" + "github.com/openfaas/of-watchdog/auth" "github.com/openfaas/of-watchdog/config" "github.com/openfaas/of-watchdog/executor" "github.com/openfaas/of-watchdog/metrics" @@ -73,6 +74,26 @@ func main() { // of the request mode. baseFunctionHandler := buildRequestHandler(watchdogConfig, watchdogConfig.PrefixLogs) requestHandler := baseFunctionHandler + // add middlewares here + policyPath := os.Getenv("OPA_POLICY") + if policyPath != "" { + log.Printf("loading OPA policy(s): %q", policyPath) + // TODO: support remote authorizer if it starts with http:// or https:// + authorizer, err := auth.NewAuthorizer(policyPath) + if err != nil { + fmt.Fprintf(os.Stderr, "%s", err.Error()) + os.Exit(1) + } + + inputCfg, err := auth.InputConfigFromEnv() + if err != nil { + fmt.Fprintf(os.Stderr, "%s", err.Error()) + os.Exit(1) + } + + authMiddleware := auth.New(authorizer, inputCfg) + requestHandler = authMiddleware(requestHandler) + } var limit limiter.Limiter if watchdogConfig.MaxInflight > 0 { diff --git a/vendor/github.com/OneOfOne/xxhash/.gitignore b/vendor/github.com/OneOfOne/xxhash/.gitignore new file mode 100644 index 00000000..f4faa7f8 --- /dev/null +++ b/vendor/github.com/OneOfOne/xxhash/.gitignore @@ -0,0 +1,4 @@ +*.txt +*.pprof +cmap2/ +cache/ diff --git a/vendor/github.com/OneOfOne/xxhash/.travis.yml b/vendor/github.com/OneOfOne/xxhash/.travis.yml new file mode 100644 index 00000000..1c6dc55b --- /dev/null +++ b/vendor/github.com/OneOfOne/xxhash/.travis.yml @@ -0,0 +1,13 @@ +language: go +sudo: false + +go: + - "1.10" + - "1.11" + - "1.12" + - master + +script: + - go test -tags safe ./... + - go test ./... + - diff --git a/vendor/github.com/OneOfOne/xxhash/LICENSE b/vendor/github.com/OneOfOne/xxhash/LICENSE new file mode 100644 index 00000000..9e30b4f3 --- /dev/null +++ b/vendor/github.com/OneOfOne/xxhash/LICENSE @@ -0,0 +1,187 @@ + Apache License + Version 2.0, January 2004 + http://www.apache.org/licenses/ + + TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION + + 1. Definitions. + + "License" shall mean the terms and conditions for use, reproduction, + and distribution as defined by Sections 1 through 9 of this document. + + "Licensor" shall mean the copyright owner or entity authorized by + the copyright owner that is granting the License. + + "Legal Entity" shall mean the union of the acting entity and all + other entities that control, are controlled by, or are under common + control with that entity. For the purposes of this definition, + "control" means (i) the power, direct or indirect, to cause the + direction or management of such entity, whether by contract or + otherwise, or (ii) ownership of fifty percent (50%) or more of the + outstanding shares, or (iii) beneficial ownership of such entity. + + "You" (or "Your") shall mean an individual or Legal Entity + exercising permissions granted by this License. + + "Source" form shall mean the preferred form for making modifications, + including but not limited to software source code, documentation + source, and configuration files. + + "Object" form shall mean any form resulting from mechanical + transformation or translation of a Source form, including but + not limited to compiled object code, generated documentation, + and conversions to other media types. + + "Work" shall mean the work of authorship, whether in Source or + Object form, made available under the License, as indicated by a + copyright notice that is included in or attached to the work + (an example is provided in the Appendix below). + + "Derivative Works" shall mean any work, whether in Source or Object + form, that is based on (or derived from) the Work and for which the + editorial revisions, annotations, elaborations, or other modifications + represent, as a whole, an original work of authorship. For the purposes + of this License, Derivative Works shall not include works that remain + separable from, or merely link (or bind by name) to the interfaces of, + the Work and Derivative Works thereof. + + "Contribution" shall mean any work of authorship, including + the original version of the Work and any modifications or additions + to that Work or Derivative Works thereof, that is intentionally + submitted to Licensor for inclusion in the Work by the copyright owner + or by an individual or Legal Entity authorized to submit on behalf of + the copyright owner. For the purposes of this definition, "submitted" + means any form of electronic, verbal, or written communication sent + to the Licensor or its representatives, including but not limited to + communication on electronic mailing lists, source code control systems, + and issue tracking systems that are managed by, or on behalf of, the + Licensor for the purpose of discussing and improving the Work, but + excluding communication that is conspicuously marked or otherwise + designated in writing by the copyright owner as "Not a Contribution." + + "Contributor" shall mean Licensor and any individual or Legal Entity + on behalf of whom a Contribution has been received by Licensor and + subsequently incorporated within the Work. + + 2. Grant of Copyright License. Subject to the terms and conditions of + this License, each Contributor hereby grants to You a perpetual, + worldwide, non-exclusive, no-charge, royalty-free, irrevocable + copyright license to reproduce, prepare Derivative Works of, + publicly display, publicly perform, sublicense, and distribute the + Work and such Derivative Works in Source or Object form. + + 3. Grant of Patent License. Subject to the terms and conditions of + this License, each Contributor hereby grants to You a perpetual, + worldwide, non-exclusive, no-charge, royalty-free, irrevocable + (except as stated in this section) patent license to make, have made, + use, offer to sell, sell, import, and otherwise transfer the Work, + where such license applies only to those patent claims licensable + by such Contributor that are necessarily infringed by their + Contribution(s) alone or by combination of their Contribution(s) + with the Work to which such Contribution(s) was submitted. If You + institute patent litigation against any entity (including a + cross-claim or counterclaim in a lawsuit) alleging that the Work + or a Contribution incorporated within the Work constitutes direct + or contributory patent infringement, then any patent licenses + granted to You under this License for that Work shall terminate + as of the date such litigation is filed. + + 4. Redistribution. You may reproduce and distribute copies of the + Work or Derivative Works thereof in any medium, with or without + modifications, and in Source or Object form, provided that You + meet the following conditions: + + (a) You must give any other recipients of the Work or + Derivative Works a copy of this License; and + + (b) You must cause any modified files to carry prominent notices + stating that You changed the files; and + + (c) You must retain, in the Source form of any Derivative Works + that You distribute, all copyright, patent, trademark, and + attribution notices from the Source form of the Work, + excluding those notices that do not pertain to any part of + the Derivative Works; and + + (d) If the Work includes a "NOTICE" text file as part of its + distribution, then any Derivative Works that You distribute must + include a readable copy of the attribution notices contained + within such NOTICE file, excluding those notices that do not + pertain to any part of the Derivative Works, in at least one + of the following places: within a NOTICE text file distributed + as part of the Derivative Works; within the Source form or + documentation, if provided along with the Derivative Works; or, + within a display generated by the Derivative Works, if and + wherever such third-party notices normally appear. The contents + of the NOTICE file are for informational purposes only and + do not modify the License. You may add Your own attribution + notices within Derivative Works that You distribute, alongside + or as an addendum to the NOTICE text from the Work, provided + that such additional attribution notices cannot be construed + as modifying the License. + + You may add Your own copyright statement to Your modifications and + may provide additional or different license terms and conditions + for use, reproduction, or distribution of Your modifications, or + for any such Derivative Works as a whole, provided Your use, + reproduction, and distribution of the Work otherwise complies with + the conditions stated in this License. + + 5. Submission of Contributions. Unless You explicitly state otherwise, + any Contribution intentionally submitted for inclusion in the Work + by You to the Licensor shall be under the terms and conditions of + this License, without any additional terms or conditions. + Notwithstanding the above, nothing herein shall supersede or modify + the terms of any separate license agreement you may have executed + with Licensor regarding such Contributions. + + 6. Trademarks. This License does not grant permission to use the trade + names, trademarks, service marks, or product names of the Licensor, + except as required for reasonable and customary use in describing the + origin of the Work and reproducing the content of the NOTICE file. + + 7. Disclaimer of Warranty. Unless required by applicable law or + agreed to in writing, Licensor provides the Work (and each + Contributor provides its Contributions) on an "AS IS" BASIS, + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or + implied, including, without limitation, any warranties or conditions + of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A + PARTICULAR PURPOSE. You are solely responsible for determining the + appropriateness of using or redistributing the Work and assume any + risks associated with Your exercise of permissions under this License. + + 8. Limitation of Liability. In no event and under no legal theory, + whether in tort (including negligence), contract, or otherwise, + unless required by applicable law (such as deliberate and grossly + negligent acts) or agreed to in writing, shall any Contributor be + liable to You for damages, including any direct, indirect, special, + incidental, or consequential damages of any character arising as a + result of this License or out of the use or inability to use the + Work (including but not limited to damages for loss of goodwill, + work stoppage, computer failure or malfunction, or any and all + other commercial damages or losses), even if such Contributor + has been advised of the possibility of such damages. + + 9. Accepting Warranty or Additional Liability. While redistributing + the Work or Derivative Works thereof, You may choose to offer, + and charge a fee for, acceptance of support, warranty, indemnity, + or other liability obligations and/or rights consistent with this + License. However, in accepting such obligations, You may act only + on Your own behalf and on Your sole responsibility, not on behalf + of any other Contributor, and only if You agree to indemnify, + defend, and hold each Contributor harmless for any liability + incurred by, or claims asserted against, such Contributor by reason + of your accepting any such warranty or additional liability. + + END OF TERMS AND CONDITIONS + + APPENDIX: How to apply the Apache License to your work. + + To apply the Apache License to your work, attach the following + boilerplate notice, with the fields enclosed by brackets "{}" + replaced with your own identifying information. (Don't include + the brackets!) The text should be enclosed in the appropriate + comment syntax for the file format. We also recommend that a + file or class name and description of purpose be included on the + same "printed page" as the copyright notice for easier + identification within third-party archives. diff --git a/vendor/github.com/OneOfOne/xxhash/README.md b/vendor/github.com/OneOfOne/xxhash/README.md new file mode 100644 index 00000000..8eea28c3 --- /dev/null +++ b/vendor/github.com/OneOfOne/xxhash/README.md @@ -0,0 +1,74 @@ +# xxhash [![GoDoc](https://godoc.org/github.com/OneOfOne/xxhash?status.svg)](https://godoc.org/github.com/OneOfOne/xxhash) [![Build Status](https://travis-ci.org/OneOfOne/xxhash.svg?branch=master)](https://travis-ci.org/OneOfOne/xxhash) [![Coverage](https://gocover.io/_badge/github.com/OneOfOne/xxhash)](https://gocover.io/github.com/OneOfOne/xxhash) + +This is a native Go implementation of the excellent [xxhash](https://github.com/Cyan4973/xxHash)* algorithm, an extremely fast non-cryptographic Hash algorithm, working at speeds close to RAM limits. + +* The C implementation is ([Copyright](https://github.com/Cyan4973/xxHash/blob/master/LICENSE) (c) 2012-2014, Yann Collet) + +## Install + + go get github.com/OneOfOne/xxhash + +## Features + +* On Go 1.7+ the pure go version is faster than CGO for all inputs. +* Supports ChecksumString{32,64} xxhash{32,64}.WriteString, which uses no copies when it can, falls back to copy on appengine. +* The native version falls back to a less optimized version on appengine due to the lack of unsafe. +* Almost as fast as the mostly pure assembly version written by the brilliant [cespare](https://github.com/cespare/xxhash), while also supporting seeds. +* To manually toggle the appengine version build with `-tags safe`. + +## Benchmark + +### Core i7-4790 @ 3.60GHz, Linux 4.12.6-1-ARCH (64bit), Go tip (+ff90f4af66 2017-08-19) + +```bash +➤ go test -bench '64' -count 5 -tags cespare | benchstat /dev/stdin +name time/op + +# https://github.com/cespare/xxhash +XXSum64Cespare/Func-8 160ns ± 2% +XXSum64Cespare/Struct-8 173ns ± 1% +XXSum64ShortCespare/Func-8 6.78ns ± 1% +XXSum64ShortCespare/Struct-8 19.6ns ± 2% + +# this package (default mode, using unsafe) +XXSum64/Func-8 170ns ± 1% +XXSum64/Struct-8 182ns ± 1% +XXSum64Short/Func-8 13.5ns ± 3% +XXSum64Short/Struct-8 20.4ns ± 0% + +# this package (appengine, *not* using unsafe) +XXSum64/Func-8 241ns ± 5% +XXSum64/Struct-8 243ns ± 6% +XXSum64Short/Func-8 15.2ns ± 2% +XXSum64Short/Struct-8 23.7ns ± 5% + +CRC64ISO-8 1.23µs ± 1% +CRC64ISOString-8 2.71µs ± 4% +CRC64ISOShort-8 22.2ns ± 3% + +Fnv64-8 2.34µs ± 1% +Fnv64Short-8 74.7ns ± 8% +``` + +## Usage + +```go + h := xxhash.New64() + // r, err := os.Open("......") + // defer f.Close() + r := strings.NewReader(F) + io.Copy(h, r) + fmt.Println("xxhash.Backend:", xxhash.Backend) + fmt.Println("File checksum:", h.Sum64()) +``` + +[playground](https://play.golang.org/p/wHKBwfu6CPV) + +## TODO + +* Rewrite the 32bit version to be more optimized. +* General cleanup as the Go inliner gets smarter. + +## License + +This project is released under the Apache v2. license. See [LICENSE](LICENSE) for more details. diff --git a/vendor/github.com/OneOfOne/xxhash/xxhash.go b/vendor/github.com/OneOfOne/xxhash/xxhash.go new file mode 100644 index 00000000..af2496b7 --- /dev/null +++ b/vendor/github.com/OneOfOne/xxhash/xxhash.go @@ -0,0 +1,294 @@ +package xxhash + +import ( + "encoding/binary" + "errors" + "hash" +) + +const ( + prime32x1 uint32 = 2654435761 + prime32x2 uint32 = 2246822519 + prime32x3 uint32 = 3266489917 + prime32x4 uint32 = 668265263 + prime32x5 uint32 = 374761393 + + prime64x1 uint64 = 11400714785074694791 + prime64x2 uint64 = 14029467366897019727 + prime64x3 uint64 = 1609587929392839161 + prime64x4 uint64 = 9650029242287828579 + prime64x5 uint64 = 2870177450012600261 + + maxInt32 int32 = (1<<31 - 1) + + // precomputed zero Vs for seed 0 + zero64x1 = 0x60ea27eeadc0b5d6 + zero64x2 = 0xc2b2ae3d27d4eb4f + zero64x3 = 0x0 + zero64x4 = 0x61c8864e7a143579 +) + +const ( + magic32 = "xxh\x07" + magic64 = "xxh\x08" + marshaled32Size = len(magic32) + 4*7 + 16 + marshaled64Size = len(magic64) + 8*6 + 32 + 1 +) + +func NewHash32() hash.Hash { return New32() } +func NewHash64() hash.Hash { return New64() } + +// Checksum32 returns the checksum of the input data with the seed set to 0. +func Checksum32(in []byte) uint32 { + return Checksum32S(in, 0) +} + +// ChecksumString32 returns the checksum of the input data, without creating a copy, with the seed set to 0. +func ChecksumString32(s string) uint32 { + return ChecksumString32S(s, 0) +} + +type XXHash32 struct { + mem [16]byte + ln, memIdx int32 + v1, v2, v3, v4 uint32 + seed uint32 +} + +// Size returns the number of bytes Sum will return. +func (xx *XXHash32) Size() int { + return 4 +} + +// BlockSize returns the hash's underlying block size. +// The Write method must be able to accept any amount +// of data, but it may operate more efficiently if all writes +// are a multiple of the block size. +func (xx *XXHash32) BlockSize() int { + return 16 +} + +// NewS32 creates a new hash.Hash32 computing the 32bit xxHash checksum starting with the specific seed. +func NewS32(seed uint32) (xx *XXHash32) { + xx = &XXHash32{ + seed: seed, + } + xx.Reset() + return +} + +// New32 creates a new hash.Hash32 computing the 32bit xxHash checksum starting with the seed set to 0. +func New32() *XXHash32 { + return NewS32(0) +} + +func (xx *XXHash32) Reset() { + xx.v1 = xx.seed + prime32x1 + prime32x2 + xx.v2 = xx.seed + prime32x2 + xx.v3 = xx.seed + xx.v4 = xx.seed - prime32x1 + xx.ln, xx.memIdx = 0, 0 +} + +// Sum appends the current hash to b and returns the resulting slice. +// It does not change the underlying hash state. +func (xx *XXHash32) Sum(in []byte) []byte { + s := xx.Sum32() + return append(in, byte(s>>24), byte(s>>16), byte(s>>8), byte(s)) +} + +// MarshalBinary implements the encoding.BinaryMarshaler interface. +func (xx *XXHash32) MarshalBinary() ([]byte, error) { + b := make([]byte, 0, marshaled32Size) + b = append(b, magic32...) + b = appendUint32(b, xx.v1) + b = appendUint32(b, xx.v2) + b = appendUint32(b, xx.v3) + b = appendUint32(b, xx.v4) + b = appendUint32(b, xx.seed) + b = appendInt32(b, xx.ln) + b = appendInt32(b, xx.memIdx) + b = append(b, xx.mem[:]...) + return b, nil +} + +// UnmarshalBinary implements the encoding.BinaryUnmarshaler interface. +func (xx *XXHash32) UnmarshalBinary(b []byte) error { + if len(b) < len(magic32) || string(b[:len(magic32)]) != magic32 { + return errors.New("xxhash: invalid hash state identifier") + } + if len(b) != marshaled32Size { + return errors.New("xxhash: invalid hash state size") + } + b = b[len(magic32):] + b, xx.v1 = consumeUint32(b) + b, xx.v2 = consumeUint32(b) + b, xx.v3 = consumeUint32(b) + b, xx.v4 = consumeUint32(b) + b, xx.seed = consumeUint32(b) + b, xx.ln = consumeInt32(b) + b, xx.memIdx = consumeInt32(b) + copy(xx.mem[:], b) + return nil +} + +// Checksum64 an alias for Checksum64S(in, 0) +func Checksum64(in []byte) uint64 { + return Checksum64S(in, 0) +} + +// ChecksumString64 returns the checksum of the input data, without creating a copy, with the seed set to 0. +func ChecksumString64(s string) uint64 { + return ChecksumString64S(s, 0) +} + +type XXHash64 struct { + v1, v2, v3, v4 uint64 + seed uint64 + ln uint64 + mem [32]byte + memIdx int8 +} + +// Size returns the number of bytes Sum will return. +func (xx *XXHash64) Size() int { + return 8 +} + +// BlockSize returns the hash's underlying block size. +// The Write method must be able to accept any amount +// of data, but it may operate more efficiently if all writes +// are a multiple of the block size. +func (xx *XXHash64) BlockSize() int { + return 32 +} + +// NewS64 creates a new hash.Hash64 computing the 64bit xxHash checksum starting with the specific seed. +func NewS64(seed uint64) (xx *XXHash64) { + xx = &XXHash64{ + seed: seed, + } + xx.Reset() + return +} + +// New64 creates a new hash.Hash64 computing the 64bit xxHash checksum starting with the seed set to 0x0. +func New64() *XXHash64 { + return NewS64(0) +} + +func (xx *XXHash64) Reset() { + xx.ln, xx.memIdx = 0, 0 + xx.v1, xx.v2, xx.v3, xx.v4 = resetVs64(xx.seed) +} + +// Sum appends the current hash to b and returns the resulting slice. +// It does not change the underlying hash state. +func (xx *XXHash64) Sum(in []byte) []byte { + s := xx.Sum64() + return append(in, byte(s>>56), byte(s>>48), byte(s>>40), byte(s>>32), byte(s>>24), byte(s>>16), byte(s>>8), byte(s)) +} + +// MarshalBinary implements the encoding.BinaryMarshaler interface. +func (xx *XXHash64) MarshalBinary() ([]byte, error) { + b := make([]byte, 0, marshaled64Size) + b = append(b, magic64...) + b = appendUint64(b, xx.v1) + b = appendUint64(b, xx.v2) + b = appendUint64(b, xx.v3) + b = appendUint64(b, xx.v4) + b = appendUint64(b, xx.seed) + b = appendUint64(b, xx.ln) + b = append(b, byte(xx.memIdx)) + b = append(b, xx.mem[:]...) + return b, nil +} + +// UnmarshalBinary implements the encoding.BinaryUnmarshaler interface. +func (xx *XXHash64) UnmarshalBinary(b []byte) error { + if len(b) < len(magic64) || string(b[:len(magic64)]) != magic64 { + return errors.New("xxhash: invalid hash state identifier") + } + if len(b) != marshaled64Size { + return errors.New("xxhash: invalid hash state size") + } + b = b[len(magic64):] + b, xx.v1 = consumeUint64(b) + b, xx.v2 = consumeUint64(b) + b, xx.v3 = consumeUint64(b) + b, xx.v4 = consumeUint64(b) + b, xx.seed = consumeUint64(b) + b, xx.ln = consumeUint64(b) + xx.memIdx = int8(b[0]) + b = b[1:] + copy(xx.mem[:], b) + return nil +} + +func appendInt32(b []byte, x int32) []byte { return appendUint32(b, uint32(x)) } + +func appendUint32(b []byte, x uint32) []byte { + var a [4]byte + binary.LittleEndian.PutUint32(a[:], x) + return append(b, a[:]...) +} + +func appendUint64(b []byte, x uint64) []byte { + var a [8]byte + binary.LittleEndian.PutUint64(a[:], x) + return append(b, a[:]...) +} + +func consumeInt32(b []byte) ([]byte, int32) { bn, x := consumeUint32(b); return bn, int32(x) } +func consumeUint32(b []byte) ([]byte, uint32) { x := u32(b); return b[4:], x } +func consumeUint64(b []byte) ([]byte, uint64) { x := u64(b); return b[8:], x } + +// force the compiler to use ROTL instructions + +func rotl32_1(x uint32) uint32 { return (x << 1) | (x >> (32 - 1)) } +func rotl32_7(x uint32) uint32 { return (x << 7) | (x >> (32 - 7)) } +func rotl32_11(x uint32) uint32 { return (x << 11) | (x >> (32 - 11)) } +func rotl32_12(x uint32) uint32 { return (x << 12) | (x >> (32 - 12)) } +func rotl32_13(x uint32) uint32 { return (x << 13) | (x >> (32 - 13)) } +func rotl32_17(x uint32) uint32 { return (x << 17) | (x >> (32 - 17)) } +func rotl32_18(x uint32) uint32 { return (x << 18) | (x >> (32 - 18)) } + +func rotl64_1(x uint64) uint64 { return (x << 1) | (x >> (64 - 1)) } +func rotl64_7(x uint64) uint64 { return (x << 7) | (x >> (64 - 7)) } +func rotl64_11(x uint64) uint64 { return (x << 11) | (x >> (64 - 11)) } +func rotl64_12(x uint64) uint64 { return (x << 12) | (x >> (64 - 12)) } +func rotl64_18(x uint64) uint64 { return (x << 18) | (x >> (64 - 18)) } +func rotl64_23(x uint64) uint64 { return (x << 23) | (x >> (64 - 23)) } +func rotl64_27(x uint64) uint64 { return (x << 27) | (x >> (64 - 27)) } +func rotl64_31(x uint64) uint64 { return (x << 31) | (x >> (64 - 31)) } + +func mix64(h uint64) uint64 { + h ^= h >> 33 + h *= prime64x2 + h ^= h >> 29 + h *= prime64x3 + h ^= h >> 32 + return h +} + +func resetVs64(seed uint64) (v1, v2, v3, v4 uint64) { + if seed == 0 { + return zero64x1, zero64x2, zero64x3, zero64x4 + } + return (seed + prime64x1 + prime64x2), (seed + prime64x2), (seed), (seed - prime64x1) +} + +// borrowed from cespare +func round64(h, v uint64) uint64 { + h += v * prime64x2 + h = rotl64_31(h) + h *= prime64x1 + return h +} + +func mergeRound64(h, v uint64) uint64 { + v = round64(0, v) + h ^= v + h = h*prime64x1 + prime64x4 + return h +} diff --git a/vendor/github.com/OneOfOne/xxhash/xxhash_go17.go b/vendor/github.com/OneOfOne/xxhash/xxhash_go17.go new file mode 100644 index 00000000..ae48e0c5 --- /dev/null +++ b/vendor/github.com/OneOfOne/xxhash/xxhash_go17.go @@ -0,0 +1,161 @@ +package xxhash + +func u32(in []byte) uint32 { + return uint32(in[0]) | uint32(in[1])<<8 | uint32(in[2])<<16 | uint32(in[3])<<24 +} + +func u64(in []byte) uint64 { + return uint64(in[0]) | uint64(in[1])<<8 | uint64(in[2])<<16 | uint64(in[3])<<24 | uint64(in[4])<<32 | uint64(in[5])<<40 | uint64(in[6])<<48 | uint64(in[7])<<56 +} + +// Checksum32S returns the checksum of the input bytes with the specific seed. +func Checksum32S(in []byte, seed uint32) (h uint32) { + var i int + + if len(in) > 15 { + var ( + v1 = seed + prime32x1 + prime32x2 + v2 = seed + prime32x2 + v3 = seed + 0 + v4 = seed - prime32x1 + ) + for ; i < len(in)-15; i += 16 { + in := in[i : i+16 : len(in)] + v1 += u32(in[0:4:len(in)]) * prime32x2 + v1 = rotl32_13(v1) * prime32x1 + + v2 += u32(in[4:8:len(in)]) * prime32x2 + v2 = rotl32_13(v2) * prime32x1 + + v3 += u32(in[8:12:len(in)]) * prime32x2 + v3 = rotl32_13(v3) * prime32x1 + + v4 += u32(in[12:16:len(in)]) * prime32x2 + v4 = rotl32_13(v4) * prime32x1 + } + + h = rotl32_1(v1) + rotl32_7(v2) + rotl32_12(v3) + rotl32_18(v4) + + } else { + h = seed + prime32x5 + } + + h += uint32(len(in)) + for ; i <= len(in)-4; i += 4 { + in := in[i : i+4 : len(in)] + h += u32(in[0:4:len(in)]) * prime32x3 + h = rotl32_17(h) * prime32x4 + } + + for ; i < len(in); i++ { + h += uint32(in[i]) * prime32x5 + h = rotl32_11(h) * prime32x1 + } + + h ^= h >> 15 + h *= prime32x2 + h ^= h >> 13 + h *= prime32x3 + h ^= h >> 16 + + return +} + +func (xx *XXHash32) Write(in []byte) (n int, err error) { + i, ml := 0, int(xx.memIdx) + n = len(in) + xx.ln += int32(n) + + if d := 16 - ml; ml > 0 && ml+len(in) > 16 { + xx.memIdx += int32(copy(xx.mem[xx.memIdx:], in[:d])) + ml, in = 16, in[d:len(in):len(in)] + } else if ml+len(in) < 16 { + xx.memIdx += int32(copy(xx.mem[xx.memIdx:], in)) + return + } + + if ml > 0 { + i += 16 - ml + xx.memIdx += int32(copy(xx.mem[xx.memIdx:len(xx.mem):len(xx.mem)], in)) + in := xx.mem[:16:len(xx.mem)] + + xx.v1 += u32(in[0:4:len(in)]) * prime32x2 + xx.v1 = rotl32_13(xx.v1) * prime32x1 + + xx.v2 += u32(in[4:8:len(in)]) * prime32x2 + xx.v2 = rotl32_13(xx.v2) * prime32x1 + + xx.v3 += u32(in[8:12:len(in)]) * prime32x2 + xx.v3 = rotl32_13(xx.v3) * prime32x1 + + xx.v4 += u32(in[12:16:len(in)]) * prime32x2 + xx.v4 = rotl32_13(xx.v4) * prime32x1 + + xx.memIdx = 0 + } + + for ; i <= len(in)-16; i += 16 { + in := in[i : i+16 : len(in)] + xx.v1 += u32(in[0:4:len(in)]) * prime32x2 + xx.v1 = rotl32_13(xx.v1) * prime32x1 + + xx.v2 += u32(in[4:8:len(in)]) * prime32x2 + xx.v2 = rotl32_13(xx.v2) * prime32x1 + + xx.v3 += u32(in[8:12:len(in)]) * prime32x2 + xx.v3 = rotl32_13(xx.v3) * prime32x1 + + xx.v4 += u32(in[12:16:len(in)]) * prime32x2 + xx.v4 = rotl32_13(xx.v4) * prime32x1 + } + + if len(in)-i != 0 { + xx.memIdx += int32(copy(xx.mem[xx.memIdx:], in[i:len(in):len(in)])) + } + + return +} + +func (xx *XXHash32) Sum32() (h uint32) { + var i int32 + if xx.ln > 15 { + h = rotl32_1(xx.v1) + rotl32_7(xx.v2) + rotl32_12(xx.v3) + rotl32_18(xx.v4) + } else { + h = xx.seed + prime32x5 + } + + h += uint32(xx.ln) + + if xx.memIdx > 0 { + for ; i < xx.memIdx-3; i += 4 { + in := xx.mem[i : i+4 : len(xx.mem)] + h += u32(in[0:4:len(in)]) * prime32x3 + h = rotl32_17(h) * prime32x4 + } + + for ; i < xx.memIdx; i++ { + h += uint32(xx.mem[i]) * prime32x5 + h = rotl32_11(h) * prime32x1 + } + } + h ^= h >> 15 + h *= prime32x2 + h ^= h >> 13 + h *= prime32x3 + h ^= h >> 16 + + return +} + +// Checksum64S returns the 64bit xxhash checksum for a single input +func Checksum64S(in []byte, seed uint64) uint64 { + if len(in) == 0 && seed == 0 { + return 0xef46db3751d8e999 + } + + if len(in) > 31 { + return checksum64(in, seed) + } + + return checksum64Short(in, seed) +} diff --git a/vendor/github.com/OneOfOne/xxhash/xxhash_safe.go b/vendor/github.com/OneOfOne/xxhash/xxhash_safe.go new file mode 100644 index 00000000..e92ec29e --- /dev/null +++ b/vendor/github.com/OneOfOne/xxhash/xxhash_safe.go @@ -0,0 +1,183 @@ +// +build appengine safe ppc64le ppc64be mipsle mips s390x + +package xxhash + +// Backend returns the current version of xxhash being used. +const Backend = "GoSafe" + +func ChecksumString32S(s string, seed uint32) uint32 { + return Checksum32S([]byte(s), seed) +} + +func (xx *XXHash32) WriteString(s string) (int, error) { + if len(s) == 0 { + return 0, nil + } + return xx.Write([]byte(s)) +} + +func ChecksumString64S(s string, seed uint64) uint64 { + return Checksum64S([]byte(s), seed) +} + +func (xx *XXHash64) WriteString(s string) (int, error) { + if len(s) == 0 { + return 0, nil + } + return xx.Write([]byte(s)) +} + +func checksum64(in []byte, seed uint64) (h uint64) { + var ( + v1, v2, v3, v4 = resetVs64(seed) + + i int + ) + + for ; i < len(in)-31; i += 32 { + in := in[i : i+32 : len(in)] + v1 = round64(v1, u64(in[0:8:len(in)])) + v2 = round64(v2, u64(in[8:16:len(in)])) + v3 = round64(v3, u64(in[16:24:len(in)])) + v4 = round64(v4, u64(in[24:32:len(in)])) + } + + h = rotl64_1(v1) + rotl64_7(v2) + rotl64_12(v3) + rotl64_18(v4) + + h = mergeRound64(h, v1) + h = mergeRound64(h, v2) + h = mergeRound64(h, v3) + h = mergeRound64(h, v4) + + h += uint64(len(in)) + + for ; i < len(in)-7; i += 8 { + h ^= round64(0, u64(in[i:len(in):len(in)])) + h = rotl64_27(h)*prime64x1 + prime64x4 + } + + for ; i < len(in)-3; i += 4 { + h ^= uint64(u32(in[i:len(in):len(in)])) * prime64x1 + h = rotl64_23(h)*prime64x2 + prime64x3 + } + + for ; i < len(in); i++ { + h ^= uint64(in[i]) * prime64x5 + h = rotl64_11(h) * prime64x1 + } + + return mix64(h) +} + +func checksum64Short(in []byte, seed uint64) uint64 { + var ( + h = seed + prime64x5 + uint64(len(in)) + i int + ) + + for ; i < len(in)-7; i += 8 { + k := u64(in[i : i+8 : len(in)]) + h ^= round64(0, k) + h = rotl64_27(h)*prime64x1 + prime64x4 + } + + for ; i < len(in)-3; i += 4 { + h ^= uint64(u32(in[i:i+4:len(in)])) * prime64x1 + h = rotl64_23(h)*prime64x2 + prime64x3 + } + + for ; i < len(in); i++ { + h ^= uint64(in[i]) * prime64x5 + h = rotl64_11(h) * prime64x1 + } + + return mix64(h) +} + +func (xx *XXHash64) Write(in []byte) (n int, err error) { + var ( + ml = int(xx.memIdx) + d = 32 - ml + ) + + n = len(in) + xx.ln += uint64(n) + + if ml+len(in) < 32 { + xx.memIdx += int8(copy(xx.mem[xx.memIdx:len(xx.mem):len(xx.mem)], in)) + return + } + + i, v1, v2, v3, v4 := 0, xx.v1, xx.v2, xx.v3, xx.v4 + if ml > 0 && ml+len(in) > 32 { + xx.memIdx += int8(copy(xx.mem[xx.memIdx:len(xx.mem):len(xx.mem)], in[:d:len(in)])) + in = in[d:len(in):len(in)] + + in := xx.mem[0:32:len(xx.mem)] + + v1 = round64(v1, u64(in[0:8:len(in)])) + v2 = round64(v2, u64(in[8:16:len(in)])) + v3 = round64(v3, u64(in[16:24:len(in)])) + v4 = round64(v4, u64(in[24:32:len(in)])) + + xx.memIdx = 0 + } + + for ; i < len(in)-31; i += 32 { + in := in[i : i+32 : len(in)] + v1 = round64(v1, u64(in[0:8:len(in)])) + v2 = round64(v2, u64(in[8:16:len(in)])) + v3 = round64(v3, u64(in[16:24:len(in)])) + v4 = round64(v4, u64(in[24:32:len(in)])) + } + + if len(in)-i != 0 { + xx.memIdx += int8(copy(xx.mem[xx.memIdx:], in[i:len(in):len(in)])) + } + + xx.v1, xx.v2, xx.v3, xx.v4 = v1, v2, v3, v4 + + return +} + +func (xx *XXHash64) Sum64() (h uint64) { + var i int + if xx.ln > 31 { + v1, v2, v3, v4 := xx.v1, xx.v2, xx.v3, xx.v4 + h = rotl64_1(v1) + rotl64_7(v2) + rotl64_12(v3) + rotl64_18(v4) + + h = mergeRound64(h, v1) + h = mergeRound64(h, v2) + h = mergeRound64(h, v3) + h = mergeRound64(h, v4) + } else { + h = xx.seed + prime64x5 + } + + h += uint64(xx.ln) + if xx.memIdx > 0 { + in := xx.mem[:xx.memIdx] + for ; i < int(xx.memIdx)-7; i += 8 { + in := in[i : i+8 : len(in)] + k := u64(in[0:8:len(in)]) + k *= prime64x2 + k = rotl64_31(k) + k *= prime64x1 + h ^= k + h = rotl64_27(h)*prime64x1 + prime64x4 + } + + for ; i < int(xx.memIdx)-3; i += 4 { + in := in[i : i+4 : len(in)] + h ^= uint64(u32(in[0:4:len(in)])) * prime64x1 + h = rotl64_23(h)*prime64x2 + prime64x3 + } + + for ; i < int(xx.memIdx); i++ { + h ^= uint64(in[i]) * prime64x5 + h = rotl64_11(h) * prime64x1 + } + } + + return mix64(h) +} diff --git a/vendor/github.com/OneOfOne/xxhash/xxhash_unsafe.go b/vendor/github.com/OneOfOne/xxhash/xxhash_unsafe.go new file mode 100644 index 00000000..1e2b5e8f --- /dev/null +++ b/vendor/github.com/OneOfOne/xxhash/xxhash_unsafe.go @@ -0,0 +1,240 @@ +// +build !safe +// +build !appengine +// +build !ppc64le +// +build !mipsle +// +build !ppc64be +// +build !mips +// +build !s390x + +package xxhash + +import ( + "reflect" + "unsafe" +) + +// Backend returns the current version of xxhash being used. +const Backend = "GoUnsafe" + +// ChecksumString32S returns the checksum of the input data, without creating a copy, with the specific seed. +func ChecksumString32S(s string, seed uint32) uint32 { + if len(s) == 0 { + return Checksum32S(nil, seed) + } + ss := (*reflect.StringHeader)(unsafe.Pointer(&s)) + return Checksum32S((*[maxInt32]byte)(unsafe.Pointer(ss.Data))[:len(s):len(s)], seed) +} + +func (xx *XXHash32) WriteString(s string) (int, error) { + if len(s) == 0 { + return 0, nil + } + + ss := (*reflect.StringHeader)(unsafe.Pointer(&s)) + return xx.Write((*[maxInt32]byte)(unsafe.Pointer(ss.Data))[:len(s):len(s)]) +} + +// ChecksumString64S returns the checksum of the input data, without creating a copy, with the specific seed. +func ChecksumString64S(s string, seed uint64) uint64 { + if len(s) == 0 { + return Checksum64S(nil, seed) + } + + ss := (*reflect.StringHeader)(unsafe.Pointer(&s)) + return Checksum64S((*[maxInt32]byte)(unsafe.Pointer(ss.Data))[:len(s):len(s)], seed) +} + +func (xx *XXHash64) WriteString(s string) (int, error) { + if len(s) == 0 { + return 0, nil + } + ss := (*reflect.StringHeader)(unsafe.Pointer(&s)) + return xx.Write((*[maxInt32]byte)(unsafe.Pointer(ss.Data))[:len(s):len(s)]) +} + +//go:nocheckptr +func checksum64(in []byte, seed uint64) uint64 { + var ( + wordsLen = len(in) >> 3 + words = ((*[maxInt32 / 8]uint64)(unsafe.Pointer(&in[0])))[:wordsLen:wordsLen] + + v1, v2, v3, v4 = resetVs64(seed) + + h uint64 + i int + ) + + for ; i < len(words)-3; i += 4 { + words := (*[4]uint64)(unsafe.Pointer(&words[i])) + + v1 = round64(v1, words[0]) + v2 = round64(v2, words[1]) + v3 = round64(v3, words[2]) + v4 = round64(v4, words[3]) + } + + h = rotl64_1(v1) + rotl64_7(v2) + rotl64_12(v3) + rotl64_18(v4) + + h = mergeRound64(h, v1) + h = mergeRound64(h, v2) + h = mergeRound64(h, v3) + h = mergeRound64(h, v4) + + h += uint64(len(in)) + + for _, k := range words[i:] { + h ^= round64(0, k) + h = rotl64_27(h)*prime64x1 + prime64x4 + } + + if in = in[wordsLen<<3 : len(in) : len(in)]; len(in) > 3 { + words := (*[1]uint32)(unsafe.Pointer(&in[0])) + h ^= uint64(words[0]) * prime64x1 + h = rotl64_23(h)*prime64x2 + prime64x3 + + in = in[4:len(in):len(in)] + } + + for _, b := range in { + h ^= uint64(b) * prime64x5 + h = rotl64_11(h) * prime64x1 + } + + return mix64(h) +} + +//go:nocheckptr +func checksum64Short(in []byte, seed uint64) uint64 { + var ( + h = seed + prime64x5 + uint64(len(in)) + i int + ) + + if len(in) > 7 { + var ( + wordsLen = len(in) >> 3 + words = ((*[maxInt32 / 8]uint64)(unsafe.Pointer(&in[0])))[:wordsLen:wordsLen] + ) + + for i := range words { + h ^= round64(0, words[i]) + h = rotl64_27(h)*prime64x1 + prime64x4 + } + + i = wordsLen << 3 + } + + if in = in[i:len(in):len(in)]; len(in) > 3 { + words := (*[1]uint32)(unsafe.Pointer(&in[0])) + h ^= uint64(words[0]) * prime64x1 + h = rotl64_23(h)*prime64x2 + prime64x3 + + in = in[4:len(in):len(in)] + } + + for _, b := range in { + h ^= uint64(b) * prime64x5 + h = rotl64_11(h) * prime64x1 + } + + return mix64(h) +} + +func (xx *XXHash64) Write(in []byte) (n int, err error) { + mem, idx := xx.mem[:], int(xx.memIdx) + + xx.ln, n = xx.ln+uint64(len(in)), len(in) + + if idx+len(in) < 32 { + xx.memIdx += int8(copy(mem[idx:len(mem):len(mem)], in)) + return + } + + var ( + v1, v2, v3, v4 = xx.v1, xx.v2, xx.v3, xx.v4 + + i int + ) + + if d := 32 - int(idx); d > 0 && int(idx)+len(in) > 31 { + copy(mem[idx:len(mem):len(mem)], in[:len(in):len(in)]) + + words := (*[4]uint64)(unsafe.Pointer(&mem[0])) + + v1 = round64(v1, words[0]) + v2 = round64(v2, words[1]) + v3 = round64(v3, words[2]) + v4 = round64(v4, words[3]) + + if in, xx.memIdx = in[d:len(in):len(in)], 0; len(in) == 0 { + goto RET + } + } + + for ; i < len(in)-31; i += 32 { + words := (*[4]uint64)(unsafe.Pointer(&in[i])) + + v1 = round64(v1, words[0]) + v2 = round64(v2, words[1]) + v3 = round64(v3, words[2]) + v4 = round64(v4, words[3]) + } + + if len(in)-i != 0 { + xx.memIdx += int8(copy(mem[xx.memIdx:len(mem):len(mem)], in[i:len(in):len(in)])) + } + +RET: + xx.v1, xx.v2, xx.v3, xx.v4 = v1, v2, v3, v4 + + return +} + +func (xx *XXHash64) Sum64() (h uint64) { + if seed := xx.seed; xx.ln > 31 { + v1, v2, v3, v4 := xx.v1, xx.v2, xx.v3, xx.v4 + h = rotl64_1(v1) + rotl64_7(v2) + rotl64_12(v3) + rotl64_18(v4) + + h = mergeRound64(h, v1) + h = mergeRound64(h, v2) + h = mergeRound64(h, v3) + h = mergeRound64(h, v4) + } else if seed == 0 { + h = prime64x5 + } else { + h = seed + prime64x5 + } + + h += uint64(xx.ln) + + if xx.memIdx == 0 { + return mix64(h) + } + + var ( + in = xx.mem[:xx.memIdx:xx.memIdx] + wordsLen = len(in) >> 3 + words = ((*[maxInt32 / 8]uint64)(unsafe.Pointer(&in[0])))[:wordsLen:wordsLen] + ) + + for _, k := range words { + h ^= round64(0, k) + h = rotl64_27(h)*prime64x1 + prime64x4 + } + + if in = in[wordsLen<<3 : len(in) : len(in)]; len(in) > 3 { + words := (*[1]uint32)(unsafe.Pointer(&in[0])) + + h ^= uint64(words[0]) * prime64x1 + h = rotl64_23(h)*prime64x2 + prime64x3 + + in = in[4:len(in):len(in)] + } + + for _, b := range in { + h ^= uint64(b) * prime64x5 + h = rotl64_11(h) * prime64x1 + } + + return mix64(h) +} diff --git a/vendor/github.com/ghodss/yaml/.gitignore b/vendor/github.com/ghodss/yaml/.gitignore new file mode 100644 index 00000000..e256a31e --- /dev/null +++ b/vendor/github.com/ghodss/yaml/.gitignore @@ -0,0 +1,20 @@ +# OSX leaves these everywhere on SMB shares +._* + +# Eclipse files +.classpath +.project +.settings/** + +# Emacs save files +*~ + +# Vim-related files +[._]*.s[a-w][a-z] +[._]s[a-w][a-z] +*.un~ +Session.vim +.netrwhist + +# Go test binaries +*.test diff --git a/vendor/github.com/ghodss/yaml/.travis.yml b/vendor/github.com/ghodss/yaml/.travis.yml new file mode 100644 index 00000000..0e9d6edc --- /dev/null +++ b/vendor/github.com/ghodss/yaml/.travis.yml @@ -0,0 +1,7 @@ +language: go +go: + - 1.3 + - 1.4 +script: + - go test + - go build diff --git a/vendor/github.com/ghodss/yaml/LICENSE b/vendor/github.com/ghodss/yaml/LICENSE new file mode 100644 index 00000000..7805d36d --- /dev/null +++ b/vendor/github.com/ghodss/yaml/LICENSE @@ -0,0 +1,50 @@ +The MIT License (MIT) + +Copyright (c) 2014 Sam Ghods + +Permission is hereby granted, free of charge, to any person obtaining a copy +of this software and associated documentation files (the "Software"), to deal +in the Software without restriction, including without limitation the rights +to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the Software is +furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in all +copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE +SOFTWARE. + + +Copyright (c) 2012 The Go Authors. All rights reserved. + +Redistribution and use in source and binary forms, with or without +modification, are permitted provided that the following conditions are +met: + + * Redistributions of source code must retain the above copyright +notice, this list of conditions and the following disclaimer. + * Redistributions in binary form must reproduce the above +copyright notice, this list of conditions and the following disclaimer +in the documentation and/or other materials provided with the +distribution. + * Neither the name of Google Inc. nor the names of its +contributors may be used to endorse or promote products derived from +this software without specific prior written permission. + +THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +"AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. diff --git a/vendor/github.com/ghodss/yaml/README.md b/vendor/github.com/ghodss/yaml/README.md new file mode 100644 index 00000000..0200f75b --- /dev/null +++ b/vendor/github.com/ghodss/yaml/README.md @@ -0,0 +1,121 @@ +# YAML marshaling and unmarshaling support for Go + +[![Build Status](https://travis-ci.org/ghodss/yaml.svg)](https://travis-ci.org/ghodss/yaml) + +## Introduction + +A wrapper around [go-yaml](https://github.com/go-yaml/yaml) designed to enable a better way of handling YAML when marshaling to and from structs. + +In short, this library first converts YAML to JSON using go-yaml and then uses `json.Marshal` and `json.Unmarshal` to convert to or from the struct. This means that it effectively reuses the JSON struct tags as well as the custom JSON methods `MarshalJSON` and `UnmarshalJSON` unlike go-yaml. For a detailed overview of the rationale behind this method, [see this blog post](http://ghodss.com/2014/the-right-way-to-handle-yaml-in-golang/). + +## Compatibility + +This package uses [go-yaml](https://github.com/go-yaml/yaml) and therefore supports [everything go-yaml supports](https://github.com/go-yaml/yaml#compatibility). + +## Caveats + +**Caveat #1:** When using `yaml.Marshal` and `yaml.Unmarshal`, binary data should NOT be preceded with the `!!binary` YAML tag. If you do, go-yaml will convert the binary data from base64 to native binary data, which is not compatible with JSON. You can still use binary in your YAML files though - just store them without the `!!binary` tag and decode the base64 in your code (e.g. in the custom JSON methods `MarshalJSON` and `UnmarshalJSON`). This also has the benefit that your YAML and your JSON binary data will be decoded exactly the same way. As an example: + +``` +BAD: + exampleKey: !!binary gIGC + +GOOD: + exampleKey: gIGC +... and decode the base64 data in your code. +``` + +**Caveat #2:** When using `YAMLToJSON` directly, maps with keys that are maps will result in an error since this is not supported by JSON. This error will occur in `Unmarshal` as well since you can't unmarshal map keys anyways since struct fields can't be keys. + +## Installation and usage + +To install, run: + +``` +$ go get github.com/ghodss/yaml +``` + +And import using: + +``` +import "github.com/ghodss/yaml" +``` + +Usage is very similar to the JSON library: + +```go +package main + +import ( + "fmt" + + "github.com/ghodss/yaml" +) + +type Person struct { + Name string `json:"name"` // Affects YAML field names too. + Age int `json:"age"` +} + +func main() { + // Marshal a Person struct to YAML. + p := Person{"John", 30} + y, err := yaml.Marshal(p) + if err != nil { + fmt.Printf("err: %v\n", err) + return + } + fmt.Println(string(y)) + /* Output: + age: 30 + name: John + */ + + // Unmarshal the YAML back into a Person struct. + var p2 Person + err = yaml.Unmarshal(y, &p2) + if err != nil { + fmt.Printf("err: %v\n", err) + return + } + fmt.Println(p2) + /* Output: + {John 30} + */ +} +``` + +`yaml.YAMLToJSON` and `yaml.JSONToYAML` methods are also available: + +```go +package main + +import ( + "fmt" + + "github.com/ghodss/yaml" +) + +func main() { + j := []byte(`{"name": "John", "age": 30}`) + y, err := yaml.JSONToYAML(j) + if err != nil { + fmt.Printf("err: %v\n", err) + return + } + fmt.Println(string(y)) + /* Output: + name: John + age: 30 + */ + j2, err := yaml.YAMLToJSON(y) + if err != nil { + fmt.Printf("err: %v\n", err) + return + } + fmt.Println(string(j2)) + /* Output: + {"age":30,"name":"John"} + */ +} +``` diff --git a/vendor/github.com/ghodss/yaml/fields.go b/vendor/github.com/ghodss/yaml/fields.go new file mode 100644 index 00000000..58600740 --- /dev/null +++ b/vendor/github.com/ghodss/yaml/fields.go @@ -0,0 +1,501 @@ +// Copyright 2013 The Go Authors. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. +package yaml + +import ( + "bytes" + "encoding" + "encoding/json" + "reflect" + "sort" + "strings" + "sync" + "unicode" + "unicode/utf8" +) + +// indirect walks down v allocating pointers as needed, +// until it gets to a non-pointer. +// if it encounters an Unmarshaler, indirect stops and returns that. +// if decodingNull is true, indirect stops at the last pointer so it can be set to nil. +func indirect(v reflect.Value, decodingNull bool) (json.Unmarshaler, encoding.TextUnmarshaler, reflect.Value) { + // If v is a named type and is addressable, + // start with its address, so that if the type has pointer methods, + // we find them. + if v.Kind() != reflect.Ptr && v.Type().Name() != "" && v.CanAddr() { + v = v.Addr() + } + for { + // Load value from interface, but only if the result will be + // usefully addressable. + if v.Kind() == reflect.Interface && !v.IsNil() { + e := v.Elem() + if e.Kind() == reflect.Ptr && !e.IsNil() && (!decodingNull || e.Elem().Kind() == reflect.Ptr) { + v = e + continue + } + } + + if v.Kind() != reflect.Ptr { + break + } + + if v.Elem().Kind() != reflect.Ptr && decodingNull && v.CanSet() { + break + } + if v.IsNil() { + if v.CanSet() { + v.Set(reflect.New(v.Type().Elem())) + } else { + v = reflect.New(v.Type().Elem()) + } + } + if v.Type().NumMethod() > 0 { + if u, ok := v.Interface().(json.Unmarshaler); ok { + return u, nil, reflect.Value{} + } + if u, ok := v.Interface().(encoding.TextUnmarshaler); ok { + return nil, u, reflect.Value{} + } + } + v = v.Elem() + } + return nil, nil, v +} + +// A field represents a single field found in a struct. +type field struct { + name string + nameBytes []byte // []byte(name) + equalFold func(s, t []byte) bool // bytes.EqualFold or equivalent + + tag bool + index []int + typ reflect.Type + omitEmpty bool + quoted bool +} + +func fillField(f field) field { + f.nameBytes = []byte(f.name) + f.equalFold = foldFunc(f.nameBytes) + return f +} + +// byName sorts field by name, breaking ties with depth, +// then breaking ties with "name came from json tag", then +// breaking ties with index sequence. +type byName []field + +func (x byName) Len() int { return len(x) } + +func (x byName) Swap(i, j int) { x[i], x[j] = x[j], x[i] } + +func (x byName) Less(i, j int) bool { + if x[i].name != x[j].name { + return x[i].name < x[j].name + } + if len(x[i].index) != len(x[j].index) { + return len(x[i].index) < len(x[j].index) + } + if x[i].tag != x[j].tag { + return x[i].tag + } + return byIndex(x).Less(i, j) +} + +// byIndex sorts field by index sequence. +type byIndex []field + +func (x byIndex) Len() int { return len(x) } + +func (x byIndex) Swap(i, j int) { x[i], x[j] = x[j], x[i] } + +func (x byIndex) Less(i, j int) bool { + for k, xik := range x[i].index { + if k >= len(x[j].index) { + return false + } + if xik != x[j].index[k] { + return xik < x[j].index[k] + } + } + return len(x[i].index) < len(x[j].index) +} + +// typeFields returns a list of fields that JSON should recognize for the given type. +// The algorithm is breadth-first search over the set of structs to include - the top struct +// and then any reachable anonymous structs. +func typeFields(t reflect.Type) []field { + // Anonymous fields to explore at the current level and the next. + current := []field{} + next := []field{{typ: t}} + + // Count of queued names for current level and the next. + count := map[reflect.Type]int{} + nextCount := map[reflect.Type]int{} + + // Types already visited at an earlier level. + visited := map[reflect.Type]bool{} + + // Fields found. + var fields []field + + for len(next) > 0 { + current, next = next, current[:0] + count, nextCount = nextCount, map[reflect.Type]int{} + + for _, f := range current { + if visited[f.typ] { + continue + } + visited[f.typ] = true + + // Scan f.typ for fields to include. + for i := 0; i < f.typ.NumField(); i++ { + sf := f.typ.Field(i) + if sf.PkgPath != "" { // unexported + continue + } + tag := sf.Tag.Get("json") + if tag == "-" { + continue + } + name, opts := parseTag(tag) + if !isValidTag(name) { + name = "" + } + index := make([]int, len(f.index)+1) + copy(index, f.index) + index[len(f.index)] = i + + ft := sf.Type + if ft.Name() == "" && ft.Kind() == reflect.Ptr { + // Follow pointer. + ft = ft.Elem() + } + + // Record found field and index sequence. + if name != "" || !sf.Anonymous || ft.Kind() != reflect.Struct { + tagged := name != "" + if name == "" { + name = sf.Name + } + fields = append(fields, fillField(field{ + name: name, + tag: tagged, + index: index, + typ: ft, + omitEmpty: opts.Contains("omitempty"), + quoted: opts.Contains("string"), + })) + if count[f.typ] > 1 { + // If there were multiple instances, add a second, + // so that the annihilation code will see a duplicate. + // It only cares about the distinction between 1 or 2, + // so don't bother generating any more copies. + fields = append(fields, fields[len(fields)-1]) + } + continue + } + + // Record new anonymous struct to explore in next round. + nextCount[ft]++ + if nextCount[ft] == 1 { + next = append(next, fillField(field{name: ft.Name(), index: index, typ: ft})) + } + } + } + } + + sort.Sort(byName(fields)) + + // Delete all fields that are hidden by the Go rules for embedded fields, + // except that fields with JSON tags are promoted. + + // The fields are sorted in primary order of name, secondary order + // of field index length. Loop over names; for each name, delete + // hidden fields by choosing the one dominant field that survives. + out := fields[:0] + for advance, i := 0, 0; i < len(fields); i += advance { + // One iteration per name. + // Find the sequence of fields with the name of this first field. + fi := fields[i] + name := fi.name + for advance = 1; i+advance < len(fields); advance++ { + fj := fields[i+advance] + if fj.name != name { + break + } + } + if advance == 1 { // Only one field with this name + out = append(out, fi) + continue + } + dominant, ok := dominantField(fields[i : i+advance]) + if ok { + out = append(out, dominant) + } + } + + fields = out + sort.Sort(byIndex(fields)) + + return fields +} + +// dominantField looks through the fields, all of which are known to +// have the same name, to find the single field that dominates the +// others using Go's embedding rules, modified by the presence of +// JSON tags. If there are multiple top-level fields, the boolean +// will be false: This condition is an error in Go and we skip all +// the fields. +func dominantField(fields []field) (field, bool) { + // The fields are sorted in increasing index-length order. The winner + // must therefore be one with the shortest index length. Drop all + // longer entries, which is easy: just truncate the slice. + length := len(fields[0].index) + tagged := -1 // Index of first tagged field. + for i, f := range fields { + if len(f.index) > length { + fields = fields[:i] + break + } + if f.tag { + if tagged >= 0 { + // Multiple tagged fields at the same level: conflict. + // Return no field. + return field{}, false + } + tagged = i + } + } + if tagged >= 0 { + return fields[tagged], true + } + // All remaining fields have the same length. If there's more than one, + // we have a conflict (two fields named "X" at the same level) and we + // return no field. + if len(fields) > 1 { + return field{}, false + } + return fields[0], true +} + +var fieldCache struct { + sync.RWMutex + m map[reflect.Type][]field +} + +// cachedTypeFields is like typeFields but uses a cache to avoid repeated work. +func cachedTypeFields(t reflect.Type) []field { + fieldCache.RLock() + f := fieldCache.m[t] + fieldCache.RUnlock() + if f != nil { + return f + } + + // Compute fields without lock. + // Might duplicate effort but won't hold other computations back. + f = typeFields(t) + if f == nil { + f = []field{} + } + + fieldCache.Lock() + if fieldCache.m == nil { + fieldCache.m = map[reflect.Type][]field{} + } + fieldCache.m[t] = f + fieldCache.Unlock() + return f +} + +func isValidTag(s string) bool { + if s == "" { + return false + } + for _, c := range s { + switch { + case strings.ContainsRune("!#$%&()*+-./:<=>?@[]^_{|}~ ", c): + // Backslash and quote chars are reserved, but + // otherwise any punctuation chars are allowed + // in a tag name. + default: + if !unicode.IsLetter(c) && !unicode.IsDigit(c) { + return false + } + } + } + return true +} + +const ( + caseMask = ^byte(0x20) // Mask to ignore case in ASCII. + kelvin = '\u212a' + smallLongEss = '\u017f' +) + +// foldFunc returns one of four different case folding equivalence +// functions, from most general (and slow) to fastest: +// +// 1) bytes.EqualFold, if the key s contains any non-ASCII UTF-8 +// 2) equalFoldRight, if s contains special folding ASCII ('k', 'K', 's', 'S') +// 3) asciiEqualFold, no special, but includes non-letters (including _) +// 4) simpleLetterEqualFold, no specials, no non-letters. +// +// The letters S and K are special because they map to 3 runes, not just 2: +// * S maps to s and to U+017F 'ſ' Latin small letter long s +// * k maps to K and to U+212A 'K' Kelvin sign +// See http://play.golang.org/p/tTxjOc0OGo +// +// The returned function is specialized for matching against s and +// should only be given s. It's not curried for performance reasons. +func foldFunc(s []byte) func(s, t []byte) bool { + nonLetter := false + special := false // special letter + for _, b := range s { + if b >= utf8.RuneSelf { + return bytes.EqualFold + } + upper := b & caseMask + if upper < 'A' || upper > 'Z' { + nonLetter = true + } else if upper == 'K' || upper == 'S' { + // See above for why these letters are special. + special = true + } + } + if special { + return equalFoldRight + } + if nonLetter { + return asciiEqualFold + } + return simpleLetterEqualFold +} + +// equalFoldRight is a specialization of bytes.EqualFold when s is +// known to be all ASCII (including punctuation), but contains an 's', +// 'S', 'k', or 'K', requiring a Unicode fold on the bytes in t. +// See comments on foldFunc. +func equalFoldRight(s, t []byte) bool { + for _, sb := range s { + if len(t) == 0 { + return false + } + tb := t[0] + if tb < utf8.RuneSelf { + if sb != tb { + sbUpper := sb & caseMask + if 'A' <= sbUpper && sbUpper <= 'Z' { + if sbUpper != tb&caseMask { + return false + } + } else { + return false + } + } + t = t[1:] + continue + } + // sb is ASCII and t is not. t must be either kelvin + // sign or long s; sb must be s, S, k, or K. + tr, size := utf8.DecodeRune(t) + switch sb { + case 's', 'S': + if tr != smallLongEss { + return false + } + case 'k', 'K': + if tr != kelvin { + return false + } + default: + return false + } + t = t[size:] + + } + if len(t) > 0 { + return false + } + return true +} + +// asciiEqualFold is a specialization of bytes.EqualFold for use when +// s is all ASCII (but may contain non-letters) and contains no +// special-folding letters. +// See comments on foldFunc. +func asciiEqualFold(s, t []byte) bool { + if len(s) != len(t) { + return false + } + for i, sb := range s { + tb := t[i] + if sb == tb { + continue + } + if ('a' <= sb && sb <= 'z') || ('A' <= sb && sb <= 'Z') { + if sb&caseMask != tb&caseMask { + return false + } + } else { + return false + } + } + return true +} + +// simpleLetterEqualFold is a specialization of bytes.EqualFold for +// use when s is all ASCII letters (no underscores, etc) and also +// doesn't contain 'k', 'K', 's', or 'S'. +// See comments on foldFunc. +func simpleLetterEqualFold(s, t []byte) bool { + if len(s) != len(t) { + return false + } + for i, b := range s { + if b&caseMask != t[i]&caseMask { + return false + } + } + return true +} + +// tagOptions is the string following a comma in a struct field's "json" +// tag, or the empty string. It does not include the leading comma. +type tagOptions string + +// parseTag splits a struct field's json tag into its name and +// comma-separated options. +func parseTag(tag string) (string, tagOptions) { + if idx := strings.Index(tag, ","); idx != -1 { + return tag[:idx], tagOptions(tag[idx+1:]) + } + return tag, tagOptions("") +} + +// Contains reports whether a comma-separated list of options +// contains a particular substr flag. substr must be surrounded by a +// string boundary or commas. +func (o tagOptions) Contains(optionName string) bool { + if len(o) == 0 { + return false + } + s := string(o) + for s != "" { + var next string + i := strings.Index(s, ",") + if i >= 0 { + s, next = s[:i], s[i+1:] + } + if s == optionName { + return true + } + s = next + } + return false +} diff --git a/vendor/github.com/ghodss/yaml/yaml.go b/vendor/github.com/ghodss/yaml/yaml.go new file mode 100644 index 00000000..4fb4054a --- /dev/null +++ b/vendor/github.com/ghodss/yaml/yaml.go @@ -0,0 +1,277 @@ +package yaml + +import ( + "bytes" + "encoding/json" + "fmt" + "reflect" + "strconv" + + "gopkg.in/yaml.v2" +) + +// Marshals the object into JSON then converts JSON to YAML and returns the +// YAML. +func Marshal(o interface{}) ([]byte, error) { + j, err := json.Marshal(o) + if err != nil { + return nil, fmt.Errorf("error marshaling into JSON: %v", err) + } + + y, err := JSONToYAML(j) + if err != nil { + return nil, fmt.Errorf("error converting JSON to YAML: %v", err) + } + + return y, nil +} + +// Converts YAML to JSON then uses JSON to unmarshal into an object. +func Unmarshal(y []byte, o interface{}) error { + vo := reflect.ValueOf(o) + j, err := yamlToJSON(y, &vo) + if err != nil { + return fmt.Errorf("error converting YAML to JSON: %v", err) + } + + err = json.Unmarshal(j, o) + if err != nil { + return fmt.Errorf("error unmarshaling JSON: %v", err) + } + + return nil +} + +// Convert JSON to YAML. +func JSONToYAML(j []byte) ([]byte, error) { + // Convert the JSON to an object. + var jsonObj interface{} + // We are using yaml.Unmarshal here (instead of json.Unmarshal) because the + // Go JSON library doesn't try to pick the right number type (int, float, + // etc.) when unmarshalling to interface{}, it just picks float64 + // universally. go-yaml does go through the effort of picking the right + // number type, so we can preserve number type throughout this process. + err := yaml.Unmarshal(j, &jsonObj) + if err != nil { + return nil, err + } + + // Marshal this object into YAML. + return yaml.Marshal(jsonObj) +} + +// Convert YAML to JSON. Since JSON is a subset of YAML, passing JSON through +// this method should be a no-op. +// +// Things YAML can do that are not supported by JSON: +// * In YAML you can have binary and null keys in your maps. These are invalid +// in JSON. (int and float keys are converted to strings.) +// * Binary data in YAML with the !!binary tag is not supported. If you want to +// use binary data with this library, encode the data as base64 as usual but do +// not use the !!binary tag in your YAML. This will ensure the original base64 +// encoded data makes it all the way through to the JSON. +func YAMLToJSON(y []byte) ([]byte, error) { + return yamlToJSON(y, nil) +} + +func yamlToJSON(y []byte, jsonTarget *reflect.Value) ([]byte, error) { + // Convert the YAML to an object. + var yamlObj interface{} + err := yaml.Unmarshal(y, &yamlObj) + if err != nil { + return nil, err + } + + // YAML objects are not completely compatible with JSON objects (e.g. you + // can have non-string keys in YAML). So, convert the YAML-compatible object + // to a JSON-compatible object, failing with an error if irrecoverable + // incompatibilties happen along the way. + jsonObj, err := convertToJSONableObject(yamlObj, jsonTarget) + if err != nil { + return nil, err + } + + // Convert this object to JSON and return the data. + return json.Marshal(jsonObj) +} + +func convertToJSONableObject(yamlObj interface{}, jsonTarget *reflect.Value) (interface{}, error) { + var err error + + // Resolve jsonTarget to a concrete value (i.e. not a pointer or an + // interface). We pass decodingNull as false because we're not actually + // decoding into the value, we're just checking if the ultimate target is a + // string. + if jsonTarget != nil { + ju, tu, pv := indirect(*jsonTarget, false) + // We have a JSON or Text Umarshaler at this level, so we can't be trying + // to decode into a string. + if ju != nil || tu != nil { + jsonTarget = nil + } else { + jsonTarget = &pv + } + } + + // If yamlObj is a number or a boolean, check if jsonTarget is a string - + // if so, coerce. Else return normal. + // If yamlObj is a map or array, find the field that each key is + // unmarshaling to, and when you recurse pass the reflect.Value for that + // field back into this function. + switch typedYAMLObj := yamlObj.(type) { + case map[interface{}]interface{}: + // JSON does not support arbitrary keys in a map, so we must convert + // these keys to strings. + // + // From my reading of go-yaml v2 (specifically the resolve function), + // keys can only have the types string, int, int64, float64, binary + // (unsupported), or null (unsupported). + strMap := make(map[string]interface{}) + for k, v := range typedYAMLObj { + // Resolve the key to a string first. + var keyString string + switch typedKey := k.(type) { + case string: + keyString = typedKey + case int: + keyString = strconv.Itoa(typedKey) + case int64: + // go-yaml will only return an int64 as a key if the system + // architecture is 32-bit and the key's value is between 32-bit + // and 64-bit. Otherwise the key type will simply be int. + keyString = strconv.FormatInt(typedKey, 10) + case float64: + // Stolen from go-yaml to use the same conversion to string as + // the go-yaml library uses to convert float to string when + // Marshaling. + s := strconv.FormatFloat(typedKey, 'g', -1, 32) + switch s { + case "+Inf": + s = ".inf" + case "-Inf": + s = "-.inf" + case "NaN": + s = ".nan" + } + keyString = s + case bool: + if typedKey { + keyString = "true" + } else { + keyString = "false" + } + default: + return nil, fmt.Errorf("Unsupported map key of type: %s, key: %+#v, value: %+#v", + reflect.TypeOf(k), k, v) + } + + // jsonTarget should be a struct or a map. If it's a struct, find + // the field it's going to map to and pass its reflect.Value. If + // it's a map, find the element type of the map and pass the + // reflect.Value created from that type. If it's neither, just pass + // nil - JSON conversion will error for us if it's a real issue. + if jsonTarget != nil { + t := *jsonTarget + if t.Kind() == reflect.Struct { + keyBytes := []byte(keyString) + // Find the field that the JSON library would use. + var f *field + fields := cachedTypeFields(t.Type()) + for i := range fields { + ff := &fields[i] + if bytes.Equal(ff.nameBytes, keyBytes) { + f = ff + break + } + // Do case-insensitive comparison. + if f == nil && ff.equalFold(ff.nameBytes, keyBytes) { + f = ff + } + } + if f != nil { + // Find the reflect.Value of the most preferential + // struct field. + jtf := t.Field(f.index[0]) + strMap[keyString], err = convertToJSONableObject(v, &jtf) + if err != nil { + return nil, err + } + continue + } + } else if t.Kind() == reflect.Map { + // Create a zero value of the map's element type to use as + // the JSON target. + jtv := reflect.Zero(t.Type().Elem()) + strMap[keyString], err = convertToJSONableObject(v, &jtv) + if err != nil { + return nil, err + } + continue + } + } + strMap[keyString], err = convertToJSONableObject(v, nil) + if err != nil { + return nil, err + } + } + return strMap, nil + case []interface{}: + // We need to recurse into arrays in case there are any + // map[interface{}]interface{}'s inside and to convert any + // numbers to strings. + + // If jsonTarget is a slice (which it really should be), find the + // thing it's going to map to. If it's not a slice, just pass nil + // - JSON conversion will error for us if it's a real issue. + var jsonSliceElemValue *reflect.Value + if jsonTarget != nil { + t := *jsonTarget + if t.Kind() == reflect.Slice { + // By default slices point to nil, but we need a reflect.Value + // pointing to a value of the slice type, so we create one here. + ev := reflect.Indirect(reflect.New(t.Type().Elem())) + jsonSliceElemValue = &ev + } + } + + // Make and use a new array. + arr := make([]interface{}, len(typedYAMLObj)) + for i, v := range typedYAMLObj { + arr[i], err = convertToJSONableObject(v, jsonSliceElemValue) + if err != nil { + return nil, err + } + } + return arr, nil + default: + // If the target type is a string and the YAML type is a number, + // convert the YAML type to a string. + if jsonTarget != nil && (*jsonTarget).Kind() == reflect.String { + // Based on my reading of go-yaml, it may return int, int64, + // float64, or uint64. + var s string + switch typedVal := typedYAMLObj.(type) { + case int: + s = strconv.FormatInt(int64(typedVal), 10) + case int64: + s = strconv.FormatInt(typedVal, 10) + case float64: + s = strconv.FormatFloat(typedVal, 'g', -1, 32) + case uint64: + s = strconv.FormatUint(typedVal, 10) + case bool: + if typedVal { + s = "true" + } else { + s = "false" + } + } + if len(s) > 0 { + yamlObj = interface{}(s) + } + } + return yamlObj, nil + } + + return nil, nil +} diff --git a/vendor/github.com/gobwas/glob/.gitignore b/vendor/github.com/gobwas/glob/.gitignore new file mode 100644 index 00000000..b4ae623b --- /dev/null +++ b/vendor/github.com/gobwas/glob/.gitignore @@ -0,0 +1,8 @@ +glob.iml +.idea +*.cpu +*.mem +*.test +*.dot +*.png +*.svg diff --git a/vendor/github.com/gobwas/glob/.travis.yml b/vendor/github.com/gobwas/glob/.travis.yml new file mode 100644 index 00000000..e8a27682 --- /dev/null +++ b/vendor/github.com/gobwas/glob/.travis.yml @@ -0,0 +1,9 @@ +sudo: false + +language: go + +go: + - 1.5.3 + +script: + - go test -v ./... diff --git a/vendor/github.com/gobwas/glob/LICENSE b/vendor/github.com/gobwas/glob/LICENSE new file mode 100644 index 00000000..9d4735ca --- /dev/null +++ b/vendor/github.com/gobwas/glob/LICENSE @@ -0,0 +1,21 @@ +The MIT License (MIT) + +Copyright (c) 2016 Sergey Kamardin + +Permission is hereby granted, free of charge, to any person obtaining a copy +of this software and associated documentation files (the "Software"), to deal +in the Software without restriction, including without limitation the rights +to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the Software is +furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in all +copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE +SOFTWARE. \ No newline at end of file diff --git a/vendor/github.com/gobwas/glob/bench.sh b/vendor/github.com/gobwas/glob/bench.sh new file mode 100644 index 00000000..804cf22e --- /dev/null +++ b/vendor/github.com/gobwas/glob/bench.sh @@ -0,0 +1,26 @@ +#! /bin/bash + +bench() { + filename="/tmp/$1-$2.bench" + if test -e "${filename}"; + then + echo "Already exists ${filename}" + else + backup=`git rev-parse --abbrev-ref HEAD` + git checkout $1 + echo -n "Creating ${filename}... " + go test ./... -run=NONE -bench=$2 > "${filename}" -benchmem + echo "OK" + git checkout ${backup} + sleep 5 + fi +} + + +to=$1 +current=`git rev-parse --abbrev-ref HEAD` + +bench ${to} $2 +bench ${current} $2 + +benchcmp $3 "/tmp/${to}-$2.bench" "/tmp/${current}-$2.bench" diff --git a/vendor/github.com/gobwas/glob/compiler/compiler.go b/vendor/github.com/gobwas/glob/compiler/compiler.go new file mode 100644 index 00000000..02e7de80 --- /dev/null +++ b/vendor/github.com/gobwas/glob/compiler/compiler.go @@ -0,0 +1,525 @@ +package compiler + +// TODO use constructor with all matchers, and to their structs private +// TODO glue multiple Text nodes (like after QuoteMeta) + +import ( + "fmt" + "reflect" + + "github.com/gobwas/glob/match" + "github.com/gobwas/glob/syntax/ast" + "github.com/gobwas/glob/util/runes" +) + +func optimizeMatcher(matcher match.Matcher) match.Matcher { + switch m := matcher.(type) { + + case match.Any: + if len(m.Separators) == 0 { + return match.NewSuper() + } + + case match.AnyOf: + if len(m.Matchers) == 1 { + return m.Matchers[0] + } + + return m + + case match.List: + if m.Not == false && len(m.List) == 1 { + return match.NewText(string(m.List)) + } + + return m + + case match.BTree: + m.Left = optimizeMatcher(m.Left) + m.Right = optimizeMatcher(m.Right) + + r, ok := m.Value.(match.Text) + if !ok { + return m + } + + var ( + leftNil = m.Left == nil + rightNil = m.Right == nil + ) + if leftNil && rightNil { + return match.NewText(r.Str) + } + + _, leftSuper := m.Left.(match.Super) + lp, leftPrefix := m.Left.(match.Prefix) + la, leftAny := m.Left.(match.Any) + + _, rightSuper := m.Right.(match.Super) + rs, rightSuffix := m.Right.(match.Suffix) + ra, rightAny := m.Right.(match.Any) + + switch { + case leftSuper && rightSuper: + return match.NewContains(r.Str, false) + + case leftSuper && rightNil: + return match.NewSuffix(r.Str) + + case rightSuper && leftNil: + return match.NewPrefix(r.Str) + + case leftNil && rightSuffix: + return match.NewPrefixSuffix(r.Str, rs.Suffix) + + case rightNil && leftPrefix: + return match.NewPrefixSuffix(lp.Prefix, r.Str) + + case rightNil && leftAny: + return match.NewSuffixAny(r.Str, la.Separators) + + case leftNil && rightAny: + return match.NewPrefixAny(r.Str, ra.Separators) + } + + return m + } + + return matcher +} + +func compileMatchers(matchers []match.Matcher) (match.Matcher, error) { + if len(matchers) == 0 { + return nil, fmt.Errorf("compile error: need at least one matcher") + } + if len(matchers) == 1 { + return matchers[0], nil + } + if m := glueMatchers(matchers); m != nil { + return m, nil + } + + idx := -1 + maxLen := -1 + var val match.Matcher + for i, matcher := range matchers { + if l := matcher.Len(); l != -1 && l >= maxLen { + maxLen = l + idx = i + val = matcher + } + } + + if val == nil { // not found matcher with static length + r, err := compileMatchers(matchers[1:]) + if err != nil { + return nil, err + } + return match.NewBTree(matchers[0], nil, r), nil + } + + left := matchers[:idx] + var right []match.Matcher + if len(matchers) > idx+1 { + right = matchers[idx+1:] + } + + var l, r match.Matcher + var err error + if len(left) > 0 { + l, err = compileMatchers(left) + if err != nil { + return nil, err + } + } + + if len(right) > 0 { + r, err = compileMatchers(right) + if err != nil { + return nil, err + } + } + + return match.NewBTree(val, l, r), nil +} + +func glueMatchers(matchers []match.Matcher) match.Matcher { + if m := glueMatchersAsEvery(matchers); m != nil { + return m + } + if m := glueMatchersAsRow(matchers); m != nil { + return m + } + return nil +} + +func glueMatchersAsRow(matchers []match.Matcher) match.Matcher { + if len(matchers) <= 1 { + return nil + } + + var ( + c []match.Matcher + l int + ) + for _, matcher := range matchers { + if ml := matcher.Len(); ml == -1 { + return nil + } else { + c = append(c, matcher) + l += ml + } + } + return match.NewRow(l, c...) +} + +func glueMatchersAsEvery(matchers []match.Matcher) match.Matcher { + if len(matchers) <= 1 { + return nil + } + + var ( + hasAny bool + hasSuper bool + hasSingle bool + min int + separator []rune + ) + + for i, matcher := range matchers { + var sep []rune + + switch m := matcher.(type) { + case match.Super: + sep = []rune{} + hasSuper = true + + case match.Any: + sep = m.Separators + hasAny = true + + case match.Single: + sep = m.Separators + hasSingle = true + min++ + + case match.List: + if !m.Not { + return nil + } + sep = m.List + hasSingle = true + min++ + + default: + return nil + } + + // initialize + if i == 0 { + separator = sep + } + + if runes.Equal(sep, separator) { + continue + } + + return nil + } + + if hasSuper && !hasAny && !hasSingle { + return match.NewSuper() + } + + if hasAny && !hasSuper && !hasSingle { + return match.NewAny(separator) + } + + if (hasAny || hasSuper) && min > 0 && len(separator) == 0 { + return match.NewMin(min) + } + + every := match.NewEveryOf() + + if min > 0 { + every.Add(match.NewMin(min)) + + if !hasAny && !hasSuper { + every.Add(match.NewMax(min)) + } + } + + if len(separator) > 0 { + every.Add(match.NewContains(string(separator), true)) + } + + return every +} + +func minimizeMatchers(matchers []match.Matcher) []match.Matcher { + var done match.Matcher + var left, right, count int + + for l := 0; l < len(matchers); l++ { + for r := len(matchers); r > l; r-- { + if glued := glueMatchers(matchers[l:r]); glued != nil { + var swap bool + + if done == nil { + swap = true + } else { + cl, gl := done.Len(), glued.Len() + swap = cl > -1 && gl > -1 && gl > cl + swap = swap || count < r-l + } + + if swap { + done = glued + left = l + right = r + count = r - l + } + } + } + } + + if done == nil { + return matchers + } + + next := append(append([]match.Matcher{}, matchers[:left]...), done) + if right < len(matchers) { + next = append(next, matchers[right:]...) + } + + if len(next) == len(matchers) { + return next + } + + return minimizeMatchers(next) +} + +// minimizeAnyOf tries to apply some heuristics to minimize number of nodes in given tree +func minimizeTree(tree *ast.Node) *ast.Node { + switch tree.Kind { + case ast.KindAnyOf: + return minimizeTreeAnyOf(tree) + default: + return nil + } +} + +// minimizeAnyOf tries to find common children of given node of AnyOf pattern +// it searches for common children from left and from right +// if any common children are found – then it returns new optimized ast tree +// else it returns nil +func minimizeTreeAnyOf(tree *ast.Node) *ast.Node { + if !areOfSameKind(tree.Children, ast.KindPattern) { + return nil + } + + commonLeft, commonRight := commonChildren(tree.Children) + commonLeftCount, commonRightCount := len(commonLeft), len(commonRight) + if commonLeftCount == 0 && commonRightCount == 0 { // there are no common parts + return nil + } + + var result []*ast.Node + if commonLeftCount > 0 { + result = append(result, ast.NewNode(ast.KindPattern, nil, commonLeft...)) + } + + var anyOf []*ast.Node + for _, child := range tree.Children { + reuse := child.Children[commonLeftCount : len(child.Children)-commonRightCount] + var node *ast.Node + if len(reuse) == 0 { + // this pattern is completely reduced by commonLeft and commonRight patterns + // so it become nothing + node = ast.NewNode(ast.KindNothing, nil) + } else { + node = ast.NewNode(ast.KindPattern, nil, reuse...) + } + anyOf = appendIfUnique(anyOf, node) + } + switch { + case len(anyOf) == 1 && anyOf[0].Kind != ast.KindNothing: + result = append(result, anyOf[0]) + case len(anyOf) > 1: + result = append(result, ast.NewNode(ast.KindAnyOf, nil, anyOf...)) + } + + if commonRightCount > 0 { + result = append(result, ast.NewNode(ast.KindPattern, nil, commonRight...)) + } + + return ast.NewNode(ast.KindPattern, nil, result...) +} + +func commonChildren(nodes []*ast.Node) (commonLeft, commonRight []*ast.Node) { + if len(nodes) <= 1 { + return + } + + // find node that has least number of children + idx := leastChildren(nodes) + if idx == -1 { + return + } + tree := nodes[idx] + treeLength := len(tree.Children) + + // allocate max able size for rightCommon slice + // to get ability insert elements in reverse order (from end to start) + // without sorting + commonRight = make([]*ast.Node, treeLength) + lastRight := treeLength // will use this to get results as commonRight[lastRight:] + + var ( + breakLeft bool + breakRight bool + commonTotal int + ) + for i, j := 0, treeLength-1; commonTotal < treeLength && j >= 0 && !(breakLeft && breakRight); i, j = i+1, j-1 { + treeLeft := tree.Children[i] + treeRight := tree.Children[j] + + for k := 0; k < len(nodes) && !(breakLeft && breakRight); k++ { + // skip least children node + if k == idx { + continue + } + + restLeft := nodes[k].Children[i] + restRight := nodes[k].Children[j+len(nodes[k].Children)-treeLength] + + breakLeft = breakLeft || !treeLeft.Equal(restLeft) + + // disable searching for right common parts, if left part is already overlapping + breakRight = breakRight || (!breakLeft && j <= i) + breakRight = breakRight || !treeRight.Equal(restRight) + } + + if !breakLeft { + commonTotal++ + commonLeft = append(commonLeft, treeLeft) + } + if !breakRight { + commonTotal++ + lastRight = j + commonRight[j] = treeRight + } + } + + commonRight = commonRight[lastRight:] + + return +} + +func appendIfUnique(target []*ast.Node, val *ast.Node) []*ast.Node { + for _, n := range target { + if reflect.DeepEqual(n, val) { + return target + } + } + return append(target, val) +} + +func areOfSameKind(nodes []*ast.Node, kind ast.Kind) bool { + for _, n := range nodes { + if n.Kind != kind { + return false + } + } + return true +} + +func leastChildren(nodes []*ast.Node) int { + min := -1 + idx := -1 + for i, n := range nodes { + if idx == -1 || (len(n.Children) < min) { + min = len(n.Children) + idx = i + } + } + return idx +} + +func compileTreeChildren(tree *ast.Node, sep []rune) ([]match.Matcher, error) { + var matchers []match.Matcher + for _, desc := range tree.Children { + m, err := compile(desc, sep) + if err != nil { + return nil, err + } + matchers = append(matchers, optimizeMatcher(m)) + } + return matchers, nil +} + +func compile(tree *ast.Node, sep []rune) (m match.Matcher, err error) { + switch tree.Kind { + case ast.KindAnyOf: + // todo this could be faster on pattern_alternatives_combine_lite (see glob_test.go) + if n := minimizeTree(tree); n != nil { + return compile(n, sep) + } + matchers, err := compileTreeChildren(tree, sep) + if err != nil { + return nil, err + } + return match.NewAnyOf(matchers...), nil + + case ast.KindPattern: + if len(tree.Children) == 0 { + return match.NewNothing(), nil + } + matchers, err := compileTreeChildren(tree, sep) + if err != nil { + return nil, err + } + m, err = compileMatchers(minimizeMatchers(matchers)) + if err != nil { + return nil, err + } + + case ast.KindAny: + m = match.NewAny(sep) + + case ast.KindSuper: + m = match.NewSuper() + + case ast.KindSingle: + m = match.NewSingle(sep) + + case ast.KindNothing: + m = match.NewNothing() + + case ast.KindList: + l := tree.Value.(ast.List) + m = match.NewList([]rune(l.Chars), l.Not) + + case ast.KindRange: + r := tree.Value.(ast.Range) + m = match.NewRange(r.Lo, r.Hi, r.Not) + + case ast.KindText: + t := tree.Value.(ast.Text) + m = match.NewText(t.Text) + + default: + return nil, fmt.Errorf("could not compile tree: unknown node type") + } + + return optimizeMatcher(m), nil +} + +func Compile(tree *ast.Node, sep []rune) (match.Matcher, error) { + m, err := compile(tree, sep) + if err != nil { + return nil, err + } + + return m, nil +} diff --git a/vendor/github.com/gobwas/glob/glob.go b/vendor/github.com/gobwas/glob/glob.go new file mode 100644 index 00000000..2afde343 --- /dev/null +++ b/vendor/github.com/gobwas/glob/glob.go @@ -0,0 +1,80 @@ +package glob + +import ( + "github.com/gobwas/glob/compiler" + "github.com/gobwas/glob/syntax" +) + +// Glob represents compiled glob pattern. +type Glob interface { + Match(string) bool +} + +// Compile creates Glob for given pattern and strings (if any present after pattern) as separators. +// The pattern syntax is: +// +// pattern: +// { term } +// +// term: +// `*` matches any sequence of non-separator characters +// `**` matches any sequence of characters +// `?` matches any single non-separator character +// `[` [ `!` ] { character-range } `]` +// character class (must be non-empty) +// `{` pattern-list `}` +// pattern alternatives +// c matches character c (c != `*`, `**`, `?`, `\`, `[`, `{`, `}`) +// `\` c matches character c +// +// character-range: +// c matches character c (c != `\\`, `-`, `]`) +// `\` c matches character c +// lo `-` hi matches character c for lo <= c <= hi +// +// pattern-list: +// pattern { `,` pattern } +// comma-separated (without spaces) patterns +// +func Compile(pattern string, separators ...rune) (Glob, error) { + ast, err := syntax.Parse(pattern) + if err != nil { + return nil, err + } + + matcher, err := compiler.Compile(ast, separators) + if err != nil { + return nil, err + } + + return matcher, nil +} + +// MustCompile is the same as Compile, except that if Compile returns error, this will panic +func MustCompile(pattern string, separators ...rune) Glob { + g, err := Compile(pattern, separators...) + if err != nil { + panic(err) + } + + return g +} + +// QuoteMeta returns a string that quotes all glob pattern meta characters +// inside the argument text; For example, QuoteMeta(`{foo*}`) returns `\[foo\*\]`. +func QuoteMeta(s string) string { + b := make([]byte, 2*len(s)) + + // a byte loop is correct because all meta characters are ASCII + j := 0 + for i := 0; i < len(s); i++ { + if syntax.Special(s[i]) { + b[j] = '\\' + j++ + } + b[j] = s[i] + j++ + } + + return string(b[0:j]) +} diff --git a/vendor/github.com/gobwas/glob/match/any.go b/vendor/github.com/gobwas/glob/match/any.go new file mode 100644 index 00000000..514a9a5c --- /dev/null +++ b/vendor/github.com/gobwas/glob/match/any.go @@ -0,0 +1,45 @@ +package match + +import ( + "fmt" + "github.com/gobwas/glob/util/strings" +) + +type Any struct { + Separators []rune +} + +func NewAny(s []rune) Any { + return Any{s} +} + +func (self Any) Match(s string) bool { + return strings.IndexAnyRunes(s, self.Separators) == -1 +} + +func (self Any) Index(s string) (int, []int) { + found := strings.IndexAnyRunes(s, self.Separators) + switch found { + case -1: + case 0: + return 0, segments0 + default: + s = s[:found] + } + + segments := acquireSegments(len(s)) + for i := range s { + segments = append(segments, i) + } + segments = append(segments, len(s)) + + return 0, segments +} + +func (self Any) Len() int { + return lenNo +} + +func (self Any) String() string { + return fmt.Sprintf("", string(self.Separators)) +} diff --git a/vendor/github.com/gobwas/glob/match/any_of.go b/vendor/github.com/gobwas/glob/match/any_of.go new file mode 100644 index 00000000..8e65356c --- /dev/null +++ b/vendor/github.com/gobwas/glob/match/any_of.go @@ -0,0 +1,82 @@ +package match + +import "fmt" + +type AnyOf struct { + Matchers Matchers +} + +func NewAnyOf(m ...Matcher) AnyOf { + return AnyOf{Matchers(m)} +} + +func (self *AnyOf) Add(m Matcher) error { + self.Matchers = append(self.Matchers, m) + return nil +} + +func (self AnyOf) Match(s string) bool { + for _, m := range self.Matchers { + if m.Match(s) { + return true + } + } + + return false +} + +func (self AnyOf) Index(s string) (int, []int) { + index := -1 + + segments := acquireSegments(len(s)) + for _, m := range self.Matchers { + idx, seg := m.Index(s) + if idx == -1 { + continue + } + + if index == -1 || idx < index { + index = idx + segments = append(segments[:0], seg...) + continue + } + + if idx > index { + continue + } + + // here idx == index + segments = appendMerge(segments, seg) + } + + if index == -1 { + releaseSegments(segments) + return -1, nil + } + + return index, segments +} + +func (self AnyOf) Len() (l int) { + l = -1 + for _, m := range self.Matchers { + ml := m.Len() + switch { + case l == -1: + l = ml + continue + + case ml == -1: + return -1 + + case l != ml: + return -1 + } + } + + return +} + +func (self AnyOf) String() string { + return fmt.Sprintf("", self.Matchers) +} diff --git a/vendor/github.com/gobwas/glob/match/btree.go b/vendor/github.com/gobwas/glob/match/btree.go new file mode 100644 index 00000000..a8130e93 --- /dev/null +++ b/vendor/github.com/gobwas/glob/match/btree.go @@ -0,0 +1,146 @@ +package match + +import ( + "fmt" + "unicode/utf8" +) + +type BTree struct { + Value Matcher + Left Matcher + Right Matcher + ValueLengthRunes int + LeftLengthRunes int + RightLengthRunes int + LengthRunes int +} + +func NewBTree(Value, Left, Right Matcher) (tree BTree) { + tree.Value = Value + tree.Left = Left + tree.Right = Right + + lenOk := true + if tree.ValueLengthRunes = Value.Len(); tree.ValueLengthRunes == -1 { + lenOk = false + } + + if Left != nil { + if tree.LeftLengthRunes = Left.Len(); tree.LeftLengthRunes == -1 { + lenOk = false + } + } + + if Right != nil { + if tree.RightLengthRunes = Right.Len(); tree.RightLengthRunes == -1 { + lenOk = false + } + } + + if lenOk { + tree.LengthRunes = tree.LeftLengthRunes + tree.ValueLengthRunes + tree.RightLengthRunes + } else { + tree.LengthRunes = -1 + } + + return tree +} + +func (self BTree) Len() int { + return self.LengthRunes +} + +// todo? +func (self BTree) Index(s string) (int, []int) { + return -1, nil +} + +func (self BTree) Match(s string) bool { + inputLen := len(s) + + // self.Length, self.RLen and self.LLen are values meaning the length of runes for each part + // here we manipulating byte length for better optimizations + // but these checks still works, cause minLen of 1-rune string is 1 byte. + if self.LengthRunes != -1 && self.LengthRunes > inputLen { + return false + } + + // try to cut unnecessary parts + // by knowledge of length of right and left part + var offset, limit int + if self.LeftLengthRunes >= 0 { + offset = self.LeftLengthRunes + } + if self.RightLengthRunes >= 0 { + limit = inputLen - self.RightLengthRunes + } else { + limit = inputLen + } + + for offset < limit { + // search for matching part in substring + index, segments := self.Value.Index(s[offset:limit]) + if index == -1 { + releaseSegments(segments) + return false + } + + l := s[:offset+index] + var left bool + if self.Left != nil { + left = self.Left.Match(l) + } else { + left = l == "" + } + + if left { + for i := len(segments) - 1; i >= 0; i-- { + length := segments[i] + + var right bool + var r string + // if there is no string for the right branch + if inputLen <= offset+index+length { + r = "" + } else { + r = s[offset+index+length:] + } + + if self.Right != nil { + right = self.Right.Match(r) + } else { + right = r == "" + } + + if right { + releaseSegments(segments) + return true + } + } + } + + _, step := utf8.DecodeRuneInString(s[offset+index:]) + offset += index + step + + releaseSegments(segments) + } + + return false +} + +func (self BTree) String() string { + const n string = "" + var l, r string + if self.Left == nil { + l = n + } else { + l = self.Left.String() + } + if self.Right == nil { + r = n + } else { + r = self.Right.String() + } + + return fmt.Sprintf("%s]>", l, self.Value, r) +} diff --git a/vendor/github.com/gobwas/glob/match/contains.go b/vendor/github.com/gobwas/glob/match/contains.go new file mode 100644 index 00000000..0998e95b --- /dev/null +++ b/vendor/github.com/gobwas/glob/match/contains.go @@ -0,0 +1,58 @@ +package match + +import ( + "fmt" + "strings" +) + +type Contains struct { + Needle string + Not bool +} + +func NewContains(needle string, not bool) Contains { + return Contains{needle, not} +} + +func (self Contains) Match(s string) bool { + return strings.Contains(s, self.Needle) != self.Not +} + +func (self Contains) Index(s string) (int, []int) { + var offset int + + idx := strings.Index(s, self.Needle) + + if !self.Not { + if idx == -1 { + return -1, nil + } + + offset = idx + len(self.Needle) + if len(s) <= offset { + return 0, []int{offset} + } + s = s[offset:] + } else if idx != -1 { + s = s[:idx] + } + + segments := acquireSegments(len(s) + 1) + for i := range s { + segments = append(segments, offset+i) + } + + return 0, append(segments, offset+len(s)) +} + +func (self Contains) Len() int { + return lenNo +} + +func (self Contains) String() string { + var not string + if self.Not { + not = "!" + } + return fmt.Sprintf("", not, self.Needle) +} diff --git a/vendor/github.com/gobwas/glob/match/every_of.go b/vendor/github.com/gobwas/glob/match/every_of.go new file mode 100644 index 00000000..7c968ee3 --- /dev/null +++ b/vendor/github.com/gobwas/glob/match/every_of.go @@ -0,0 +1,99 @@ +package match + +import ( + "fmt" +) + +type EveryOf struct { + Matchers Matchers +} + +func NewEveryOf(m ...Matcher) EveryOf { + return EveryOf{Matchers(m)} +} + +func (self *EveryOf) Add(m Matcher) error { + self.Matchers = append(self.Matchers, m) + return nil +} + +func (self EveryOf) Len() (l int) { + for _, m := range self.Matchers { + if ml := m.Len(); l > 0 { + l += ml + } else { + return -1 + } + } + + return +} + +func (self EveryOf) Index(s string) (int, []int) { + var index int + var offset int + + // make `in` with cap as len(s), + // cause it is the maximum size of output segments values + next := acquireSegments(len(s)) + current := acquireSegments(len(s)) + + sub := s + for i, m := range self.Matchers { + idx, seg := m.Index(sub) + if idx == -1 { + releaseSegments(next) + releaseSegments(current) + return -1, nil + } + + if i == 0 { + // we use copy here instead of `current = seg` + // cause seg is a slice from reusable buffer `in` + // and it could be overwritten in next iteration + current = append(current, seg...) + } else { + // clear the next + next = next[:0] + + delta := index - (idx + offset) + for _, ex := range current { + for _, n := range seg { + if ex+delta == n { + next = append(next, n) + } + } + } + + if len(next) == 0 { + releaseSegments(next) + releaseSegments(current) + return -1, nil + } + + current = append(current[:0], next...) + } + + index = idx + offset + sub = s[index:] + offset += idx + } + + releaseSegments(next) + + return index, current +} + +func (self EveryOf) Match(s string) bool { + for _, m := range self.Matchers { + if !m.Match(s) { + return false + } + } + + return true +} + +func (self EveryOf) String() string { + return fmt.Sprintf("", self.Matchers) +} diff --git a/vendor/github.com/gobwas/glob/match/list.go b/vendor/github.com/gobwas/glob/match/list.go new file mode 100644 index 00000000..7fd763ec --- /dev/null +++ b/vendor/github.com/gobwas/glob/match/list.go @@ -0,0 +1,49 @@ +package match + +import ( + "fmt" + "github.com/gobwas/glob/util/runes" + "unicode/utf8" +) + +type List struct { + List []rune + Not bool +} + +func NewList(list []rune, not bool) List { + return List{list, not} +} + +func (self List) Match(s string) bool { + r, w := utf8.DecodeRuneInString(s) + if len(s) > w { + return false + } + + inList := runes.IndexRune(self.List, r) != -1 + return inList == !self.Not +} + +func (self List) Len() int { + return lenOne +} + +func (self List) Index(s string) (int, []int) { + for i, r := range s { + if self.Not == (runes.IndexRune(self.List, r) == -1) { + return i, segmentsByRuneLength[utf8.RuneLen(r)] + } + } + + return -1, nil +} + +func (self List) String() string { + var not string + if self.Not { + not = "!" + } + + return fmt.Sprintf("", not, string(self.List)) +} diff --git a/vendor/github.com/gobwas/glob/match/match.go b/vendor/github.com/gobwas/glob/match/match.go new file mode 100644 index 00000000..f80e007f --- /dev/null +++ b/vendor/github.com/gobwas/glob/match/match.go @@ -0,0 +1,81 @@ +package match + +// todo common table of rune's length + +import ( + "fmt" + "strings" +) + +const lenOne = 1 +const lenZero = 0 +const lenNo = -1 + +type Matcher interface { + Match(string) bool + Index(string) (int, []int) + Len() int + String() string +} + +type Matchers []Matcher + +func (m Matchers) String() string { + var s []string + for _, matcher := range m { + s = append(s, fmt.Sprint(matcher)) + } + + return fmt.Sprintf("%s", strings.Join(s, ",")) +} + +// appendMerge merges and sorts given already SORTED and UNIQUE segments. +func appendMerge(target, sub []int) []int { + lt, ls := len(target), len(sub) + out := make([]int, 0, lt+ls) + + for x, y := 0, 0; x < lt || y < ls; { + if x >= lt { + out = append(out, sub[y:]...) + break + } + + if y >= ls { + out = append(out, target[x:]...) + break + } + + xValue := target[x] + yValue := sub[y] + + switch { + + case xValue == yValue: + out = append(out, xValue) + x++ + y++ + + case xValue < yValue: + out = append(out, xValue) + x++ + + case yValue < xValue: + out = append(out, yValue) + y++ + + } + } + + target = append(target[:0], out...) + + return target +} + +func reverseSegments(input []int) { + l := len(input) + m := l / 2 + + for i := 0; i < m; i++ { + input[i], input[l-i-1] = input[l-i-1], input[i] + } +} diff --git a/vendor/github.com/gobwas/glob/match/max.go b/vendor/github.com/gobwas/glob/match/max.go new file mode 100644 index 00000000..d72f69ef --- /dev/null +++ b/vendor/github.com/gobwas/glob/match/max.go @@ -0,0 +1,49 @@ +package match + +import ( + "fmt" + "unicode/utf8" +) + +type Max struct { + Limit int +} + +func NewMax(l int) Max { + return Max{l} +} + +func (self Max) Match(s string) bool { + var l int + for range s { + l += 1 + if l > self.Limit { + return false + } + } + + return true +} + +func (self Max) Index(s string) (int, []int) { + segments := acquireSegments(self.Limit + 1) + segments = append(segments, 0) + var count int + for i, r := range s { + count++ + if count > self.Limit { + break + } + segments = append(segments, i+utf8.RuneLen(r)) + } + + return 0, segments +} + +func (self Max) Len() int { + return lenNo +} + +func (self Max) String() string { + return fmt.Sprintf("", self.Limit) +} diff --git a/vendor/github.com/gobwas/glob/match/min.go b/vendor/github.com/gobwas/glob/match/min.go new file mode 100644 index 00000000..db57ac8e --- /dev/null +++ b/vendor/github.com/gobwas/glob/match/min.go @@ -0,0 +1,57 @@ +package match + +import ( + "fmt" + "unicode/utf8" +) + +type Min struct { + Limit int +} + +func NewMin(l int) Min { + return Min{l} +} + +func (self Min) Match(s string) bool { + var l int + for range s { + l += 1 + if l >= self.Limit { + return true + } + } + + return false +} + +func (self Min) Index(s string) (int, []int) { + var count int + + c := len(s) - self.Limit + 1 + if c <= 0 { + return -1, nil + } + + segments := acquireSegments(c) + for i, r := range s { + count++ + if count >= self.Limit { + segments = append(segments, i+utf8.RuneLen(r)) + } + } + + if len(segments) == 0 { + return -1, nil + } + + return 0, segments +} + +func (self Min) Len() int { + return lenNo +} + +func (self Min) String() string { + return fmt.Sprintf("", self.Limit) +} diff --git a/vendor/github.com/gobwas/glob/match/nothing.go b/vendor/github.com/gobwas/glob/match/nothing.go new file mode 100644 index 00000000..0d4ecd36 --- /dev/null +++ b/vendor/github.com/gobwas/glob/match/nothing.go @@ -0,0 +1,27 @@ +package match + +import ( + "fmt" +) + +type Nothing struct{} + +func NewNothing() Nothing { + return Nothing{} +} + +func (self Nothing) Match(s string) bool { + return len(s) == 0 +} + +func (self Nothing) Index(s string) (int, []int) { + return 0, segments0 +} + +func (self Nothing) Len() int { + return lenZero +} + +func (self Nothing) String() string { + return fmt.Sprintf("") +} diff --git a/vendor/github.com/gobwas/glob/match/prefix.go b/vendor/github.com/gobwas/glob/match/prefix.go new file mode 100644 index 00000000..a7347250 --- /dev/null +++ b/vendor/github.com/gobwas/glob/match/prefix.go @@ -0,0 +1,50 @@ +package match + +import ( + "fmt" + "strings" + "unicode/utf8" +) + +type Prefix struct { + Prefix string +} + +func NewPrefix(p string) Prefix { + return Prefix{p} +} + +func (self Prefix) Index(s string) (int, []int) { + idx := strings.Index(s, self.Prefix) + if idx == -1 { + return -1, nil + } + + length := len(self.Prefix) + var sub string + if len(s) > idx+length { + sub = s[idx+length:] + } else { + sub = "" + } + + segments := acquireSegments(len(sub) + 1) + segments = append(segments, length) + for i, r := range sub { + segments = append(segments, length+i+utf8.RuneLen(r)) + } + + return idx, segments +} + +func (self Prefix) Len() int { + return lenNo +} + +func (self Prefix) Match(s string) bool { + return strings.HasPrefix(s, self.Prefix) +} + +func (self Prefix) String() string { + return fmt.Sprintf("", self.Prefix) +} diff --git a/vendor/github.com/gobwas/glob/match/prefix_any.go b/vendor/github.com/gobwas/glob/match/prefix_any.go new file mode 100644 index 00000000..8ee58fe1 --- /dev/null +++ b/vendor/github.com/gobwas/glob/match/prefix_any.go @@ -0,0 +1,55 @@ +package match + +import ( + "fmt" + "strings" + "unicode/utf8" + + sutil "github.com/gobwas/glob/util/strings" +) + +type PrefixAny struct { + Prefix string + Separators []rune +} + +func NewPrefixAny(s string, sep []rune) PrefixAny { + return PrefixAny{s, sep} +} + +func (self PrefixAny) Index(s string) (int, []int) { + idx := strings.Index(s, self.Prefix) + if idx == -1 { + return -1, nil + } + + n := len(self.Prefix) + sub := s[idx+n:] + i := sutil.IndexAnyRunes(sub, self.Separators) + if i > -1 { + sub = sub[:i] + } + + seg := acquireSegments(len(sub) + 1) + seg = append(seg, n) + for i, r := range sub { + seg = append(seg, n+i+utf8.RuneLen(r)) + } + + return idx, seg +} + +func (self PrefixAny) Len() int { + return lenNo +} + +func (self PrefixAny) Match(s string) bool { + if !strings.HasPrefix(s, self.Prefix) { + return false + } + return sutil.IndexAnyRunes(s[len(self.Prefix):], self.Separators) == -1 +} + +func (self PrefixAny) String() string { + return fmt.Sprintf("", self.Prefix, string(self.Separators)) +} diff --git a/vendor/github.com/gobwas/glob/match/prefix_suffix.go b/vendor/github.com/gobwas/glob/match/prefix_suffix.go new file mode 100644 index 00000000..8208085a --- /dev/null +++ b/vendor/github.com/gobwas/glob/match/prefix_suffix.go @@ -0,0 +1,62 @@ +package match + +import ( + "fmt" + "strings" +) + +type PrefixSuffix struct { + Prefix, Suffix string +} + +func NewPrefixSuffix(p, s string) PrefixSuffix { + return PrefixSuffix{p, s} +} + +func (self PrefixSuffix) Index(s string) (int, []int) { + prefixIdx := strings.Index(s, self.Prefix) + if prefixIdx == -1 { + return -1, nil + } + + suffixLen := len(self.Suffix) + if suffixLen <= 0 { + return prefixIdx, []int{len(s) - prefixIdx} + } + + if (len(s) - prefixIdx) <= 0 { + return -1, nil + } + + segments := acquireSegments(len(s) - prefixIdx) + for sub := s[prefixIdx:]; ; { + suffixIdx := strings.LastIndex(sub, self.Suffix) + if suffixIdx == -1 { + break + } + + segments = append(segments, suffixIdx+suffixLen) + sub = sub[:suffixIdx] + } + + if len(segments) == 0 { + releaseSegments(segments) + return -1, nil + } + + reverseSegments(segments) + + return prefixIdx, segments +} + +func (self PrefixSuffix) Len() int { + return lenNo +} + +func (self PrefixSuffix) Match(s string) bool { + return strings.HasPrefix(s, self.Prefix) && strings.HasSuffix(s, self.Suffix) +} + +func (self PrefixSuffix) String() string { + return fmt.Sprintf("", self.Prefix, self.Suffix) +} diff --git a/vendor/github.com/gobwas/glob/match/range.go b/vendor/github.com/gobwas/glob/match/range.go new file mode 100644 index 00000000..ce30245a --- /dev/null +++ b/vendor/github.com/gobwas/glob/match/range.go @@ -0,0 +1,48 @@ +package match + +import ( + "fmt" + "unicode/utf8" +) + +type Range struct { + Lo, Hi rune + Not bool +} + +func NewRange(lo, hi rune, not bool) Range { + return Range{lo, hi, not} +} + +func (self Range) Len() int { + return lenOne +} + +func (self Range) Match(s string) bool { + r, w := utf8.DecodeRuneInString(s) + if len(s) > w { + return false + } + + inRange := r >= self.Lo && r <= self.Hi + + return inRange == !self.Not +} + +func (self Range) Index(s string) (int, []int) { + for i, r := range s { + if self.Not != (r >= self.Lo && r <= self.Hi) { + return i, segmentsByRuneLength[utf8.RuneLen(r)] + } + } + + return -1, nil +} + +func (self Range) String() string { + var not string + if self.Not { + not = "!" + } + return fmt.Sprintf("", not, string(self.Lo), string(self.Hi)) +} diff --git a/vendor/github.com/gobwas/glob/match/row.go b/vendor/github.com/gobwas/glob/match/row.go new file mode 100644 index 00000000..4379042e --- /dev/null +++ b/vendor/github.com/gobwas/glob/match/row.go @@ -0,0 +1,77 @@ +package match + +import ( + "fmt" +) + +type Row struct { + Matchers Matchers + RunesLength int + Segments []int +} + +func NewRow(len int, m ...Matcher) Row { + return Row{ + Matchers: Matchers(m), + RunesLength: len, + Segments: []int{len}, + } +} + +func (self Row) matchAll(s string) bool { + var idx int + for _, m := range self.Matchers { + length := m.Len() + + var next, i int + for next = range s[idx:] { + i++ + if i == length { + break + } + } + + if i < length || !m.Match(s[idx:idx+next+1]) { + return false + } + + idx += next + 1 + } + + return true +} + +func (self Row) lenOk(s string) bool { + var i int + for range s { + i++ + if i > self.RunesLength { + return false + } + } + return self.RunesLength == i +} + +func (self Row) Match(s string) bool { + return self.lenOk(s) && self.matchAll(s) +} + +func (self Row) Len() (l int) { + return self.RunesLength +} + +func (self Row) Index(s string) (int, []int) { + for i := range s { + if len(s[i:]) < self.RunesLength { + break + } + if self.matchAll(s[i:]) { + return i, self.Segments + } + } + return -1, nil +} + +func (self Row) String() string { + return fmt.Sprintf("", self.RunesLength, self.Matchers) +} diff --git a/vendor/github.com/gobwas/glob/match/segments.go b/vendor/github.com/gobwas/glob/match/segments.go new file mode 100644 index 00000000..9ea6f309 --- /dev/null +++ b/vendor/github.com/gobwas/glob/match/segments.go @@ -0,0 +1,91 @@ +package match + +import ( + "sync" +) + +type SomePool interface { + Get() []int + Put([]int) +} + +var segmentsPools [1024]sync.Pool + +func toPowerOfTwo(v int) int { + v-- + v |= v >> 1 + v |= v >> 2 + v |= v >> 4 + v |= v >> 8 + v |= v >> 16 + v++ + + return v +} + +const ( + cacheFrom = 16 + cacheToAndHigher = 1024 + cacheFromIndex = 15 + cacheToAndHigherIndex = 1023 +) + +var ( + segments0 = []int{0} + segments1 = []int{1} + segments2 = []int{2} + segments3 = []int{3} + segments4 = []int{4} +) + +var segmentsByRuneLength [5][]int = [5][]int{ + 0: segments0, + 1: segments1, + 2: segments2, + 3: segments3, + 4: segments4, +} + +func init() { + for i := cacheToAndHigher; i >= cacheFrom; i >>= 1 { + func(i int) { + segmentsPools[i-1] = sync.Pool{New: func() interface{} { + return make([]int, 0, i) + }} + }(i) + } +} + +func getTableIndex(c int) int { + p := toPowerOfTwo(c) + switch { + case p >= cacheToAndHigher: + return cacheToAndHigherIndex + case p <= cacheFrom: + return cacheFromIndex + default: + return p - 1 + } +} + +func acquireSegments(c int) []int { + // make []int with less capacity than cacheFrom + // is faster than acquiring it from pool + if c < cacheFrom { + return make([]int, 0, c) + } + + return segmentsPools[getTableIndex(c)].Get().([]int)[:0] +} + +func releaseSegments(s []int) { + c := cap(s) + + // make []int with less capacity than cacheFrom + // is faster than acquiring it from pool + if c < cacheFrom { + return + } + + segmentsPools[getTableIndex(c)].Put(s) +} diff --git a/vendor/github.com/gobwas/glob/match/single.go b/vendor/github.com/gobwas/glob/match/single.go new file mode 100644 index 00000000..ee6e3954 --- /dev/null +++ b/vendor/github.com/gobwas/glob/match/single.go @@ -0,0 +1,43 @@ +package match + +import ( + "fmt" + "github.com/gobwas/glob/util/runes" + "unicode/utf8" +) + +// single represents ? +type Single struct { + Separators []rune +} + +func NewSingle(s []rune) Single { + return Single{s} +} + +func (self Single) Match(s string) bool { + r, w := utf8.DecodeRuneInString(s) + if len(s) > w { + return false + } + + return runes.IndexRune(self.Separators, r) == -1 +} + +func (self Single) Len() int { + return lenOne +} + +func (self Single) Index(s string) (int, []int) { + for i, r := range s { + if runes.IndexRune(self.Separators, r) == -1 { + return i, segmentsByRuneLength[utf8.RuneLen(r)] + } + } + + return -1, nil +} + +func (self Single) String() string { + return fmt.Sprintf("", string(self.Separators)) +} diff --git a/vendor/github.com/gobwas/glob/match/suffix.go b/vendor/github.com/gobwas/glob/match/suffix.go new file mode 100644 index 00000000..85bea8c6 --- /dev/null +++ b/vendor/github.com/gobwas/glob/match/suffix.go @@ -0,0 +1,35 @@ +package match + +import ( + "fmt" + "strings" +) + +type Suffix struct { + Suffix string +} + +func NewSuffix(s string) Suffix { + return Suffix{s} +} + +func (self Suffix) Len() int { + return lenNo +} + +func (self Suffix) Match(s string) bool { + return strings.HasSuffix(s, self.Suffix) +} + +func (self Suffix) Index(s string) (int, []int) { + idx := strings.Index(s, self.Suffix) + if idx == -1 { + return -1, nil + } + + return 0, []int{idx + len(self.Suffix)} +} + +func (self Suffix) String() string { + return fmt.Sprintf("", self.Suffix) +} diff --git a/vendor/github.com/gobwas/glob/match/suffix_any.go b/vendor/github.com/gobwas/glob/match/suffix_any.go new file mode 100644 index 00000000..c5106f81 --- /dev/null +++ b/vendor/github.com/gobwas/glob/match/suffix_any.go @@ -0,0 +1,43 @@ +package match + +import ( + "fmt" + "strings" + + sutil "github.com/gobwas/glob/util/strings" +) + +type SuffixAny struct { + Suffix string + Separators []rune +} + +func NewSuffixAny(s string, sep []rune) SuffixAny { + return SuffixAny{s, sep} +} + +func (self SuffixAny) Index(s string) (int, []int) { + idx := strings.Index(s, self.Suffix) + if idx == -1 { + return -1, nil + } + + i := sutil.LastIndexAnyRunes(s[:idx], self.Separators) + 1 + + return i, []int{idx + len(self.Suffix) - i} +} + +func (self SuffixAny) Len() int { + return lenNo +} + +func (self SuffixAny) Match(s string) bool { + if !strings.HasSuffix(s, self.Suffix) { + return false + } + return sutil.IndexAnyRunes(s[:len(s)-len(self.Suffix)], self.Separators) == -1 +} + +func (self SuffixAny) String() string { + return fmt.Sprintf("", string(self.Separators), self.Suffix) +} diff --git a/vendor/github.com/gobwas/glob/match/super.go b/vendor/github.com/gobwas/glob/match/super.go new file mode 100644 index 00000000..3875950b --- /dev/null +++ b/vendor/github.com/gobwas/glob/match/super.go @@ -0,0 +1,33 @@ +package match + +import ( + "fmt" +) + +type Super struct{} + +func NewSuper() Super { + return Super{} +} + +func (self Super) Match(s string) bool { + return true +} + +func (self Super) Len() int { + return lenNo +} + +func (self Super) Index(s string) (int, []int) { + segments := acquireSegments(len(s) + 1) + for i := range s { + segments = append(segments, i) + } + segments = append(segments, len(s)) + + return 0, segments +} + +func (self Super) String() string { + return fmt.Sprintf("") +} diff --git a/vendor/github.com/gobwas/glob/match/text.go b/vendor/github.com/gobwas/glob/match/text.go new file mode 100644 index 00000000..0a17616d --- /dev/null +++ b/vendor/github.com/gobwas/glob/match/text.go @@ -0,0 +1,45 @@ +package match + +import ( + "fmt" + "strings" + "unicode/utf8" +) + +// raw represents raw string to match +type Text struct { + Str string + RunesLength int + BytesLength int + Segments []int +} + +func NewText(s string) Text { + return Text{ + Str: s, + RunesLength: utf8.RuneCountInString(s), + BytesLength: len(s), + Segments: []int{len(s)}, + } +} + +func (self Text) Match(s string) bool { + return self.Str == s +} + +func (self Text) Len() int { + return self.RunesLength +} + +func (self Text) Index(s string) (int, []int) { + index := strings.Index(s, self.Str) + if index == -1 { + return -1, nil + } + + return index, self.Segments +} + +func (self Text) String() string { + return fmt.Sprintf("", self.Str) +} diff --git a/vendor/github.com/gobwas/glob/readme.md b/vendor/github.com/gobwas/glob/readme.md new file mode 100644 index 00000000..f58144e7 --- /dev/null +++ b/vendor/github.com/gobwas/glob/readme.md @@ -0,0 +1,148 @@ +# glob.[go](https://golang.org) + +[![GoDoc][godoc-image]][godoc-url] [![Build Status][travis-image]][travis-url] + +> Go Globbing Library. + +## Install + +```shell + go get github.com/gobwas/glob +``` + +## Example + +```go + +package main + +import "github.com/gobwas/glob" + +func main() { + var g glob.Glob + + // create simple glob + g = glob.MustCompile("*.github.com") + g.Match("api.github.com") // true + + // quote meta characters and then create simple glob + g = glob.MustCompile(glob.QuoteMeta("*.github.com")) + g.Match("*.github.com") // true + + // create new glob with set of delimiters as ["."] + g = glob.MustCompile("api.*.com", '.') + g.Match("api.github.com") // true + g.Match("api.gi.hub.com") // false + + // create new glob with set of delimiters as ["."] + // but now with super wildcard + g = glob.MustCompile("api.**.com", '.') + g.Match("api.github.com") // true + g.Match("api.gi.hub.com") // true + + // create glob with single symbol wildcard + g = glob.MustCompile("?at") + g.Match("cat") // true + g.Match("fat") // true + g.Match("at") // false + + // create glob with single symbol wildcard and delimiters ['f'] + g = glob.MustCompile("?at", 'f') + g.Match("cat") // true + g.Match("fat") // false + g.Match("at") // false + + // create glob with character-list matchers + g = glob.MustCompile("[abc]at") + g.Match("cat") // true + g.Match("bat") // true + g.Match("fat") // false + g.Match("at") // false + + // create glob with character-list matchers + g = glob.MustCompile("[!abc]at") + g.Match("cat") // false + g.Match("bat") // false + g.Match("fat") // true + g.Match("at") // false + + // create glob with character-range matchers + g = glob.MustCompile("[a-c]at") + g.Match("cat") // true + g.Match("bat") // true + g.Match("fat") // false + g.Match("at") // false + + // create glob with character-range matchers + g = glob.MustCompile("[!a-c]at") + g.Match("cat") // false + g.Match("bat") // false + g.Match("fat") // true + g.Match("at") // false + + // create glob with pattern-alternatives list + g = glob.MustCompile("{cat,bat,[fr]at}") + g.Match("cat") // true + g.Match("bat") // true + g.Match("fat") // true + g.Match("rat") // true + g.Match("at") // false + g.Match("zat") // false +} + +``` + +## Performance + +This library is created for compile-once patterns. This means, that compilation could take time, but +strings matching is done faster, than in case when always parsing template. + +If you will not use compiled `glob.Glob` object, and do `g := glob.MustCompile(pattern); g.Match(...)` every time, then your code will be much more slower. + +Run `go test -bench=.` from source root to see the benchmarks: + +Pattern | Fixture | Match | Speed (ns/op) +--------|---------|-------|-------------- +`[a-z][!a-x]*cat*[h][!b]*eyes*` | `my cat has very bright eyes` | `true` | 432 +`[a-z][!a-x]*cat*[h][!b]*eyes*` | `my dog has very bright eyes` | `false` | 199 +`https://*.google.*` | `https://account.google.com` | `true` | 96 +`https://*.google.*` | `https://google.com` | `false` | 66 +`{https://*.google.*,*yandex.*,*yahoo.*,*mail.ru}` | `http://yahoo.com` | `true` | 163 +`{https://*.google.*,*yandex.*,*yahoo.*,*mail.ru}` | `http://google.com` | `false` | 197 +`{https://*gobwas.com,http://exclude.gobwas.com}` | `https://safe.gobwas.com` | `true` | 22 +`{https://*gobwas.com,http://exclude.gobwas.com}` | `http://safe.gobwas.com` | `false` | 24 +`abc*` | `abcdef` | `true` | 8.15 +`abc*` | `af` | `false` | 5.68 +`*def` | `abcdef` | `true` | 8.84 +`*def` | `af` | `false` | 5.74 +`ab*ef` | `abcdef` | `true` | 15.2 +`ab*ef` | `af` | `false` | 10.4 + +The same things with `regexp` package: + +Pattern | Fixture | Match | Speed (ns/op) +--------|---------|-------|-------------- +`^[a-z][^a-x].*cat.*[h][^b].*eyes.*$` | `my cat has very bright eyes` | `true` | 2553 +`^[a-z][^a-x].*cat.*[h][^b].*eyes.*$` | `my dog has very bright eyes` | `false` | 1383 +`^https:\/\/.*\.google\..*$` | `https://account.google.com` | `true` | 1205 +`^https:\/\/.*\.google\..*$` | `https://google.com` | `false` | 767 +`^(https:\/\/.*\.google\..*|.*yandex\..*|.*yahoo\..*|.*mail\.ru)$` | `http://yahoo.com` | `true` | 1435 +`^(https:\/\/.*\.google\..*|.*yandex\..*|.*yahoo\..*|.*mail\.ru)$` | `http://google.com` | `false` | 1674 +`^(https:\/\/.*gobwas\.com|http://exclude.gobwas.com)$` | `https://safe.gobwas.com` | `true` | 1039 +`^(https:\/\/.*gobwas\.com|http://exclude.gobwas.com)$` | `http://safe.gobwas.com` | `false` | 272 +`^abc.*$` | `abcdef` | `true` | 237 +`^abc.*$` | `af` | `false` | 100 +`^.*def$` | `abcdef` | `true` | 464 +`^.*def$` | `af` | `false` | 265 +`^ab.*ef$` | `abcdef` | `true` | 375 +`^ab.*ef$` | `af` | `false` | 145 + +[godoc-image]: https://godoc.org/github.com/gobwas/glob?status.svg +[godoc-url]: https://godoc.org/github.com/gobwas/glob +[travis-image]: https://travis-ci.org/gobwas/glob.svg?branch=master +[travis-url]: https://travis-ci.org/gobwas/glob + +## Syntax + +Syntax is inspired by [standard wildcards](http://tldp.org/LDP/GNU-Linux-Tools-Summary/html/x11655.htm), +except that `**` is aka super-asterisk, that do not sensitive for separators. \ No newline at end of file diff --git a/vendor/github.com/gobwas/glob/syntax/ast/ast.go b/vendor/github.com/gobwas/glob/syntax/ast/ast.go new file mode 100644 index 00000000..3220a694 --- /dev/null +++ b/vendor/github.com/gobwas/glob/syntax/ast/ast.go @@ -0,0 +1,122 @@ +package ast + +import ( + "bytes" + "fmt" +) + +type Node struct { + Parent *Node + Children []*Node + Value interface{} + Kind Kind +} + +func NewNode(k Kind, v interface{}, ch ...*Node) *Node { + n := &Node{ + Kind: k, + Value: v, + } + for _, c := range ch { + Insert(n, c) + } + return n +} + +func (a *Node) Equal(b *Node) bool { + if a.Kind != b.Kind { + return false + } + if a.Value != b.Value { + return false + } + if len(a.Children) != len(b.Children) { + return false + } + for i, c := range a.Children { + if !c.Equal(b.Children[i]) { + return false + } + } + return true +} + +func (a *Node) String() string { + var buf bytes.Buffer + buf.WriteString(a.Kind.String()) + if a.Value != nil { + buf.WriteString(" =") + buf.WriteString(fmt.Sprintf("%v", a.Value)) + } + if len(a.Children) > 0 { + buf.WriteString(" [") + for i, c := range a.Children { + if i > 0 { + buf.WriteString(", ") + } + buf.WriteString(c.String()) + } + buf.WriteString("]") + } + return buf.String() +} + +func Insert(parent *Node, children ...*Node) { + parent.Children = append(parent.Children, children...) + for _, ch := range children { + ch.Parent = parent + } +} + +type List struct { + Not bool + Chars string +} + +type Range struct { + Not bool + Lo, Hi rune +} + +type Text struct { + Text string +} + +type Kind int + +const ( + KindNothing Kind = iota + KindPattern + KindList + KindRange + KindText + KindAny + KindSuper + KindSingle + KindAnyOf +) + +func (k Kind) String() string { + switch k { + case KindNothing: + return "Nothing" + case KindPattern: + return "Pattern" + case KindList: + return "List" + case KindRange: + return "Range" + case KindText: + return "Text" + case KindAny: + return "Any" + case KindSuper: + return "Super" + case KindSingle: + return "Single" + case KindAnyOf: + return "AnyOf" + default: + return "" + } +} diff --git a/vendor/github.com/gobwas/glob/syntax/ast/parser.go b/vendor/github.com/gobwas/glob/syntax/ast/parser.go new file mode 100644 index 00000000..429b4094 --- /dev/null +++ b/vendor/github.com/gobwas/glob/syntax/ast/parser.go @@ -0,0 +1,157 @@ +package ast + +import ( + "errors" + "fmt" + "github.com/gobwas/glob/syntax/lexer" + "unicode/utf8" +) + +type Lexer interface { + Next() lexer.Token +} + +type parseFn func(*Node, Lexer) (parseFn, *Node, error) + +func Parse(lexer Lexer) (*Node, error) { + var parser parseFn + + root := NewNode(KindPattern, nil) + + var ( + tree *Node + err error + ) + for parser, tree = parserMain, root; parser != nil; { + parser, tree, err = parser(tree, lexer) + if err != nil { + return nil, err + } + } + + return root, nil +} + +func parserMain(tree *Node, lex Lexer) (parseFn, *Node, error) { + for { + token := lex.Next() + switch token.Type { + case lexer.EOF: + return nil, tree, nil + + case lexer.Error: + return nil, tree, errors.New(token.Raw) + + case lexer.Text: + Insert(tree, NewNode(KindText, Text{token.Raw})) + return parserMain, tree, nil + + case lexer.Any: + Insert(tree, NewNode(KindAny, nil)) + return parserMain, tree, nil + + case lexer.Super: + Insert(tree, NewNode(KindSuper, nil)) + return parserMain, tree, nil + + case lexer.Single: + Insert(tree, NewNode(KindSingle, nil)) + return parserMain, tree, nil + + case lexer.RangeOpen: + return parserRange, tree, nil + + case lexer.TermsOpen: + a := NewNode(KindAnyOf, nil) + Insert(tree, a) + + p := NewNode(KindPattern, nil) + Insert(a, p) + + return parserMain, p, nil + + case lexer.Separator: + p := NewNode(KindPattern, nil) + Insert(tree.Parent, p) + + return parserMain, p, nil + + case lexer.TermsClose: + return parserMain, tree.Parent.Parent, nil + + default: + return nil, tree, fmt.Errorf("unexpected token: %s", token) + } + } + return nil, tree, fmt.Errorf("unknown error") +} + +func parserRange(tree *Node, lex Lexer) (parseFn, *Node, error) { + var ( + not bool + lo rune + hi rune + chars string + ) + for { + token := lex.Next() + switch token.Type { + case lexer.EOF: + return nil, tree, errors.New("unexpected end") + + case lexer.Error: + return nil, tree, errors.New(token.Raw) + + case lexer.Not: + not = true + + case lexer.RangeLo: + r, w := utf8.DecodeRuneInString(token.Raw) + if len(token.Raw) > w { + return nil, tree, fmt.Errorf("unexpected length of lo character") + } + lo = r + + case lexer.RangeBetween: + // + + case lexer.RangeHi: + r, w := utf8.DecodeRuneInString(token.Raw) + if len(token.Raw) > w { + return nil, tree, fmt.Errorf("unexpected length of lo character") + } + + hi = r + + if hi < lo { + return nil, tree, fmt.Errorf("hi character '%s' should be greater than lo '%s'", string(hi), string(lo)) + } + + case lexer.Text: + chars = token.Raw + + case lexer.RangeClose: + isRange := lo != 0 && hi != 0 + isChars := chars != "" + + if isChars == isRange { + return nil, tree, fmt.Errorf("could not parse range") + } + + if isRange { + Insert(tree, NewNode(KindRange, Range{ + Lo: lo, + Hi: hi, + Not: not, + })) + } else { + Insert(tree, NewNode(KindList, List{ + Chars: chars, + Not: not, + })) + } + + return parserMain, tree, nil + } + } +} diff --git a/vendor/github.com/gobwas/glob/syntax/lexer/lexer.go b/vendor/github.com/gobwas/glob/syntax/lexer/lexer.go new file mode 100644 index 00000000..a1c8d196 --- /dev/null +++ b/vendor/github.com/gobwas/glob/syntax/lexer/lexer.go @@ -0,0 +1,273 @@ +package lexer + +import ( + "bytes" + "fmt" + "github.com/gobwas/glob/util/runes" + "unicode/utf8" +) + +const ( + char_any = '*' + char_comma = ',' + char_single = '?' + char_escape = '\\' + char_range_open = '[' + char_range_close = ']' + char_terms_open = '{' + char_terms_close = '}' + char_range_not = '!' + char_range_between = '-' +) + +var specials = []byte{ + char_any, + char_single, + char_escape, + char_range_open, + char_range_close, + char_terms_open, + char_terms_close, +} + +func Special(c byte) bool { + return bytes.IndexByte(specials, c) != -1 +} + +type tokens []Token + +func (i *tokens) shift() (ret Token) { + ret = (*i)[0] + copy(*i, (*i)[1:]) + *i = (*i)[:len(*i)-1] + return +} + +func (i *tokens) push(v Token) { + *i = append(*i, v) +} + +func (i *tokens) empty() bool { + return len(*i) == 0 +} + +var eof rune = 0 + +type lexer struct { + data string + pos int + err error + + tokens tokens + termsLevel int + + lastRune rune + lastRuneSize int + hasRune bool +} + +func NewLexer(source string) *lexer { + l := &lexer{ + data: source, + tokens: tokens(make([]Token, 0, 4)), + } + return l +} + +func (l *lexer) Next() Token { + if l.err != nil { + return Token{Error, l.err.Error()} + } + if !l.tokens.empty() { + return l.tokens.shift() + } + + l.fetchItem() + return l.Next() +} + +func (l *lexer) peek() (r rune, w int) { + if l.pos == len(l.data) { + return eof, 0 + } + + r, w = utf8.DecodeRuneInString(l.data[l.pos:]) + if r == utf8.RuneError { + l.errorf("could not read rune") + r = eof + w = 0 + } + + return +} + +func (l *lexer) read() rune { + if l.hasRune { + l.hasRune = false + l.seek(l.lastRuneSize) + return l.lastRune + } + + r, s := l.peek() + l.seek(s) + + l.lastRune = r + l.lastRuneSize = s + + return r +} + +func (l *lexer) seek(w int) { + l.pos += w +} + +func (l *lexer) unread() { + if l.hasRune { + l.errorf("could not unread rune") + return + } + l.seek(-l.lastRuneSize) + l.hasRune = true +} + +func (l *lexer) errorf(f string, v ...interface{}) { + l.err = fmt.Errorf(f, v...) +} + +func (l *lexer) inTerms() bool { + return l.termsLevel > 0 +} + +func (l *lexer) termsEnter() { + l.termsLevel++ +} + +func (l *lexer) termsLeave() { + l.termsLevel-- +} + +var inTextBreakers = []rune{char_single, char_any, char_range_open, char_terms_open} +var inTermsBreakers = append(inTextBreakers, char_terms_close, char_comma) + +func (l *lexer) fetchItem() { + r := l.read() + switch { + case r == eof: + l.tokens.push(Token{EOF, ""}) + + case r == char_terms_open: + l.termsEnter() + l.tokens.push(Token{TermsOpen, string(r)}) + + case r == char_comma && l.inTerms(): + l.tokens.push(Token{Separator, string(r)}) + + case r == char_terms_close && l.inTerms(): + l.tokens.push(Token{TermsClose, string(r)}) + l.termsLeave() + + case r == char_range_open: + l.tokens.push(Token{RangeOpen, string(r)}) + l.fetchRange() + + case r == char_single: + l.tokens.push(Token{Single, string(r)}) + + case r == char_any: + if l.read() == char_any { + l.tokens.push(Token{Super, string(r) + string(r)}) + } else { + l.unread() + l.tokens.push(Token{Any, string(r)}) + } + + default: + l.unread() + + var breakers []rune + if l.inTerms() { + breakers = inTermsBreakers + } else { + breakers = inTextBreakers + } + l.fetchText(breakers) + } +} + +func (l *lexer) fetchRange() { + var wantHi bool + var wantClose bool + var seenNot bool + for { + r := l.read() + if r == eof { + l.errorf("unexpected end of input") + return + } + + if wantClose { + if r != char_range_close { + l.errorf("expected close range character") + } else { + l.tokens.push(Token{RangeClose, string(r)}) + } + return + } + + if wantHi { + l.tokens.push(Token{RangeHi, string(r)}) + wantClose = true + continue + } + + if !seenNot && r == char_range_not { + l.tokens.push(Token{Not, string(r)}) + seenNot = true + continue + } + + if n, w := l.peek(); n == char_range_between { + l.seek(w) + l.tokens.push(Token{RangeLo, string(r)}) + l.tokens.push(Token{RangeBetween, string(n)}) + wantHi = true + continue + } + + l.unread() // unread first peek and fetch as text + l.fetchText([]rune{char_range_close}) + wantClose = true + } +} + +func (l *lexer) fetchText(breakers []rune) { + var data []rune + var escaped bool + +reading: + for { + r := l.read() + if r == eof { + break + } + + if !escaped { + if r == char_escape { + escaped = true + continue + } + + if runes.IndexRune(breakers, r) != -1 { + l.unread() + break reading + } + } + + escaped = false + data = append(data, r) + } + + if len(data) > 0 { + l.tokens.push(Token{Text, string(data)}) + } +} diff --git a/vendor/github.com/gobwas/glob/syntax/lexer/token.go b/vendor/github.com/gobwas/glob/syntax/lexer/token.go new file mode 100644 index 00000000..2797c4e8 --- /dev/null +++ b/vendor/github.com/gobwas/glob/syntax/lexer/token.go @@ -0,0 +1,88 @@ +package lexer + +import "fmt" + +type TokenType int + +const ( + EOF TokenType = iota + Error + Text + Char + Any + Super + Single + Not + Separator + RangeOpen + RangeClose + RangeLo + RangeHi + RangeBetween + TermsOpen + TermsClose +) + +func (tt TokenType) String() string { + switch tt { + case EOF: + return "eof" + + case Error: + return "error" + + case Text: + return "text" + + case Char: + return "char" + + case Any: + return "any" + + case Super: + return "super" + + case Single: + return "single" + + case Not: + return "not" + + case Separator: + return "separator" + + case RangeOpen: + return "range_open" + + case RangeClose: + return "range_close" + + case RangeLo: + return "range_lo" + + case RangeHi: + return "range_hi" + + case RangeBetween: + return "range_between" + + case TermsOpen: + return "terms_open" + + case TermsClose: + return "terms_close" + + default: + return "undef" + } +} + +type Token struct { + Type TokenType + Raw string +} + +func (t Token) String() string { + return fmt.Sprintf("%v<%q>", t.Type, t.Raw) +} diff --git a/vendor/github.com/gobwas/glob/syntax/syntax.go b/vendor/github.com/gobwas/glob/syntax/syntax.go new file mode 100644 index 00000000..1d168b14 --- /dev/null +++ b/vendor/github.com/gobwas/glob/syntax/syntax.go @@ -0,0 +1,14 @@ +package syntax + +import ( + "github.com/gobwas/glob/syntax/ast" + "github.com/gobwas/glob/syntax/lexer" +) + +func Parse(s string) (*ast.Node, error) { + return ast.Parse(lexer.NewLexer(s)) +} + +func Special(b byte) bool { + return lexer.Special(b) +} diff --git a/vendor/github.com/gobwas/glob/util/runes/runes.go b/vendor/github.com/gobwas/glob/util/runes/runes.go new file mode 100644 index 00000000..a7235564 --- /dev/null +++ b/vendor/github.com/gobwas/glob/util/runes/runes.go @@ -0,0 +1,154 @@ +package runes + +func Index(s, needle []rune) int { + ls, ln := len(s), len(needle) + + switch { + case ln == 0: + return 0 + case ln == 1: + return IndexRune(s, needle[0]) + case ln == ls: + if Equal(s, needle) { + return 0 + } + return -1 + case ln > ls: + return -1 + } + +head: + for i := 0; i < ls && ls-i >= ln; i++ { + for y := 0; y < ln; y++ { + if s[i+y] != needle[y] { + continue head + } + } + + return i + } + + return -1 +} + +func LastIndex(s, needle []rune) int { + ls, ln := len(s), len(needle) + + switch { + case ln == 0: + if ls == 0 { + return 0 + } + return ls + case ln == 1: + return IndexLastRune(s, needle[0]) + case ln == ls: + if Equal(s, needle) { + return 0 + } + return -1 + case ln > ls: + return -1 + } + +head: + for i := ls - 1; i >= 0 && i >= ln; i-- { + for y := ln - 1; y >= 0; y-- { + if s[i-(ln-y-1)] != needle[y] { + continue head + } + } + + return i - ln + 1 + } + + return -1 +} + +// IndexAny returns the index of the first instance of any Unicode code point +// from chars in s, or -1 if no Unicode code point from chars is present in s. +func IndexAny(s, chars []rune) int { + if len(chars) > 0 { + for i, c := range s { + for _, m := range chars { + if c == m { + return i + } + } + } + } + return -1 +} + +func Contains(s, needle []rune) bool { + return Index(s, needle) >= 0 +} + +func Max(s []rune) (max rune) { + for _, r := range s { + if r > max { + max = r + } + } + + return +} + +func Min(s []rune) rune { + min := rune(-1) + for _, r := range s { + if min == -1 { + min = r + continue + } + + if r < min { + min = r + } + } + + return min +} + +func IndexRune(s []rune, r rune) int { + for i, c := range s { + if c == r { + return i + } + } + return -1 +} + +func IndexLastRune(s []rune, r rune) int { + for i := len(s) - 1; i >= 0; i-- { + if s[i] == r { + return i + } + } + + return -1 +} + +func Equal(a, b []rune) bool { + if len(a) == len(b) { + for i := 0; i < len(a); i++ { + if a[i] != b[i] { + return false + } + } + + return true + } + + return false +} + +// HasPrefix tests whether the string s begins with prefix. +func HasPrefix(s, prefix []rune) bool { + return len(s) >= len(prefix) && Equal(s[0:len(prefix)], prefix) +} + +// HasSuffix tests whether the string s ends with suffix. +func HasSuffix(s, suffix []rune) bool { + return len(s) >= len(suffix) && Equal(s[len(s)-len(suffix):], suffix) +} diff --git a/vendor/github.com/gobwas/glob/util/strings/strings.go b/vendor/github.com/gobwas/glob/util/strings/strings.go new file mode 100644 index 00000000..e8ee1920 --- /dev/null +++ b/vendor/github.com/gobwas/glob/util/strings/strings.go @@ -0,0 +1,39 @@ +package strings + +import ( + "strings" + "unicode/utf8" +) + +func IndexAnyRunes(s string, rs []rune) int { + for _, r := range rs { + if i := strings.IndexRune(s, r); i != -1 { + return i + } + } + + return -1 +} + +func LastIndexAnyRunes(s string, rs []rune) int { + for _, r := range rs { + i := -1 + if 0 <= r && r < utf8.RuneSelf { + i = strings.LastIndexByte(s, byte(r)) + } else { + sub := s + for len(sub) > 0 { + j := strings.IndexRune(s, r) + if j == -1 { + break + } + i = j + sub = sub[i+1:] + } + } + if i != -1 { + return i + } + } + return -1 +} diff --git a/vendor/github.com/open-policy-agent/opa/LICENSE b/vendor/github.com/open-policy-agent/opa/LICENSE new file mode 100644 index 00000000..8f71f43f --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/LICENSE @@ -0,0 +1,202 @@ + Apache License + Version 2.0, January 2004 + http://www.apache.org/licenses/ + + TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION + + 1. Definitions. + + "License" shall mean the terms and conditions for use, reproduction, + and distribution as defined by Sections 1 through 9 of this document. + + "Licensor" shall mean the copyright owner or entity authorized by + the copyright owner that is granting the License. + + "Legal Entity" shall mean the union of the acting entity and all + other entities that control, are controlled by, or are under common + control with that entity. For the purposes of this definition, + "control" means (i) the power, direct or indirect, to cause the + direction or management of such entity, whether by contract or + otherwise, or (ii) ownership of fifty percent (50%) or more of the + outstanding shares, or (iii) beneficial ownership of such entity. + + "You" (or "Your") shall mean an individual or Legal Entity + exercising permissions granted by this License. + + "Source" form shall mean the preferred form for making modifications, + including but not limited to software source code, documentation + source, and configuration files. + + "Object" form shall mean any form resulting from mechanical + transformation or translation of a Source form, including but + not limited to compiled object code, generated documentation, + and conversions to other media types. + + "Work" shall mean the work of authorship, whether in Source or + Object form, made available under the License, as indicated by a + copyright notice that is included in or attached to the work + (an example is provided in the Appendix below). + + "Derivative Works" shall mean any work, whether in Source or Object + form, that is based on (or derived from) the Work and for which the + editorial revisions, annotations, elaborations, or other modifications + represent, as a whole, an original work of authorship. For the purposes + of this License, Derivative Works shall not include works that remain + separable from, or merely link (or bind by name) to the interfaces of, + the Work and Derivative Works thereof. + + "Contribution" shall mean any work of authorship, including + the original version of the Work and any modifications or additions + to that Work or Derivative Works thereof, that is intentionally + submitted to Licensor for inclusion in the Work by the copyright owner + or by an individual or Legal Entity authorized to submit on behalf of + the copyright owner. For the purposes of this definition, "submitted" + means any form of electronic, verbal, or written communication sent + to the Licensor or its representatives, including but not limited to + communication on electronic mailing lists, source code control systems, + and issue tracking systems that are managed by, or on behalf of, the + Licensor for the purpose of discussing and improving the Work, but + excluding communication that is conspicuously marked or otherwise + designated in writing by the copyright owner as "Not a Contribution." + + "Contributor" shall mean Licensor and any individual or Legal Entity + on behalf of whom a Contribution has been received by Licensor and + subsequently incorporated within the Work. + + 2. Grant of Copyright License. Subject to the terms and conditions of + this License, each Contributor hereby grants to You a perpetual, + worldwide, non-exclusive, no-charge, royalty-free, irrevocable + copyright license to reproduce, prepare Derivative Works of, + publicly display, publicly perform, sublicense, and distribute the + Work and such Derivative Works in Source or Object form. + + 3. Grant of Patent License. Subject to the terms and conditions of + this License, each Contributor hereby grants to You a perpetual, + worldwide, non-exclusive, no-charge, royalty-free, irrevocable + (except as stated in this section) patent license to make, have made, + use, offer to sell, sell, import, and otherwise transfer the Work, + where such license applies only to those patent claims licensable + by such Contributor that are necessarily infringed by their + Contribution(s) alone or by combination of their Contribution(s) + with the Work to which such Contribution(s) was submitted. If You + institute patent litigation against any entity (including a + cross-claim or counterclaim in a lawsuit) alleging that the Work + or a Contribution incorporated within the Work constitutes direct + or contributory patent infringement, then any patent licenses + granted to You under this License for that Work shall terminate + as of the date such litigation is filed. + + 4. Redistribution. You may reproduce and distribute copies of the + Work or Derivative Works thereof in any medium, with or without + modifications, and in Source or Object form, provided that You + meet the following conditions: + + (a) You must give any other recipients of the Work or + Derivative Works a copy of this License; and + + (b) You must cause any modified files to carry prominent notices + stating that You changed the files; and + + (c) You must retain, in the Source form of any Derivative Works + that You distribute, all copyright, patent, trademark, and + attribution notices from the Source form of the Work, + excluding those notices that do not pertain to any part of + the Derivative Works; and + + (d) If the Work includes a "NOTICE" text file as part of its + distribution, then any Derivative Works that You distribute must + include a readable copy of the attribution notices contained + within such NOTICE file, excluding those notices that do not + pertain to any part of the Derivative Works, in at least one + of the following places: within a NOTICE text file distributed + as part of the Derivative Works; within the Source form or + documentation, if provided along with the Derivative Works; or, + within a display generated by the Derivative Works, if and + wherever such third-party notices normally appear. The contents + of the NOTICE file are for informational purposes only and + do not modify the License. You may add Your own attribution + notices within Derivative Works that You distribute, alongside + or as an addendum to the NOTICE text from the Work, provided + that such additional attribution notices cannot be construed + as modifying the License. + + You may add Your own copyright statement to Your modifications and + may provide additional or different license terms and conditions + for use, reproduction, or distribution of Your modifications, or + for any such Derivative Works as a whole, provided Your use, + reproduction, and distribution of the Work otherwise complies with + the conditions stated in this License. + + 5. Submission of Contributions. Unless You explicitly state otherwise, + any Contribution intentionally submitted for inclusion in the Work + by You to the Licensor shall be under the terms and conditions of + this License, without any additional terms or conditions. + Notwithstanding the above, nothing herein shall supersede or modify + the terms of any separate license agreement you may have executed + with Licensor regarding such Contributions. + + 6. Trademarks. This License does not grant permission to use the trade + names, trademarks, service marks, or product names of the Licensor, + except as required for reasonable and customary use in describing the + origin of the Work and reproducing the content of the NOTICE file. + + 7. Disclaimer of Warranty. Unless required by applicable law or + agreed to in writing, Licensor provides the Work (and each + Contributor provides its Contributions) on an "AS IS" BASIS, + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or + implied, including, without limitation, any warranties or conditions + of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A + PARTICULAR PURPOSE. You are solely responsible for determining the + appropriateness of using or redistributing the Work and assume any + risks associated with Your exercise of permissions under this License. + + 8. Limitation of Liability. In no event and under no legal theory, + whether in tort (including negligence), contract, or otherwise, + unless required by applicable law (such as deliberate and grossly + negligent acts) or agreed to in writing, shall any Contributor be + liable to You for damages, including any direct, indirect, special, + incidental, or consequential damages of any character arising as a + result of this License or out of the use or inability to use the + Work (including but not limited to damages for loss of goodwill, + work stoppage, computer failure or malfunction, or any and all + other commercial damages or losses), even if such Contributor + has been advised of the possibility of such damages. + + 9. Accepting Warranty or Additional Liability. While redistributing + the Work or Derivative Works thereof, You may choose to offer, + and charge a fee for, acceptance of support, warranty, indemnity, + or other liability obligations and/or rights consistent with this + License. However, in accepting such obligations, You may act only + on Your own behalf and on Your sole responsibility, not on behalf + of any other Contributor, and only if You agree to indemnify, + defend, and hold each Contributor harmless for any liability + incurred by, or claims asserted against, such Contributor by reason + of your accepting any such warranty or additional liability. + + END OF TERMS AND CONDITIONS + + APPENDIX: How to apply the Apache License to your work. + + To apply the Apache License to your work, attach the following + boilerplate notice, with the fields enclosed by brackets "{}" + replaced with your own identifying information. (Don't include + the brackets!) The text should be enclosed in the appropriate + comment syntax for the file format. We also recommend that a + file or class name and description of purpose be included on the + same "printed page" as the copyright notice for easier + identification within third-party archives. + + Copyright {yyyy} {name of copyright owner} + + Licensed under the Apache License, Version 2.0 (the "License"); + you may not use this file except in compliance with the License. + You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + + Unless required by applicable law or agreed to in writing, software + distributed under the License is distributed on an "AS IS" BASIS, + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + See the License for the specific language governing permissions and + limitations under the License. + diff --git a/vendor/github.com/open-policy-agent/opa/ast/builtins.go b/vendor/github.com/open-policy-agent/opa/ast/builtins.go new file mode 100644 index 00000000..4afa98f3 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/ast/builtins.go @@ -0,0 +1,2523 @@ +// Copyright 2016 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package ast + +import ( + "strings" + + "github.com/open-policy-agent/opa/types" +) + +// Builtins is the registry of built-in functions supported by OPA. +// Call RegisterBuiltin to add a new built-in. +var Builtins []*Builtin + +// RegisterBuiltin adds a new built-in function to the registry. +func RegisterBuiltin(b *Builtin) { + Builtins = append(Builtins, b) + BuiltinMap[b.Name] = b + if len(b.Infix) > 0 { + BuiltinMap[b.Infix] = b + } +} + +// DefaultBuiltins is the registry of built-in functions supported in OPA +// by default. When adding a new built-in function to OPA, update this +// list. +var DefaultBuiltins = [...]*Builtin{ + // Unification/equality ("=") + Equality, + + // Assignment (":=") + Assign, + + // Membership, infix "in": `x in xs` + Member, + MemberWithKey, + + // Comparisons + GreaterThan, + GreaterThanEq, + LessThan, + LessThanEq, + NotEqual, + Equal, + + // Arithmetic + Plus, + Minus, + Multiply, + Divide, + Ceil, + Floor, + Round, + Abs, + Rem, + + // Bitwise Arithmetic + BitsOr, + BitsAnd, + BitsNegate, + BitsXOr, + BitsShiftLeft, + BitsShiftRight, + + // Binary + And, + Or, + + // Aggregates + Count, + Sum, + Product, + Max, + Min, + Any, + All, + + // Arrays + ArrayConcat, + ArraySlice, + ArrayReverse, + + // Conversions + ToNumber, + + // Casts (DEPRECATED) + CastObject, + CastNull, + CastBoolean, + CastString, + CastSet, + CastArray, + + // Regular Expressions + RegexIsValid, + RegexMatch, + RegexMatchDeprecated, + RegexSplit, + GlobsMatch, + RegexTemplateMatch, + RegexFind, + RegexFindAllStringSubmatch, + + // Sets + SetDiff, + Intersection, + Union, + + // Strings + Concat, + FormatInt, + IndexOf, + Substring, + Lower, + Upper, + Contains, + StartsWith, + EndsWith, + Split, + Replace, + ReplaceN, + Trim, + TrimLeft, + TrimPrefix, + TrimRight, + TrimSuffix, + TrimSpace, + Sprintf, + StringReverse, + + // Numbers + NumbersRange, + RandIntn, + + // Encoding + JSONMarshal, + JSONUnmarshal, + JSONIsValid, + Base64Encode, + Base64Decode, + Base64IsValid, + Base64UrlEncode, + Base64UrlEncodeNoPad, + Base64UrlDecode, + URLQueryDecode, + URLQueryEncode, + URLQueryEncodeObject, + URLQueryDecodeObject, + YAMLMarshal, + YAMLUnmarshal, + YAMLIsValid, + HexEncode, + HexDecode, + + // Object Manipulation + ObjectUnion, + ObjectRemove, + ObjectFilter, + ObjectGet, + + // JSON Object Manipulation + JSONFilter, + JSONRemove, + JSONPatch, + + // Tokens + JWTDecode, + JWTVerifyRS256, + JWTVerifyRS384, + JWTVerifyRS512, + JWTVerifyPS256, + JWTVerifyPS384, + JWTVerifyPS512, + JWTVerifyES256, + JWTVerifyES384, + JWTVerifyES512, + JWTVerifyHS256, + JWTVerifyHS384, + JWTVerifyHS512, + JWTDecodeVerify, + JWTEncodeSignRaw, + JWTEncodeSign, + + // Time + NowNanos, + ParseNanos, + ParseRFC3339Nanos, + ParseDurationNanos, + Date, + Clock, + Weekday, + AddDate, + Diff, + + // Crypto + CryptoX509ParseCertificates, + CryptoX509ParseAndVerifyCertificates, + CryptoMd5, + CryptoSha1, + CryptoSha256, + CryptoX509ParseCertificateRequest, + CryptoX509ParseRSAPrivateKey, + CryptoHmacMd5, + CryptoHmacSha1, + CryptoHmacSha256, + CryptoHmacSha512, + + // Graphs + WalkBuiltin, + ReachableBuiltin, + + // Sort + Sort, + + // Types + IsNumber, + IsString, + IsBoolean, + IsArray, + IsSet, + IsObject, + IsNull, + TypeNameBuiltin, + + // HTTP + HTTPSend, + + // Rego + RegoParseModule, + + // OPA + OPARuntime, + + // Tracing + Trace, + + // Networking + NetCIDROverlap, + NetCIDRIntersects, + NetCIDRContains, + NetCIDRContainsMatches, + NetCIDRExpand, + NetCIDRMerge, + NetLookupIPAddr, + + // Glob + GlobMatch, + GlobQuoteMeta, + + // Units + UnitsParseBytes, + + // UUIDs + UUIDRFC4122, + + //SemVers + SemVerIsValid, + SemVerCompare, + + // Printing + Print, + InternalPrint, +} + +// BuiltinMap provides a convenient mapping of built-in names to +// built-in definitions. +var BuiltinMap map[string]*Builtin + +// IgnoreDuringPartialEval is a set of built-in functions that should not be +// evaluated during partial evaluation. These functions are not partially +// evaluated because they are not pure. +var IgnoreDuringPartialEval = []*Builtin{ + NowNanos, + HTTPSend, + UUIDRFC4122, + RandIntn, + NetLookupIPAddr, +} + +/** + * Unification + */ + +// Equality represents the "=" operator. +var Equality = &Builtin{ + Name: "eq", + Infix: "=", + Decl: types.NewFunction( + types.Args(types.A, types.A), + types.B, + ), +} + +/** + * Assignment + */ + +// Assign represents the assignment (":=") operator. +var Assign = &Builtin{ + Name: "assign", + Infix: ":=", + Decl: types.NewFunction( + types.Args(types.A, types.A), + types.B, + ), +} + +// Member represents the `in` (infix) operator. +var Member = &Builtin{ + Name: "internal.member_2", + Infix: "in", + Decl: types.NewFunction( + types.Args( + types.A, + types.A, + ), + types.B, + ), +} + +// MemberWithKey represents the `in` (infix) operator when used +// with two terms on the lhs, i.e., `k, v in obj`. +var MemberWithKey = &Builtin{ + Name: "internal.member_3", + Infix: "in", + Decl: types.NewFunction( + types.Args( + types.A, + types.A, + types.A, + ), + types.B, + ), +} + +/** + * Comparisons + */ + +// GreaterThan represents the ">" comparison operator. +var GreaterThan = &Builtin{ + Name: "gt", + Infix: ">", + Decl: types.NewFunction( + types.Args(types.A, types.A), + types.B, + ), +} + +// GreaterThanEq represents the ">=" comparison operator. +var GreaterThanEq = &Builtin{ + Name: "gte", + Infix: ">=", + Decl: types.NewFunction( + types.Args(types.A, types.A), + types.B, + ), +} + +// LessThan represents the "<" comparison operator. +var LessThan = &Builtin{ + Name: "lt", + Infix: "<", + Decl: types.NewFunction( + types.Args(types.A, types.A), + types.B, + ), +} + +// LessThanEq represents the "<=" comparison operator. +var LessThanEq = &Builtin{ + Name: "lte", + Infix: "<=", + Decl: types.NewFunction( + types.Args(types.A, types.A), + types.B, + ), +} + +// NotEqual represents the "!=" comparison operator. +var NotEqual = &Builtin{ + Name: "neq", + Infix: "!=", + Decl: types.NewFunction( + types.Args(types.A, types.A), + types.B, + ), +} + +// Equal represents the "==" comparison operator. +var Equal = &Builtin{ + Name: "equal", + Infix: "==", + Decl: types.NewFunction( + types.Args(types.A, types.A), + types.B, + ), +} + +/** + * Arithmetic + */ + +// Plus adds two numbers together. +var Plus = &Builtin{ + Name: "plus", + Infix: "+", + Decl: types.NewFunction( + types.Args(types.N, types.N), + types.N, + ), +} + +// Minus subtracts the second number from the first number or computes the diff +// between two sets. +var Minus = &Builtin{ + Name: "minus", + Infix: "-", + Decl: types.NewFunction( + types.Args( + types.NewAny(types.N, types.NewSet(types.A)), + types.NewAny(types.N, types.NewSet(types.A)), + ), + types.NewAny(types.N, types.NewSet(types.A)), + ), +} + +// Multiply multiplies two numbers together. +var Multiply = &Builtin{ + Name: "mul", + Infix: "*", + Decl: types.NewFunction( + types.Args(types.N, types.N), + types.N, + ), +} + +// Divide divides the first number by the second number. +var Divide = &Builtin{ + Name: "div", + Infix: "/", + Decl: types.NewFunction( + types.Args(types.N, types.N), + types.N, + ), +} + +// Round rounds the number to the nearest integer. +var Round = &Builtin{ + Name: "round", + Decl: types.NewFunction( + types.Args(types.N), + types.N, + ), +} + +// Ceil rounds the number up to the nearest integer. +var Ceil = &Builtin{ + Name: "ceil", + Decl: types.NewFunction( + types.Args(types.N), + types.N, + ), +} + +// Floor rounds the number down to the nearest integer. +var Floor = &Builtin{ + Name: "floor", + Decl: types.NewFunction( + types.Args(types.N), + types.N, + ), +} + +// Abs returns the number without its sign. +var Abs = &Builtin{ + Name: "abs", + Decl: types.NewFunction( + types.Args(types.N), + types.N, + ), +} + +// Rem returns the remainder for x%y for y != 0. +var Rem = &Builtin{ + Name: "rem", + Infix: "%", + Decl: types.NewFunction( + types.Args(types.N, types.N), + types.N, + ), +} + +/** + * Bitwise + */ + +// BitsOr returns the bitwise "or" of two integers. +var BitsOr = &Builtin{ + Name: "bits.or", + Decl: types.NewFunction( + types.Args(types.N, types.N), + types.N, + ), +} + +// BitsAnd returns the bitwise "and" of two integers. +var BitsAnd = &Builtin{ + Name: "bits.and", + Decl: types.NewFunction( + types.Args(types.N, types.N), + types.N, + ), +} + +// BitsNegate returns the bitwise "negation" of an integer (i.e. flips each +// bit). +var BitsNegate = &Builtin{ + Name: "bits.negate", + Decl: types.NewFunction( + types.Args(types.N), + types.N, + ), +} + +// BitsXOr returns the bitwise "exclusive-or" of two integers. +var BitsXOr = &Builtin{ + Name: "bits.xor", + Decl: types.NewFunction( + types.Args(types.N, types.N), + types.N, + ), +} + +// BitsShiftLeft returns a new integer with its bits shifted some value to the +// left. +var BitsShiftLeft = &Builtin{ + Name: "bits.lsh", + Decl: types.NewFunction( + types.Args(types.N, types.N), + types.N, + ), +} + +// BitsShiftRight returns a new integer with its bits shifted some value to the +// right. +var BitsShiftRight = &Builtin{ + Name: "bits.rsh", + Decl: types.NewFunction( + types.Args(types.N, types.N), + types.N, + ), +} + +/** + * Sets + */ + +// And performs an intersection operation on sets. +var And = &Builtin{ + Name: "and", + Infix: "&", + Decl: types.NewFunction( + types.Args( + types.NewSet(types.A), + types.NewSet(types.A), + ), + types.NewSet(types.A), + ), +} + +// Or performs a union operation on sets. +var Or = &Builtin{ + Name: "or", + Infix: "|", + Decl: types.NewFunction( + types.Args( + types.NewSet(types.A), + types.NewSet(types.A), + ), + types.NewSet(types.A), + ), +} + +/** + * Aggregates + */ + +// Count takes a collection or string and counts the number of elements in it. +var Count = &Builtin{ + Name: "count", + Decl: types.NewFunction( + types.Args( + types.NewAny( + types.NewSet(types.A), + types.NewArray(nil, types.A), + types.NewObject(nil, types.NewDynamicProperty(types.A, types.A)), + types.S, + ), + ), + types.N, + ), +} + +// Sum takes an array or set of numbers and sums them. +var Sum = &Builtin{ + Name: "sum", + Decl: types.NewFunction( + types.Args( + types.NewAny( + types.NewSet(types.N), + types.NewArray(nil, types.N), + ), + ), + types.N, + ), +} + +// Product takes an array or set of numbers and multiplies them. +var Product = &Builtin{ + Name: "product", + Decl: types.NewFunction( + types.Args( + types.NewAny( + types.NewSet(types.N), + types.NewArray(nil, types.N), + ), + ), + types.N, + ), +} + +// Max returns the maximum value in a collection. +var Max = &Builtin{ + Name: "max", + Decl: types.NewFunction( + types.Args( + types.NewAny( + types.NewSet(types.A), + types.NewArray(nil, types.A), + ), + ), + types.A, + ), +} + +// Min returns the minimum value in a collection. +var Min = &Builtin{ + Name: "min", + Decl: types.NewFunction( + types.Args( + types.NewAny( + types.NewSet(types.A), + types.NewArray(nil, types.A), + ), + ), + types.A, + ), +} + +// All takes a list and returns true if all of the items +// are true. A collection of length 0 returns true. +var All = &Builtin{ + Name: "all", + Decl: types.NewFunction( + types.Args( + types.NewAny( + types.NewSet(types.A), + types.NewArray(nil, types.A), + ), + ), + types.B, + ), +} + +// Any takes a collection and returns true if any of the items +// is true. A collection of length 0 returns false. +var Any = &Builtin{ + Name: "any", + Decl: types.NewFunction( + types.Args( + types.NewAny( + types.NewSet(types.A), + types.NewArray(nil, types.A), + ), + ), + types.B, + ), +} + +/** + * Arrays + */ + +// ArrayConcat returns the result of concatenating two arrays together. +var ArrayConcat = &Builtin{ + Name: "array.concat", + Decl: types.NewFunction( + types.Args( + types.NewArray(nil, types.A), + types.NewArray(nil, types.A), + ), + types.NewArray(nil, types.A), + ), +} + +// ArraySlice returns a slice of a given array +var ArraySlice = &Builtin{ + Name: "array.slice", + Decl: types.NewFunction( + types.Args( + types.NewArray(nil, types.A), + types.NewNumber(), + types.NewNumber(), + ), + types.NewArray(nil, types.A), + ), +} + +// ArrayReverse returns a given array, reversed +var ArrayReverse = &Builtin{ + Name: "array.reverse", + Decl: types.NewFunction( + types.Args( + types.NewArray(nil, types.A), + ), + types.NewArray(nil, types.A), + ), +} + +/** + * Conversions + */ + +// ToNumber takes a string, bool, or number value and converts it to a number. +// Strings are converted to numbers using strconv.Atoi. +// Boolean false is converted to 0 and boolean true is converted to 1. +var ToNumber = &Builtin{ + Name: "to_number", + Decl: types.NewFunction( + types.Args( + types.NewAny( + types.N, + types.S, + types.B, + types.NewNull(), + ), + ), + types.N, + ), +} + +/** + * Regular Expressions + */ + +// RegexMatch takes two strings and evaluates to true if the string in the second +// position matches the pattern in the first position. +var RegexMatch = &Builtin{ + Name: "regex.match", + Decl: types.NewFunction( + types.Args( + types.S, + types.S, + ), + types.B, + ), +} + +// RegexIsValid returns true if the regex pattern string is valid, otherwise false. +var RegexIsValid = &Builtin{ + Name: "regex.is_valid", + Decl: types.NewFunction( + types.Args( + types.S, + ), + types.B, + ), +} + +// RegexFindAllStringSubmatch returns an array of all successive matches of the expression. +// It takes two strings and a number, the pattern, the value and number of matches to +// return, -1 means all matches. +var RegexFindAllStringSubmatch = &Builtin{ + Name: "regex.find_all_string_submatch_n", + Decl: types.NewFunction( + types.Args( + types.S, + types.S, + types.N, + ), + types.NewArray(nil, types.NewArray(nil, types.S)), + ), +} + +// RegexTemplateMatch takes two strings and evaluates to true if the string in the second +// position matches the pattern in the first position. +var RegexTemplateMatch = &Builtin{ + Name: "regex.template_match", + Decl: types.NewFunction( + types.Args( + types.S, + types.S, + types.S, + types.S, + ), + types.B, + ), +} + +// RegexSplit splits the input string by the occurrences of the given pattern. +var RegexSplit = &Builtin{ + Name: "regex.split", + Decl: types.NewFunction( + types.Args( + types.S, + types.S, + ), + types.NewArray(nil, types.S), + ), +} + +// RegexFind takes two strings and a number, the pattern, the value and number of match values to +// return, -1 means all match values. +var RegexFind = &Builtin{ + Name: "regex.find_n", + Decl: types.NewFunction( + types.Args( + types.S, + types.S, + types.N, + ), + types.NewArray(nil, types.S), + ), +} + +// GlobsMatch takes two strings regexp-style strings and evaluates to true if their +// intersection matches a non-empty set of non-empty strings. +// Examples: +// - "a.a." and ".b.b" -> true. +// - "[a-z]*" and [0-9]+" -> not true. +var GlobsMatch = &Builtin{ + Name: "regex.globs_match", + Decl: types.NewFunction( + types.Args( + types.S, + types.S, + ), + types.B, + ), +} + +/** + * Strings + */ + +// Concat joins an array of strings with an input string. +var Concat = &Builtin{ + Name: "concat", + Decl: types.NewFunction( + types.Args( + types.S, + types.NewAny( + types.NewSet(types.S), + types.NewArray(nil, types.S), + ), + ), + types.S, + ), +} + +// FormatInt returns the string representation of the number in the given base after converting it to an integer value. +var FormatInt = &Builtin{ + Name: "format_int", + Decl: types.NewFunction( + types.Args( + types.N, + types.N, + ), + types.S, + ), +} + +// IndexOf returns the index of a substring contained inside a string +var IndexOf = &Builtin{ + Name: "indexof", + Decl: types.NewFunction( + types.Args( + types.S, + types.S, + ), + types.N, + ), +} + +// Substring returns the portion of a string for a given start index and a length. +// If the length is less than zero, then substring returns the remainder of the string. +var Substring = &Builtin{ + Name: "substring", + Decl: types.NewFunction( + types.Args( + types.S, + types.N, + types.N, + ), + types.S, + ), +} + +// Contains returns true if the search string is included in the base string +var Contains = &Builtin{ + Name: "contains", + Decl: types.NewFunction( + types.Args( + types.S, + types.S, + ), + types.B, + ), +} + +// StartsWith returns true if the search string begins with the base string +var StartsWith = &Builtin{ + Name: "startswith", + Decl: types.NewFunction( + types.Args( + types.S, + types.S, + ), + types.B, + ), +} + +// EndsWith returns true if the search string begins with the base string +var EndsWith = &Builtin{ + Name: "endswith", + Decl: types.NewFunction( + types.Args( + types.S, + types.S, + ), + types.B, + ), +} + +// Lower returns the input string but with all characters in lower-case +var Lower = &Builtin{ + Name: "lower", + Decl: types.NewFunction( + types.Args(types.S), + types.S, + ), +} + +// Upper returns the input string but with all characters in upper-case +var Upper = &Builtin{ + Name: "upper", + Decl: types.NewFunction( + types.Args(types.S), + types.S, + ), +} + +// Split returns an array containing elements of the input string split on a delimiter. +var Split = &Builtin{ + Name: "split", + Decl: types.NewFunction( + types.Args( + types.S, + types.S, + ), + types.NewArray(nil, types.S), + ), +} + +// Replace returns the given string with all instances of the second argument replaced +// by the third. +var Replace = &Builtin{ + Name: "replace", + Decl: types.NewFunction( + types.Args( + types.S, + types.S, + types.S, + ), + types.S, + ), +} + +// ReplaceN replaces a string from a list of old, new string pairs. +// Replacements are performed in the order they appear in the target string, without overlapping matches. +// The old string comparisons are done in argument order. +var ReplaceN = &Builtin{ + Name: "strings.replace_n", + Decl: types.NewFunction( + types.Args( + types.NewObject( + nil, + types.NewDynamicProperty( + types.S, + types.S)), + types.S, + ), + types.S, + ), +} + +// Trim returns the given string with all leading or trailing instances of the second +// argument removed. +var Trim = &Builtin{ + Name: "trim", + Decl: types.NewFunction( + types.Args( + types.S, + types.S, + ), + types.S, + ), +} + +// TrimLeft returns the given string with all leading instances of second argument removed. +var TrimLeft = &Builtin{ + Name: "trim_left", + Decl: types.NewFunction( + types.Args( + types.S, + types.S, + ), + types.S, + ), +} + +// TrimPrefix returns the given string without the second argument prefix string. +// If the given string doesn't start with prefix, it is returned unchanged. +var TrimPrefix = &Builtin{ + Name: "trim_prefix", + Decl: types.NewFunction( + types.Args( + types.S, + types.S, + ), + types.S, + ), +} + +// TrimRight returns the given string with all trailing instances of second argument removed. +var TrimRight = &Builtin{ + Name: "trim_right", + Decl: types.NewFunction( + types.Args( + types.S, + types.S, + ), + types.S, + ), +} + +// TrimSuffix returns the given string without the second argument suffix string. +// If the given string doesn't end with suffix, it is returned unchanged. +var TrimSuffix = &Builtin{ + Name: "trim_suffix", + Decl: types.NewFunction( + types.Args( + types.S, + types.S, + ), + types.S, + ), +} + +// TrimSpace return the given string with all leading and trailing white space removed. +var TrimSpace = &Builtin{ + Name: "trim_space", + Decl: types.NewFunction( + types.Args( + types.S, + ), + types.S, + ), +} + +// Sprintf returns the given string, formatted. +var Sprintf = &Builtin{ + Name: "sprintf", + Decl: types.NewFunction( + types.Args( + types.S, + types.NewArray(nil, types.A), + ), + types.S, + ), +} + +// StringReverse returns the given string, reversed. +var StringReverse = &Builtin{ + Name: "strings.reverse", + Decl: types.NewFunction( + types.Args( + types.S, + ), + types.S, + ), +} + +/** + * Numbers + */ + +// RandIntn returns a random number 0 - n +var RandIntn = &Builtin{ + Name: "rand.intn", + Decl: types.NewFunction( + types.Args( + types.S, + types.N, + ), + types.N, + ), +} + +// NumbersRange returns an array of numbers in the given inclusive range. +var NumbersRange = &Builtin{ + Name: "numbers.range", + Decl: types.NewFunction( + types.Args( + types.N, + types.N, + ), + types.NewArray(nil, types.N), + ), +} + +/** + * Units + */ + +// UnitsParseBytes converts strings like 10GB, 5K, 4mb, and the like into an +// integer number of bytes. +var UnitsParseBytes = &Builtin{ + Name: "units.parse_bytes", + Decl: types.NewFunction( + types.Args( + types.S, + ), + types.N, + ), +} + +// +/** + * Type + */ + +// UUIDRFC4122 returns a version 4 UUID string. +var UUIDRFC4122 = &Builtin{ + Name: "uuid.rfc4122", + Decl: types.NewFunction( + types.Args(types.S), + types.S, + ), +} + +/** + * JSON + */ + +// JSONMarshal serializes the input term. +var JSONMarshal = &Builtin{ + Name: "json.marshal", + Decl: types.NewFunction( + types.Args(types.A), + types.S, + ), +} + +// JSONUnmarshal deserializes the input string. +var JSONUnmarshal = &Builtin{ + Name: "json.unmarshal", + Decl: types.NewFunction( + types.Args(types.S), + types.A, + ), +} + +// JSONIsValid verifies the input string is a valid JSON document. +var JSONIsValid = &Builtin{ + Name: "json.is_valid", + Decl: types.NewFunction( + types.Args(types.S), + types.B, + ), +} + +// JSONFilter filters the JSON object +var JSONFilter = &Builtin{ + Name: "json.filter", + Decl: types.NewFunction( + types.Args( + types.NewObject( + nil, + types.NewDynamicProperty(types.A, types.A), + ), + types.NewAny( + types.NewArray( + nil, + types.NewAny( + types.S, + types.NewArray( + nil, + types.A, + ), + ), + ), + types.NewSet( + types.NewAny( + types.S, + types.NewArray( + nil, + types.A, + ), + ), + ), + ), + ), + types.A, + ), +} + +// JSONRemove removes paths in the JSON object +var JSONRemove = &Builtin{ + Name: "json.remove", + Decl: types.NewFunction( + types.Args( + types.NewObject( + nil, + types.NewDynamicProperty(types.A, types.A), + ), + types.NewAny( + types.NewArray( + nil, + types.NewAny( + types.S, + types.NewArray( + nil, + types.A, + ), + ), + ), + types.NewSet( + types.NewAny( + types.S, + types.NewArray( + nil, + types.A, + ), + ), + ), + ), + ), + types.A, + ), +} + +// JSONPatch patches a JSON object according to RFC6902 +var JSONPatch = &Builtin{ + Name: "json.patch", + Decl: types.NewFunction( + types.Args( + types.A, + types.NewArray( + nil, + types.NewObject( + []*types.StaticProperty{ + {Key: "op", Value: types.S}, + {Key: "path", Value: types.A}, + }, + types.NewDynamicProperty(types.A, types.A), + ), + ), + ), + types.A, + ), +} + +// ObjectGet returns takes an object and returns a value under its key if +// present, otherwise it returns the default. +var ObjectGet = &Builtin{ + Name: "object.get", + Decl: types.NewFunction( + types.Args( + types.NewObject(nil, types.NewDynamicProperty(types.A, types.A)), + types.A, + types.A, + ), + types.A, + ), +} + +// ObjectUnion creates a new object that is the asymmetric union of two objects +var ObjectUnion = &Builtin{ + Name: "object.union", + Decl: types.NewFunction( + types.Args( + types.NewObject( + nil, + types.NewDynamicProperty(types.A, types.A), + ), + types.NewObject( + nil, + types.NewDynamicProperty(types.A, types.A), + ), + ), + types.A, + ), +} + +// ObjectRemove Removes specified keys from an object +var ObjectRemove = &Builtin{ + Name: "object.remove", + Decl: types.NewFunction( + types.Args( + types.NewObject( + nil, + types.NewDynamicProperty(types.A, types.A), + ), + types.NewAny( + types.NewArray(nil, types.A), + types.NewSet(types.A), + types.NewObject(nil, types.NewDynamicProperty(types.A, types.A)), + ), + ), + types.A, + ), +} + +// ObjectFilter filters the object by keeping only specified keys +var ObjectFilter = &Builtin{ + Name: "object.filter", + Decl: types.NewFunction( + types.Args( + types.NewObject( + nil, + types.NewDynamicProperty(types.A, types.A), + ), + types.NewAny( + types.NewArray(nil, types.A), + types.NewSet(types.A), + types.NewObject(nil, types.NewDynamicProperty(types.A, types.A)), + ), + ), + types.A, + ), +} + +// Base64Encode serializes the input string into base64 encoding. +var Base64Encode = &Builtin{ + Name: "base64.encode", + Decl: types.NewFunction( + types.Args(types.S), + types.S, + ), +} + +// Base64Decode deserializes the base64 encoded input string. +var Base64Decode = &Builtin{ + Name: "base64.decode", + Decl: types.NewFunction( + types.Args(types.S), + types.S, + ), +} + +// Base64IsValid verifies the input string is base64 encoded. +var Base64IsValid = &Builtin{ + Name: "base64.is_valid", + Decl: types.NewFunction( + types.Args(types.S), + types.B, + ), +} + +// Base64UrlEncode serializes the input string into base64url encoding. +var Base64UrlEncode = &Builtin{ + Name: "base64url.encode", + Decl: types.NewFunction( + types.Args(types.S), + types.S, + ), +} + +// Base64UrlEncodeNoPad serializes the input string into base64url encoding without padding. +var Base64UrlEncodeNoPad = &Builtin{ + Name: "base64url.encode_no_pad", + Decl: types.NewFunction( + types.Args(types.S), + types.S, + ), +} + +// Base64UrlDecode deserializes the base64url encoded input string. +var Base64UrlDecode = &Builtin{ + Name: "base64url.decode", + Decl: types.NewFunction( + types.Args(types.S), + types.S, + ), +} + +// URLQueryDecode decodes a URL encoded input string. +var URLQueryDecode = &Builtin{ + Name: "urlquery.decode", + Decl: types.NewFunction( + types.Args(types.S), + types.S, + ), +} + +// URLQueryEncode encodes the input string into a URL encoded string. +var URLQueryEncode = &Builtin{ + Name: "urlquery.encode", + Decl: types.NewFunction( + types.Args(types.S), + types.S, + ), +} + +// URLQueryEncodeObject encodes the given JSON into a URL encoded query string. +var URLQueryEncodeObject = &Builtin{ + Name: "urlquery.encode_object", + Decl: types.NewFunction( + types.Args( + types.NewObject( + nil, + types.NewDynamicProperty( + types.S, + types.NewAny( + types.S, + types.NewArray(nil, types.S), + types.NewSet(types.S))))), + types.S, + ), +} + +// URLQueryDecodeObject decodes the given URL query string into an object. +var URLQueryDecodeObject = &Builtin{ + Name: "urlquery.decode_object", + Decl: types.NewFunction( + types.Args(types.S), + types.NewObject(nil, types.NewDynamicProperty( + types.S, + types.NewArray(nil, types.S))), + ), +} + +// YAMLMarshal serializes the input term. +var YAMLMarshal = &Builtin{ + Name: "yaml.marshal", + Decl: types.NewFunction( + types.Args(types.A), + types.S, + ), +} + +// YAMLUnmarshal deserializes the input string. +var YAMLUnmarshal = &Builtin{ + Name: "yaml.unmarshal", + Decl: types.NewFunction( + types.Args(types.S), + types.A, + ), +} + +// YAMLIsValid verifies the input string is a valid YAML document. +var YAMLIsValid = &Builtin{ + Name: "yaml.is_valid", + Decl: types.NewFunction( + types.Args(types.S), + types.B, + ), +} + +// HexEncode serializes the input string into hex encoding. +var HexEncode = &Builtin{ + Name: "hex.encode", + Decl: types.NewFunction( + types.Args(types.S), + types.S, + ), +} + +// HexDecode deserializes the hex encoded input string. +var HexDecode = &Builtin{ + Name: "hex.decode", + Decl: types.NewFunction( + types.Args(types.S), + types.S, + ), +} + +/** + * Tokens + */ + +// JWTDecode decodes a JSON Web Token and outputs it as an Object. +var JWTDecode = &Builtin{ + Name: "io.jwt.decode", + Decl: types.NewFunction( + types.Args(types.S), + types.NewArray([]types.Type{ + types.NewObject(nil, types.NewDynamicProperty(types.A, types.A)), + types.NewObject(nil, types.NewDynamicProperty(types.A, types.A)), + types.S, + }, nil), + ), +} + +// JWTVerifyRS256 verifies if a RS256 JWT signature is valid or not. +var JWTVerifyRS256 = &Builtin{ + Name: "io.jwt.verify_rs256", + Decl: types.NewFunction( + types.Args( + types.S, + types.S, + ), + types.B, + ), +} + +// JWTVerifyRS384 verifies if a RS384 JWT signature is valid or not. +var JWTVerifyRS384 = &Builtin{ + Name: "io.jwt.verify_rs384", + Decl: types.NewFunction( + types.Args( + types.S, + types.S, + ), + types.B, + ), +} + +// JWTVerifyRS512 verifies if a RS512 JWT signature is valid or not. +var JWTVerifyRS512 = &Builtin{ + Name: "io.jwt.verify_rs512", + Decl: types.NewFunction( + types.Args( + types.S, + types.S, + ), + types.B, + ), +} + +// JWTVerifyPS256 verifies if a PS256 JWT signature is valid or not. +var JWTVerifyPS256 = &Builtin{ + Name: "io.jwt.verify_ps256", + Decl: types.NewFunction( + types.Args( + types.S, + types.S, + ), + types.B, + ), +} + +// JWTVerifyPS384 verifies if a PS384 JWT signature is valid or not. +var JWTVerifyPS384 = &Builtin{ + Name: "io.jwt.verify_ps384", + Decl: types.NewFunction( + types.Args( + types.S, + types.S, + ), + types.B, + ), +} + +// JWTVerifyPS512 verifies if a PS512 JWT signature is valid or not. +var JWTVerifyPS512 = &Builtin{ + Name: "io.jwt.verify_ps512", + Decl: types.NewFunction( + types.Args( + types.S, + types.S, + ), + types.B, + ), +} + +// JWTVerifyES256 verifies if a ES256 JWT signature is valid or not. +var JWTVerifyES256 = &Builtin{ + Name: "io.jwt.verify_es256", + Decl: types.NewFunction( + types.Args( + types.S, + types.S, + ), + types.B, + ), +} + +// JWTVerifyES384 verifies if a ES384 JWT signature is valid or not. +var JWTVerifyES384 = &Builtin{ + Name: "io.jwt.verify_es384", + Decl: types.NewFunction( + types.Args( + types.S, + types.S, + ), + types.B, + ), +} + +// JWTVerifyES512 verifies if a ES512 JWT signature is valid or not. +var JWTVerifyES512 = &Builtin{ + Name: "io.jwt.verify_es512", + Decl: types.NewFunction( + types.Args( + types.S, + types.S, + ), + types.B, + ), +} + +// JWTVerifyHS256 verifies if a HS256 (secret) JWT signature is valid or not. +var JWTVerifyHS256 = &Builtin{ + Name: "io.jwt.verify_hs256", + Decl: types.NewFunction( + types.Args( + types.S, + types.S, + ), + types.B, + ), +} + +// JWTVerifyHS384 verifies if a HS384 (secret) JWT signature is valid or not. +var JWTVerifyHS384 = &Builtin{ + Name: "io.jwt.verify_hs384", + Decl: types.NewFunction( + types.Args( + types.S, + types.S, + ), + types.B, + ), +} + +// JWTVerifyHS512 verifies if a HS512 (secret) JWT signature is valid or not. +var JWTVerifyHS512 = &Builtin{ + Name: "io.jwt.verify_hs512", + Decl: types.NewFunction( + types.Args( + types.S, + types.S, + ), + types.B, + ), +} + +// JWTDecodeVerify verifies a JWT signature under parameterized constraints and decodes the claims if it is valid. +var JWTDecodeVerify = &Builtin{ + Name: "io.jwt.decode_verify", + Decl: types.NewFunction( + types.Args( + types.S, + types.NewObject(nil, types.NewDynamicProperty(types.S, types.A)), + ), + types.NewArray([]types.Type{ + types.B, + types.NewObject(nil, types.NewDynamicProperty(types.A, types.A)), + types.NewObject(nil, types.NewDynamicProperty(types.A, types.A)), + }, nil), + ), +} + +// JWTEncodeSignRaw encodes and optionally sign a JSON Web Token. +// Inputs are protected headers, payload, secret +var JWTEncodeSignRaw = &Builtin{ + Name: "io.jwt.encode_sign_raw", + Decl: types.NewFunction( + types.Args( + types.S, + types.S, + types.S, + ), + types.S, + ), +} + +// JWTEncodeSign encodes and optionally sign a JSON Web Token. +// Inputs are protected headers, payload, secret +var JWTEncodeSign = &Builtin{ + Name: "io.jwt.encode_sign", + Decl: types.NewFunction( + types.Args( + types.NewObject(nil, types.NewDynamicProperty(types.S, types.A)), + types.NewObject(nil, types.NewDynamicProperty(types.S, types.A)), + types.NewObject(nil, types.NewDynamicProperty(types.S, types.A)), + ), + types.S, + ), +} + +/** + * Time + */ + +// NowNanos returns the current time since epoch in nanoseconds. +var NowNanos = &Builtin{ + Name: "time.now_ns", + Decl: types.NewFunction( + nil, + types.N, + ), +} + +// ParseNanos returns the time in nanoseconds parsed from the string in the given format. +var ParseNanos = &Builtin{ + Name: "time.parse_ns", + Decl: types.NewFunction( + types.Args( + types.S, + types.S, + ), + types.N, + ), +} + +// ParseRFC3339Nanos returns the time in nanoseconds parsed from the string in RFC3339 format. +var ParseRFC3339Nanos = &Builtin{ + Name: "time.parse_rfc3339_ns", + Decl: types.NewFunction( + types.Args(types.S), + types.N, + ), +} + +// ParseDurationNanos returns the duration in nanoseconds represented by a duration string. +// Duration string is similar to the Go time.ParseDuration string +var ParseDurationNanos = &Builtin{ + Name: "time.parse_duration_ns", + Decl: types.NewFunction( + types.Args(types.S), + types.N, + ), +} + +// Date returns the [year, month, day] for the nanoseconds since epoch. +var Date = &Builtin{ + Name: "time.date", + Decl: types.NewFunction( + types.Args( + types.NewAny( + types.N, + types.NewArray([]types.Type{types.N, types.S}, nil), + ), + ), + types.NewArray([]types.Type{types.N, types.N, types.N}, nil), + ), +} + +// Clock returns the [hour, minute, second] of the day for the nanoseconds since epoch. +var Clock = &Builtin{ + Name: "time.clock", + Decl: types.NewFunction( + types.Args( + types.NewAny( + types.N, + types.NewArray([]types.Type{types.N, types.S}, nil), + ), + ), + types.NewArray([]types.Type{types.N, types.N, types.N}, nil), + ), +} + +// Weekday returns the day of the week (Monday, Tuesday, ...) for the nanoseconds since epoch. +var Weekday = &Builtin{ + Name: "time.weekday", + Decl: types.NewFunction( + types.Args( + types.NewAny( + types.N, + types.NewArray([]types.Type{types.N, types.S}, nil), + ), + ), + types.S, + ), +} + +// AddDate returns the nanoseconds since epoch after adding years, months and days to nanoseconds. +var AddDate = &Builtin{ + Name: "time.add_date", + Decl: types.NewFunction( + types.Args( + types.N, + types.N, + types.N, + types.N, + ), + types.N, + ), +} + +// Diff returns the difference [years, months, days, hours, minutes, seconds] between two unix timestamps in nanoseconds +var Diff = &Builtin{ + Name: "time.diff", + Decl: types.NewFunction( + types.Args( + types.NewAny( + types.N, + types.NewArray([]types.Type{types.N, types.S}, nil), + ), + types.NewAny( + types.N, + types.NewArray([]types.Type{types.N, types.S}, nil), + ), + ), + types.NewArray([]types.Type{types.N, types.N, types.N, types.N, types.N, types.N}, nil), + ), +} + +/** + * Crypto. + */ + +// CryptoX509ParseCertificates returns one or more certificates from the given +// base64 encoded string containing DER encoded certificates that have been +// concatenated. +var CryptoX509ParseCertificates = &Builtin{ + Name: "crypto.x509.parse_certificates", + Decl: types.NewFunction( + types.Args(types.S), + types.NewArray(nil, types.NewObject(nil, types.NewDynamicProperty(types.S, types.A))), + ), +} + +// CryptoX509ParseAndVerifyCertificates returns one or more certificates from the given +// string containing PEM or base64 encoded DER certificates after verifying the supplied +// certificates form a complete certificate chain back to a trusted root. +// +// The first certificate is treated as the root and the last is treated as the leaf, +// with all others being treated as intermediates +var CryptoX509ParseAndVerifyCertificates = &Builtin{ + Name: "crypto.x509.parse_and_verify_certificates", + Decl: types.NewFunction( + types.Args(types.S), + types.NewArray([]types.Type{ + types.B, + types.NewArray(nil, types.NewObject(nil, types.NewDynamicProperty(types.S, types.A))), + }, nil), + ), +} + +// CryptoX509ParseCertificateRequest returns a PKCS #10 certificate signing +// request from the given PEM-encoded PKCS#10 certificate signing request. +var CryptoX509ParseCertificateRequest = &Builtin{ + Name: "crypto.x509.parse_certificate_request", + Decl: types.NewFunction( + types.Args(types.S), + types.NewObject(nil, types.NewDynamicProperty(types.S, types.A)), + ), +} + +// CryptoX509ParseRSAPrivateKey returns a JWK for signing a JWT from the given +// PEM-encoded RSA private key. +var CryptoX509ParseRSAPrivateKey = &Builtin{ + Name: "crypto.x509.parse_rsa_private_key", + Decl: types.NewFunction( + types.Args(types.S), + types.NewObject(nil, types.NewDynamicProperty(types.S, types.A)), + ), +} + +// CryptoMd5 returns a string representing the input string hashed with the md5 function +var CryptoMd5 = &Builtin{ + Name: "crypto.md5", + Decl: types.NewFunction( + types.Args(types.S), + types.S, + ), +} + +// CryptoSha1 returns a string representing the input string hashed with the sha1 function +var CryptoSha1 = &Builtin{ + Name: "crypto.sha1", + Decl: types.NewFunction( + types.Args(types.S), + types.S, + ), +} + +// CryptoSha256 returns a string representing the input string hashed with the sha256 function +var CryptoSha256 = &Builtin{ + Name: "crypto.sha256", + Decl: types.NewFunction( + types.Args(types.S), + types.S, + ), +} + +// CryptoHmacMd5 returns a string representing the MD-5 HMAC of the input message using the input key +// Inputs are message, key +var CryptoHmacMd5 = &Builtin{ + Name: "crypto.hmac.md5", + Decl: types.NewFunction( + types.Args( + types.S, + types.S, + ), + types.S, + ), +} + +// CryptoHmacSha1 returns a string representing the SHA-1 HMAC of the input message using the input key +// Inputs are message, key +var CryptoHmacSha1 = &Builtin{ + Name: "crypto.hmac.sha1", + Decl: types.NewFunction( + types.Args( + types.S, + types.S, + ), + types.S, + ), +} + +// CryptoHmacSha256 returns a string representing the SHA-256 HMAC of the input message using the input key +// Inputs are message, key +var CryptoHmacSha256 = &Builtin{ + Name: "crypto.hmac.sha256", + Decl: types.NewFunction( + types.Args( + types.S, + types.S, + ), + types.S, + ), +} + +// CryptoHmacSha512 returns a string representing the SHA-512 HMAC of the input message using the input key +// Inputs are message, key +var CryptoHmacSha512 = &Builtin{ + Name: "crypto.hmac.sha512", + Decl: types.NewFunction( + types.Args( + types.S, + types.S, + ), + types.S, + ), +} + +/** + * Graphs. + */ + +// WalkBuiltin generates [path, value] tuples for all nested documents +// (recursively). +var WalkBuiltin = &Builtin{ + Name: "walk", + Relation: true, + Decl: types.NewFunction( + types.Args(types.A), + types.NewArray( + []types.Type{ + types.NewArray(nil, types.A), + types.A, + }, + nil, + ), + ), +} + +// ReachableBuiltin computes the set of reachable nodes in the graph from a set +// of starting nodes. +var ReachableBuiltin = &Builtin{ + Name: "graph.reachable", + Decl: types.NewFunction( + types.Args( + types.NewObject( + nil, + types.NewDynamicProperty( + types.A, + types.NewAny( + types.NewSet(types.A), + types.NewArray(nil, types.A)), + )), + types.NewAny(types.NewSet(types.A), types.NewArray(nil, types.A)), + ), + types.NewSet(types.A), + ), +} + +/** + * Sorting + */ + +// Sort returns a sorted array. +var Sort = &Builtin{ + Name: "sort", + Decl: types.NewFunction( + types.Args( + types.NewAny( + types.NewArray(nil, types.A), + types.NewSet(types.A), + ), + ), + types.NewArray(nil, types.A), + ), +} + +/** + * Type + */ + +// IsNumber returns true if the input value is a number +var IsNumber = &Builtin{ + Name: "is_number", + Decl: types.NewFunction( + types.Args( + types.A, + ), + types.B, + ), +} + +// IsString returns true if the input value is a string. +var IsString = &Builtin{ + Name: "is_string", + Decl: types.NewFunction( + types.Args( + types.A, + ), + types.B, + ), +} + +// IsBoolean returns true if the input value is a boolean. +var IsBoolean = &Builtin{ + Name: "is_boolean", + Decl: types.NewFunction( + types.Args( + types.A, + ), + types.B, + ), +} + +// IsArray returns true if the input value is an array. +var IsArray = &Builtin{ + Name: "is_array", + Decl: types.NewFunction( + types.Args( + types.A, + ), + types.B, + ), +} + +// IsSet returns true if the input value is a set. +var IsSet = &Builtin{ + Name: "is_set", + Decl: types.NewFunction( + types.Args( + types.A, + ), + types.B, + ), +} + +// IsObject returns true if the input value is an object. +var IsObject = &Builtin{ + Name: "is_object", + Decl: types.NewFunction( + types.Args( + types.A, + ), + types.B, + ), +} + +// IsNull returns true if the input value is null. +var IsNull = &Builtin{ + Name: "is_null", + Decl: types.NewFunction( + types.Args( + types.A, + ), + types.B, + ), +} + +/** + * Type Name + */ + +// TypeNameBuiltin returns the type of the input. +var TypeNameBuiltin = &Builtin{ + Name: "type_name", + Decl: types.NewFunction( + types.Args( + types.NewAny( + types.A, + ), + ), + types.S, + ), +} + +/** + * HTTP Request + */ + +// HTTPSend returns a HTTP response to the given HTTP request. +var HTTPSend = &Builtin{ + Name: "http.send", + Decl: types.NewFunction( + types.Args( + types.NewObject(nil, types.NewDynamicProperty(types.S, types.A)), + ), + types.NewObject(nil, types.NewDynamicProperty(types.A, types.A)), + ), +} + +/** + * Rego + */ + +// RegoParseModule parses the input Rego file and returns a JSON representation +// of the AST. +var RegoParseModule = &Builtin{ + Name: "rego.parse_module", + Decl: types.NewFunction( + types.Args( + types.S, + types.S, + ), + types.NewObject(nil, types.NewDynamicProperty(types.S, types.A)), // TODO(tsandall): import AST schema + ), +} + +/** + * OPA + */ + +// OPARuntime returns an object containing OPA runtime information such as the +// configuration that OPA was booted with. +var OPARuntime = &Builtin{ + Name: "opa.runtime", + Decl: types.NewFunction( + nil, + types.NewObject(nil, types.NewDynamicProperty(types.S, types.A)), + ), +} + +/** + * Trace + */ + +// Trace prints a note that is included in the query explanation. +var Trace = &Builtin{ + Name: "trace", + Decl: types.NewFunction( + types.Args( + types.S, + ), + types.B, + ), +} + +/** + * Set + */ + +// Intersection returns the intersection of the given input sets +var Intersection = &Builtin{ + Name: "intersection", + Decl: types.NewFunction( + types.Args( + types.NewSet(types.NewSet(types.A)), + ), + types.NewSet(types.A), + ), +} + +// Union returns the union of the given input sets +var Union = &Builtin{ + Name: "union", + Decl: types.NewFunction( + types.Args( + types.NewSet(types.NewSet(types.A)), + ), + types.NewSet(types.A), + ), +} + +/** + * Glob + */ + +// GlobMatch - not to be confused with regex.globs_match - parses and matches strings against the glob notation. +var GlobMatch = &Builtin{ + Name: "glob.match", + Decl: types.NewFunction( + types.Args( + types.S, + types.NewArray(nil, types.S), + types.S, + ), + types.B, + ), +} + +// GlobQuoteMeta returns a string which represents a version of the pattern where all asterisks have been escaped. +var GlobQuoteMeta = &Builtin{ + Name: "glob.quote_meta", + Decl: types.NewFunction( + types.Args( + types.S, + ), + types.S, + ), +} + +/** + * Networking + */ + +// NetCIDRIntersects checks if a cidr intersects with another cidr and returns true or false +var NetCIDRIntersects = &Builtin{ + Name: "net.cidr_intersects", + Decl: types.NewFunction( + types.Args( + types.S, + types.S, + ), + types.B, + ), +} + +// NetCIDRExpand returns a set of hosts inside the specified cidr. +var NetCIDRExpand = &Builtin{ + Name: "net.cidr_expand", + Decl: types.NewFunction( + types.Args( + types.S, + ), + types.NewSet(types.S), + ), +} + +// NetCIDRContains checks if a cidr or ip is contained within another cidr and returns true or false +var NetCIDRContains = &Builtin{ + Name: "net.cidr_contains", + Decl: types.NewFunction( + types.Args( + types.S, + types.S, + ), + types.B, + ), +} + +// NetCIDRContainsMatches checks if collections of cidrs or ips are contained within another collection of cidrs and returns matches. +var NetCIDRContainsMatches = &Builtin{ + Name: "net.cidr_contains_matches", + Decl: types.NewFunction( + types.Args(netCidrContainsMatchesOperandType, netCidrContainsMatchesOperandType), + types.NewSet(types.NewArray([]types.Type{types.A, types.A}, nil)), + ), +} + +// NetCIDRMerge merges IP addresses and subnets into the smallest possible list of CIDRs. +var NetCIDRMerge = &Builtin{ + Name: "net.cidr_merge", + Decl: types.NewFunction( + types.Args(netCidrMergeOperandType), + types.NewSet(types.S), + ), +} + +var netCidrMergeOperandType = types.NewAny( + types.NewArray(nil, types.NewAny(types.S)), + types.NewSet(types.S), +) + +var netCidrContainsMatchesOperandType = types.NewAny( + types.S, + types.NewArray(nil, types.NewAny( + types.S, + types.NewArray(nil, types.A), + )), + types.NewSet(types.NewAny( + types.S, + types.NewArray(nil, types.A), + )), + types.NewObject(nil, types.NewDynamicProperty( + types.S, + types.NewAny( + types.S, + types.NewArray(nil, types.A), + ), + )), +) + +// NetLookupIPAddr returns the set of IP addresses (as strings, both v4 and v6) +// that the passed-in name (string) resolves to using the standard name resolution +// mechanisms available. +var NetLookupIPAddr = &Builtin{ + Name: "net.lookup_ip_addr", + Decl: types.NewFunction( + types.Args(types.S), + types.NewSet(types.S), + ), +} + +/** + * Semantic Versions + */ + +// SemVerIsValid validiates a the term is a valid SemVer as a string, returns +// false for all other input +var SemVerIsValid = &Builtin{ + Name: "semver.is_valid", + Decl: types.NewFunction( + types.Args( + types.A, + ), + types.B, + ), +} + +// SemVerCompare compares valid SemVer formatted version strings. Given two +// version strings, if A < B returns -1, if A > B returns 1. If A == B, returns +// 0 +var SemVerCompare = &Builtin{ + Name: "semver.compare", + Decl: types.NewFunction( + types.Args( + types.S, + types.S, + ), + types.N, + ), +} + +/** + * Printing + */ + +// Print is a special built-in function that writes zero or more operands +// to a message buffer. The caller controls how the buffer is displayed. The +// operands may be of any type. Furthermore, unlike other built-in functions, +// undefined operands DO NOT cause the print() function to fail during +// evaluation. +var Print = &Builtin{ + Name: "print", + Decl: types.NewVariadicFunction(nil, types.A, nil), +} + +// InternalPrint represents the internal implementation of the print() function. +// The compiler rewrites print() calls to refer to the internal implementation. +var InternalPrint = &Builtin{ + Name: "internal.print", + Decl: types.NewFunction([]types.Type{types.NewArray(nil, types.NewSet(types.A))}, nil), +} + +/** + * Deprecated built-ins. + */ + +// SetDiff has been replaced by the minus built-in. +var SetDiff = &Builtin{ + Name: "set_diff", + Decl: types.NewFunction( + types.Args( + types.NewSet(types.A), + types.NewSet(types.A), + ), + types.NewSet(types.A), + ), +} + +// NetCIDROverlap has been replaced by the `net.cidr_contains` built-in. +var NetCIDROverlap = &Builtin{ + Name: "net.cidr_overlap", + Decl: types.NewFunction( + types.Args( + types.S, + types.S, + ), + types.B, + ), +} + +// CastArray checks the underlying type of the input. If it is array or set, an array +// containing the values is returned. If it is not an array, an error is thrown. +var CastArray = &Builtin{ + Name: "cast_array", + Decl: types.NewFunction( + types.Args(types.A), + types.NewArray(nil, types.A), + ), +} + +// CastSet checks the underlying type of the input. +// If it is a set, the set is returned. +// If it is an array, the array is returned in set form (all duplicates removed) +// If neither, an error is thrown +var CastSet = &Builtin{ + Name: "cast_set", + Decl: types.NewFunction( + types.Args(types.A), + types.NewSet(types.A), + ), +} + +// CastString returns input if it is a string; if not returns error. +// For formatting variables, see sprintf +var CastString = &Builtin{ + Name: "cast_string", + Decl: types.NewFunction( + types.Args(types.A), + types.S, + ), +} + +// CastBoolean returns input if it is a boolean; if not returns error. +var CastBoolean = &Builtin{ + Name: "cast_boolean", + Decl: types.NewFunction( + types.Args(types.A), + types.B, + ), +} + +// CastNull returns null if input is null; if not returns error. +var CastNull = &Builtin{ + Name: "cast_null", + Decl: types.NewFunction( + types.Args(types.A), + types.NewNull(), + ), +} + +// CastObject returns the given object if it is null; throws an error otherwise +var CastObject = &Builtin{ + Name: "cast_object", + Decl: types.NewFunction( + types.Args(types.A), + types.NewObject(nil, types.NewDynamicProperty(types.A, types.A)), + ), +} + +// RegexMatchDeprecated declares `re_match` which has been deprecated. Use `regex.match` instead. +var RegexMatchDeprecated = &Builtin{ + Name: "re_match", + Decl: types.NewFunction( + types.Args( + types.S, + types.S, + ), + types.B, + ), +} + +// Builtin represents a built-in function supported by OPA. Every built-in +// function is uniquely identified by a name. +type Builtin struct { + Name string `json:"name"` // Unique name of built-in function, e.g., (arg1,arg2,...,argN) + Decl *types.Function `json:"decl"` // Built-in function type declaration. + Infix string `json:"infix,omitempty"` // Unique name of infix operator. Default should be unset. + Relation bool `json:"relation,omitempty"` // Indicates if the built-in acts as a relation. +} + +// Expr creates a new expression for the built-in with the given operands. +func (b *Builtin) Expr(operands ...*Term) *Expr { + ts := make([]*Term, len(operands)+1) + ts[0] = NewTerm(b.Ref()) + for i := range operands { + ts[i+1] = operands[i] + } + return &Expr{ + Terms: ts, + } +} + +// Call creates a new term for the built-in with the given operands. +func (b *Builtin) Call(operands ...*Term) *Term { + call := make(Call, len(operands)+1) + call[0] = NewTerm(b.Ref()) + for i := range operands { + call[i+1] = operands[i] + } + return NewTerm(call) +} + +// Ref returns a Ref that refers to the built-in function. +func (b *Builtin) Ref() Ref { + parts := strings.Split(b.Name, ".") + ref := make(Ref, len(parts)) + ref[0] = VarTerm(parts[0]) + for i := 1; i < len(parts); i++ { + ref[i] = StringTerm(parts[i]) + } + return ref +} + +// IsTargetPos returns true if a variable in the i-th position will be bound by +// evaluating the call expression. +func (b *Builtin) IsTargetPos(i int) bool { + return len(b.Decl.Args()) == i +} + +func init() { + BuiltinMap = map[string]*Builtin{} + for _, b := range DefaultBuiltins { + RegisterBuiltin(b) + } +} diff --git a/vendor/github.com/open-policy-agent/opa/ast/capabilities.go b/vendor/github.com/open-policy-agent/opa/ast/capabilities.go new file mode 100644 index 00000000..a6b162ed --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/ast/capabilities.go @@ -0,0 +1,65 @@ +// Copyright 2020 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package ast + +import ( + "io" + "sort" + + "github.com/open-policy-agent/opa/internal/wasm/sdk/opa/capabilities" + "github.com/open-policy-agent/opa/util" +) + +// Capabilities defines a structure containing data that describes the capablilities +// or features supported by a particular version of OPA. +type Capabilities struct { + Builtins []*Builtin `json:"builtins"` + FutureKeywords []string `json:"future_keywords"` + WasmABIVersions []WasmABIVersion `json:"wasm_abi_versions"` + + // allow_net is an array of hostnames or IP addresses, that an OPA instance is + // allowed to connect to. + // If omitted, ANY host can be connected to. If empty, NO host can be connected to. + // As of now, this only controls fetching remote refs for using JSON Schemas in + // the type checker. + // TODO(sr): support ports to further restrict connection peers + // TODO(sr): support restricting `http.send` using the same mechanism (see https://github.com/open-policy-agent/opa/issues/3665) + AllowNet []string `json:"allow_net,omitempty"` +} + +// WasmABIVersion captures the Wasm ABI version. Its `Minor` version is indicating +// backwards-compatible changes. +type WasmABIVersion struct { + Version int `json:"version"` + Minor int `json:"minor_version"` +} + +// CapabilitiesForThisVersion returns the capabilities of this version of OPA. +func CapabilitiesForThisVersion() *Capabilities { + f := &Capabilities{} + + for _, vers := range capabilities.ABIVersions() { + f.WasmABIVersions = append(f.WasmABIVersions, WasmABIVersion{Version: vers[0], Minor: vers[1]}) + } + + f.Builtins = append(f.Builtins, Builtins...) + sort.Slice(f.Builtins, func(i, j int) bool { + return f.Builtins[i].Name < f.Builtins[j].Name + }) + + for kw := range futureKeywords { + f.FutureKeywords = append(f.FutureKeywords, kw) + } + sort.Strings(f.FutureKeywords) + + return f +} + +// LoadCapabilitiesJSON loads a JSON serialized capabilities structure from the reader r. +func LoadCapabilitiesJSON(r io.Reader) (*Capabilities, error) { + d := util.NewJSONDecoder(r) + var c Capabilities + return &c, d.Decode(&c) +} diff --git a/vendor/github.com/open-policy-agent/opa/ast/check.go b/vendor/github.com/open-policy-agent/opa/ast/check.go new file mode 100644 index 00000000..1e0971bc --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/ast/check.go @@ -0,0 +1,1372 @@ +// Copyright 2017 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package ast + +import ( + "fmt" + "sort" + "strings" + + "github.com/open-policy-agent/opa/types" + "github.com/open-policy-agent/opa/util" +) + +type varRewriter func(Ref) Ref + +// exprChecker defines the interface for executing type checking on a single +// expression. The exprChecker must update the provided TypeEnv with inferred +// types of vars. +type exprChecker func(*TypeEnv, *Expr) *Error + +// typeChecker implements type checking on queries and rules. Errors are +// accumulated on the typeChecker so that a single run can report multiple +// issues. +type typeChecker struct { + errs Errors + exprCheckers map[string]exprChecker + varRewriter varRewriter + ss *SchemaSet + allowNet []string + input types.Type +} + +// newTypeChecker returns a new typeChecker object that has no errors. +func newTypeChecker() *typeChecker { + tc := &typeChecker{} + tc.exprCheckers = map[string]exprChecker{ + "eq": tc.checkExprEq, + } + return tc +} + +func (tc *typeChecker) newEnv(exist *TypeEnv) *TypeEnv { + if exist != nil { + return exist.wrap() + } + env := newTypeEnv(tc.copy) + if tc.input != nil { + env.tree.Put(InputRootRef, tc.input) + } + return env +} + +func (tc *typeChecker) copy() *typeChecker { + return newTypeChecker(). + WithVarRewriter(tc.varRewriter). + WithSchemaSet(tc.ss). + WithAllowNet(tc.allowNet). + WithInputType(tc.input) +} + +func (tc *typeChecker) WithSchemaSet(ss *SchemaSet) *typeChecker { + tc.ss = ss + return tc +} + +func (tc *typeChecker) WithAllowNet(hosts []string) *typeChecker { + tc.allowNet = hosts + return tc +} + +func (tc *typeChecker) WithVarRewriter(f varRewriter) *typeChecker { + tc.varRewriter = f + return tc +} + +func (tc *typeChecker) WithInputType(tpe types.Type) *typeChecker { + tc.input = tpe + return tc +} + +// Env returns a type environment for the specified built-ins with any other +// global types configured on the checker. In practice, this is the default +// environment that other statements will be checked against. +func (tc *typeChecker) Env(builtins map[string]*Builtin) *TypeEnv { + env := tc.newEnv(nil) + for _, bi := range builtins { + env.tree.Put(bi.Ref(), bi.Decl) + } + return env +} + +// CheckBody runs type checking on the body and returns a TypeEnv if no errors +// are found. The resulting TypeEnv wraps the provided one. The resulting +// TypeEnv will be able to resolve types of vars contained in the body. +func (tc *typeChecker) CheckBody(env *TypeEnv, body Body) (*TypeEnv, Errors) { + + errors := []*Error{} + env = tc.newEnv(env) + + WalkExprs(body, func(expr *Expr) bool { + + closureErrs := tc.checkClosures(env, expr) + for _, err := range closureErrs { + errors = append(errors, err) + } + + hasClosureErrors := len(closureErrs) > 0 + + vis := newRefChecker(env, tc.varRewriter) + NewGenericVisitor(vis.Visit).Walk(expr) + for _, err := range vis.errs { + errors = append(errors, err) + } + + hasRefErrors := len(vis.errs) > 0 + + if err := tc.checkExpr(env, expr); err != nil { + // Suppress this error if a more actionable one has occurred. In + // this case, if an error occurred in a ref or closure contained in + // this expression, and the error is due to a nil type, then it's + // likely to be the result of the more specific error. + skip := (hasClosureErrors || hasRefErrors) && causedByNilType(err) + if !skip { + errors = append(errors, err) + } + } + return true + }) + + tc.err(errors) + return env, errors +} + +// CheckTypes runs type checking on the rules returns a TypeEnv if no errors +// are found. The resulting TypeEnv wraps the provided one. The resulting +// TypeEnv will be able to resolve types of refs that refer to rules. +func (tc *typeChecker) CheckTypes(env *TypeEnv, sorted []util.T) (*TypeEnv, Errors) { + env = tc.newEnv(env) + var as *annotationSet + if tc.ss != nil { + var errs Errors + as, errs = buildAnnotationSet(sorted) + if len(errs) > 0 { + return env, errs + } + } + for _, s := range sorted { + tc.checkRule(env, as, s.(*Rule)) + } + tc.errs.Sort() + return env, tc.errs +} + +func (tc *typeChecker) checkClosures(env *TypeEnv, expr *Expr) Errors { + var result Errors + WalkClosures(expr, func(x interface{}) bool { + switch x := x.(type) { + case *ArrayComprehension: + _, errs := tc.copy().CheckBody(env, x.Body) + if len(errs) > 0 { + result = errs + return true + } + case *SetComprehension: + _, errs := tc.copy().CheckBody(env, x.Body) + if len(errs) > 0 { + result = errs + return true + } + case *ObjectComprehension: + _, errs := tc.copy().CheckBody(env, x.Body) + if len(errs) > 0 { + result = errs + return true + } + } + return false + }) + return result +} + +func (tc *typeChecker) checkRule(env *TypeEnv, as *annotationSet, rule *Rule) { + + env = env.wrap() + + if schemaAnnots := getRuleAnnotation(as, rule); schemaAnnots != nil { + for _, schemaAnnot := range schemaAnnots { + ref, refType, err := processAnnotation(tc.ss, schemaAnnot, rule, tc.allowNet) + if err != nil { + tc.err([]*Error{err}) + continue + } + prefixRef, t := getPrefix(env, ref) + if t == nil || len(prefixRef) == len(ref) { + env.tree.Put(ref, refType) + } else { + newType, err := override(ref[len(prefixRef):], t, refType, rule) + if err != nil { + tc.err([]*Error{err}) + continue + } + env.tree.Put(prefixRef, newType) + } + } + } + + cpy, err := tc.CheckBody(env, rule.Body) + env = env.next + path := rule.Path() + + if len(err) > 0 { + // if the rule/function contains an error, add it to the type env so + // that expressions that refer to this rule/function do not encounter + // type errors. + env.tree.Put(path, types.A) + return + } + + var tpe types.Type + + if len(rule.Head.Args) > 0 { + + // If args are not referred to in body, infer as any. + WalkVars(rule.Head.Args, func(v Var) bool { + if cpy.Get(v) == nil { + cpy.tree.PutOne(v, types.A) + } + return false + }) + + // Construct function type. + args := make([]types.Type, len(rule.Head.Args)) + for i := 0; i < len(rule.Head.Args); i++ { + args[i] = cpy.Get(rule.Head.Args[i]) + } + + f := types.NewFunction(args, cpy.Get(rule.Head.Value)) + + // Union with existing. + exist := env.tree.Get(path) + tpe = types.Or(exist, f) + + } else { + switch rule.Head.DocKind() { + case CompleteDoc: + typeV := cpy.Get(rule.Head.Value) + if typeV != nil { + exist := env.tree.Get(path) + tpe = types.Or(typeV, exist) + } + case PartialObjectDoc: + typeK := cpy.Get(rule.Head.Key) + typeV := cpy.Get(rule.Head.Value) + if typeK != nil && typeV != nil { + exist := env.tree.Get(path) + typeV = types.Or(types.Values(exist), typeV) + typeK = types.Or(types.Keys(exist), typeK) + tpe = types.NewObject(nil, types.NewDynamicProperty(typeK, typeV)) + } + case PartialSetDoc: + typeK := cpy.Get(rule.Head.Key) + if typeK != nil { + exist := env.tree.Get(path) + typeK = types.Or(types.Keys(exist), typeK) + tpe = types.NewSet(typeK) + } + } + } + + if tpe != nil { + env.tree.Put(path, tpe) + } +} + +func (tc *typeChecker) checkExpr(env *TypeEnv, expr *Expr) *Error { + if !expr.IsCall() { + return nil + } + + checker := tc.exprCheckers[expr.Operator().String()] + if checker != nil { + return checker(env, expr) + } + + return tc.checkExprBuiltin(env, expr) +} + +func (tc *typeChecker) checkExprBuiltin(env *TypeEnv, expr *Expr) *Error { + + args := expr.Operands() + pre := getArgTypes(env, args) + + // NOTE(tsandall): undefined functions will have been caught earlier in the + // compiler. We check for undefined functions before the safety check so + // that references to non-existent functions result in undefined function + // errors as opposed to unsafe var errors. + // + // We cannot run type checking before the safety check because part of the + // type checker relies on reordering (in particular for references to local + // vars). + name := expr.Operator() + tpe := env.Get(name) + + if tpe == nil { + return NewError(TypeErr, expr.Location, "undefined function %v", name) + } + + // check if the expression refers to a function that contains an error + _, ok := tpe.(types.Any) + if ok { + return nil + } + + ftpe, ok := tpe.(*types.Function) + if !ok { + return NewError(TypeErr, expr.Location, "undefined function %v", name) + } + + fargs := ftpe.FuncArgs() + + if ftpe.Result() != nil { + fargs.Args = append(fargs.Args, ftpe.Result()) + } + + if len(args) > len(fargs.Args) && fargs.Variadic == nil { + return newArgError(expr.Location, name, "too many arguments", pre, fargs) + } + + if len(args) < len(ftpe.FuncArgs().Args) { + return newArgError(expr.Location, name, "too few arguments", pre, fargs) + } + + for i := range args { + if !unify1(env, args[i], fargs.Arg(i), false) { + post := make([]types.Type, len(args)) + for i := range args { + post[i] = env.Get(args[i]) + } + return newArgError(expr.Location, name, "invalid argument(s)", post, fargs) + } + } + + return nil +} + +func (tc *typeChecker) checkExprEq(env *TypeEnv, expr *Expr) *Error { + + pre := getArgTypes(env, expr.Operands()) + exp := Equality.Decl.FuncArgs() + + if len(pre) < len(exp.Args) { + return newArgError(expr.Location, expr.Operator(), "too few arguments", pre, exp) + } + + if len(exp.Args) < len(pre) { + return newArgError(expr.Location, expr.Operator(), "too many arguments", pre, exp) + } + + a, b := expr.Operand(0), expr.Operand(1) + typeA, typeB := env.Get(a), env.Get(b) + + if !unify2(env, a, typeA, b, typeB) { + err := NewError(TypeErr, expr.Location, "match error") + err.Details = &UnificationErrDetail{ + Left: typeA, + Right: typeB, + } + return err + } + + return nil +} + +func unify2(env *TypeEnv, a *Term, typeA types.Type, b *Term, typeB types.Type) bool { + + nilA := types.Nil(typeA) + nilB := types.Nil(typeB) + + if nilA && !nilB { + return unify1(env, a, typeB, false) + } else if nilB && !nilA { + return unify1(env, b, typeA, false) + } else if !nilA && !nilB { + return unifies(typeA, typeB) + } + + switch a.Value.(type) { + case *Array: + return unify2Array(env, a, b) + case *object: + return unify2Object(env, a, b) + case Var: + switch b.Value.(type) { + case Var: + return unify1(env, a, types.A, false) && unify1(env, b, env.Get(a), false) + case *Array: + return unify2Array(env, b, a) + case *object: + return unify2Object(env, b, a) + } + } + + return false +} + +func unify2Array(env *TypeEnv, a *Term, b *Term) bool { + arr := a.Value.(*Array) + switch bv := b.Value.(type) { + case *Array: + if arr.Len() == bv.Len() { + for i := 0; i < arr.Len(); i++ { + if !unify2(env, arr.Elem(i), env.Get(arr.Elem(i)), bv.Elem(i), env.Get(bv.Elem(i))) { + return false + } + } + return true + } + case Var: + return unify1(env, a, types.A, false) && unify1(env, b, env.Get(a), false) + } + return false +} + +func unify2Object(env *TypeEnv, a *Term, b *Term) bool { + obj := a.Value.(Object) + switch bv := b.Value.(type) { + case *object: + cv := obj.Intersect(bv) + if obj.Len() == bv.Len() && bv.Len() == len(cv) { + for i := range cv { + if !unify2(env, cv[i][1], env.Get(cv[i][1]), cv[i][2], env.Get(cv[i][2])) { + return false + } + } + return true + } + case Var: + return unify1(env, a, types.A, false) && unify1(env, b, env.Get(a), false) + } + return false +} + +func unify1(env *TypeEnv, term *Term, tpe types.Type, union bool) bool { + switch v := term.Value.(type) { + case *Array: + switch tpe := tpe.(type) { + case *types.Array: + return unify1Array(env, v, tpe, union) + case types.Any: + if types.Compare(tpe, types.A) == 0 { + for i := 0; i < v.Len(); i++ { + unify1(env, v.Elem(i), types.A, true) + } + return true + } + unifies := false + for i := range tpe { + unifies = unify1(env, term, tpe[i], true) || unifies + } + return unifies + } + return false + case *object: + switch tpe := tpe.(type) { + case *types.Object: + return unify1Object(env, v, tpe, union) + case types.Any: + if types.Compare(tpe, types.A) == 0 { + v.Foreach(func(key, value *Term) { + unify1(env, key, types.A, true) + unify1(env, value, types.A, true) + }) + return true + } + unifies := false + for i := range tpe { + unifies = unify1(env, term, tpe[i], true) || unifies + } + return unifies + } + return false + case Set: + switch tpe := tpe.(type) { + case *types.Set: + return unify1Set(env, v, tpe, union) + case types.Any: + if types.Compare(tpe, types.A) == 0 { + v.Foreach(func(elem *Term) { + unify1(env, elem, types.A, true) + }) + return true + } + unifies := false + for i := range tpe { + unifies = unify1(env, term, tpe[i], true) || unifies + } + return unifies + } + return false + case Ref, *ArrayComprehension, *ObjectComprehension, *SetComprehension: + return unifies(env.Get(v), tpe) + case Var: + if !union { + if exist := env.Get(v); exist != nil { + return unifies(exist, tpe) + } + env.tree.PutOne(term.Value, tpe) + } else { + env.tree.PutOne(term.Value, types.Or(env.Get(v), tpe)) + } + return true + default: + if !IsConstant(v) { + panic("unreachable") + } + return unifies(env.Get(term), tpe) + } +} + +func unify1Array(env *TypeEnv, val *Array, tpe *types.Array, union bool) bool { + if val.Len() != tpe.Len() && tpe.Dynamic() == nil { + return false + } + for i := 0; i < val.Len(); i++ { + if !unify1(env, val.Elem(i), tpe.Select(i), union) { + return false + } + } + return true +} + +func unify1Object(env *TypeEnv, val Object, tpe *types.Object, union bool) bool { + if val.Len() != len(tpe.Keys()) && tpe.DynamicValue() == nil { + return false + } + stop := val.Until(func(k, v *Term) bool { + if IsConstant(k.Value) { + if child := selectConstant(tpe, k); child != nil { + if !unify1(env, v, child, union) { + return true + } + } else { + return true + } + } else { + // Inferring type of value under dynamic key would involve unioning + // with all property values of tpe whose keys unify. For now, type + // these values as Any. We can investigate stricter inference in + // the future. + unify1(env, v, types.A, union) + } + return false + }) + return !stop +} + +func unify1Set(env *TypeEnv, val Set, tpe *types.Set, union bool) bool { + of := types.Values(tpe) + return !val.Until(func(elem *Term) bool { + return !unify1(env, elem, of, union) + }) +} + +func (tc *typeChecker) err(errors []*Error) { + tc.errs = append(tc.errs, errors...) +} + +type refChecker struct { + env *TypeEnv + errs Errors + varRewriter varRewriter +} + +func rewriteVarsNop(node Ref) Ref { + return node +} + +func newRefChecker(env *TypeEnv, f varRewriter) *refChecker { + + if f == nil { + f = rewriteVarsNop + } + + return &refChecker{ + env: env, + errs: nil, + varRewriter: f, + } +} + +func (rc *refChecker) Visit(x interface{}) bool { + switch x := x.(type) { + case *ArrayComprehension, *ObjectComprehension, *SetComprehension: + return true + case *Expr: + switch terms := x.Terms.(type) { + case []*Term: + for i := 1; i < len(terms); i++ { + NewGenericVisitor(rc.Visit).Walk(terms[i]) + } + return true + case *Term: + NewGenericVisitor(rc.Visit).Walk(terms) + return true + } + case Ref: + if err := rc.checkApply(rc.env, x); err != nil { + rc.errs = append(rc.errs, err) + return true + } + if err := rc.checkRef(rc.env, rc.env.tree, x, 0); err != nil { + rc.errs = append(rc.errs, err) + } + } + return false +} + +func (rc *refChecker) checkApply(curr *TypeEnv, ref Ref) *Error { + if tpe := curr.Get(ref); tpe != nil { + if _, ok := tpe.(*types.Function); ok { + return newRefErrUnsupported(ref[0].Location, rc.varRewriter(ref), len(ref)-1, tpe) + } + } + return nil +} + +func (rc *refChecker) checkRef(curr *TypeEnv, node *typeTreeNode, ref Ref, idx int) *Error { + + if idx == len(ref) { + return nil + } + + head := ref[idx] + + // Handle constant ref operands, i.e., strings or the ref head. + if _, ok := head.Value.(String); ok || idx == 0 { + + child := node.Child(head.Value) + if child == nil { + + if curr.next != nil { + next := curr.next + return rc.checkRef(next, next.tree, ref, 0) + } + + if RootDocumentNames.Contains(ref[0]) { + return rc.checkRefLeaf(types.A, ref, 1) + } + + return rc.checkRefLeaf(types.A, ref, 0) + } + + if child.Leaf() { + return rc.checkRefLeaf(child.Value(), ref, idx+1) + } + + return rc.checkRef(curr, child, ref, idx+1) + } + + // Handle dynamic ref operands. + switch value := head.Value.(type) { + + case Var: + + if exist := rc.env.Get(value); exist != nil { + if !unifies(types.S, exist) { + return newRefErrInvalid(ref[0].Location, rc.varRewriter(ref), idx, exist, types.S, getOneOfForNode(node)) + } + } else { + rc.env.tree.PutOne(value, types.S) + } + + case Ref: + + exist := rc.env.Get(value) + if exist == nil { + // If ref type is unknown, an error will already be reported so + // stop here. + return nil + } + + if !unifies(types.S, exist) { + return newRefErrInvalid(ref[0].Location, rc.varRewriter(ref), idx, exist, types.S, getOneOfForNode(node)) + } + + // Catch other ref operand types here. Non-leaf nodes must be referred to + // with string values. + default: + return newRefErrInvalid(ref[0].Location, rc.varRewriter(ref), idx, nil, types.S, getOneOfForNode(node)) + } + + // Run checking on remaining portion of the ref. Note, since the ref + // potentially refers to data for which no type information exists, + // checking should never fail. + node.Children().Iter(func(_, child util.T) bool { + _ = rc.checkRef(curr, child.(*typeTreeNode), ref, idx+1) // ignore error + return false + }) + + return nil +} + +func (rc *refChecker) checkRefLeaf(tpe types.Type, ref Ref, idx int) *Error { + + if idx == len(ref) { + return nil + } + + head := ref[idx] + + keys := types.Keys(tpe) + if keys == nil { + return newRefErrUnsupported(ref[0].Location, rc.varRewriter(ref), idx-1, tpe) + } + + switch value := head.Value.(type) { + + case Var: + if exist := rc.env.Get(value); exist != nil { + if !unifies(exist, keys) { + return newRefErrInvalid(ref[0].Location, rc.varRewriter(ref), idx, exist, keys, getOneOfForType(tpe)) + } + } else { + rc.env.tree.PutOne(value, types.Keys(tpe)) + } + + case Ref: + if exist := rc.env.Get(value); exist != nil { + if !unifies(exist, keys) { + return newRefErrInvalid(ref[0].Location, rc.varRewriter(ref), idx, exist, keys, getOneOfForType(tpe)) + } + } + + case *Array, Object, Set: + if !unify1(rc.env, head, keys, false) { + return newRefErrInvalid(ref[0].Location, rc.varRewriter(ref), idx, rc.env.Get(head), keys, nil) + } + + default: + child := selectConstant(tpe, head) + if child == nil { + return newRefErrInvalid(ref[0].Location, rc.varRewriter(ref), idx, nil, types.Keys(tpe), getOneOfForType(tpe)) + } + return rc.checkRefLeaf(child, ref, idx+1) + } + + return rc.checkRefLeaf(types.Values(tpe), ref, idx+1) +} + +func unifies(a, b types.Type) bool { + + if a == nil || b == nil { + return false + } + + anyA, ok1 := a.(types.Any) + if ok1 { + if unifiesAny(anyA, b) { + return true + } + } + + anyB, ok2 := b.(types.Any) + if ok2 { + if unifiesAny(anyB, a) { + return true + } + } + + if ok1 || ok2 { + return false + } + + switch a := a.(type) { + case types.Null: + _, ok := b.(types.Null) + return ok + case types.Boolean: + _, ok := b.(types.Boolean) + return ok + case types.Number: + _, ok := b.(types.Number) + return ok + case types.String: + _, ok := b.(types.String) + return ok + case *types.Array: + b, ok := b.(*types.Array) + if !ok { + return false + } + return unifiesArrays(a, b) + case *types.Object: + b, ok := b.(*types.Object) + if !ok { + return false + } + return unifiesObjects(a, b) + case *types.Set: + b, ok := b.(*types.Set) + if !ok { + return false + } + return unifies(types.Values(a), types.Values(b)) + case *types.Function: + // TODO(tsandall): revisit once functions become first-class values. + return false + default: + panic("unreachable") + } +} + +func unifiesAny(a types.Any, b types.Type) bool { + if _, ok := b.(*types.Function); ok { + return false + } + for i := range a { + if unifies(a[i], b) { + return true + } + } + return len(a) == 0 +} + +func unifiesArrays(a, b *types.Array) bool { + + if !unifiesArraysStatic(a, b) { + return false + } + + if !unifiesArraysStatic(b, a) { + return false + } + + return a.Dynamic() == nil || b.Dynamic() == nil || unifies(a.Dynamic(), b.Dynamic()) +} + +func unifiesArraysStatic(a, b *types.Array) bool { + if a.Len() != 0 { + for i := 0; i < a.Len(); i++ { + if !unifies(a.Select(i), b.Select(i)) { + return false + } + } + } + return true +} + +func unifiesObjects(a, b *types.Object) bool { + if !unifiesObjectsStatic(a, b) { + return false + } + + if !unifiesObjectsStatic(b, a) { + return false + } + + return a.DynamicValue() == nil || b.DynamicValue() == nil || unifies(a.DynamicValue(), b.DynamicValue()) +} + +func unifiesObjectsStatic(a, b *types.Object) bool { + for _, k := range a.Keys() { + if !unifies(a.Select(k), b.Select(k)) { + return false + } + } + return true +} + +// typeErrorCause defines an interface to determine the reason for a type +// error. The type error details implement this interface so that type checking +// can report more actionable errors. +type typeErrorCause interface { + nilType() bool +} + +func causedByNilType(err *Error) bool { + cause, ok := err.Details.(typeErrorCause) + if !ok { + return false + } + return cause.nilType() +} + +// ArgErrDetail represents a generic argument error. +type ArgErrDetail struct { + Have []types.Type `json:"have"` + Want types.FuncArgs `json:"want"` +} + +// Lines returns the string representation of the detail. +func (d *ArgErrDetail) Lines() []string { + lines := make([]string, 2) + lines[0] = "have: " + formatArgs(d.Have) + lines[1] = "want: " + fmt.Sprint(d.Want) + return lines +} + +func (d *ArgErrDetail) nilType() bool { + for i := range d.Have { + if types.Nil(d.Have[i]) { + return true + } + } + return false +} + +// UnificationErrDetail describes a type mismatch error when two values are +// unified (e.g., x = [1,2,y]). +type UnificationErrDetail struct { + Left types.Type `json:"a"` + Right types.Type `json:"b"` +} + +func (a *UnificationErrDetail) nilType() bool { + return types.Nil(a.Left) || types.Nil(a.Right) +} + +// Lines returns the string representation of the detail. +func (a *UnificationErrDetail) Lines() []string { + lines := make([]string, 2) + lines[0] = fmt.Sprint("left : ", types.Sprint(a.Left)) + lines[1] = fmt.Sprint("right : ", types.Sprint(a.Right)) + return lines +} + +// RefErrUnsupportedDetail describes an undefined reference error where the +// referenced value does not support dereferencing (e.g., scalars). +type RefErrUnsupportedDetail struct { + Ref Ref `json:"ref"` // invalid ref + Pos int `json:"pos"` // invalid element + Have types.Type `json:"have"` // referenced type +} + +// Lines returns the string representation of the detail. +func (r *RefErrUnsupportedDetail) Lines() []string { + lines := []string{ + r.Ref.String(), + strings.Repeat("^", len(r.Ref[:r.Pos+1].String())), + fmt.Sprintf("have: %v", r.Have), + } + return lines +} + +// RefErrInvalidDetail describes an undefined reference error where the referenced +// value does not support the reference operand (e.g., missing object key, +// invalid key type, etc.) +type RefErrInvalidDetail struct { + Ref Ref `json:"ref"` // invalid ref + Pos int `json:"pos"` // invalid element + Have types.Type `json:"have,omitempty"` // type of invalid element (for var/ref elements) + Want types.Type `json:"want"` // allowed type (for non-object values) + OneOf []Value `json:"oneOf"` // allowed values (e.g., for object keys) +} + +// Lines returns the string representation of the detail. +func (r *RefErrInvalidDetail) Lines() []string { + lines := []string{r.Ref.String()} + offset := len(r.Ref[:r.Pos].String()) + 1 + pad := strings.Repeat(" ", offset) + lines = append(lines, fmt.Sprintf("%s^", pad)) + if r.Have != nil { + lines = append(lines, fmt.Sprintf("%shave (type): %v", pad, r.Have)) + } else { + lines = append(lines, fmt.Sprintf("%shave: %v", pad, r.Ref[r.Pos])) + } + if len(r.OneOf) > 0 { + lines = append(lines, fmt.Sprintf("%swant (one of): %v", pad, r.OneOf)) + } else { + lines = append(lines, fmt.Sprintf("%swant (type): %v", pad, r.Want)) + } + return lines +} + +func formatArgs(args []types.Type) string { + buf := make([]string, len(args)) + for i := range args { + buf[i] = types.Sprint(args[i]) + } + return "(" + strings.Join(buf, ", ") + ")" +} + +func newRefErrInvalid(loc *Location, ref Ref, idx int, have, want types.Type, oneOf []Value) *Error { + err := newRefError(loc, ref) + err.Details = &RefErrInvalidDetail{ + Ref: ref, + Pos: idx, + Have: have, + Want: want, + OneOf: oneOf, + } + return err +} + +func newRefErrUnsupported(loc *Location, ref Ref, idx int, have types.Type) *Error { + err := newRefError(loc, ref) + err.Details = &RefErrUnsupportedDetail{ + Ref: ref, + Pos: idx, + Have: have, + } + return err +} + +func newRefError(loc *Location, ref Ref) *Error { + return NewError(TypeErr, loc, "undefined ref: %v", ref) +} + +func newArgError(loc *Location, builtinName Ref, msg string, have []types.Type, want types.FuncArgs) *Error { + err := NewError(TypeErr, loc, "%v: %v", builtinName, msg) + err.Details = &ArgErrDetail{ + Have: have, + Want: want, + } + return err +} + +func getOneOfForNode(node *typeTreeNode) (result []Value) { + node.Children().Iter(func(k, _ util.T) bool { + result = append(result, k.(Value)) + return false + }) + + sortValueSlice(result) + return result +} + +func getOneOfForType(tpe types.Type) (result []Value) { + switch tpe := tpe.(type) { + case *types.Object: + for _, k := range tpe.Keys() { + v, err := InterfaceToValue(k) + if err != nil { + panic(err) + } + result = append(result, v) + } + + case types.Any: + for _, object := range tpe { + objRes := getOneOfForType(object) + result = append(result, objRes...) + } + } + + result = removeDuplicate(result) + sortValueSlice(result) + return result +} + +func sortValueSlice(sl []Value) { + sort.Slice(sl, func(i, j int) bool { + return sl[i].Compare(sl[j]) < 0 + }) +} + +func removeDuplicate(list []Value) []Value { + seen := make(map[Value]bool) + var newResult []Value + for _, item := range list { + if !seen[item] { + newResult = append(newResult, item) + seen[item] = true + } + } + return newResult +} + +func getArgTypes(env *TypeEnv, args []*Term) []types.Type { + pre := make([]types.Type, len(args)) + for i := range args { + pre[i] = env.Get(args[i]) + } + return pre +} + +// getPrefix returns the shortest prefix of ref that exists in env +func getPrefix(env *TypeEnv, ref Ref) (Ref, types.Type) { + if len(ref) == 1 { + t := env.Get(ref) + if t != nil { + return ref, t + } + } + for i := 1; i < len(ref); i++ { + t := env.Get(ref[:i]) + if t != nil { + return ref[:i], t + } + } + return nil, nil +} + +// override takes a type t and returns a type obtained from t where the path represented by ref within it has type o (overriding the original type of that path) +func override(ref Ref, t types.Type, o types.Type, rule *Rule) (types.Type, *Error) { + var newStaticProps []*types.StaticProperty + obj, ok := t.(*types.Object) + if !ok { + newType, err := getObjectType(ref, o, rule, types.NewDynamicProperty(types.A, types.A)) + if err != nil { + return nil, err + } + return newType, nil + } + found := false + if ok { + staticProps := obj.StaticProperties() + for _, prop := range staticProps { + valueCopy := prop.Value + key, err := InterfaceToValue(prop.Key) + if err != nil { + return nil, NewError(TypeErr, rule.Location, "unexpected error in override: %s", err.Error()) + } + if len(ref) > 0 && ref[0].Value.Compare(key) == 0 { + found = true + if len(ref) == 1 { + valueCopy = o + } else { + newVal, err := override(ref[1:], valueCopy, o, rule) + if err != nil { + return nil, err + } + valueCopy = newVal + } + } + newStaticProps = append(newStaticProps, types.NewStaticProperty(prop.Key, valueCopy)) + } + } + + // ref[0] is not a top-level key in staticProps, so it must be added + if !found { + newType, err := getObjectType(ref, o, rule, obj.DynamicProperties()) + if err != nil { + return nil, err + } + newStaticProps = append(newStaticProps, newType.StaticProperties()...) + } + return types.NewObject(newStaticProps, obj.DynamicProperties()), nil +} + +func getKeys(ref Ref, rule *Rule) ([]interface{}, *Error) { + keys := []interface{}{} + for _, refElem := range ref { + key, err := JSON(refElem.Value) + if err != nil { + return nil, NewError(TypeErr, rule.Location, "error getting key from value: %s", err.Error()) + } + keys = append(keys, key) + } + return keys, nil +} + +func getObjectTypeRec(keys []interface{}, o types.Type, d *types.DynamicProperty) *types.Object { + if len(keys) == 1 { + staticProps := []*types.StaticProperty{types.NewStaticProperty(keys[0], o)} + return types.NewObject(staticProps, d) + } + + staticProps := []*types.StaticProperty{types.NewStaticProperty(keys[0], getObjectTypeRec(keys[1:], o, d))} + return types.NewObject(staticProps, d) +} + +func getObjectType(ref Ref, o types.Type, rule *Rule, d *types.DynamicProperty) (*types.Object, *Error) { + keys, err := getKeys(ref, rule) + if err != nil { + return nil, err + } + return getObjectTypeRec(keys, o, d), nil +} + +func getRuleAnnotation(as *annotationSet, rule *Rule) (result []*SchemaAnnotation) { + + for _, x := range as.GetSubpackagesScope(rule.Module.Package.Path) { + result = append(result, x.Schemas...) + } + + if x := as.GetPackageScope(rule.Module.Package); x != nil { + result = append(result, x.Schemas...) + } + + if x := as.GetDocumentScope(rule.Path()); x != nil { + result = append(result, x.Schemas...) + } + + for _, x := range as.GetRuleScope(rule) { + result = append(result, x.Schemas...) + } + + return result +} + +func processAnnotation(ss *SchemaSet, annot *SchemaAnnotation, rule *Rule, allowNet []string) (Ref, types.Type, *Error) { + + var schema interface{} + + if annot.Schema != nil { + schema = ss.Get(annot.Schema) + if schema == nil { + return nil, nil, NewError(TypeErr, rule.Location, "undefined schema: %v", annot.Schema) + } + } else if annot.Definition != nil { + schema = *annot.Definition + } + + tpe, err := loadSchema(schema, allowNet) + if err != nil { + return nil, nil, NewError(TypeErr, rule.Location, err.Error()) + } + + return annot.Path, tpe, nil +} + +func errAnnotationRedeclared(a *Annotations, other *Location) *Error { + return NewError(TypeErr, a.Location, "%v annotation redeclared: %v", a.Scope, other) +} + +type annotationSet struct { + byRule map[*Rule][]*Annotations + byPackage map[*Package]*Annotations + byPath *annotationTreeNode +} + +func buildAnnotationSet(rules []util.T) (*annotationSet, Errors) { + as := newAnnotationSet() + processed := map[*Module]struct{}{} + var errs Errors + for _, x := range rules { + module := x.(*Rule).Module + if _, ok := processed[module]; ok { + continue + } + processed[module] = struct{}{} + for _, a := range module.Annotations { + if err := as.Add(a); err != nil { + errs = append(errs, err) + } + } + } + if len(errs) > 0 { + return nil, errs + } + return as, nil +} + +func newAnnotationSet() *annotationSet { + return &annotationSet{ + byRule: map[*Rule][]*Annotations{}, + byPackage: map[*Package]*Annotations{}, + byPath: newAnnotationTree(), + } +} + +func (as *annotationSet) Add(a *Annotations) *Error { + switch a.Scope { + case annotationScopeRule: + rule := a.node.(*Rule) + as.byRule[rule] = append(as.byRule[rule], a) + case annotationScopePackage: + pkg := a.node.(*Package) + if exist, ok := as.byPackage[pkg]; ok { + return errAnnotationRedeclared(a, exist.Location) + } + as.byPackage[pkg] = a + case annotationScopeDocument: + rule := a.node.(*Rule) + path := rule.Path() + x := as.byPath.Get(path) + if x != nil { + return errAnnotationRedeclared(a, x.Value.Location) + } + as.byPath.Insert(path, a) + case annotationScopeSubpackages: + pkg := a.node.(*Package) + x := as.byPath.Get(pkg.Path) + if x != nil { + return errAnnotationRedeclared(a, x.Value.Location) + } + as.byPath.Insert(pkg.Path, a) + } + return nil +} + +func (as *annotationSet) GetRuleScope(r *Rule) []*Annotations { + if as == nil { + return nil + } + return as.byRule[r] +} + +func (as *annotationSet) GetSubpackagesScope(path Ref) []*Annotations { + if as == nil { + return nil + } + return as.byPath.Ancestors(path) +} + +func (as *annotationSet) GetDocumentScope(path Ref) *Annotations { + if as == nil { + return nil + } + if node := as.byPath.Get(path); node != nil { + return node.Value + } + return nil +} + +func (as *annotationSet) GetPackageScope(pkg *Package) *Annotations { + if as == nil { + return nil + } + return as.byPackage[pkg] +} + +type annotationTreeNode struct { + Value *Annotations + Children map[Value]*annotationTreeNode // we assume key elements are hashable (vars and strings only!) +} + +func newAnnotationTree() *annotationTreeNode { + return &annotationTreeNode{ + Value: nil, + Children: map[Value]*annotationTreeNode{}, + } +} + +func (t *annotationTreeNode) Insert(path Ref, value *Annotations) { + node := t + for _, k := range path { + child, ok := node.Children[k.Value] + if !ok { + child = newAnnotationTree() + node.Children[k.Value] = child + } + node = child + } + node.Value = value +} + +func (t *annotationTreeNode) Get(path Ref) *annotationTreeNode { + node := t + for _, k := range path { + if node == nil { + return nil + } + child, ok := node.Children[k.Value] + if !ok { + return nil + } + node = child + } + return node +} + +func (t *annotationTreeNode) Ancestors(path Ref) (result []*Annotations) { + node := t + for _, k := range path { + if node == nil { + return result + } + child, ok := node.Children[k.Value] + if !ok { + return result + } + if child.Value != nil { + result = append(result, child.Value) + } + node = child + } + return result +} diff --git a/vendor/github.com/open-policy-agent/opa/ast/compare.go b/vendor/github.com/open-policy-agent/opa/ast/compare.go new file mode 100644 index 00000000..685082da --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/ast/compare.go @@ -0,0 +1,386 @@ +// Copyright 2016 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package ast + +import ( + "encoding/json" + "fmt" + "math/big" +) + +// Compare returns an integer indicating whether two AST values are less than, +// equal to, or greater than each other. +// +// If a is less than b, the return value is negative. If a is greater than b, +// the return value is positive. If a is equal to b, the return value is zero. +// +// Different types are never equal to each other. For comparison purposes, types +// are sorted as follows: +// +// nil < Null < Boolean < Number < String < Var < Ref < Array < Object < Set < +// ArrayComprehension < ObjectComprehension < SetComprehension < Expr < SomeDecl +// < With < Body < Rule < Import < Package < Module. +// +// Arrays and Refs are equal if and only if both a and b have the same length +// and all corresponding elements are equal. If one element is not equal, the +// return value is the same as for the first differing element. If all elements +// are equal but a and b have different lengths, the shorter is considered less +// than the other. +// +// Objects are considered equal if and only if both a and b have the same sorted +// (key, value) pairs and are of the same length. Other comparisons are +// consistent but not defined. +// +// Sets are considered equal if and only if the symmetric difference of a and b +// is empty. +// Other comparisons are consistent but not defined. +func Compare(a, b interface{}) int { + + if t, ok := a.(*Term); ok { + if t == nil { + a = nil + } else { + a = t.Value + } + } + + if t, ok := b.(*Term); ok { + if t == nil { + b = nil + } else { + b = t.Value + } + } + + if a == nil { + if b == nil { + return 0 + } + return -1 + } + if b == nil { + return 1 + } + + sortA := sortOrder(a) + sortB := sortOrder(b) + + if sortA < sortB { + return -1 + } else if sortB < sortA { + return 1 + } + + switch a := a.(type) { + case Null: + return 0 + case Boolean: + b := b.(Boolean) + if a.Equal(b) { + return 0 + } + if !a { + return -1 + } + return 1 + case Number: + if ai, err := json.Number(a).Int64(); err == nil { + if bi, err := json.Number(b.(Number)).Int64(); err == nil { + if ai == bi { + return 0 + } + if ai < bi { + return -1 + } + return 1 + } + } + + // We use big.Rat for comparing big numbers. + // It replaces big.Float due to following reason: + // big.Float comes with a default precision of 64, and setting a + // larger precision results in more memory being allocated + // (regardless of the actual number we are parsing with SetString). + // + // Note: If we're so close to zero that big.Float says we are zero, do + // *not* big.Rat).SetString on the original string it'll potentially + // take very long. + var bigA, bigB *big.Rat + fa, ok := new(big.Float).SetString(string(a)) + if !ok { + panic("illegal value") + } + if fa.IsInt() { + if i, _ := fa.Int64(); i == 0 { + bigA = new(big.Rat).SetInt64(0) + } + } + if bigA == nil { + bigA, ok = new(big.Rat).SetString(string(a)) + if !ok { + panic("illegal value") + } + } + + fb, ok := new(big.Float).SetString(string(b.(Number))) + if !ok { + panic("illegal value") + } + if fb.IsInt() { + if i, _ := fb.Int64(); i == 0 { + bigB = new(big.Rat).SetInt64(0) + } + } + if bigB == nil { + bigB, ok = new(big.Rat).SetString(string(b.(Number))) + if !ok { + panic("illegal value") + } + } + + return bigA.Cmp(bigB) + case String: + b := b.(String) + if a.Equal(b) { + return 0 + } + if a < b { + return -1 + } + return 1 + case Var: + b := b.(Var) + if a.Equal(b) { + return 0 + } + if a < b { + return -1 + } + return 1 + case Ref: + b := b.(Ref) + return termSliceCompare(a, b) + case *Array: + b := b.(*Array) + return termSliceCompare(a.elems, b.elems) + case *object: + b := b.(*object) + return a.Compare(b) + case Set: + b := b.(Set) + return a.Compare(b) + case *ArrayComprehension: + b := b.(*ArrayComprehension) + if cmp := Compare(a.Term, b.Term); cmp != 0 { + return cmp + } + return Compare(a.Body, b.Body) + case *ObjectComprehension: + b := b.(*ObjectComprehension) + if cmp := Compare(a.Key, b.Key); cmp != 0 { + return cmp + } + if cmp := Compare(a.Value, b.Value); cmp != 0 { + return cmp + } + return Compare(a.Body, b.Body) + case *SetComprehension: + b := b.(*SetComprehension) + if cmp := Compare(a.Term, b.Term); cmp != 0 { + return cmp + } + return Compare(a.Body, b.Body) + case Call: + b := b.(Call) + return termSliceCompare(a, b) + case *Expr: + b := b.(*Expr) + return a.Compare(b) + case *SomeDecl: + b := b.(*SomeDecl) + return a.Compare(b) + case *With: + b := b.(*With) + return a.Compare(b) + case Body: + b := b.(Body) + return a.Compare(b) + case *Head: + b := b.(*Head) + return a.Compare(b) + case *Rule: + b := b.(*Rule) + return a.Compare(b) + case Args: + b := b.(Args) + return termSliceCompare(a, b) + case *Import: + b := b.(*Import) + return a.Compare(b) + case *Package: + b := b.(*Package) + return a.Compare(b) + case *Annotations: + b := b.(*Annotations) + return a.Compare(b) + case *Module: + b := b.(*Module) + return a.Compare(b) + } + panic(fmt.Sprintf("illegal value: %T", a)) +} + +type termSlice []*Term + +func (s termSlice) Less(i, j int) bool { return Compare(s[i].Value, s[j].Value) < 0 } +func (s termSlice) Swap(i, j int) { x := s[i]; s[i] = s[j]; s[j] = x } +func (s termSlice) Len() int { return len(s) } + +func sortOrder(x interface{}) int { + switch x.(type) { + case Null: + return 0 + case Boolean: + return 1 + case Number: + return 2 + case String: + return 3 + case Var: + return 4 + case Ref: + return 5 + case *Array: + return 6 + case Object: + return 7 + case Set: + return 8 + case *ArrayComprehension: + return 9 + case *ObjectComprehension: + return 10 + case *SetComprehension: + return 11 + case Call: + return 12 + case Args: + return 13 + case *Expr: + return 100 + case *SomeDecl: + return 101 + case *With: + return 110 + case *Head: + return 120 + case Body: + return 200 + case *Rule: + return 1000 + case *Import: + return 1001 + case *Package: + return 1002 + case *Annotations: + return 1003 + case *Module: + return 10000 + } + panic(fmt.Sprintf("illegal value: %T", x)) +} + +func importsCompare(a, b []*Import) int { + minLen := len(a) + if len(b) < minLen { + minLen = len(b) + } + for i := 0; i < minLen; i++ { + if cmp := a[i].Compare(b[i]); cmp != 0 { + return cmp + } + } + if len(a) < len(b) { + return -1 + } + if len(b) < len(a) { + return 1 + } + return 0 +} + +func annotationsCompare(a, b []*Annotations) int { + minLen := len(a) + if len(b) < minLen { + minLen = len(b) + } + for i := 0; i < minLen; i++ { + if cmp := a[i].Compare(b[i]); cmp != 0 { + return cmp + } + } + if len(a) < len(b) { + return -1 + } + if len(b) < len(a) { + return 1 + } + return 0 +} + +func rulesCompare(a, b []*Rule) int { + minLen := len(a) + if len(b) < minLen { + minLen = len(b) + } + for i := 0; i < minLen; i++ { + if cmp := a[i].Compare(b[i]); cmp != 0 { + return cmp + } + } + if len(a) < len(b) { + return -1 + } + if len(b) < len(a) { + return 1 + } + return 0 +} + +func termSliceCompare(a, b []*Term) int { + minLen := len(a) + if len(b) < minLen { + minLen = len(b) + } + for i := 0; i < minLen; i++ { + if cmp := Compare(a[i], b[i]); cmp != 0 { + return cmp + } + } + if len(a) < len(b) { + return -1 + } else if len(b) < len(a) { + return 1 + } + return 0 +} + +func withSliceCompare(a, b []*With) int { + minLen := len(a) + if len(b) < minLen { + minLen = len(b) + } + for i := 0; i < minLen; i++ { + if cmp := Compare(a[i], b[i]); cmp != 0 { + return cmp + } + } + if len(a) < len(b) { + return -1 + } else if len(b) < len(a) { + return 1 + } + return 0 +} diff --git a/vendor/github.com/open-policy-agent/opa/ast/compile.go b/vendor/github.com/open-policy-agent/opa/ast/compile.go new file mode 100644 index 00000000..e0c27e4b --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/ast/compile.go @@ -0,0 +1,4411 @@ +// Copyright 2016 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package ast + +import ( + "fmt" + "io" + "sort" + "strconv" + "strings" + + "github.com/open-policy-agent/opa/internal/debug" + "github.com/open-policy-agent/opa/internal/gojsonschema" + "github.com/open-policy-agent/opa/metrics" + "github.com/open-policy-agent/opa/types" + "github.com/open-policy-agent/opa/util" +) + +// CompileErrorLimitDefault is the default number errors a compiler will allow before +// exiting. +const CompileErrorLimitDefault = 10 + +var errLimitReached = NewError(CompileErr, nil, "error limit reached") + +// Compiler contains the state of a compilation process. +type Compiler struct { + + // Errors contains errors that occurred during the compilation process. + // If there are one or more errors, the compilation process is considered + // "failed". + Errors Errors + + // Modules contains the compiled modules. The compiled modules are the + // output of the compilation process. If the compilation process failed, + // there is no guarantee about the state of the modules. + Modules map[string]*Module + + // ModuleTree organizes the modules into a tree where each node is keyed by + // an element in the module's package path. E.g., given modules containing + // the following package directives: "a", "a.b", "a.c", and "a.b", the + // resulting module tree would be: + // + // root + // | + // +--- data (no modules) + // | + // +--- a (1 module) + // | + // +--- b (2 modules) + // | + // +--- c (1 module) + // + ModuleTree *ModuleTreeNode + + // RuleTree organizes rules into a tree where each node is keyed by an + // element in the rule's path. The rule path is the concatenation of the + // containing package and the stringified rule name. E.g., given the + // following module: + // + // package ex + // p[1] { true } + // p[2] { true } + // q = true + // + // root + // | + // +--- data (no rules) + // | + // +--- ex (no rules) + // | + // +--- p (2 rules) + // | + // +--- q (1 rule) + RuleTree *TreeNode + + // Graph contains dependencies between rules. An edge (u,v) is added to the + // graph if rule 'u' refers to the virtual document defined by 'v'. + Graph *Graph + + // TypeEnv holds type information for values inferred by the compiler. + TypeEnv *TypeEnv + + // RewrittenVars is a mapping of variables that have been rewritten + // with the key being the generated name and value being the original. + RewrittenVars map[Var]Var + + localvargen *localVarGenerator + moduleLoader ModuleLoader + ruleIndices *util.HashMap + stages []struct { + name string + metricName string + f func() + } + maxErrs int + sorted []string // list of sorted module names + pathExists func([]string) (bool, error) + after map[string][]CompilerStageDefinition + metrics metrics.Metrics + capabilities *Capabilities // user-supplied capabilities + builtins map[string]*Builtin // universe of built-in functions + customBuiltins map[string]*Builtin // user-supplied custom built-in functions (deprecated: use capabilities) + unsafeBuiltinsMap map[string]struct{} // user-supplied set of unsafe built-ins functions to block (deprecated: use capabilities) + enablePrintStatements bool // indicates if print statements should be elided (default) + comprehensionIndices map[*Term]*ComprehensionIndex // comprehension key index + initialized bool // indicates if init() has been called + debug debug.Debug // emits debug information produced during compilation + schemaSet *SchemaSet // user-supplied schemas for input and data documents + inputType types.Type // global input type retrieved from schema set +} + +// CompilerStage defines the interface for stages in the compiler. +type CompilerStage func(*Compiler) *Error + +// CompilerStageDefinition defines a compiler stage +type CompilerStageDefinition struct { + Name string + MetricName string + Stage CompilerStage +} + +// RulesOptions defines the options for retrieving rules by Ref from the +// compiler. +type RulesOptions struct { + // IncludeHiddenModules determines if the result contains hidden modules, + // currently only the "system" namespace, i.e. "data.system.*". + IncludeHiddenModules bool +} + +// QueryContext contains contextual information for running an ad-hoc query. +// +// Ad-hoc queries can be run in the context of a package and imports may be +// included to provide concise access to data. +type QueryContext struct { + Package *Package + Imports []*Import +} + +// NewQueryContext returns a new QueryContext object. +func NewQueryContext() *QueryContext { + return &QueryContext{} +} + +// WithPackage sets the pkg on qc. +func (qc *QueryContext) WithPackage(pkg *Package) *QueryContext { + if qc == nil { + qc = NewQueryContext() + } + qc.Package = pkg + return qc +} + +// WithImports sets the imports on qc. +func (qc *QueryContext) WithImports(imports []*Import) *QueryContext { + if qc == nil { + qc = NewQueryContext() + } + qc.Imports = imports + return qc +} + +// Copy returns a deep copy of qc. +func (qc *QueryContext) Copy() *QueryContext { + if qc == nil { + return nil + } + cpy := *qc + if cpy.Package != nil { + cpy.Package = qc.Package.Copy() + } + cpy.Imports = make([]*Import, len(qc.Imports)) + for i := range qc.Imports { + cpy.Imports[i] = qc.Imports[i].Copy() + } + return &cpy +} + +// QueryCompiler defines the interface for compiling ad-hoc queries. +type QueryCompiler interface { + + // Compile should be called to compile ad-hoc queries. The return value is + // the compiled version of the query. + Compile(q Body) (Body, error) + + // TypeEnv returns the type environment built after running type checking + // on the query. + TypeEnv() *TypeEnv + + // WithContext sets the QueryContext on the QueryCompiler. Subsequent calls + // to Compile will take the QueryContext into account. + WithContext(qctx *QueryContext) QueryCompiler + + // WithEnablePrintStatements enables print statements in queries compiled + // with the QueryCompiler. + WithEnablePrintStatements(yes bool) QueryCompiler + + // WithUnsafeBuiltins sets the built-in functions to treat as unsafe and not + // allow inside of queries. By default the query compiler inherits the + // compiler's unsafe built-in functions. This function allows callers to + // override that set. If an empty (non-nil) map is provided, all built-ins + // are allowed. + WithUnsafeBuiltins(unsafe map[string]struct{}) QueryCompiler + + // WithStageAfter registers a stage to run during query compilation after + // the named stage. + WithStageAfter(after string, stage QueryCompilerStageDefinition) QueryCompiler + + // RewrittenVars maps generated vars in the compiled query to vars from the + // parsed query. For example, given the query "input := 1" the rewritten + // query would be "__local0__ = 1". The mapping would then be {__local0__: input}. + RewrittenVars() map[Var]Var + + // ComprehensionIndex returns an index data structure for the given comprehension + // term. If no index is found, returns nil. + ComprehensionIndex(term *Term) *ComprehensionIndex +} + +// QueryCompilerStage defines the interface for stages in the query compiler. +type QueryCompilerStage func(QueryCompiler, Body) (Body, error) + +// QueryCompilerStageDefinition defines a QueryCompiler stage +type QueryCompilerStageDefinition struct { + Name string + MetricName string + Stage QueryCompilerStage +} + +// NewCompiler returns a new empty compiler. +func NewCompiler() *Compiler { + + c := &Compiler{ + Modules: map[string]*Module{}, + RewrittenVars: map[Var]Var{}, + ruleIndices: util.NewHashMap(func(a, b util.T) bool { + r1, r2 := a.(Ref), b.(Ref) + return r1.Equal(r2) + }, func(x util.T) int { + return x.(Ref).Hash() + }), + maxErrs: CompileErrorLimitDefault, + after: map[string][]CompilerStageDefinition{}, + unsafeBuiltinsMap: map[string]struct{}{}, + comprehensionIndices: map[*Term]*ComprehensionIndex{}, + debug: debug.Discard(), + } + + c.ModuleTree = NewModuleTree(nil) + c.RuleTree = NewRuleTree(c.ModuleTree) + + c.stages = []struct { + name string + metricName string + f func() + }{ + // Reference resolution should run first as it may be used to lazily + // load additional modules. If any stages run before resolution, they + // need to be re-run after resolution. + {"ResolveRefs", "compile_stage_resolve_refs", c.resolveAllRefs}, + {"SetModuleTree", "compile_stage_set_module_tree", c.setModuleTree}, + {"SetRuleTree", "compile_stage_set_rule_tree", c.setRuleTree}, + // The local variable generator must be initialized after references are + // resolved and the dynamic module loader has run but before subsequent + // stages that need to generate variables. + {"InitLocalVarGen", "compile_stage_init_local_var_gen", c.initLocalVarGen}, + {"RewriteLocalVars", "compile_stage_rewrite_local_vars", c.rewriteLocalVars}, + {"CheckVoidCalls", "compile_stage_check_void_calls", c.checkVoidCalls}, + {"RewritePrintCalls", "compile_stage_rewrite_print_calls", c.rewritePrintCalls}, + {"RewriteExprTerms", "compile_stage_rewrite_expr_terms", c.rewriteExprTerms}, + {"SetGraph", "compile_stage_set_graph", c.setGraph}, + {"RewriteComprehensionTerms", "compile_stage_rewrite_comprehension_terms", c.rewriteComprehensionTerms}, + {"RewriteRefsInHead", "compile_stage_rewrite_refs_in_head", c.rewriteRefsInHead}, + {"RewriteWithValues", "compile_stage_rewrite_with_values", c.rewriteWithModifiers}, + {"CheckRuleConflicts", "compile_stage_check_rule_conflicts", c.checkRuleConflicts}, + {"CheckUndefinedFuncs", "compile_stage_check_undefined_funcs", c.checkUndefinedFuncs}, + {"CheckSafetyRuleHeads", "compile_stage_check_safety_rule_heads", c.checkSafetyRuleHeads}, + {"CheckSafetyRuleBodies", "compile_stage_check_safety_rule_bodies", c.checkSafetyRuleBodies}, + {"RewriteEquals", "compile_stage_rewrite_equals", c.rewriteEquals}, + {"RewriteDynamicTerms", "compile_stage_rewrite_dynamic_terms", c.rewriteDynamicTerms}, + {"CheckRecursion", "compile_stage_check_recursion", c.checkRecursion}, + {"CheckTypes", "compile_stage_check_types", c.checkTypes}, + {"CheckUnsafeBuiltins", "compile_state_check_unsafe_builtins", c.checkUnsafeBuiltins}, + {"BuildRuleIndices", "compile_stage_rebuild_indices", c.buildRuleIndices}, + {"BuildComprehensionIndices", "compile_stage_rebuild_comprehension_indices", c.buildComprehensionIndices}, + } + + return c +} + +// SetErrorLimit sets the number of errors the compiler can encounter before it +// quits. Zero or a negative number indicates no limit. +func (c *Compiler) SetErrorLimit(limit int) *Compiler { + c.maxErrs = limit + return c +} + +// WithEnablePrintStatements enables print statements inside of modules compiled +// by the compiler. If print statements are not enabled, calls to print() are +// erased at compile-time. +func (c *Compiler) WithEnablePrintStatements(yes bool) *Compiler { + c.enablePrintStatements = yes + return c +} + +// WithPathConflictsCheck enables base-virtual document conflict +// detection. The compiler will check that rules don't overlap with +// paths that exist as determined by the provided callable. +func (c *Compiler) WithPathConflictsCheck(fn func([]string) (bool, error)) *Compiler { + c.pathExists = fn + return c +} + +// WithStageAfter registers a stage to run during compilation after +// the named stage. +func (c *Compiler) WithStageAfter(after string, stage CompilerStageDefinition) *Compiler { + c.after[after] = append(c.after[after], stage) + return c +} + +// WithMetrics will set a metrics.Metrics and be used for profiling +// the Compiler instance. +func (c *Compiler) WithMetrics(metrics metrics.Metrics) *Compiler { + c.metrics = metrics + return c +} + +// WithCapabilities sets capabilities to enable during compilation. Capabilities allow the caller +// to specify the set of built-in functions available to the policy. In the future, capabilities +// may be able to restrict access to other language features. Capabilities allow callers to check +// if policies are compatible with a particular version of OPA. If policies are a compiled for a +// specific version of OPA, there is no guarantee that _this_ version of OPA can evaluate them +// successfully. +func (c *Compiler) WithCapabilities(capabilities *Capabilities) *Compiler { + c.capabilities = capabilities + return c +} + +// Capabilities returns the capabilities enabled during compilation. +func (c *Compiler) Capabilities() *Capabilities { + return c.capabilities +} + +// WithDebug sets where debug messages are written to. Passing `nil` has no +// effect. +func (c *Compiler) WithDebug(sink io.Writer) *Compiler { + if sink != nil { + c.debug = debug.New(sink) + } + return c +} + +// WithBuiltins is deprecated. Use WithCapabilities instead. +func (c *Compiler) WithBuiltins(builtins map[string]*Builtin) *Compiler { + c.customBuiltins = make(map[string]*Builtin) + for k, v := range builtins { + c.customBuiltins[k] = v + } + return c +} + +// WithUnsafeBuiltins is deprecated. Use WithCapabilities instead. +func (c *Compiler) WithUnsafeBuiltins(unsafeBuiltins map[string]struct{}) *Compiler { + for name := range unsafeBuiltins { + c.unsafeBuiltinsMap[name] = struct{}{} + } + return c +} + +// QueryCompiler returns a new QueryCompiler object. +func (c *Compiler) QueryCompiler() QueryCompiler { + c.init() + return newQueryCompiler(c) +} + +// Compile runs the compilation process on the input modules. The compiled +// version of the modules and associated data structures are stored on the +// compiler. If the compilation process fails for any reason, the compiler will +// contain a slice of errors. +func (c *Compiler) Compile(modules map[string]*Module) { + + c.init() + + c.Modules = make(map[string]*Module, len(modules)) + + for k, v := range modules { + c.Modules[k] = v.Copy() + c.sorted = append(c.sorted, k) + } + + sort.Strings(c.sorted) + + c.compile() +} + +// WithSchemas sets a schemaSet to the compiler +func (c *Compiler) WithSchemas(schemas *SchemaSet) *Compiler { + c.schemaSet = schemas + return c +} + +// Failed returns true if a compilation error has been encountered. +func (c *Compiler) Failed() bool { + return len(c.Errors) > 0 +} + +// ComprehensionIndex returns a data structure specifying how to index comprehension +// results so that callers do not have to recompute the comprehension more than once. +// If no index is found, returns nil. +func (c *Compiler) ComprehensionIndex(term *Term) *ComprehensionIndex { + return c.comprehensionIndices[term] +} + +// GetArity returns the number of args a function referred to by ref takes. If +// ref refers to built-in function, the built-in declaration is consulted, +// otherwise, the ref is used to perform a ruleset lookup. +func (c *Compiler) GetArity(ref Ref) int { + if bi := c.builtins[ref.String()]; bi != nil { + return len(bi.Decl.Args()) + } + rules := c.GetRulesExact(ref) + if len(rules) == 0 { + return -1 + } + return len(rules[0].Head.Args) +} + +// GetRulesExact returns a slice of rules referred to by the reference. +// +// E.g., given the following module: +// +// package a.b.c +// +// p[k] = v { ... } # rule1 +// p[k1] = v1 { ... } # rule2 +// +// The following calls yield the rules on the right. +// +// GetRulesExact("data.a.b.c.p") => [rule1, rule2] +// GetRulesExact("data.a.b.c.p.x") => nil +// GetRulesExact("data.a.b.c") => nil +func (c *Compiler) GetRulesExact(ref Ref) (rules []*Rule) { + node := c.RuleTree + + for _, x := range ref { + if node = node.Child(x.Value); node == nil { + return nil + } + } + + return extractRules(node.Values) +} + +// GetRulesForVirtualDocument returns a slice of rules that produce the virtual +// document referred to by the reference. +// +// E.g., given the following module: +// +// package a.b.c +// +// p[k] = v { ... } # rule1 +// p[k1] = v1 { ... } # rule2 +// +// The following calls yield the rules on the right. +// +// GetRulesForVirtualDocument("data.a.b.c.p") => [rule1, rule2] +// GetRulesForVirtualDocument("data.a.b.c.p.x") => [rule1, rule2] +// GetRulesForVirtualDocument("data.a.b.c") => nil +func (c *Compiler) GetRulesForVirtualDocument(ref Ref) (rules []*Rule) { + + node := c.RuleTree + + for _, x := range ref { + if node = node.Child(x.Value); node == nil { + return nil + } + if len(node.Values) > 0 { + return extractRules(node.Values) + } + } + + return extractRules(node.Values) +} + +// GetRulesWithPrefix returns a slice of rules that share the prefix ref. +// +// E.g., given the following module: +// +// package a.b.c +// +// p[x] = y { ... } # rule1 +// p[k] = v { ... } # rule2 +// q { ... } # rule3 +// +// The following calls yield the rules on the right. +// +// GetRulesWithPrefix("data.a.b.c.p") => [rule1, rule2] +// GetRulesWithPrefix("data.a.b.c.p.a") => nil +// GetRulesWithPrefix("data.a.b.c") => [rule1, rule2, rule3] +func (c *Compiler) GetRulesWithPrefix(ref Ref) (rules []*Rule) { + + node := c.RuleTree + + for _, x := range ref { + if node = node.Child(x.Value); node == nil { + return nil + } + } + + var acc func(node *TreeNode) + + acc = func(node *TreeNode) { + rules = append(rules, extractRules(node.Values)...) + for _, child := range node.Children { + if child.Hide { + continue + } + acc(child) + } + } + + acc(node) + + return rules +} + +func extractRules(s []util.T) (rules []*Rule) { + for _, r := range s { + rules = append(rules, r.(*Rule)) + } + return rules +} + +// GetRules returns a slice of rules that are referred to by ref. +// +// E.g., given the following module: +// +// package a.b.c +// +// p[x] = y { q[x] = y; ... } # rule1 +// q[x] = y { ... } # rule2 +// +// The following calls yield the rules on the right. +// +// GetRules("data.a.b.c.p") => [rule1] +// GetRules("data.a.b.c.p.x") => [rule1] +// GetRules("data.a.b.c.q") => [rule2] +// GetRules("data.a.b.c") => [rule1, rule2] +// GetRules("data.a.b.d") => nil +func (c *Compiler) GetRules(ref Ref) (rules []*Rule) { + + set := map[*Rule]struct{}{} + + for _, rule := range c.GetRulesForVirtualDocument(ref) { + set[rule] = struct{}{} + } + + for _, rule := range c.GetRulesWithPrefix(ref) { + set[rule] = struct{}{} + } + + for rule := range set { + rules = append(rules, rule) + } + + return rules +} + +// GetRulesDynamic returns a slice of rules that could be referred to by a ref. +// +// Deprecated: use GetRulesDynamicWithOpts +func (c *Compiler) GetRulesDynamic(ref Ref) []*Rule { + return c.GetRulesDynamicWithOpts(ref, RulesOptions{}) +} + +// GetRulesDynamicWithOpts returns a slice of rules that could be referred to by +// a ref. +// When parts of the ref are statically known, we use that information to narrow +// down which rules the ref could refer to, but in the most general case this +// will be an over-approximation. +// +// E.g., given the following modules: +// +// package a.b.c +// +// r1 = 1 # rule1 +// +// and: +// +// package a.d.c +// +// r2 = 2 # rule2 +// +// The following calls yield the rules on the right. +// +// GetRulesDynamicWithOpts("data.a[x].c[y]", opts) => [rule1, rule2] +// GetRulesDynamicWithOpts("data.a[x].c.r2", opts) => [rule2] +// GetRulesDynamicWithOpts("data.a.b[x][y]", opts) => [rule1] +// +// Using the RulesOptions parameter, the inclusion of hidden modules can be +// controlled: +// +// With +// +// package system.main +// +// r3 = 3 # rule3 +// +// We'd get this result: +// +// GetRulesDynamicWithOpts("data[x]", RulesOptions{IncludeHiddenModules: true}) => [rule1, rule2, rule3] +// +// Without the options, it would be excluded. +func (c *Compiler) GetRulesDynamicWithOpts(ref Ref, opts RulesOptions) []*Rule { + node := c.RuleTree + + set := map[*Rule]struct{}{} + var walk func(node *TreeNode, i int) + walk = func(node *TreeNode, i int) { + if i >= len(ref) { + // We've reached the end of the reference and want to collect everything + // under this "prefix". + node.DepthFirst(func(descendant *TreeNode) bool { + insertRules(set, descendant.Values) + if opts.IncludeHiddenModules { + return false + } + return descendant.Hide + }) + } else if i == 0 || IsConstant(ref[i].Value) { + // The head of the ref is always grounded. In case another part of the + // ref is also grounded, we can lookup the exact child. If it's not found + // we can immediately return... + if child := node.Child(ref[i].Value); child == nil { + return + } else if len(child.Values) > 0 { + // If there are any rules at this position, it's what the ref would + // refer to. We can just append those and stop here. + insertRules(set, child.Values) + } else { + // Otherwise, we continue using the child node. + walk(child, i+1) + } + } else { + // This part of the ref is a dynamic term. We can't know what it refers + // to and will just need to try all of the children. + for _, child := range node.Children { + if child.Hide && !opts.IncludeHiddenModules { + continue + } + insertRules(set, child.Values) + walk(child, i+1) + } + } + } + + walk(node, 0) + rules := make([]*Rule, 0, len(set)) + for rule := range set { + rules = append(rules, rule) + } + return rules +} + +// Utility: add all rule values to the set. +func insertRules(set map[*Rule]struct{}, rules []util.T) { + for _, rule := range rules { + set[rule.(*Rule)] = struct{}{} + } +} + +// RuleIndex returns a RuleIndex built for the rule set referred to by path. +// The path must refer to the rule set exactly, i.e., given a rule set at path +// data.a.b.c.p, refs data.a.b.c.p.x and data.a.b.c would not return a +// RuleIndex built for the rule. +func (c *Compiler) RuleIndex(path Ref) RuleIndex { + r, ok := c.ruleIndices.Get(path) + if !ok { + return nil + } + return r.(RuleIndex) +} + +// PassesTypeCheck determines whether the given body passes type checking +func (c *Compiler) PassesTypeCheck(body Body) bool { + checker := newTypeChecker().WithSchemaSet(c.schemaSet).WithInputType(c.inputType) + env := c.TypeEnv + _, errs := checker.CheckBody(env, body) + return len(errs) == 0 +} + +// ModuleLoader defines the interface that callers can implement to enable lazy +// loading of modules during compilation. +type ModuleLoader func(resolved map[string]*Module) (parsed map[string]*Module, err error) + +// WithModuleLoader sets f as the ModuleLoader on the compiler. +// +// The compiler will invoke the ModuleLoader after resolving all references in +// the current set of input modules. The ModuleLoader can return a new +// collection of parsed modules that are to be included in the compilation +// process. This process will repeat until the ModuleLoader returns an empty +// collection or an error. If an error is returned, compilation will stop +// immediately. +func (c *Compiler) WithModuleLoader(f ModuleLoader) *Compiler { + c.moduleLoader = f + return c +} + +func (c *Compiler) counterAdd(name string, n uint64) { + if c.metrics == nil { + return + } + c.metrics.Counter(name).Add(n) +} + +func (c *Compiler) buildRuleIndices() { + + c.RuleTree.DepthFirst(func(node *TreeNode) bool { + if len(node.Values) == 0 { + return false + } + index := newBaseDocEqIndex(func(ref Ref) bool { + return isVirtual(c.RuleTree, ref.GroundPrefix()) + }) + if rules := extractRules(node.Values); index.Build(rules) { + c.ruleIndices.Put(rules[0].Path(), index) + } + return false + }) + +} + +func (c *Compiler) buildComprehensionIndices() { + for _, name := range c.sorted { + WalkRules(c.Modules[name], func(r *Rule) bool { + candidates := r.Head.Args.Vars() + candidates.Update(ReservedVars) + n := buildComprehensionIndices(c.debug, c.GetArity, candidates, c.RewrittenVars, r.Body, c.comprehensionIndices) + c.counterAdd(compileStageComprehensionIndexBuild, n) + return false + }) + } +} + +// checkRecursion ensures that there are no recursive definitions, i.e., there are +// no cycles in the Graph. +func (c *Compiler) checkRecursion() { + eq := func(a, b util.T) bool { + return a.(*Rule) == b.(*Rule) + } + + c.RuleTree.DepthFirst(func(node *TreeNode) bool { + for _, rule := range node.Values { + for node := rule.(*Rule); node != nil; node = node.Else { + c.checkSelfPath(node.Loc(), eq, node, node) + } + } + return false + }) +} + +func (c *Compiler) checkSelfPath(loc *Location, eq func(a, b util.T) bool, a, b util.T) { + tr := NewGraphTraversal(c.Graph) + if p := util.DFSPath(tr, eq, a, b); len(p) > 0 { + n := []string{} + for _, x := range p { + n = append(n, astNodeToString(x)) + } + c.err(NewError(RecursionErr, loc, "rule %v is recursive: %v", astNodeToString(a), strings.Join(n, " -> "))) + } +} + +func astNodeToString(x interface{}) string { + switch x := x.(type) { + case *Rule: + return string(x.Head.Name) + default: + panic("not reached") + } +} + +// checkRuleConflicts ensures that rules definitions are not in conflict. +func (c *Compiler) checkRuleConflicts() { + c.RuleTree.DepthFirst(func(node *TreeNode) bool { + if len(node.Values) == 0 { + return false + } + + kinds := map[DocKind]struct{}{} + defaultRules := 0 + arities := map[int]struct{}{} + declared := false + + for _, rule := range node.Values { + r := rule.(*Rule) + kinds[r.Head.DocKind()] = struct{}{} + arities[len(r.Head.Args)] = struct{}{} + if r.Head.Assign { + declared = true + } + if r.Default { + defaultRules++ + } + } + + name := Var(node.Key.(String)) + + if declared && len(node.Values) > 1 { + c.err(NewError(TypeErr, node.Values[0].(*Rule).Loc(), "rule named %v redeclared at %v", name, node.Values[1].(*Rule).Loc())) + } else if len(kinds) > 1 || len(arities) > 1 { + c.err(NewError(TypeErr, node.Values[0].(*Rule).Loc(), "conflicting rules named %v found", name)) + } else if defaultRules > 1 { + c.err(NewError(TypeErr, node.Values[0].(*Rule).Loc(), "multiple default rules named %s found", name)) + } + + return false + }) + + if c.pathExists != nil { + for _, err := range CheckPathConflicts(c, c.pathExists) { + c.err(err) + } + } + + c.ModuleTree.DepthFirst(func(node *ModuleTreeNode) bool { + for _, mod := range node.Modules { + for _, rule := range mod.Rules { + if childNode, ok := node.Children[String(rule.Head.Name)]; ok { + for _, childMod := range childNode.Modules { + msg := fmt.Sprintf("%v conflicts with rule defined at %v", childMod.Package, rule.Loc()) + c.err(NewError(TypeErr, mod.Package.Loc(), msg)) + } + } + } + } + return false + }) +} + +func (c *Compiler) checkUndefinedFuncs() { + for _, name := range c.sorted { + m := c.Modules[name] + for _, err := range checkUndefinedFuncs(c.TypeEnv, m, c.GetArity, c.RewrittenVars) { + c.err(err) + } + } +} + +func checkUndefinedFuncs(env *TypeEnv, x interface{}, arity func(Ref) int, rwVars map[Var]Var) Errors { + + var errs Errors + + WalkExprs(x, func(expr *Expr) bool { + if !expr.IsCall() { + return false + } + ref := expr.Operator() + if arity := arity(ref); arity >= 0 { + operands := len(expr.Operands()) + if expr.Generated { // an output var was added + if !expr.IsEquality() && operands != arity+1 { + ref = rewriteVarsInRef(rwVars)(ref) + errs = append(errs, arityMismatchError(env, ref, expr, arity, operands-1)) + return true + } + } else { // either output var or not + if operands != arity && operands != arity+1 { + ref = rewriteVarsInRef(rwVars)(ref) + errs = append(errs, arityMismatchError(env, ref, expr, arity, operands)) + return true + } + } + return false + } + ref = rewriteVarsInRef(rwVars)(ref) + errs = append(errs, NewError(TypeErr, expr.Loc(), "undefined function %v", ref)) + return true + }) + + return errs +} + +func arityMismatchError(env *TypeEnv, f Ref, expr *Expr, exp, act int) *Error { + if want, ok := env.Get(f).(*types.Function); ok { // generate richer error for built-in functions + have := make([]types.Type, len(expr.Operands())) + for i, op := range expr.Operands() { + have[i] = env.Get(op) + } + return newArgError(expr.Loc(), f, "arity mismatch", have, want.FuncArgs()) + } + if act != 1 { + return NewError(TypeErr, expr.Loc(), "function %v has arity %d, got %d arguments", f, exp, act) + } + return NewError(TypeErr, expr.Loc(), "function %v has arity %d, got %d argument", f, exp, act) +} + +// checkSafetyRuleBodies ensures that variables appearing in negated expressions or non-target +// positions of built-in expressions will be bound when evaluating the rule from left +// to right, re-ordering as necessary. +func (c *Compiler) checkSafetyRuleBodies() { + for _, name := range c.sorted { + m := c.Modules[name] + WalkRules(m, func(r *Rule) bool { + safe := ReservedVars.Copy() + safe.Update(r.Head.Args.Vars()) + r.Body = c.checkBodySafety(safe, r.Body) + return false + }) + } +} + +func (c *Compiler) checkBodySafety(safe VarSet, b Body) Body { + reordered, unsafe := reorderBodyForSafety(c.builtins, c.GetArity, safe, b) + if errs := safetyErrorSlice(unsafe, c.RewrittenVars); len(errs) > 0 { + for _, err := range errs { + c.err(err) + } + return b + } + return reordered +} + +// SafetyCheckVisitorParams defines the AST visitor parameters to use for collecting +// variables during the safety check. This has to be exported because it's relied on +// by the copy propagation implementation in topdown. +var SafetyCheckVisitorParams = VarVisitorParams{ + SkipRefCallHead: true, + SkipClosures: true, +} + +// checkSafetyRuleHeads ensures that variables appearing in the head of a +// rule also appear in the body. +func (c *Compiler) checkSafetyRuleHeads() { + + for _, name := range c.sorted { + m := c.Modules[name] + WalkRules(m, func(r *Rule) bool { + safe := r.Body.Vars(SafetyCheckVisitorParams) + safe.Update(r.Head.Args.Vars()) + unsafe := r.Head.Vars().Diff(safe) + for v := range unsafe { + if w, ok := c.RewrittenVars[v]; ok { + v = w + } + if !v.IsGenerated() { + c.err(NewError(UnsafeVarErr, r.Loc(), "var %v is unsafe", v)) + } + } + return false + }) + } +} + +func compileSchema(goSchema interface{}, allowNet []string) (*gojsonschema.Schema, error) { + gojsonschema.SetAllowNet(allowNet) + + var refLoader gojsonschema.JSONLoader + sl := gojsonschema.NewSchemaLoader() + + if goSchema != nil { + refLoader = gojsonschema.NewGoLoader(goSchema) + } else { + return nil, fmt.Errorf("no schema as input to compile") + } + schemasCompiled, err := sl.Compile(refLoader) + if err != nil { + return nil, fmt.Errorf("unable to compile the schema: %w", err) + } + return schemasCompiled, nil +} + +func mergeSchemas(schemas ...*gojsonschema.SubSchema) (*gojsonschema.SubSchema, error) { + if len(schemas) == 0 { + return nil, nil + } + var result = schemas[0] + + for i := range schemas { + if len(schemas[i].PropertiesChildren) > 0 { + if !schemas[i].Types.Contains("object") { + if err := schemas[i].Types.Add("object"); err != nil { + return nil, fmt.Errorf("unable to set the type in schemas") + } + } + } else if len(schemas[i].ItemsChildren) > 0 { + if !schemas[i].Types.Contains("array") { + if err := schemas[i].Types.Add("array"); err != nil { + return nil, fmt.Errorf("unable to set the type in schemas") + } + } + } + } + + for i := 1; i < len(schemas); i++ { + if result.Types.String() != schemas[i].Types.String() { + return nil, fmt.Errorf("unable to merge these schemas: type mismatch: %v and %v", result.Types.String(), schemas[i].Types.String()) + } else if result.Types.Contains("object") && len(result.PropertiesChildren) > 0 && schemas[i].Types.Contains("object") && len(schemas[i].PropertiesChildren) > 0 { + result.PropertiesChildren = append(result.PropertiesChildren, schemas[i].PropertiesChildren...) + } else if result.Types.Contains("array") && len(result.ItemsChildren) > 0 && schemas[i].Types.Contains("array") && len(schemas[i].ItemsChildren) > 0 { + for j := 0; j < len(schemas[i].ItemsChildren); j++ { + if len(result.ItemsChildren)-1 < j && !(len(schemas[i].ItemsChildren)-1 < j) { + result.ItemsChildren = append(result.ItemsChildren, schemas[i].ItemsChildren[j]) + } + if result.ItemsChildren[j].Types.String() != schemas[i].ItemsChildren[j].Types.String() { + return nil, fmt.Errorf("unable to merge these schemas") + } + } + } + } + return result, nil +} + +func parseSchema(schema interface{}) (types.Type, error) { + subSchema, ok := schema.(*gojsonschema.SubSchema) + if !ok { + return nil, fmt.Errorf("unexpected schema type %v", subSchema) + } + + // Handle referenced schemas, returns directly when a $ref is found + if subSchema.RefSchema != nil { + return parseSchema(subSchema.RefSchema) + } + + // Handle anyOf + if subSchema.AnyOf != nil { + var orType types.Type + + // If there is a core schema, find its type first + if subSchema.Types.IsTyped() { + copySchema := *subSchema + copySchemaRef := ©Schema + copySchemaRef.AnyOf = nil + coreType, err := parseSchema(copySchemaRef) + if err != nil { + return nil, fmt.Errorf("unexpected schema type %v: %w", subSchema, err) + } + + // Only add Object type with static props to orType + if objType, ok := coreType.(*types.Object); ok { + if objType.StaticProperties() != nil && objType.DynamicProperties() == nil { + orType = types.Or(orType, coreType) + } + } + } + + // Iterate through every property of AnyOf and add it to orType + for _, pSchema := range subSchema.AnyOf { + newtype, err := parseSchema(pSchema) + if err != nil { + return nil, fmt.Errorf("unexpected schema type %v: %w", pSchema, err) + } + orType = types.Or(newtype, orType) + } + + return orType, nil + } + + if subSchema.AllOf != nil { + subSchemaArray := subSchema.AllOf + allOfResult, err := mergeSchemas(subSchemaArray...) + if err != nil { + return nil, err + } + + if subSchema.Types.IsTyped() { + if (subSchema.Types.Contains("object") && allOfResult.Types.Contains("object")) || (subSchema.Types.Contains("array") && allOfResult.Types.Contains("array")) { + objectOrArrayResult, err := mergeSchemas(allOfResult, subSchema) + if err != nil { + return nil, err + } + return parseSchema(objectOrArrayResult) + } else if subSchema.Types.String() != allOfResult.Types.String() { + return nil, fmt.Errorf("unable to merge these schemas") + } + } + return parseSchema(allOfResult) + } + + if subSchema.Types.IsTyped() { + if subSchema.Types.Contains("boolean") { + return types.B, nil + + } else if subSchema.Types.Contains("string") { + return types.S, nil + + } else if subSchema.Types.Contains("integer") || subSchema.Types.Contains("number") { + return types.N, nil + + } else if subSchema.Types.Contains("object") { + if len(subSchema.PropertiesChildren) > 0 { + staticProps := make([]*types.StaticProperty, 0, len(subSchema.PropertiesChildren)) + for _, pSchema := range subSchema.PropertiesChildren { + newtype, err := parseSchema(pSchema) + if err != nil { + return nil, fmt.Errorf("unexpected schema type %v: %w", pSchema, err) + } + staticProps = append(staticProps, types.NewStaticProperty(pSchema.Property, newtype)) + } + return types.NewObject(staticProps, nil), nil + } + return types.NewObject(nil, types.NewDynamicProperty(types.A, types.A)), nil + + } else if subSchema.Types.Contains("array") { + if len(subSchema.ItemsChildren) > 0 { + if subSchema.ItemsChildrenIsSingleSchema { + iSchema := subSchema.ItemsChildren[0] + newtype, err := parseSchema(iSchema) + if err != nil { + return nil, fmt.Errorf("unexpected schema type %v", iSchema) + } + return types.NewArray(nil, newtype), nil + } + newTypes := make([]types.Type, 0, len(subSchema.ItemsChildren)) + for i := 0; i != len(subSchema.ItemsChildren); i++ { + iSchema := subSchema.ItemsChildren[i] + newtype, err := parseSchema(iSchema) + if err != nil { + return nil, fmt.Errorf("unexpected schema type %v", iSchema) + } + newTypes = append(newTypes, newtype) + } + return types.NewArray(newTypes, nil), nil + } + return types.NewArray(nil, types.A), nil + } + } + + // Assume types if not specified in schema + if len(subSchema.PropertiesChildren) > 0 { + if err := subSchema.Types.Add("object"); err == nil { + return parseSchema(subSchema) + } + } else if len(subSchema.ItemsChildren) > 0 { + if err := subSchema.Types.Add("array"); err == nil { + return parseSchema(subSchema) + } + } + + return types.A, nil +} + +// checkTypes runs the type checker on all rules. The type checker builds a +// TypeEnv that is stored on the compiler. +func (c *Compiler) checkTypes() { + // Recursion is caught in earlier step, so this cannot fail. + sorted, _ := c.Graph.Sort() + checker := newTypeChecker(). + WithSchemaSet(c.schemaSet). + WithInputType(c.inputType). + WithVarRewriter(rewriteVarsInRef(c.RewrittenVars)) + env, errs := checker.CheckTypes(c.TypeEnv, sorted) + for _, err := range errs { + c.err(err) + } + c.TypeEnv = env +} + +func (c *Compiler) checkUnsafeBuiltins() { + for _, name := range c.sorted { + errs := checkUnsafeBuiltins(c.unsafeBuiltinsMap, c.Modules[name]) + for _, err := range errs { + c.err(err) + } + } +} + +func (c *Compiler) runStage(metricName string, f func()) { + if c.metrics != nil { + c.metrics.Timer(metricName).Start() + defer c.metrics.Timer(metricName).Stop() + } + f() +} + +func (c *Compiler) runStageAfter(metricName string, s CompilerStage) *Error { + if c.metrics != nil { + c.metrics.Timer(metricName).Start() + defer c.metrics.Timer(metricName).Stop() + } + return s(c) +} + +func (c *Compiler) compile() { + + defer func() { + if r := recover(); r != nil && r != errLimitReached { + panic(r) + } + }() + + for _, s := range c.stages { + c.runStage(s.metricName, s.f) + if c.Failed() { + return + } + for _, s := range c.after[s.name] { + err := c.runStageAfter(s.MetricName, s.Stage) + if err != nil { + c.err(err) + } + } + } +} + +func (c *Compiler) init() { + + if c.initialized { + return + } + + if c.capabilities == nil { + c.capabilities = CapabilitiesForThisVersion() + } + + c.builtins = make(map[string]*Builtin, len(c.capabilities.Builtins)+len(c.customBuiltins)) + + for _, bi := range c.capabilities.Builtins { + c.builtins[bi.Name] = bi + } + + for name, bi := range c.customBuiltins { + c.builtins[name] = bi + } + + // Load the global input schema if one was provided. + if c.schemaSet != nil { + if schema := c.schemaSet.Get(SchemaRootRef); schema != nil { + tpe, err := loadSchema(schema, c.capabilities.AllowNet) + if err != nil { + c.err(NewError(TypeErr, nil, err.Error())) + } else { + c.inputType = tpe + } + } + } + + c.TypeEnv = newTypeChecker(). + WithSchemaSet(c.schemaSet). + WithInputType(c.inputType). + Env(c.builtins) + + c.initialized = true +} + +func (c *Compiler) err(err *Error) { + if c.maxErrs > 0 && len(c.Errors) >= c.maxErrs { + c.Errors = append(c.Errors, errLimitReached) + panic(errLimitReached) + } + c.Errors = append(c.Errors, err) +} + +func (c *Compiler) getExports() *util.HashMap { + + rules := util.NewHashMap(func(a, b util.T) bool { + r1 := a.(Ref) + r2 := a.(Ref) + return r1.Equal(r2) + }, func(v util.T) int { + return v.(Ref).Hash() + }) + + for _, name := range c.sorted { + mod := c.Modules[name] + rv, ok := rules.Get(mod.Package.Path) + if !ok { + rv = []Var{} + } + rvs := rv.([]Var) + + for _, rule := range mod.Rules { + rvs = append(rvs, rule.Head.Name) + } + rules.Put(mod.Package.Path, rvs) + } + + return rules +} + +// resolveAllRefs resolves references in expressions to their fully qualified values. +// +// For instance, given the following module: +// +// package a.b +// import data.foo.bar +// p[x] { bar[_] = x } +// +// The reference "bar[_]" would be resolved to "data.foo.bar[_]". +func (c *Compiler) resolveAllRefs() { + + rules := c.getExports() + + for _, name := range c.sorted { + mod := c.Modules[name] + + var ruleExports []Var + if x, ok := rules.Get(mod.Package.Path); ok { + ruleExports = x.([]Var) + } + + globals := getGlobals(mod.Package, ruleExports, mod.Imports) + + WalkRules(mod, func(rule *Rule) bool { + err := resolveRefsInRule(globals, rule) + if err != nil { + c.err(NewError(CompileErr, rule.Location, err.Error())) + } + return false + }) + + // Once imports have been resolved, they are no longer needed. + mod.Imports = nil + } + + if c.moduleLoader != nil { + + parsed, err := c.moduleLoader(c.Modules) + if err != nil { + c.err(NewError(CompileErr, nil, err.Error())) + return + } + + if len(parsed) == 0 { + return + } + + for id, module := range parsed { + c.Modules[id] = module.Copy() + c.sorted = append(c.sorted, id) + } + + sort.Strings(c.sorted) + c.resolveAllRefs() + } +} + +func (c *Compiler) initLocalVarGen() { + c.localvargen = newLocalVarGeneratorForModuleSet(c.sorted, c.Modules) +} + +func (c *Compiler) rewriteComprehensionTerms() { + f := newEqualityFactory(c.localvargen) + for _, name := range c.sorted { + mod := c.Modules[name] + _, _ = rewriteComprehensionTerms(f, mod) // ignore error + } +} + +func (c *Compiler) rewriteExprTerms() { + for _, name := range c.sorted { + mod := c.Modules[name] + WalkRules(mod, func(rule *Rule) bool { + rewriteExprTermsInHead(c.localvargen, rule) + rule.Body = rewriteExprTermsInBody(c.localvargen, rule.Body) + return false + }) + } +} + +func (c *Compiler) checkVoidCalls() { + for _, name := range c.sorted { + mod := c.Modules[name] + for _, err := range checkVoidCalls(c.TypeEnv, mod) { + c.err(err) + } + } +} + +func (c *Compiler) rewritePrintCalls() { + if !c.enablePrintStatements { + for _, name := range c.sorted { + erasePrintCalls(c.Modules[name]) + } + return + } + for _, name := range c.sorted { + mod := c.Modules[name] + WalkRules(mod, func(r *Rule) bool { + safe := r.Head.Args.Vars() + safe.Update(ReservedVars) + WalkBodies(r, func(b Body) bool { + for _, err := range rewritePrintCalls(c.localvargen, c.GetArity, safe, b) { + c.err(err) + } + return false + }) + return false + }) + } +} + +// checkVoidCalls returns errors for any expressions that treat void function +// calls as values. The only void functions in Rego are specific built-ins like +// print(). +func checkVoidCalls(env *TypeEnv, x interface{}) Errors { + var errs Errors + WalkTerms(x, func(x *Term) bool { + if call, ok := x.Value.(Call); ok { + if tpe, ok := env.Get(call[0]).(*types.Function); ok && tpe.Result() == nil { + errs = append(errs, NewError(TypeErr, x.Loc(), "%v used as value", call)) + } + } + return false + }) + return errs +} + +// rewritePrintCalls will rewrite the body so that print operands are captured +// in local variables and their evaluation occurs within a comprehension. +// Wrapping the terms inside of a comprehension ensures that undefined values do +// not short-circuit evaluation. +// +// For example, given the following print statement: +// +// print("the value of x is:", input.x) +// +// The expression would be rewritten to: +// +// print({__local0__ | __local0__ = "the value of x is:"}, {__local1__ | __local1__ = input.x}) +func rewritePrintCalls(gen *localVarGenerator, getArity func(Ref) int, globals VarSet, body Body) Errors { + + var errs Errors + + // Visit comprehension bodies recursively to ensure print statements inside + // those bodies only close over variables that are safe. + for i := range body { + if ContainsComprehensions(body[i]) { + safe := outputVarsForBody(body[:i], getArity, globals) + safe.Update(globals) + WalkClosures(body[i], func(x interface{}) bool { + switch x := x.(type) { + case *SetComprehension: + errs = rewritePrintCalls(gen, getArity, safe, x.Body) + case *ArrayComprehension: + errs = rewritePrintCalls(gen, getArity, safe, x.Body) + case *ObjectComprehension: + errs = rewritePrintCalls(gen, getArity, safe, x.Body) + } + return true + }) + if len(errs) > 0 { + return errs + } + } + } + + for i := range body { + + if !isPrintCall(body[i]) { + continue + } + + var errs Errors + safe := outputVarsForBody(body[:i], getArity, globals) + safe.Update(globals) + args := body[i].Operands() + + for j := range args { + vis := NewVarVisitor().WithParams(SafetyCheckVisitorParams) + vis.Walk(args[j]) + unsafe := vis.Vars().Diff(safe) + for _, v := range unsafe.Sorted() { + errs = append(errs, NewError(CompileErr, args[j].Loc(), "var %v is undeclared", v)) + } + } + + if len(errs) > 0 { + return errs + } + + arr := NewArray() + + for j := range args { + x := NewTerm(gen.Generate()).SetLocation(args[j].Loc()) + capture := Equality.Expr(x, args[j]).SetLocation(args[j].Loc()) + arr = arr.Append(SetComprehensionTerm(x, NewBody(capture)).SetLocation(args[j].Loc())) + } + + body.Set(NewExpr([]*Term{ + NewTerm(InternalPrint.Ref()).SetLocation(body[i].Loc()), + NewTerm(arr).SetLocation(body[i].Loc()), + }).SetLocation(body[i].Loc()), i) + } + + return nil +} + +func erasePrintCalls(node interface{}) { + NewGenericVisitor(func(x interface{}) bool { + switch x := x.(type) { + case *Rule: + x.Body = erasePrintCallsInBody(x.Body) + case *ArrayComprehension: + x.Body = erasePrintCallsInBody(x.Body) + case *SetComprehension: + x.Body = erasePrintCallsInBody(x.Body) + case *ObjectComprehension: + x.Body = erasePrintCallsInBody(x.Body) + } + return false + }).Walk(node) +} + +func erasePrintCallsInBody(x Body) Body { + + if !containsPrintCall(x) { + return x + } + + var cpy Body + + for i := range x { + + // Recursively visit any comprehensions contained in this expression. + erasePrintCalls(x[i]) + + if !isPrintCall(x[i]) { + cpy.Append(x[i]) + } + } + + if len(cpy) == 0 { + term := BooleanTerm(true).SetLocation(x.Loc()) + expr := NewExpr(term).SetLocation(x.Loc()) + cpy.Append(expr) + } + + return cpy +} + +func containsPrintCall(x Body) bool { + var found bool + WalkExprs(x, func(expr *Expr) bool { + if !found { + if isPrintCall(expr) { + found = true + } + } + return found + }) + return found +} + +func isPrintCall(x *Expr) bool { + return x.IsCall() && x.Operator().Equal(Print.Ref()) +} + +// rewriteTermsInHead will rewrite rules so that the head does not contain any +// terms that require evaluation (e.g., refs or comprehensions). If the key or +// value contains one or more of these terms, the key or value will be moved +// into the body and assigned to a new variable. The new variable will replace +// the key or value in the head. +// +// For instance, given the following rule: +// +// p[{"foo": data.foo[i]}] { i < 100 } +// +// The rule would be re-written as: +// +// p[__local0__] { i < 100; __local0__ = {"foo": data.foo[i]} } +func (c *Compiler) rewriteRefsInHead() { + f := newEqualityFactory(c.localvargen) + for _, name := range c.sorted { + mod := c.Modules[name] + WalkRules(mod, func(rule *Rule) bool { + if requiresEval(rule.Head.Key) { + expr := f.Generate(rule.Head.Key) + rule.Head.Key = expr.Operand(0) + rule.Body.Append(expr) + } + if requiresEval(rule.Head.Value) { + expr := f.Generate(rule.Head.Value) + rule.Head.Value = expr.Operand(0) + rule.Body.Append(expr) + } + for i := 0; i < len(rule.Head.Args); i++ { + if requiresEval(rule.Head.Args[i]) { + expr := f.Generate(rule.Head.Args[i]) + rule.Head.Args[i] = expr.Operand(0) + rule.Body.Append(expr) + } + } + return false + }) + } +} + +func (c *Compiler) rewriteEquals() { + for _, name := range c.sorted { + mod := c.Modules[name] + rewriteEquals(mod) + } +} + +func (c *Compiler) rewriteDynamicTerms() { + f := newEqualityFactory(c.localvargen) + for _, name := range c.sorted { + mod := c.Modules[name] + WalkRules(mod, func(rule *Rule) bool { + rule.Body = rewriteDynamics(f, rule.Body) + return false + }) + } +} + +func (c *Compiler) rewriteLocalVars() { + + for _, name := range c.sorted { + mod := c.Modules[name] + gen := c.localvargen + + WalkRules(mod, func(rule *Rule) bool { + + // Rewrite assignments contained in head of rule. Assignments can + // occur in rule head if they're inside a comprehension. Note, + // assigned vars in comprehensions in the head will be rewritten + // first to preserve scoping rules. For example: + // + // p = [x | x := 1] { x := 2 } becomes p = [__local0__ | __local0__ = 1] { __local1__ = 2 } + // + // This behaviour is consistent scoping inside the body. For example: + // + // p = xs { x := 2; xs = [x | x := 1] } becomes p = xs { __local0__ = 2; xs = [__local1__ | __local1__ = 1] } + nestedXform := &rewriteNestedHeadVarLocalTransform{ + gen: gen, + RewrittenVars: c.RewrittenVars, + } + + NewGenericVisitor(nestedXform.Visit).Walk(rule.Head) + + for _, err := range nestedXform.errs { + c.err(err) + } + + // Rewrite assignments in body. + used := NewVarSet() + + if rule.Head.Key != nil { + used.Update(rule.Head.Key.Vars()) + } + + if rule.Head.Value != nil { + used.Update(rule.Head.Value.Vars()) + } + + stack := newLocalDeclaredVars() + + c.rewriteLocalArgVars(gen, stack, rule) + + body, declared, errs := rewriteLocalVars(gen, stack, used, rule.Body) + for _, err := range errs { + c.err(err) + } + + // For rewritten vars use the collection of all variables that + // were in the stack at some point in time. + for k, v := range stack.rewritten { + c.RewrittenVars[k] = v + } + + rule.Body = body + + // Rewrite vars in head that refer to locally declared vars in the body. + localXform := rewriteHeadVarLocalTransform{declared: declared} + + for i := range rule.Head.Args { + rule.Head.Args[i], _ = transformTerm(localXform, rule.Head.Args[i]) + } + + if rule.Head.Key != nil { + rule.Head.Key, _ = transformTerm(localXform, rule.Head.Key) + } + + if rule.Head.Value != nil { + rule.Head.Value, _ = transformTerm(localXform, rule.Head.Value) + } + + return false + }) + } +} + +type rewriteNestedHeadVarLocalTransform struct { + gen *localVarGenerator + errs Errors + RewrittenVars map[Var]Var +} + +func (xform *rewriteNestedHeadVarLocalTransform) Visit(x interface{}) bool { + + if term, ok := x.(*Term); ok { + + stop := false + stack := newLocalDeclaredVars() + + switch x := term.Value.(type) { + case *object: + cpy, _ := x.Map(func(k, v *Term) (*Term, *Term, error) { + kcpy := k.Copy() + NewGenericVisitor(xform.Visit).Walk(kcpy) + vcpy := v.Copy() + NewGenericVisitor(xform.Visit).Walk(vcpy) + return kcpy, vcpy, nil + }) + term.Value = cpy + stop = true + case *set: + cpy, _ := x.Map(func(v *Term) (*Term, error) { + vcpy := v.Copy() + NewGenericVisitor(xform.Visit).Walk(vcpy) + return vcpy, nil + }) + term.Value = cpy + stop = true + case *ArrayComprehension: + xform.errs = rewriteDeclaredVarsInArrayComprehension(xform.gen, stack, x, xform.errs) + stop = true + case *SetComprehension: + xform.errs = rewriteDeclaredVarsInSetComprehension(xform.gen, stack, x, xform.errs) + stop = true + case *ObjectComprehension: + xform.errs = rewriteDeclaredVarsInObjectComprehension(xform.gen, stack, x, xform.errs) + stop = true + } + + for k, v := range stack.rewritten { + xform.RewrittenVars[k] = v + } + + return stop + } + + return false +} + +type rewriteHeadVarLocalTransform struct { + declared map[Var]Var +} + +func (xform rewriteHeadVarLocalTransform) Transform(x interface{}) (interface{}, error) { + if v, ok := x.(Var); ok { + if gv, ok := xform.declared[v]; ok { + return gv, nil + } + } + return x, nil +} + +func (c *Compiler) rewriteLocalArgVars(gen *localVarGenerator, stack *localDeclaredVars, rule *Rule) { + + vis := &ruleArgLocalRewriter{ + stack: stack, + gen: gen, + } + + for i := range rule.Head.Args { + Walk(vis, rule.Head.Args[i]) + } + + for i := range vis.errs { + c.err(vis.errs[i]) + } +} + +type ruleArgLocalRewriter struct { + stack *localDeclaredVars + gen *localVarGenerator + errs []*Error +} + +func (vis *ruleArgLocalRewriter) Visit(x interface{}) Visitor { + + t, ok := x.(*Term) + if !ok { + return vis + } + + switch v := t.Value.(type) { + case Var: + gv, ok := vis.stack.Declared(v) + if !ok { + gv = vis.gen.Generate() + vis.stack.Insert(v, gv, argVar) + } + t.Value = gv + return nil + case *object: + if cpy, err := v.Map(func(k, v *Term) (*Term, *Term, error) { + vcpy := v.Copy() + Walk(vis, vcpy) + return k, vcpy, nil + }); err != nil { + vis.errs = append(vis.errs, NewError(CompileErr, t.Location, err.Error())) + } else { + t.Value = cpy + } + return nil + case Null, Boolean, Number, String, *ArrayComprehension, *SetComprehension, *ObjectComprehension, Set: + // Scalars are no-ops. Comprehensions are handled above. Sets must not + // contain variables. + return nil + case Call: + vis.errs = append(vis.errs, NewError(CompileErr, t.Location, "rule arguments cannot contain calls")) + return nil + default: + // Recurse on refs and arrays. Any embedded + // variables can be rewritten. + return vis + } +} + +func (c *Compiler) rewriteWithModifiers() { + f := newEqualityFactory(c.localvargen) + for _, name := range c.sorted { + mod := c.Modules[name] + t := NewGenericTransformer(func(x interface{}) (interface{}, error) { + body, ok := x.(Body) + if !ok { + return x, nil + } + body, err := rewriteWithModifiersInBody(c, f, body) + if err != nil { + c.err(err) + } + + return body, nil + }) + _, _ = Transform(t, mod) // ignore error + } +} + +func (c *Compiler) setModuleTree() { + c.ModuleTree = NewModuleTree(c.Modules) +} + +func (c *Compiler) setRuleTree() { + c.RuleTree = NewRuleTree(c.ModuleTree) +} + +func (c *Compiler) setGraph() { + list := func(r Ref) []*Rule { + return c.GetRulesDynamicWithOpts(r, RulesOptions{IncludeHiddenModules: true}) + } + c.Graph = NewGraph(c.Modules, list) +} + +type queryCompiler struct { + compiler *Compiler + qctx *QueryContext + typeEnv *TypeEnv + rewritten map[Var]Var + after map[string][]QueryCompilerStageDefinition + unsafeBuiltins map[string]struct{} + comprehensionIndices map[*Term]*ComprehensionIndex + enablePrintStatements bool +} + +func newQueryCompiler(compiler *Compiler) QueryCompiler { + qc := &queryCompiler{ + compiler: compiler, + qctx: nil, + after: map[string][]QueryCompilerStageDefinition{}, + comprehensionIndices: map[*Term]*ComprehensionIndex{}, + } + return qc +} + +func (qc *queryCompiler) WithEnablePrintStatements(yes bool) QueryCompiler { + qc.enablePrintStatements = yes + return qc +} + +func (qc *queryCompiler) WithContext(qctx *QueryContext) QueryCompiler { + qc.qctx = qctx + return qc +} + +func (qc *queryCompiler) WithStageAfter(after string, stage QueryCompilerStageDefinition) QueryCompiler { + qc.after[after] = append(qc.after[after], stage) + return qc +} + +func (qc *queryCompiler) WithUnsafeBuiltins(unsafe map[string]struct{}) QueryCompiler { + qc.unsafeBuiltins = unsafe + return qc +} + +func (qc *queryCompiler) RewrittenVars() map[Var]Var { + return qc.rewritten +} + +func (qc *queryCompiler) ComprehensionIndex(term *Term) *ComprehensionIndex { + if result, ok := qc.comprehensionIndices[term]; ok { + return result + } else if result, ok := qc.compiler.comprehensionIndices[term]; ok { + return result + } + return nil +} + +func (qc *queryCompiler) runStage(metricName string, qctx *QueryContext, query Body, s func(*QueryContext, Body) (Body, error)) (Body, error) { + if qc.compiler.metrics != nil { + qc.compiler.metrics.Timer(metricName).Start() + defer qc.compiler.metrics.Timer(metricName).Stop() + } + return s(qctx, query) +} + +func (qc *queryCompiler) runStageAfter(metricName string, query Body, s QueryCompilerStage) (Body, error) { + if qc.compiler.metrics != nil { + qc.compiler.metrics.Timer(metricName).Start() + defer qc.compiler.metrics.Timer(metricName).Stop() + } + return s(qc, query) +} + +func (qc *queryCompiler) Compile(query Body) (Body, error) { + if len(query) == 0 { + return nil, Errors{NewError(CompileErr, nil, "empty query cannot be compiled")} + } + + query = query.Copy() + + stages := []struct { + name string + metricName string + f func(*QueryContext, Body) (Body, error) + }{ + {"ResolveRefs", "query_compile_stage_resolve_refs", qc.resolveRefs}, + {"RewriteLocalVars", "query_compile_stage_rewrite_local_vars", qc.rewriteLocalVars}, + {"CheckVoidCalls", "query_compile_stage_check_void_calls", qc.checkVoidCalls}, + {"RewritePrintCalls", "query_compile_stage_rewrite_print_calls", qc.rewritePrintCalls}, + {"RewriteExprTerms", "query_compile_stage_rewrite_expr_terms", qc.rewriteExprTerms}, + {"RewriteComprehensionTerms", "query_compile_stage_rewrite_comprehension_terms", qc.rewriteComprehensionTerms}, + {"RewriteWithValues", "query_compile_stage_rewrite_with_values", qc.rewriteWithModifiers}, + {"CheckUndefinedFuncs", "query_compile_stage_check_undefined_funcs", qc.checkUndefinedFuncs}, + {"CheckSafety", "query_compile_stage_check_safety", qc.checkSafety}, + {"RewriteDynamicTerms", "query_compile_stage_rewrite_dynamic_terms", qc.rewriteDynamicTerms}, + {"CheckTypes", "query_compile_stage_check_types", qc.checkTypes}, + {"CheckUnsafeBuiltins", "query_compile_stage_check_unsafe_builtins", qc.checkUnsafeBuiltins}, + {"BuildComprehensionIndex", "query_compile_stage_build_comprehension_index", qc.buildComprehensionIndices}, + } + + qctx := qc.qctx.Copy() + + for _, s := range stages { + var err error + query, err = qc.runStage(s.metricName, qctx, query, s.f) + if err != nil { + return nil, qc.applyErrorLimit(err) + } + for _, s := range qc.after[s.name] { + query, err = qc.runStageAfter(s.MetricName, query, s.Stage) + if err != nil { + return nil, qc.applyErrorLimit(err) + } + } + } + + return query, nil +} + +func (qc *queryCompiler) TypeEnv() *TypeEnv { + return qc.typeEnv +} + +func (qc *queryCompiler) applyErrorLimit(err error) error { + if errs, ok := err.(Errors); ok { + if qc.compiler.maxErrs > 0 && len(errs) > qc.compiler.maxErrs { + err = append(errs[:qc.compiler.maxErrs], errLimitReached) + } + } + return err +} + +func (qc *queryCompiler) resolveRefs(qctx *QueryContext, body Body) (Body, error) { + + var globals map[Var]Ref + + if qctx != nil { + pkg := qctx.Package + // Query compiler ought to generate a package if one was not provided and one or more imports were provided. + // The generated package name could even be an empty string to avoid conflicts (it doesn't have to be valid syntactically) + if pkg == nil && len(qctx.Imports) > 0 { + pkg = &Package{Path: RefTerm(VarTerm("")).Value.(Ref)} + } + if pkg != nil { + var ruleExports []Var + rules := qc.compiler.getExports() + if exist, ok := rules.Get(pkg.Path); ok { + ruleExports = exist.([]Var) + } + + globals = getGlobals(qctx.Package, ruleExports, qctx.Imports) + qctx.Imports = nil + } + } + + ignore := &declaredVarStack{declaredVars(body)} + + return resolveRefsInBody(globals, ignore, body), nil +} + +func (qc *queryCompiler) rewriteComprehensionTerms(_ *QueryContext, body Body) (Body, error) { + gen := newLocalVarGenerator("q", body) + f := newEqualityFactory(gen) + node, err := rewriteComprehensionTerms(f, body) + if err != nil { + return nil, err + } + return node.(Body), nil +} + +func (qc *queryCompiler) rewriteDynamicTerms(_ *QueryContext, body Body) (Body, error) { + gen := newLocalVarGenerator("q", body) + f := newEqualityFactory(gen) + return rewriteDynamics(f, body), nil +} + +func (qc *queryCompiler) rewriteExprTerms(_ *QueryContext, body Body) (Body, error) { + gen := newLocalVarGenerator("q", body) + return rewriteExprTermsInBody(gen, body), nil +} + +func (qc *queryCompiler) rewriteLocalVars(_ *QueryContext, body Body) (Body, error) { + gen := newLocalVarGenerator("q", body) + stack := newLocalDeclaredVars() + body, _, err := rewriteLocalVars(gen, stack, nil, body) + if len(err) != 0 { + return nil, err + } + qc.rewritten = make(map[Var]Var, len(stack.rewritten)) + for k, v := range stack.rewritten { + // The vars returned during the rewrite will include all seen vars, + // even if they're not declared with an assignment operation. We don't + // want to include these inside the rewritten set though. + qc.rewritten[k] = v + } + return body, nil +} + +func (qc *queryCompiler) rewritePrintCalls(_ *QueryContext, body Body) (Body, error) { + if !qc.enablePrintStatements { + return erasePrintCallsInBody(body), nil + } + gen := newLocalVarGenerator("q", body) + if errs := rewritePrintCalls(gen, qc.compiler.GetArity, ReservedVars, body); len(errs) > 0 { + return nil, errs + } + return body, nil +} + +func (qc *queryCompiler) checkVoidCalls(_ *QueryContext, body Body) (Body, error) { + if errs := checkVoidCalls(qc.compiler.TypeEnv, body); len(errs) > 0 { + return nil, errs + } + return body, nil +} + +func (qc *queryCompiler) checkUndefinedFuncs(_ *QueryContext, body Body) (Body, error) { + if errs := checkUndefinedFuncs(qc.compiler.TypeEnv, body, qc.compiler.GetArity, qc.rewritten); len(errs) > 0 { + return nil, errs + } + return body, nil +} + +func (qc *queryCompiler) checkSafety(_ *QueryContext, body Body) (Body, error) { + safe := ReservedVars.Copy() + reordered, unsafe := reorderBodyForSafety(qc.compiler.builtins, qc.compiler.GetArity, safe, body) + if errs := safetyErrorSlice(unsafe, qc.RewrittenVars()); len(errs) > 0 { + return nil, errs + } + return reordered, nil +} + +func (qc *queryCompiler) checkTypes(_ *QueryContext, body Body) (Body, error) { + var errs Errors + checker := newTypeChecker(). + WithSchemaSet(qc.compiler.schemaSet). + WithInputType(qc.compiler.inputType). + WithVarRewriter(rewriteVarsInRef(qc.rewritten, qc.compiler.RewrittenVars)) + qc.typeEnv, errs = checker.CheckBody(qc.compiler.TypeEnv, body) + if len(errs) > 0 { + return nil, errs + } + + return body, nil +} + +func (qc *queryCompiler) checkUnsafeBuiltins(_ *QueryContext, body Body) (Body, error) { + var unsafe map[string]struct{} + if qc.unsafeBuiltins != nil { + unsafe = qc.unsafeBuiltins + } else { + unsafe = qc.compiler.unsafeBuiltinsMap + } + errs := checkUnsafeBuiltins(unsafe, body) + if len(errs) > 0 { + return nil, errs + } + return body, nil +} + +func (qc *queryCompiler) rewriteWithModifiers(_ *QueryContext, body Body) (Body, error) { + f := newEqualityFactory(newLocalVarGenerator("q", body)) + body, err := rewriteWithModifiersInBody(qc.compiler, f, body) + if err != nil { + return nil, Errors{err} + } + return body, nil +} + +func (qc *queryCompiler) buildComprehensionIndices(_ *QueryContext, body Body) (Body, error) { + // NOTE(tsandall): The query compiler does not have a metrics object so we + // cannot record index metrics currently. + _ = buildComprehensionIndices(qc.compiler.debug, qc.compiler.GetArity, ReservedVars, qc.RewrittenVars(), body, qc.comprehensionIndices) + return body, nil +} + +// ComprehensionIndex specifies how the comprehension term can be indexed. The keys +// tell the evaluator what variables to use for indexing. In the future, the index +// could be expanded with more information that would allow the evaluator to index +// a larger fragment of comprehensions (e.g., by closing over variables in the outer +// query.) +type ComprehensionIndex struct { + Term *Term + Keys []*Term +} + +func (ci *ComprehensionIndex) String() string { + if ci == nil { + return "" + } + return fmt.Sprintf("", NewArray(ci.Keys...)) +} + +func buildComprehensionIndices(dbg debug.Debug, arity func(Ref) int, candidates VarSet, rwVars map[Var]Var, node interface{}, result map[*Term]*ComprehensionIndex) uint64 { + var n uint64 + cpy := candidates.Copy() + WalkBodies(node, func(b Body) bool { + for _, expr := range b { + index := getComprehensionIndex(dbg, arity, cpy, rwVars, expr) + if index != nil { + result[index.Term] = index + n++ + } + // Any variables appearing in the expressions leading up to the comprehension + // are fair-game to be used as index keys. + cpy.Update(expr.Vars(VarVisitorParams{SkipClosures: true, SkipRefCallHead: true})) + } + return false + }) + return n +} + +func getComprehensionIndex(dbg debug.Debug, arity func(Ref) int, candidates VarSet, rwVars map[Var]Var, expr *Expr) *ComprehensionIndex { + + // Ignore everything except = expressions. Extract + // the comprehension term from the expression. + if !expr.IsEquality() || expr.Negated || len(expr.With) > 0 { + // No debug message, these are assumed to be known hinderances + // to comprehension indexing. + return nil + } + + var term *Term + + lhs, rhs := expr.Operand(0), expr.Operand(1) + + if _, ok := lhs.Value.(Var); ok && IsComprehension(rhs.Value) { + term = rhs + } else if _, ok := rhs.Value.(Var); ok && IsComprehension(lhs.Value) { + term = lhs + } + + if term == nil { + // no debug for this, it's the ordinary "nothing to do here" case + return nil + } + + // Ignore comprehensions that contain expressions that close over variables + // in the outer body if those variables are not also output variables in the + // comprehension body. In other words, ignore comprehensions that we cannot + // safely evaluate without bindings from the outer body. For example: + // + // x = [1] + // [true | data.y[z] = x] # safe to evaluate w/o outer body + // [true | data.y[z] = x[0]] # NOT safe to evaluate because 'x' would be unsafe. + // + // By identifying output variables in the body we also know what to index on by + // intersecting with candidate variables from the outer query. + // + // For example: + // + // x = data.foo[_] + // _ = [y | data.bar[y] = x] # index on 'x' + // + // This query goes from O(data.foo*data.bar) to O(data.foo+data.bar). + var body Body + + switch x := term.Value.(type) { + case *ArrayComprehension: + body = x.Body + case *SetComprehension: + body = x.Body + case *ObjectComprehension: + body = x.Body + } + + outputs := outputVarsForBody(body, arity, ReservedVars) + unsafe := body.Vars(SafetyCheckVisitorParams).Diff(outputs).Diff(ReservedVars) + + if len(unsafe) > 0 { + dbg.Printf("%s: comprehension index: unsafe vars: %v", expr.Location, unsafe) + return nil + } + + // Similarly, ignore comprehensions that contain references with output variables + // that intersect with the candidates. Indexing these comprehensions could worsen + // performance. + regressionVis := newComprehensionIndexRegressionCheckVisitor(candidates) + regressionVis.Walk(body) + if regressionVis.worse { + dbg.Printf("%s: comprehension index: output vars intersect candidates", expr.Location) + return nil + } + + // Check if any nested comprehensions close over candidates. If any intersection is found + // the comprehension cannot be cached because it would require closing over the candidates + // which the evaluator does not support today. + nestedVis := newComprehensionIndexNestedCandidateVisitor(candidates) + nestedVis.Walk(body) + if nestedVis.found { + dbg.Printf("%s: comprehension index: nested comprehensions close over candidates", expr.Location) + return nil + } + + // Make a sorted set of variable names that will serve as the index key set. + // Sort to ensure deterministic indexing. In future this could be relaxed + // if we can decide that one ordering is better than another. If the set is + // empty, there is no indexing to do. + indexVars := candidates.Intersect(outputs) + if len(indexVars) == 0 { + dbg.Printf("%s: comprehension index: no index vars", expr.Location) + return nil + } + + result := make([]*Term, 0, len(indexVars)) + + for v := range indexVars { + result = append(result, NewTerm(v)) + } + + sort.Slice(result, func(i, j int) bool { + return result[i].Value.Compare(result[j].Value) < 0 + }) + + debugRes := make([]*Term, len(result)) + for i, r := range result { + if o, ok := rwVars[r.Value.(Var)]; ok { + debugRes[i] = NewTerm(o) + } else { + debugRes[i] = r + } + } + dbg.Printf("%s: comprehension index: built with keys: %v", expr.Location, debugRes) + return &ComprehensionIndex{Term: term, Keys: result} +} + +type comprehensionIndexRegressionCheckVisitor struct { + candidates VarSet + seen VarSet + worse bool +} + +// TODO(tsandall): Improve this so that users can either supply this list explicitly +// or the information is maintained on the built-in function declaration. What we really +// need to know is whether the built-in function allows callers to push down output +// values or not. It's unlikely that anything outside of OPA does this today so this +// solution is fine for now. +var comprehensionIndexBlacklist = map[string]int{ + WalkBuiltin.Name: len(WalkBuiltin.Decl.Args()), +} + +func newComprehensionIndexRegressionCheckVisitor(candidates VarSet) *comprehensionIndexRegressionCheckVisitor { + return &comprehensionIndexRegressionCheckVisitor{ + candidates: candidates, + seen: NewVarSet(), + } +} + +func (vis *comprehensionIndexRegressionCheckVisitor) Walk(x interface{}) { + NewGenericVisitor(vis.visit).Walk(x) +} + +func (vis *comprehensionIndexRegressionCheckVisitor) visit(x interface{}) bool { + if !vis.worse { + switch x := x.(type) { + case *Expr: + operands := x.Operands() + if pos := comprehensionIndexBlacklist[x.Operator().String()]; pos > 0 && pos < len(operands) { + vis.assertEmptyIntersection(operands[pos].Vars()) + } + case Ref: + vis.assertEmptyIntersection(x.OutputVars()) + case Var: + vis.seen.Add(x) + // Always skip comprehensions. We do not have to visit their bodies here. + case *ArrayComprehension, *SetComprehension, *ObjectComprehension: + return true + } + } + return vis.worse +} + +func (vis *comprehensionIndexRegressionCheckVisitor) assertEmptyIntersection(vs VarSet) { + for v := range vs { + if vis.candidates.Contains(v) && !vis.seen.Contains(v) { + vis.worse = true + return + } + } +} + +type comprehensionIndexNestedCandidateVisitor struct { + candidates VarSet + found bool +} + +func newComprehensionIndexNestedCandidateVisitor(candidates VarSet) *comprehensionIndexNestedCandidateVisitor { + return &comprehensionIndexNestedCandidateVisitor{ + candidates: candidates, + } +} + +func (vis *comprehensionIndexNestedCandidateVisitor) Walk(x interface{}) { + NewGenericVisitor(vis.visit).Walk(x) +} + +func (vis *comprehensionIndexNestedCandidateVisitor) visit(x interface{}) bool { + + if vis.found { + return true + } + + if v, ok := x.(Value); ok && IsComprehension(v) { + varVis := NewVarVisitor().WithParams(VarVisitorParams{SkipRefHead: true}) + varVis.Walk(v) + vis.found = len(varVis.Vars().Intersect(vis.candidates)) > 0 + return true + } + + return false +} + +// ModuleTreeNode represents a node in the module tree. The module +// tree is keyed by the package path. +type ModuleTreeNode struct { + Key Value + Modules []*Module + Children map[Value]*ModuleTreeNode + Hide bool +} + +// NewModuleTree returns a new ModuleTreeNode that represents the root +// of the module tree populated with the given modules. +func NewModuleTree(mods map[string]*Module) *ModuleTreeNode { + root := &ModuleTreeNode{ + Children: map[Value]*ModuleTreeNode{}, + } + for _, m := range mods { + node := root + for i, x := range m.Package.Path { + c, ok := node.Children[x.Value] + if !ok { + var hide bool + if i == 1 && x.Value.Compare(SystemDocumentKey) == 0 { + hide = true + } + c = &ModuleTreeNode{ + Key: x.Value, + Children: map[Value]*ModuleTreeNode{}, + Hide: hide, + } + node.Children[x.Value] = c + } + node = c + } + node.Modules = append(node.Modules, m) + } + return root +} + +// Size returns the number of modules in the tree. +func (n *ModuleTreeNode) Size() int { + s := len(n.Modules) + for _, c := range n.Children { + s += c.Size() + } + return s +} + +// DepthFirst performs a depth-first traversal of the module tree rooted at n. +// If f returns true, traversal will not continue to the children of n. +func (n *ModuleTreeNode) DepthFirst(f func(node *ModuleTreeNode) bool) { + if !f(n) { + for _, node := range n.Children { + node.DepthFirst(f) + } + } +} + +// TreeNode represents a node in the rule tree. The rule tree is keyed by +// rule path. +type TreeNode struct { + Key Value + Values []util.T + Children map[Value]*TreeNode + Sorted []Value + Hide bool +} + +// NewRuleTree returns a new TreeNode that represents the root +// of the rule tree populated with the given rules. +func NewRuleTree(mtree *ModuleTreeNode) *TreeNode { + + ruleSets := map[String][]util.T{} + + // Build rule sets for this package. + for _, mod := range mtree.Modules { + for _, rule := range mod.Rules { + key := String(rule.Head.Name) + ruleSets[key] = append(ruleSets[key], rule) + } + } + + // Each rule set becomes a leaf node. + children := map[Value]*TreeNode{} + sorted := make([]Value, 0, len(ruleSets)) + + for key, rules := range ruleSets { + sorted = append(sorted, key) + children[key] = &TreeNode{ + Key: key, + Children: nil, + Values: rules, + } + } + + // Each module in subpackage becomes child node. + for key, child := range mtree.Children { + sorted = append(sorted, key) + children[child.Key] = NewRuleTree(child) + } + + sort.Slice(sorted, func(i, j int) bool { + return sorted[i].Compare(sorted[j]) < 0 + }) + + return &TreeNode{ + Key: mtree.Key, + Values: nil, + Children: children, + Sorted: sorted, + Hide: mtree.Hide, + } +} + +// Size returns the number of rules in the tree. +func (n *TreeNode) Size() int { + s := len(n.Values) + for _, c := range n.Children { + s += c.Size() + } + return s +} + +// Child returns n's child with key k. +func (n *TreeNode) Child(k Value) *TreeNode { + switch k.(type) { + case String, Var: + return n.Children[k] + } + return nil +} + +// DepthFirst performs a depth-first traversal of the rule tree rooted at n. If +// f returns true, traversal will not continue to the children of n. +func (n *TreeNode) DepthFirst(f func(node *TreeNode) bool) { + if !f(n) { + for _, node := range n.Children { + node.DepthFirst(f) + } + } +} + +// Graph represents the graph of dependencies between rules. +type Graph struct { + adj map[util.T]map[util.T]struct{} + radj map[util.T]map[util.T]struct{} + nodes map[util.T]struct{} + sorted []util.T +} + +// NewGraph returns a new Graph based on modules. The list function must return +// the rules referred to directly by the ref. +func NewGraph(modules map[string]*Module, list func(Ref) []*Rule) *Graph { + + graph := &Graph{ + adj: map[util.T]map[util.T]struct{}{}, + radj: map[util.T]map[util.T]struct{}{}, + nodes: map[util.T]struct{}{}, + sorted: nil, + } + + // Create visitor to walk a rule AST and add edges to the rule graph for + // each dependency. + vis := func(a *Rule) *GenericVisitor { + stop := false + return NewGenericVisitor(func(x interface{}) bool { + switch x := x.(type) { + case Ref: + for _, b := range list(x) { + for node := b; node != nil; node = node.Else { + graph.addDependency(a, node) + } + } + case *Rule: + if stop { + // Do not recurse into else clauses (which will be handled + // by the outer visitor.) + return true + } + stop = true + } + return false + }) + } + + // Walk over all rules, add them to graph, and build adjencency lists. + for _, module := range modules { + WalkRules(module, func(a *Rule) bool { + graph.addNode(a) + vis(a).Walk(a) + return false + }) + } + + return graph +} + +// Dependencies returns the set of rules that x depends on. +func (g *Graph) Dependencies(x util.T) map[util.T]struct{} { + return g.adj[x] +} + +// Dependents returns the set of rules that depend on x. +func (g *Graph) Dependents(x util.T) map[util.T]struct{} { + return g.radj[x] +} + +// Sort returns a slice of rules sorted by dependencies. If a cycle is found, +// ok is set to false. +func (g *Graph) Sort() (sorted []util.T, ok bool) { + if g.sorted != nil { + return g.sorted, true + } + + sorter := &graphSort{ + sorted: make([]util.T, 0, len(g.nodes)), + deps: g.Dependencies, + marked: map[util.T]struct{}{}, + temp: map[util.T]struct{}{}, + } + + for node := range g.nodes { + if !sorter.Visit(node) { + return nil, false + } + } + + g.sorted = sorter.sorted + return g.sorted, true +} + +func (g *Graph) addDependency(u util.T, v util.T) { + + if _, ok := g.nodes[u]; !ok { + g.addNode(u) + } + + if _, ok := g.nodes[v]; !ok { + g.addNode(v) + } + + edges, ok := g.adj[u] + if !ok { + edges = map[util.T]struct{}{} + g.adj[u] = edges + } + + edges[v] = struct{}{} + + edges, ok = g.radj[v] + if !ok { + edges = map[util.T]struct{}{} + g.radj[v] = edges + } + + edges[u] = struct{}{} +} + +func (g *Graph) addNode(n util.T) { + g.nodes[n] = struct{}{} +} + +type graphSort struct { + sorted []util.T + deps func(util.T) map[util.T]struct{} + marked map[util.T]struct{} + temp map[util.T]struct{} +} + +func (sort *graphSort) Marked(node util.T) bool { + _, marked := sort.marked[node] + return marked +} + +func (sort *graphSort) Visit(node util.T) (ok bool) { + if _, ok := sort.temp[node]; ok { + return false + } + if sort.Marked(node) { + return true + } + sort.temp[node] = struct{}{} + for other := range sort.deps(node) { + if !sort.Visit(other) { + return false + } + } + sort.marked[node] = struct{}{} + delete(sort.temp, node) + sort.sorted = append(sort.sorted, node) + return true +} + +// GraphTraversal is a Traversal that understands the dependency graph +type GraphTraversal struct { + graph *Graph + visited map[util.T]struct{} +} + +// NewGraphTraversal returns a Traversal for the dependency graph +func NewGraphTraversal(graph *Graph) *GraphTraversal { + return &GraphTraversal{ + graph: graph, + visited: map[util.T]struct{}{}, + } +} + +// Edges lists all dependency connections for a given node +func (g *GraphTraversal) Edges(x util.T) []util.T { + r := []util.T{} + for v := range g.graph.Dependencies(x) { + r = append(r, v) + } + return r +} + +// Visited returns whether a node has been visited, setting a node to visited if not +func (g *GraphTraversal) Visited(u util.T) bool { + _, ok := g.visited[u] + g.visited[u] = struct{}{} + return ok +} + +type unsafePair struct { + Expr *Expr + Vars VarSet +} + +type unsafeVarLoc struct { + Var Var + Loc *Location +} + +type unsafeVars map[*Expr]VarSet + +func (vs unsafeVars) Add(e *Expr, v Var) { + if u, ok := vs[e]; ok { + u[v] = struct{}{} + } else { + vs[e] = VarSet{v: struct{}{}} + } +} + +func (vs unsafeVars) Set(e *Expr, s VarSet) { + vs[e] = s +} + +func (vs unsafeVars) Update(o unsafeVars) { + for k, v := range o { + if _, ok := vs[k]; !ok { + vs[k] = VarSet{} + } + vs[k].Update(v) + } +} + +func (vs unsafeVars) Vars() (result []unsafeVarLoc) { + + locs := map[Var]*Location{} + + // If var appears in multiple sets then pick first by location. + for expr, vars := range vs { + for v := range vars { + if locs[v].Compare(expr.Location) > 0 { + locs[v] = expr.Location + } + } + } + + for v, loc := range locs { + result = append(result, unsafeVarLoc{ + Var: v, + Loc: loc, + }) + } + + sort.Slice(result, func(i, j int) bool { + return result[i].Loc.Compare(result[j].Loc) < 0 + }) + + return result +} + +func (vs unsafeVars) Slice() (result []unsafePair) { + for expr, vs := range vs { + result = append(result, unsafePair{ + Expr: expr, + Vars: vs, + }) + } + return +} + +// reorderBodyForSafety returns a copy of the body ordered such that +// left to right evaluation of the body will not encounter unbound variables +// in input positions or negated expressions. +// +// Expressions are added to the re-ordered body as soon as they are considered +// safe. If multiple expressions become safe in the same pass, they are added +// in their original order. This results in minimal re-ordering of the body. +// +// If the body cannot be reordered to ensure safety, the second return value +// contains a mapping of expressions to unsafe variables in those expressions. +func reorderBodyForSafety(builtins map[string]*Builtin, arity func(Ref) int, globals VarSet, body Body) (Body, unsafeVars) { + + body, unsafe := reorderBodyForClosures(arity, globals, body) + if len(unsafe) != 0 { + return nil, unsafe + } + + reordered := Body{} + safe := VarSet{} + + for _, e := range body { + for v := range e.Vars(SafetyCheckVisitorParams) { + if globals.Contains(v) { + safe.Add(v) + } else { + unsafe.Add(e, v) + } + } + } + + for { + n := len(reordered) + + for _, e := range body { + if reordered.Contains(e) { + continue + } + + safe.Update(outputVarsForExpr(e, arity, safe)) + + for v := range unsafe[e] { + if safe.Contains(v) { + delete(unsafe[e], v) + } + } + + if len(unsafe[e]) == 0 { + delete(unsafe, e) + reordered.Append(e) + } + } + + if len(reordered) == n { + break + } + } + + // Recursively visit closures and perform the safety checks on them. + // Update the globals at each expression to include the variables that could + // be closed over. + g := globals.Copy() + for i, e := range reordered { + if i > 0 { + g.Update(reordered[i-1].Vars(SafetyCheckVisitorParams)) + } + xform := &bodySafetyTransformer{ + builtins: builtins, + arity: arity, + current: e, + globals: g, + unsafe: unsafe, + } + NewGenericVisitor(xform.Visit).Walk(e) + } + + return reordered, unsafe +} + +type bodySafetyTransformer struct { + builtins map[string]*Builtin + arity func(Ref) int + current *Expr + globals VarSet + unsafe unsafeVars +} + +func (xform *bodySafetyTransformer) Visit(x interface{}) bool { + if term, ok := x.(*Term); ok { + switch x := term.Value.(type) { + case *object: + cpy, _ := x.Map(func(k, v *Term) (*Term, *Term, error) { + kcpy := k.Copy() + NewGenericVisitor(xform.Visit).Walk(kcpy) + vcpy := v.Copy() + NewGenericVisitor(xform.Visit).Walk(vcpy) + return kcpy, vcpy, nil + }) + term.Value = cpy + return true + case *set: + cpy, _ := x.Map(func(v *Term) (*Term, error) { + vcpy := v.Copy() + NewGenericVisitor(xform.Visit).Walk(vcpy) + return vcpy, nil + }) + term.Value = cpy + return true + case *ArrayComprehension: + xform.reorderArrayComprehensionSafety(x) + return true + case *ObjectComprehension: + xform.reorderObjectComprehensionSafety(x) + return true + case *SetComprehension: + xform.reorderSetComprehensionSafety(x) + return true + } + } + return false +} + +func (xform *bodySafetyTransformer) reorderComprehensionSafety(tv VarSet, body Body) Body { + bv := body.Vars(SafetyCheckVisitorParams) + bv.Update(xform.globals) + uv := tv.Diff(bv) + for v := range uv { + xform.unsafe.Add(xform.current, v) + } + + r, u := reorderBodyForSafety(xform.builtins, xform.arity, xform.globals, body) + if len(u) == 0 { + return r + } + + xform.unsafe.Update(u) + return body +} + +func (xform *bodySafetyTransformer) reorderArrayComprehensionSafety(ac *ArrayComprehension) { + ac.Body = xform.reorderComprehensionSafety(ac.Term.Vars(), ac.Body) +} + +func (xform *bodySafetyTransformer) reorderObjectComprehensionSafety(oc *ObjectComprehension) { + tv := oc.Key.Vars() + tv.Update(oc.Value.Vars()) + oc.Body = xform.reorderComprehensionSafety(tv, oc.Body) +} + +func (xform *bodySafetyTransformer) reorderSetComprehensionSafety(sc *SetComprehension) { + sc.Body = xform.reorderComprehensionSafety(sc.Term.Vars(), sc.Body) +} + +// reorderBodyForClosures returns a copy of the body ordered such that +// expressions (such as array comprehensions) that close over variables are ordered +// after other expressions that contain the same variable in an output position. +func reorderBodyForClosures(arity func(Ref) int, globals VarSet, body Body) (Body, unsafeVars) { + + reordered := Body{} + unsafe := unsafeVars{} + + for { + n := len(reordered) + + for _, e := range body { + if reordered.Contains(e) { + continue + } + + // Collect vars that are contained in closures within this + // expression. + vs := VarSet{} + WalkClosures(e, func(x interface{}) bool { + vis := &VarVisitor{vars: vs} + vis.Walk(x) + return true + }) + + // Compute vars that are closed over from the body but not yet + // contained in the output position of an expression in the reordered + // body. These vars are considered unsafe. + cv := vs.Intersect(body.Vars(SafetyCheckVisitorParams)).Diff(globals) + uv := cv.Diff(outputVarsForBody(reordered, arity, globals)) + + if len(uv) == 0 { + reordered = append(reordered, e) + delete(unsafe, e) + } else { + unsafe.Set(e, uv) + } + } + + if len(reordered) == n { + break + } + } + + return reordered, unsafe +} + +// OutputVarsFromBody returns all variables which are the "output" for +// the given body. For safety checks this means that they would be +// made safe by the body. +func OutputVarsFromBody(c *Compiler, body Body, safe VarSet) VarSet { + return outputVarsForBody(body, c.GetArity, safe) +} + +func outputVarsForBody(body Body, arity func(Ref) int, safe VarSet) VarSet { + o := safe.Copy() + for _, e := range body { + o.Update(outputVarsForExpr(e, arity, o)) + } + return o.Diff(safe) +} + +// OutputVarsFromExpr returns all variables which are the "output" for +// the given expression. For safety checks this means that they would be +// made safe by the expr. +func OutputVarsFromExpr(c *Compiler, expr *Expr, safe VarSet) VarSet { + return outputVarsForExpr(expr, c.GetArity, safe) +} + +func outputVarsForExpr(expr *Expr, arity func(Ref) int, safe VarSet) VarSet { + + // Negated expressions must be safe. + if expr.Negated { + return VarSet{} + } + + // With modifier inputs must be safe. + for _, with := range expr.With { + unsafe := false + WalkVars(with, func(v Var) bool { + if !safe.Contains(v) { + unsafe = true + return true + } + return false + }) + if unsafe { + return VarSet{} + } + } + + switch terms := expr.Terms.(type) { + case *Term: + return outputVarsForTerms(expr, safe) + case []*Term: + if expr.IsEquality() { + return outputVarsForExprEq(expr, safe) + } + + operator, ok := terms[0].Value.(Ref) + if !ok { + return VarSet{} + } + + ar := arity(operator) + if ar < 0 { + return VarSet{} + } + + return outputVarsForExprCall(expr, ar, safe, terms) + default: + panic("illegal expression") + } +} + +func outputVarsForExprEq(expr *Expr, safe VarSet) VarSet { + + if !validEqAssignArgCount(expr) { + return safe + } + + output := outputVarsForTerms(expr, safe) + output.Update(safe) + output.Update(Unify(output, expr.Operand(0), expr.Operand(1))) + + return output.Diff(safe) +} + +func outputVarsForExprCall(expr *Expr, arity int, safe VarSet, terms []*Term) VarSet { + + output := outputVarsForTerms(expr, safe) + + numInputTerms := arity + 1 + if numInputTerms >= len(terms) { + return output + } + + vis := NewVarVisitor().WithParams(VarVisitorParams{ + SkipClosures: true, + SkipSets: true, + SkipObjectKeys: true, + SkipRefHead: true, + }) + + vis.Walk(Args(terms[:numInputTerms])) + unsafe := vis.Vars().Diff(output).Diff(safe) + + if len(unsafe) > 0 { + return VarSet{} + } + + vis = NewVarVisitor().WithParams(VarVisitorParams{ + SkipRefHead: true, + SkipSets: true, + SkipObjectKeys: true, + SkipClosures: true, + }) + + vis.Walk(Args(terms[numInputTerms:])) + output.Update(vis.vars) + return output +} + +func outputVarsForTerms(expr *Expr, safe VarSet) VarSet { + output := VarSet{} + WalkTerms(expr, func(x *Term) bool { + switch r := x.Value.(type) { + case *SetComprehension, *ArrayComprehension, *ObjectComprehension: + return true + case Ref: + if !isRefSafe(r, safe) { + return true + } + output.Update(r.OutputVars()) + return false + } + return false + }) + return output +} + +type equalityFactory struct { + gen *localVarGenerator +} + +func newEqualityFactory(gen *localVarGenerator) *equalityFactory { + return &equalityFactory{gen} +} + +func (f *equalityFactory) Generate(other *Term) *Expr { + term := NewTerm(f.gen.Generate()).SetLocation(other.Location) + expr := Equality.Expr(term, other) + expr.Generated = true + expr.Location = other.Location + return expr +} + +type localVarGenerator struct { + exclude VarSet + suffix string + next int +} + +func newLocalVarGeneratorForModuleSet(sorted []string, modules map[string]*Module) *localVarGenerator { + exclude := NewVarSet() + vis := &VarVisitor{vars: exclude} + for _, key := range sorted { + vis.Walk(modules[key]) + } + return &localVarGenerator{exclude: exclude, next: 0} +} + +func newLocalVarGenerator(suffix string, node interface{}) *localVarGenerator { + exclude := NewVarSet() + vis := &VarVisitor{vars: exclude} + vis.Walk(node) + return &localVarGenerator{exclude: exclude, suffix: suffix, next: 0} +} + +func (l *localVarGenerator) Generate() Var { + for { + result := Var("__local" + l.suffix + strconv.Itoa(l.next) + "__") + l.next++ + if !l.exclude.Contains(result) { + return result + } + } +} + +func getGlobals(pkg *Package, rules []Var, imports []*Import) map[Var]Ref { + + globals := map[Var]Ref{} + + // Populate globals with exports within the package. + for _, v := range rules { + global := append(Ref{}, pkg.Path...) + global = append(global, &Term{Value: String(v)}) + globals[v] = global + } + + // Populate globals with imports. + for _, i := range imports { + if len(i.Alias) > 0 { + path := i.Path.Value.(Ref) + globals[i.Alias] = path + } else { + path := i.Path.Value.(Ref) + if len(path) == 1 { + globals[path[0].Value.(Var)] = path + } else { + v := path[len(path)-1].Value.(String) + globals[Var(v)] = path + } + } + } + + return globals +} + +func requiresEval(x *Term) bool { + if x == nil { + return false + } + return ContainsRefs(x) || ContainsComprehensions(x) +} + +func resolveRef(globals map[Var]Ref, ignore *declaredVarStack, ref Ref) Ref { + + r := Ref{} + for i, x := range ref { + switch v := x.Value.(type) { + case Var: + if g, ok := globals[v]; ok && !ignore.Contains(v) { + cpy := g.Copy() + for i := range cpy { + cpy[i].SetLocation(x.Location) + } + if i == 0 { + r = cpy + } else { + r = append(r, NewTerm(cpy).SetLocation(x.Location)) + } + } else { + r = append(r, x) + } + case Ref, *Array, Object, Set, *ArrayComprehension, *SetComprehension, *ObjectComprehension, Call: + r = append(r, resolveRefsInTerm(globals, ignore, x)) + default: + r = append(r, x) + } + } + + return r +} + +func resolveRefsInRule(globals map[Var]Ref, rule *Rule) error { + ignore := &declaredVarStack{} + + vars := NewVarSet() + var vis *GenericVisitor + var err error + + // Walk args to collect vars and transform body so that callers can shadow + // root documents. + vis = NewGenericVisitor(func(x interface{}) bool { + if err != nil { + return true + } + switch x := x.(type) { + case Var: + vars.Add(x) + + // Object keys cannot be pattern matched so only walk values. + case *object: + x.Foreach(func(k, v *Term) { + vis.Walk(v) + }) + + // Skip terms that could contain vars that cannot be pattern matched. + case Set, *ArrayComprehension, *SetComprehension, *ObjectComprehension, Call: + return true + + case *Term: + if _, ok := x.Value.(Ref); ok { + if RootDocumentRefs.Contains(x) { + // We could support args named input, data, etc. however + // this would require rewriting terms in the head and body. + // Preventing root document shadowing is simpler, and + // arguably, will prevent confusing names from being used. + err = fmt.Errorf("args must not shadow %v (use a different variable name)", x) + return true + } + } + } + return false + }) + + vis.Walk(rule.Head.Args) + + if err != nil { + return err + } + + ignore.Push(vars) + ignore.Push(declaredVars(rule.Body)) + + if rule.Head.Key != nil { + rule.Head.Key = resolveRefsInTerm(globals, ignore, rule.Head.Key) + } + + if rule.Head.Value != nil { + rule.Head.Value = resolveRefsInTerm(globals, ignore, rule.Head.Value) + } + + rule.Body = resolveRefsInBody(globals, ignore, rule.Body) + return nil +} + +func resolveRefsInBody(globals map[Var]Ref, ignore *declaredVarStack, body Body) Body { + r := Body{} + for _, expr := range body { + r = append(r, resolveRefsInExpr(globals, ignore, expr)) + } + return r +} + +func resolveRefsInExpr(globals map[Var]Ref, ignore *declaredVarStack, expr *Expr) *Expr { + cpy := *expr + switch ts := expr.Terms.(type) { + case *Term: + cpy.Terms = resolveRefsInTerm(globals, ignore, ts) + case []*Term: + buf := make([]*Term, len(ts)) + for i := 0; i < len(ts); i++ { + buf[i] = resolveRefsInTerm(globals, ignore, ts[i]) + } + cpy.Terms = buf + case *SomeDecl: + if val, ok := ts.Symbols[0].Value.(Call); ok { + cpy.Terms = &SomeDecl{Symbols: []*Term{CallTerm(resolveRefsInTermSlice(globals, ignore, val)...)}} + } + } + for _, w := range cpy.With { + w.Target = resolveRefsInTerm(globals, ignore, w.Target) + w.Value = resolveRefsInTerm(globals, ignore, w.Value) + } + return &cpy +} + +func resolveRefsInTerm(globals map[Var]Ref, ignore *declaredVarStack, term *Term) *Term { + switch v := term.Value.(type) { + case Var: + if g, ok := globals[v]; ok && !ignore.Contains(v) { + cpy := g.Copy() + for i := range cpy { + cpy[i].SetLocation(term.Location) + } + return NewTerm(cpy).SetLocation(term.Location) + } + return term + case Ref: + fqn := resolveRef(globals, ignore, v) + cpy := *term + cpy.Value = fqn + return &cpy + case *object: + cpy := *term + cpy.Value, _ = v.Map(func(k, v *Term) (*Term, *Term, error) { + k = resolveRefsInTerm(globals, ignore, k) + v = resolveRefsInTerm(globals, ignore, v) + return k, v, nil + }) + return &cpy + case *Array: + cpy := *term + cpy.Value = NewArray(resolveRefsInTermArray(globals, ignore, v)...) + return &cpy + case Call: + cpy := *term + cpy.Value = Call(resolveRefsInTermSlice(globals, ignore, v)) + return &cpy + case Set: + s, _ := v.Map(func(e *Term) (*Term, error) { + return resolveRefsInTerm(globals, ignore, e), nil + }) + cpy := *term + cpy.Value = s + return &cpy + case *ArrayComprehension: + ac := &ArrayComprehension{} + ignore.Push(declaredVars(v.Body)) + ac.Term = resolveRefsInTerm(globals, ignore, v.Term) + ac.Body = resolveRefsInBody(globals, ignore, v.Body) + cpy := *term + cpy.Value = ac + ignore.Pop() + return &cpy + case *ObjectComprehension: + oc := &ObjectComprehension{} + ignore.Push(declaredVars(v.Body)) + oc.Key = resolveRefsInTerm(globals, ignore, v.Key) + oc.Value = resolveRefsInTerm(globals, ignore, v.Value) + oc.Body = resolveRefsInBody(globals, ignore, v.Body) + cpy := *term + cpy.Value = oc + ignore.Pop() + return &cpy + case *SetComprehension: + sc := &SetComprehension{} + ignore.Push(declaredVars(v.Body)) + sc.Term = resolveRefsInTerm(globals, ignore, v.Term) + sc.Body = resolveRefsInBody(globals, ignore, v.Body) + cpy := *term + cpy.Value = sc + ignore.Pop() + return &cpy + default: + return term + } +} + +func resolveRefsInTermArray(globals map[Var]Ref, ignore *declaredVarStack, terms *Array) []*Term { + cpy := make([]*Term, terms.Len()) + for i := 0; i < terms.Len(); i++ { + cpy[i] = resolveRefsInTerm(globals, ignore, terms.Elem(i)) + } + return cpy +} + +func resolveRefsInTermSlice(globals map[Var]Ref, ignore *declaredVarStack, terms []*Term) []*Term { + cpy := make([]*Term, len(terms)) + for i := 0; i < len(terms); i++ { + cpy[i] = resolveRefsInTerm(globals, ignore, terms[i]) + } + return cpy +} + +type declaredVarStack []VarSet + +func (s declaredVarStack) Contains(v Var) bool { + for i := len(s) - 1; i >= 0; i-- { + if _, ok := s[i][v]; ok { + return ok + } + } + return false +} + +func (s declaredVarStack) Add(v Var) { + s[len(s)-1].Add(v) +} + +func (s *declaredVarStack) Push(vs VarSet) { + *s = append(*s, vs) +} + +func (s *declaredVarStack) Pop() { + curr := *s + *s = curr[:len(curr)-1] +} + +func declaredVars(x interface{}) VarSet { + vars := NewVarSet() + vis := NewGenericVisitor(func(x interface{}) bool { + switch x := x.(type) { + case *Expr: + if x.IsAssignment() && validEqAssignArgCount(x) { + WalkVars(x.Operand(0), func(v Var) bool { + vars.Add(v) + return false + }) + } else if decl, ok := x.Terms.(*SomeDecl); ok { + for i := range decl.Symbols { + switch val := decl.Symbols[i].Value.(type) { + case Var: + vars.Add(val) + case Call: + args := val[1:] + if len(args) == 3 { // some x, y in xs + WalkVars(args[1], func(v Var) bool { + vars.Add(v) + return false + }) + } + // some x in xs + WalkVars(args[0], func(v Var) bool { + vars.Add(v) + return false + }) + } + } + } + case *ArrayComprehension, *SetComprehension, *ObjectComprehension: + return true + } + return false + }) + vis.Walk(x) + return vars +} + +// rewriteComprehensionTerms will rewrite comprehensions so that the term part +// is bound to a variable in the body. This allows any type of term to be used +// in the term part (even if the term requires evaluation.) +// +// For instance, given the following comprehension: +// +// [x[0] | x = y[_]; y = [1,2,3]] +// +// The comprehension would be rewritten as: +// +// [__local0__ | x = y[_]; y = [1,2,3]; __local0__ = x[0]] +func rewriteComprehensionTerms(f *equalityFactory, node interface{}) (interface{}, error) { + return TransformComprehensions(node, func(x interface{}) (Value, error) { + switch x := x.(type) { + case *ArrayComprehension: + if requiresEval(x.Term) { + expr := f.Generate(x.Term) + x.Term = expr.Operand(0) + x.Body.Append(expr) + } + return x, nil + case *SetComprehension: + if requiresEval(x.Term) { + expr := f.Generate(x.Term) + x.Term = expr.Operand(0) + x.Body.Append(expr) + } + return x, nil + case *ObjectComprehension: + if requiresEval(x.Key) { + expr := f.Generate(x.Key) + x.Key = expr.Operand(0) + x.Body.Append(expr) + } + if requiresEval(x.Value) { + expr := f.Generate(x.Value) + x.Value = expr.Operand(0) + x.Body.Append(expr) + } + return x, nil + } + panic("illegal type") + }) +} + +// rewriteEquals will rewrite exprs under x as unification calls instead of == +// calls. For example: +// +// data.foo == data.bar is rewritten as data.foo = data.bar +// +// This stage should only run the safety check (since == is a built-in with no +// outputs, so the inputs must not be marked as safe.) +// +// This stage is not executed by the query compiler by default because when +// callers specify == instead of = they expect to receive a true/false/undefined +// result back whereas with = the result is only ever true/undefined. For +// partial evaluation cases we do want to rewrite == to = to simplify the +// result. +func rewriteEquals(x interface{}) { + doubleEq := Equal.Ref() + unifyOp := Equality.Ref() + t := NewGenericTransformer(func(x interface{}) (interface{}, error) { + if x, ok := x.(*Expr); ok && x.IsCall() { + operator := x.Operator() + if operator.Equal(doubleEq) && len(x.Operands()) == 2 { + x.SetOperator(NewTerm(unifyOp)) + } + } + return x, nil + }) + _, _ = Transform(t, x) // ignore error +} + +// rewriteDynamics will rewrite the body so that dynamic terms (i.e., refs and +// comprehensions) are bound to vars earlier in the query. This translation +// results in eager evaluation. +// +// For instance, given the following query: +// +// foo(data.bar) = 1 +// +// The rewritten version will be: +// +// __local0__ = data.bar; foo(__local0__) = 1 +func rewriteDynamics(f *equalityFactory, body Body) Body { + result := make(Body, 0, len(body)) + for _, expr := range body { + if expr.IsEquality() { + result = rewriteDynamicsEqExpr(f, expr, result) + } else if expr.IsCall() { + result = rewriteDynamicsCallExpr(f, expr, result) + } else { + result = rewriteDynamicsTermExpr(f, expr, result) + } + } + return result +} + +func appendExpr(body Body, expr *Expr) Body { + body.Append(expr) + return body +} + +func rewriteDynamicsEqExpr(f *equalityFactory, expr *Expr, result Body) Body { + if !validEqAssignArgCount(expr) { + return appendExpr(result, expr) + } + terms := expr.Terms.([]*Term) + result, terms[1] = rewriteDynamicsInTerm(expr, f, terms[1], result) + result, terms[2] = rewriteDynamicsInTerm(expr, f, terms[2], result) + return appendExpr(result, expr) +} + +func rewriteDynamicsCallExpr(f *equalityFactory, expr *Expr, result Body) Body { + terms := expr.Terms.([]*Term) + for i := 1; i < len(terms); i++ { + result, terms[i] = rewriteDynamicsOne(expr, f, terms[i], result) + } + return appendExpr(result, expr) +} + +func rewriteDynamicsTermExpr(f *equalityFactory, expr *Expr, result Body) Body { + term := expr.Terms.(*Term) + result, expr.Terms = rewriteDynamicsInTerm(expr, f, term, result) + return appendExpr(result, expr) +} + +func rewriteDynamicsInTerm(original *Expr, f *equalityFactory, term *Term, result Body) (Body, *Term) { + switch v := term.Value.(type) { + case Ref: + for i := 1; i < len(v); i++ { + result, v[i] = rewriteDynamicsOne(original, f, v[i], result) + } + case *ArrayComprehension: + v.Body = rewriteDynamics(f, v.Body) + case *SetComprehension: + v.Body = rewriteDynamics(f, v.Body) + case *ObjectComprehension: + v.Body = rewriteDynamics(f, v.Body) + default: + result, term = rewriteDynamicsOne(original, f, term, result) + } + return result, term +} + +func rewriteDynamicsOne(original *Expr, f *equalityFactory, term *Term, result Body) (Body, *Term) { + switch v := term.Value.(type) { + case Ref: + for i := 1; i < len(v); i++ { + result, v[i] = rewriteDynamicsOne(original, f, v[i], result) + } + generated := f.Generate(term) + generated.With = original.With + result.Append(generated) + return result, result[len(result)-1].Operand(0) + case *Array: + for i := 0; i < v.Len(); i++ { + var t *Term + result, t = rewriteDynamicsOne(original, f, v.Elem(i), result) + v.set(i, t) + } + return result, term + case *object: + cpy := NewObject() + v.Foreach(func(key, value *Term) { + result, key = rewriteDynamicsOne(original, f, key, result) + result, value = rewriteDynamicsOne(original, f, value, result) + cpy.Insert(key, value) + }) + return result, NewTerm(cpy).SetLocation(term.Location) + case Set: + cpy := NewSet() + for _, term := range v.Slice() { + var rw *Term + result, rw = rewriteDynamicsOne(original, f, term, result) + cpy.Add(rw) + } + return result, NewTerm(cpy).SetLocation(term.Location) + case *ArrayComprehension: + var extra *Expr + v.Body, extra = rewriteDynamicsComprehensionBody(original, f, v.Body, term) + result.Append(extra) + return result, result[len(result)-1].Operand(0) + case *SetComprehension: + var extra *Expr + v.Body, extra = rewriteDynamicsComprehensionBody(original, f, v.Body, term) + result.Append(extra) + return result, result[len(result)-1].Operand(0) + case *ObjectComprehension: + var extra *Expr + v.Body, extra = rewriteDynamicsComprehensionBody(original, f, v.Body, term) + result.Append(extra) + return result, result[len(result)-1].Operand(0) + } + return result, term +} + +func rewriteDynamicsComprehensionBody(original *Expr, f *equalityFactory, body Body, term *Term) (Body, *Expr) { + body = rewriteDynamics(f, body) + generated := f.Generate(term) + generated.With = original.With + return body, generated +} + +func rewriteExprTermsInHead(gen *localVarGenerator, rule *Rule) { + for i := range rule.Head.Args { + support, output := expandExprTerm(gen, rule.Head.Args[i]) + for j := range support { + rule.Body.Append(support[j]) + } + rule.Head.Args[i] = output + } + if rule.Head.Key != nil { + support, output := expandExprTerm(gen, rule.Head.Key) + for i := range support { + rule.Body.Append(support[i]) + } + rule.Head.Key = output + } + if rule.Head.Value != nil { + support, output := expandExprTerm(gen, rule.Head.Value) + for i := range support { + rule.Body.Append(support[i]) + } + rule.Head.Value = output + } +} + +func rewriteExprTermsInBody(gen *localVarGenerator, body Body) Body { + cpy := make(Body, 0, len(body)) + for i := 0; i < len(body); i++ { + for _, expr := range expandExpr(gen, body[i]) { + cpy.Append(expr) + } + } + return cpy +} + +func expandExpr(gen *localVarGenerator, expr *Expr) (result []*Expr) { + for i := range expr.With { + extras, value := expandExprTerm(gen, expr.With[i].Value) + expr.With[i].Value = value + result = append(result, extras...) + } + switch terms := expr.Terms.(type) { + case *Term: + extras, term := expandExprTerm(gen, terms) + if len(expr.With) > 0 { + for i := range extras { + extras[i].With = expr.With + } + } + result = append(result, extras...) + expr.Terms = term + result = append(result, expr) + case []*Term: + for i := 1; i < len(terms); i++ { + var extras []*Expr + extras, terms[i] = expandExprTerm(gen, terms[i]) + if len(expr.With) > 0 { + for i := range extras { + extras[i].With = expr.With + } + } + result = append(result, extras...) + } + result = append(result, expr) + } + return +} + +func expandExprTerm(gen *localVarGenerator, term *Term) (support []*Expr, output *Term) { + output = term + switch v := term.Value.(type) { + case Call: + for i := 1; i < len(v); i++ { + var extras []*Expr + extras, v[i] = expandExprTerm(gen, v[i]) + support = append(support, extras...) + } + output = NewTerm(gen.Generate()).SetLocation(term.Location) + expr := v.MakeExpr(output).SetLocation(term.Location) + expr.Generated = true + support = append(support, expr) + case Ref: + support = expandExprRef(gen, v) + case *Array: + support = expandExprTermArray(gen, v) + case *object: + cpy, _ := v.Map(func(k, v *Term) (*Term, *Term, error) { + extras1, expandedKey := expandExprTerm(gen, k) + extras2, expandedValue := expandExprTerm(gen, v) + support = append(support, extras1...) + support = append(support, extras2...) + return expandedKey, expandedValue, nil + }) + output = NewTerm(cpy).SetLocation(term.Location) + case Set: + cpy, _ := v.Map(func(x *Term) (*Term, error) { + extras, expanded := expandExprTerm(gen, x) + support = append(support, extras...) + return expanded, nil + }) + output = NewTerm(cpy).SetLocation(term.Location) + case *ArrayComprehension: + support, term := expandExprTerm(gen, v.Term) + for i := range support { + v.Body.Append(support[i]) + } + v.Term = term + v.Body = rewriteExprTermsInBody(gen, v.Body) + case *SetComprehension: + support, term := expandExprTerm(gen, v.Term) + for i := range support { + v.Body.Append(support[i]) + } + v.Term = term + v.Body = rewriteExprTermsInBody(gen, v.Body) + case *ObjectComprehension: + support, key := expandExprTerm(gen, v.Key) + for i := range support { + v.Body.Append(support[i]) + } + v.Key = key + support, value := expandExprTerm(gen, v.Value) + for i := range support { + v.Body.Append(support[i]) + } + v.Value = value + v.Body = rewriteExprTermsInBody(gen, v.Body) + } + return +} + +func expandExprRef(gen *localVarGenerator, v []*Term) (support []*Expr) { + // Start by calling a normal expandExprTerm on all terms. + support = expandExprTermSlice(gen, v) + + // Rewrite references in order to support indirect references. We rewrite + // e.g. + // + // [1, 2, 3][i] + // + // to + // + // __local_var = [1, 2, 3] + // __local_var[i] + // + // to support these. This only impacts the reference subject, i.e. the + // first item in the slice. + var subject = v[0] + switch subject.Value.(type) { + case *Array, Object, Set, *ArrayComprehension, *SetComprehension, *ObjectComprehension, Call: + f := newEqualityFactory(gen) + assignToLocal := f.Generate(subject) + support = append(support, assignToLocal) + v[0] = assignToLocal.Operand(0) + } + return +} + +func expandExprTermArray(gen *localVarGenerator, arr *Array) (support []*Expr) { + for i := 0; i < arr.Len(); i++ { + extras, v := expandExprTerm(gen, arr.Elem(i)) + arr.set(i, v) + support = append(support, extras...) + } + return +} + +func expandExprTermSlice(gen *localVarGenerator, v []*Term) (support []*Expr) { + for i := 0; i < len(v); i++ { + var extras []*Expr + extras, v[i] = expandExprTerm(gen, v[i]) + support = append(support, extras...) + } + return +} + +type localDeclaredVars struct { + vars []*declaredVarSet + + // rewritten contains a mapping of *all* user-defined variables + // that have been rewritten whereas vars contains the state + // from the current query (not not any nested queries, and all + // vars seen). + rewritten map[Var]Var +} + +type varOccurrence int + +const ( + newVar varOccurrence = iota + argVar + seenVar + assignedVar + declaredVar +) + +type declaredVarSet struct { + vs map[Var]Var + reverse map[Var]Var + occurrence map[Var]varOccurrence +} + +func newDeclaredVarSet() *declaredVarSet { + return &declaredVarSet{ + vs: map[Var]Var{}, + reverse: map[Var]Var{}, + occurrence: map[Var]varOccurrence{}, + } +} + +func newLocalDeclaredVars() *localDeclaredVars { + return &localDeclaredVars{ + vars: []*declaredVarSet{newDeclaredVarSet()}, + rewritten: map[Var]Var{}, + } +} + +func (s *localDeclaredVars) Push() { + s.vars = append(s.vars, newDeclaredVarSet()) +} + +func (s *localDeclaredVars) Pop() *declaredVarSet { + sl := s.vars + curr := sl[len(sl)-1] + s.vars = sl[:len(sl)-1] + return curr +} + +func (s localDeclaredVars) Peek() *declaredVarSet { + return s.vars[len(s.vars)-1] +} + +func (s localDeclaredVars) Insert(x, y Var, occurrence varOccurrence) { + elem := s.vars[len(s.vars)-1] + elem.vs[x] = y + elem.reverse[y] = x + elem.occurrence[x] = occurrence + + // If the variable has been rewritten (where x != y, with y being + // the generated value), store it in the map of rewritten vars. + // Assume that the generated values are unique for the compilation. + if !x.Equal(y) { + s.rewritten[y] = x + } +} + +func (s localDeclaredVars) Declared(x Var) (y Var, ok bool) { + for i := len(s.vars) - 1; i >= 0; i-- { + if y, ok = s.vars[i].vs[x]; ok { + return + } + } + return +} + +// Occurrence returns a flag that indicates whether x has occurred in the +// current scope. +func (s localDeclaredVars) Occurrence(x Var) varOccurrence { + return s.vars[len(s.vars)-1].occurrence[x] +} + +// GlobalOccurrence returns a flag that indicates whether x has occurred in the +// global scope. +func (s localDeclaredVars) GlobalOccurrence(x Var) (varOccurrence, bool) { + for i := len(s.vars) - 1; i >= 0; i-- { + if occ, ok := s.vars[i].occurrence[x]; ok { + return occ, true + } + } + return newVar, false +} + +// rewriteLocalVars rewrites bodies to remove assignment/declaration +// expressions. For example: +// +// a := 1; p[a] +// +// Is rewritten to: +// +// __local0__ = 1; p[__local0__] +// +// During rewriting, assignees are validated to prevent use before declaration. +func rewriteLocalVars(g *localVarGenerator, stack *localDeclaredVars, used VarSet, body Body) (Body, map[Var]Var, Errors) { + var errs Errors + body, errs = rewriteDeclaredVarsInBody(g, stack, used, body, errs) + return body, stack.Pop().vs, errs +} + +func rewriteDeclaredVarsInBody(g *localVarGenerator, stack *localDeclaredVars, used VarSet, body Body, errs Errors) (Body, Errors) { + + var cpy Body + + for i := range body { + var expr *Expr + if body[i].IsAssignment() { + expr, errs = rewriteDeclaredAssignment(g, stack, body[i], errs) + } else if _, ok := body[i].Terms.(*SomeDecl); ok { + expr, errs = rewriteSomeDeclStatement(g, stack, body[i], errs) + } else { + expr, errs = rewriteDeclaredVarsInExpr(g, stack, body[i], errs) + } + if expr != nil { + cpy.Append(expr) + } + } + + // If the body only contained a var statement it will be empty at this + // point. Append true to the body to ensure that it's non-empty (zero length + // bodies are not supported.) + if len(cpy) == 0 { + cpy.Append(NewExpr(BooleanTerm(true))) + } + + return cpy, checkUnusedDeclaredVars(body[0].Loc(), stack, used, cpy, errs) +} + +func checkUnusedDeclaredVars(loc *Location, stack *localDeclaredVars, used VarSet, cpy Body, errs Errors) Errors { + + // NOTE(tsandall): Do not generate more errors if there are existing + // declaration errors. + if len(errs) > 0 { + return errs + } + + dvs := stack.Peek() + declared := NewVarSet() + + for v, occ := range dvs.occurrence { + if occ == declaredVar { + declared.Add(dvs.vs[v]) + } + } + + bodyvars := cpy.Vars(VarVisitorParams{}) + + for v := range used { + if gv, ok := stack.Declared(v); ok { + bodyvars.Add(gv) + } else { + bodyvars.Add(v) + } + } + + unused := declared.Diff(bodyvars).Diff(used) + + for _, gv := range unused.Sorted() { + errs = append(errs, NewError(CompileErr, loc, "declared var %v unused", dvs.reverse[gv])) + } + + return errs +} + +func rewriteSomeDeclStatement(g *localVarGenerator, stack *localDeclaredVars, expr *Expr, errs Errors) (*Expr, Errors) { + e := expr.Copy() + decl := e.Terms.(*SomeDecl) + for i := range decl.Symbols { + switch v := decl.Symbols[i].Value.(type) { + case Var: + if _, err := rewriteDeclaredVar(g, stack, v, declaredVar); err != nil { + return nil, append(errs, NewError(CompileErr, decl.Loc(), err.Error())) + } + case Call: + var key, val, container *Term + switch len(v) { + case 4: // member3 + key = v[1] + val = v[2] + container = v[3] + case 3: // member + key = NewTerm(g.Generate()) + val = v[1] + container = v[2] + } + + var rhs *Term + switch c := container.Value.(type) { + case Ref: + rhs = RefTerm(append(c, key)...) + default: + rhs = RefTerm(container, key) + } + e.Terms = []*Term{ + RefTerm(VarTerm(Equality.Name)), val, rhs, + } + + for _, v0 := range outputVarsForExprEq(e, container.Vars()).Sorted() { + if _, err := rewriteDeclaredVar(g, stack, v0, declaredVar); err != nil { + return nil, append(errs, NewError(CompileErr, decl.Loc(), err.Error())) + } + } + return rewriteDeclaredVarsInExpr(g, stack, e, errs) + } + } + return nil, errs +} + +func rewriteDeclaredVarsInExpr(g *localVarGenerator, stack *localDeclaredVars, expr *Expr, errs Errors) (*Expr, Errors) { + vis := NewGenericVisitor(func(x interface{}) bool { + var stop bool + switch x := x.(type) { + case *Term: + stop, errs = rewriteDeclaredVarsInTerm(g, stack, x, errs) + case *With: + _, errs = rewriteDeclaredVarsInTerm(g, stack, x.Value, errs) + stop = true + } + return stop + }) + vis.Walk(expr) + return expr, errs +} + +func rewriteDeclaredAssignment(g *localVarGenerator, stack *localDeclaredVars, expr *Expr, errs Errors) (*Expr, Errors) { + + if expr.Negated { + errs = append(errs, NewError(CompileErr, expr.Location, "cannot assign vars inside negated expression")) + return expr, errs + } + + numErrsBefore := len(errs) + + if !validEqAssignArgCount(expr) { + return expr, errs + } + + // Rewrite terms on right hand side capture seen vars and recursively + // process comprehensions before left hand side is processed. Also + // rewrite with modifier. + errs = rewriteDeclaredVarsInTermRecursive(g, stack, expr.Operand(1), errs) + + for _, w := range expr.With { + errs = rewriteDeclaredVarsInTermRecursive(g, stack, w.Value, errs) + } + + // Rewrite vars on left hand side with unique names. Catch redeclaration + // and invalid term types here. + var vis func(t *Term) bool + + vis = func(t *Term) bool { + switch v := t.Value.(type) { + case Var: + if gv, err := rewriteDeclaredVar(g, stack, v, assignedVar); err != nil { + errs = append(errs, NewError(CompileErr, t.Location, err.Error())) + } else { + t.Value = gv + } + return true + case *Array: + return false + case *object: + v.Foreach(func(_, v *Term) { + WalkTerms(v, vis) + }) + return true + case Ref: + if RootDocumentRefs.Contains(t) { + if gv, err := rewriteDeclaredVar(g, stack, v[0].Value.(Var), assignedVar); err != nil { + errs = append(errs, NewError(CompileErr, t.Location, err.Error())) + } else { + t.Value = gv + } + return true + } + } + errs = append(errs, NewError(CompileErr, t.Location, "cannot assign to %v", TypeName(t.Value))) + return true + } + + WalkTerms(expr.Operand(0), vis) + + if len(errs) == numErrsBefore { + loc := expr.Operator()[0].Location + expr.SetOperator(RefTerm(VarTerm(Equality.Name).SetLocation(loc)).SetLocation(loc)) + } + + return expr, errs +} + +func rewriteDeclaredVarsInTerm(g *localVarGenerator, stack *localDeclaredVars, term *Term, errs Errors) (bool, Errors) { + switch v := term.Value.(type) { + case Var: + if gv, ok := stack.Declared(v); ok { + term.Value = gv + } else if stack.Occurrence(v) == newVar { + stack.Insert(v, v, seenVar) + } + case Ref: + if RootDocumentRefs.Contains(term) { + x := v[0].Value.(Var) + if occ, ok := stack.GlobalOccurrence(x); ok && occ != seenVar { + gv, _ := stack.Declared(x) + term.Value = gv + } + + return true, errs + } + return false, errs + case *object: + cpy, _ := v.Map(func(k, v *Term) (*Term, *Term, error) { + kcpy := k.Copy() + errs = rewriteDeclaredVarsInTermRecursive(g, stack, kcpy, errs) + errs = rewriteDeclaredVarsInTermRecursive(g, stack, v, errs) + return kcpy, v, nil + }) + term.Value = cpy + case Set: + cpy, _ := v.Map(func(elem *Term) (*Term, error) { + elemcpy := elem.Copy() + errs = rewriteDeclaredVarsInTermRecursive(g, stack, elemcpy, errs) + return elemcpy, nil + }) + term.Value = cpy + case *ArrayComprehension: + errs = rewriteDeclaredVarsInArrayComprehension(g, stack, v, errs) + case *SetComprehension: + errs = rewriteDeclaredVarsInSetComprehension(g, stack, v, errs) + case *ObjectComprehension: + errs = rewriteDeclaredVarsInObjectComprehension(g, stack, v, errs) + default: + return false, errs + } + return true, errs +} + +func rewriteDeclaredVarsInTermRecursive(g *localVarGenerator, stack *localDeclaredVars, term *Term, errs Errors) Errors { + WalkNodes(term, func(n Node) bool { + var stop bool + switch n := n.(type) { + case *With: + _, errs = rewriteDeclaredVarsInTerm(g, stack, n.Value, errs) + stop = true + case *Term: + stop, errs = rewriteDeclaredVarsInTerm(g, stack, n, errs) + } + return stop + }) + return errs +} + +func rewriteDeclaredVarsInArrayComprehension(g *localVarGenerator, stack *localDeclaredVars, v *ArrayComprehension, errs Errors) Errors { + stack.Push() + v.Body, errs = rewriteDeclaredVarsInBody(g, stack, nil, v.Body, errs) + errs = rewriteDeclaredVarsInTermRecursive(g, stack, v.Term, errs) + stack.Pop() + return errs +} + +func rewriteDeclaredVarsInSetComprehension(g *localVarGenerator, stack *localDeclaredVars, v *SetComprehension, errs Errors) Errors { + stack.Push() + v.Body, errs = rewriteDeclaredVarsInBody(g, stack, nil, v.Body, errs) + errs = rewriteDeclaredVarsInTermRecursive(g, stack, v.Term, errs) + stack.Pop() + return errs +} + +func rewriteDeclaredVarsInObjectComprehension(g *localVarGenerator, stack *localDeclaredVars, v *ObjectComprehension, errs Errors) Errors { + stack.Push() + v.Body, errs = rewriteDeclaredVarsInBody(g, stack, nil, v.Body, errs) + errs = rewriteDeclaredVarsInTermRecursive(g, stack, v.Key, errs) + errs = rewriteDeclaredVarsInTermRecursive(g, stack, v.Value, errs) + stack.Pop() + return errs +} + +func rewriteDeclaredVar(g *localVarGenerator, stack *localDeclaredVars, v Var, occ varOccurrence) (gv Var, err error) { + switch stack.Occurrence(v) { + case seenVar: + return gv, fmt.Errorf("var %v referenced above", v) + case assignedVar: + return gv, fmt.Errorf("var %v assigned above", v) + case declaredVar: + return gv, fmt.Errorf("var %v declared above", v) + case argVar: + return gv, fmt.Errorf("arg %v redeclared", v) + } + gv = g.Generate() + stack.Insert(v, gv, occ) + return +} + +// rewriteWithModifiersInBody will rewrite the body so that with modifiers do +// not contain terms that require evaluation as values. If this function +// encounters an invalid with modifier target then it will raise an error. +func rewriteWithModifiersInBody(c *Compiler, f *equalityFactory, body Body) (Body, *Error) { + var result Body + for i := range body { + exprs, err := rewriteWithModifier(c, f, body[i]) + if err != nil { + return nil, err + } + if len(exprs) > 0 { + for _, expr := range exprs { + result.Append(expr) + } + } else { + result.Append(body[i]) + } + } + return result, nil +} + +func rewriteWithModifier(c *Compiler, f *equalityFactory, expr *Expr) ([]*Expr, *Error) { + + var result []*Expr + for i := range expr.With { + err := validateTarget(c, expr.With[i].Target) + if err != nil { + return nil, err + } + + if requiresEval(expr.With[i].Value) { + eq := f.Generate(expr.With[i].Value) + result = append(result, eq) + expr.With[i].Value = eq.Operand(0) + } + } + + // If any of the with modifiers in this expression were rewritten then result + // will be non-empty. In this case, the expression will have been modified and + // it should also be added to the result. + if len(result) > 0 { + result = append(result, expr) + } + return result, nil +} + +func validateTarget(c *Compiler, term *Term) *Error { + if !isInputRef(term) && !isDataRef(term) { + return NewError(TypeErr, term.Location, "with keyword target must start with %v or %v", InputRootDocument, DefaultRootDocument) + } + + if isDataRef(term) { + ref := term.Value.(Ref) + node := c.RuleTree + for i := 0; i < len(ref)-1; i++ { + child := node.Child(ref[i].Value) + if child == nil { + break + } else if len(child.Values) > 0 { + return NewError(CompileErr, term.Loc(), "with keyword cannot partially replace virtual document(s)") + } + node = child + } + + if node != nil { + if child := node.Child(ref[len(ref)-1].Value); child != nil { + for _, value := range child.Values { + if len(value.(*Rule).Head.Args) > 0 { + return NewError(CompileErr, term.Loc(), "with keyword cannot replace functions") + } + } + } + } + + } + return nil +} + +func isInputRef(term *Term) bool { + if ref, ok := term.Value.(Ref); ok { + if ref.HasPrefix(InputRootRef) { + return true + } + } + return false +} + +func isDataRef(term *Term) bool { + if ref, ok := term.Value.(Ref); ok { + if ref.HasPrefix(DefaultRootRef) { + return true + } + } + return false +} + +func isVirtual(node *TreeNode, ref Ref) bool { + for i := 0; i < len(ref); i++ { + child := node.Child(ref[i].Value) + if child == nil { + return false + } else if len(child.Values) > 0 { + return true + } + node = child + } + return true +} + +func safetyErrorSlice(unsafe unsafeVars, rewritten map[Var]Var) (result Errors) { + + if len(unsafe) == 0 { + return + } + + for _, pair := range unsafe.Vars() { + v := pair.Var + if w, ok := rewritten[v]; ok { + v = w + } + if !v.IsGenerated() { + if _, ok := futureKeywords[string(v)]; ok { + result = append(result, NewError(UnsafeVarErr, pair.Loc, + "var %[1]v is unsafe (hint: `import future.keywords.%[1]v` to import a future keyword)", v)) + continue + } + result = append(result, NewError(UnsafeVarErr, pair.Loc, "var %v is unsafe", v)) + } + } + + if len(result) > 0 { + return + } + + // If the expression contains unsafe generated variables, report which + // expressions are unsafe instead of the variables that are unsafe (since + // the latter are not meaningful to the user.) + pairs := unsafe.Slice() + + sort.Slice(pairs, func(i, j int) bool { + return pairs[i].Expr.Location.Compare(pairs[j].Expr.Location) < 0 + }) + + // Report at most one error per generated variable. + seen := NewVarSet() + + for _, expr := range pairs { + before := len(seen) + for v := range expr.Vars { + if v.IsGenerated() { + seen.Add(v) + } + } + if len(seen) > before { + result = append(result, NewError(UnsafeVarErr, expr.Expr.Location, "expression is unsafe")) + } + } + + return +} + +func checkUnsafeBuiltins(unsafeBuiltinsMap map[string]struct{}, node interface{}) Errors { + errs := make(Errors, 0) + WalkExprs(node, func(x *Expr) bool { + if x.IsCall() { + operator := x.Operator().String() + if _, ok := unsafeBuiltinsMap[operator]; ok { + errs = append(errs, NewError(TypeErr, x.Loc(), "unsafe built-in function calls in expression: %v", operator)) + } + } + return false + }) + return errs +} + +func rewriteVarsInRef(vars ...map[Var]Var) varRewriter { + return func(node Ref) Ref { + i, _ := TransformVars(node, func(v Var) (Value, error) { + for _, m := range vars { + if u, ok := m[v]; ok { + return u, nil + } + } + return v, nil + }) + return i.(Ref) + } +} diff --git a/vendor/github.com/open-policy-agent/opa/ast/compilehelper.go b/vendor/github.com/open-policy-agent/opa/ast/compilehelper.go new file mode 100644 index 00000000..ca75dfab --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/ast/compilehelper.go @@ -0,0 +1,59 @@ +// Copyright 2016 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package ast + +// CompileModules takes a set of Rego modules represented as strings and +// compiles them for evaluation. The keys of the map are used as filenames. +func CompileModules(modules map[string]string) (*Compiler, error) { + return CompileModulesWithOpt(modules, CompileOpts{}) +} + +// CompileOpts defines a set of options for the compiler. +type CompileOpts struct { + EnablePrintStatements bool +} + +// CompileModulesWithOpt takes a set of Rego modules represented as strings and +// compiles them for evaluation. The keys of the map are used as filenames. +func CompileModulesWithOpt(modules map[string]string, opts CompileOpts) (*Compiler, error) { + + parsed := make(map[string]*Module, len(modules)) + + for f, module := range modules { + var pm *Module + var err error + if pm, err = ParseModule(f, module); err != nil { + return nil, err + } + parsed[f] = pm + } + + compiler := NewCompiler().WithEnablePrintStatements(opts.EnablePrintStatements) + compiler.Compile(parsed) + + if compiler.Failed() { + return nil, compiler.Errors + } + + return compiler, nil +} + +// MustCompileModules compiles a set of Rego modules represented as strings. If +// the compilation process fails, this function panics. +func MustCompileModules(modules map[string]string) *Compiler { + return MustCompileModulesWithOpts(modules, CompileOpts{}) +} + +// MustCompileModulesWithOpts compiles a set of Rego modules represented as strings. If +// the compilation process fails, this function panics. +func MustCompileModulesWithOpts(modules map[string]string, opts CompileOpts) *Compiler { + + compiler, err := CompileModulesWithOpt(modules, opts) + if err != nil { + panic(err) + } + + return compiler +} diff --git a/vendor/github.com/open-policy-agent/opa/ast/compilemetrics.go b/vendor/github.com/open-policy-agent/opa/ast/compilemetrics.go new file mode 100644 index 00000000..5d952258 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/ast/compilemetrics.go @@ -0,0 +1,9 @@ +// Copyright 2020 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package ast + +const ( + compileStageComprehensionIndexBuild = "compile_stage_comprehension_index_build" +) diff --git a/vendor/github.com/open-policy-agent/opa/ast/conflicts.go b/vendor/github.com/open-policy-agent/opa/ast/conflicts.go new file mode 100644 index 00000000..d1013cce --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/ast/conflicts.go @@ -0,0 +1,48 @@ +// Copyright 2019 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package ast + +import ( + "strings" +) + +// CheckPathConflicts returns a set of errors indicating paths that +// are in conflict with the result of the provided callable. +func CheckPathConflicts(c *Compiler, exists func([]string) (bool, error)) Errors { + var errs Errors + + root := c.RuleTree.Child(DefaultRootDocument.Value) + if root == nil { + return nil + } + + for _, node := range root.Children { + errs = append(errs, checkDocumentConflicts(node, exists, nil)...) + } + + return errs +} + +func checkDocumentConflicts(node *TreeNode, exists func([]string) (bool, error), path []string) Errors { + + path = append(path, string(node.Key.(String))) + + if len(node.Values) > 0 { + s := strings.Join(path, "/") + if ok, err := exists(path); err != nil { + return Errors{NewError(CompileErr, node.Values[0].(*Rule).Loc(), "conflict check for data path %v: %v", s, err.Error())} + } else if ok { + return Errors{NewError(CompileErr, node.Values[0].(*Rule).Loc(), "conflicting rule for data path %v found", s)} + } + } + + var errs Errors + + for _, child := range node.Children { + errs = append(errs, checkDocumentConflicts(child, exists, path)...) + } + + return errs +} diff --git a/vendor/github.com/open-policy-agent/opa/ast/doc.go b/vendor/github.com/open-policy-agent/opa/ast/doc.go new file mode 100644 index 00000000..363660cf --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/ast/doc.go @@ -0,0 +1,36 @@ +// Copyright 2016 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +// Package ast declares Rego syntax tree types and also includes a parser and compiler for preparing policies for execution in the policy engine. +// +// Rego policies are defined using a relatively small set of types: modules, package and import declarations, rules, expressions, and terms. At their core, policies consist of rules that are defined by one or more expressions over documents available to the policy engine. The expressions are defined by intrinsic values (terms) such as strings, objects, variables, etc. +// +// Rego policies are typically defined in text files and then parsed and compiled by the policy engine at runtime. The parsing stage takes the text or string representation of the policy and converts it into an abstract syntax tree (AST) that consists of the types mentioned above. The AST is organized as follows: +// +// Module +// | +// +--- Package (Reference) +// | +// +--- Imports +// | | +// | +--- Import (Term) +// | +// +--- Rules +// | +// +--- Rule +// | +// +--- Head +// | | +// | +--- Name (Variable) +// | | +// | +--- Key (Term) +// | | +// | +--- Value (Term) +// | +// +--- Body +// | +// +--- Expression (Term | Terms | Variable Declaration) +// +// At query time, the policy engine expects policies to have been compiled. The compilation stage takes one or more modules and compiles them into a format that the policy engine supports. +package ast diff --git a/vendor/github.com/open-policy-agent/opa/ast/env.go b/vendor/github.com/open-policy-agent/opa/ast/env.go new file mode 100644 index 00000000..60006baa --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/ast/env.go @@ -0,0 +1,327 @@ +// Copyright 2017 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package ast + +import ( + "github.com/open-policy-agent/opa/types" + "github.com/open-policy-agent/opa/util" +) + +// TypeEnv contains type info for static analysis such as type checking. +type TypeEnv struct { + tree *typeTreeNode + next *TypeEnv + newChecker func() *typeChecker +} + +// newTypeEnv returns an empty TypeEnv. The constructor is not exported because +// type environments should only be created by the type checker. +func newTypeEnv(f func() *typeChecker) *TypeEnv { + return &TypeEnv{ + tree: newTypeTree(), + newChecker: f, + } +} + +// Get returns the type of x. +func (env *TypeEnv) Get(x interface{}) types.Type { + + if term, ok := x.(*Term); ok { + x = term.Value + } + + switch x := x.(type) { + + // Scalars. + case Null: + return types.NewNull() + case Boolean: + return types.NewBoolean() + case Number: + return types.NewNumber() + case String: + return types.NewString() + + // Composites. + case *Array: + static := make([]types.Type, x.Len()) + for i := range static { + tpe := env.Get(x.Elem(i).Value) + static[i] = tpe + } + + var dynamic types.Type + if len(static) == 0 { + dynamic = types.A + } + + return types.NewArray(static, dynamic) + + case *object: + static := []*types.StaticProperty{} + var dynamic *types.DynamicProperty + + x.Foreach(func(k, v *Term) { + if IsConstant(k.Value) { + kjson, err := JSON(k.Value) + if err == nil { + tpe := env.Get(v) + static = append(static, types.NewStaticProperty(kjson, tpe)) + return + } + } + // Can't handle it as a static property, fallback to dynamic + typeK := env.Get(k.Value) + typeV := env.Get(v.Value) + dynamic = types.NewDynamicProperty(typeK, typeV) + }) + + if len(static) == 0 && dynamic == nil { + dynamic = types.NewDynamicProperty(types.A, types.A) + } + + return types.NewObject(static, dynamic) + + case Set: + var tpe types.Type + x.Foreach(func(elem *Term) { + other := env.Get(elem.Value) + tpe = types.Or(tpe, other) + }) + if tpe == nil { + tpe = types.A + } + return types.NewSet(tpe) + + // Comprehensions. + case *ArrayComprehension: + cpy, errs := env.newChecker().CheckBody(env, x.Body) + if len(errs) == 0 { + return types.NewArray(nil, cpy.Get(x.Term)) + } + return nil + case *ObjectComprehension: + cpy, errs := env.newChecker().CheckBody(env, x.Body) + if len(errs) == 0 { + return types.NewObject(nil, types.NewDynamicProperty(cpy.Get(x.Key), cpy.Get(x.Value))) + } + return nil + case *SetComprehension: + cpy, errs := env.newChecker().CheckBody(env, x.Body) + if len(errs) == 0 { + return types.NewSet(cpy.Get(x.Term)) + } + return nil + + // Refs. + case Ref: + return env.getRef(x) + + // Vars. + case Var: + if node := env.tree.Child(x); node != nil { + return node.Value() + } + if env.next != nil { + return env.next.Get(x) + } + return nil + + // Calls. + case Call: + return nil + + default: + panic("unreachable") + } +} + +func (env *TypeEnv) getRef(ref Ref) types.Type { + + node := env.tree.Child(ref[0].Value) + if node == nil { + return env.getRefFallback(ref) + } + + return env.getRefRec(node, ref, ref[1:]) +} + +func (env *TypeEnv) getRefFallback(ref Ref) types.Type { + + if env.next != nil { + return env.next.Get(ref) + } + + if RootDocumentNames.Contains(ref[0]) { + return types.A + } + + return nil +} + +func (env *TypeEnv) getRefRec(node *typeTreeNode, ref, tail Ref) types.Type { + if len(tail) == 0 { + return env.getRefRecExtent(node) + } + + if node.Leaf() { + return selectRef(node.Value(), tail) + } + + if !IsConstant(tail[0].Value) { + return selectRef(env.getRefRecExtent(node), tail) + } + + child := node.Child(tail[0].Value) + if child == nil { + return env.getRefFallback(ref) + } + + return env.getRefRec(child, ref, tail[1:]) +} + +func (env *TypeEnv) getRefRecExtent(node *typeTreeNode) types.Type { + + if node.Leaf() { + return node.Value() + } + + children := []*types.StaticProperty{} + + node.Children().Iter(func(k, v util.T) bool { + key := k.(Value) + child := v.(*typeTreeNode) + + tpe := env.getRefRecExtent(child) + // TODO(tsandall): handle non-string keys? + if s, ok := key.(String); ok { + children = append(children, types.NewStaticProperty(string(s), tpe)) + } + return false + }) + + // TODO(tsandall): for now, these objects can have any dynamic properties + // because we don't have schema for base docs. Once schemas are supported + // we can improve this. + return types.NewObject(children, types.NewDynamicProperty(types.S, types.A)) +} + +func (env *TypeEnv) wrap() *TypeEnv { + cpy := *env + cpy.next = env + cpy.tree = newTypeTree() + return &cpy +} + +// typeTreeNode is used to store type information in a tree. +type typeTreeNode struct { + key Value + value types.Type + children *util.HashMap +} + +func newTypeTree() *typeTreeNode { + return &typeTreeNode{ + key: nil, + value: nil, + children: util.NewHashMap(valueEq, valueHash), + } +} + +func (n *typeTreeNode) Child(key Value) *typeTreeNode { + value, ok := n.children.Get(key) + if !ok { + return nil + } + return value.(*typeTreeNode) +} + +func (n *typeTreeNode) Children() *util.HashMap { + return n.children +} + +func (n *typeTreeNode) Get(path Ref) types.Type { + curr := n + for _, term := range path { + child, ok := curr.children.Get(term.Value) + if !ok { + return nil + } + curr = child.(*typeTreeNode) + } + return curr.Value() +} + +func (n *typeTreeNode) Leaf() bool { + return n.value != nil +} + +func (n *typeTreeNode) PutOne(key Value, tpe types.Type) { + c, ok := n.children.Get(key) + + var child *typeTreeNode + if !ok { + child = newTypeTree() + child.key = key + n.children.Put(key, child) + } else { + child = c.(*typeTreeNode) + } + + child.value = tpe +} + +func (n *typeTreeNode) Put(path Ref, tpe types.Type) { + curr := n + for _, term := range path { + c, ok := curr.children.Get(term.Value) + + var child *typeTreeNode + if !ok { + child = newTypeTree() + child.key = term.Value + curr.children.Put(child.key, child) + } else { + child = c.(*typeTreeNode) + } + + curr = child + } + curr.value = tpe +} + +func (n *typeTreeNode) Value() types.Type { + return n.value +} + +// selectConstant returns the attribute of the type referred to by the term. If +// the attribute type cannot be determined, nil is returned. +func selectConstant(tpe types.Type, term *Term) types.Type { + x, err := JSON(term.Value) + if err == nil { + return types.Select(tpe, x) + } + return nil +} + +// selectRef returns the type of the nested attribute referred to by ref. If +// the attribute type cannot be determined, nil is returned. If the ref +// contains vars or refs, then the returned type will be a union of the +// possible types. +func selectRef(tpe types.Type, ref Ref) types.Type { + + if tpe == nil || len(ref) == 0 { + return tpe + } + + head, tail := ref[0], ref[1:] + + switch head.Value.(type) { + case Var, Ref, *Array, Object, Set: + return selectRef(types.Values(tpe), tail) + default: + return selectRef(selectConstant(tpe, head), tail) + } +} diff --git a/vendor/github.com/open-policy-agent/opa/ast/errors.go b/vendor/github.com/open-policy-agent/opa/ast/errors.go new file mode 100644 index 00000000..11348b3d --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/ast/errors.go @@ -0,0 +1,132 @@ +// Copyright 2016 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package ast + +import ( + "fmt" + "sort" + "strings" +) + +// Errors represents a series of errors encountered during parsing, compiling, +// etc. +type Errors []*Error + +func (e Errors) Error() string { + + if len(e) == 0 { + return "no error(s)" + } + + if len(e) == 1 { + return fmt.Sprintf("1 error occurred: %v", e[0].Error()) + } + + s := make([]string, len(e)) + for i, err := range e { + s[i] = err.Error() + } + + return fmt.Sprintf("%d errors occurred:\n%s", len(e), strings.Join(s, "\n")) +} + +// Sort sorts the error slice by location. If the locations are equal then the +// error message is compared. +func (e Errors) Sort() { + sort.Slice(e, func(i, j int) bool { + a := e[i] + b := e[j] + + if cmp := a.Location.Compare(b.Location); cmp != 0 { + return cmp < 0 + } + + return a.Error() < b.Error() + }) +} + +const ( + // ParseErr indicates an unclassified parse error occurred. + ParseErr = "rego_parse_error" + + // CompileErr indicates an unclassified compile error occurred. + CompileErr = "rego_compile_error" + + // TypeErr indicates a type error was caught. + TypeErr = "rego_type_error" + + // UnsafeVarErr indicates an unsafe variable was found during compilation. + UnsafeVarErr = "rego_unsafe_var_error" + + // RecursionErr indicates recursion was found during compilation. + RecursionErr = "rego_recursion_error" +) + +// IsError returns true if err is an AST error with code. +func IsError(code string, err error) bool { + if err, ok := err.(*Error); ok { + return err.Code == code + } + return false +} + +// ErrorDetails defines the interface for detailed error messages. +type ErrorDetails interface { + Lines() []string +} + +// Error represents a single error caught during parsing, compiling, etc. +type Error struct { + Code string `json:"code"` + Message string `json:"message"` + Location *Location `json:"location,omitempty"` + Details ErrorDetails `json:"details,omitempty"` +} + +func (e *Error) Error() string { + + var prefix string + + if e.Location != nil { + + if len(e.Location.File) > 0 { + prefix += e.Location.File + ":" + fmt.Sprint(e.Location.Row) + } else { + prefix += fmt.Sprint(e.Location.Row) + ":" + fmt.Sprint(e.Location.Col) + } + } + + msg := fmt.Sprintf("%v: %v", e.Code, e.Message) + + if len(prefix) > 0 { + msg = prefix + ": " + msg + } + + if e.Details != nil { + for _, line := range e.Details.Lines() { + msg += "\n\t" + line + } + } + + return msg +} + +// NewError returns a new Error object. +func NewError(code string, loc *Location, f string, a ...interface{}) *Error { + return &Error{ + Code: code, + Location: loc, + Message: fmt.Sprintf(f, a...), + } +} + +var ( + errPartialRuleAssignOperator = fmt.Errorf("partial rules must use = operator (not := operator)") + errFunctionAssignOperator = fmt.Errorf("functions must use = operator (not := operator)") +) + +func errTermAssignOperator(x interface{}) error { + return fmt.Errorf("cannot assign to %v", TypeName(x)) +} diff --git a/vendor/github.com/open-policy-agent/opa/ast/fuzz.go b/vendor/github.com/open-policy-agent/opa/ast/fuzz.go new file mode 100644 index 00000000..6ff7e35a --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/ast/fuzz.go @@ -0,0 +1,16 @@ +// +build gofuzz + +package ast + +func Fuzz(data []byte) int { + + str := string(data) + _, _, err := ParseStatements("", str) + + if err == nil { + CompileModules(map[string]string{"": str}) + return 1 + } + + return 0 +} diff --git a/vendor/github.com/open-policy-agent/opa/ast/index.go b/vendor/github.com/open-policy-agent/opa/ast/index.go new file mode 100644 index 00000000..bcbb5c17 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/ast/index.go @@ -0,0 +1,884 @@ +// Copyright 2017 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package ast + +import ( + "fmt" + "sort" + "strings" + + "github.com/open-policy-agent/opa/util" +) + +// RuleIndex defines the interface for rule indices. +type RuleIndex interface { + + // Build tries to construct an index for the given rules. If the index was + // constructed, it returns true, otherwise false. + Build(rules []*Rule) bool + + // Lookup searches the index for rules that will match the provided + // resolver. If the resolver returns an error, it is returned via err. + Lookup(resolver ValueResolver) (*IndexResult, error) + + // AllRules traverses the index and returns all rules that will match + // the provided resolver without any optimizations (effectively with + // indexing disabled). If the resolver returns an error, it is returned + // via err. + AllRules(resolver ValueResolver) (*IndexResult, error) +} + +// IndexResult contains the result of an index lookup. +type IndexResult struct { + Kind DocKind + Rules []*Rule + Else map[*Rule][]*Rule + Default *Rule + EarlyExit bool +} + +// NewIndexResult returns a new IndexResult object. +func NewIndexResult(kind DocKind) *IndexResult { + return &IndexResult{ + Kind: kind, + Else: map[*Rule][]*Rule{}, + } +} + +// Empty returns true if there are no rules to evaluate. +func (ir *IndexResult) Empty() bool { + return len(ir.Rules) == 0 && ir.Default == nil +} + +type baseDocEqIndex struct { + skipIndexing Set + isVirtual func(Ref) bool + root *trieNode + defaultRule *Rule + kind DocKind +} + +func newBaseDocEqIndex(isVirtual func(Ref) bool) *baseDocEqIndex { + return &baseDocEqIndex{ + skipIndexing: NewSet(NewTerm(InternalPrint.Ref())), + isVirtual: isVirtual, + root: newTrieNodeImpl(), + } +} + +func (i *baseDocEqIndex) Build(rules []*Rule) bool { + if len(rules) == 0 { + return false + } + + i.kind = rules[0].Head.DocKind() + indices := newrefindices(i.isVirtual) + + // build indices for each rule. + for idx := range rules { + WalkRules(rules[idx], func(rule *Rule) bool { + if rule.Default { + i.defaultRule = rule + return false + } + var skip bool + for _, expr := range rule.Body { + if op := expr.OperatorTerm(); op != nil && i.skipIndexing.Contains(op) { + skip = true + break + } + } + if !skip { + for _, expr := range rule.Body { + indices.Update(rule, expr) + } + } + return false + }) + } + + // build trie out of indices. + for idx := range rules { + var prio int + WalkRules(rules[idx], func(rule *Rule) bool { + if rule.Default { + return false + } + node := i.root + if indices.Indexed(rule) { + for _, ref := range indices.Sorted() { + node = node.Insert(ref, indices.Value(rule, ref), indices.Mapper(rule, ref)) + } + } + // Insert rule into trie with (insertion order, priority order) + // tuple. Retaining the insertion order allows us to return rules + // in the order they were passed to this function. + node.append([...]int{idx, prio}, rule) + prio++ + return false + }) + } + return true +} + +func (i *baseDocEqIndex) Lookup(resolver ValueResolver) (*IndexResult, error) { + + tr := newTrieTraversalResult() + + err := i.root.Traverse(resolver, tr) + if err != nil { + return nil, err + } + + result := NewIndexResult(i.kind) + result.Default = i.defaultRule + result.Rules = make([]*Rule, 0, len(tr.ordering)) + + for _, pos := range tr.ordering { + sort.Slice(tr.unordered[pos], func(i, j int) bool { + return tr.unordered[pos][i].prio[1] < tr.unordered[pos][j].prio[1] + }) + nodes := tr.unordered[pos] + root := nodes[0].rule + + result.Rules = append(result.Rules, root) + if len(nodes) > 1 { + result.Else[root] = make([]*Rule, len(nodes)-1) + for i := 1; i < len(nodes); i++ { + result.Else[root][i-1] = nodes[i].rule + } + } + } + + result.EarlyExit = tr.values.Len() == 1 && tr.values.Slice()[0].IsGround() + + return result, nil +} + +func (i *baseDocEqIndex) AllRules(resolver ValueResolver) (*IndexResult, error) { + tr := newTrieTraversalResult() + + // Walk over the rule trie and accumulate _all_ rules + rw := &ruleWalker{result: tr} + i.root.Do(rw) + + result := NewIndexResult(i.kind) + result.Default = i.defaultRule + result.Rules = make([]*Rule, 0, len(tr.ordering)) + + for _, pos := range tr.ordering { + sort.Slice(tr.unordered[pos], func(i, j int) bool { + return tr.unordered[pos][i].prio[1] < tr.unordered[pos][j].prio[1] + }) + nodes := tr.unordered[pos] + root := nodes[0].rule + result.Rules = append(result.Rules, root) + if len(nodes) > 1 { + result.Else[root] = make([]*Rule, len(nodes)-1) + for i := 1; i < len(nodes); i++ { + result.Else[root][i-1] = nodes[i].rule + } + } + } + + result.EarlyExit = tr.values.Len() == 1 && tr.values.Slice()[0].IsGround() + + return result, nil +} + +type ruleWalker struct { + result *trieTraversalResult +} + +func (r *ruleWalker) Do(x interface{}) trieWalker { + tn := x.(*trieNode) + r.result.Add(tn) + return r +} + +type valueMapper struct { + Key string + MapValue func(Value) Value +} + +type refindex struct { + Ref Ref + Value Value + Mapper *valueMapper +} + +type refindices struct { + isVirtual func(Ref) bool + rules map[*Rule][]*refindex + frequency *util.HashMap + sorted []Ref +} + +func newrefindices(isVirtual func(Ref) bool) *refindices { + return &refindices{ + isVirtual: isVirtual, + rules: map[*Rule][]*refindex{}, + frequency: util.NewHashMap(func(a, b util.T) bool { + r1, r2 := a.(Ref), b.(Ref) + return r1.Equal(r2) + }, func(x util.T) int { + return x.(Ref).Hash() + }), + } +} + +// Update attempts to update the refindices for the given expression in the +// given rule. If the expression cannot be indexed the update does not affect +// the indices. +func (i *refindices) Update(rule *Rule, expr *Expr) { + + if expr.Negated { + return + } + + if len(expr.With) > 0 { + // NOTE(tsandall): In the future, we may need to consider expressions + // that have with statements applied to them. + return + } + + op := expr.Operator() + + if op.Equal(Equality.Ref()) { + i.updateEq(rule, expr) + } else if op.Equal(Equal.Ref()) && len(expr.Operands()) == 2 { + // NOTE(tsandall): if equal() is called with more than two arguments the + // output value is being captured in which case the indexer cannot + // exclude the rule if the equal() call would return false (because the + // false value must still be produced.) + i.updateEq(rule, expr) + } else if op.Equal(GlobMatch.Ref()) { + i.updateGlobMatch(rule, expr) + } +} + +// Sorted returns a sorted list of references that the indices were built from. +// References that appear more frequently in the indexed rules are ordered +// before less frequently appearing references. +func (i *refindices) Sorted() []Ref { + + if i.sorted == nil { + counts := make([]int, 0, i.frequency.Len()) + i.sorted = make([]Ref, 0, i.frequency.Len()) + + i.frequency.Iter(func(k, v util.T) bool { + counts = append(counts, v.(int)) + i.sorted = append(i.sorted, k.(Ref)) + return false + }) + + sort.Slice(i.sorted, func(a, b int) bool { + if counts[a] > counts[b] { + return true + } else if counts[b] > counts[a] { + return false + } + return i.sorted[a][0].Loc().Compare(i.sorted[b][0].Loc()) < 0 + }) + } + + return i.sorted +} + +func (i *refindices) Indexed(rule *Rule) bool { + return len(i.rules[rule]) > 0 +} + +func (i *refindices) Value(rule *Rule, ref Ref) Value { + if index := i.index(rule, ref); index != nil { + return index.Value + } + return nil +} + +func (i *refindices) Mapper(rule *Rule, ref Ref) *valueMapper { + if index := i.index(rule, ref); index != nil { + return index.Mapper + } + return nil +} + +func (i *refindices) updateEq(rule *Rule, expr *Expr) { + a, b := expr.Operand(0), expr.Operand(1) + args := rule.Head.Args + if idx, ok := eqOperandsToRefAndValue(i.isVirtual, args, a, b); ok { + i.insert(rule, idx) + return + } + if idx, ok := eqOperandsToRefAndValue(i.isVirtual, args, b, a); ok { + i.insert(rule, idx) + return + } +} + +func (i *refindices) updateGlobMatch(rule *Rule, expr *Expr) { + args := rule.Head.Args + + delim, ok := globDelimiterToString(expr.Operand(1)) + if !ok { + return + } + + if arr := globPatternToArray(expr.Operand(0), delim); arr != nil { + // The 3rd operand of glob.match is the value to match. We assume the + // 3rd operand was a reference that has been rewritten and bound to a + // variable earlier in the query OR a function argument variable. + match := expr.Operand(2) + if _, ok := match.Value.(Var); ok { + var ref Ref + for _, other := range i.rules[rule] { + if _, ok := other.Value.(Var); ok && other.Value.Compare(match.Value) == 0 { + ref = other.Ref + } + } + if ref == nil { + for j, arg := range args { + if arg.Equal(match) { + ref = Ref{FunctionArgRootDocument, IntNumberTerm(j)} + } + } + } + if ref != nil { + i.insert(rule, &refindex{ + Ref: ref, + Value: arr.Value, + Mapper: &valueMapper{ + Key: delim, + MapValue: func(v Value) Value { + if s, ok := v.(String); ok { + return stringSliceToArray(splitStringEscaped(string(s), delim)) + } + return v + }, + }, + }) + } + } + } +} + +func (i *refindices) insert(rule *Rule, index *refindex) { + + count, ok := i.frequency.Get(index.Ref) + if !ok { + count = 0 + } + + i.frequency.Put(index.Ref, count.(int)+1) + + for pos, other := range i.rules[rule] { + if other.Ref.Equal(index.Ref) { + i.rules[rule][pos] = index + return + } + } + + i.rules[rule] = append(i.rules[rule], index) +} + +func (i *refindices) index(rule *Rule, ref Ref) *refindex { + for _, index := range i.rules[rule] { + if index.Ref.Equal(ref) { + return index + } + } + return nil +} + +type trieWalker interface { + Do(x interface{}) trieWalker +} + +type trieTraversalResult struct { + unordered map[int][]*ruleNode + ordering []int + values Set +} + +func newTrieTraversalResult() *trieTraversalResult { + return &trieTraversalResult{ + unordered: map[int][]*ruleNode{}, + values: NewSet(), + } +} + +func (tr *trieTraversalResult) Add(t *trieNode) { + for _, node := range t.rules { + root := node.prio[0] + nodes, ok := tr.unordered[root] + if !ok { + tr.ordering = append(tr.ordering, root) + } + tr.unordered[root] = append(nodes, node) + } + if t.values != nil { + t.values.Foreach(func(v *Term) { tr.values.Add(v) }) + } +} + +type trieNode struct { + ref Ref + values Set + mappers []*valueMapper + next *trieNode + any *trieNode + undefined *trieNode + scalars map[Value]*trieNode + array *trieNode + rules []*ruleNode +} + +func (node *trieNode) String() string { + var flags []string + flags = append(flags, fmt.Sprintf("self:%p", node)) + if len(node.ref) > 0 { + flags = append(flags, node.ref.String()) + } + if node.next != nil { + flags = append(flags, fmt.Sprintf("next:%p", node.next)) + } + if node.any != nil { + flags = append(flags, fmt.Sprintf("any:%p", node.any)) + } + if node.undefined != nil { + flags = append(flags, fmt.Sprintf("undefined:%p", node.undefined)) + } + if node.array != nil { + flags = append(flags, fmt.Sprintf("array:%p", node.array)) + } + if len(node.scalars) > 0 { + buf := make([]string, 0, len(node.scalars)) + for k, v := range node.scalars { + buf = append(buf, fmt.Sprintf("scalar(%v):%p", k, v)) + } + sort.Strings(buf) + flags = append(flags, strings.Join(buf, " ")) + } + if len(node.rules) > 0 { + flags = append(flags, fmt.Sprintf("%d rule(s)", len(node.rules))) + } + if len(node.mappers) > 0 { + flags = append(flags, fmt.Sprintf("%d mapper(s)", len(node.mappers))) + } + if l := node.values.Len(); l > 0 { + flags = append(flags, fmt.Sprintf("%d value(s)", l)) + } + return strings.Join(flags, " ") +} + +func (node *trieNode) append(prio [2]int, rule *Rule) { + node.rules = append(node.rules, &ruleNode{prio, rule}) + + if node.values != nil { + node.values.Add(rule.Head.Value) + return + } + + if node.values == nil && rule.Head.DocKind() == CompleteDoc { + node.values = NewSet(rule.Head.Value) + } +} + +type ruleNode struct { + prio [2]int + rule *Rule +} + +func newTrieNodeImpl() *trieNode { + return &trieNode{ + scalars: map[Value]*trieNode{}, + } +} + +func (node *trieNode) Do(walker trieWalker) { + next := walker.Do(node) + if next == nil { + return + } + if node.any != nil { + node.any.Do(next) + } + if node.undefined != nil { + node.undefined.Do(next) + } + for _, child := range node.scalars { + child.Do(next) + } + if node.array != nil { + node.array.Do(next) + } + if node.next != nil { + node.next.Do(next) + } +} + +func (node *trieNode) Insert(ref Ref, value Value, mapper *valueMapper) *trieNode { + + if node.next == nil { + node.next = newTrieNodeImpl() + node.next.ref = ref + } + + if mapper != nil { + node.next.addMapper(mapper) + } + + return node.next.insertValue(value) +} + +func (node *trieNode) Traverse(resolver ValueResolver, tr *trieTraversalResult) error { + + if node == nil { + return nil + } + + tr.Add(node) + + return node.next.traverse(resolver, tr) +} + +func (node *trieNode) addMapper(mapper *valueMapper) { + for i := range node.mappers { + if node.mappers[i].Key == mapper.Key { + return + } + } + node.mappers = append(node.mappers, mapper) +} + +func (node *trieNode) insertValue(value Value) *trieNode { + + switch value := value.(type) { + case nil: + if node.undefined == nil { + node.undefined = newTrieNodeImpl() + } + return node.undefined + case Var: + if node.any == nil { + node.any = newTrieNodeImpl() + } + return node.any + case Null, Boolean, Number, String: + child, ok := node.scalars[value] + if !ok { + child = newTrieNodeImpl() + node.scalars[value] = child + } + return child + case *Array: + if node.array == nil { + node.array = newTrieNodeImpl() + } + return node.array.insertArray(value) + } + + panic("illegal value") +} + +func (node *trieNode) insertArray(arr *Array) *trieNode { + + if arr.Len() == 0 { + return node + } + + switch head := arr.Elem(0).Value.(type) { + case Var: + if node.any == nil { + node.any = newTrieNodeImpl() + } + return node.any.insertArray(arr.Slice(1, -1)) + case Null, Boolean, Number, String: + child, ok := node.scalars[head] + if !ok { + child = newTrieNodeImpl() + node.scalars[head] = child + } + return child.insertArray(arr.Slice(1, -1)) + } + + panic("illegal value") +} + +func (node *trieNode) traverse(resolver ValueResolver, tr *trieTraversalResult) error { + + if node == nil { + return nil + } + + v, err := resolver.Resolve(node.ref) + if err != nil { + if IsUnknownValueErr(err) { + return node.traverseUnknown(resolver, tr) + } + return err + } + + if node.undefined != nil { + err = node.undefined.Traverse(resolver, tr) + if err != nil { + return err + } + } + + if v == nil { + return nil + } + + if node.any != nil { + err = node.any.Traverse(resolver, tr) + if err != nil { + return err + } + } + + if err := node.traverseValue(resolver, tr, v); err != nil { + return err + } + + for i := range node.mappers { + if err := node.traverseValue(resolver, tr, node.mappers[i].MapValue(v)); err != nil { + return err + } + } + + return nil +} + +func (node *trieNode) traverseValue(resolver ValueResolver, tr *trieTraversalResult, value Value) error { + + switch value := value.(type) { + case *Array: + if node.array == nil { + return nil + } + return node.array.traverseArray(resolver, tr, value) + + case Null, Boolean, Number, String: + child, ok := node.scalars[value] + if !ok { + return nil + } + return child.Traverse(resolver, tr) + } + + return nil +} + +func (node *trieNode) traverseArray(resolver ValueResolver, tr *trieTraversalResult, arr *Array) error { + + if arr.Len() == 0 { + return node.Traverse(resolver, tr) + } + + head := arr.Elem(0).Value + + if !IsScalar(head) { + return nil + } + + if node.any != nil { + err := node.any.traverseArray(resolver, tr, arr.Slice(1, -1)) + if err != nil { + return err + } + } + + child, ok := node.scalars[head] + if !ok { + return nil + } + + return child.traverseArray(resolver, tr, arr.Slice(1, -1)) +} + +func (node *trieNode) traverseUnknown(resolver ValueResolver, tr *trieTraversalResult) error { + + if node == nil { + return nil + } + + if err := node.Traverse(resolver, tr); err != nil { + return err + } + + if err := node.undefined.traverseUnknown(resolver, tr); err != nil { + return err + } + + if err := node.any.traverseUnknown(resolver, tr); err != nil { + return err + } + + if err := node.array.traverseUnknown(resolver, tr); err != nil { + return err + } + + for _, child := range node.scalars { + if err := child.traverseUnknown(resolver, tr); err != nil { + return err + } + } + + return nil +} + +// If term `a` is one of the function's operands, we store a Ref: `args[0]` +// for the argument number. So for `f(x, y) { x = 10; y = 12 }`, we'll +// bind `args[0]` and `args[1]` to this rule when called for (x=10) and +// (y=12) respectively. +func eqOperandsToRefAndValue(isVirtual func(Ref) bool, args []*Term, a, b *Term) (*refindex, bool) { + switch v := a.Value.(type) { + case Var: + for i, arg := range args { + if arg.Value.Compare(v) == 0 { + if bval, ok := indexValue(b); ok { + return &refindex{Ref: Ref{FunctionArgRootDocument, IntNumberTerm(i)}, Value: bval}, true + } + } + } + case Ref: + if !RootDocumentNames.Contains(v[0]) { + return nil, false + } + if isVirtual(v) { + return nil, false + } + if v.IsNested() || !v.IsGround() { + return nil, false + } + if bval, ok := indexValue(b); ok { + return &refindex{Ref: v, Value: bval}, true + } + } + return nil, false +} + +func indexValue(b *Term) (Value, bool) { + switch b := b.Value.(type) { + case Null, Boolean, Number, String, Var: + return b, true + case *Array: + stop := false + first := true + vis := NewGenericVisitor(func(x interface{}) bool { + if first { + first = false + return false + } + switch x.(type) { + // No nested structures or values that require evaluation (other than var). + case *Array, Object, Set, *ArrayComprehension, *ObjectComprehension, *SetComprehension, Ref: + stop = true + } + return stop + }) + vis.Walk(b) + if !stop { + return b, true + } + } + + return nil, false +} + +func globDelimiterToString(delim *Term) (string, bool) { + + arr, ok := delim.Value.(*Array) + if !ok { + return "", false + } + + var result string + + if arr.Len() == 0 { + result = "." + } else { + for i := 0; i < arr.Len(); i++ { + term := arr.Elem(i) + s, ok := term.Value.(String) + if !ok { + return "", false + } + result += string(s) + } + } + + return result, true +} + +func globPatternToArray(pattern *Term, delim string) *Term { + + s, ok := pattern.Value.(String) + if !ok { + return nil + } + + parts := splitStringEscaped(string(s), delim) + arr := make([]*Term, len(parts)) + + for i := range parts { + if parts[i] == "*" { + arr[i] = VarTerm("$globwildcard") + } else { + var escaped bool + for _, c := range parts[i] { + if c == '\\' { + escaped = !escaped + continue + } + if !escaped { + switch c { + case '[', '?', '{', '*': + // TODO(tsandall): super glob and character pattern + // matching not supported yet. + return nil + } + } + escaped = false + } + arr[i] = StringTerm(parts[i]) + } + } + + return NewTerm(NewArray(arr...)) +} + +// splits s on characters in delim except if delim characters have been escaped +// with reverse solidus. +func splitStringEscaped(s string, delim string) []string { + + var last, curr int + var escaped bool + var result []string + + for ; curr < len(s); curr++ { + if s[curr] == '\\' || escaped { + escaped = !escaped + continue + } + if strings.ContainsRune(delim, rune(s[curr])) { + result = append(result, s[last:curr]) + last = curr + 1 + } + } + + result = append(result, s[last:]) + + return result +} + +func stringSliceToArray(s []string) *Array { + arr := make([]*Term, len(s)) + for i, v := range s { + arr[i] = StringTerm(v) + } + return NewArray(arr...) +} diff --git a/vendor/github.com/open-policy-agent/opa/ast/internal/scanner/scanner.go b/vendor/github.com/open-policy-agent/opa/ast/internal/scanner/scanner.go new file mode 100644 index 00000000..9402749e --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/ast/internal/scanner/scanner.go @@ -0,0 +1,411 @@ +// Copyright 2020 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package scanner + +import ( + "fmt" + "io" + "io/ioutil" + "unicode" + "unicode/utf8" + + "github.com/open-policy-agent/opa/ast/internal/tokens" +) + +const bom = 0xFEFF + +// Scanner is used to tokenize an input stream of +// Rego source code. +type Scanner struct { + offset int + row int + col int + bs []byte + curr rune + width int + errors []Error + keywords map[string]tokens.Token +} + +// Error represents a scanner error. +type Error struct { + Pos Position + Message string +} + +// Position represents a point in the scanned source code. +type Position struct { + Offset int // start offset in bytes + End int // end offset in bytes + Row int // line number computed in bytes + Col int // column number computed in bytes +} + +// New returns an initialized scanner that will scan +// through the source code provided by the io.Reader. +func New(r io.Reader) (*Scanner, error) { + + bs, err := ioutil.ReadAll(r) + if err != nil { + return nil, err + } + + s := &Scanner{ + offset: 0, + row: 1, + col: 0, + bs: bs, + curr: -1, + width: 0, + keywords: tokens.Keywords(), + } + + s.next() + + if s.curr == bom { + s.next() + } + + return s, nil +} + +// Bytes returns the raw bytes for the full source +// which the scanner has read in. +func (s *Scanner) Bytes() []byte { + return s.bs +} + +// String returns a human readable string of the current scanner state. +func (s *Scanner) String() string { + return fmt.Sprintf("", s.curr, s.offset, len(s.bs)) +} + +// Keyword will return a token for the passed in +// literal value. If the value is a Rego keyword +// then the appropriate token is returned. Everything +// else is an Ident. +func (s *Scanner) Keyword(lit string) tokens.Token { + if tok, ok := s.keywords[lit]; ok { + return tok + } + return tokens.Ident +} + +// AddKeyword adds a string -> token mapping to this Scanner instance. +func (s *Scanner) AddKeyword(kw string, tok tokens.Token) { + s.keywords[kw] = tok +} + +// WithKeywords returns a new copy of the Scanner struct `s`, with the set +// of known keywords being that of `s` with `kws` added. +func (s *Scanner) WithKeywords(kws map[string]tokens.Token) *Scanner { + cpy := *s + cpy.keywords = make(map[string]tokens.Token, len(s.keywords)+len(kws)) + for kw, tok := range s.keywords { + cpy.AddKeyword(kw, tok) + } + for k, t := range kws { + cpy.AddKeyword(k, t) + } + return &cpy +} + +// Scan will increment the scanners position in the source +// code until the next token is found. The token, starting position +// of the token, string literal, and any errors encountered are +// returned. A token will always be returned, the caller must check +// for any errors before using the other values. +func (s *Scanner) Scan() (tokens.Token, Position, string, []Error) { + + pos := Position{Offset: s.offset - s.width, Row: s.row, Col: s.col} + var tok tokens.Token + var lit string + + if s.isWhitespace() { + lit = string(s.curr) + s.next() + tok = tokens.Whitespace + } else if isLetter(s.curr) { + lit = s.scanIdentifier() + tok = s.Keyword(lit) + } else if isDecimal(s.curr) { + lit = s.scanNumber() + tok = tokens.Number + } else { + ch := s.curr + s.next() + switch ch { + case -1: + tok = tokens.EOF + case '#': + lit = s.scanComment() + tok = tokens.Comment + case '"': + lit = s.scanString() + tok = tokens.String + case '`': + lit = s.scanRawString() + tok = tokens.String + case '[': + tok = tokens.LBrack + case ']': + tok = tokens.RBrack + case '{': + tok = tokens.LBrace + case '}': + tok = tokens.RBrace + case '(': + tok = tokens.LParen + case ')': + tok = tokens.RParen + case ',': + tok = tokens.Comma + case ':': + if s.curr == '=' { + s.next() + tok = tokens.Assign + } else { + tok = tokens.Colon + } + case '+': + tok = tokens.Add + case '-': + tok = tokens.Sub + case '*': + tok = tokens.Mul + case '/': + tok = tokens.Quo + case '%': + tok = tokens.Rem + case '&': + tok = tokens.And + case '|': + tok = tokens.Or + case '=': + if s.curr == '=' { + s.next() + tok = tokens.Equal + } else { + tok = tokens.Unify + } + case '>': + if s.curr == '=' { + s.next() + tok = tokens.Gte + } else { + tok = tokens.Gt + } + case '<': + if s.curr == '=' { + s.next() + tok = tokens.Lte + } else { + tok = tokens.Lt + } + case '!': + if s.curr == '=' { + s.next() + tok = tokens.Neq + } else { + s.error("illegal ! character") + } + case ';': + tok = tokens.Semicolon + case '.': + tok = tokens.Dot + } + } + + pos.End = s.offset - s.width + errs := s.errors + s.errors = nil + + return tok, pos, lit, errs +} + +func (s *Scanner) scanIdentifier() string { + start := s.offset - 1 + for isLetter(s.curr) || isDigit(s.curr) { + s.next() + } + return string(s.bs[start : s.offset-1]) +} + +func (s *Scanner) scanNumber() string { + + start := s.offset - 1 + + if s.curr != '.' { + for isDecimal(s.curr) { + s.next() + } + } + + if s.curr == '.' { + s.next() + var found bool + for isDecimal(s.curr) { + s.next() + found = true + } + if !found { + s.error("expected fraction") + } + } + + if lower(s.curr) == 'e' { + s.next() + if s.curr == '+' || s.curr == '-' { + s.next() + } + var found bool + for isDecimal(s.curr) { + s.next() + found = true + } + if !found { + s.error("expected exponent") + } + } + + // Scan any digits following the decimals to get the + // entire invalid number/identifier. + // Example: 0a2b should be a single invalid number "0a2b" + // rather than a number "0", followed by identifier "a2b". + if isLetter(s.curr) { + s.error("illegal number format") + for isLetter(s.curr) || isDigit(s.curr) { + s.next() + } + } + + return string(s.bs[start : s.offset-1]) +} + +func (s *Scanner) scanString() string { + start := s.literalStart() + for { + ch := s.curr + + if ch == '\n' || ch < 0 { + s.error("non-terminated string") + break + } + + s.next() + + if ch == '"' { + break + } + + if ch == '\\' { + switch s.curr { + case '\\', '"', '/', 'b', 'f', 'n', 'r', 't': + s.next() + case 'u': + s.next() + s.next() + s.next() + s.next() + default: + s.error("illegal escape sequence") + } + } + } + + return string(s.bs[start : s.offset-1]) +} + +func (s *Scanner) scanRawString() string { + start := s.literalStart() + for { + ch := s.curr + s.next() + if ch == '`' { + break + } else if ch < 0 { + s.error("non-terminated string") + break + } + } + return string(s.bs[start : s.offset-1]) +} + +func (s *Scanner) scanComment() string { + start := s.literalStart() + for s.curr != '\n' && s.curr != -1 { + s.next() + } + end := s.offset - 1 + // Trim carriage returns that precede the newline + if s.offset > 1 && s.bs[s.offset-2] == '\r' { + end = end - 1 + } + return string(s.bs[start:end]) +} + +func (s *Scanner) next() { + + if s.offset >= len(s.bs) { + s.curr = -1 + s.offset = len(s.bs) + 1 + return + } + + s.curr = rune(s.bs[s.offset]) + s.width = 1 + + if s.curr == 0 { + s.error("illegal null character") + } else if s.curr >= utf8.RuneSelf { + s.curr, s.width = utf8.DecodeRune(s.bs[s.offset:]) + if s.curr == utf8.RuneError && s.width == 1 { + s.error("illegal utf-8 character") + } else if s.curr == bom && s.offset > 0 { + s.error("illegal byte-order mark") + } + } + + s.offset += s.width + + if s.curr == '\n' { + s.row++ + s.col = 0 + } else { + s.col++ + } +} + +func (s *Scanner) literalStart() int { + // The current offset is at the first character past the literal delimiter (#, ", `, etc.) + // Need to subtract width of first character (plus one for the delimiter). + return s.offset - (s.width + 1) +} + +// From the Go scanner (src/go/scanner/scanner.go) + +func isLetter(ch rune) bool { + return 'a' <= lower(ch) && lower(ch) <= 'z' || ch == '_' +} + +func isDigit(ch rune) bool { + return isDecimal(ch) || ch >= utf8.RuneSelf && unicode.IsDigit(ch) +} + +func isDecimal(ch rune) bool { return '0' <= ch && ch <= '9' } + +func lower(ch rune) rune { return ('a' - 'A') | ch } // returns lower-case ch iff ch is ASCII letter + +func (s *Scanner) isWhitespace() bool { + return s.curr == ' ' || s.curr == '\t' || s.curr == '\n' || s.curr == '\r' +} + +func (s *Scanner) error(reason string) { + s.errors = append(s.errors, Error{Pos: Position{ + Offset: s.offset, + Row: s.row, + Col: s.col, + }, Message: reason}) +} diff --git a/vendor/github.com/open-policy-agent/opa/ast/internal/tokens/tokens.go b/vendor/github.com/open-policy-agent/opa/ast/internal/tokens/tokens.go new file mode 100644 index 00000000..ce053c18 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/ast/internal/tokens/tokens.go @@ -0,0 +1,138 @@ +// Copyright 2020 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package tokens + +// Token represents a single Rego source code token +// for use by the Parser. +type Token int + +func (t Token) String() string { + if t < 0 || int(t) >= len(strings) { + return "unknown" + } + return strings[t] +} + +// All tokens must be defined here +const ( + Illegal Token = iota + EOF + Whitespace + Ident + Comment + + Package + Import + As + Default + Else + Not + Some + With + Null + True + False + + Number + String + + LBrack + RBrack + LBrace + RBrace + LParen + RParen + Comma + Colon + + Add + Sub + Mul + Quo + Rem + And + Or + Unify + Equal + Assign + In + Neq + Gt + Lt + Gte + Lte + Dot + Semicolon +) + +var strings = [...]string{ + Illegal: "illegal", + EOF: "eof", + Whitespace: "whitespace", + Comment: "comment", + Ident: "ident", + Package: "package", + Import: "import", + As: "as", + Default: "default", + Else: "else", + Not: "not", + Some: "some", + With: "with", + Null: "null", + True: "true", + False: "false", + Number: "number", + String: "string", + LBrack: "[", + RBrack: "]", + LBrace: "{", + RBrace: "}", + LParen: "(", + RParen: ")", + Comma: ",", + Colon: ":", + Add: "plus", + Sub: "minus", + Mul: "mul", + Quo: "div", + Rem: "rem", + And: "and", + Or: "or", + Unify: "eq", + Equal: "equal", + Assign: "assign", + In: "in", + Neq: "neq", + Gt: "gt", + Lt: "lt", + Gte: "gte", + Lte: "lte", + Dot: ".", + Semicolon: ";", +} + +var keywords = map[string]Token{ + "package": Package, + "import": Import, + "as": As, + "default": Default, + "else": Else, + "not": Not, + "some": Some, + "with": With, + "null": Null, + "true": True, + "false": False, +} + +// Keywords returns a copy of the default string -> Token keyword map. +func Keywords() map[string]Token { + cpy := make(map[string]Token, len(keywords)) + for k, v := range keywords { + cpy[k] = v + } + return cpy +} diff --git a/vendor/github.com/open-policy-agent/opa/ast/location/location.go b/vendor/github.com/open-policy-agent/opa/ast/location/location.go new file mode 100644 index 00000000..13ae6e35 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/ast/location/location.go @@ -0,0 +1,90 @@ +// Package location defines locations in Rego source code. +package location + +import ( + "bytes" + "fmt" + + "github.com/pkg/errors" +) + +// Location records a position in source code +type Location struct { + Text []byte `json:"-"` // The original text fragment from the source. + File string `json:"file"` // The name of the source file (which may be empty). + Row int `json:"row"` // The line in the source. + Col int `json:"col"` // The column in the row. + Offset int `json:"-"` // The byte offset for the location in the source. +} + +// NewLocation returns a new Location object. +func NewLocation(text []byte, file string, row int, col int) *Location { + return &Location{Text: text, File: file, Row: row, Col: col} +} + +// Equal checks if two locations are equal to each other. +func (loc *Location) Equal(other *Location) bool { + return bytes.Equal(loc.Text, other.Text) && + loc.File == other.File && + loc.Row == other.Row && + loc.Col == other.Col +} + +// Errorf returns a new error value with a message formatted to include the location +// info (e.g., line, column, filename, etc.) +func (loc *Location) Errorf(f string, a ...interface{}) error { + return errors.New(loc.Format(f, a...)) +} + +// Wrapf returns a new error value that wraps an existing error with a message formatted +// to include the location info (e.g., line, column, filename, etc.) +func (loc *Location) Wrapf(err error, f string, a ...interface{}) error { + return errors.Wrap(err, loc.Format(f, a...)) +} + +// Format returns a formatted string prefixed with the location information. +func (loc *Location) Format(f string, a ...interface{}) string { + if len(loc.File) > 0 { + f = fmt.Sprintf("%v:%v: %v", loc.File, loc.Row, f) + } else { + f = fmt.Sprintf("%v:%v: %v", loc.Row, loc.Col, f) + } + return fmt.Sprintf(f, a...) +} + +func (loc *Location) String() string { + if len(loc.File) > 0 { + return fmt.Sprintf("%v:%v", loc.File, loc.Row) + } + if len(loc.Text) > 0 { + return string(loc.Text) + } + return fmt.Sprintf("%v:%v", loc.Row, loc.Col) +} + +// Compare returns -1, 0, or 1 to indicate if this loc is less than, equal to, +// or greater than the other. Comparison is performed on the file, row, and +// column of the Location (but not on the text.) Nil locations are greater than +// non-nil locations. +func (loc *Location) Compare(other *Location) int { + if loc == nil && other == nil { + return 0 + } else if loc == nil { + return 1 + } else if other == nil { + return -1 + } else if loc.File < other.File { + return -1 + } else if loc.File > other.File { + return 1 + } else if loc.Row < other.Row { + return -1 + } else if loc.Row > other.Row { + return 1 + } else if loc.Col < other.Col { + return -1 + } else if loc.Col > other.Col { + return 1 + } + return 0 +} diff --git a/vendor/github.com/open-policy-agent/opa/ast/map.go b/vendor/github.com/open-policy-agent/opa/ast/map.go new file mode 100644 index 00000000..b0cc9eb6 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/ast/map.go @@ -0,0 +1,133 @@ +// Copyright 2016 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package ast + +import ( + "encoding/json" + + "github.com/open-policy-agent/opa/util" +) + +// ValueMap represents a key/value map between AST term values. Any type of term +// can be used as a key in the map. +type ValueMap struct { + hashMap *util.HashMap +} + +// NewValueMap returns a new ValueMap. +func NewValueMap() *ValueMap { + vs := &ValueMap{ + hashMap: util.NewHashMap(valueEq, valueHash), + } + return vs +} + +// MarshalJSON provides a custom marshaller for the ValueMap which +// will include the key, value, and value type. +func (vs *ValueMap) MarshalJSON() ([]byte, error) { + var tmp []map[string]interface{} + vs.Iter(func(k Value, v Value) bool { + tmp = append(tmp, map[string]interface{}{ + "name": k.String(), + "type": TypeName(v), + "value": v, + }) + return false + }) + return json.Marshal(tmp) +} + +// Copy returns a shallow copy of the ValueMap. +func (vs *ValueMap) Copy() *ValueMap { + if vs == nil { + return nil + } + cpy := NewValueMap() + cpy.hashMap = vs.hashMap.Copy() + return cpy +} + +// Equal returns true if this ValueMap equals the other. +func (vs *ValueMap) Equal(other *ValueMap) bool { + if vs == nil { + return other == nil || other.Len() == 0 + } + if other == nil { + return vs == nil || vs.Len() == 0 + } + return vs.hashMap.Equal(other.hashMap) +} + +// Len returns the number of elements in the map. +func (vs *ValueMap) Len() int { + if vs == nil { + return 0 + } + return vs.hashMap.Len() +} + +// Get returns the value in the map for k. +func (vs *ValueMap) Get(k Value) Value { + if vs != nil { + if v, ok := vs.hashMap.Get(k); ok { + return v.(Value) + } + } + return nil +} + +// Hash returns a hash code for this ValueMap. +func (vs *ValueMap) Hash() int { + if vs == nil { + return 0 + } + return vs.hashMap.Hash() +} + +// Iter calls the iter function for each key/value pair in the map. If the iter +// function returns true, iteration stops. +func (vs *ValueMap) Iter(iter func(Value, Value) bool) bool { + if vs == nil { + return false + } + return vs.hashMap.Iter(func(kt, vt util.T) bool { + k := kt.(Value) + v := vt.(Value) + return iter(k, v) + }) +} + +// Put inserts a key k into the map with value v. +func (vs *ValueMap) Put(k, v Value) { + if vs == nil { + panic("put on nil value map") + } + vs.hashMap.Put(k, v) +} + +// Delete removes a key k from the map. +func (vs *ValueMap) Delete(k Value) { + if vs == nil { + return + } + vs.hashMap.Delete(k) +} + +func (vs *ValueMap) String() string { + if vs == nil { + return "{}" + } + return vs.hashMap.String() +} + +func valueHash(v util.T) int { + return v.(Value).Hash() +} + +func valueEq(a, b util.T) bool { + av := a.(Value) + bv := b.(Value) + return av.Compare(bv) == 0 +} diff --git a/vendor/github.com/open-policy-agent/opa/ast/parser.go b/vendor/github.com/open-policy-agent/opa/ast/parser.go new file mode 100644 index 00000000..0990a6a5 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/ast/parser.go @@ -0,0 +1,2040 @@ +// Copyright 2020 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package ast + +import ( + "bytes" + "encoding/json" + "fmt" + "io" + "math/big" + "regexp" + "strconv" + "strings" + + "github.com/pkg/errors" + "gopkg.in/yaml.v2" + + "github.com/open-policy-agent/opa/ast/internal/scanner" + "github.com/open-policy-agent/opa/ast/internal/tokens" + "github.com/open-policy-agent/opa/ast/location" +) + +// Note: This state is kept isolated from the parser so that we +// can do efficient shallow copies of these values when doing a +// save() and restore(). +type state struct { + s *scanner.Scanner + lastEnd int + skippedNL bool + tok tokens.Token + tokEnd int + lit string + loc Location + errors Errors + hints []string + comments []*Comment + wildcard int +} + +func (s *state) String() string { + return fmt.Sprintf("", s.s, s.tok, s.lit, s.loc, len(s.errors), len(s.comments)) +} + +func (s *state) Loc() *location.Location { + cpy := s.loc + return &cpy +} + +func (s *state) Text(offset, end int) []byte { + bs := s.s.Bytes() + if offset >= 0 && offset < len(bs) { + if end >= offset && end <= len(bs) { + return bs[offset:end] + } + } + return nil +} + +// Parser is used to parse Rego statements. +type Parser struct { + r io.Reader + s *state + po ParserOptions + cache parsedTermCache +} + +type parsedTermCacheItem struct { + t *Term + post *state // post is the post-state that's restored on a cache-hit + offset int + next *parsedTermCacheItem +} + +type parsedTermCache struct { + m *parsedTermCacheItem +} + +func (c parsedTermCache) String() string { + s := strings.Builder{} + s.WriteRune('{') + var e *parsedTermCacheItem + for e = c.m; e != nil; e = e.next { + fmt.Fprintf(&s, "%v", e) + } + s.WriteRune('}') + return s.String() +} + +func (e *parsedTermCacheItem) String() string { + return fmt.Sprintf("<%d:%v>", e.offset, e.t) +} + +// ParserOptions defines the options for parsing Rego statements. +type ParserOptions struct { + Capabilities *Capabilities + ProcessAnnotation bool + AllFutureKeywords bool + FutureKeywords []string +} + +// NewParser creates and initializes a Parser. +func NewParser() *Parser { + p := &Parser{ + s: &state{}, + po: ParserOptions{}, + } + return p +} + +// WithFilename provides the filename for Location details +// on parsed statements. +func (p *Parser) WithFilename(filename string) *Parser { + p.s.loc.File = filename + return p +} + +// WithReader provides the io.Reader that the parser will +// use as its source. +func (p *Parser) WithReader(r io.Reader) *Parser { + p.r = r + return p +} + +// WithProcessAnnotation enables or disables the processing of +// annotations by the Parser +func (p *Parser) WithProcessAnnotation(processAnnotation bool) *Parser { + p.po.ProcessAnnotation = processAnnotation + return p +} + +// WithFutureKeywords enables "future" keywords, i.e., keywords that can +// be imported via +// +// import future.keywords.kw +// import future.keywords.other +// +// but in a more direct way. The equivalent of this import would be +// +// WithFutureKeywords("kw", "other") +func (p *Parser) WithFutureKeywords(kws ...string) *Parser { + p.po.FutureKeywords = kws + return p +} + +// WithAllFutureKeywords enables all "future" keywords, i.e., the +// ParserOption equivalent of +// +// import future.keywords +func (p *Parser) WithAllFutureKeywords(yes bool) *Parser { + p.po.AllFutureKeywords = yes + return p +} + +// WithCapabilities sets the capabilities structure on the parser. +func (p *Parser) WithCapabilities(c *Capabilities) *Parser { + p.po.Capabilities = c + return p +} + +const ( + annotationScopePackage = "package" + annotationScopeImport = "import" + annotationScopeRule = "rule" + annotationScopeDocument = "document" + annotationScopeSubpackages = "subpackages" +) + +func (p *Parser) parsedTermCacheLookup() (*Term, *state) { + l := p.s.loc.Offset + // stop comparing once the cached offsets are lower than l + for h := p.cache.m; h != nil && h.offset >= l; h = h.next { + if h.offset == l { + return h.t, h.post + } + } + return nil, nil +} + +func (p *Parser) parsedTermCachePush(t *Term, s0 *state) { + s1 := p.save() + o0 := s0.loc.Offset + entry := parsedTermCacheItem{t: t, post: s1, offset: o0} + + // find the first one whose offset is smaller than ours + var e *parsedTermCacheItem + for e = p.cache.m; e != nil; e = e.next { + if e.offset < o0 { + break + } + } + entry.next = e + p.cache.m = &entry +} + +// futureParser returns a shallow copy of `p` with an empty +// cache, and a scanner that knows all future keywords. +// It's used to present hints in errors, when statements would +// only parse successfully if some future keyword is enabled. +func (p *Parser) futureParser() *Parser { + q := *p + q.s = p.save() + q.s.s = p.s.s.WithKeywords(futureKeywords) + q.cache = parsedTermCache{} + return &q +} + +// Parse will read the Rego source and parse statements and +// comments as they are found. Any errors encountered while +// parsing will be accumulated and returned as a list of Errors. +func (p *Parser) Parse() ([]Statement, []*Comment, Errors) { + + if p.po.Capabilities == nil { + p.po.Capabilities = CapabilitiesForThisVersion() + } + + allowedFutureKeywords := map[string]tokens.Token{} + + for _, kw := range p.po.Capabilities.FutureKeywords { + var ok bool + allowedFutureKeywords[kw], ok = futureKeywords[kw] + if !ok { + return nil, nil, Errors{ + &Error{ + Code: ParseErr, + Message: fmt.Sprintf("illegal capabilities: unknown keyword: %v", kw), + Location: nil, + }, + } + } + } + + var err error + p.s.s, err = scanner.New(p.r) + if err != nil { + return nil, nil, Errors{ + &Error{ + Code: ParseErr, + Message: err.Error(), + Location: nil, + }, + } + } + + selected := map[string]tokens.Token{} + if p.po.AllFutureKeywords { + for kw, tok := range allowedFutureKeywords { + selected[kw] = tok + } + } else { + for _, kw := range p.po.FutureKeywords { + tok, ok := allowedFutureKeywords[kw] + if !ok { + return nil, nil, Errors{ + &Error{ + Code: ParseErr, + Message: fmt.Sprintf("unknown future keyword: %v", kw), + Location: nil, + }, + } + } + selected[kw] = tok + } + } + p.s.s = p.s.s.WithKeywords(selected) + + // read the first token to initialize the parser + p.scan() + + var stmts []Statement + + // Read from the scanner until the last token is reached or no statements + // can be parsed. Attempt to parse package statements, import statements, + // rule statements, and then body/query statements (in that order). If a + // statement cannot be parsed, restore the parser state before trying the + // next type of statement. If a statement can be parsed, continue from that + // point trying to parse packages, imports, etc. in the same order. + for p.s.tok != tokens.EOF { + + s := p.save() + + if pkg := p.parsePackage(); pkg != nil { + stmts = append(stmts, pkg) + continue + } else if len(p.s.errors) > 0 { + break + } + + p.restore(s) + s = p.save() + + if imp := p.parseImport(); imp != nil { + if FutureRootDocument.Equal(imp.Path.Value.(Ref)[0]) { + p.futureImport(imp, allowedFutureKeywords) + } + stmts = append(stmts, imp) + continue + } else if len(p.s.errors) > 0 { + break + } + + p.restore(s) + s = p.save() + + if rules := p.parseRules(); rules != nil { + for i := range rules { + stmts = append(stmts, rules[i]) + } + continue + } else if len(p.s.errors) > 0 { + break + } + + p.restore(s) + + if body := p.parseQuery(true, tokens.EOF); body != nil { + stmts = append(stmts, body) + continue + } + + break + } + + if p.po.ProcessAnnotation { + stmts = p.parseAnnotations(stmts) + } + + return stmts, p.s.comments, p.s.errors +} + +func (p *Parser) parseAnnotations(stmts []Statement) []Statement { + + var hint = []byte("METADATA") + var curr *metadataParser + var blocks []*metadataParser + + for i := 0; i < len(p.s.comments); i++ { + if curr != nil { + if p.s.comments[i].Location.Row == p.s.comments[i-1].Location.Row+1 && p.s.comments[i].Location.Col == 1 { + curr.Append(p.s.comments[i]) + continue + } + curr = nil + } + if bytes.HasPrefix(bytes.TrimSpace(p.s.comments[i].Text), hint) { + curr = newMetadataParser(p.s.comments[i].Location) + blocks = append(blocks, curr) + } + } + + for _, b := range blocks { + a, err := b.Parse() + if err != nil { + p.error(b.loc, err.Error()) + } else { + stmts = append(stmts, a) + } + } + + return stmts +} + +func (p *Parser) parsePackage() *Package { + + var pkg Package + pkg.SetLoc(p.s.Loc()) + + if p.s.tok != tokens.Package { + return nil + } + + p.scan() + if p.s.tok != tokens.Ident { + p.illegalToken() + return nil + } + + term := p.parseTerm() + + if term != nil { + switch v := term.Value.(type) { + case Var: + pkg.Path = Ref{ + DefaultRootDocument.Copy().SetLocation(term.Location), + StringTerm(string(v)).SetLocation(term.Location), + } + case Ref: + pkg.Path = make(Ref, len(v)+1) + pkg.Path[0] = DefaultRootDocument.Copy().SetLocation(v[0].Location) + first, ok := v[0].Value.(Var) + if !ok { + p.errorf(v[0].Location, "unexpected %v token: expecting var", TypeName(v[0].Value)) + return nil + } + pkg.Path[1] = StringTerm(string(first)).SetLocation(v[0].Location) + for i := 2; i < len(pkg.Path); i++ { + switch v[i-1].Value.(type) { + case String: + pkg.Path[i] = v[i-1] + default: + p.errorf(v[i-1].Location, "unexpected %v token: expecting string", TypeName(v[i-1].Value)) + return nil + } + } + default: + p.illegalToken() + return nil + } + } + + if pkg.Path == nil { + if len(p.s.errors) == 0 { + p.error(p.s.Loc(), "expected path") + } + return nil + } + + return &pkg +} + +func (p *Parser) parseImport() *Import { + + var imp Import + imp.SetLoc(p.s.Loc()) + + if p.s.tok != tokens.Import { + return nil + } + + p.scan() + if p.s.tok != tokens.Ident { + p.error(p.s.Loc(), "expected ident") + return nil + } + + term := p.parseTerm() + if term != nil { + switch v := term.Value.(type) { + case Var: + imp.Path = RefTerm(term).SetLocation(term.Location) + case Ref: + for i := 1; i < len(v); i++ { + if _, ok := v[i].Value.(String); !ok { + p.errorf(v[i].Location, "unexpected %v token: expecting string", TypeName(v[i].Value)) + return nil + } + } + imp.Path = term + } + } + + if imp.Path == nil { + p.error(p.s.Loc(), "expected path") + return nil + } + + path := imp.Path.Value.(Ref) + + if !RootDocumentNames.Contains(path[0]) && !FutureRootDocument.Equal(path[0]) { + p.errorf(imp.Path.Location, "unexpected import path, must begin with one of: %v, got: %v", + RootDocumentNames.Union(NewSet(FutureRootDocument)), + path[0]) + return nil + } + + if p.s.tok == tokens.As { + p.scan() + + if p.s.tok != tokens.Ident { + p.illegal("expected var") + return nil + } + + if alias := p.parseTerm(); alias != nil { + v, ok := alias.Value.(Var) + if ok { + imp.Alias = v + return &imp + } + } + p.illegal("expected var") + return nil + } + + return &imp +} + +func (p *Parser) parseRules() []*Rule { + + var rule Rule + rule.SetLoc(p.s.Loc()) + + if p.s.tok == tokens.Default { + p.scan() + rule.Default = true + } + + if p.s.tok != tokens.Ident { + return nil + } + + if rule.Head = p.parseHead(rule.Default); rule.Head == nil { + return nil + } + + if rule.Default { + if !p.validateDefaultRuleValue(&rule) { + return nil + } + + rule.Body = NewBody(NewExpr(BooleanTerm(true).SetLocation(rule.Location)).SetLocation(rule.Location)) + return []*Rule{&rule} + } + + if p.s.tok == tokens.LBrace { + p.scan() + if rule.Body = p.parseBody(tokens.RBrace); rule.Body == nil { + return nil + } + p.scan() + } else { + return nil + } + + if p.s.tok == tokens.Else { + + if rule.Head.Assign { + p.error(p.s.Loc(), "else keyword cannot be used on rule declared with := operator") + return nil + } + + if rule.Head.Key != nil { + p.error(p.s.Loc(), "else keyword cannot be used on partial rules") + return nil + } + + if rule.Else = p.parseElse(rule.Head); rule.Else == nil { + return nil + } + } + + rule.Location.Text = p.s.Text(rule.Location.Offset, p.s.lastEnd) + + var rules []*Rule + + rules = append(rules, &rule) + + for p.s.tok == tokens.LBrace { + + if rule.Else != nil { + p.error(p.s.Loc(), "expected else keyword") + return nil + } + + loc := p.s.Loc() + + p.scan() + var next Rule + + if next.Body = p.parseBody(tokens.RBrace); next.Body == nil { + return nil + } + p.scan() + + loc.Text = p.s.Text(loc.Offset, p.s.lastEnd) + next.SetLoc(loc) + + // Chained rule head's keep the original + // rule's head AST but have their location + // set to the rule body. + next.Head = rule.Head.Copy() + setLocRecursive(next.Head, loc) + + rules = append(rules, &next) + } + + return rules +} + +func (p *Parser) parseElse(head *Head) *Rule { + + var rule Rule + rule.SetLoc(p.s.Loc()) + + rule.Head = head.Copy() + rule.Head.SetLoc(p.s.Loc()) + + defer func() { + rule.Location.Text = p.s.Text(rule.Location.Offset, p.s.lastEnd) + }() + + p.scan() + + switch p.s.tok { + case tokens.LBrace: + rule.Head.Value = BooleanTerm(true) + case tokens.Unify: + p.scan() + rule.Head.Value = p.parseTermInfixCall() + if rule.Head.Value == nil { + return nil + } + rule.Head.Location.Text = p.s.Text(rule.Head.Location.Offset, p.s.lastEnd) + default: + p.illegal("expected else value term or rule body") + return nil + } + + if p.s.tok != tokens.LBrace { + rule.Body = NewBody(NewExpr(BooleanTerm(true))) + setLocRecursive(rule.Body, rule.Location) + return &rule + } + + p.scan() + + if rule.Body = p.parseBody(tokens.RBrace); rule.Body == nil { + return nil + } + + p.scan() + + if p.s.tok == tokens.Else { + if rule.Else = p.parseElse(head); rule.Else == nil { + return nil + } + } + return &rule +} + +func (p *Parser) parseHead(defaultRule bool) *Head { + + var head Head + head.SetLoc(p.s.Loc()) + + defer func() { + head.Location.Text = p.s.Text(head.Location.Offset, p.s.lastEnd) + }() + + if term := p.parseVar(); term != nil { + head.Name = term.Value.(Var) + } else { + p.illegal("expected rule head name") + } + + p.scan() + + if p.s.tok == tokens.LParen { + p.scan() + if p.s.tok != tokens.RParen { + head.Args = p.parseTermList(tokens.RParen, nil) + if head.Args == nil { + return nil + } + } + p.scan() + + if p.s.tok == tokens.LBrack { + return nil + } + } + + if p.s.tok == tokens.LBrack { + p.scan() + head.Key = p.parseTermInfixCall() + if head.Key == nil { + p.illegal("expected rule key term (e.g., %s[] { ... })", head.Name) + } + if p.s.tok != tokens.RBrack { + if _, ok := futureKeywords[head.Name.String()]; ok { + p.hint("`import future.keywords.%[1]s` for '%[1]s' keyword", head.Name.String()) + } + p.illegal("non-terminated rule key") + } + p.scan() + } + + if p.s.tok == tokens.Unify { + p.scan() + head.Value = p.parseTermInfixCall() + if head.Value == nil { + p.illegal("expected rule value term (e.g., %s[%s] = { ... })", head.Name, head.Key) + } + } else if p.s.tok == tokens.Assign { + + if defaultRule { + p.error(p.s.Loc(), "default rules must use = operator (not := operator)") + return nil + } else if head.Key != nil { + p.error(p.s.Loc(), "partial rules must use = operator (not := operator)") + return nil + } else if len(head.Args) > 0 { + p.error(p.s.Loc(), "functions must use = operator (not := operator)") + return nil + } + + p.scan() + head.Assign = true + head.Value = p.parseTermInfixCall() + if head.Value == nil { + p.illegal("expected rule value term (e.g., %s := { ... })", head.Name) + } + } + + if head.Value == nil && head.Key == nil { + head.Value = BooleanTerm(true).SetLocation(head.Location) + } + + return &head +} + +func (p *Parser) parseBody(end tokens.Token) Body { + return p.parseQuery(false, end) +} + +func (p *Parser) parseQuery(requireSemi bool, end tokens.Token) Body { + body := Body{} + + if p.s.tok == end { + p.error(p.s.Loc(), "found empty body") + return nil + } + + for { + + expr := p.parseLiteral() + if expr == nil { + return nil + } + + body.Append(expr) + + if p.s.tok == tokens.Semicolon { + p.scan() + continue + } + + if p.s.tok == end || requireSemi { + return body + } + + if !p.s.skippedNL { + // If there was already an error then don't pile this one on + if len(p.s.errors) == 0 { + p.illegal(`expected \n or %s or %s`, tokens.Semicolon, end) + } + return nil + } + } +} + +func (p *Parser) parseLiteral() (expr *Expr) { + + offset := p.s.loc.Offset + loc := p.s.Loc() + + defer func() { + if expr != nil { + loc.Text = p.s.Text(offset, p.s.lastEnd) + expr.SetLoc(loc) + } + }() + + var negated bool + switch p.s.tok { + case tokens.Some: + return p.parseSome() + case tokens.Not: + p.scan() + negated = true + fallthrough + default: + expr := p.parseExpr() + if expr != nil { + expr.Negated = negated + if p.s.tok == tokens.With { + if expr.With = p.parseWith(); expr.With == nil { + return nil + } + } + return expr + } + return nil + } +} + +func (p *Parser) parseWith() []*With { + + withs := []*With{} + + for { + + with := With{ + Location: p.s.Loc(), + } + p.scan() + + if p.s.tok != tokens.Ident { + p.illegal("expected ident") + return nil + } + + if with.Target = p.parseTerm(); with.Target == nil { + return nil + } + + switch with.Target.Value.(type) { + case Ref, Var: + break + default: + p.illegal("expected with target path") + } + + if p.s.tok != tokens.As { + p.illegal("expected as keyword") + return nil + } + + p.scan() + + if with.Value = p.parseTermInfixCall(); with.Value == nil { + return nil + } + + with.Location.Text = p.s.Text(with.Location.Offset, p.s.lastEnd) + + withs = append(withs, &with) + + if p.s.tok != tokens.With { + break + } + } + + return withs +} + +func (p *Parser) parseSome() *Expr { + + decl := &SomeDecl{} + decl.SetLoc(p.s.Loc()) + + // Attempt to parse "some x in xs", which will end up in + // SomeDecl{Symbols: ["member(x, xs)"]} + s := p.save() + p.scan() + if term := p.parseTermInfixCall(); term != nil { + if call, ok := term.Value.(Call); ok { + switch call[0].String() { + case Member.Name, MemberWithKey.Name: // OK + default: + p.illegal("expected `x in xs` or `x, y in xs` expression") + return nil + } + + decl.Symbols = []*Term{term} + return NewExpr(decl).SetLocation(decl.Location) + } + } + + p.restore(s) + s = p.save() // new copy for later + var hint bool + p.scan() + if term := p.futureParser().parseTermInfixCall(); term != nil { + if call, ok := term.Value.(Call); ok { + switch call[0].String() { + case Member.Name, MemberWithKey.Name: + hint = true + } + } + } + + // go on as before, it's `some x[...]` or illegal + p.restore(s) + if hint { + p.hint("`import future.keywords.in` for `some x in xs` expressions") + } + + for { // collecting var args + + p.scan() + + if p.s.tok != tokens.Ident { + p.illegal("expected var") + return nil + } + + decl.Symbols = append(decl.Symbols, p.parseVar()) + + p.scan() + + if p.s.tok != tokens.Comma { + break + } + } + + return NewExpr(decl).SetLocation(decl.Location) +} + +func (p *Parser) parseExpr() *Expr { + + lhs := p.parseTermInfixCall() + if lhs == nil { + return nil + } + + if op := p.parseTermOp(tokens.Assign, tokens.Unify); op != nil { + if rhs := p.parseTermInfixCall(); rhs != nil { + return NewExpr([]*Term{op, lhs, rhs}) + } + return nil + } + + // NOTE(tsandall): the top-level call term is converted to an expr because + // the evaluator does not support the call term type (nested calls are + // rewritten by the compiler.) + if call, ok := lhs.Value.(Call); ok { + return NewExpr([]*Term(call)) + } + + return NewExpr(lhs) +} + +// parseTermInfixCall consumes the next term from the input and returns it. If a +// term cannot be parsed the return value is nil and error will be recorded. The +// scanner will be advanced to the next token before returning. +// By starting out with infix relations (==, !=, <, etc) and further calling the +// other binary operators (|, &, arithmetics), it constitutes the binding +// precedence. +func (p *Parser) parseTermInfixCall() *Term { + return p.parseTermIn(nil, true, p.s.loc.Offset) +} + +func (p *Parser) parseTermInfixCallInList() *Term { + return p.parseTermIn(nil, false, p.s.loc.Offset) +} + +func (p *Parser) parseTermIn(lhs *Term, keyVal bool, offset int) *Term { + // NOTE(sr): `in` is a bit special: besides `lhs in rhs`, it also + // supports `key, val in rhs`, so it can have an optional second lhs. + // `keyVal` triggers if we attempt to parse a second lhs argument (`mhs`). + if lhs == nil { + lhs = p.parseTermRelation(nil, offset) + } + if lhs != nil { + if keyVal && p.s.tok == tokens.Comma { // second "lhs", or "middle hand side" + s := p.save() + p.scan() + if mhs := p.parseTermRelation(nil, offset); mhs != nil { + if op := p.parseTermOpName(MemberWithKey.Ref(), tokens.In); op != nil { + if rhs := p.parseTermRelation(nil, p.s.loc.Offset); rhs != nil { + call := p.setLoc(CallTerm(op, lhs, mhs, rhs), lhs.Location, offset, p.s.lastEnd) + switch p.s.tok { + case tokens.In: + return p.parseTermIn(call, keyVal, offset) + default: + return call + } + } + } + } + p.restore(s) + return nil + } + if op := p.parseTermOpName(Member.Ref(), tokens.In); op != nil { + if rhs := p.parseTermRelation(nil, p.s.loc.Offset); rhs != nil { + call := p.setLoc(CallTerm(op, lhs, rhs), lhs.Location, offset, p.s.lastEnd) + switch p.s.tok { + case tokens.In: + return p.parseTermIn(call, keyVal, offset) + default: + return call + } + } + } + } + return lhs +} + +func (p *Parser) parseTermRelation(lhs *Term, offset int) *Term { + if lhs == nil { + lhs = p.parseTermOr(nil, offset) + } + if lhs != nil { + if op := p.parseTermOp(tokens.Equal, tokens.Neq, tokens.Lt, tokens.Gt, tokens.Lte, tokens.Gte); op != nil { + if rhs := p.parseTermOr(nil, p.s.loc.Offset); rhs != nil { + call := p.setLoc(CallTerm(op, lhs, rhs), lhs.Location, offset, p.s.lastEnd) + switch p.s.tok { + case tokens.Equal, tokens.Neq, tokens.Lt, tokens.Gt, tokens.Lte, tokens.Gte: + return p.parseTermRelation(call, offset) + default: + return call + } + } + } + } + return lhs +} + +func (p *Parser) parseTermOr(lhs *Term, offset int) *Term { + if lhs == nil { + lhs = p.parseTermAnd(nil, offset) + } + if lhs != nil { + if op := p.parseTermOp(tokens.Or); op != nil { + if rhs := p.parseTermAnd(nil, p.s.loc.Offset); rhs != nil { + call := p.setLoc(CallTerm(op, lhs, rhs), lhs.Location, offset, p.s.lastEnd) + switch p.s.tok { + case tokens.Or: + return p.parseTermOr(call, offset) + default: + return call + } + } + } + return lhs + } + return nil +} + +func (p *Parser) parseTermAnd(lhs *Term, offset int) *Term { + if lhs == nil { + lhs = p.parseTermArith(nil, offset) + } + if lhs != nil { + if op := p.parseTermOp(tokens.And); op != nil { + if rhs := p.parseTermArith(nil, p.s.loc.Offset); rhs != nil { + call := p.setLoc(CallTerm(op, lhs, rhs), lhs.Location, offset, p.s.lastEnd) + switch p.s.tok { + case tokens.And: + return p.parseTermAnd(call, offset) + default: + return call + } + } + } + return lhs + } + return nil +} + +func (p *Parser) parseTermArith(lhs *Term, offset int) *Term { + if lhs == nil { + lhs = p.parseTermFactor(nil, offset) + } + if lhs != nil { + if op := p.parseTermOp(tokens.Add, tokens.Sub); op != nil { + if rhs := p.parseTermFactor(nil, p.s.loc.Offset); rhs != nil { + call := p.setLoc(CallTerm(op, lhs, rhs), lhs.Location, offset, p.s.lastEnd) + switch p.s.tok { + case tokens.Add, tokens.Sub: + return p.parseTermArith(call, offset) + default: + return call + } + } + } + } + return lhs +} + +func (p *Parser) parseTermFactor(lhs *Term, offset int) *Term { + if lhs == nil { + lhs = p.parseTerm() + } + if lhs != nil { + if op := p.parseTermOp(tokens.Mul, tokens.Quo, tokens.Rem); op != nil { + if rhs := p.parseTerm(); rhs != nil { + call := p.setLoc(CallTerm(op, lhs, rhs), lhs.Location, offset, p.s.lastEnd) + switch p.s.tok { + case tokens.Mul, tokens.Quo, tokens.Rem: + return p.parseTermFactor(call, offset) + default: + return call + } + } + } + } + return lhs +} + +func (p *Parser) parseTerm() *Term { + if term, s := p.parsedTermCacheLookup(); s != nil { + p.restore(s) + return term + } + s0 := p.save() + + var term *Term + switch p.s.tok { + case tokens.Null: + term = NullTerm().SetLocation(p.s.Loc()) + case tokens.True: + term = BooleanTerm(true).SetLocation(p.s.Loc()) + case tokens.False: + term = BooleanTerm(false).SetLocation(p.s.Loc()) + case tokens.Sub, tokens.Dot, tokens.Number: + term = p.parseNumber() + case tokens.String: + term = p.parseString() + case tokens.Ident: + term = p.parseVar() + case tokens.LBrack: + term = p.parseArray() + case tokens.LBrace: + term = p.parseSetOrObject() + case tokens.LParen: + offset := p.s.loc.Offset + p.scan() + if r := p.parseTermInfixCall(); r != nil { + if p.s.tok == tokens.RParen { + r.Location.Text = p.s.Text(offset, p.s.tokEnd) + term = r + } else { + p.error(p.s.Loc(), "non-terminated expression") + } + } + default: + p.illegalToken() + } + + term = p.parseTermFinish(term) + p.parsedTermCachePush(term, s0) + return term +} + +func (p *Parser) parseTermFinish(head *Term) *Term { + if head == nil { + return nil + } + offset := p.s.loc.Offset + p.scanWS() + switch p.s.tok { + case tokens.LParen, tokens.Dot, tokens.LBrack: + return p.parseRef(head, offset) + case tokens.Whitespace: + p.scan() + fallthrough + default: + if _, ok := head.Value.(Var); ok && RootDocumentNames.Contains(head) { + return RefTerm(head).SetLocation(head.Location) + } + return head + } +} + +func (p *Parser) parseNumber() *Term { + var prefix string + loc := p.s.Loc() + if p.s.tok == tokens.Sub { + prefix = "-" + p.scan() + switch p.s.tok { + case tokens.Number, tokens.Dot: + break + default: + p.illegal("expected number") + return nil + } + } + if p.s.tok == tokens.Dot { + prefix += "." + p.scan() + if p.s.tok != tokens.Number { + p.illegal("expected number") + return nil + } + } + + // Check for multiple leading 0's, parsed by math/big.Float.Parse as decimal 0: + // https://golang.org/pkg/math/big/#Float.Parse + if ((len(prefix) != 0 && prefix[0] == '-') || len(prefix) == 0) && + len(p.s.lit) > 1 && p.s.lit[0] == '0' && p.s.lit[1] == '0' { + p.illegal("expected number") + return nil + } + + // Ensure that the number is valid + s := prefix + p.s.lit + f, ok := new(big.Float).SetString(s) + if !ok { + p.illegal("invalid float") + return nil + } + + // Put limit on size of exponent to prevent non-linear cost of String() + // function on big.Float from causing denial of service: https://github.com/golang/go/issues/11068 + // + // n == sign * mantissa * 2^exp + // 0.5 <= mantissa < 1.0 + // + // The limit is arbitrary. + exp := f.MantExp(nil) + if exp > 1e5 || exp < -1e5 || f.IsInf() { // +/- inf, exp is 0 + p.error(p.s.Loc(), "number too big") + return nil + } + + // Note: Use the original string, do *not* round trip from + // the big.Float as it can cause precision loss. + r := NumberTerm(json.Number(s)).SetLocation(loc) + return r +} + +func (p *Parser) parseString() *Term { + if p.s.lit[0] == '"' { + var s string + err := json.Unmarshal([]byte(p.s.lit), &s) + if err != nil { + p.errorf(p.s.Loc(), "illegal string literal: %s", p.s.lit) + return nil + } + term := StringTerm(s).SetLocation(p.s.Loc()) + return term + } + return p.parseRawString() +} + +func (p *Parser) parseRawString() *Term { + if len(p.s.lit) < 2 { + return nil + } + term := StringTerm(p.s.lit[1 : len(p.s.lit)-1]).SetLocation(p.s.Loc()) + return term +} + +// this is the name to use for instantiating an empty set, e.g., `set()`. +var setConstructor = RefTerm(VarTerm("set")) + +func (p *Parser) parseCall(operator *Term, offset int) (term *Term) { + + loc := operator.Location + var end int + + defer func() { + p.setLoc(term, loc, offset, end) + }() + + p.scan() // steps over '(' + + if p.s.tok == tokens.RParen { // no args, i.e. set() or any.func() + end = p.s.tokEnd + p.scanWS() + if operator.Equal(setConstructor) { + return SetTerm() + } + return CallTerm(operator) + } + + if r := p.parseTermList(tokens.RParen, []*Term{operator}); r != nil { + end = p.s.tokEnd + p.scanWS() + return CallTerm(r...) + } + + return nil +} + +func (p *Parser) parseRef(head *Term, offset int) (term *Term) { + + loc := head.Location + var end int + + defer func() { + p.setLoc(term, loc, offset, end) + }() + + switch h := head.Value.(type) { + case Var, *Array, Object, Set, *ArrayComprehension, *ObjectComprehension, *SetComprehension, Call: + // ok + default: + p.errorf(loc, "illegal ref (head cannot be %v)", TypeName(h)) + } + + ref := []*Term{head} + + for { + switch p.s.tok { + case tokens.Dot: + p.scanWS() + if p.s.tok != tokens.Ident { + p.illegal("expected %v", tokens.Ident) + return nil + } + ref = append(ref, StringTerm(p.s.lit).SetLocation(p.s.Loc())) + p.scanWS() + case tokens.LParen: + term = p.parseCall(p.setLoc(RefTerm(ref...), loc, offset, p.s.loc.Offset), offset) + if term != nil { + switch p.s.tok { + case tokens.Whitespace: + p.scan() + end = p.s.lastEnd + return term + case tokens.Dot, tokens.LBrack: + term = p.parseRef(term, offset) + } + } + end = p.s.tokEnd + return term + case tokens.LBrack: + p.scan() + if term := p.parseTermInfixCall(); term != nil { + if p.s.tok != tokens.RBrack { + p.illegal("expected %v", tokens.LBrack) + return nil + } + ref = append(ref, term) + p.scanWS() + } else { + return nil + } + case tokens.Whitespace: + end = p.s.lastEnd + p.scan() + return RefTerm(ref...) + default: + end = p.s.lastEnd + return RefTerm(ref...) + } + } +} + +func (p *Parser) parseArray() (term *Term) { + + loc := p.s.Loc() + offset := p.s.loc.Offset + + defer func() { + p.setLoc(term, loc, offset, p.s.tokEnd) + }() + + p.scan() + + if p.s.tok == tokens.RBrack { + return ArrayTerm() + } + + potentialComprehension := true + + // Skip leading commas, eg [, x, y] + // Supported for backwards compatibility. In the future + // we should make this a parse error. + if p.s.tok == tokens.Comma { + potentialComprehension = false + p.scan() + } + + s := p.save() + + // NOTE(tsandall): The parser cannot attempt a relational term here because + // of ambiguity around comprehensions. For example, given: + // + // {1 | 1} + // + // Does this represent a set comprehension or a set containing binary OR + // call? We resolve the ambiguity by prioritizing comprehensions. + head := p.parseTerm() + + if head == nil { + return nil + } + + switch p.s.tok { + case tokens.RBrack: + return ArrayTerm(head) + case tokens.Comma: + p.scan() + if terms := p.parseTermList(tokens.RBrack, []*Term{head}); terms != nil { + return NewTerm(NewArray(terms...)) + } + return nil + case tokens.Or: + if potentialComprehension { + // Try to parse as if it is an array comprehension + p.scan() + if body := p.parseBody(tokens.RBrack); body != nil { + return ArrayComprehensionTerm(head, body) + } + if p.s.tok != tokens.Comma { + return nil + } + } + // fall back to parsing as a normal array definition + } + + p.restore(s) + + if terms := p.parseTermList(tokens.RBrack, nil); terms != nil { + return NewTerm(NewArray(terms...)) + } + return nil +} + +func (p *Parser) parseSetOrObject() (term *Term) { + loc := p.s.Loc() + offset := p.s.loc.Offset + + defer func() { + p.setLoc(term, loc, offset, p.s.tokEnd) + }() + + p.scan() + + if p.s.tok == tokens.RBrace { + return ObjectTerm() + } + + potentialComprehension := true + + // Skip leading commas, eg {, x, y} + // Supported for backwards compatibility. In the future + // we should make this a parse error. + if p.s.tok == tokens.Comma { + potentialComprehension = false + p.scan() + } + + s := p.save() + + // Try parsing just a single term first to give comprehensions higher + // priority to "or" calls in ambiguous situations. Eg: { a | b } + // will be a set comprehension. + // + // Note: We don't know yet if it is a set or object being defined. + head := p.parseTerm() + if head == nil { + return nil + } + + switch p.s.tok { + case tokens.Or: + if potentialComprehension { + return p.parseSet(s, head, potentialComprehension) + } + case tokens.RBrace, tokens.Comma: + return p.parseSet(s, head, potentialComprehension) + case tokens.Colon: + return p.parseObject(head, potentialComprehension) + } + + p.restore(s) + + head = p.parseTermInfixCallInList() + if head == nil { + return nil + } + + switch p.s.tok { + case tokens.RBrace, tokens.Comma: + return p.parseSet(s, head, false) + case tokens.Colon: + // It still might be an object comprehension, eg { a+1: b | ... } + return p.parseObject(head, potentialComprehension) + } + + p.illegal("non-terminated set") + return nil +} + +func (p *Parser) parseSet(s *state, head *Term, potentialComprehension bool) *Term { + switch p.s.tok { + case tokens.RBrace: + return SetTerm(head) + case tokens.Comma: + p.scan() + if terms := p.parseTermList(tokens.RBrace, []*Term{head}); terms != nil { + return SetTerm(terms...) + } + case tokens.Or: + if potentialComprehension { + // Try to parse as if it is a set comprehension + p.scan() + if body := p.parseBody(tokens.RBrace); body != nil { + return SetComprehensionTerm(head, body) + } + if p.s.tok != tokens.Comma { + return nil + } + } + // Fall back to parsing as normal set definition + p.restore(s) + if terms := p.parseTermList(tokens.RBrace, nil); terms != nil { + return SetTerm(terms...) + } + } + return nil +} + +func (p *Parser) parseObject(k *Term, potentialComprehension bool) *Term { + // NOTE(tsandall): Assumption: this function is called after parsing the key + // of the head element and then receiving a colon token from the scanner. + // Advance beyond the colon and attempt to parse an object. + if p.s.tok != tokens.Colon { + panic("expected colon") + } + p.scan() + + s := p.save() + + // NOTE(sr): We first try to parse the value as a term (`v`), and see + // if we can parse `{ x: v | ...}` as a comprehension. + // However, if we encounter either a Comma or an RBace, it cannot be + // parsed as a comprehension -- so we save double work further down + // where `parseObjectFinish(k, v, false)` would only exercise the + // same code paths once more. + v := p.parseTerm() + if v == nil { + return nil + } + + potentialRelation := true + if potentialComprehension { + switch p.s.tok { + case tokens.RBrace, tokens.Comma: + potentialRelation = false + fallthrough + case tokens.Or: + if term := p.parseObjectFinish(k, v, true); term != nil { + return term + } + } + } + + p.restore(s) + + if potentialRelation { + v := p.parseTermInfixCallInList() + if v == nil { + return nil + } + + switch p.s.tok { + case tokens.RBrace, tokens.Comma: + return p.parseObjectFinish(k, v, false) + } + } + + p.illegal("non-terminated object") + return nil +} + +func (p *Parser) parseObjectFinish(key, val *Term, potentialComprehension bool) *Term { + switch p.s.tok { + case tokens.RBrace: + return ObjectTerm([2]*Term{key, val}) + case tokens.Or: + if potentialComprehension { + p.scan() + if body := p.parseBody(tokens.RBrace); body != nil { + return ObjectComprehensionTerm(key, val, body) + } + } else { + p.illegal("non-terminated object") + } + case tokens.Comma: + p.scan() + if r := p.parseTermPairList(tokens.RBrace, [][2]*Term{{key, val}}); r != nil { + return ObjectTerm(r...) + } + } + return nil +} + +func (p *Parser) parseTermList(end tokens.Token, r []*Term) []*Term { + if p.s.tok == end { + return r + } + for { + term := p.parseTermInfixCallInList() + if term != nil { + r = append(r, term) + switch p.s.tok { + case end: + return r + case tokens.Comma: + p.scan() + if p.s.tok == end { + return r + } + continue + default: + p.illegal(fmt.Sprintf("expected %q or %q", tokens.Comma, end)) + return nil + } + } + return nil + } +} + +func (p *Parser) parseTermPairList(end tokens.Token, r [][2]*Term) [][2]*Term { + if p.s.tok == end { + return r + } + for { + key := p.parseTermInfixCallInList() + if key != nil { + switch p.s.tok { + case tokens.Colon: + p.scan() + if val := p.parseTermInfixCallInList(); val != nil { + r = append(r, [2]*Term{key, val}) + switch p.s.tok { + case end: + return r + case tokens.Comma: + p.scan() + if p.s.tok == end { + return r + } + continue + default: + p.illegal(fmt.Sprintf("expected %q or %q", tokens.Comma, end)) + return nil + } + } + default: + p.illegal(fmt.Sprintf("expected %q", tokens.Colon)) + return nil + } + } + return nil + } +} + +func (p *Parser) parseTermOp(values ...tokens.Token) *Term { + for i := range values { + if p.s.tok == values[i] { + r := RefTerm(VarTerm(fmt.Sprint(p.s.tok)).SetLocation(p.s.Loc())).SetLocation(p.s.Loc()) + p.scan() + return r + } + } + return nil +} + +func (p *Parser) parseTermOpName(ref Ref, values ...tokens.Token) *Term { + for i := range values { + if p.s.tok == values[i] { + for _, r := range ref { + r.SetLocation(p.s.Loc()) + } + t := RefTerm(ref...) + t.SetLocation(p.s.Loc()) + p.scan() + return t + } + } + return nil +} + +func (p *Parser) parseVar() *Term { + + s := p.s.lit + + term := VarTerm(s).SetLocation(p.s.Loc()) + + // Update wildcard values with unique identifiers + if term.Equal(Wildcard) { + term.Value = Var(p.genwildcard()) + } + + return term +} + +func (p *Parser) genwildcard() string { + c := p.s.wildcard + p.s.wildcard++ + return fmt.Sprintf("%v%d", WildcardPrefix, c) +} + +func (p *Parser) error(loc *location.Location, reason string) { + p.errorf(loc, reason) +} + +func (p *Parser) errorf(loc *location.Location, f string, a ...interface{}) { + msg := strings.Builder{} + fmt.Fprintf(&msg, f, a...) + + switch len(p.s.hints) { + case 0: // nothing to do + case 1: + msg.WriteString(" (hint: ") + msg.WriteString(p.s.hints[0]) + msg.WriteRune(')') + default: + msg.WriteString(" (hints: ") + for i, h := range p.s.hints { + if i > 0 { + msg.WriteString(", ") + } + msg.WriteString(h) + } + msg.WriteRune(')') + } + + p.s.errors = append(p.s.errors, &Error{ + Code: ParseErr, + Message: msg.String(), + Location: loc, + Details: newParserErrorDetail(p.s.s.Bytes(), loc.Offset), + }) + p.s.hints = nil +} + +func (p *Parser) hint(f string, a ...interface{}) { + p.s.hints = append(p.s.hints, fmt.Sprintf(f, a...)) +} + +func (p *Parser) illegal(note string, a ...interface{}) { + tok := p.s.tok.String() + + if p.s.tok == tokens.Illegal { + p.errorf(p.s.Loc(), "illegal token") + return + } + + tokType := "token" + if p.s.tok >= tokens.Package && p.s.tok <= tokens.False { + tokType = "keyword" + } + + note = fmt.Sprintf(note, a...) + if len(note) > 0 { + p.errorf(p.s.Loc(), "unexpected %s %s: %s", tok, tokType, note) + } else { + p.errorf(p.s.Loc(), "unexpected %s %s", tok, tokType) + } +} + +func (p *Parser) illegalToken() { + p.illegal("") +} + +func (p *Parser) scan() { + p.doScan(true) +} + +func (p *Parser) scanWS() { + p.doScan(false) +} + +func (p *Parser) doScan(skipws bool) { + + // NOTE(tsandall): the last position is used to compute the "text" field for + // complex AST nodes. Whitespace never affects the last position of an AST + // node so do not update it when scanning. + if p.s.tok != tokens.Whitespace { + p.s.lastEnd = p.s.tokEnd + p.s.skippedNL = false + } + + var errs []scanner.Error + for { + var pos scanner.Position + p.s.tok, pos, p.s.lit, errs = p.s.s.Scan() + + p.s.tokEnd = pos.End + p.s.loc.Row = pos.Row + p.s.loc.Col = pos.Col + p.s.loc.Offset = pos.Offset + p.s.loc.Text = p.s.Text(pos.Offset, pos.End) + + for _, err := range errs { + p.error(p.s.Loc(), err.Message) + } + + if len(errs) > 0 { + p.s.tok = tokens.Illegal + } + + if p.s.tok == tokens.Whitespace { + if p.s.lit == "\n" { + p.s.skippedNL = true + } + if skipws { + continue + } + } + + if p.s.tok != tokens.Comment { + break + } + + // For backwards compatibility leave a nil + // Text value if there is no text rather than + // an empty string. + var commentText []byte + if len(p.s.lit) > 1 { + commentText = []byte(p.s.lit[1:]) + } + comment := NewComment(commentText) + comment.SetLoc(p.s.Loc()) + p.s.comments = append(p.s.comments, comment) + } +} + +func (p *Parser) save() *state { + cpy := *p.s + s := *cpy.s + cpy.s = &s + return &cpy +} + +func (p *Parser) restore(s *state) { + p.s = s +} + +func setLocRecursive(x interface{}, loc *location.Location) { + NewGenericVisitor(func(x interface{}) bool { + if node, ok := x.(Node); ok { + node.SetLoc(loc) + } + return false + }).Walk(x) +} + +func (p *Parser) setLoc(term *Term, loc *location.Location, offset, end int) *Term { + if term != nil { + cpy := *loc + term.Location = &cpy + term.Location.Text = p.s.Text(offset, end) + } + return term +} + +func (p *Parser) validateDefaultRuleValue(rule *Rule) bool { + if rule.Head.Value == nil { + p.error(rule.Loc(), "illegal default rule (must have a value)") + return false + } + + valid := true + vis := NewGenericVisitor(func(x interface{}) bool { + switch x.(type) { + case *ArrayComprehension, *ObjectComprehension, *SetComprehension: // skip closures + return true + case Ref, Var, Call: + p.error(rule.Loc(), fmt.Sprintf("illegal default rule (value cannot contain %v)", TypeName(x))) + valid = false + return true + } + return false + }) + + vis.Walk(rule.Head.Value.Value) + return valid +} + +type rawAnnotation struct { + Scope string `json:"scope"` + Schemas []rawSchemaAnnotation `json:"schemas"` +} + +type rawSchemaAnnotation map[string]interface{} + +type metadataParser struct { + buf *bytes.Buffer + comments []*Comment + loc *location.Location +} + +func newMetadataParser(loc *Location) *metadataParser { + return &metadataParser{loc: loc, buf: bytes.NewBuffer(nil)} +} + +func (b *metadataParser) Append(c *Comment) { + b.buf.Write(bytes.TrimPrefix(c.Text, []byte(" "))) + b.buf.WriteByte('\n') + b.comments = append(b.comments, c) +} + +var yamlLineErrRegex = regexp.MustCompile(`^yaml: line ([[:digit:]]+):`) + +func (b *metadataParser) Parse() (*Annotations, error) { + + var raw rawAnnotation + + if len(bytes.TrimSpace(b.buf.Bytes())) == 0 { + return nil, fmt.Errorf("expected METADATA block, found whitespace") + } + + if err := yaml.Unmarshal(b.buf.Bytes(), &raw); err != nil { + match := yamlLineErrRegex.FindStringSubmatch(err.Error()) + if len(match) == 2 { + n, err2 := strconv.Atoi(match[1]) + if err2 == nil { + index := n - 1 // line numbering is 1-based so subtract one from row + if index >= len(b.comments) { + b.loc = b.comments[len(b.comments)-1].Location + } else { + b.loc = b.comments[index].Location + } + } + } + return nil, err + } + + var result Annotations + result.Scope = raw.Scope + + for _, pair := range raw.Schemas { + var k string + var v interface{} + for k, v = range pair { + } + + var a SchemaAnnotation + var err error + + a.Path, err = ParseRef(k) + if err != nil { + return nil, fmt.Errorf("invalid document reference") + } + + switch v := v.(type) { + case string: + a.Schema, err = parseSchemaRef(v) + if err != nil { + return nil, err + } + case map[interface{}]interface{}: + w, err := convertYAMLMapKeyTypes(v, nil) + if err != nil { + return nil, errors.Wrap(err, "invalid schema definition") + } + a.Definition = &w + default: + return nil, fmt.Errorf("invalid schema declaration for path %q", k) + } + + result.Schemas = append(result.Schemas, &a) + } + + result.Location = b.loc + return &result, nil +} + +var errInvalidSchemaRef = fmt.Errorf("invalid schema reference") + +// NOTE(tsandall): 'schema' is not registered as a root because it's not +// supported by the compiler or evaluator today. Once we fix that, we can remove +// this function. +func parseSchemaRef(s string) (Ref, error) { + + term, err := ParseTerm(s) + if err == nil { + switch v := term.Value.(type) { + case Var: + if term.Equal(SchemaRootDocument) { + return SchemaRootRef.Copy(), nil + } + case Ref: + if v.HasPrefix(SchemaRootRef) { + return v, nil + } + } + } + + return nil, errInvalidSchemaRef +} + +func convertYAMLMapKeyTypes(x interface{}, path []string) (interface{}, error) { + var err error + switch x := x.(type) { + case map[interface{}]interface{}: + result := make(map[string]interface{}, len(x)) + for k, v := range x { + str, ok := k.(string) + if !ok { + return nil, fmt.Errorf("invalid map key type(s): %v", strings.Join(path, "/")) + } + result[str], err = convertYAMLMapKeyTypes(v, append(path, str)) + if err != nil { + return nil, err + } + } + return result, nil + case []interface{}: + for i := range x { + x[i], err = convertYAMLMapKeyTypes(x[i], append(path, fmt.Sprintf("%d", i))) + if err != nil { + return nil, err + } + } + return x, nil + default: + return x, nil + } +} + +// futureKeywords is the source of truth for future keywords that will +// eventually become standard keywords inside of Rego. +var futureKeywords = map[string]tokens.Token{ + "in": tokens.In, +} + +func (p *Parser) futureImport(imp *Import, allowedFutureKeywords map[string]tokens.Token) { + path := imp.Path.Value.(Ref) + if len(path) == 1 { + p.errorf(imp.Path.Location, "invalid import, use `import future.keywords` or `import.future.keywords.in`") + return + } + + if !path[1].Equal(StringTerm("keywords")) { + p.errorf(imp.Path.Location, "invalid import, must be `future.keywords`") + return + } + + if imp.Alias != "" { + p.errorf(imp.Path.Location, "future keyword imports cannot be aliased") + return + } + + kwds := make([]string, 0, len(allowedFutureKeywords)) + for k := range allowedFutureKeywords { + kwds = append(kwds, k) + } + switch len(path) { + case 2: // all keywords imported, nothing to do + case 3: // one keyword imported + kw, ok := path[2].Value.(String) + if !ok { + p.errorf(imp.Path.Location, "invalid import, must be `future.keywords.x`, e.g. `import future.keywords.in`") + return + } + keyword := string(kw) + _, ok = allowedFutureKeywords[keyword] + if !ok { + p.errorf(imp.Path.Location, "unexpected keyword, must be one of %v", kwds) + return + } + + kwds = []string{keyword} // overwrite + } + for _, kw := range kwds { + p.s.s.AddKeyword(kw, allowedFutureKeywords[kw]) + } +} diff --git a/vendor/github.com/open-policy-agent/opa/ast/parser_ext.go b/vendor/github.com/open-policy-agent/opa/ast/parser_ext.go new file mode 100644 index 00000000..749dbd62 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/ast/parser_ext.go @@ -0,0 +1,768 @@ +// Copyright 2016 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +// This file contains extra functions for parsing Rego. +// Most of the parsing is handled by the code in parser.go, +// however, there are additional utilities that are +// helpful for dealing with Rego source inputs (e.g., REPL +// statements, source files, etc.) + +package ast + +import ( + "bytes" + "fmt" + "strings" + "unicode" + + "github.com/pkg/errors" +) + +// MustParseBody returns a parsed body. +// If an error occurs during parsing, panic. +func MustParseBody(input string) Body { + parsed, err := ParseBody(input) + if err != nil { + panic(err) + } + return parsed +} + +// MustParseExpr returns a parsed expression. +// If an error occurs during parsing, panic. +func MustParseExpr(input string) *Expr { + parsed, err := ParseExpr(input) + if err != nil { + panic(err) + } + return parsed +} + +// MustParseImports returns a slice of imports. +// If an error occurs during parsing, panic. +func MustParseImports(input string) []*Import { + parsed, err := ParseImports(input) + if err != nil { + panic(err) + } + return parsed +} + +// MustParseModule returns a parsed module. +// If an error occurs during parsing, panic. +func MustParseModule(input string) *Module { + parsed, err := ParseModule("", input) + if err != nil { + panic(err) + } + return parsed +} + +// MustParsePackage returns a Package. +// If an error occurs during parsing, panic. +func MustParsePackage(input string) *Package { + parsed, err := ParsePackage(input) + if err != nil { + panic(err) + } + return parsed +} + +// MustParseStatements returns a slice of parsed statements. +// If an error occurs during parsing, panic. +func MustParseStatements(input string) []Statement { + parsed, _, err := ParseStatements("", input) + if err != nil { + panic(err) + } + return parsed +} + +// MustParseStatement returns exactly one statement. +// If an error occurs during parsing, panic. +func MustParseStatement(input string) Statement { + parsed, err := ParseStatement(input) + if err != nil { + panic(err) + } + return parsed +} + +// MustParseRef returns a parsed reference. +// If an error occurs during parsing, panic. +func MustParseRef(input string) Ref { + parsed, err := ParseRef(input) + if err != nil { + panic(err) + } + return parsed +} + +// MustParseRule returns a parsed rule. +// If an error occurs during parsing, panic. +func MustParseRule(input string) *Rule { + parsed, err := ParseRule(input) + if err != nil { + panic(err) + } + return parsed +} + +// MustParseTerm returns a parsed term. +// If an error occurs during parsing, panic. +func MustParseTerm(input string) *Term { + parsed, err := ParseTerm(input) + if err != nil { + panic(err) + } + return parsed +} + +// ParseRuleFromBody returns a rule if the body can be interpreted as a rule +// definition. Otherwise, an error is returned. +func ParseRuleFromBody(module *Module, body Body) (*Rule, error) { + + if len(body) != 1 { + return nil, fmt.Errorf("multiple expressions cannot be used for rule head") + } + + return ParseRuleFromExpr(module, body[0]) +} + +// ParseRuleFromExpr returns a rule if the expression can be interpreted as a +// rule definition. +func ParseRuleFromExpr(module *Module, expr *Expr) (*Rule, error) { + + if len(expr.With) > 0 { + return nil, fmt.Errorf("expressions using with keyword cannot be used for rule head") + } + + if expr.Negated { + return nil, fmt.Errorf("negated expressions cannot be used for rule head") + } + + if _, ok := expr.Terms.(*SomeDecl); ok { + return nil, errors.New("some declarations cannot be used for rule head") + } + + if term, ok := expr.Terms.(*Term); ok { + switch v := term.Value.(type) { + case Ref: + return ParsePartialSetDocRuleFromTerm(module, term) + default: + return nil, fmt.Errorf("%v cannot be used for rule name", TypeName(v)) + } + } + + if _, ok := expr.Terms.([]*Term); !ok { + // This is a defensive check in case other kinds of expression terms are + // introduced in the future. + return nil, errors.New("expression cannot be used for rule head") + } + + if expr.IsAssignment() { + + lhs, rhs := expr.Operand(0), expr.Operand(1) + if lhs == nil || rhs == nil { + return nil, errors.New("assignment requires two operands") + } + + rule, err := ParseCompleteDocRuleFromAssignmentExpr(module, lhs, rhs) + + if err == nil { + rule.Location = expr.Location + rule.Head.Location = expr.Location + return rule, nil + } else if _, ok := lhs.Value.(Call); ok { + return nil, errFunctionAssignOperator + } else if _, ok := lhs.Value.(Ref); ok { + return nil, errPartialRuleAssignOperator + } + + return nil, errTermAssignOperator(lhs.Value) + } + + if expr.IsEquality() { + return parseCompleteRuleFromEq(module, expr) + } + + if _, ok := BuiltinMap[expr.Operator().String()]; ok { + return nil, fmt.Errorf("rule name conflicts with built-in function") + } + + return ParseRuleFromCallExpr(module, expr.Terms.([]*Term)) +} + +func parseCompleteRuleFromEq(module *Module, expr *Expr) (rule *Rule, err error) { + + // ensure the rule location is set to the expr location + // the helper functions called below try to set the location based + // on the terms they've been provided but that is not as accurate. + defer func() { + if rule != nil { + rule.Location = expr.Location + rule.Head.Location = expr.Location + } + }() + + lhs, rhs := expr.Operand(0), expr.Operand(1) + if lhs == nil || rhs == nil { + return nil, errors.New("assignment requires two operands") + } + + rule, err = ParseCompleteDocRuleFromEqExpr(module, lhs, rhs) + + if err == nil { + return rule, nil + } + + rule, err = ParseRuleFromCallEqExpr(module, lhs, rhs) + if err == nil { + return rule, nil + } + + return ParsePartialObjectDocRuleFromEqExpr(module, lhs, rhs) +} + +// ParseCompleteDocRuleFromAssignmentExpr returns a rule if the expression can +// be interpreted as a complete document definition declared with the assignment +// operator. +func ParseCompleteDocRuleFromAssignmentExpr(module *Module, lhs, rhs *Term) (*Rule, error) { + + rule, err := ParseCompleteDocRuleFromEqExpr(module, lhs, rhs) + if err != nil { + return nil, err + } + + rule.Head.Assign = true + + return rule, nil +} + +// ParseCompleteDocRuleFromEqExpr returns a rule if the expression can be +// interpreted as a complete document definition. +func ParseCompleteDocRuleFromEqExpr(module *Module, lhs, rhs *Term) (*Rule, error) { + + var name Var + + if RootDocumentRefs.Contains(lhs) { + name = lhs.Value.(Ref)[0].Value.(Var) + } else if v, ok := lhs.Value.(Var); ok { + name = v + } else { + return nil, fmt.Errorf("%v cannot be used for rule name", TypeName(lhs.Value)) + } + + rule := &Rule{ + Location: lhs.Location, + Head: &Head{ + Location: lhs.Location, + Name: name, + Value: rhs, + }, + Body: NewBody( + NewExpr(BooleanTerm(true).SetLocation(rhs.Location)).SetLocation(rhs.Location), + ), + Module: module, + } + + return rule, nil +} + +// ParsePartialObjectDocRuleFromEqExpr returns a rule if the expression can be +// interpreted as a partial object document definition. +func ParsePartialObjectDocRuleFromEqExpr(module *Module, lhs, rhs *Term) (*Rule, error) { + + ref, ok := lhs.Value.(Ref) + if !ok || len(ref) != 2 { + return nil, fmt.Errorf("%v cannot be used as rule name", TypeName(lhs.Value)) + } + + if _, ok := ref[0].Value.(Var); !ok { + return nil, fmt.Errorf("%vs cannot be used as rule name", TypeName(ref[0].Value)) + } + + name := ref[0].Value.(Var) + key := ref[1] + + rule := &Rule{ + Location: rhs.Location, + Head: &Head{ + Location: rhs.Location, + Name: name, + Key: key, + Value: rhs, + }, + Body: NewBody( + NewExpr(BooleanTerm(true).SetLocation(rhs.Location)).SetLocation(rhs.Location), + ), + Module: module, + } + + return rule, nil +} + +// ParsePartialSetDocRuleFromTerm returns a rule if the term can be interpreted +// as a partial set document definition. +func ParsePartialSetDocRuleFromTerm(module *Module, term *Term) (*Rule, error) { + + ref, ok := term.Value.(Ref) + if !ok { + return nil, fmt.Errorf("%vs cannot be used for rule head", TypeName(term.Value)) + } + + if len(ref) != 2 { + return nil, fmt.Errorf("refs cannot be used for rule") + } + + name, ok := ref[0].Value.(Var) + if !ok { + return nil, fmt.Errorf("%vs cannot be used as rule name", TypeName(ref[0].Value)) + } + + rule := &Rule{ + Location: term.Location, + Head: &Head{ + Location: term.Location, + Name: name, + Key: ref[1], + }, + Body: NewBody( + NewExpr(BooleanTerm(true).SetLocation(term.Location)).SetLocation(term.Location), + ), + Module: module, + } + + return rule, nil +} + +// ParseRuleFromCallEqExpr returns a rule if the term can be interpreted as a +// function definition (e.g., f(x) = y => f(x) = y { true }). +func ParseRuleFromCallEqExpr(module *Module, lhs, rhs *Term) (*Rule, error) { + + call, ok := lhs.Value.(Call) + if !ok { + return nil, fmt.Errorf("must be call") + } + + ref, ok := call[0].Value.(Ref) + if !ok { + return nil, fmt.Errorf("%vs cannot be used in function signature", TypeName(call[0].Value)) + } + + name, ok := ref[0].Value.(Var) + if !ok { + return nil, fmt.Errorf("%vs cannot be used in function signature", TypeName(ref[0].Value)) + } + + rule := &Rule{ + Location: lhs.Location, + Head: &Head{ + Location: lhs.Location, + Name: name, + Args: Args(call[1:]), + Value: rhs, + }, + Body: NewBody(NewExpr(BooleanTerm(true).SetLocation(rhs.Location)).SetLocation(rhs.Location)), + Module: module, + } + + return rule, nil +} + +// ParseRuleFromCallExpr returns a rule if the terms can be interpreted as a +// function returning true or some value (e.g., f(x) => f(x) = true { true }). +func ParseRuleFromCallExpr(module *Module, terms []*Term) (*Rule, error) { + + if len(terms) <= 1 { + return nil, fmt.Errorf("rule argument list must take at least one argument") + } + + loc := terms[0].Location + args := terms[1:] + value := BooleanTerm(true).SetLocation(loc) + + rule := &Rule{ + Location: loc, + Head: &Head{ + Location: loc, + Name: Var(terms[0].String()), + Args: args, + Value: value, + }, + Module: module, + Body: NewBody(NewExpr(BooleanTerm(true).SetLocation(loc)).SetLocation(loc)), + } + return rule, nil +} + +// ParseImports returns a slice of Import objects. +func ParseImports(input string) ([]*Import, error) { + stmts, _, err := ParseStatements("", input) + if err != nil { + return nil, err + } + result := []*Import{} + for _, stmt := range stmts { + if imp, ok := stmt.(*Import); ok { + result = append(result, imp) + } else { + return nil, fmt.Errorf("expected import but got %T", stmt) + } + } + return result, nil +} + +// ParseModule returns a parsed Module object. +// For details on Module objects and their fields, see policy.go. +// Empty input will return nil, nil. +func ParseModule(filename, input string) (*Module, error) { + return ParseModuleWithOpts(filename, input, ParserOptions{}) +} + +// ParseModuleWithOpts returns a parsed Module object, and has an additional input ParserOptions +// For details on Module objects and their fields, see policy.go. +// Empty input will return nil, nil. +func ParseModuleWithOpts(filename, input string, popts ParserOptions) (*Module, error) { + stmts, comments, err := ParseStatementsWithOpts(filename, input, popts) + if err != nil { + return nil, err + } + return parseModule(filename, stmts, comments) +} + +// ParseBody returns exactly one body. +// If multiple bodies are parsed, an error is returned. +func ParseBody(input string) (Body, error) { + return ParseBodyWithOpts(input, ParserOptions{}) +} + +func ParseBodyWithOpts(input string, popts ParserOptions) (Body, error) { + stmts, _, err := ParseStatementsWithOpts("", input, popts) + if err != nil { + return nil, err + } + + result := Body{} + + for _, stmt := range stmts { + switch stmt := stmt.(type) { + case Body: + for i := range stmt { + result.Append(stmt[i]) + } + case *Comment: + // skip + default: + return nil, fmt.Errorf("expected body but got %T", stmt) + } + } + + return result, nil +} + +// ParseExpr returns exactly one expression. +// If multiple expressions are parsed, an error is returned. +func ParseExpr(input string) (*Expr, error) { + body, err := ParseBody(input) + if err != nil { + return nil, errors.Wrap(err, "failed to parse expression") + } + if len(body) != 1 { + return nil, fmt.Errorf("expected exactly one expression but got: %v", body) + } + return body[0], nil +} + +// ParsePackage returns exactly one Package. +// If multiple statements are parsed, an error is returned. +func ParsePackage(input string) (*Package, error) { + stmt, err := ParseStatement(input) + if err != nil { + return nil, err + } + pkg, ok := stmt.(*Package) + if !ok { + return nil, fmt.Errorf("expected package but got %T", stmt) + } + return pkg, nil +} + +// ParseTerm returns exactly one term. +// If multiple terms are parsed, an error is returned. +func ParseTerm(input string) (*Term, error) { + body, err := ParseBody(input) + if err != nil { + return nil, errors.Wrap(err, "failed to parse term") + } + if len(body) != 1 { + return nil, fmt.Errorf("expected exactly one term but got: %v", body) + } + term, ok := body[0].Terms.(*Term) + if !ok { + return nil, fmt.Errorf("expected term but got %v", body[0].Terms) + } + return term, nil +} + +// ParseRef returns exactly one reference. +func ParseRef(input string) (Ref, error) { + term, err := ParseTerm(input) + if err != nil { + return nil, errors.Wrap(err, "failed to parse ref") + } + ref, ok := term.Value.(Ref) + if !ok { + return nil, fmt.Errorf("expected ref but got %v", term) + } + return ref, nil +} + +// ParseRule returns exactly one rule. +// If multiple rules are parsed, an error is returned. +func ParseRule(input string) (*Rule, error) { + stmts, _, err := ParseStatements("", input) + if err != nil { + return nil, err + } + if len(stmts) != 1 { + return nil, fmt.Errorf("expected exactly one statement (rule)") + } + rule, ok := stmts[0].(*Rule) + if !ok { + return nil, fmt.Errorf("expected rule but got %T", stmts[0]) + } + return rule, nil +} + +// ParseStatement returns exactly one statement. +// A statement might be a term, expression, rule, etc. Regardless, +// this function expects *exactly* one statement. If multiple +// statements are parsed, an error is returned. +func ParseStatement(input string) (Statement, error) { + stmts, _, err := ParseStatements("", input) + if err != nil { + return nil, err + } + if len(stmts) != 1 { + return nil, fmt.Errorf("expected exactly one statement") + } + return stmts[0], nil +} + +// ParseStatements is deprecated. Use ParseStatementWithOpts instead. +func ParseStatements(filename, input string) ([]Statement, []*Comment, error) { + return ParseStatementsWithOpts(filename, input, ParserOptions{}) +} + +// ParseStatementsWithOpts returns a slice of parsed statements. This is the +// default return value from the parser. +func ParseStatementsWithOpts(filename, input string, popts ParserOptions) ([]Statement, []*Comment, error) { + + parser := NewParser(). + WithFilename(filename). + WithReader(bytes.NewBufferString(input)). + WithProcessAnnotation(popts.ProcessAnnotation). + WithFutureKeywords(popts.FutureKeywords...). + WithAllFutureKeywords(popts.AllFutureKeywords). + WithCapabilities(popts.Capabilities) + + stmts, comments, errs := parser.Parse() + + if len(errs) > 0 { + return nil, nil, errs + } + + return stmts, comments, nil +} + +func parseModule(filename string, stmts []Statement, comments []*Comment) (*Module, error) { + + if len(stmts) == 0 { + return nil, NewError(ParseErr, &Location{File: filename}, "empty module") + } + + var errs Errors + + _package, ok := stmts[0].(*Package) + if !ok { + loc := stmts[0].Loc() + errs = append(errs, NewError(ParseErr, loc, "package expected")) + } + + mod := &Module{ + Package: _package, + } + + // The comments slice only holds comments that were not their own statements. + mod.Comments = append(mod.Comments, comments...) + + for i, stmt := range stmts[1:] { + switch stmt := stmt.(type) { + case *Import: + mod.Imports = append(mod.Imports, stmt) + case *Rule: + setRuleModule(stmt, mod) + mod.Rules = append(mod.Rules, stmt) + case Body: + rule, err := ParseRuleFromBody(mod, stmt) + if err != nil { + errs = append(errs, NewError(ParseErr, stmt[0].Location, err.Error())) + } else { + mod.Rules = append(mod.Rules, rule) + + // NOTE(tsandall): the statement should now be interpreted as a + // rule so update the statement list. This is important for the + // logic below that associates annotations with statements. + stmts[i+1] = rule + } + case *Package: + errs = append(errs, NewError(ParseErr, stmt.Loc(), "unexpected package")) + case *Annotations: + mod.Annotations = append(mod.Annotations, stmt) + case *Comment: + // Ignore comments, they're handled above. + default: + panic("illegal value") // Indicates grammar is out-of-sync with code. + } + } + + if len(errs) > 0 { + return nil, errs + } + + // Find first non-annotation statement following each annotation and attach + // the annotation to that statement. + for _, a := range mod.Annotations { + for _, stmt := range stmts { + _, ok := stmt.(*Annotations) + if !ok { + if stmt.Loc().Row > a.Location.Row { + a.node = stmt + break + } + } + } + + if a.Scope == "" { + switch a.node.(type) { + case *Rule: + a.Scope = annotationScopeRule + case *Package: + a.Scope = annotationScopePackage + case *Import: + a.Scope = annotationScopeImport + } + } + + if err := validateAnnotationScopeAttachment(a); err != nil { + errs = append(errs, err) + } + } + + if len(errs) > 0 { + return nil, errs + } + + return mod, nil +} + +func validateAnnotationScopeAttachment(a *Annotations) *Error { + + switch a.Scope { + case annotationScopeRule, annotationScopeDocument: + if _, ok := a.node.(*Rule); ok { + return nil + } + return newScopeAttachmentErr(a, "rule") + case annotationScopePackage, annotationScopeSubpackages: + if _, ok := a.node.(*Package); ok { + return nil + } + return newScopeAttachmentErr(a, "package") + } + + return NewError(ParseErr, a.Loc(), "invalid annotation scope '%v'", a.Scope) +} + +func newScopeAttachmentErr(a *Annotations, want string) *Error { + var have string + if a.node != nil { + have = fmt.Sprintf(" (have %v)", TypeName(a.node)) + } + return NewError(ParseErr, a.Loc(), "annotation scope '%v' must be applied to %v%v", a.Scope, want, have) +} + +func setRuleModule(rule *Rule, module *Module) { + rule.Module = module + if rule.Else != nil { + setRuleModule(rule.Else, module) + } +} + +// ParserErrorDetail holds additional details for parser errors. +type ParserErrorDetail struct { + Line string `json:"line"` + Idx int `json:"idx"` +} + +func newParserErrorDetail(bs []byte, offset int) *ParserErrorDetail { + + // Find first non-space character at or before offset position. + if offset >= len(bs) { + offset = len(bs) - 1 + } else if offset < 0 { + offset = 0 + } + + for offset > 0 && unicode.IsSpace(rune(bs[offset])) { + offset-- + } + + // Find beginning of line containing offset. + begin := offset + + for begin > 0 && !isNewLineChar(bs[begin]) { + begin-- + } + + if isNewLineChar(bs[begin]) { + begin++ + } + + // Find end of line containing offset. + end := offset + + for end < len(bs) && !isNewLineChar(bs[end]) { + end++ + } + + if begin > end { + begin = end + } + + // Extract line and compute index of offset byte in line. + line := bs[begin:end] + index := offset - begin + + return &ParserErrorDetail{ + Line: string(line), + Idx: index, + } +} + +// Lines returns the pretty formatted line output for the error details. +func (d ParserErrorDetail) Lines() []string { + line := strings.TrimLeft(d.Line, "\t") // remove leading tabs + tabCount := len(d.Line) - len(line) + indent := d.Idx - tabCount + if indent < 0 { + indent = 0 + } + return []string{line, strings.Repeat(" ", indent) + "^"} +} + +func isNewLineChar(b byte) bool { + return b == '\r' || b == '\n' +} diff --git a/vendor/github.com/open-policy-agent/opa/ast/policy.go b/vendor/github.com/open-policy-agent/opa/ast/policy.go new file mode 100644 index 00000000..26b6d78e --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/ast/policy.go @@ -0,0 +1,1622 @@ +// Copyright 2016 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package ast + +import ( + "bytes" + "encoding/json" + "fmt" + "math/rand" + "strings" + "time" + + "github.com/open-policy-agent/opa/util" +) + +// Initialize seed for term hashing. This is intentionally placed before the +// root document sets are constructed to ensure they use the same hash seed as +// subsequent lookups. If the hash seeds are out of sync, lookups will fail. +var hashSeed = rand.New(rand.NewSource(time.Now().UnixNano())) +var hashSeed0 = (uint64(hashSeed.Uint32()) << 32) | uint64(hashSeed.Uint32()) + +// DefaultRootDocument is the default root document. +// +// All package directives inside source files are implicitly prefixed with the +// DefaultRootDocument value. +var DefaultRootDocument = VarTerm("data") + +// InputRootDocument names the document containing query arguments. +var InputRootDocument = VarTerm("input") + +// SchemaRootDocument names the document containing external data schemas. +var SchemaRootDocument = VarTerm("schema") + +// FunctionArgRootDocument names the document containing function arguments. +// It's only for internal usage, for referencing function arguments between +// the index and topdown. +var FunctionArgRootDocument = VarTerm("args") + +// FutureRootDocument names the document containing new, to-become-default, +// features. +var FutureRootDocument = VarTerm("future") + +// RootDocumentNames contains the names of top-level documents that can be +// referred to in modules and queries. +// +// Note, the schema document is not currently implemented in the evaluator so it +// is not registered as a root document name (yet). +var RootDocumentNames = NewSet( + DefaultRootDocument, + InputRootDocument, +) + +// DefaultRootRef is a reference to the root of the default document. +// +// All refs to data in the policy engine's storage layer are prefixed with this ref. +var DefaultRootRef = Ref{DefaultRootDocument} + +// InputRootRef is a reference to the root of the input document. +// +// All refs to query arguments are prefixed with this ref. +var InputRootRef = Ref{InputRootDocument} + +// SchemaRootRef is a reference to the root of the schema document. +// +// All refs to schema documents are prefixed with this ref. Note, the schema +// document is not currently implemented in the evaluator so it is not +// registered as a root document ref (yet). +var SchemaRootRef = Ref{SchemaRootDocument} + +// RootDocumentRefs contains the prefixes of top-level documents that all +// non-local references start with. +var RootDocumentRefs = NewSet( + NewTerm(DefaultRootRef), + NewTerm(InputRootRef), +) + +// SystemDocumentKey is the name of the top-level key that identifies the system +// document. +var SystemDocumentKey = String("system") + +// ReservedVars is the set of names that refer to implicitly ground vars. +var ReservedVars = NewVarSet( + DefaultRootDocument.Value.(Var), + InputRootDocument.Value.(Var), +) + +// Wildcard represents the wildcard variable as defined in the language. +var Wildcard = &Term{Value: Var("_")} + +// WildcardPrefix is the special character that all wildcard variables are +// prefixed with when the statement they are contained in is parsed. +var WildcardPrefix = "$" + +// Keywords contains strings that map to language keywords. +var Keywords = [...]string{ + "not", + "package", + "import", + "as", + "default", + "else", + "with", + "null", + "true", + "false", + "some", +} + +// IsKeyword returns true if s is a language keyword. +func IsKeyword(s string) bool { + for _, x := range Keywords { + if x == s { + return true + } + } + return false +} + +type ( + // Node represents a node in an AST. Nodes may be statements in a policy module + // or elements of an ad-hoc query, expression, etc. + Node interface { + fmt.Stringer + Loc() *Location + SetLoc(*Location) + } + + // Statement represents a single statement in a policy module. + Statement interface { + Node + } +) + +type ( + + // Module represents a collection of policies (defined by rules) + // within a namespace (defined by the package) and optional + // dependencies on external documents (defined by imports). + Module struct { + Package *Package `json:"package"` + Imports []*Import `json:"imports,omitempty"` + Annotations []*Annotations `json:"annotations,omitempty"` + Rules []*Rule `json:"rules,omitempty"` + Comments []*Comment `json:"comments,omitempty"` + } + + // Comment contains the raw text from the comment in the definition. + Comment struct { + Text []byte + Location *Location + } + + // Annotations represents metadata attached to other AST nodes such as rules. + Annotations struct { + Location *Location `json:"-"` + Scope string `json:"scope"` + Schemas []*SchemaAnnotation `json:"schemas,omitempty"` + node Node + } + + // SchemaAnnotation contains a schema declaration for the document identified by the path. + SchemaAnnotation struct { + Path Ref `json:"path"` + Schema Ref `json:"schema,omitempty"` + Definition *interface{} `json:"definition,omitempty"` + } + + // Package represents the namespace of the documents produced + // by rules inside the module. + Package struct { + Location *Location `json:"-"` + Path Ref `json:"path"` + } + + // Import represents a dependency on a document outside of the policy + // namespace. Imports are optional. + Import struct { + Location *Location `json:"-"` + Path *Term `json:"path"` + Alias Var `json:"alias,omitempty"` + } + + // Rule represents a rule as defined in the language. Rules define the + // content of documents that represent policy decisions. + Rule struct { + Location *Location `json:"-"` + Default bool `json:"default,omitempty"` + Head *Head `json:"head"` + Body Body `json:"body"` + Else *Rule `json:"else,omitempty"` + + // Module is a pointer to the module containing this rule. If the rule + // was NOT created while parsing/constructing a module, this should be + // left unset. The pointer is not included in any standard operations + // on the rule (e.g., printing, comparison, visiting, etc.) + Module *Module `json:"-"` + } + + // Head represents the head of a rule. + Head struct { + Location *Location `json:"-"` + Name Var `json:"name"` + Args Args `json:"args,omitempty"` + Key *Term `json:"key,omitempty"` + Value *Term `json:"value,omitempty"` + Assign bool `json:"assign,omitempty"` + } + + // Args represents zero or more arguments to a rule. + Args []*Term + + // Body represents one or more expressions contained inside a rule or user + // function. + Body []*Expr + + // Expr represents a single expression contained inside the body of a rule. + Expr struct { + With []*With `json:"with,omitempty"` + Terms interface{} `json:"terms"` + Location *Location `json:"-"` + Index int `json:"index"` + Generated bool `json:"generated,omitempty"` + Negated bool `json:"negated,omitempty"` + } + + // SomeDecl represents a variable declaration statement. The symbols are variables. + SomeDecl struct { + Location *Location `json:"-"` + Symbols []*Term `json:"symbols"` + } + + // With represents a modifier on an expression. + With struct { + Location *Location `json:"-"` + Target *Term `json:"target"` + Value *Term `json:"value"` + } +) + +func (s *Annotations) String() string { + bs, _ := json.Marshal(s) + return string(bs) +} + +// Loc returns the location of this annotation. +func (s *Annotations) Loc() *Location { + return s.Location +} + +// SetLoc updates the location of this annotation. +func (s *Annotations) SetLoc(l *Location) { + s.Location = l +} + +// Compare returns an integer indicating if s is less than, equal to, or greater +// than other. +func (s *Annotations) Compare(other *Annotations) int { + + if cmp := scopeCompare(s.Scope, other.Scope); cmp != 0 { + return cmp + } + + max := len(s.Schemas) + if len(other.Schemas) < max { + max = len(other.Schemas) + } + + for i := 0; i < max; i++ { + if cmp := s.Schemas[i].Compare(other.Schemas[i]); cmp != 0 { + return cmp + } + } + + if len(s.Schemas) > len(other.Schemas) { + return 1 + } else if len(s.Schemas) < len(other.Schemas) { + return -1 + } + + return 0 +} + +// Copy returns a deep copy of s. +func (s *Annotations) Copy(node Node) *Annotations { + cpy := *s + cpy.Schemas = make([]*SchemaAnnotation, len(s.Schemas)) + for i := range cpy.Schemas { + cpy.Schemas[i] = s.Schemas[i].Copy() + } + cpy.node = node + return &cpy +} + +// Copy returns a deep copy of s. +func (s *SchemaAnnotation) Copy() *SchemaAnnotation { + cpy := *s + return &cpy +} + +// Compare returns an integer indicating if s is less than, equal to, or greater +// than other. +func (s *SchemaAnnotation) Compare(other *SchemaAnnotation) int { + + if cmp := s.Path.Compare(other.Path); cmp != 0 { + return cmp + } + + if cmp := s.Schema.Compare(other.Schema); cmp != 0 { + return cmp + } + + if s.Definition != nil && other.Definition == nil { + return -1 + } else if s.Definition == nil && other.Definition != nil { + return 1 + } else if s.Definition != nil && other.Definition != nil { + return util.Compare(*s.Definition, *other.Definition) + } + + return 0 +} + +func (s *SchemaAnnotation) String() string { + bs, _ := json.Marshal(s) + return string(bs) +} + +func scopeCompare(s1, s2 string) int { + + o1 := scopeOrder(s1) + o2 := scopeOrder(s2) + + if o2 < o1 { + return 1 + } else if o2 > o1 { + return -1 + } + + if s1 < s2 { + return -1 + } else if s2 < s1 { + return 1 + } + + return 0 +} + +func scopeOrder(s string) int { + switch s { + case annotationScopeRule: + return 1 + } + return 0 +} + +// Compare returns an integer indicating whether mod is less than, equal to, +// or greater than other. +func (mod *Module) Compare(other *Module) int { + if mod == nil { + if other == nil { + return 0 + } + return -1 + } else if other == nil { + return 1 + } + if cmp := mod.Package.Compare(other.Package); cmp != 0 { + return cmp + } + if cmp := importsCompare(mod.Imports, other.Imports); cmp != 0 { + return cmp + } + if cmp := annotationsCompare(mod.Annotations, other.Annotations); cmp != 0 { + return cmp + } + return rulesCompare(mod.Rules, other.Rules) +} + +// Copy returns a deep copy of mod. +func (mod *Module) Copy() *Module { + cpy := *mod + cpy.Rules = make([]*Rule, len(mod.Rules)) + + var nodes map[Node]Node + + if len(mod.Annotations) > 0 { + nodes = make(map[Node]Node) + } + + for i := range mod.Rules { + cpy.Rules[i] = mod.Rules[i].Copy() + cpy.Rules[i].Module = &cpy + if nodes != nil { + nodes[mod.Rules[i]] = cpy.Rules[i] + } + } + + cpy.Imports = make([]*Import, len(mod.Imports)) + for i := range mod.Imports { + cpy.Imports[i] = mod.Imports[i].Copy() + if nodes != nil { + nodes[mod.Imports[i]] = cpy.Imports[i] + } + } + + cpy.Package = mod.Package.Copy() + if nodes != nil { + nodes[mod.Package] = cpy.Package + } + + cpy.Annotations = make([]*Annotations, len(mod.Annotations)) + for i := range mod.Annotations { + cpy.Annotations[i] = mod.Annotations[i].Copy(nodes[mod.Annotations[i].node]) + } + + cpy.Comments = make([]*Comment, len(mod.Comments)) + for i := range mod.Comments { + cpy.Comments[i] = mod.Comments[i].Copy() + } + + return &cpy +} + +// Equal returns true if mod equals other. +func (mod *Module) Equal(other *Module) bool { + return mod.Compare(other) == 0 +} + +func (mod *Module) String() string { + byNode := map[Node][]*Annotations{} + for _, a := range mod.Annotations { + byNode[a.node] = append(byNode[a.node], a) + } + + appendAnnotationStrings := func(buf []string, node Node) []string { + if as, ok := byNode[node]; ok { + for i := range as { + buf = append(buf, "# METADATA") + buf = append(buf, "# "+as[i].String()) + } + } + return buf + } + + buf := []string{} + buf = appendAnnotationStrings(buf, mod.Package) + buf = append(buf, mod.Package.String()) + + if len(mod.Imports) > 0 { + buf = append(buf, "") + for _, imp := range mod.Imports { + buf = appendAnnotationStrings(buf, imp) + buf = append(buf, imp.String()) + } + } + if len(mod.Rules) > 0 { + buf = append(buf, "") + for _, rule := range mod.Rules { + buf = appendAnnotationStrings(buf, rule) + buf = append(buf, rule.String()) + } + } + return strings.Join(buf, "\n") +} + +// RuleSet returns a RuleSet containing named rules in the mod. +func (mod *Module) RuleSet(name Var) RuleSet { + rs := NewRuleSet() + for _, rule := range mod.Rules { + if rule.Head.Name.Equal(name) { + rs.Add(rule) + } + } + return rs +} + +// UnmarshalJSON parses bs and stores the result in mod. The rules in the module +// will have their module pointer set to mod. +func (mod *Module) UnmarshalJSON(bs []byte) error { + + // Declare a new type and use a type conversion to avoid recursively calling + // Module#UnmarshalJSON. + type module Module + + if err := util.UnmarshalJSON(bs, (*module)(mod)); err != nil { + return err + } + + WalkRules(mod, func(rule *Rule) bool { + rule.Module = mod + return false + }) + + return nil +} + +// NewComment returns a new Comment object. +func NewComment(text []byte) *Comment { + return &Comment{ + Text: text, + } +} + +// Loc returns the location of the comment in the definition. +func (c *Comment) Loc() *Location { + if c == nil { + return nil + } + return c.Location +} + +// SetLoc sets the location on c. +func (c *Comment) SetLoc(loc *Location) { + c.Location = loc +} + +func (c *Comment) String() string { + return "#" + string(c.Text) +} + +// Copy returns a deep copy of c. +func (c *Comment) Copy() *Comment { + cpy := *c + cpy.Text = make([]byte, len(c.Text)) + copy(cpy.Text, c.Text) + return &cpy +} + +// Equal returns true if this comment equals the other comment. +// Unlike other equality checks on AST nodes, comment equality +// depends on location. +func (c *Comment) Equal(other *Comment) bool { + return c.Location.Equal(other.Location) && bytes.Equal(c.Text, other.Text) +} + +// Compare returns an integer indicating whether pkg is less than, equal to, +// or greater than other. +func (pkg *Package) Compare(other *Package) int { + return Compare(pkg.Path, other.Path) +} + +// Copy returns a deep copy of pkg. +func (pkg *Package) Copy() *Package { + cpy := *pkg + cpy.Path = pkg.Path.Copy() + return &cpy +} + +// Equal returns true if pkg is equal to other. +func (pkg *Package) Equal(other *Package) bool { + return pkg.Compare(other) == 0 +} + +// Loc returns the location of the Package in the definition. +func (pkg *Package) Loc() *Location { + if pkg == nil { + return nil + } + return pkg.Location +} + +// SetLoc sets the location on pkg. +func (pkg *Package) SetLoc(loc *Location) { + pkg.Location = loc +} + +func (pkg *Package) String() string { + if pkg == nil { + return "" + } else if len(pkg.Path) <= 1 { + return fmt.Sprintf("package ", pkg.Path) + } + // Omit head as all packages have the DefaultRootDocument prepended at parse time. + path := make(Ref, len(pkg.Path)-1) + path[0] = VarTerm(string(pkg.Path[1].Value.(String))) + copy(path[1:], pkg.Path[2:]) + return fmt.Sprintf("package %v", path) +} + +// IsValidImportPath returns an error indicating if the import path is invalid. +// If the import path is invalid, err is nil. +func IsValidImportPath(v Value) (err error) { + switch v := v.(type) { + case Var: + if !v.Equal(DefaultRootDocument.Value) && !v.Equal(InputRootDocument.Value) { + return fmt.Errorf("invalid path %v: path must begin with input or data", v) + } + case Ref: + if err := IsValidImportPath(v[0].Value); err != nil { + return fmt.Errorf("invalid path %v: path must begin with input or data", v) + } + for _, e := range v[1:] { + if _, ok := e.Value.(String); !ok { + return fmt.Errorf("invalid path %v: path elements must be strings", v) + } + } + default: + return fmt.Errorf("invalid path %v: path must be ref or var", v) + } + return nil +} + +// Compare returns an integer indicating whether imp is less than, equal to, +// or greater than other. +func (imp *Import) Compare(other *Import) int { + if imp == nil { + if other == nil { + return 0 + } + return -1 + } else if other == nil { + return 1 + } + if cmp := Compare(imp.Path, other.Path); cmp != 0 { + return cmp + } + return Compare(imp.Alias, other.Alias) +} + +// Copy returns a deep copy of imp. +func (imp *Import) Copy() *Import { + cpy := *imp + cpy.Path = imp.Path.Copy() + return &cpy +} + +// Equal returns true if imp is equal to other. +func (imp *Import) Equal(other *Import) bool { + return imp.Compare(other) == 0 +} + +// Loc returns the location of the Import in the definition. +func (imp *Import) Loc() *Location { + if imp == nil { + return nil + } + return imp.Location +} + +// SetLoc sets the location on imp. +func (imp *Import) SetLoc(loc *Location) { + imp.Location = loc +} + +// Name returns the variable that is used to refer to the imported virtual +// document. This is the alias if defined otherwise the last element in the +// path. +func (imp *Import) Name() Var { + if len(imp.Alias) != 0 { + return imp.Alias + } + switch v := imp.Path.Value.(type) { + case Var: + return v + case Ref: + if len(v) == 1 { + return v[0].Value.(Var) + } + return Var(v[len(v)-1].Value.(String)) + } + panic("illegal import") +} + +func (imp *Import) String() string { + buf := []string{"import", imp.Path.String()} + if len(imp.Alias) > 0 { + buf = append(buf, "as "+imp.Alias.String()) + } + return strings.Join(buf, " ") +} + +// Compare returns an integer indicating whether rule is less than, equal to, +// or greater than other. +func (rule *Rule) Compare(other *Rule) int { + if rule == nil { + if other == nil { + return 0 + } + return -1 + } else if other == nil { + return 1 + } + if cmp := rule.Head.Compare(other.Head); cmp != 0 { + return cmp + } + if cmp := util.Compare(rule.Default, other.Default); cmp != 0 { + return cmp + } + if cmp := rule.Body.Compare(other.Body); cmp != 0 { + return cmp + } + return rule.Else.Compare(other.Else) +} + +// Copy returns a deep copy of rule. +func (rule *Rule) Copy() *Rule { + cpy := *rule + cpy.Head = rule.Head.Copy() + cpy.Body = rule.Body.Copy() + if cpy.Else != nil { + cpy.Else = rule.Else.Copy() + } + return &cpy +} + +// Equal returns true if rule is equal to other. +func (rule *Rule) Equal(other *Rule) bool { + return rule.Compare(other) == 0 +} + +// Loc returns the location of the Rule in the definition. +func (rule *Rule) Loc() *Location { + if rule == nil { + return nil + } + return rule.Location +} + +// SetLoc sets the location on rule. +func (rule *Rule) SetLoc(loc *Location) { + rule.Location = loc +} + +// Path returns a ref referring to the document produced by this rule. If rule +// is not contained in a module, this function panics. +func (rule *Rule) Path() Ref { + if rule.Module == nil { + panic("assertion failed") + } + return rule.Module.Package.Path.Append(StringTerm(string(rule.Head.Name))) +} + +func (rule *Rule) String() string { + buf := []string{} + if rule.Default { + buf = append(buf, "default") + } + buf = append(buf, rule.Head.String()) + if !rule.Default { + buf = append(buf, "{") + buf = append(buf, rule.Body.String()) + buf = append(buf, "}") + } + if rule.Else != nil { + buf = append(buf, rule.Else.elseString()) + } + return strings.Join(buf, " ") +} + +func (rule *Rule) elseString() string { + var buf []string + + buf = append(buf, "else") + + value := rule.Head.Value + if value != nil { + buf = append(buf, "=") + buf = append(buf, value.String()) + } + + buf = append(buf, "{") + buf = append(buf, rule.Body.String()) + buf = append(buf, "}") + + if rule.Else != nil { + buf = append(buf, rule.Else.elseString()) + } + + return strings.Join(buf, " ") +} + +// NewHead returns a new Head object. If args are provided, the first will be +// used for the key and the second will be used for the value. +func NewHead(name Var, args ...*Term) *Head { + head := &Head{ + Name: name, + } + if len(args) == 0 { + return head + } + head.Key = args[0] + if len(args) == 1 { + return head + } + head.Value = args[1] + return head +} + +// DocKind represents the collection of document types that can be produced by rules. +type DocKind int + +const ( + // CompleteDoc represents a document that is completely defined by the rule. + CompleteDoc = iota + + // PartialSetDoc represents a set document that is partially defined by the rule. + PartialSetDoc + + // PartialObjectDoc represents an object document that is partially defined by the rule. + PartialObjectDoc +) + +// DocKind returns the type of document produced by this rule. +func (head *Head) DocKind() DocKind { + if head.Key != nil { + if head.Value != nil { + return PartialObjectDoc + } + return PartialSetDoc + } + return CompleteDoc +} + +// Compare returns an integer indicating whether head is less than, equal to, +// or greater than other. +func (head *Head) Compare(other *Head) int { + if head == nil { + if other == nil { + return 0 + } + return -1 + } else if other == nil { + return 1 + } + if head.Assign && !other.Assign { + return -1 + } else if !head.Assign && other.Assign { + return 1 + } + if cmp := Compare(head.Args, other.Args); cmp != 0 { + return cmp + } + if cmp := Compare(head.Name, other.Name); cmp != 0 { + return cmp + } + if cmp := Compare(head.Key, other.Key); cmp != 0 { + return cmp + } + return Compare(head.Value, other.Value) +} + +// Copy returns a deep copy of head. +func (head *Head) Copy() *Head { + cpy := *head + cpy.Args = head.Args.Copy() + cpy.Key = head.Key.Copy() + cpy.Value = head.Value.Copy() + return &cpy +} + +// Equal returns true if this head equals other. +func (head *Head) Equal(other *Head) bool { + return head.Compare(other) == 0 +} + +func (head *Head) String() string { + var buf []string + if len(head.Args) != 0 { + buf = append(buf, head.Name.String()+head.Args.String()) + } else if head.Key != nil { + buf = append(buf, head.Name.String()+"["+head.Key.String()+"]") + } else { + buf = append(buf, head.Name.String()) + } + if head.Value != nil { + if head.Assign { + buf = append(buf, ":=") + } else { + buf = append(buf, "=") + } + buf = append(buf, head.Value.String()) + } + return strings.Join(buf, " ") +} + +// Vars returns a set of vars found in the head. +func (head *Head) Vars() VarSet { + vis := &VarVisitor{vars: VarSet{}} + // TODO: improve test coverage for this. + if head.Args != nil { + vis.Walk(head.Args) + } + if head.Key != nil { + vis.Walk(head.Key) + } + if head.Value != nil { + vis.Walk(head.Value) + } + return vis.vars +} + +// Loc returns the Location of head. +func (head *Head) Loc() *Location { + if head == nil { + return nil + } + return head.Location +} + +// SetLoc sets the location on head. +func (head *Head) SetLoc(loc *Location) { + head.Location = loc +} + +// Copy returns a deep copy of a. +func (a Args) Copy() Args { + cpy := Args{} + for _, t := range a { + cpy = append(cpy, t.Copy()) + } + return cpy +} + +func (a Args) String() string { + var buf []string + for _, t := range a { + buf = append(buf, t.String()) + } + return "(" + strings.Join(buf, ", ") + ")" +} + +// Loc returns the Location of a. +func (a Args) Loc() *Location { + if len(a) == 0 { + return nil + } + return a[0].Location +} + +// SetLoc sets the location on a. +func (a Args) SetLoc(loc *Location) { + if len(a) != 0 { + a[0].SetLocation(loc) + } +} + +// Vars returns a set of vars that appear in a. +func (a Args) Vars() VarSet { + vis := &VarVisitor{vars: VarSet{}} + vis.Walk(a) + return vis.vars +} + +// NewBody returns a new Body containing the given expressions. The indices of +// the immediate expressions will be reset. +func NewBody(exprs ...*Expr) Body { + for i, expr := range exprs { + expr.Index = i + } + return Body(exprs) +} + +// MarshalJSON returns JSON encoded bytes representing body. +func (body Body) MarshalJSON() ([]byte, error) { + // Serialize empty Body to empty array. This handles both the empty case and the + // nil case (whereas by default the result would be null if body was nil.) + if len(body) == 0 { + return []byte(`[]`), nil + } + return json.Marshal([]*Expr(body)) +} + +// Append adds the expr to the body and updates the expr's index accordingly. +func (body *Body) Append(expr *Expr) { + n := len(*body) + expr.Index = n + *body = append(*body, expr) +} + +// Set sets the expr in the body at the specified position and updates the +// expr's index accordingly. +func (body Body) Set(expr *Expr, pos int) { + body[pos] = expr + expr.Index = pos +} + +// Compare returns an integer indicating whether body is less than, equal to, +// or greater than other. +// +// If body is a subset of other, it is considered less than (and vice versa). +func (body Body) Compare(other Body) int { + minLen := len(body) + if len(other) < minLen { + minLen = len(other) + } + for i := 0; i < minLen; i++ { + if cmp := body[i].Compare(other[i]); cmp != 0 { + return cmp + } + } + if len(body) < len(other) { + return -1 + } + if len(other) < len(body) { + return 1 + } + return 0 +} + +// Copy returns a deep copy of body. +func (body Body) Copy() Body { + cpy := make(Body, len(body)) + for i := range body { + cpy[i] = body[i].Copy() + } + return cpy +} + +// Contains returns true if this body contains the given expression. +func (body Body) Contains(x *Expr) bool { + for _, e := range body { + if e.Equal(x) { + return true + } + } + return false +} + +// Equal returns true if this Body is equal to the other Body. +func (body Body) Equal(other Body) bool { + return body.Compare(other) == 0 +} + +// Hash returns the hash code for the Body. +func (body Body) Hash() int { + s := 0 + for _, e := range body { + s += e.Hash() + } + return s +} + +// IsGround returns true if all of the expressions in the Body are ground. +func (body Body) IsGround() bool { + for _, e := range body { + if !e.IsGround() { + return false + } + } + return true +} + +// Loc returns the location of the Body in the definition. +func (body Body) Loc() *Location { + if len(body) == 0 { + return nil + } + return body[0].Location +} + +// SetLoc sets the location on body. +func (body Body) SetLoc(loc *Location) { + if len(body) != 0 { + body[0].SetLocation(loc) + } +} + +func (body Body) String() string { + var buf []string + for _, v := range body { + buf = append(buf, v.String()) + } + return strings.Join(buf, "; ") +} + +// Vars returns a VarSet containing variables in body. The params can be set to +// control which vars are included. +func (body Body) Vars(params VarVisitorParams) VarSet { + vis := NewVarVisitor().WithParams(params) + vis.Walk(body) + return vis.Vars() +} + +// NewExpr returns a new Expr object. +func NewExpr(terms interface{}) *Expr { + return &Expr{ + Negated: false, + Terms: terms, + Index: 0, + With: nil, + } +} + +// Complement returns a copy of this expression with the negation flag flipped. +func (expr *Expr) Complement() *Expr { + cpy := *expr + cpy.Negated = !cpy.Negated + return &cpy +} + +// Equal returns true if this Expr equals the other Expr. +func (expr *Expr) Equal(other *Expr) bool { + return expr.Compare(other) == 0 +} + +// Compare returns an integer indicating whether expr is less than, equal to, +// or greater than other. +// +// Expressions are compared as follows: +// +// 1. Declarations are always less than other expressions. +// 2. Preceding expression (by Index) is always less than the other expression. +// 3. Non-negated expressions are always less than than negated expressions. +// 4. Single term expressions are always less than built-in expressions. +// +// Otherwise, the expression terms are compared normally. If both expressions +// have the same terms, the modifiers are compared. +func (expr *Expr) Compare(other *Expr) int { + + if expr == nil { + if other == nil { + return 0 + } + return -1 + } else if other == nil { + return 1 + } + + o1 := expr.sortOrder() + o2 := other.sortOrder() + if o1 < o2 { + return -1 + } else if o2 < o1 { + return 1 + } + + switch { + case expr.Index < other.Index: + return -1 + case expr.Index > other.Index: + return 1 + } + + switch { + case expr.Negated && !other.Negated: + return 1 + case !expr.Negated && other.Negated: + return -1 + } + + switch t := expr.Terms.(type) { + case *Term: + if cmp := Compare(t.Value, other.Terms.(*Term).Value); cmp != 0 { + return cmp + } + case []*Term: + if cmp := termSliceCompare(t, other.Terms.([]*Term)); cmp != 0 { + return cmp + } + case *SomeDecl: + if cmp := Compare(t, other.Terms.(*SomeDecl)); cmp != 0 { + return cmp + } + } + + return withSliceCompare(expr.With, other.With) +} + +func (expr *Expr) sortOrder() int { + switch expr.Terms.(type) { + case *SomeDecl: + return 0 + case *Term: + return 1 + case []*Term: + return 2 + } + return -1 +} + +// Copy returns a deep copy of expr. +func (expr *Expr) Copy() *Expr { + + cpy := *expr + + switch ts := expr.Terms.(type) { + case *SomeDecl: + cpy.Terms = ts.Copy() + case []*Term: + cpyTs := make([]*Term, len(ts)) + for i := range ts { + cpyTs[i] = ts[i].Copy() + } + cpy.Terms = cpyTs + case *Term: + cpy.Terms = ts.Copy() + } + + cpy.With = make([]*With, len(expr.With)) + for i := range expr.With { + cpy.With[i] = expr.With[i].Copy() + } + + return &cpy +} + +// Hash returns the hash code of the Expr. +func (expr *Expr) Hash() int { + s := expr.Index + switch ts := expr.Terms.(type) { + case *SomeDecl: + s += ts.Hash() + case []*Term: + for _, t := range ts { + s += t.Value.Hash() + } + case *Term: + s += ts.Value.Hash() + } + if expr.Negated { + s++ + } + for _, w := range expr.With { + s += w.Hash() + } + return s +} + +// IncludeWith returns a copy of expr with the with modifier appended. +func (expr *Expr) IncludeWith(target *Term, value *Term) *Expr { + cpy := *expr + cpy.With = append(cpy.With, &With{Target: target, Value: value}) + return &cpy +} + +// NoWith returns a copy of expr where the with modifier has been removed. +func (expr *Expr) NoWith() *Expr { + cpy := *expr + cpy.With = nil + return &cpy +} + +// IsEquality returns true if this is an equality expression. +func (expr *Expr) IsEquality() bool { + return isGlobalBuiltin(expr, Var(Equality.Name)) +} + +// IsAssignment returns true if this an assignment expression. +func (expr *Expr) IsAssignment() bool { + return isGlobalBuiltin(expr, Var(Assign.Name)) +} + +// IsCall returns true if this expression calls a function. +func (expr *Expr) IsCall() bool { + _, ok := expr.Terms.([]*Term) + return ok +} + +// Operator returns the name of the function or built-in this expression refers +// to. If this expression is not a function call, returns nil. +func (expr *Expr) Operator() Ref { + op := expr.OperatorTerm() + if op == nil { + return nil + } + return op.Value.(Ref) +} + +// OperatorTerm returns the name of the function or built-in this expression +// refers to. If this expression is not a function call, returns nil. +func (expr *Expr) OperatorTerm() *Term { + terms, ok := expr.Terms.([]*Term) + if !ok || len(terms) == 0 { + return nil + } + return terms[0] +} + +// Operand returns the term at the zero-based pos. If the expr does not include +// at least pos+1 terms, this function returns nil. +func (expr *Expr) Operand(pos int) *Term { + terms, ok := expr.Terms.([]*Term) + if !ok { + return nil + } + idx := pos + 1 + if idx < len(terms) { + return terms[idx] + } + return nil +} + +// Operands returns the built-in function operands. +func (expr *Expr) Operands() []*Term { + terms, ok := expr.Terms.([]*Term) + if !ok { + return nil + } + return terms[1:] +} + +// IsGround returns true if all of the expression terms are ground. +func (expr *Expr) IsGround() bool { + switch ts := expr.Terms.(type) { + case []*Term: + for _, t := range ts[1:] { + if !t.IsGround() { + return false + } + } + case *Term: + return ts.IsGround() + } + return true +} + +// SetOperator sets the expr's operator and returns the expr itself. If expr is +// not a call expr, this function will panic. +func (expr *Expr) SetOperator(term *Term) *Expr { + expr.Terms.([]*Term)[0] = term + return expr +} + +// SetLocation sets the expr's location and returns the expr itself. +func (expr *Expr) SetLocation(loc *Location) *Expr { + expr.Location = loc + return expr +} + +// Loc returns the Location of expr. +func (expr *Expr) Loc() *Location { + if expr == nil { + return nil + } + return expr.Location +} + +// SetLoc sets the location on expr. +func (expr *Expr) SetLoc(loc *Location) { + expr.SetLocation(loc) +} + +func (expr *Expr) String() string { + var buf []string + if expr.Negated { + buf = append(buf, "not") + } + switch t := expr.Terms.(type) { + case []*Term: + if expr.IsEquality() && validEqAssignArgCount(expr) { + buf = append(buf, fmt.Sprintf("%v %v %v", t[1], Equality.Infix, t[2])) + } else { + buf = append(buf, Call(t).String()) + } + case *Term: + buf = append(buf, t.String()) + case *SomeDecl: + buf = append(buf, t.String()) + } + + for i := range expr.With { + buf = append(buf, expr.With[i].String()) + } + + return strings.Join(buf, " ") +} + +// UnmarshalJSON parses the byte array and stores the result in expr. +func (expr *Expr) UnmarshalJSON(bs []byte) error { + v := map[string]interface{}{} + if err := util.UnmarshalJSON(bs, &v); err != nil { + return err + } + return unmarshalExpr(expr, v) +} + +// Vars returns a VarSet containing variables in expr. The params can be set to +// control which vars are included. +func (expr *Expr) Vars(params VarVisitorParams) VarSet { + vis := NewVarVisitor().WithParams(params) + vis.Walk(expr) + return vis.Vars() +} + +// NewBuiltinExpr creates a new Expr object with the supplied terms. +// The builtin operator must be the first term. +func NewBuiltinExpr(terms ...*Term) *Expr { + return &Expr{Terms: terms} +} + +func (d *SomeDecl) String() string { + if call, ok := d.Symbols[0].Value.(Call); ok { + if len(call) == 4 { + return "some " + call[1].String() + ", " + call[2].String() + " in " + call[3].String() + } + return "some " + call[1].String() + " in " + call[2].String() + } + buf := make([]string, len(d.Symbols)) + for i := range buf { + buf[i] = d.Symbols[i].String() + } + return "some " + strings.Join(buf, ", ") +} + +// SetLoc sets the Location on d. +func (d *SomeDecl) SetLoc(loc *Location) { + d.Location = loc +} + +// Loc returns the Location of d. +func (d *SomeDecl) Loc() *Location { + return d.Location +} + +// Copy returns a deep copy of d. +func (d *SomeDecl) Copy() *SomeDecl { + cpy := *d + cpy.Symbols = termSliceCopy(d.Symbols) + return &cpy +} + +// Compare returns an integer indicating whether d is less than, equal to, or +// greater than other. +func (d *SomeDecl) Compare(other *SomeDecl) int { + return termSliceCompare(d.Symbols, other.Symbols) +} + +// Hash returns a hash code of d. +func (d *SomeDecl) Hash() int { + return termSliceHash(d.Symbols) +} + +func (w *With) String() string { + return "with " + w.Target.String() + " as " + w.Value.String() +} + +// Equal returns true if this With is equals the other With. +func (w *With) Equal(other *With) bool { + return Compare(w, other) == 0 +} + +// Compare returns an integer indicating whether w is less than, equal to, or +// greater than other. +func (w *With) Compare(other *With) int { + if w == nil { + if other == nil { + return 0 + } + return -1 + } else if other == nil { + return 1 + } + if cmp := Compare(w.Target, other.Target); cmp != 0 { + return cmp + } + return Compare(w.Value, other.Value) +} + +// Copy returns a deep copy of w. +func (w *With) Copy() *With { + cpy := *w + cpy.Value = w.Value.Copy() + cpy.Target = w.Target.Copy() + return &cpy +} + +// Hash returns the hash code of the With. +func (w With) Hash() int { + return w.Target.Hash() + w.Value.Hash() +} + +// SetLocation sets the location on w. +func (w *With) SetLocation(loc *Location) *With { + w.Location = loc + return w +} + +// Loc returns the Location of w. +func (w *With) Loc() *Location { + if w == nil { + return nil + } + return w.Location +} + +// SetLoc sets the location on w. +func (w *With) SetLoc(loc *Location) { + w.Location = loc +} + +// Copy returns a deep copy of the AST node x. If x is not an AST node, x is returned unmodified. +func Copy(x interface{}) interface{} { + switch x := x.(type) { + case *Module: + return x.Copy() + case *Package: + return x.Copy() + case *Import: + return x.Copy() + case *Rule: + return x.Copy() + case *Head: + return x.Copy() + case Args: + return x.Copy() + case Body: + return x.Copy() + case *Expr: + return x.Copy() + case *With: + return x.Copy() + case *SomeDecl: + return x.Copy() + case *Term: + return x.Copy() + case *ArrayComprehension: + return x.Copy() + case *SetComprehension: + return x.Copy() + case *ObjectComprehension: + return x.Copy() + case Set: + return x.Copy() + case *object: + return x.Copy() + case *Array: + return x.Copy() + case Ref: + return x.Copy() + case Call: + return x.Copy() + case *Comment: + return x.Copy() + } + return x +} + +// RuleSet represents a collection of rules that produce a virtual document. +type RuleSet []*Rule + +// NewRuleSet returns a new RuleSet containing the given rules. +func NewRuleSet(rules ...*Rule) RuleSet { + rs := make(RuleSet, 0, len(rules)) + for _, rule := range rules { + rs.Add(rule) + } + return rs +} + +// Add inserts the rule into rs. +func (rs *RuleSet) Add(rule *Rule) { + for _, exist := range *rs { + if exist.Equal(rule) { + return + } + } + *rs = append(*rs, rule) +} + +// Contains returns true if rs contains rule. +func (rs RuleSet) Contains(rule *Rule) bool { + for i := range rs { + if rs[i].Equal(rule) { + return true + } + } + return false +} + +// Diff returns a new RuleSet containing rules in rs that are not in other. +func (rs RuleSet) Diff(other RuleSet) RuleSet { + result := NewRuleSet() + for i := range rs { + if !other.Contains(rs[i]) { + result.Add(rs[i]) + } + } + return result +} + +// Equal returns true if rs equals other. +func (rs RuleSet) Equal(other RuleSet) bool { + return len(rs.Diff(other)) == 0 && len(other.Diff(rs)) == 0 +} + +// Merge returns a ruleset containing the union of rules from rs an other. +func (rs RuleSet) Merge(other RuleSet) RuleSet { + result := NewRuleSet() + for i := range rs { + result.Add(rs[i]) + } + for i := range other { + result.Add(other[i]) + } + return result +} + +func (rs RuleSet) String() string { + buf := make([]string, 0, len(rs)) + for _, rule := range rs { + buf = append(buf, rule.String()) + } + return "{" + strings.Join(buf, ", ") + "}" +} + +// Returns true if the equality or assignment expression referred to by expr +// has a valid number of arguments. +func validEqAssignArgCount(expr *Expr) bool { + return len(expr.Operands()) == 2 +} + +// this function checks if the expr refers to a non-namespaced (global) built-in +// function like eq, gt, plus, etc. +func isGlobalBuiltin(expr *Expr, name Var) bool { + terms, ok := expr.Terms.([]*Term) + if !ok { + return false + } + + // NOTE(tsandall): do not use Term#Equal or Value#Compare to avoid + // allocation here. + ref, ok := terms[0].Value.(Ref) + if !ok || len(ref) != 1 { + return false + } + if head, ok := ref[0].Value.(Var); ok { + return head.Equal(name) + } + return false +} diff --git a/vendor/github.com/open-policy-agent/opa/ast/pretty.go b/vendor/github.com/open-policy-agent/opa/ast/pretty.go new file mode 100644 index 00000000..b4f05ad5 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/ast/pretty.go @@ -0,0 +1,82 @@ +// Copyright 2018 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package ast + +import ( + "fmt" + "io" + "strings" +) + +// Pretty writes a pretty representation of the AST rooted at x to w. +// +// This is function is intended for debug purposes when inspecting ASTs. +func Pretty(w io.Writer, x interface{}) { + pp := &prettyPrinter{ + depth: -1, + w: w, + } + NewBeforeAfterVisitor(pp.Before, pp.After).Walk(x) +} + +type prettyPrinter struct { + depth int + w io.Writer +} + +func (pp *prettyPrinter) Before(x interface{}) bool { + switch x.(type) { + case *Term: + default: + pp.depth++ + } + + switch x := x.(type) { + case *Term: + return false + case Args: + if len(x) == 0 { + return false + } + pp.writeType(x) + case *Expr: + extras := []string{} + if x.Negated { + extras = append(extras, "negated") + } + extras = append(extras, fmt.Sprintf("index=%d", x.Index)) + pp.writeIndent("%v %v", TypeName(x), strings.Join(extras, " ")) + case Null, Boolean, Number, String, Var: + pp.writeValue(x) + default: + pp.writeType(x) + } + return false +} + +func (pp *prettyPrinter) After(x interface{}) { + switch x.(type) { + case *Term: + default: + pp.depth-- + } +} + +func (pp *prettyPrinter) writeValue(x interface{}) { + pp.writeIndent(fmt.Sprint(x)) +} + +func (pp *prettyPrinter) writeType(x interface{}) { + pp.writeIndent(TypeName(x)) +} + +func (pp *prettyPrinter) writeIndent(f string, a ...interface{}) { + pad := strings.Repeat(" ", pp.depth) + pp.write(pad+f, a...) +} + +func (pp *prettyPrinter) write(f string, a ...interface{}) { + fmt.Fprintf(pp.w, f+"\n", a...) +} diff --git a/vendor/github.com/open-policy-agent/opa/ast/schema.go b/vendor/github.com/open-policy-agent/opa/ast/schema.go new file mode 100644 index 00000000..76bd4756 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/ast/schema.go @@ -0,0 +1,63 @@ +// Copyright 2021 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package ast + +import ( + "fmt" + + "github.com/open-policy-agent/opa/types" + "github.com/open-policy-agent/opa/util" +) + +// SchemaSet holds a map from a path to a schema. +type SchemaSet struct { + m *util.HashMap +} + +// NewSchemaSet returns an empty SchemaSet. +func NewSchemaSet() *SchemaSet { + + eqFunc := func(a, b util.T) bool { + return a.(Ref).Equal(b.(Ref)) + } + + hashFunc := func(x util.T) int { return x.(Ref).Hash() } + + return &SchemaSet{ + m: util.NewHashMap(eqFunc, hashFunc), + } +} + +// Put inserts a raw schema into the set. +func (ss *SchemaSet) Put(path Ref, raw interface{}) { + ss.m.Put(path, raw) +} + +// Get returns the raw schema identified by the path. +func (ss *SchemaSet) Get(path Ref) interface{} { + if ss == nil { + return nil + } + x, ok := ss.m.Get(path) + if !ok { + return nil + } + return x +} + +func loadSchema(raw interface{}, allowNet []string) (types.Type, error) { + + jsonSchema, err := compileSchema(raw, allowNet) + if err != nil { + return nil, err + } + + tpe, err := parseSchema(jsonSchema.RootSchema) + if err != nil { + return nil, fmt.Errorf("type checking: %w", err) + } + + return tpe, nil +} diff --git a/vendor/github.com/open-policy-agent/opa/ast/strings.go b/vendor/github.com/open-policy-agent/opa/ast/strings.go new file mode 100644 index 00000000..8f992801 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/ast/strings.go @@ -0,0 +1,15 @@ +// Copyright 2016 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package ast + +import ( + "reflect" + "strings" +) + +// TypeName returns a human readable name for the AST element type. +func TypeName(x interface{}) string { + return strings.ToLower(reflect.Indirect(reflect.ValueOf(x)).Type().Name()) +} diff --git a/vendor/github.com/open-policy-agent/opa/ast/term.go b/vendor/github.com/open-policy-agent/opa/ast/term.go new file mode 100644 index 00000000..ce254685 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/ast/term.go @@ -0,0 +1,2875 @@ +// Copyright 2016 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +// nolint: deadcode // Public API. +package ast + +import ( + "bytes" + "encoding/json" + "fmt" + "io" + "math" + "math/big" + "net/url" + "regexp" + "sort" + "strconv" + "strings" + + "github.com/OneOfOne/xxhash" + "github.com/pkg/errors" + + "github.com/open-policy-agent/opa/ast/location" + "github.com/open-policy-agent/opa/util" +) + +var errFindNotFound = fmt.Errorf("find: not found") + +// Location records a position in source code. +type Location = location.Location + +// NewLocation returns a new Location object. +func NewLocation(text []byte, file string, row int, col int) *Location { + return location.NewLocation(text, file, row, col) +} + +// Value declares the common interface for all Term values. Every kind of Term value +// in the language is represented as a type that implements this interface: +// +// - Null, Boolean, Number, String +// - Object, Array, Set +// - Variables, References +// - Array, Set, and Object Comprehensions +// - Calls +type Value interface { + Compare(other Value) int // Compare returns <0, 0, or >0 if this Value is less than, equal to, or greater than other, respectively. + Find(path Ref) (Value, error) // Find returns value referred to by path or an error if path is not found. + Hash() int // Returns hash code of the value. + IsGround() bool // IsGround returns true if this value is not a variable or contains no variables. + String() string // String returns a human readable string representation of the value. +} + +// InterfaceToValue converts a native Go value x to a Value. +func InterfaceToValue(x interface{}) (Value, error) { + switch x := x.(type) { + case nil: + return Null{}, nil + case bool: + return Boolean(x), nil + case json.Number: + return Number(x), nil + case int64: + return int64Number(x), nil + case uint64: + return uint64Number(x), nil + case float64: + return floatNumber(x), nil + case int: + return intNumber(x), nil + case string: + return String(x), nil + case []interface{}: + r := make([]*Term, len(x)) + for i, e := range x { + e, err := InterfaceToValue(e) + if err != nil { + return nil, err + } + r[i] = &Term{Value: e} + } + return NewArray(r...), nil + case map[string]interface{}: + r := newobject(len(x)) + for k, v := range x { + k, err := InterfaceToValue(k) + if err != nil { + return nil, err + } + v, err := InterfaceToValue(v) + if err != nil { + return nil, err + } + r.Insert(NewTerm(k), NewTerm(v)) + } + return r, nil + case map[string]string: + r := newobject(len(x)) + for k, v := range x { + k, err := InterfaceToValue(k) + if err != nil { + return nil, err + } + v, err := InterfaceToValue(v) + if err != nil { + return nil, err + } + r.Insert(NewTerm(k), NewTerm(v)) + } + return r, nil + default: + ptr := util.Reference(x) + if err := util.RoundTrip(ptr); err != nil { + return nil, fmt.Errorf("ast: interface conversion: %w", err) + } + return InterfaceToValue(*ptr) + } +} + +// ValueFromReader returns an AST value from a JSON serialized value in the reader. +func ValueFromReader(r io.Reader) (Value, error) { + var x interface{} + if err := util.NewJSONDecoder(r).Decode(&x); err != nil { + return nil, err + } + return InterfaceToValue(x) +} + +// As converts v into a Go native type referred to by x. +func As(v Value, x interface{}) error { + return util.NewJSONDecoder(bytes.NewBufferString(v.String())).Decode(x) +} + +// Resolver defines the interface for resolving references to native Go values. +type Resolver interface { + Resolve(ref Ref) (interface{}, error) +} + +// ValueResolver defines the interface for resolving references to AST values. +type ValueResolver interface { + Resolve(ref Ref) (Value, error) +} + +// UnknownValueErr indicates a ValueResolver was unable to resolve a reference +// because the reference refers to an unknown value. +type UnknownValueErr struct{} + +func (UnknownValueErr) Error() string { + return "unknown value" +} + +// IsUnknownValueErr returns true if the err is an UnknownValueErr. +func IsUnknownValueErr(err error) bool { + _, ok := err.(UnknownValueErr) + return ok +} + +type illegalResolver struct{} + +func (illegalResolver) Resolve(ref Ref) (interface{}, error) { + return nil, fmt.Errorf("illegal value: %v", ref) +} + +// ValueToInterface returns the Go representation of an AST value. The AST +// value should not contain any values that require evaluation (e.g., vars, +// comprehensions, etc.) +func ValueToInterface(v Value, resolver Resolver) (interface{}, error) { + return valueToInterface(v, resolver, JSONOpt{}) +} + +func valueToInterface(v Value, resolver Resolver, opt JSONOpt) (interface{}, error) { + switch v := v.(type) { + case Null: + return nil, nil + case Boolean: + return bool(v), nil + case Number: + return json.Number(v), nil + case String: + return string(v), nil + case *Array: + buf := []interface{}{} + for i := 0; i < v.Len(); i++ { + x1, err := valueToInterface(v.Elem(i).Value, resolver, opt) + if err != nil { + return nil, err + } + buf = append(buf, x1) + } + return buf, nil + case *object: + buf := make(map[string]interface{}, v.Len()) + err := v.Iter(func(k, v *Term) error { + ki, err := valueToInterface(k.Value, resolver, opt) + if err != nil { + return err + } + var str string + var ok bool + if str, ok = ki.(string); !ok { + var buf bytes.Buffer + if err := json.NewEncoder(&buf).Encode(ki); err != nil { + return err + } + str = strings.TrimSpace(buf.String()) + } + vi, err := valueToInterface(v.Value, resolver, opt) + if err != nil { + return err + } + buf[str] = vi + return nil + }) + if err != nil { + return nil, err + } + return buf, nil + case Set: + buf := []interface{}{} + iter := func(x *Term) error { + x1, err := valueToInterface(x.Value, resolver, opt) + if err != nil { + return err + } + buf = append(buf, x1) + return nil + } + var err error + if opt.SortSets { + err = v.Sorted().Iter(iter) + } else { + err = v.Iter(iter) + } + if err != nil { + return nil, err + } + return buf, nil + case Ref: + return resolver.Resolve(v) + default: + return nil, fmt.Errorf("%v requires evaluation", TypeName(v)) + } +} + +// JSON returns the JSON representation of v. The value must not contain any +// refs or terms that require evaluation (e.g., vars, comprehensions, etc.) +func JSON(v Value) (interface{}, error) { + return JSONWithOpt(v, JSONOpt{}) +} + +// JSONOpt defines parameters for AST to JSON conversion. +type JSONOpt struct { + SortSets bool // sort sets before serializing (this makes conversion more expensive) +} + +// JSONWithOpt returns the JSON representation of v. The value must not contain any +// refs or terms that require evaluation (e.g., vars, comprehensions, etc.) +func JSONWithOpt(v Value, opt JSONOpt) (interface{}, error) { + return valueToInterface(v, illegalResolver{}, opt) +} + +// MustJSON returns the JSON representation of v. The value must not contain any +// refs or terms that require evaluation (e.g., vars, comprehensions, etc.) If +// the conversion fails, this function will panic. This function is mostly for +// test purposes. +func MustJSON(v Value) interface{} { + r, err := JSON(v) + if err != nil { + panic(err) + } + return r +} + +// MustInterfaceToValue converts a native Go value x to a Value. If the +// conversion fails, this function will panic. This function is mostly for test +// purposes. +func MustInterfaceToValue(x interface{}) Value { + v, err := InterfaceToValue(x) + if err != nil { + panic(err) + } + return v +} + +// Term is an argument to a function. +type Term struct { + Value Value `json:"value"` // the value of the Term as represented in Go + Location *Location `json:"-"` // the location of the Term in the source +} + +// NewTerm returns a new Term object. +func NewTerm(v Value) *Term { + return &Term{ + Value: v, + } +} + +// SetLocation updates the term's Location and returns the term itself. +func (term *Term) SetLocation(loc *Location) *Term { + term.Location = loc + return term +} + +// Loc returns the Location of term. +func (term *Term) Loc() *Location { + if term == nil { + return nil + } + return term.Location +} + +// SetLoc sets the location on term. +func (term *Term) SetLoc(loc *Location) { + term.SetLocation(loc) +} + +// Copy returns a deep copy of term. +func (term *Term) Copy() *Term { + + if term == nil { + return nil + } + + cpy := *term + + switch v := term.Value.(type) { + case Null, Boolean, Number, String, Var: + cpy.Value = v + case Ref: + cpy.Value = v.Copy() + case *Array: + cpy.Value = v.Copy() + case Set: + cpy.Value = v.Copy() + case *object: + cpy.Value = v.Copy() + case *ArrayComprehension: + cpy.Value = v.Copy() + case *ObjectComprehension: + cpy.Value = v.Copy() + case *SetComprehension: + cpy.Value = v.Copy() + case Call: + cpy.Value = v.Copy() + } + + return &cpy +} + +// Equal returns true if this term equals the other term. Equality is +// defined for each kind of term. +func (term *Term) Equal(other *Term) bool { + if term == nil && other != nil { + return false + } + if term != nil && other == nil { + return false + } + if term == other { + return true + } + + // TODO(tsandall): This early-exit avoids allocations for types that have + // Equal() functions that just use == underneath. We should revisit the + // other types and implement Equal() functions that do not require + // allocations. + switch v := term.Value.(type) { + case Null: + return v.Equal(other.Value) + case Boolean: + return v.Equal(other.Value) + case Number: + return v.Equal(other.Value) + case String: + return v.Equal(other.Value) + case Var: + return v.Equal(other.Value) + } + + return term.Value.Compare(other.Value) == 0 +} + +// Get returns a value referred to by name from the term. +func (term *Term) Get(name *Term) *Term { + switch v := term.Value.(type) { + case *Array: + return v.Get(name) + case *object: + return v.Get(name) + case Set: + if v.Contains(name) { + return name + } + } + return nil +} + +// Hash returns the hash code of the Term's Value. Its Location +// is ignored. +func (term *Term) Hash() int { + return term.Value.Hash() +} + +// IsGround returns true if this term's Value is ground. +func (term *Term) IsGround() bool { + return term.Value.IsGround() +} + +// MarshalJSON returns the JSON encoding of the term. +// +// Specialized marshalling logic is required to include a type hint for Value. +func (term *Term) MarshalJSON() ([]byte, error) { + d := map[string]interface{}{ + "type": TypeName(term.Value), + "value": term.Value, + } + return json.Marshal(d) +} + +func (term *Term) String() string { + return term.Value.String() +} + +// UnmarshalJSON parses the byte array and stores the result in term. +// Specialized unmarshalling is required to handle Value. +func (term *Term) UnmarshalJSON(bs []byte) error { + v := map[string]interface{}{} + if err := util.UnmarshalJSON(bs, &v); err != nil { + return err + } + val, err := unmarshalValue(v) + if err != nil { + return err + } + term.Value = val + return nil +} + +// Vars returns a VarSet with variables contained in this term. +func (term *Term) Vars() VarSet { + vis := &VarVisitor{vars: VarSet{}} + vis.Walk(term) + return vis.vars +} + +// IsConstant returns true if the AST value is constant. +func IsConstant(v Value) bool { + found := false + vis := GenericVisitor{ + func(x interface{}) bool { + switch x.(type) { + case Var, Ref, *ArrayComprehension, *ObjectComprehension, *SetComprehension, Call: + found = true + return true + } + return false + }, + } + vis.Walk(v) + return !found +} + +// IsComprehension returns true if the supplied value is a comprehension. +func IsComprehension(x Value) bool { + switch x.(type) { + case *ArrayComprehension, *ObjectComprehension, *SetComprehension: + return true + } + return false +} + +// ContainsRefs returns true if the Value v contains refs. +func ContainsRefs(v interface{}) bool { + found := false + WalkRefs(v, func(Ref) bool { + found = true + return found + }) + return found +} + +// ContainsComprehensions returns true if the Value v contains comprehensions. +func ContainsComprehensions(v interface{}) bool { + found := false + WalkClosures(v, func(x interface{}) bool { + switch x.(type) { + case *ArrayComprehension, *ObjectComprehension, *SetComprehension: + found = true + return found + } + return found + }) + return found +} + +// IsScalar returns true if the AST value is a scalar. +func IsScalar(v Value) bool { + switch v.(type) { + case String: + return true + case Number: + return true + case Boolean: + return true + case Null: + return true + } + return false +} + +// Null represents the null value defined by JSON. +type Null struct{} + +// NullTerm creates a new Term with a Null value. +func NullTerm() *Term { + return &Term{Value: Null{}} +} + +// Equal returns true if the other term Value is also Null. +func (null Null) Equal(other Value) bool { + switch other.(type) { + case Null: + return true + default: + return false + } +} + +// Compare compares null to other, return <0, 0, or >0 if it is less than, equal to, +// or greater than other. +func (null Null) Compare(other Value) int { + return Compare(null, other) +} + +// Find returns the current value or a not found error. +func (null Null) Find(path Ref) (Value, error) { + if len(path) == 0 { + return null, nil + } + return nil, errFindNotFound +} + +// Hash returns the hash code for the Value. +func (null Null) Hash() int { + return 0 +} + +// IsGround always returns true. +func (Null) IsGround() bool { + return true +} + +func (null Null) String() string { + return "null" +} + +// Boolean represents a boolean value defined by JSON. +type Boolean bool + +// BooleanTerm creates a new Term with a Boolean value. +func BooleanTerm(b bool) *Term { + return &Term{Value: Boolean(b)} +} + +// Equal returns true if the other Value is a Boolean and is equal. +func (bol Boolean) Equal(other Value) bool { + switch other := other.(type) { + case Boolean: + return bol == other + default: + return false + } +} + +// Compare compares bol to other, return <0, 0, or >0 if it is less than, equal to, +// or greater than other. +func (bol Boolean) Compare(other Value) int { + return Compare(bol, other) +} + +// Find returns the current value or a not found error. +func (bol Boolean) Find(path Ref) (Value, error) { + if len(path) == 0 { + return bol, nil + } + return nil, errFindNotFound +} + +// Hash returns the hash code for the Value. +func (bol Boolean) Hash() int { + if bol { + return 1 + } + return 0 +} + +// IsGround always returns true. +func (Boolean) IsGround() bool { + return true +} + +func (bol Boolean) String() string { + return strconv.FormatBool(bool(bol)) +} + +// Number represents a numeric value as defined by JSON. +type Number json.Number + +// NumberTerm creates a new Term with a Number value. +func NumberTerm(n json.Number) *Term { + return &Term{Value: Number(n)} +} + +// IntNumberTerm creates a new Term with an integer Number value. +func IntNumberTerm(i int) *Term { + return &Term{Value: Number(strconv.Itoa(i))} +} + +// UIntNumberTerm creates a new Term with an unsigned integer Number value. +func UIntNumberTerm(u uint64) *Term { + return &Term{Value: uint64Number(u)} +} + +// FloatNumberTerm creates a new Term with a floating point Number value. +func FloatNumberTerm(f float64) *Term { + s := strconv.FormatFloat(f, 'g', -1, 64) + return &Term{Value: Number(s)} +} + +// Equal returns true if the other Value is a Number and is equal. +func (num Number) Equal(other Value) bool { + switch other := other.(type) { + case Number: + return Compare(num, other) == 0 + default: + return false + } +} + +// Compare compares num to other, return <0, 0, or >0 if it is less than, equal to, +// or greater than other. +func (num Number) Compare(other Value) int { + return Compare(num, other) +} + +// Find returns the current value or a not found error. +func (num Number) Find(path Ref) (Value, error) { + if len(path) == 0 { + return num, nil + } + return nil, errFindNotFound +} + +// Hash returns the hash code for the Value. +func (num Number) Hash() int { + f, err := json.Number(num).Float64() + if err != nil { + bs := []byte(num) + h := xxhash.Checksum64(bs) + return int(h) + } + return int(f) +} + +// Int returns the int representation of num if possible. +func (num Number) Int() (int, bool) { + i64, ok := num.Int64() + return int(i64), ok +} + +// Int64 returns the int64 representation of num if possible. +func (num Number) Int64() (int64, bool) { + i, err := json.Number(num).Int64() + if err != nil { + return 0, false + } + return i, true +} + +// Float64 returns the float64 representation of num if possible. +func (num Number) Float64() (float64, bool) { + f, err := json.Number(num).Float64() + if err != nil { + return 0, false + } + return f, true +} + +// IsGround always returns true. +func (Number) IsGround() bool { + return true +} + +// MarshalJSON returns JSON encoded bytes representing num. +func (num Number) MarshalJSON() ([]byte, error) { + return json.Marshal(json.Number(num)) +} + +func (num Number) String() string { + return string(num) +} + +func intNumber(i int) Number { + return Number(strconv.Itoa(i)) +} + +func int64Number(i int64) Number { + return Number(strconv.FormatInt(i, 10)) +} + +func uint64Number(u uint64) Number { + return Number(strconv.FormatUint(u, 10)) +} + +func floatNumber(f float64) Number { + return Number(strconv.FormatFloat(f, 'g', -1, 64)) +} + +// String represents a string value as defined by JSON. +type String string + +// StringTerm creates a new Term with a String value. +func StringTerm(s string) *Term { + return &Term{Value: String(s)} +} + +// Equal returns true if the other Value is a String and is equal. +func (str String) Equal(other Value) bool { + switch other := other.(type) { + case String: + return str == other + default: + return false + } +} + +// Compare compares str to other, return <0, 0, or >0 if it is less than, equal to, +// or greater than other. +func (str String) Compare(other Value) int { + return Compare(str, other) +} + +// Find returns the current value or a not found error. +func (str String) Find(path Ref) (Value, error) { + if len(path) == 0 { + return str, nil + } + return nil, errFindNotFound +} + +// IsGround always returns true. +func (String) IsGround() bool { + return true +} + +func (str String) String() string { + return strconv.Quote(string(str)) +} + +// Hash returns the hash code for the Value. +func (str String) Hash() int { + h := xxhash.ChecksumString64S(string(str), hashSeed0) + return int(h) +} + +// Var represents a variable as defined by the language. +type Var string + +// VarTerm creates a new Term with a Variable value. +func VarTerm(v string) *Term { + return &Term{Value: Var(v)} +} + +// Equal returns true if the other Value is a Variable and has the same value +// (name). +func (v Var) Equal(other Value) bool { + switch other := other.(type) { + case Var: + return v == other + default: + return false + } +} + +// Compare compares v to other, return <0, 0, or >0 if it is less than, equal to, +// or greater than other. +func (v Var) Compare(other Value) int { + return Compare(v, other) +} + +// Find returns the current value or a not found error. +func (v Var) Find(path Ref) (Value, error) { + if len(path) == 0 { + return v, nil + } + return nil, errFindNotFound +} + +// Hash returns the hash code for the Value. +func (v Var) Hash() int { + h := xxhash.ChecksumString64S(string(v), hashSeed0) + return int(h) +} + +// IsGround always returns false. +func (Var) IsGround() bool { + return false +} + +// IsWildcard returns true if this is a wildcard variable. +func (v Var) IsWildcard() bool { + return strings.HasPrefix(string(v), WildcardPrefix) +} + +// IsGenerated returns true if this variable was generated during compilation. +func (v Var) IsGenerated() bool { + return strings.HasPrefix(string(v), "__local") +} + +func (v Var) String() string { + // Special case for wildcard so that string representation is parseable. The + // parser mangles wildcard variables to make their names unique and uses an + // illegal variable name character (WildcardPrefix) to avoid conflicts. When + // we serialize the variable here, we need to make sure it's parseable. + if v.IsWildcard() { + return Wildcard.String() + } + return string(v) +} + +// Ref represents a reference as defined by the language. +type Ref []*Term + +// EmptyRef returns a new, empty reference. +func EmptyRef() Ref { + return Ref([]*Term{}) +} + +// PtrRef returns a new reference against the head for the pointer +// s. Path components in the pointer are unescaped. +func PtrRef(head *Term, s string) (Ref, error) { + s = strings.Trim(s, "/") + if s == "" { + return Ref{head}, nil + } + parts := strings.Split(s, "/") + if max := math.MaxInt32; len(parts) >= max { + return nil, fmt.Errorf("path too long: %s, %d > %d (max)", s, len(parts), max) + } + ref := make(Ref, uint(len(parts))+1) + ref[0] = head + for i := 0; i < len(parts); i++ { + var err error + parts[i], err = url.PathUnescape(parts[i]) + if err != nil { + return nil, err + } + ref[i+1] = StringTerm(parts[i]) + } + return ref, nil +} + +// RefTerm creates a new Term with a Ref value. +func RefTerm(r ...*Term) *Term { + return &Term{Value: Ref(r)} +} + +// Append returns a copy of ref with the term appended to the end. +func (ref Ref) Append(term *Term) Ref { + n := len(ref) + dst := make(Ref, n+1) + copy(dst, ref) + dst[n] = term + return dst +} + +// Insert returns a copy of the ref with x inserted at pos. If pos < len(ref), +// existing elements are shifted to the right. If pos > len(ref)+1 this +// function panics. +func (ref Ref) Insert(x *Term, pos int) Ref { + if pos == len(ref) { + return ref.Append(x) + } else if pos > len(ref)+1 { + panic("illegal index") + } + cpy := make(Ref, len(ref)+1) + for i := 0; i < pos; i++ { + cpy[i] = ref[i] + } + cpy[pos] = x + for i := pos; i < len(ref); i++ { + cpy[i+1] = ref[i] + } + return cpy +} + +// Extend returns a copy of ref with the terms from other appended. The head of +// other will be converted to a string. +func (ref Ref) Extend(other Ref) Ref { + dst := make(Ref, len(ref)+len(other)) + copy(dst, ref) + + head := other[0].Copy() + head.Value = String(head.Value.(Var)) + offset := len(ref) + dst[offset] = head + for i := range other[1:] { + dst[offset+i+1] = other[i+1] + } + return dst +} + +// Concat returns a ref with the terms appended. +func (ref Ref) Concat(terms []*Term) Ref { + if len(terms) == 0 { + return ref + } + cpy := make(Ref, len(ref)+len(terms)) + copy(cpy, ref) + + for i := range terms { + cpy[len(ref)+i] = terms[i] + } + return cpy +} + +// Dynamic returns the offset of the first non-constant operand of ref. +func (ref Ref) Dynamic() int { + switch ref[0].Value.(type) { + case Call: + return 0 + } + for i := 1; i < len(ref); i++ { + if !IsConstant(ref[i].Value) { + return i + } + } + return -1 +} + +// Copy returns a deep copy of ref. +func (ref Ref) Copy() Ref { + return termSliceCopy(ref) +} + +// Equal returns true if ref is equal to other. +func (ref Ref) Equal(other Value) bool { + return Compare(ref, other) == 0 +} + +// Compare compares ref to other, return <0, 0, or >0 if it is less than, equal to, +// or greater than other. +func (ref Ref) Compare(other Value) int { + return Compare(ref, other) +} + +// Find returns the current value or a "not found" error. +func (ref Ref) Find(path Ref) (Value, error) { + if len(path) == 0 { + return ref, nil + } + return nil, errFindNotFound +} + +// Hash returns the hash code for the Value. +func (ref Ref) Hash() int { + return termSliceHash(ref) +} + +// HasPrefix returns true if the other ref is a prefix of this ref. +func (ref Ref) HasPrefix(other Ref) bool { + if len(other) > len(ref) { + return false + } + for i := range other { + if !ref[i].Equal(other[i]) { + return false + } + } + return true +} + +// ConstantPrefix returns the constant portion of the ref starting from the head. +func (ref Ref) ConstantPrefix() Ref { + ref = ref.Copy() + + i := ref.Dynamic() + if i < 0 { + return ref + } + return ref[:i] +} + +// GroundPrefix returns the ground portion of the ref starting from the head. By +// definition, the head of the reference is always ground. +func (ref Ref) GroundPrefix() Ref { + prefix := make(Ref, 0, len(ref)) + + for i, x := range ref { + if i > 0 && !x.IsGround() { + break + } + prefix = append(prefix, x) + } + + return prefix +} + +// IsGround returns true if all of the parts of the Ref are ground. +func (ref Ref) IsGround() bool { + if len(ref) == 0 { + return true + } + return termSliceIsGround(ref[1:]) +} + +// IsNested returns true if this ref contains other Refs. +func (ref Ref) IsNested() bool { + for _, x := range ref { + if _, ok := x.Value.(Ref); ok { + return true + } + } + return false +} + +// Ptr returns a slash-separated path string for this ref. If the ref +// contains non-string terms this function returns an error. Path +// components are escaped. +func (ref Ref) Ptr() (string, error) { + parts := make([]string, 0, len(ref)-1) + for _, term := range ref[1:] { + if str, ok := term.Value.(String); ok { + parts = append(parts, url.PathEscape(string(str))) + } else { + return "", fmt.Errorf("invalid path value type") + } + } + return strings.Join(parts, "/"), nil +} + +var varRegexp = regexp.MustCompile("^[[:alpha:]_][[:alpha:][:digit:]_]*$") + +func (ref Ref) String() string { + if len(ref) == 0 { + return "" + } + buf := []string{ref[0].Value.String()} + path := ref[1:] + for _, p := range path { + switch p := p.Value.(type) { + case String: + str := string(p) + if varRegexp.MatchString(str) && len(buf) > 0 && !IsKeyword(str) { + buf = append(buf, "."+str) + } else { + buf = append(buf, "["+p.String()+"]") + } + default: + buf = append(buf, "["+p.String()+"]") + } + } + return strings.Join(buf, "") +} + +// OutputVars returns a VarSet containing variables that would be bound by evaluating +// this expression in isolation. +func (ref Ref) OutputVars() VarSet { + vis := NewVarVisitor().WithParams(VarVisitorParams{SkipRefHead: true}) + vis.Walk(ref) + return vis.Vars() +} + +// QueryIterator defines the interface for querying AST documents with references. +type QueryIterator func(map[Var]Value, Value) error + +// ArrayTerm creates a new Term with an Array value. +func ArrayTerm(a ...*Term) *Term { + return &Term{Value: &Array{elems: a, hash: 0, ground: termSliceIsGround(a)}} +} + +// NewArray creates an Array with the terms provided. The array will +// use the provided term slice. +func NewArray(a ...*Term) *Array { + return &Array{elems: a, hash: 0, ground: termSliceIsGround(a)} +} + +// Array represents an array as defined by the language. Arrays are similar to the +// same types as defined by JSON with the exception that they can contain Vars +// and References. +type Array struct { + elems []*Term + hash int + ground bool +} + +// Copy returns a deep copy of arr. +func (arr *Array) Copy() *Array { + return &Array{ + elems: termSliceCopy(arr.elems), + hash: arr.hash, + ground: arr.IsGround()} +} + +// Equal returns true if arr is equal to other. +func (arr *Array) Equal(other Value) bool { + return Compare(arr, other) == 0 +} + +// Compare compares arr to other, return <0, 0, or >0 if it is less than, equal to, +// or greater than other. +func (arr *Array) Compare(other Value) int { + return Compare(arr, other) +} + +// Find returns the value at the index or an out-of-range error. +func (arr *Array) Find(path Ref) (Value, error) { + if len(path) == 0 { + return arr, nil + } + num, ok := path[0].Value.(Number) + if !ok { + return nil, errFindNotFound + } + i, ok := num.Int() + if !ok { + return nil, errFindNotFound + } + if i < 0 || i >= arr.Len() { + return nil, errFindNotFound + } + return arr.Elem(i).Value.Find(path[1:]) +} + +// Get returns the element at pos or nil if not possible. +func (arr *Array) Get(pos *Term) *Term { + num, ok := pos.Value.(Number) + if !ok { + return nil + } + + i, ok := num.Int() + if !ok { + return nil + } + + if i >= 0 && i < len(arr.elems) { + return arr.elems[i] + } + + return nil +} + +// Sorted returns a new Array that contains the sorted elements of arr. +func (arr *Array) Sorted() *Array { + cpy := make([]*Term, len(arr.elems)) + for i := range cpy { + cpy[i] = arr.elems[i] + } + sort.Sort(termSlice(cpy)) + a := NewArray(cpy...) + a.hash = arr.hash + return a +} + +// Hash returns the hash code for the Value. +func (arr *Array) Hash() int { + if arr.hash == 0 { + arr.hash = termSliceHash(arr.elems) + } + + return arr.hash +} + +// IsGround returns true if all of the Array elements are ground. +func (arr *Array) IsGround() bool { + return arr.ground +} + +// MarshalJSON returns JSON encoded bytes representing arr. +func (arr *Array) MarshalJSON() ([]byte, error) { + if len(arr.elems) == 0 { + return []byte(`[]`), nil + } + return json.Marshal(arr.elems) +} + +func (arr *Array) String() string { + var b strings.Builder + b.WriteRune('[') + for i, e := range arr.elems { + if i > 0 { + b.WriteString(", ") + } + b.WriteString(e.String()) + } + b.WriteRune(']') + return b.String() +} + +// Len returns the number of elements in the array. +func (arr *Array) Len() int { + return len(arr.elems) +} + +// Elem returns the element i of arr. +func (arr *Array) Elem(i int) *Term { + return arr.elems[i] +} + +// set sets the element i of arr. +func (arr *Array) set(i int, v *Term) { + arr.ground = arr.ground && v.IsGround() + arr.elems[i] = v + arr.hash = 0 +} + +// Slice returns a slice of arr starting from i index to j. -1 +// indicates the end of the array. The returned value array is not a +// copy and any modifications to either of arrays may be reflected to +// the other. +func (arr *Array) Slice(i, j int) *Array { + var elems []*Term + if j == -1 { + elems = arr.elems[i:] + } else { + elems = arr.elems[i:j] + } + // If arr is ground, the slice is, too. + // If it's not, the slice could still be. + gr := arr.ground || termSliceIsGround(elems) + return &Array{elems: elems, ground: gr} +} + +// Iter calls f on each element in arr. If f returns an error, +// iteration stops and the return value is the error. +func (arr *Array) Iter(f func(*Term) error) error { + for i := range arr.elems { + if err := f(arr.elems[i]); err != nil { + return err + } + } + return nil +} + +// Until calls f on each element in arr. If f returns true, iteration stops. +func (arr *Array) Until(f func(*Term) bool) bool { + err := arr.Iter(func(t *Term) error { + if f(t) { + return errStop + } + return nil + }) + return err != nil +} + +// Foreach calls f on each element in arr. +func (arr *Array) Foreach(f func(*Term)) { + _ = arr.Iter(func(t *Term) error { + f(t) + return nil + }) // ignore error +} + +// Append appends a term to arr, returning the appended array. +func (arr *Array) Append(v *Term) *Array { + cpy := *arr + cpy.elems = append(arr.elems, v) + cpy.hash = 0 + cpy.ground = arr.ground && v.IsGround() + return &cpy +} + +// Set represents a set as defined by the language. +type Set interface { + Value + Len() int + Copy() Set + Diff(Set) Set + Intersect(Set) Set + Union(Set) Set + Add(*Term) + Iter(func(*Term) error) error + Until(func(*Term) bool) bool + Foreach(func(*Term)) + Contains(*Term) bool + Map(func(*Term) (*Term, error)) (Set, error) + Reduce(*Term, func(*Term, *Term) (*Term, error)) (*Term, error) + Sorted() *Array + Slice() []*Term +} + +// NewSet returns a new Set containing t. +func NewSet(t ...*Term) Set { + s := newset(len(t)) + for i := range t { + s.Add(t[i]) + } + return s +} + +func newset(n int) *set { + var keys []*Term + if n > 0 { + keys = make([]*Term, 0, n) + } + return &set{ + elems: make(map[int]*Term, n), + keys: keys, + hash: 0, + ground: true, + } +} + +// SetTerm returns a new Term representing a set containing terms t. +func SetTerm(t ...*Term) *Term { + set := NewSet(t...) + return &Term{ + Value: set, + } +} + +type set struct { + elems map[int]*Term + keys []*Term + hash int + ground bool +} + +// Copy returns a deep copy of s. +func (s *set) Copy() Set { + cpy := newset(s.Len()) + s.Foreach(func(x *Term) { + cpy.Add(x.Copy()) + }) + cpy.hash = s.hash + cpy.ground = s.ground + return cpy +} + +// IsGround returns true if all terms in s are ground. +func (s *set) IsGround() bool { + return s.ground +} + +// Hash returns a hash code for s. +func (s *set) Hash() int { + if s.hash == 0 { + s.Foreach(func(x *Term) { + s.hash += x.Hash() + }) + } + return s.hash +} + +func (s *set) String() string { + if s.Len() == 0 { + return "set()" + } + var b strings.Builder + b.WriteRune('{') + for i := range s.keys { + if i > 0 { + b.WriteString(", ") + } + b.WriteString(s.keys[i].Value.String()) + } + b.WriteRune('}') + return b.String() +} + +// Compare compares s to other, return <0, 0, or >0 if it is less than, equal to, +// or greater than other. +func (s *set) Compare(other Value) int { + o1 := sortOrder(s) + o2 := sortOrder(other) + if o1 < o2 { + return -1 + } else if o1 > o2 { + return 1 + } + t := other.(*set) + return termSliceCompare(s.keys, t.keys) +} + +// Find returns the set or dereferences the element itself. +func (s *set) Find(path Ref) (Value, error) { + if len(path) == 0 { + return s, nil + } + if !s.Contains(path[0]) { + return nil, errFindNotFound + } + return path[0].Value.Find(path[1:]) +} + +// Diff returns elements in s that are not in other. +func (s *set) Diff(other Set) Set { + r := NewSet() + s.Foreach(func(x *Term) { + if !other.Contains(x) { + r.Add(x) + } + }) + return r +} + +// Intersect returns the set containing elements in both s and other. +func (s *set) Intersect(other Set) Set { + o := other.(*set) + n, m := s.Len(), o.Len() + ss := s + so := o + if m < n { + ss = o + so = s + n = m + } + + r := newset(n) + ss.Foreach(func(x *Term) { + if so.Contains(x) { + r.Add(x) + } + }) + return r +} + +// Union returns the set containing all elements of s and other. +func (s *set) Union(other Set) Set { + r := NewSet() + s.Foreach(func(x *Term) { + r.Add(x) + }) + other.Foreach(func(x *Term) { + r.Add(x) + }) + return r +} + +// Add updates s to include t. +func (s *set) Add(t *Term) { + s.insert(t) +} + +// Iter calls f on each element in s. If f returns an error, iteration stops +// and the return value is the error. +func (s *set) Iter(f func(*Term) error) error { + for i := range s.keys { + if err := f(s.keys[i]); err != nil { + return err + } + } + return nil +} + +var errStop = errors.New("stop") + +// Until calls f on each element in s. If f returns true, iteration stops. +func (s *set) Until(f func(*Term) bool) bool { + err := s.Iter(func(t *Term) error { + if f(t) { + return errStop + } + return nil + }) + return err != nil +} + +// Foreach calls f on each element in s. +func (s *set) Foreach(f func(*Term)) { + _ = s.Iter(func(t *Term) error { + f(t) + return nil + }) // ignore error +} + +// Map returns a new Set obtained by applying f to each value in s. +func (s *set) Map(f func(*Term) (*Term, error)) (Set, error) { + set := NewSet() + err := s.Iter(func(x *Term) error { + term, err := f(x) + if err != nil { + return err + } + set.Add(term) + return nil + }) + if err != nil { + return nil, err + } + return set, nil +} + +// Reduce returns a Term produced by applying f to each value in s. The first +// argument to f is the reduced value (starting with i) and the second argument +// to f is the element in s. +func (s *set) Reduce(i *Term, f func(*Term, *Term) (*Term, error)) (*Term, error) { + err := s.Iter(func(x *Term) error { + var err error + i, err = f(i, x) + if err != nil { + return err + } + return nil + }) + return i, err +} + +// Contains returns true if t is in s. +func (s *set) Contains(t *Term) bool { + return s.get(t) != nil +} + +// Len returns the number of elements in the set. +func (s *set) Len() int { + return len(s.keys) +} + +// MarshalJSON returns JSON encoded bytes representing s. +func (s *set) MarshalJSON() ([]byte, error) { + if s.keys == nil { + return []byte(`[]`), nil + } + return json.Marshal(s.keys) +} + +// Sorted returns an Array that contains the sorted elements of s. +func (s *set) Sorted() *Array { + cpy := make([]*Term, len(s.keys)) + for i := range s.keys { + cpy[i] = s.keys[i] + } + sort.Sort(termSlice(cpy)) + return NewArray(cpy...) +} + +// Slice returns a slice of terms contained in the set. +func (s *set) Slice() []*Term { + return s.keys +} + +func (s *set) insert(x *Term) { + hash := x.Hash() + // This `equal` utility is duplicated and manually inlined a number of + // time in this file. Inlining it avoids heap allocations, so it makes + // a big performance difference: some operations like lookup become twice + // as slow without it. + var equal func(v Value) bool + + switch x := x.Value.(type) { + case Null, Boolean, String, Var: + equal = func(y Value) bool { return x == y } + case Number: + if xi, err := json.Number(x).Int64(); err == nil { + equal = func(y Value) bool { + if y, ok := y.(Number); ok { + if yi, err := json.Number(y).Int64(); err == nil { + return xi == yi + } + } + + return false + } + break + } + + // We use big.Rat for comparing big numbers. + // It replaces big.Float due to following reason: + // big.Float comes with a default precision of 64, and setting a + // larger precision results in more memory being allocated + // (regardless of the actual number we are parsing with SetString). + // + // Note: If we're so close to zero that big.Float says we are zero, do + // *not* big.Rat).SetString on the original string it'll potentially + // take very long. + var a *big.Rat + fa, ok := new(big.Float).SetString(string(x)) + if !ok { + panic("illegal value") + } + if fa.IsInt() { + if i, _ := fa.Int64(); i == 0 { + a = new(big.Rat).SetInt64(0) + } + } + if a == nil { + a, ok = new(big.Rat).SetString(string(x)) + if !ok { + panic("illegal value") + } + } + + equal = func(b Value) bool { + if bNum, ok := b.(Number); ok { + var b *big.Rat + fb, ok := new(big.Float).SetString(string(bNum)) + if !ok { + panic("illegal value") + } + if fb.IsInt() { + if i, _ := fb.Int64(); i == 0 { + b = new(big.Rat).SetInt64(0) + } + } + if b == nil { + b, ok = new(big.Rat).SetString(string(bNum)) + if !ok { + panic("illegal value") + } + } + + return a.Cmp(b) == 0 + } + + return false + } + default: + equal = func(y Value) bool { return Compare(x, y) == 0 } + } + + for curr, ok := s.elems[hash]; ok; { + if equal(curr.Value) { + return + } + + hash++ + curr, ok = s.elems[hash] + } + + s.elems[hash] = x + i := sort.Search(len(s.keys), func(i int) bool { return Compare(x, s.keys[i]) < 0 }) + if i < len(s.keys) { + // insert at position `i`: + s.keys = append(s.keys, nil) // add some space + copy(s.keys[i+1:], s.keys[i:]) // move things over + s.keys[i] = x // drop it in position + } else { + s.keys = append(s.keys, x) + } + + s.hash = 0 + s.ground = s.ground && x.IsGround() +} + +func (s *set) get(x *Term) *Term { + hash := x.Hash() + // This `equal` utility is duplicated and manually inlined a number of + // time in this file. Inlining it avoids heap allocations, so it makes + // a big performance difference: some operations like lookup become twice + // as slow without it. + var equal func(v Value) bool + + switch x := x.Value.(type) { + case Null, Boolean, String, Var: + equal = func(y Value) bool { return x == y } + case Number: + if xi, err := json.Number(x).Int64(); err == nil { + equal = func(y Value) bool { + if y, ok := y.(Number); ok { + if yi, err := json.Number(y).Int64(); err == nil { + return xi == yi + } + } + + return false + } + break + } + + // We use big.Rat for comparing big numbers. + // It replaces big.Float due to following reason: + // big.Float comes with a default precision of 64, and setting a + // larger precision results in more memory being allocated + // (regardless of the actual number we are parsing with SetString). + // + // Note: If we're so close to zero that big.Float says we are zero, do + // *not* big.Rat).SetString on the original string it'll potentially + // take very long. + var a *big.Rat + fa, ok := new(big.Float).SetString(string(x)) + if !ok { + panic("illegal value") + } + if fa.IsInt() { + if i, _ := fa.Int64(); i == 0 { + a = new(big.Rat).SetInt64(0) + } + } + if a == nil { + a, ok = new(big.Rat).SetString(string(x)) + if !ok { + panic("illegal value") + } + } + + equal = func(b Value) bool { + if bNum, ok := b.(Number); ok { + var b *big.Rat + fb, ok := new(big.Float).SetString(string(bNum)) + if !ok { + panic("illegal value") + } + if fb.IsInt() { + if i, _ := fb.Int64(); i == 0 { + b = new(big.Rat).SetInt64(0) + } + } + if b == nil { + b, ok = new(big.Rat).SetString(string(bNum)) + if !ok { + panic("illegal value") + } + } + + return a.Cmp(b) == 0 + } + return false + + } + + default: + equal = func(y Value) bool { return Compare(x, y) == 0 } + } + + for curr, ok := s.elems[hash]; ok; { + if equal(curr.Value) { + return curr + } + + hash++ + curr, ok = s.elems[hash] + } + return nil +} + +// Object represents an object as defined by the language. +type Object interface { + Value + Len() int + Get(*Term) *Term + Copy() Object + Insert(*Term, *Term) + Iter(func(*Term, *Term) error) error + Until(func(*Term, *Term) bool) bool + Foreach(func(*Term, *Term)) + Map(func(*Term, *Term) (*Term, *Term, error)) (Object, error) + Diff(other Object) Object + Intersect(other Object) [][3]*Term + Merge(other Object) (Object, bool) + MergeWith(other Object, conflictResolver func(v1, v2 *Term) (*Term, bool)) (Object, bool) + Filter(filter Object) (Object, error) + Keys() []*Term + Elem(i int) (*Term, *Term) + get(k *Term) *objectElem // To prevent external implementations +} + +// NewObject creates a new Object with t. +func NewObject(t ...[2]*Term) Object { + obj := newobject(len(t)) + for i := range t { + obj.Insert(t[i][0], t[i][1]) + } + return obj +} + +// ObjectTerm creates a new Term with an Object value. +func ObjectTerm(o ...[2]*Term) *Term { + return &Term{Value: NewObject(o...)} +} + +type object struct { + elems map[int]*objectElem + keys objectElemSlice + ground int // number of key and value grounds. Counting is + // required to support insert's key-value replace. + hash int +} + +func newobject(n int) *object { + var keys objectElemSlice + if n > 0 { + keys = make(objectElemSlice, 0, n) + } + return &object{ + elems: make(map[int]*objectElem, n), + keys: keys, + ground: 0, + hash: 0, + } +} + +type objectElem struct { + key *Term + value *Term + next *objectElem +} + +type objectElemSlice []*objectElem + +func (s objectElemSlice) Less(i, j int) bool { return Compare(s[i].key.Value, s[j].key.Value) < 0 } +func (s objectElemSlice) Swap(i, j int) { x := s[i]; s[i] = s[j]; s[j] = x } +func (s objectElemSlice) Len() int { return len(s) } + +// Item is a helper for constructing an tuple containing two Terms +// representing a key/value pair in an Object. +func Item(key, value *Term) [2]*Term { + return [2]*Term{key, value} +} + +// Compare compares obj to other, return <0, 0, or >0 if it is less than, equal to, +// or greater than other. +func (obj *object) Compare(other Value) int { + o1 := sortOrder(obj) + o2 := sortOrder(other) + if o1 < o2 { + return -1 + } else if o2 < o1 { + return 1 + } + a := obj + b := other.(*object) + minLen := len(a.keys) + if len(b.keys) < len(a.keys) { + minLen = len(b.keys) + } + for i := 0; i < minLen; i++ { + keysCmp := Compare(a.keys[i].key, b.keys[i].key) + if keysCmp < 0 { + return -1 + } + if keysCmp > 0 { + return 1 + } + valA := a.keys[i].value + valB := b.keys[i].value + valCmp := Compare(valA, valB) + if valCmp != 0 { + return valCmp + } + } + if len(a.keys) < len(b.keys) { + return -1 + } + if len(b.keys) < len(a.keys) { + return 1 + } + return 0 +} + +// Find returns the value at the key or undefined. +func (obj *object) Find(path Ref) (Value, error) { + if len(path) == 0 { + return obj, nil + } + value := obj.Get(path[0]) + if value == nil { + return nil, errFindNotFound + } + return value.Value.Find(path[1:]) +} + +func (obj *object) Insert(k, v *Term) { + obj.insert(k, v) +} + +// Get returns the value of k in obj if k exists, otherwise nil. +func (obj *object) Get(k *Term) *Term { + if elem := obj.get(k); elem != nil { + return elem.value + } + return nil +} + +// Hash returns the hash code for the Value. +func (obj *object) Hash() int { + if obj.hash == 0 { + for h, curr := range obj.elems { + for ; curr != nil; curr = curr.next { + obj.hash += h + obj.hash += curr.value.Hash() + } + } + } + return obj.hash +} + +// IsGround returns true if all of the Object key/value pairs are ground. +func (obj *object) IsGround() bool { + return obj.ground == 2*len(obj.keys) +} + +// Copy returns a deep copy of obj. +func (obj *object) Copy() Object { + cpy, _ := obj.Map(func(k, v *Term) (*Term, *Term, error) { + return k.Copy(), v.Copy(), nil + }) + cpy.(*object).hash = obj.hash + return cpy +} + +// Diff returns a new Object that contains only the key/value pairs that exist in obj. +func (obj *object) Diff(other Object) Object { + r := NewObject() + obj.Foreach(func(k, v *Term) { + if other.Get(k) == nil { + r.Insert(k, v) + } + }) + return r +} + +// Intersect returns a slice of term triplets that represent the intersection of keys +// between obj and other. For each intersecting key, the values from obj and other are included +// as the last two terms in the triplet (respectively). +func (obj *object) Intersect(other Object) [][3]*Term { + r := [][3]*Term{} + obj.Foreach(func(k, v *Term) { + if v2 := other.Get(k); v2 != nil { + r = append(r, [3]*Term{k, v, v2}) + } + }) + return r +} + +// Iter calls the function f for each key-value pair in the object. If f +// returns an error, iteration stops and the error is returned. +func (obj *object) Iter(f func(*Term, *Term) error) error { + for _, node := range obj.keys { + if err := f(node.key, node.value); err != nil { + return err + } + } + return nil +} + +// Until calls f for each key-value pair in the object. If f returns +// true, iteration stops and Until returns true. Otherwise, return +// false. +func (obj *object) Until(f func(*Term, *Term) bool) bool { + err := obj.Iter(func(k, v *Term) error { + if f(k, v) { + return errStop + } + return nil + }) + return err != nil +} + +// Foreach calls f for each key-value pair in the object. +func (obj *object) Foreach(f func(*Term, *Term)) { + _ = obj.Iter(func(k, v *Term) error { + f(k, v) + return nil + }) // ignore error +} + +// Map returns a new Object constructed by mapping each element in the object +// using the function f. +func (obj *object) Map(f func(*Term, *Term) (*Term, *Term, error)) (Object, error) { + cpy := newobject(obj.Len()) + err := obj.Iter(func(k, v *Term) error { + var err error + k, v, err = f(k, v) + if err != nil { + return err + } + cpy.insert(k, v) + return nil + }) + if err != nil { + return nil, err + } + return cpy, nil +} + +// Keys returns the keys of obj. +func (obj *object) Keys() []*Term { + keys := make([]*Term, len(obj.keys)) + + for i, elem := range obj.keys { + keys[i] = elem.key + } + + return keys +} + +func (obj *object) Elem(i int) (*Term, *Term) { + return obj.keys[i].key, obj.keys[i].value +} + +// MarshalJSON returns JSON encoded bytes representing obj. +func (obj *object) MarshalJSON() ([]byte, error) { + sl := make([][2]*Term, obj.Len()) + for i, node := range obj.keys { + sl[i] = Item(node.key, node.value) + } + return json.Marshal(sl) +} + +// Merge returns a new Object containing the non-overlapping keys of obj and other. If there are +// overlapping keys between obj and other, the values of associated with the keys are merged. Only +// objects can be merged with other objects. If the values cannot be merged, the second turn value +// will be false. +func (obj object) Merge(other Object) (Object, bool) { + return obj.MergeWith(other, func(v1, v2 *Term) (*Term, bool) { + obj1, ok1 := v1.Value.(Object) + obj2, ok2 := v2.Value.(Object) + if !ok1 || !ok2 { + return nil, true + } + obj3, ok := obj1.Merge(obj2) + if !ok { + return nil, true + } + return NewTerm(obj3), false + }) +} + +// MergeWith returns a new Object containing the merged keys of obj and other. +// If there are overlapping keys between obj and other, the conflictResolver +// is called. The conflictResolver can return a merged value and a boolean +// indicating if the merge has failed and should stop. +func (obj object) MergeWith(other Object, conflictResolver func(v1, v2 *Term) (*Term, bool)) (Object, bool) { + result := NewObject() + stop := obj.Until(func(k, v *Term) bool { + v2 := other.Get(k) + // The key didn't exist in other, keep the original value + if v2 == nil { + result.Insert(k, v) + return false + } + + // The key exists in both, resolve the conflict if possible + merged, stop := conflictResolver(v, v2) + if !stop { + result.Insert(k, merged) + } + return stop + }) + + if stop { + return nil, false + } + + // Copy in any values from other for keys that don't exist in obj + other.Foreach(func(k, v *Term) { + if v2 := obj.Get(k); v2 == nil { + result.Insert(k, v) + } + }) + return result, true +} + +// Filter returns a new object from values in obj where the keys are +// found in filter. Array indices for values can be specified as +// number strings. +func (obj *object) Filter(filter Object) (Object, error) { + filtered, err := filterObject(obj, filter) + if err != nil { + return nil, err + } + return filtered.(Object), nil +} + +// Len returns the number of elements in the object. +func (obj object) Len() int { + return len(obj.keys) +} + +func (obj object) String() string { + var b strings.Builder + b.WriteRune('{') + + for i, elem := range obj.keys { + if i > 0 { + b.WriteString(", ") + } + b.WriteString(elem.key.String()) + b.WriteString(": ") + b.WriteString(elem.value.String()) + } + b.WriteRune('}') + return b.String() +} + +func (obj *object) get(k *Term) *objectElem { + hash := k.Hash() + + // This `equal` utility is duplicated and manually inlined a number of + // time in this file. Inlining it avoids heap allocations, so it makes + // a big performance difference: some operations like lookup become twice + // as slow without it. + var equal func(v Value) bool + + switch x := k.Value.(type) { + case Null, Boolean, String, Var: + equal = func(y Value) bool { return x == y } + case Number: + if xi, err := json.Number(x).Int64(); err == nil { + equal = func(y Value) bool { + if y, ok := y.(Number); ok { + if yi, err := json.Number(y).Int64(); err == nil { + return xi == yi + } + } + + return false + } + break + } + + // We use big.Rat for comparing big numbers. + // It replaces big.Float due to following reason: + // big.Float comes with a default precision of 64, and setting a + // larger precision results in more memory being allocated + // (regardless of the actual number we are parsing with SetString). + // + // Note: If we're so close to zero that big.Float says we are zero, do + // *not* big.Rat).SetString on the original string it'll potentially + // take very long. + var a *big.Rat + fa, ok := new(big.Float).SetString(string(x)) + if !ok { + panic("illegal value") + } + if fa.IsInt() { + if i, _ := fa.Int64(); i == 0 { + a = new(big.Rat).SetInt64(0) + } + } + if a == nil { + a, ok = new(big.Rat).SetString(string(x)) + if !ok { + panic("illegal value") + } + } + + equal = func(b Value) bool { + if bNum, ok := b.(Number); ok { + var b *big.Rat + fb, ok := new(big.Float).SetString(string(bNum)) + if !ok { + panic("illegal value") + } + if fb.IsInt() { + if i, _ := fb.Int64(); i == 0 { + b = new(big.Rat).SetInt64(0) + } + } + if b == nil { + b, ok = new(big.Rat).SetString(string(bNum)) + if !ok { + panic("illegal value") + } + } + + return a.Cmp(b) == 0 + } + + return false + } + default: + equal = func(y Value) bool { return Compare(x, y) == 0 } + } + + for curr := obj.elems[hash]; curr != nil; curr = curr.next { + if equal(curr.key.Value) { + return curr + } + } + return nil +} + +func (obj *object) insert(k, v *Term) { + hash := k.Hash() + head := obj.elems[hash] + // This `equal` utility is duplicated and manually inlined a number of + // time in this file. Inlining it avoids heap allocations, so it makes + // a big performance difference: some operations like lookup become twice + // as slow without it. + var equal func(v Value) bool + + switch x := k.Value.(type) { + case Null, Boolean, String, Var: + equal = func(y Value) bool { return x == y } + case Number: + if xi, err := json.Number(x).Int64(); err == nil { + equal = func(y Value) bool { + if y, ok := y.(Number); ok { + if yi, err := json.Number(y).Int64(); err == nil { + return xi == yi + } + } + + return false + } + break + } + + // We use big.Rat for comparing big numbers. + // It replaces big.Float due to following reason: + // big.Float comes with a default precision of 64, and setting a + // larger precision results in more memory being allocated + // (regardless of the actual number we are parsing with SetString). + // + // Note: If we're so close to zero that big.Float says we are zero, do + // *not* big.Rat).SetString on the original string it'll potentially + // take very long. + var a *big.Rat + fa, ok := new(big.Float).SetString(string(x)) + if !ok { + panic("illegal value") + } + if fa.IsInt() { + if i, _ := fa.Int64(); i == 0 { + a = new(big.Rat).SetInt64(0) + } + } + if a == nil { + a, ok = new(big.Rat).SetString(string(x)) + if !ok { + panic("illegal value") + } + } + + equal = func(b Value) bool { + if bNum, ok := b.(Number); ok { + var b *big.Rat + fb, ok := new(big.Float).SetString(string(bNum)) + if !ok { + panic("illegal value") + } + if fb.IsInt() { + if i, _ := fb.Int64(); i == 0 { + b = new(big.Rat).SetInt64(0) + } + } + if b == nil { + b, ok = new(big.Rat).SetString(string(bNum)) + if !ok { + panic("illegal value") + } + } + + return a.Cmp(b) == 0 + } + + return false + } + default: + equal = func(y Value) bool { return Compare(x, y) == 0 } + } + + for curr := head; curr != nil; curr = curr.next { + if equal(curr.key.Value) { + // The ground bit of the value may change in + // replace, hence adjust the counter per old + // and new value. + + if curr.value.IsGround() { + obj.ground-- + } + if v.IsGround() { + obj.ground++ + } + + curr.value = v + obj.hash = 0 + return + } + } + elem := &objectElem{ + key: k, + value: v, + next: head, + } + obj.elems[hash] = elem + i := sort.Search(len(obj.keys), func(i int) bool { return Compare(elem.key, obj.keys[i].key) < 0 }) + if i < len(obj.keys) { + // insert at position `i`: + obj.keys = append(obj.keys, nil) // add some space + copy(obj.keys[i+1:], obj.keys[i:]) // move things over + obj.keys[i] = elem // drop it in position + } else { + obj.keys = append(obj.keys, elem) + } + obj.hash = 0 + + if k.IsGround() { + obj.ground++ + } + if v.IsGround() { + obj.ground++ + } +} + +func filterObject(o Value, filter Value) (Value, error) { + if filter.Compare(Null{}) == 0 { + return o, nil + } + + filteredObj, ok := filter.(*object) + if !ok { + return nil, fmt.Errorf("invalid filter value %q, expected an object", filter) + } + + switch v := o.(type) { + case String, Number, Boolean, Null: + return o, nil + case *Array: + values := NewArray() + for i := 0; i < v.Len(); i++ { + subFilter := filteredObj.Get(StringTerm(strconv.Itoa(i))) + if subFilter != nil { + filteredValue, err := filterObject(v.Elem(i).Value, subFilter.Value) + if err != nil { + return nil, err + } + values = values.Append(NewTerm(filteredValue)) + } + } + return values, nil + case Set: + values := NewSet() + err := v.Iter(func(t *Term) error { + if filteredObj.Get(t) != nil { + filteredValue, err := filterObject(t.Value, filteredObj.Get(t).Value) + if err != nil { + return err + } + values.Add(NewTerm(filteredValue)) + } + return nil + }) + return values, err + case *object: + values := NewObject() + + iterObj := v + other := filteredObj + if v.Len() < filteredObj.Len() { + iterObj = filteredObj + other = v + } + + err := iterObj.Iter(func(key *Term, value *Term) error { + if other.Get(key) != nil { + filteredValue, err := filterObject(v.Get(key).Value, filteredObj.Get(key).Value) + if err != nil { + return err + } + values.Insert(key, NewTerm(filteredValue)) + } + return nil + }) + return values, err + default: + return nil, fmt.Errorf("invalid object value type %q", v) + } +} + +// ArrayComprehension represents an array comprehension as defined in the language. +type ArrayComprehension struct { + Term *Term `json:"term"` + Body Body `json:"body"` +} + +// ArrayComprehensionTerm creates a new Term with an ArrayComprehension value. +func ArrayComprehensionTerm(term *Term, body Body) *Term { + return &Term{ + Value: &ArrayComprehension{ + Term: term, + Body: body, + }, + } +} + +// Copy returns a deep copy of ac. +func (ac *ArrayComprehension) Copy() *ArrayComprehension { + cpy := *ac + cpy.Body = ac.Body.Copy() + cpy.Term = ac.Term.Copy() + return &cpy +} + +// Equal returns true if ac is equal to other. +func (ac *ArrayComprehension) Equal(other Value) bool { + return Compare(ac, other) == 0 +} + +// Compare compares ac to other, return <0, 0, or >0 if it is less than, equal to, +// or greater than other. +func (ac *ArrayComprehension) Compare(other Value) int { + return Compare(ac, other) +} + +// Find returns the current value or a not found error. +func (ac *ArrayComprehension) Find(path Ref) (Value, error) { + if len(path) == 0 { + return ac, nil + } + return nil, errFindNotFound +} + +// Hash returns the hash code of the Value. +func (ac *ArrayComprehension) Hash() int { + return ac.Term.Hash() + ac.Body.Hash() +} + +// IsGround returns true if the Term and Body are ground. +func (ac *ArrayComprehension) IsGround() bool { + return ac.Term.IsGround() && ac.Body.IsGround() +} + +func (ac *ArrayComprehension) String() string { + return "[" + ac.Term.String() + " | " + ac.Body.String() + "]" +} + +// ObjectComprehension represents an object comprehension as defined in the language. +type ObjectComprehension struct { + Key *Term `json:"key"` + Value *Term `json:"value"` + Body Body `json:"body"` +} + +// ObjectComprehensionTerm creates a new Term with an ObjectComprehension value. +func ObjectComprehensionTerm(key, value *Term, body Body) *Term { + return &Term{ + Value: &ObjectComprehension{ + Key: key, + Value: value, + Body: body, + }, + } +} + +// Copy returns a deep copy of oc. +func (oc *ObjectComprehension) Copy() *ObjectComprehension { + cpy := *oc + cpy.Body = oc.Body.Copy() + cpy.Key = oc.Key.Copy() + cpy.Value = oc.Value.Copy() + return &cpy +} + +// Equal returns true if oc is equal to other. +func (oc *ObjectComprehension) Equal(other Value) bool { + return Compare(oc, other) == 0 +} + +// Compare compares oc to other, return <0, 0, or >0 if it is less than, equal to, +// or greater than other. +func (oc *ObjectComprehension) Compare(other Value) int { + return Compare(oc, other) +} + +// Find returns the current value or a not found error. +func (oc *ObjectComprehension) Find(path Ref) (Value, error) { + if len(path) == 0 { + return oc, nil + } + return nil, errFindNotFound +} + +// Hash returns the hash code of the Value. +func (oc *ObjectComprehension) Hash() int { + return oc.Key.Hash() + oc.Value.Hash() + oc.Body.Hash() +} + +// IsGround returns true if the Key, Value and Body are ground. +func (oc *ObjectComprehension) IsGround() bool { + return oc.Key.IsGround() && oc.Value.IsGround() && oc.Body.IsGround() +} + +func (oc *ObjectComprehension) String() string { + return "{" + oc.Key.String() + ": " + oc.Value.String() + " | " + oc.Body.String() + "}" +} + +// SetComprehension represents a set comprehension as defined in the language. +type SetComprehension struct { + Term *Term `json:"term"` + Body Body `json:"body"` +} + +// SetComprehensionTerm creates a new Term with an SetComprehension value. +func SetComprehensionTerm(term *Term, body Body) *Term { + return &Term{ + Value: &SetComprehension{ + Term: term, + Body: body, + }, + } +} + +// Copy returns a deep copy of sc. +func (sc *SetComprehension) Copy() *SetComprehension { + cpy := *sc + cpy.Body = sc.Body.Copy() + cpy.Term = sc.Term.Copy() + return &cpy +} + +// Equal returns true if sc is equal to other. +func (sc *SetComprehension) Equal(other Value) bool { + return Compare(sc, other) == 0 +} + +// Compare compares sc to other, return <0, 0, or >0 if it is less than, equal to, +// or greater than other. +func (sc *SetComprehension) Compare(other Value) int { + return Compare(sc, other) +} + +// Find returns the current value or a not found error. +func (sc *SetComprehension) Find(path Ref) (Value, error) { + if len(path) == 0 { + return sc, nil + } + return nil, errFindNotFound +} + +// Hash returns the hash code of the Value. +func (sc *SetComprehension) Hash() int { + return sc.Term.Hash() + sc.Body.Hash() +} + +// IsGround returns true if the Term and Body are ground. +func (sc *SetComprehension) IsGround() bool { + return sc.Term.IsGround() && sc.Body.IsGround() +} + +func (sc *SetComprehension) String() string { + return "{" + sc.Term.String() + " | " + sc.Body.String() + "}" +} + +// Call represents as function call in the language. +type Call []*Term + +// CallTerm returns a new Term with a Call value defined by terms. The first +// term is the operator and the rest are operands. +func CallTerm(terms ...*Term) *Term { + return NewTerm(Call(terms)) +} + +// Copy returns a deep copy of c. +func (c Call) Copy() Call { + return termSliceCopy(c) +} + +// Compare compares c to other, return <0, 0, or >0 if it is less than, equal to, +// or greater than other. +func (c Call) Compare(other Value) int { + return Compare(c, other) +} + +// Find returns the current value or a not found error. +func (c Call) Find(Ref) (Value, error) { + return nil, errFindNotFound +} + +// Hash returns the hash code for the Value. +func (c Call) Hash() int { + return termSliceHash(c) +} + +// IsGround returns true if the Value is ground. +func (c Call) IsGround() bool { + return termSliceIsGround(c) +} + +// MakeExpr returns an ew Expr from this call. +func (c Call) MakeExpr(output *Term) *Expr { + terms := []*Term(c) + return NewExpr(append(terms, output)) +} + +func (c Call) String() string { + args := make([]string, len(c)-1) + for i := 1; i < len(c); i++ { + args[i-1] = c[i].String() + } + return fmt.Sprintf("%v(%v)", c[0], strings.Join(args, ", ")) +} + +func termSliceCopy(a []*Term) []*Term { + cpy := make([]*Term, len(a)) + for i := range a { + cpy[i] = a[i].Copy() + } + return cpy +} + +func termSliceEqual(a, b []*Term) bool { + if len(a) == len(b) { + for i := range a { + if !a[i].Equal(b[i]) { + return false + } + } + return true + } + return false +} + +func termSliceHash(a []*Term) int { + var hash int + for _, v := range a { + hash += v.Value.Hash() + } + return hash +} + +func termSliceIsGround(a []*Term) bool { + for _, v := range a { + if !v.IsGround() { + return false + } + } + return true +} + +// NOTE(tsandall): The unmarshalling errors in these functions are not +// helpful for callers because they do not identify the source of the +// unmarshalling error. Because OPA doesn't accept JSON describing ASTs +// from callers, this is acceptable (for now). If that changes in the future, +// the error messages should be revisited. The current approach focuses +// on the happy path and treats all errors the same. If better error +// reporting is needed, the error paths will need to be fleshed out. + +func unmarshalBody(b []interface{}) (Body, error) { + buf := Body{} + for _, e := range b { + if m, ok := e.(map[string]interface{}); ok { + expr := &Expr{} + if err := unmarshalExpr(expr, m); err == nil { + buf = append(buf, expr) + continue + } + } + goto unmarshal_error + } + return buf, nil +unmarshal_error: + return nil, fmt.Errorf("ast: unable to unmarshal body") +} + +func unmarshalExpr(expr *Expr, v map[string]interface{}) error { + if x, ok := v["negated"]; ok { + if b, ok := x.(bool); ok { + expr.Negated = b + } else { + return fmt.Errorf("ast: unable to unmarshal negated field with type: %T (expected true or false)", v["negated"]) + } + } + if err := unmarshalExprIndex(expr, v); err != nil { + return err + } + switch ts := v["terms"].(type) { + case map[string]interface{}: + t, err := unmarshalTerm(ts) + if err != nil { + return err + } + expr.Terms = t + case []interface{}: + terms, err := unmarshalTermSlice(ts) + if err != nil { + return err + } + expr.Terms = terms + default: + return fmt.Errorf(`ast: unable to unmarshal terms field with type: %T (expected {"value": ..., "type": ...} or [{"value": ..., "type": ...}, ...])`, v["terms"]) + } + if x, ok := v["with"]; ok { + if sl, ok := x.([]interface{}); ok { + ws := make([]*With, len(sl)) + for i := range sl { + var err error + ws[i], err = unmarshalWith(sl[i]) + if err != nil { + return err + } + } + expr.With = ws + } + } + return nil +} + +func unmarshalExprIndex(expr *Expr, v map[string]interface{}) error { + if x, ok := v["index"]; ok { + if n, ok := x.(json.Number); ok { + i, err := n.Int64() + if err == nil { + expr.Index = int(i) + return nil + } + } + } + return fmt.Errorf("ast: unable to unmarshal index field with type: %T (expected integer)", v["index"]) +} + +func unmarshalTerm(m map[string]interface{}) (*Term, error) { + v, err := unmarshalValue(m) + if err != nil { + return nil, err + } + return &Term{Value: v}, nil +} + +func unmarshalTermSlice(s []interface{}) ([]*Term, error) { + buf := []*Term{} + for _, x := range s { + if m, ok := x.(map[string]interface{}); ok { + if t, err := unmarshalTerm(m); err == nil { + buf = append(buf, t) + continue + } else { + return nil, err + } + } + return nil, fmt.Errorf("ast: unable to unmarshal term") + } + return buf, nil +} + +func unmarshalTermSliceValue(d map[string]interface{}) ([]*Term, error) { + if s, ok := d["value"].([]interface{}); ok { + return unmarshalTermSlice(s) + } + return nil, fmt.Errorf(`ast: unable to unmarshal term (expected {"value": [...], "type": ...} where type is one of: ref, array, or set)`) +} + +func unmarshalWith(i interface{}) (*With, error) { + if m, ok := i.(map[string]interface{}); ok { + tgt, _ := m["target"].(map[string]interface{}) + target, err := unmarshalTerm(tgt) + if err == nil { + val, _ := m["value"].(map[string]interface{}) + value, err := unmarshalTerm(val) + if err == nil { + return &With{ + Target: target, + Value: value, + }, nil + } + return nil, err + } + return nil, err + } + return nil, fmt.Errorf(`ast: unable to unmarshal with modifier (expected {"target": {...}, "value": {...}})`) +} + +func unmarshalValue(d map[string]interface{}) (Value, error) { + v := d["value"] + switch d["type"] { + case "null": + return Null{}, nil + case "boolean": + if b, ok := v.(bool); ok { + return Boolean(b), nil + } + case "number": + if n, ok := v.(json.Number); ok { + return Number(n), nil + } + case "string": + if s, ok := v.(string); ok { + return String(s), nil + } + case "var": + if s, ok := v.(string); ok { + return Var(s), nil + } + case "ref": + if s, err := unmarshalTermSliceValue(d); err == nil { + return Ref(s), nil + } + case "array": + if s, err := unmarshalTermSliceValue(d); err == nil { + return NewArray(s...), nil + } + case "set": + if s, err := unmarshalTermSliceValue(d); err == nil { + set := NewSet() + for _, x := range s { + set.Add(x) + } + return set, nil + } + case "object": + if s, ok := v.([]interface{}); ok { + buf := NewObject() + for _, x := range s { + if i, ok := x.([]interface{}); ok && len(i) == 2 { + p, err := unmarshalTermSlice(i) + if err == nil { + buf.Insert(p[0], p[1]) + continue + } + } + goto unmarshal_error + } + return buf, nil + } + case "arraycomprehension", "setcomprehension": + if m, ok := v.(map[string]interface{}); ok { + t, ok := m["term"].(map[string]interface{}) + if !ok { + goto unmarshal_error + } + + term, err := unmarshalTerm(t) + if err != nil { + goto unmarshal_error + } + + b, ok := m["body"].([]interface{}) + if !ok { + goto unmarshal_error + } + + body, err := unmarshalBody(b) + if err != nil { + goto unmarshal_error + } + + if d["type"] == "arraycomprehension" { + return &ArrayComprehension{Term: term, Body: body}, nil + } + return &SetComprehension{Term: term, Body: body}, nil + } + case "objectcomprehension": + if m, ok := v.(map[string]interface{}); ok { + k, ok := m["key"].(map[string]interface{}) + if !ok { + goto unmarshal_error + } + + key, err := unmarshalTerm(k) + if err != nil { + goto unmarshal_error + } + + v, ok := m["value"].(map[string]interface{}) + if !ok { + goto unmarshal_error + } + + value, err := unmarshalTerm(v) + if err != nil { + goto unmarshal_error + } + + b, ok := m["body"].([]interface{}) + if !ok { + goto unmarshal_error + } + + body, err := unmarshalBody(b) + if err != nil { + goto unmarshal_error + } + + return &ObjectComprehension{Key: key, Value: value, Body: body}, nil + } + case "call": + if s, err := unmarshalTermSliceValue(d); err == nil { + return Call(s), nil + } + } +unmarshal_error: + return nil, fmt.Errorf("ast: unable to unmarshal term") +} diff --git a/vendor/github.com/open-policy-agent/opa/ast/transform.go b/vendor/github.com/open-policy-agent/opa/ast/transform.go new file mode 100644 index 00000000..c7fa4c8f --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/ast/transform.go @@ -0,0 +1,396 @@ +// Copyright 2016 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package ast + +import ( + "fmt" +) + +// Transformer defines the interface for transforming AST elements. If the +// transformer returns nil and does not indicate an error, the AST element will +// be set to nil and no transformations will be applied to children of the +// element. +type Transformer interface { + Transform(v interface{}) (interface{}, error) +} + +// Transform iterates the AST and calls the Transform function on the +// Transformer t for x before recursing. +func Transform(t Transformer, x interface{}) (interface{}, error) { + + if term, ok := x.(*Term); ok { + return Transform(t, term.Value) + } + + y, err := t.Transform(x) + if err != nil { + return x, err + } + + if y == nil { + return nil, nil + } + + var ok bool + switch y := y.(type) { + case *Module: + p, err := Transform(t, y.Package) + if err != nil { + return nil, err + } + if y.Package, ok = p.(*Package); !ok { + return nil, fmt.Errorf("illegal transform: %T != %T", y.Package, p) + } + for i := range y.Imports { + imp, err := Transform(t, y.Imports[i]) + if err != nil { + return nil, err + } + if y.Imports[i], ok = imp.(*Import); !ok { + return nil, fmt.Errorf("illegal transform: %T != %T", y.Imports[i], imp) + } + } + for i := range y.Rules { + rule, err := Transform(t, y.Rules[i]) + if err != nil { + return nil, err + } + if y.Rules[i], ok = rule.(*Rule); !ok { + return nil, fmt.Errorf("illegal transform: %T != %T", y.Rules[i], rule) + } + } + for i := range y.Annotations { + a, err := Transform(t, y.Annotations[i]) + if err != nil { + return nil, err + } + if y.Annotations[i], ok = a.(*Annotations); !ok { + return nil, fmt.Errorf("illegal transform: %T != %T", y.Annotations[i], a) + } + } + for i := range y.Comments { + comment, err := Transform(t, y.Comments[i]) + if err != nil { + return nil, err + } + if y.Comments[i], ok = comment.(*Comment); !ok { + return nil, fmt.Errorf("illegal transform: %T != %T", y.Comments[i], comment) + } + } + return y, nil + case *Package: + ref, err := Transform(t, y.Path) + if err != nil { + return nil, err + } + if y.Path, ok = ref.(Ref); !ok { + return nil, fmt.Errorf("illegal transform: %T != %T", y.Path, ref) + } + return y, nil + case *Import: + y.Path, err = transformTerm(t, y.Path) + if err != nil { + return nil, err + } + if y.Alias, err = transformVar(t, y.Alias); err != nil { + return nil, err + } + return y, nil + case *Rule: + if y.Head, err = transformHead(t, y.Head); err != nil { + return nil, err + } + if y.Body, err = transformBody(t, y.Body); err != nil { + return nil, err + } + if y.Else != nil { + rule, err := Transform(t, y.Else) + if err != nil { + return nil, err + } + if y.Else, ok = rule.(*Rule); !ok { + return nil, fmt.Errorf("illegal transform: %T != %T", y.Else, rule) + } + } + return y, nil + case *Head: + if y.Name, err = transformVar(t, y.Name); err != nil { + return nil, err + } + if y.Args, err = transformArgs(t, y.Args); err != nil { + return nil, err + } + if y.Key != nil { + if y.Key, err = transformTerm(t, y.Key); err != nil { + return nil, err + } + } + if y.Value != nil { + if y.Value, err = transformTerm(t, y.Value); err != nil { + return nil, err + } + } + return y, nil + case Args: + for i := range y { + if y[i], err = transformTerm(t, y[i]); err != nil { + return nil, err + } + } + return y, nil + case Body: + for i, e := range y { + e, err := Transform(t, e) + if err != nil { + return nil, err + } + if y[i], ok = e.(*Expr); !ok { + return nil, fmt.Errorf("illegal transform: %T != %T", y[i], e) + } + } + return y, nil + case *Expr: + switch ts := y.Terms.(type) { + case *SomeDecl: + decl, err := Transform(t, ts) + if err != nil { + return nil, err + } + if y.Terms, ok = decl.(*SomeDecl); !ok { + return nil, fmt.Errorf("illegal transform: %T != %T", y, decl) + } + return y, nil + case []*Term: + for i := range ts { + if ts[i], err = transformTerm(t, ts[i]); err != nil { + return nil, err + } + } + case *Term: + if y.Terms, err = transformTerm(t, ts); err != nil { + return nil, err + } + } + for i, w := range y.With { + w, err := Transform(t, w) + if err != nil { + return nil, err + } + if y.With[i], ok = w.(*With); !ok { + return nil, fmt.Errorf("illegal transform: %T != %T", y.With[i], w) + } + } + return y, nil + case *With: + if y.Target, err = transformTerm(t, y.Target); err != nil { + return nil, err + } + if y.Value, err = transformTerm(t, y.Value); err != nil { + return nil, err + } + return y, nil + case Ref: + for i, term := range y { + if y[i], err = transformTerm(t, term); err != nil { + return nil, err + } + } + return y, nil + case *object: + return y.Map(func(k, v *Term) (*Term, *Term, error) { + k, err := transformTerm(t, k) + if err != nil { + return nil, nil, err + } + v, err = transformTerm(t, v) + if err != nil { + return nil, nil, err + } + return k, v, nil + }) + case *Array: + for i := 0; i < y.Len(); i++ { + v, err := transformTerm(t, y.Elem(i)) + if err != nil { + return nil, err + } + y.set(i, v) + } + return y, nil + case Set: + y, err = y.Map(func(term *Term) (*Term, error) { + return transformTerm(t, term) + }) + if err != nil { + return nil, err + } + return y, nil + case *ArrayComprehension: + if y.Term, err = transformTerm(t, y.Term); err != nil { + return nil, err + } + if y.Body, err = transformBody(t, y.Body); err != nil { + return nil, err + } + return y, nil + case *ObjectComprehension: + if y.Key, err = transformTerm(t, y.Key); err != nil { + return nil, err + } + if y.Value, err = transformTerm(t, y.Value); err != nil { + return nil, err + } + if y.Body, err = transformBody(t, y.Body); err != nil { + return nil, err + } + return y, nil + case *SetComprehension: + if y.Term, err = transformTerm(t, y.Term); err != nil { + return nil, err + } + if y.Body, err = transformBody(t, y.Body); err != nil { + return nil, err + } + return y, nil + case Call: + for i := range y { + if y[i], err = transformTerm(t, y[i]); err != nil { + return nil, err + } + } + return y, nil + default: + return y, nil + } +} + +// TransformRefs calls the function f on all references under x. +func TransformRefs(x interface{}, f func(Ref) (Value, error)) (interface{}, error) { + t := &GenericTransformer{func(x interface{}) (interface{}, error) { + if r, ok := x.(Ref); ok { + return f(r) + } + return x, nil + }} + return Transform(t, x) +} + +// TransformVars calls the function f on all vars under x. +func TransformVars(x interface{}, f func(Var) (Value, error)) (interface{}, error) { + t := &GenericTransformer{func(x interface{}) (interface{}, error) { + if v, ok := x.(Var); ok { + return f(v) + } + return x, nil + }} + return Transform(t, x) +} + +// TransformComprehensions calls the functio nf on all comprehensions under x. +func TransformComprehensions(x interface{}, f func(interface{}) (Value, error)) (interface{}, error) { + t := &GenericTransformer{func(x interface{}) (interface{}, error) { + switch x := x.(type) { + case *ArrayComprehension: + return f(x) + case *SetComprehension: + return f(x) + case *ObjectComprehension: + return f(x) + } + return x, nil + }} + return Transform(t, x) +} + +// GenericTransformer implements the Transformer interface to provide a utility +// to transform AST nodes using a closure. +type GenericTransformer struct { + f func(x interface{}) (interface{}, error) +} + +// NewGenericTransformer returns a new GenericTransformer that will transform +// AST nodes using the function f. +func NewGenericTransformer(f func(x interface{}) (interface{}, error)) *GenericTransformer { + return &GenericTransformer{ + f: f, + } +} + +// Transform calls the function f on the GenericTransformer. +func (t *GenericTransformer) Transform(x interface{}) (interface{}, error) { + return t.f(x) +} + +func transformHead(t Transformer, head *Head) (*Head, error) { + y, err := Transform(t, head) + if err != nil { + return nil, err + } + h, ok := y.(*Head) + if !ok { + return nil, fmt.Errorf("illegal transform: %T != %T", head, y) + } + return h, nil +} + +func transformArgs(t Transformer, args Args) (Args, error) { + y, err := Transform(t, args) + if err != nil { + return nil, err + } + a, ok := y.(Args) + if !ok { + return nil, fmt.Errorf("illegal transform: %T != %T", args, y) + } + return a, nil +} + +func transformBody(t Transformer, body Body) (Body, error) { + y, err := Transform(t, body) + if err != nil { + return nil, err + } + r, ok := y.(Body) + if !ok { + return nil, fmt.Errorf("illegal transform: %T != %T", body, y) + } + return r, nil +} + +func transformTerm(t Transformer, term *Term) (*Term, error) { + v, err := transformValue(t, term.Value) + if err != nil { + return nil, err + } + r := &Term{ + Value: v, + Location: term.Location, + } + return r, nil +} + +func transformValue(t Transformer, v Value) (Value, error) { + v1, err := Transform(t, v) + if err != nil { + return nil, err + } + r, ok := v1.(Value) + if !ok { + return nil, fmt.Errorf("illegal transform: %T != %T", v, v1) + } + return r, nil +} + +func transformVar(t Transformer, v Var) (Var, error) { + v1, err := Transform(t, v) + if err != nil { + return "", err + } + r, ok := v1.(Var) + if !ok { + return "", fmt.Errorf("illegal transform: %T != %T", v, v1) + } + return r, nil +} diff --git a/vendor/github.com/open-policy-agent/opa/ast/unify.go b/vendor/github.com/open-policy-agent/opa/ast/unify.go new file mode 100644 index 00000000..60244974 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/ast/unify.go @@ -0,0 +1,235 @@ +// Copyright 2016 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package ast + +func isRefSafe(ref Ref, safe VarSet) bool { + switch head := ref[0].Value.(type) { + case Var: + return safe.Contains(head) + case Call: + return isCallSafe(head, safe) + default: + for v := range ref[0].Vars() { + if !safe.Contains(v) { + return false + } + } + return true + } +} + +func isCallSafe(call Call, safe VarSet) bool { + vis := NewVarVisitor().WithParams(SafetyCheckVisitorParams) + vis.Walk(call) + unsafe := vis.Vars().Diff(safe) + return len(unsafe) == 0 +} + +// Unify returns a set of variables that will be unified when the equality expression defined by +// terms a and b is evaluated. The unifier assumes that variables in the VarSet safe are already +// unified. +func Unify(safe VarSet, a *Term, b *Term) VarSet { + u := &unifier{ + safe: safe, + unified: VarSet{}, + unknown: map[Var]VarSet{}, + } + u.unify(a, b) + return u.unified +} + +type unifier struct { + safe VarSet + unified VarSet + unknown map[Var]VarSet +} + +func (u *unifier) isSafe(x Var) bool { + return u.safe.Contains(x) || u.unified.Contains(x) +} + +func (u *unifier) unify(a *Term, b *Term) { + + switch a := a.Value.(type) { + + case Var: + switch b := b.Value.(type) { + case Var: + if u.isSafe(b) { + u.markSafe(a) + } else if u.isSafe(a) { + u.markSafe(b) + } else { + u.markUnknown(a, b) + u.markUnknown(b, a) + } + case *Array, Object: + u.unifyAll(a, b) + case Ref: + if isRefSafe(b, u.safe) { + u.markSafe(a) + } + case Call: + if isCallSafe(b, u.safe) { + u.markSafe(a) + } + default: + u.markSafe(a) + } + + case Ref: + if isRefSafe(a, u.safe) { + switch b := b.Value.(type) { + case Var: + u.markSafe(b) + case *Array, Object: + u.markAllSafe(b) + } + } + + case Call: + if isCallSafe(a, u.safe) { + switch b := b.Value.(type) { + case Var: + u.markSafe(b) + case *Array, Object: + u.markAllSafe(b) + } + } + + case *ArrayComprehension: + switch b := b.Value.(type) { + case Var: + u.markSafe(b) + case *Array: + u.markAllSafe(b) + } + case *ObjectComprehension: + switch b := b.Value.(type) { + case Var: + u.markSafe(b) + case *object: + u.markAllSafe(b) + } + case *SetComprehension: + switch b := b.Value.(type) { + case Var: + u.markSafe(b) + } + + case *Array: + switch b := b.Value.(type) { + case Var: + u.unifyAll(b, a) + case *ArrayComprehension, *ObjectComprehension, *SetComprehension: + u.markAllSafe(a) + case Ref: + if isRefSafe(b, u.safe) { + u.markAllSafe(a) + } + case Call: + if isCallSafe(b, u.safe) { + u.markAllSafe(a) + } + case *Array: + if a.Len() == b.Len() { + for i := 0; i < a.Len(); i++ { + u.unify(a.Elem(i), b.Elem(i)) + } + } + } + + case *object: + switch b := b.Value.(type) { + case Var: + u.unifyAll(b, a) + case Ref: + if isRefSafe(b, u.safe) { + u.markAllSafe(a) + } + case Call: + if isCallSafe(b, u.safe) { + u.markAllSafe(a) + } + case *object: + if a.Len() == b.Len() { + _ = a.Iter(func(k, v *Term) error { + if v2 := b.Get(k); v2 != nil { + u.unify(v, v2) + } + return nil + }) // impossible to return error + } + } + + default: + switch b := b.Value.(type) { + case Var: + u.markSafe(b) + } + } +} + +func (u *unifier) markAllSafe(x Value) { + vis := u.varVisitor() + vis.Walk(x) + for v := range vis.Vars() { + u.markSafe(v) + } +} + +func (u *unifier) markSafe(x Var) { + u.unified.Add(x) + + // Add dependencies of 'x' to safe set + vs := u.unknown[x] + delete(u.unknown, x) + for v := range vs { + u.markSafe(v) + } + + // Add dependants of 'x' to safe set if they have no more + // dependencies. + for v, deps := range u.unknown { + if deps.Contains(x) { + delete(deps, x) + if len(deps) == 0 { + u.markSafe(v) + } + } + } +} + +func (u *unifier) markUnknown(a, b Var) { + if _, ok := u.unknown[a]; !ok { + u.unknown[a] = NewVarSet() + } + u.unknown[a].Add(b) +} + +func (u *unifier) unifyAll(a Var, b Value) { + if u.isSafe(a) { + u.markAllSafe(b) + } else { + vis := u.varVisitor() + vis.Walk(b) + unsafe := vis.Vars().Diff(u.safe).Diff(u.unified) + if len(unsafe) == 0 { + u.markSafe(a) + } else { + for v := range unsafe { + u.markUnknown(a, v) + } + } + } +} + +func (u *unifier) varVisitor() *VarVisitor { + return NewVarVisitor().WithParams(VarVisitorParams{ + SkipRefHead: true, + SkipObjectKeys: true, + SkipClosures: true, + }) +} diff --git a/vendor/github.com/open-policy-agent/opa/ast/varset.go b/vendor/github.com/open-policy-agent/opa/ast/varset.go new file mode 100644 index 00000000..16dc3f58 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/ast/varset.go @@ -0,0 +1,100 @@ +// Copyright 2016 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package ast + +import ( + "fmt" + "sort" +) + +// VarSet represents a set of variables. +type VarSet map[Var]struct{} + +// NewVarSet returns a new VarSet containing the specified variables. +func NewVarSet(vs ...Var) VarSet { + s := VarSet{} + for _, v := range vs { + s.Add(v) + } + return s +} + +// Add updates the set to include the variable "v". +func (s VarSet) Add(v Var) { + s[v] = struct{}{} +} + +// Contains returns true if the set contains the variable "v". +func (s VarSet) Contains(v Var) bool { + _, ok := s[v] + return ok +} + +// Copy returns a shallow copy of the VarSet. +func (s VarSet) Copy() VarSet { + cpy := VarSet{} + for v := range s { + cpy.Add(v) + } + return cpy +} + +// Diff returns a VarSet containing variables in s that are not in vs. +func (s VarSet) Diff(vs VarSet) VarSet { + r := VarSet{} + for v := range s { + if !vs.Contains(v) { + r.Add(v) + } + } + return r +} + +// Equal returns true if s contains exactly the same elements as vs. +func (s VarSet) Equal(vs VarSet) bool { + if len(s.Diff(vs)) > 0 { + return false + } + return len(vs.Diff(s)) == 0 +} + +// Intersect returns a VarSet containing variables in s that are in vs. +func (s VarSet) Intersect(vs VarSet) VarSet { + r := VarSet{} + for v := range s { + if vs.Contains(v) { + r.Add(v) + } + } + return r +} + +// Sorted returns a sorted slice of vars from s. +func (s VarSet) Sorted() []Var { + sorted := make([]Var, 0, len(s)) + for v := range s { + sorted = append(sorted, v) + } + sort.Slice(sorted, func(i, j int) bool { + return sorted[i].Compare(sorted[j]) < 0 + }) + return sorted +} + +// Update merges the other VarSet into this VarSet. +func (s VarSet) Update(vs VarSet) { + for v := range vs { + s.Add(v) + } +} + +func (s VarSet) String() string { + tmp := []string{} + for v := range s { + tmp = append(tmp, string(v)) + } + sort.Strings(tmp) + return fmt.Sprintf("%v", tmp) +} diff --git a/vendor/github.com/open-policy-agent/opa/ast/visit.go b/vendor/github.com/open-policy-agent/opa/ast/visit.go new file mode 100644 index 00000000..139c4de3 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/ast/visit.go @@ -0,0 +1,695 @@ +// Copyright 2016 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package ast + +// Visitor defines the interface for iterating AST elements. The Visit function +// can return a Visitor w which will be used to visit the children of the AST +// element v. If the Visit function returns nil, the children will not be +// visited. This is deprecated. +type Visitor interface { + Visit(v interface{}) (w Visitor) +} + +// BeforeAndAfterVisitor wraps Visitor to provide hooks for being called before +// and after the AST has been visited. This is deprecated. +type BeforeAndAfterVisitor interface { + Visitor + Before(x interface{}) + After(x interface{}) +} + +// Walk iterates the AST by calling the Visit function on the Visitor +// v for x before recursing. This is deprecated. +func Walk(v Visitor, x interface{}) { + if bav, ok := v.(BeforeAndAfterVisitor); !ok { + walk(v, x) + } else { + bav.Before(x) + defer bav.After(x) + walk(bav, x) + } +} + +// WalkBeforeAndAfter iterates the AST by calling the Visit function on the +// Visitor v for x before recursing. This is deprecated. +func WalkBeforeAndAfter(v BeforeAndAfterVisitor, x interface{}) { + Walk(v, x) +} + +func walk(v Visitor, x interface{}) { + w := v.Visit(x) + if w == nil { + return + } + switch x := x.(type) { + case *Module: + Walk(w, x.Package) + for _, i := range x.Imports { + Walk(w, i) + } + for _, r := range x.Rules { + Walk(w, r) + } + for _, a := range x.Annotations { + Walk(w, a) + } + for _, c := range x.Comments { + Walk(w, c) + } + case *Package: + Walk(w, x.Path) + case *Import: + Walk(w, x.Path) + Walk(w, x.Alias) + case *Rule: + Walk(w, x.Head) + Walk(w, x.Body) + if x.Else != nil { + Walk(w, x.Else) + } + case *Head: + Walk(w, x.Name) + Walk(w, x.Args) + if x.Key != nil { + Walk(w, x.Key) + } + if x.Value != nil { + Walk(w, x.Value) + } + case Body: + for _, e := range x { + Walk(w, e) + } + case Args: + for _, t := range x { + Walk(w, t) + } + case *Expr: + switch ts := x.Terms.(type) { + case *SomeDecl: + Walk(w, ts) + case []*Term: + for _, t := range ts { + Walk(w, t) + } + case *Term: + Walk(w, ts) + } + for i := range x.With { + Walk(w, x.With[i]) + } + case *With: + Walk(w, x.Target) + Walk(w, x.Value) + case *Term: + Walk(w, x.Value) + case Ref: + for _, t := range x { + Walk(w, t) + } + case *object: + x.Foreach(func(k, vv *Term) { + Walk(w, k) + Walk(w, vv) + }) + case *Array: + x.Foreach(func(t *Term) { + Walk(w, t) + }) + case Set: + x.Foreach(func(t *Term) { + Walk(w, t) + }) + case *ArrayComprehension: + Walk(w, x.Term) + Walk(w, x.Body) + case *ObjectComprehension: + Walk(w, x.Key) + Walk(w, x.Value) + Walk(w, x.Body) + case *SetComprehension: + Walk(w, x.Term) + Walk(w, x.Body) + case Call: + for _, t := range x { + Walk(w, t) + } + } +} + +// WalkVars calls the function f on all vars under x. If the function f +// returns true, AST nodes under the last node will not be visited. +func WalkVars(x interface{}, f func(Var) bool) { + vis := &GenericVisitor{func(x interface{}) bool { + if v, ok := x.(Var); ok { + return f(v) + } + return false + }} + vis.Walk(x) +} + +// WalkClosures calls the function f on all closures under x. If the function f +// returns true, AST nodes under the last node will not be visited. +func WalkClosures(x interface{}, f func(interface{}) bool) { + vis := &GenericVisitor{func(x interface{}) bool { + switch x.(type) { + case *ArrayComprehension, *ObjectComprehension, *SetComprehension: + return f(x) + } + return false + }} + vis.Walk(x) +} + +// WalkRefs calls the function f on all references under x. If the function f +// returns true, AST nodes under the last node will not be visited. +func WalkRefs(x interface{}, f func(Ref) bool) { + vis := &GenericVisitor{func(x interface{}) bool { + if r, ok := x.(Ref); ok { + return f(r) + } + return false + }} + vis.Walk(x) +} + +// WalkTerms calls the function f on all terms under x. If the function f +// returns true, AST nodes under the last node will not be visited. +func WalkTerms(x interface{}, f func(*Term) bool) { + vis := &GenericVisitor{func(x interface{}) bool { + if term, ok := x.(*Term); ok { + return f(term) + } + return false + }} + vis.Walk(x) +} + +// WalkWiths calls the function f on all with modifiers under x. If the function f +// returns true, AST nodes under the last node will not be visited. +func WalkWiths(x interface{}, f func(*With) bool) { + vis := &GenericVisitor{func(x interface{}) bool { + if w, ok := x.(*With); ok { + return f(w) + } + return false + }} + vis.Walk(x) +} + +// WalkExprs calls the function f on all expressions under x. If the function f +// returns true, AST nodes under the last node will not be visited. +func WalkExprs(x interface{}, f func(*Expr) bool) { + vis := &GenericVisitor{func(x interface{}) bool { + if r, ok := x.(*Expr); ok { + return f(r) + } + return false + }} + vis.Walk(x) +} + +// WalkBodies calls the function f on all bodies under x. If the function f +// returns true, AST nodes under the last node will not be visited. +func WalkBodies(x interface{}, f func(Body) bool) { + vis := &GenericVisitor{func(x interface{}) bool { + if b, ok := x.(Body); ok { + return f(b) + } + return false + }} + vis.Walk(x) +} + +// WalkRules calls the function f on all rules under x. If the function f +// returns true, AST nodes under the last node will not be visited. +func WalkRules(x interface{}, f func(*Rule) bool) { + vis := &GenericVisitor{func(x interface{}) bool { + if r, ok := x.(*Rule); ok { + stop := f(r) + // NOTE(tsandall): since rules cannot be embedded inside of queries + // we can stop early if there is no else block. + if stop || r.Else == nil { + return true + } + } + return false + }} + vis.Walk(x) +} + +// WalkNodes calls the function f on all nodes under x. If the function f +// returns true, AST nodes under the last node will not be visited. +func WalkNodes(x interface{}, f func(Node) bool) { + vis := &GenericVisitor{func(x interface{}) bool { + if n, ok := x.(Node); ok { + return f(n) + } + return false + }} + vis.Walk(x) +} + +// GenericVisitor provides a utility to walk over AST nodes using a +// closure. If the closure returns true, the visitor will not walk +// over AST nodes under x. +type GenericVisitor struct { + f func(x interface{}) bool +} + +// NewGenericVisitor returns a new GenericVisitor that will invoke the function +// f on AST nodes. +func NewGenericVisitor(f func(x interface{}) bool) *GenericVisitor { + return &GenericVisitor{f} +} + +// Walk iterates the AST by calling the function f on the +// GenericVisitor before recursing. Contrary to the generic Walk, this +// does not require allocating the visitor from heap. +func (vis *GenericVisitor) Walk(x interface{}) { + if vis.f(x) { + return + } + + switch x := x.(type) { + case *Module: + vis.Walk(x.Package) + for _, i := range x.Imports { + vis.Walk(i) + } + for _, r := range x.Rules { + vis.Walk(r) + } + for _, a := range x.Annotations { + vis.Walk(a) + } + for _, c := range x.Comments { + vis.Walk(c) + } + case *Package: + vis.Walk(x.Path) + case *Import: + vis.Walk(x.Path) + vis.Walk(x.Alias) + case *Rule: + vis.Walk(x.Head) + vis.Walk(x.Body) + if x.Else != nil { + vis.Walk(x.Else) + } + case *Head: + vis.Walk(x.Name) + vis.Walk(x.Args) + if x.Key != nil { + vis.Walk(x.Key) + } + if x.Value != nil { + vis.Walk(x.Value) + } + case Body: + for _, e := range x { + vis.Walk(e) + } + case Args: + for _, t := range x { + vis.Walk(t) + } + case *Expr: + switch ts := x.Terms.(type) { + case *SomeDecl: + vis.Walk(ts) + case []*Term: + for _, t := range ts { + vis.Walk(t) + } + case *Term: + vis.Walk(ts) + } + for i := range x.With { + vis.Walk(x.With[i]) + } + case *With: + vis.Walk(x.Target) + vis.Walk(x.Value) + case *Term: + vis.Walk(x.Value) + case Ref: + for _, t := range x { + vis.Walk(t) + } + case *object: + x.Foreach(func(k, v *Term) { + vis.Walk(k) + vis.Walk(x.Get(k)) + }) + case *Array: + x.Foreach(func(t *Term) { + vis.Walk(t) + }) + case Set: + for _, t := range x.Slice() { + vis.Walk(t) + } + case *ArrayComprehension: + vis.Walk(x.Term) + vis.Walk(x.Body) + case *ObjectComprehension: + vis.Walk(x.Key) + vis.Walk(x.Value) + vis.Walk(x.Body) + case *SetComprehension: + vis.Walk(x.Term) + vis.Walk(x.Body) + case Call: + for _, t := range x { + vis.Walk(t) + } + } +} + +// BeforeAfterVisitor provides a utility to walk over AST nodes using +// closures. If the before closure returns true, the visitor will not +// walk over AST nodes under x. The after closure is invoked always +// after visiting a node. +type BeforeAfterVisitor struct { + before func(x interface{}) bool + after func(x interface{}) +} + +// NewBeforeAfterVisitor returns a new BeforeAndAfterVisitor that +// will invoke the functions before and after AST nodes. +func NewBeforeAfterVisitor(before func(x interface{}) bool, after func(x interface{})) *BeforeAfterVisitor { + return &BeforeAfterVisitor{before, after} +} + +// Walk iterates the AST by calling the functions on the +// BeforeAndAfterVisitor before and after recursing. Contrary to the +// generic Walk, this does not require allocating the visitor from +// heap. +func (vis *BeforeAfterVisitor) Walk(x interface{}) { + defer vis.after(x) + if vis.before(x) { + return + } + + switch x := x.(type) { + case *Module: + vis.Walk(x.Package) + for _, i := range x.Imports { + vis.Walk(i) + } + for _, r := range x.Rules { + vis.Walk(r) + } + for _, a := range x.Annotations { + vis.Walk(a) + } + for _, c := range x.Comments { + vis.Walk(c) + } + case *Package: + vis.Walk(x.Path) + case *Import: + vis.Walk(x.Path) + vis.Walk(x.Alias) + case *Rule: + vis.Walk(x.Head) + vis.Walk(x.Body) + if x.Else != nil { + vis.Walk(x.Else) + } + case *Head: + vis.Walk(x.Name) + vis.Walk(x.Args) + if x.Key != nil { + vis.Walk(x.Key) + } + if x.Value != nil { + vis.Walk(x.Value) + } + case Body: + for _, e := range x { + vis.Walk(e) + } + case Args: + for _, t := range x { + vis.Walk(t) + } + case *Expr: + switch ts := x.Terms.(type) { + case *SomeDecl: + vis.Walk(ts) + case []*Term: + for _, t := range ts { + vis.Walk(t) + } + case *Term: + vis.Walk(ts) + } + for i := range x.With { + vis.Walk(x.With[i]) + } + case *With: + vis.Walk(x.Target) + vis.Walk(x.Value) + case *Term: + vis.Walk(x.Value) + case Ref: + for _, t := range x { + vis.Walk(t) + } + case *object: + x.Foreach(func(k, v *Term) { + vis.Walk(k) + vis.Walk(x.Get(k)) + }) + case *Array: + x.Foreach(func(t *Term) { + vis.Walk(t) + }) + case Set: + for _, t := range x.Slice() { + vis.Walk(t) + } + case *ArrayComprehension: + vis.Walk(x.Term) + vis.Walk(x.Body) + case *ObjectComprehension: + vis.Walk(x.Key) + vis.Walk(x.Value) + vis.Walk(x.Body) + case *SetComprehension: + vis.Walk(x.Term) + vis.Walk(x.Body) + case Call: + for _, t := range x { + vis.Walk(t) + } + } +} + +// VarVisitor walks AST nodes under a given node and collects all encountered +// variables. The collected variables can be controlled by specifying +// VarVisitorParams when creating the visitor. +type VarVisitor struct { + params VarVisitorParams + vars VarSet +} + +// VarVisitorParams contains settings for a VarVisitor. +type VarVisitorParams struct { + SkipRefHead bool + SkipRefCallHead bool + SkipObjectKeys bool + SkipClosures bool + SkipWithTarget bool + SkipSets bool +} + +// NewVarVisitor returns a new VarVisitor object. +func NewVarVisitor() *VarVisitor { + return &VarVisitor{ + vars: NewVarSet(), + } +} + +// WithParams sets the parameters in params on vis. +func (vis *VarVisitor) WithParams(params VarVisitorParams) *VarVisitor { + vis.params = params + return vis +} + +// Vars returns a VarSet that contains collected vars. +func (vis *VarVisitor) Vars() VarSet { + return vis.vars +} + +func (vis *VarVisitor) visit(v interface{}) bool { + if vis.params.SkipObjectKeys { + if o, ok := v.(Object); ok { + o.Foreach(func(k, v *Term) { + vis.Walk(v) + }) + return true + } + } + if vis.params.SkipRefHead { + if r, ok := v.(Ref); ok { + for _, t := range r[1:] { + vis.Walk(t) + } + return true + } + } + if vis.params.SkipClosures { + switch v.(type) { + case *ArrayComprehension, *ObjectComprehension, *SetComprehension: + return true + } + } + if vis.params.SkipWithTarget { + if v, ok := v.(*With); ok { + vis.Walk(v.Value) + return true + } + } + if vis.params.SkipSets { + if _, ok := v.(Set); ok { + return true + } + } + if vis.params.SkipRefCallHead { + switch v := v.(type) { + case *Expr: + if terms, ok := v.Terms.([]*Term); ok { + for _, t := range terms[0].Value.(Ref)[1:] { + vis.Walk(t) + } + for i := 1; i < len(terms); i++ { + vis.Walk(terms[i]) + } + for _, w := range v.With { + vis.Walk(w) + } + return true + } + case Call: + operator := v[0].Value.(Ref) + for i := 1; i < len(operator); i++ { + vis.Walk(operator[i]) + } + for i := 1; i < len(v); i++ { + vis.Walk(v[i]) + } + return true + } + } + if v, ok := v.(Var); ok { + vis.vars.Add(v) + } + return false +} + +// Walk iterates the AST by calling the function f on the +// GenericVisitor before recursing. Contrary to the generic Walk, this +// does not require allocating the visitor from heap. +func (vis *VarVisitor) Walk(x interface{}) { + if vis.visit(x) { + return + } + + switch x := x.(type) { + case *Module: + vis.Walk(x.Package) + for _, i := range x.Imports { + vis.Walk(i) + } + for _, r := range x.Rules { + vis.Walk(r) + } + for _, c := range x.Comments { + vis.Walk(c) + } + case *Package: + vis.Walk(x.Path) + case *Import: + vis.Walk(x.Path) + vis.Walk(x.Alias) + case *Rule: + vis.Walk(x.Head) + vis.Walk(x.Body) + if x.Else != nil { + vis.Walk(x.Else) + } + case *Head: + vis.Walk(x.Name) + vis.Walk(x.Args) + if x.Key != nil { + vis.Walk(x.Key) + } + if x.Value != nil { + vis.Walk(x.Value) + } + case Body: + for _, e := range x { + vis.Walk(e) + } + case Args: + for _, t := range x { + vis.Walk(t) + } + case *Expr: + switch ts := x.Terms.(type) { + case *SomeDecl: + vis.Walk(ts) + case []*Term: + for _, t := range ts { + vis.Walk(t) + } + case *Term: + vis.Walk(ts) + } + for i := range x.With { + vis.Walk(x.With[i]) + } + case *With: + vis.Walk(x.Target) + vis.Walk(x.Value) + case *Term: + vis.Walk(x.Value) + case Ref: + for _, t := range x { + vis.Walk(t) + } + case *object: + x.Foreach(func(k, v *Term) { + vis.Walk(k) + vis.Walk(x.Get(k)) + }) + case *Array: + x.Foreach(func(t *Term) { + vis.Walk(t) + }) + case Set: + for _, t := range x.Slice() { + vis.Walk(t) + } + case *ArrayComprehension: + vis.Walk(x.Term) + vis.Walk(x.Body) + case *ObjectComprehension: + vis.Walk(x.Key) + vis.Walk(x.Value) + vis.Walk(x.Body) + case *SetComprehension: + vis.Walk(x.Term) + vis.Walk(x.Body) + case Call: + for _, t := range x { + vis.Walk(t) + } + } +} diff --git a/vendor/github.com/open-policy-agent/opa/bundle/bundle.go b/vendor/github.com/open-policy-agent/opa/bundle/bundle.go new file mode 100644 index 00000000..13ec7fe5 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/bundle/bundle.go @@ -0,0 +1,1163 @@ +// Copyright 2018 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +// Package bundle implements bundle loading. +package bundle + +import ( + "archive/tar" + "bytes" + "compress/gzip" + "encoding/hex" + "encoding/json" + "fmt" + "io" + "net/url" + "path/filepath" + "reflect" + "strings" + + "github.com/pkg/errors" + + "github.com/open-policy-agent/opa/ast" + "github.com/open-policy-agent/opa/format" + "github.com/open-policy-agent/opa/internal/file/archive" + "github.com/open-policy-agent/opa/internal/merge" + "github.com/open-policy-agent/opa/metrics" + "github.com/open-policy-agent/opa/util" +) + +// Common file extensions and file names. +const ( + RegoExt = ".rego" + WasmFile = "policy.wasm" + ManifestExt = ".manifest" + SignaturesFile = "signatures.json" + dataFile = "data.json" + yamlDataFile = "data.yaml" + defaultHashingAlg = "SHA-256" + DefaultSizeLimitBytes = (1024 * 1024 * 1024) // limit bundle reads to 1GB to protect against gzip bombs +) + +// Bundle represents a loaded bundle. The bundle can contain data and policies. +type Bundle struct { + Signatures SignaturesConfig + Manifest Manifest + Data map[string]interface{} + Modules []ModuleFile + Wasm []byte // Deprecated. Use WasmModules instead + WasmModules []WasmModuleFile +} + +// SignaturesConfig represents an array of JWTs that encapsulate the signatures for the bundle. +type SignaturesConfig struct { + Signatures []string `json:"signatures,omitempty"` + Plugin string `json:"plugin,omitempty"` +} + +// isEmpty returns if the SignaturesConfig is empty. +func (s SignaturesConfig) isEmpty() bool { + return reflect.DeepEqual(s, SignaturesConfig{}) +} + +// DecodedSignature represents the decoded JWT payload. +type DecodedSignature struct { + Files []FileInfo `json:"files"` + KeyID string `json:"keyid"` // Deprecated, use kid in the JWT header instead. + Scope string `json:"scope"` + IssuedAt int64 `json:"iat"` + Issuer string `json:"iss"` +} + +// FileInfo contains the hashing algorithm used, resulting digest etc. +type FileInfo struct { + Name string `json:"name"` + Hash string `json:"hash"` + Algorithm string `json:"algorithm"` +} + +// NewFile returns a new FileInfo. +func NewFile(name, hash, alg string) FileInfo { + return FileInfo{ + Name: name, + Hash: hash, + Algorithm: alg, + } +} + +// Manifest represents the manifest from a bundle. The manifest may contain +// metadata such as the bundle revision. +type Manifest struct { + Revision string `json:"revision"` + Roots *[]string `json:"roots,omitempty"` + WasmResolvers []WasmResolver `json:"wasm,omitempty"` + Metadata map[string]interface{} `json:"metadata,omitempty"` +} + +// WasmResolver maps a wasm module to an entrypoint ref. +type WasmResolver struct { + Entrypoint string `json:"entrypoint,omitempty"` + Module string `json:"module,omitempty"` +} + +// Init initializes the manifest. If you instantiate a manifest +// manually, call Init to ensure that the roots are set properly. +func (m *Manifest) Init() { + if m.Roots == nil { + defaultRoots := []string{""} + m.Roots = &defaultRoots + } +} + +// AddRoot adds r to the roots of m. This function is idempotent. +func (m *Manifest) AddRoot(r string) { + m.Init() + if !RootPathsContain(*m.Roots, r) { + *m.Roots = append(*m.Roots, r) + } +} + +// Equal returns true if m is semantically equivalent to other. +func (m Manifest) Equal(other Manifest) bool { + + // This is safe since both are passed by value. + m.Init() + other.Init() + + if m.Revision != other.Revision { + return false + } + + if len(m.WasmResolvers) != len(other.WasmResolvers) { + return false + } + + for i := 0; i < len(m.WasmResolvers); i++ { + if m.WasmResolvers[i] != other.WasmResolvers[i] { + return false + } + } + + if !reflect.DeepEqual(m.Metadata, other.Metadata) { + return false + } + + return m.rootSet().Equal(other.rootSet()) +} + +// Copy returns a deep copy of the manifest. +func (m Manifest) Copy() Manifest { + m.Init() + roots := make([]string, len(*m.Roots)) + copy(roots, *m.Roots) + m.Roots = &roots + + wasmModules := make([]WasmResolver, len(m.WasmResolvers)) + copy(wasmModules, m.WasmResolvers) + m.WasmResolvers = wasmModules + + metadata := m.Metadata + + if metadata != nil { + m.Metadata = make(map[string]interface{}) + for k, v := range metadata { + m.Metadata[k] = v + } + } + + return m +} + +func (m Manifest) String() string { + m.Init() + return fmt.Sprintf("", m.Revision, *m.Roots, m.WasmResolvers) +} + +func (m Manifest) rootSet() stringSet { + rs := map[string]struct{}{} + + for _, r := range *m.Roots { + rs[r] = struct{}{} + } + + return stringSet(rs) +} + +type stringSet map[string]struct{} + +func (ss stringSet) Equal(other stringSet) bool { + if len(ss) != len(other) { + return false + } + for k := range other { + if _, ok := ss[k]; !ok { + return false + } + } + return true +} + +func (m *Manifest) validateAndInjectDefaults(b Bundle) error { + + m.Init() + + // Validate roots in bundle. + roots := *m.Roots + + // Standardize the roots (no starting or trailing slash) + for i := range roots { + roots[i] = strings.Trim(roots[i], "/") + } + + for i := 0; i < len(roots)-1; i++ { + for j := i + 1; j < len(roots); j++ { + if RootPathsOverlap(roots[i], roots[j]) { + return fmt.Errorf("manifest has overlapped roots: '%v' and '%v'", roots[i], roots[j]) + } + } + } + + // Validate modules in bundle. + for _, module := range b.Modules { + found := false + if path, err := module.Parsed.Package.Path.Ptr(); err == nil { + for i := range roots { + if strings.HasPrefix(path, roots[i]) { + found = true + break + } + } + } + if !found { + return fmt.Errorf("manifest roots %v do not permit '%v' in module '%v'", roots, module.Parsed.Package, module.Path) + } + } + + // Build a set of wasm module entrypoints to validate + wasmModuleToEps := map[string]string{} + seenEps := map[string]struct{}{} + for _, wm := range b.WasmModules { + wasmModuleToEps[wm.Path] = "" + } + + for _, wmConfig := range b.Manifest.WasmResolvers { + _, ok := wasmModuleToEps[wmConfig.Module] + if !ok { + return fmt.Errorf("manifest references wasm module '%s' but the module file does not exist", wmConfig.Module) + } + + // Ensure wasm module entrypoint in within bundle roots + found := false + for i := range roots { + if strings.HasPrefix(wmConfig.Entrypoint, roots[i]) { + found = true + break + } + } + + if !found { + return fmt.Errorf("manifest roots %v do not permit '%v' entrypoint for wasm module '%v'", roots, wmConfig.Entrypoint, wmConfig.Module) + } + + if _, ok := seenEps[wmConfig.Entrypoint]; ok { + return fmt.Errorf("entrypoint '%s' cannot be used by more than one wasm module", wmConfig.Entrypoint) + } + seenEps[wmConfig.Entrypoint] = struct{}{} + + wasmModuleToEps[wmConfig.Module] = wmConfig.Entrypoint + } + + // Validate data in bundle. + return dfs(b.Data, "", func(path string, node interface{}) (bool, error) { + path = strings.Trim(path, "/") + for i := range roots { + if strings.HasPrefix(path, roots[i]) { + return true, nil + } + } + if _, ok := node.(map[string]interface{}); ok { + for i := range roots { + if strings.HasPrefix(roots[i], path) { + return false, nil + } + } + } + return false, fmt.Errorf("manifest roots %v do not permit data at path '/%s' (hint: check bundle directory structure)", roots, path) + }) +} + +// ModuleFile represents a single module contained in a bundle. +type ModuleFile struct { + URL string + Path string + Raw []byte + Parsed *ast.Module +} + +// WasmModuleFile represents a single wasm module contained in a bundle. +type WasmModuleFile struct { + URL string + Path string + Entrypoints []ast.Ref + Raw []byte +} + +// Reader contains the reader to load the bundle from. +type Reader struct { + loader DirectoryLoader + includeManifestInData bool + metrics metrics.Metrics + baseDir string + verificationConfig *VerificationConfig + skipVerify bool + processAnnotations bool + files map[string]FileInfo // files in the bundle signature payload + sizeLimitBytes int64 +} + +// NewReader is deprecated. Use NewCustomReader instead. +func NewReader(r io.Reader) *Reader { + return NewCustomReader(NewTarballLoader(r)) +} + +// NewCustomReader returns a new Reader configured to use the +// specified DirectoryLoader. +func NewCustomReader(loader DirectoryLoader) *Reader { + nr := Reader{ + loader: loader, + metrics: metrics.New(), + files: make(map[string]FileInfo), + sizeLimitBytes: DefaultSizeLimitBytes + 1, + } + return &nr +} + +// IncludeManifestInData sets whether the manifest metadata should be +// included in the bundle's data. +func (r *Reader) IncludeManifestInData(includeManifestInData bool) *Reader { + r.includeManifestInData = includeManifestInData + return r +} + +// WithMetrics sets the metrics object to be used while loading bundles +func (r *Reader) WithMetrics(m metrics.Metrics) *Reader { + r.metrics = m + return r +} + +// WithBaseDir sets a base directory for file paths of loaded Rego +// modules. This will *NOT* affect the loaded path of data files. +func (r *Reader) WithBaseDir(dir string) *Reader { + r.baseDir = dir + return r +} + +// WithBundleVerificationConfig sets the key configuration used to verify a signed bundle +func (r *Reader) WithBundleVerificationConfig(config *VerificationConfig) *Reader { + r.verificationConfig = config + return r +} + +// WithSkipBundleVerification skips verification of a signed bundle +func (r *Reader) WithSkipBundleVerification(skipVerify bool) *Reader { + r.skipVerify = skipVerify + return r +} + +// WithProcessAnnotations enables annotation processing during .rego file parsing. +func (r *Reader) WithProcessAnnotations(yes bool) *Reader { + r.processAnnotations = yes + return r +} + +// WithSizeLimitBytes sets the size limit to apply to files in the bundle. If files are larger +// than this, an error will be returned by the reader. +func (r *Reader) WithSizeLimitBytes(n int64) *Reader { + r.sizeLimitBytes = n + 1 + return r +} + +// Read returns a new Bundle loaded from the reader. +func (r *Reader) Read() (Bundle, error) { + + var bundle Bundle + var descriptors []*Descriptor + var err error + + bundle.Data = map[string]interface{}{} + + bundle.Signatures, descriptors, err = listSignaturesAndDescriptors(r.loader, r.skipVerify, r.sizeLimitBytes) + if err != nil { + return bundle, err + } + + err = r.checkSignaturesAndDescriptors(bundle.Signatures) + if err != nil { + return bundle, err + } + + for _, f := range descriptors { + var buf bytes.Buffer + n, err := f.Read(&buf, r.sizeLimitBytes) + f.Close() // always close, even on error + + if err != nil && err != io.EOF { + return bundle, err + } else if err == nil && n >= r.sizeLimitBytes { + return bundle, fmt.Errorf("bundle file exceeded max size (%v bytes)", r.sizeLimitBytes-1) + } + + // verify the file content + if !bundle.Signatures.isEmpty() { + path := f.Path() + if r.baseDir != "" { + path = f.URL() + } + path = strings.TrimPrefix(path, "/") + + // check if the file is to be excluded from bundle verification + if r.isFileExcluded(path) { + delete(r.files, path) + } else { + if err = r.verifyBundleFile(path, buf); err != nil { + return bundle, err + } + } + } + + // Normalize the paths to use `/` separators + path := filepath.ToSlash(f.Path()) + + if strings.HasSuffix(path, RegoExt) { + fullPath := r.fullPath(path) + r.metrics.Timer(metrics.RegoModuleParse).Start() + module, err := ast.ParseModuleWithOpts(fullPath, buf.String(), ast.ParserOptions{ProcessAnnotation: r.processAnnotations}) + r.metrics.Timer(metrics.RegoModuleParse).Stop() + if err != nil { + return bundle, err + } + + mf := ModuleFile{ + URL: f.URL(), + Path: fullPath, + Raw: buf.Bytes(), + Parsed: module, + } + bundle.Modules = append(bundle.Modules, mf) + + } else if filepath.Base(path) == WasmFile { + bundle.WasmModules = append(bundle.WasmModules, WasmModuleFile{ + URL: f.URL(), + Path: r.fullPath(path), + Raw: buf.Bytes(), + }) + } else if filepath.Base(path) == dataFile { + var value interface{} + + r.metrics.Timer(metrics.RegoDataParse).Start() + err := util.NewJSONDecoder(&buf).Decode(&value) + r.metrics.Timer(metrics.RegoDataParse).Stop() + + if err != nil { + return bundle, errors.Wrapf(err, "bundle load failed on %v", r.fullPath(path)) + } + + if err := insertValue(&bundle, path, value); err != nil { + return bundle, err + } + + } else if filepath.Base(path) == yamlDataFile { + + var value interface{} + + r.metrics.Timer(metrics.RegoDataParse).Start() + err := util.Unmarshal(buf.Bytes(), &value) + r.metrics.Timer(metrics.RegoDataParse).Stop() + + if err != nil { + return bundle, errors.Wrapf(err, "bundle load failed on %v", r.fullPath(path)) + } + + if err := insertValue(&bundle, path, value); err != nil { + return bundle, err + } + + } else if strings.HasSuffix(path, ManifestExt) { + if err := util.NewJSONDecoder(&buf).Decode(&bundle.Manifest); err != nil { + return bundle, errors.Wrap(err, "bundle load failed on manifest decode") + } + } + } + + // check if the bundle signatures specify any files that weren't found in the bundle + if len(r.files) != 0 { + extra := []string{} + for k := range r.files { + extra = append(extra, k) + } + return bundle, fmt.Errorf("file(s) %v specified in bundle signatures but not found in the target bundle", extra) + } + + if err := bundle.Manifest.validateAndInjectDefaults(bundle); err != nil { + return bundle, err + } + + // Inject the wasm module entrypoint refs into the WasmModuleFile structs + epMap := map[string][]string{} + for _, r := range bundle.Manifest.WasmResolvers { + epMap[r.Module] = append(epMap[r.Module], r.Entrypoint) + } + for i := 0; i < len(bundle.WasmModules); i++ { + entrypoints := epMap[bundle.WasmModules[i].Path] + for _, entrypoint := range entrypoints { + ref, err := ast.PtrRef(ast.DefaultRootDocument, entrypoint) + if err != nil { + return bundle, fmt.Errorf("failed to parse wasm module entrypoint '%s': %s", entrypoint, err) + } + bundle.WasmModules[i].Entrypoints = append(bundle.WasmModules[i].Entrypoints, ref) + } + } + + if r.includeManifestInData { + var metadata map[string]interface{} + + b, err := json.Marshal(&bundle.Manifest) + if err != nil { + return bundle, errors.Wrap(err, "bundle load failed on manifest marshal") + } + + err = util.UnmarshalJSON(b, &metadata) + if err != nil { + return bundle, errors.Wrap(err, "bundle load failed on manifest unmarshal") + } + + // For backwards compatibility always write to the old unnamed manifest path + // This will *not* be correct if >1 bundle is in use... + if err := bundle.insertData(legacyManifestStoragePath, metadata); err != nil { + return bundle, errors.Wrapf(err, "bundle load failed on %v", legacyRevisionStoragePath) + } + } + + return bundle, nil +} + +func (r *Reader) isFileExcluded(path string) bool { + for _, e := range r.verificationConfig.Exclude { + match, _ := filepath.Match(e, path) + if match { + return true + } + } + return false +} + +func (r *Reader) checkSignaturesAndDescriptors(signatures SignaturesConfig) error { + if r.skipVerify { + return nil + } + + if signatures.isEmpty() && r.verificationConfig != nil && r.verificationConfig.KeyID != "" { + return fmt.Errorf("bundle missing .signatures.json file") + } + + if !signatures.isEmpty() { + if r.verificationConfig == nil { + return fmt.Errorf("verification key not provided") + } + + // verify the JWT signatures included in the `.signatures.json` file + if err := r.verifyBundleSignature(signatures); err != nil { + return err + } + } + return nil +} + +func (r *Reader) verifyBundleSignature(sc SignaturesConfig) error { + var err error + r.files, err = VerifyBundleSignature(sc, r.verificationConfig) + return err +} + +func (r *Reader) verifyBundleFile(path string, data bytes.Buffer) error { + return VerifyBundleFile(path, data, r.files) +} + +func (r *Reader) fullPath(path string) string { + if r.baseDir != "" { + path = filepath.Join(r.baseDir, path) + } + return path +} + +// Write is deprecated. Use NewWriter instead. +func Write(w io.Writer, bundle Bundle) error { + return NewWriter(w). + UseModulePath(true). + DisableFormat(true). + Write(bundle) +} + +// Writer implements bundle serialization. +type Writer struct { + usePath bool + disableFormat bool + w io.Writer +} + +// NewWriter returns a bundle writer that writes to w. +func NewWriter(w io.Writer) *Writer { + return &Writer{ + w: w, + } +} + +// UseModulePath configures the writer to use the module file path instead of the +// module file URL during serialization. This is for backwards compatibility. +func (w *Writer) UseModulePath(yes bool) *Writer { + w.usePath = yes + return w +} + +// DisableFormat configures the writer to just write out raw bytes instead +// of formatting modules before serialization. +func (w *Writer) DisableFormat(yes bool) *Writer { + w.disableFormat = yes + return w +} + +// Write writes the bundle to the writer's output stream. +func (w *Writer) Write(bundle Bundle) error { + gw := gzip.NewWriter(w.w) + tw := tar.NewWriter(gw) + + var buf bytes.Buffer + + if err := json.NewEncoder(&buf).Encode(bundle.Data); err != nil { + return err + } + + if err := archive.WriteFile(tw, "data.json", buf.Bytes()); err != nil { + return err + } + + for _, module := range bundle.Modules { + path := module.URL + if w.usePath { + path = module.Path + } + + if err := archive.WriteFile(tw, path, module.Raw); err != nil { + return err + } + } + + if err := w.writeWasm(tw, bundle); err != nil { + return err + } + + if err := writeManifest(tw, bundle); err != nil { + return err + } + + if err := writeSignatures(tw, bundle); err != nil { + return err + } + + if err := tw.Close(); err != nil { + return err + } + + return gw.Close() +} + +func (w *Writer) writeWasm(tw *tar.Writer, bundle Bundle) error { + for _, wm := range bundle.WasmModules { + path := wm.URL + if w.usePath { + path = wm.Path + } + + err := archive.WriteFile(tw, path, wm.Raw) + if err != nil { + return err + } + } + + if len(bundle.Wasm) > 0 { + err := archive.WriteFile(tw, "/"+WasmFile, bundle.Wasm) + if err != nil { + return err + } + } + + return nil +} + +func writeManifest(tw *tar.Writer, bundle Bundle) error { + + if bundle.Manifest.Equal(Manifest{}) { + return nil + } + + var buf bytes.Buffer + + if err := json.NewEncoder(&buf).Encode(bundle.Manifest); err != nil { + return err + } + + return archive.WriteFile(tw, ManifestExt, buf.Bytes()) +} + +func writeSignatures(tw *tar.Writer, bundle Bundle) error { + + if bundle.Signatures.isEmpty() { + return nil + } + + bs, err := json.MarshalIndent(bundle.Signatures, "", " ") + if err != nil { + return err + } + + return archive.WriteFile(tw, fmt.Sprintf(".%v", SignaturesFile), bs) +} + +func hashBundleFiles(hash SignatureHasher, b *Bundle) ([]FileInfo, error) { + + files := []FileInfo{} + + bs, err := hash.HashFile(b.Data) + if err != nil { + return files, err + } + files = append(files, NewFile(strings.TrimPrefix("data.json", "/"), hex.EncodeToString(bs), defaultHashingAlg)) + + if len(b.Wasm) != 0 { + bs, err := hash.HashFile(b.Wasm) + if err != nil { + return files, err + } + files = append(files, NewFile(strings.TrimPrefix(WasmFile, "/"), hex.EncodeToString(bs), defaultHashingAlg)) + } + + for _, wasmModule := range b.WasmModules { + bs, err := hash.HashFile(wasmModule.Raw) + if err != nil { + return files, err + } + files = append(files, NewFile(strings.TrimPrefix(wasmModule.Path, "/"), hex.EncodeToString(bs), defaultHashingAlg)) + } + + bs, err = hash.HashFile(b.Manifest) + if err != nil { + return files, err + } + files = append(files, NewFile(strings.TrimPrefix(ManifestExt, "/"), hex.EncodeToString(bs), defaultHashingAlg)) + + return files, err +} + +// FormatModules formats Rego modules +func (b *Bundle) FormatModules(useModulePath bool) error { + var err error + + for i, module := range b.Modules { + if module.Raw == nil { + module.Raw, err = format.Ast(module.Parsed) + if err != nil { + return err + } + } else { + path := module.URL + if useModulePath { + path = module.Path + } + + module.Raw, err = format.Source(path, module.Raw) + if err != nil { + return err + } + } + b.Modules[i].Raw = module.Raw + } + return nil +} + +// GenerateSignature generates the signature for the given bundle. +func (b *Bundle) GenerateSignature(signingConfig *SigningConfig, keyID string, useModulePath bool) error { + + hash, err := NewSignatureHasher(HashingAlgorithm(defaultHashingAlg)) + if err != nil { + return err + } + + files := []FileInfo{} + + for _, module := range b.Modules { + bytes, err := hash.HashFile(module.Raw) + if err != nil { + return err + } + + path := module.URL + if useModulePath { + path = module.Path + } + files = append(files, NewFile(strings.TrimPrefix(path, "/"), hex.EncodeToString(bytes), defaultHashingAlg)) + } + + result, err := hashBundleFiles(hash, b) + if err != nil { + return err + } + files = append(files, result...) + + // generate signed token + token, err := GenerateSignedToken(files, signingConfig, keyID) + if err != nil { + return err + } + + if b.Signatures.isEmpty() { + b.Signatures = SignaturesConfig{} + } + + if signingConfig.Plugin != "" { + b.Signatures.Plugin = signingConfig.Plugin + } + + b.Signatures.Signatures = []string{string(token)} + + return nil +} + +// ParsedModules returns a map of parsed modules with names that are +// unique and human readable for the given a bundle name. +func (b *Bundle) ParsedModules(bundleName string) map[string]*ast.Module { + + mods := make(map[string]*ast.Module, len(b.Modules)) + + for _, mf := range b.Modules { + mods[modulePathWithPrefix(bundleName, mf.Path)] = mf.Parsed + } + + return mods +} + +// Equal returns true if this bundle's contents equal the other bundle's +// contents. +func (b Bundle) Equal(other Bundle) bool { + if !reflect.DeepEqual(b.Data, other.Data) { + return false + } + + if len(b.Modules) != len(other.Modules) { + return false + } + for i := range b.Modules { + if b.Modules[i].URL != other.Modules[i].URL { + return false + } + if b.Modules[i].Path != other.Modules[i].Path { + return false + } + if !b.Modules[i].Parsed.Equal(other.Modules[i].Parsed) { + return false + } + if !bytes.Equal(b.Modules[i].Raw, other.Modules[i].Raw) { + return false + } + } + if (b.Wasm == nil && other.Wasm != nil) || (b.Wasm != nil && other.Wasm == nil) { + return false + } + + return bytes.Equal(b.Wasm, other.Wasm) +} + +// Copy returns a deep copy of the bundle. +func (b Bundle) Copy() Bundle { + + // Copy data. + var x interface{} = b.Data + + if err := util.RoundTrip(&x); err != nil { + panic(err) + } + + if x != nil { + b.Data = x.(map[string]interface{}) + } + + // Copy modules. + for i := range b.Modules { + bs := make([]byte, len(b.Modules[i].Raw)) + copy(bs, b.Modules[i].Raw) + b.Modules[i].Raw = bs + b.Modules[i].Parsed = b.Modules[i].Parsed.Copy() + } + + // Copy manifest. + b.Manifest = b.Manifest.Copy() + + return b +} + +func (b *Bundle) insertData(key []string, value interface{}) error { + // Build an object with the full structure for the value + obj, err := mktree(key, value) + if err != nil { + return err + } + + // Merge the new data in with the current bundle data object + merged, ok := merge.InterfaceMaps(b.Data, obj) + if !ok { + return fmt.Errorf("failed to insert data file from path %s", filepath.Join(key...)) + } + + b.Data = merged + + return nil +} + +func (b *Bundle) readData(key []string) *interface{} { + + if len(key) == 0 { + if len(b.Data) == 0 { + return nil + } + var result interface{} = b.Data + return &result + } + + node := b.Data + + for i := 0; i < len(key)-1; i++ { + + child, ok := node[key[i]] + if !ok { + return nil + } + + childObj, ok := child.(map[string]interface{}) + if !ok { + return nil + } + + node = childObj + } + + child, ok := node[key[len(key)-1]] + if !ok { + return nil + } + + return &child +} + +func mktree(path []string, value interface{}) (map[string]interface{}, error) { + if len(path) == 0 { + // For 0 length path the value is the full tree. + obj, ok := value.(map[string]interface{}) + if !ok { + return nil, fmt.Errorf("root value must be object") + } + return obj, nil + } + + dir := map[string]interface{}{} + for i := len(path) - 1; i > 0; i-- { + dir[path[i]] = value + value = dir + dir = map[string]interface{}{} + } + dir[path[0]] = value + + return dir, nil +} + +// Merge accepts a set of bundles and merges them into a single result bundle. If there are +// any conflicts during the merge (e.g., with roots) an error is returned. The result bundle +// will have an empty revision except in the special case where a single bundle is provided +// (and in that case the bundle is just returned unmodified.) +func Merge(bundles []*Bundle) (*Bundle, error) { + + if len(bundles) == 0 { + return nil, errors.New("expected at least one bundle") + } + + if len(bundles) == 1 { + return bundles[0], nil + } + + var roots []string + var result Bundle + + for _, b := range bundles { + + if b.Manifest.Roots == nil { + return nil, errors.New("bundle manifest not initialized") + } + + roots = append(roots, *b.Manifest.Roots...) + + result.Modules = append(result.Modules, b.Modules...) + + for _, root := range *b.Manifest.Roots { + key := strings.Split(root, "/") + if val := b.readData(key); val != nil { + if err := result.insertData(key, *val); err != nil { + return nil, err + } + } + } + + result.Manifest.WasmResolvers = append(result.Manifest.WasmResolvers, b.Manifest.WasmResolvers...) + result.WasmModules = append(result.WasmModules, b.WasmModules...) + + } + + result.Manifest.Roots = &roots + + if err := result.Manifest.validateAndInjectDefaults(result); err != nil { + return nil, err + } + + return &result, nil +} + +// RootPathsOverlap takes in two bundle root paths and returns true if they overlap. +func RootPathsOverlap(pathA string, pathB string) bool { + a := rootPathSegments(pathA) + b := rootPathSegments(pathB) + return rootContains(a, b) || rootContains(b, a) +} + +// RootPathsContain takes a set of bundle root paths and returns true if the path is contained. +func RootPathsContain(roots []string, path string) bool { + segments := rootPathSegments(path) + for i := range roots { + if rootContains(rootPathSegments(roots[i]), segments) { + return true + } + } + return false +} + +func rootPathSegments(path string) []string { + return strings.Split(path, "/") +} + +func rootContains(root []string, other []string) bool { + + // A single segment, empty string root always contains the other. + if len(root) == 1 && root[0] == "" { + return true + } + + if len(root) > len(other) { + return false + } + + for j := range root { + if root[j] != other[j] { + return false + } + } + + return true +} + +func insertValue(b *Bundle, path string, value interface{}) error { + + // Remove leading / and . characters from the directory path. If the bundle + // was written with OPA then the paths will contain a leading slash. On the + // other hand, if the path is empty, filepath.Dir will return '.'. + // Note: filepath.Dir can return paths with '\' separators, always use + // filepath.ToSlash to keep them normalized. + dirpath := strings.TrimLeft(filepath.ToSlash(filepath.Dir(path)), "/.") + var key []string + if dirpath != "" { + key = strings.Split(dirpath, "/") + } + if err := b.insertData(key, value); err != nil { + return errors.Wrapf(err, "bundle load failed on %v", path) + } + return nil +} + +func dfs(value interface{}, path string, fn func(string, interface{}) (bool, error)) error { + if stop, err := fn(path, value); err != nil { + return err + } else if stop { + return nil + } + obj, ok := value.(map[string]interface{}) + if !ok { + return nil + } + for key := range obj { + if err := dfs(obj[key], path+"/"+key, fn); err != nil { + return err + } + } + return nil +} + +func modulePathWithPrefix(bundleName string, modulePath string) string { + // Default prefix is just the bundle name + prefix := bundleName + + // Bundle names are sometimes just file paths, some of which + // are full urls (file:///foo/). Parse these and only use the path. + parsed, err := url.Parse(bundleName) + if err == nil { + prefix = filepath.Join(parsed.Host, parsed.Path) + } + + return filepath.Join(prefix, modulePath) +} + +// IsStructuredDoc checks if the file name equals a structured file extension ex. ".json" +func IsStructuredDoc(name string) bool { + return filepath.Base(name) == dataFile || filepath.Base(name) == yamlDataFile || + filepath.Base(name) == SignaturesFile || filepath.Base(name) == ManifestExt +} + +func listSignaturesAndDescriptors(loader DirectoryLoader, skipVerify bool, sizeLimitBytes int64) (SignaturesConfig, []*Descriptor, error) { + descriptors := []*Descriptor{} + var signatures SignaturesConfig + + for { + f, err := loader.NextFile() + if err == io.EOF { + break + } + + if err != nil { + return signatures, nil, errors.Wrap(err, "bundle read failed") + } + + // check for the signatures file + if !skipVerify && strings.HasSuffix(f.Path(), SignaturesFile) { + var buf bytes.Buffer + n, err := f.Read(&buf, sizeLimitBytes) + f.Close() // always close, even on error + if err != nil && err != io.EOF { + return signatures, nil, err + } else if err == nil && n >= sizeLimitBytes { + return signatures, nil, fmt.Errorf("bundle signatures file exceeded max size (%v bytes)", sizeLimitBytes-1) + } + + if err := util.NewJSONDecoder(&buf).Decode(&signatures); err != nil { + return signatures, nil, errors.Wrap(err, "bundle load failed on signatures decode") + } + } else if !strings.HasSuffix(f.Path(), SignaturesFile) { + descriptors = append(descriptors, f) + } + } + return signatures, descriptors, nil +} diff --git a/vendor/github.com/open-policy-agent/opa/bundle/file.go b/vendor/github.com/open-policy-agent/opa/bundle/file.go new file mode 100644 index 00000000..040c0af3 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/bundle/file.go @@ -0,0 +1,268 @@ +package bundle + +import ( + "archive/tar" + "bytes" + "compress/gzip" + "io" + "os" + "path" + "path/filepath" + "strings" + "sync" + + "github.com/pkg/errors" +) + +// Descriptor contains information about a file and +// can be used to read the file contents. +type Descriptor struct { + url string + path string + reader io.Reader + closer io.Closer + closeOnce *sync.Once +} + +// lazyFile defers reading the file until the first call of Read +type lazyFile struct { + path string + file *os.File +} + +// newLazyFile creates a new instance of lazyFile +func newLazyFile(path string) *lazyFile { + return &lazyFile{path: path} +} + +// Read implements io.Reader. It will check if the file has been opened +// and open it if it has not before attempting to read using the file's +// read method +func (f *lazyFile) Read(b []byte) (int, error) { + var err error + + if f.file == nil { + if f.file, err = os.Open(f.path); err != nil { + return 0, errors.Wrapf(err, "failed to open file %s", f.path) + } + } + + return f.file.Read(b) +} + +// Close closes the lazy file if it has been opened using the file's +// close method +func (f *lazyFile) Close() error { + if f.file != nil { + return f.file.Close() + } + + return nil +} + +func newDescriptor(url, path string, reader io.Reader) *Descriptor { + return &Descriptor{ + url: url, + path: path, + reader: reader, + } +} + +func (d *Descriptor) withCloser(closer io.Closer) *Descriptor { + d.closer = closer + d.closeOnce = new(sync.Once) + return d +} + +// Path returns the path of the file. +func (d *Descriptor) Path() string { + return d.path +} + +// URL returns the url of the file. +func (d *Descriptor) URL() string { + return d.url +} + +// Read will read all the contents from the file the Descriptor refers to +// into the dest writer up n bytes. Will return an io.EOF error +// if EOF is encountered before n bytes are read. +func (d *Descriptor) Read(dest io.Writer, n int64) (int64, error) { + n, err := io.CopyN(dest, d.reader, n) + return n, err +} + +// Close the file, on some Loader implementations this might be a no-op. +// It should *always* be called regardless of file. +func (d *Descriptor) Close() error { + var err error + if d.closer != nil { + d.closeOnce.Do(func() { + err = d.closer.Close() + }) + } + return err +} + +// DirectoryLoader defines an interface which can be used to load +// files from a directory by iterating over each one in the tree. +type DirectoryLoader interface { + // NextFile must return io.EOF if there is no next value. The returned + // descriptor should *always* be closed when no longer needed. + NextFile() (*Descriptor, error) +} + +type dirLoader struct { + root string + files []string + idx int +} + +// NewDirectoryLoader returns a basic DirectoryLoader implementation +// that will load files from a given root directory path. +func NewDirectoryLoader(root string) DirectoryLoader { + + if len(root) > 1 { + // Normalize relative directories, ex "./src/bundle" -> "src/bundle" + // We don't need an absolute path, but this makes the joined/trimmed + // paths more uniform. + if root[0] == '.' && root[1] == filepath.Separator { + if len(root) == 2 { + root = root[:1] // "./" -> "." + } else { + root = root[2:] // remove leading "./" + } + } + } + + d := dirLoader{ + root: root, + } + return &d +} + +// NextFile iterates to the next file in the directory tree +// and returns a file Descriptor for the file. +func (d *dirLoader) NextFile() (*Descriptor, error) { + // build a list of all files we will iterate over and read, but only one time + if d.files == nil { + d.files = []string{} + err := filepath.Walk(d.root, func(path string, info os.FileInfo, err error) error { + if info != nil && info.Mode().IsRegular() { + d.files = append(d.files, filepath.ToSlash(path)) + } + return nil + }) + if err != nil { + return nil, errors.Wrap(err, "failed to list files") + } + } + + // If done reading files then just return io.EOF + // errors for each NextFile() call + if d.idx >= len(d.files) { + return nil, io.EOF + } + + fileName := d.files[d.idx] + d.idx++ + fh := newLazyFile(fileName) + + // Trim off the root directory and return path as if chrooted + cleanedPath := strings.TrimPrefix(fileName, d.root) + if d.root == "." && filepath.Base(fileName) == ManifestExt { + cleanedPath = fileName + } + + if !strings.HasPrefix(cleanedPath, "/") { + cleanedPath = "/" + cleanedPath + } + + f := newDescriptor(path.Join(d.root, cleanedPath), cleanedPath, fh).withCloser(fh) + return f, nil +} + +type tarballLoader struct { + baseURL string + r io.Reader + tr *tar.Reader + files []file + idx int +} + +type file struct { + name string + reader io.Reader +} + +// NewTarballLoader is deprecated. Use NewTarballLoaderWithBaseURL instead. +func NewTarballLoader(r io.Reader) DirectoryLoader { + l := tarballLoader{ + r: r, + } + return &l +} + +// NewTarballLoaderWithBaseURL returns a new DirectoryLoader that reads +// files out of a gzipped tar archive. The file URLs will be prefixed +// with the baseURL. +func NewTarballLoaderWithBaseURL(r io.Reader, baseURL string) DirectoryLoader { + l := tarballLoader{ + baseURL: strings.TrimSuffix(baseURL, "/"), + r: r, + } + return &l +} + +// NextFile iterates to the next file in the directory tree +// and returns a file Descriptor for the file. +func (t *tarballLoader) NextFile() (*Descriptor, error) { + if t.tr == nil { + gr, err := gzip.NewReader(t.r) + if err != nil { + return nil, errors.Wrap(err, "archive read failed") + } + + t.tr = tar.NewReader(gr) + } + + if t.files == nil { + t.files = []file{} + + for { + header, err := t.tr.Next() + if err == io.EOF { + break + } + + if err != nil { + return nil, err + } + + // Keep iterating on the archive until we find a normal file + if header.Typeflag == tar.TypeReg { + f := file{name: header.Name} + + var buf bytes.Buffer + if _, err := io.Copy(&buf, t.tr); err != nil { + return nil, errors.Wrapf(err, "failed to copy file %s", header.Name) + } + + f.reader = &buf + + t.files = append(t.files, f) + } + } + } + + // If done reading files then just return io.EOF + // errors for each NextFile() call + if t.idx >= len(t.files) { + return nil, io.EOF + } + + f := t.files[t.idx] + t.idx++ + + return newDescriptor(path.Join(t.baseURL, f.name), f.name, f.reader), nil +} diff --git a/vendor/github.com/open-policy-agent/opa/bundle/filefs.go b/vendor/github.com/open-policy-agent/opa/bundle/filefs.go new file mode 100644 index 00000000..f587456e --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/bundle/filefs.go @@ -0,0 +1,74 @@ +//go:build go1.16 +// +build go1.16 + +package bundle + +import ( + "fmt" + "io" + "io/fs" + "sync" +) + +const ( + defaultFSLoaderRoot = "." +) + +type dirLoaderFS struct { + sync.Mutex + filesystem fs.FS + files []string + idx int +} + +// NewFSLoader returns a basic DirectoryLoader implementation +// that will load files from a fs.FS interface +func NewFSLoader(filesystem fs.FS) (DirectoryLoader, error) { + d := dirLoaderFS{ + filesystem: filesystem, + } + + err := fs.WalkDir(d.filesystem, defaultFSLoaderRoot, d.walkDir) + if err != nil { + return nil, fmt.Errorf("failed to list files: %w", err) + } + + return &d, nil +} + +func (d *dirLoaderFS) walkDir(path string, dirEntry fs.DirEntry, err error) error { + if err != nil { + return err + } + + if dirEntry != nil && dirEntry.Type().IsRegular() { + d.files = append(d.files, path) + } + + return nil +} + +// NextFile iterates to the next file in the directory tree +// and returns a file Descriptor for the file. +func (d *dirLoaderFS) NextFile() (*Descriptor, error) { + d.Lock() + defer d.Unlock() + + // If done reading files then just return io.EOF + // errors for each NextFile() call + if d.idx >= len(d.files) { + return nil, io.EOF + } + + fileName := d.files[d.idx] + d.idx++ + + fh, err := d.filesystem.Open(fileName) + if err != nil { + return nil, fmt.Errorf("failed to open file %s: %w", fileName, err) + } + + fileNameWithSlash := fmt.Sprintf("/%s", fileName) + f := newDescriptor(fileNameWithSlash, fileNameWithSlash, fh).withCloser(fh) + return f, nil +} diff --git a/vendor/github.com/open-policy-agent/opa/bundle/hash.go b/vendor/github.com/open-policy-agent/opa/bundle/hash.go new file mode 100644 index 00000000..021801bb --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/bundle/hash.go @@ -0,0 +1,141 @@ +// Copyright 2020 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package bundle + +import ( + "bytes" + "crypto/md5" + "crypto/sha1" + "crypto/sha256" + "crypto/sha512" + "encoding/json" + "fmt" + "hash" + "io" + "sort" + "strings" +) + +// HashingAlgorithm represents a subset of hashing algorithms implemented in Go +type HashingAlgorithm string + +// Supported values for HashingAlgorithm +const ( + MD5 HashingAlgorithm = "MD5" + SHA1 HashingAlgorithm = "SHA-1" + SHA224 HashingAlgorithm = "SHA-224" + SHA256 HashingAlgorithm = "SHA-256" + SHA384 HashingAlgorithm = "SHA-384" + SHA512 HashingAlgorithm = "SHA-512" + SHA512224 HashingAlgorithm = "SHA-512-224" + SHA512256 HashingAlgorithm = "SHA-512-256" +) + +// String returns the string representation of a HashingAlgorithm +func (alg HashingAlgorithm) String() string { + return string(alg) +} + +// SignatureHasher computes a signature digest for a file with (structured or unstructured) data and policy +type SignatureHasher interface { + HashFile(v interface{}) ([]byte, error) +} + +type hasher struct { + h func() hash.Hash // hash function factory +} + +// NewSignatureHasher returns a signature hasher suitable for a particular hashing algorithm +func NewSignatureHasher(alg HashingAlgorithm) (SignatureHasher, error) { + h := &hasher{} + + switch alg { + case MD5: + h.h = md5.New + case SHA1: + h.h = sha1.New + case SHA224: + h.h = sha256.New224 + case SHA256: + h.h = sha256.New + case SHA384: + h.h = sha512.New384 + case SHA512: + h.h = sha512.New + case SHA512224: + h.h = sha512.New512_224 + case SHA512256: + h.h = sha512.New512_256 + default: + return nil, fmt.Errorf("unsupported hashing algorithm: %s", alg) + } + + return h, nil +} + +// HashFile hashes the file content, JSON or binary, both in golang native format. +func (h *hasher) HashFile(v interface{}) ([]byte, error) { + hf := h.h() + walk(v, hf) + return hf.Sum(nil), nil +} + +// walk hashes the file content, JSON or binary, both in golang native format. +// +// Computation for unstructured documents is a hash of the document. +// +// Computation for the types of structured JSON document is as follows: +// +// object: Hash {, then each key (in alphabetical order) and digest of the value, then comma (between items) and finally }. +// +// array: Hash [, then digest of the value, then comma (between items) and finally ]. +func walk(v interface{}, h io.Writer) { + + switch x := v.(type) { + case map[string]interface{}: + _, _ = h.Write([]byte("{")) + + var keys []string + for k := range x { + keys = append(keys, k) + } + sort.Strings(keys) + + for i, key := range keys { + if i > 0 { + _, _ = h.Write([]byte(",")) + } + + _, _ = h.Write(encodePrimitive(key)) + _, _ = h.Write([]byte(":")) + walk(x[key], h) + } + + _, _ = h.Write([]byte("}")) + case []interface{}: + _, _ = h.Write([]byte("[")) + + for i, e := range x { + if i > 0 { + _, _ = h.Write([]byte(",")) + } + walk(e, h) + } + + _, _ = h.Write([]byte("]")) + case []byte: + _, _ = h.Write(x) + default: + _, _ = h.Write(encodePrimitive(x)) + } +} + +func encodePrimitive(v interface{}) []byte { + var buf bytes.Buffer + encoder := json.NewEncoder(&buf) + encoder.SetEscapeHTML(false) + _ = encoder.Encode(v) + return []byte(strings.Trim(buf.String(), "\n")) +} diff --git a/vendor/github.com/open-policy-agent/opa/bundle/keys.go b/vendor/github.com/open-policy-agent/opa/bundle/keys.go new file mode 100644 index 00000000..1f45ba80 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/bundle/keys.go @@ -0,0 +1,145 @@ +// Copyright 2020 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +// Package bundle provide helpers that assist in creating the verification and signing key configuration +package bundle + +import ( + "encoding/pem" + "fmt" + "io/ioutil" + "os" + + "github.com/open-policy-agent/opa/internal/jwx/jwa" + "github.com/open-policy-agent/opa/internal/jwx/jws/sign" + "github.com/open-policy-agent/opa/keys" + + "github.com/open-policy-agent/opa/util" +) + +const ( + defaultTokenSigningAlg = "RS256" +) + +// KeyConfig holds the keys used to sign or verify bundles and tokens +// Moved to own package, alias kept for backwards compatibility +type KeyConfig = keys.Config + +// VerificationConfig represents the key configuration used to verify a signed bundle +type VerificationConfig struct { + PublicKeys map[string]*KeyConfig + KeyID string `json:"keyid"` + Scope string `json:"scope"` + Exclude []string `json:"exclude_files"` +} + +// NewVerificationConfig return a new VerificationConfig +func NewVerificationConfig(keys map[string]*KeyConfig, id, scope string, exclude []string) *VerificationConfig { + return &VerificationConfig{ + PublicKeys: keys, + KeyID: id, + Scope: scope, + Exclude: exclude, + } +} + +// ValidateAndInjectDefaults validates the config and inserts default values +func (vc *VerificationConfig) ValidateAndInjectDefaults(keys map[string]*KeyConfig) error { + vc.PublicKeys = keys + + if vc.KeyID != "" { + found := false + for key := range keys { + if key == vc.KeyID { + found = true + break + } + } + + if !found { + return fmt.Errorf("key id %s not found", vc.KeyID) + } + } + return nil +} + +// GetPublicKey returns the public key corresponding to the given key id +func (vc *VerificationConfig) GetPublicKey(id string) (*KeyConfig, error) { + var kc *KeyConfig + var ok bool + + if kc, ok = vc.PublicKeys[id]; !ok { + return nil, fmt.Errorf("verification key corresponding to ID %v not found", id) + } + return kc, nil +} + +// SigningConfig represents the key configuration used to generate a signed bundle +type SigningConfig struct { + Plugin string + Key string + Algorithm string + ClaimsPath string +} + +// NewSigningConfig return a new SigningConfig +func NewSigningConfig(key, alg, claimsPath string) *SigningConfig { + if alg == "" { + alg = defaultTokenSigningAlg + } + + return &SigningConfig{ + Plugin: defaultSignerID, + Key: key, + Algorithm: alg, + ClaimsPath: claimsPath, + } +} + +// WithPlugin sets the signing plugin in the signing config +func (s *SigningConfig) WithPlugin(plugin string) *SigningConfig { + if plugin != "" { + s.Plugin = plugin + } + return s +} + +// GetPrivateKey returns the private key or secret from the signing config +func (s *SigningConfig) GetPrivateKey() (interface{}, error) { + + block, _ := pem.Decode([]byte(s.Key)) + if block != nil { + return sign.GetSigningKey(s.Key, jwa.SignatureAlgorithm(s.Algorithm)) + } + + var priv string + if _, err := os.Stat(s.Key); err == nil { + bs, err := ioutil.ReadFile(s.Key) + if err != nil { + return nil, err + } + priv = string(bs) + } else if os.IsNotExist(err) { + priv = s.Key + } else { + return nil, err + } + + return sign.GetSigningKey(priv, jwa.SignatureAlgorithm(s.Algorithm)) +} + +// GetClaims returns the claims by reading the file specified in the signing config +func (s *SigningConfig) GetClaims() (map[string]interface{}, error) { + var claims map[string]interface{} + + bs, err := ioutil.ReadFile(s.ClaimsPath) + if err != nil { + return claims, err + } + + if err := util.UnmarshalJSON(bs, &claims); err != nil { + return claims, err + } + return claims, nil +} diff --git a/vendor/github.com/open-policy-agent/opa/bundle/sign.go b/vendor/github.com/open-policy-agent/opa/bundle/sign.go new file mode 100644 index 00000000..cf9a3e18 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/bundle/sign.go @@ -0,0 +1,135 @@ +// Copyright 2020 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +// Package bundle provide helpers that assist in the creating a signed bundle +package bundle + +import ( + "crypto/rand" + "encoding/json" + "fmt" + + "github.com/open-policy-agent/opa/internal/jwx/jwa" + "github.com/open-policy-agent/opa/internal/jwx/jws" +) + +const defaultSignerID = "_default" + +var signers map[string]Signer + +// Signer is the interface expected for implementations that generate bundle signatures. +type Signer interface { + GenerateSignedToken([]FileInfo, *SigningConfig, string) (string, error) +} + +// GenerateSignedToken will retrieve the Signer implementation based on the Plugin specified +// in SigningConfig, and call its implementation of GenerateSignedToken. The signer generates +// a signed token given the list of files to be included in the payload and the bundle +// signing config. The keyID if non-empty, represents the value for the "keyid" claim in the token. +func GenerateSignedToken(files []FileInfo, sc *SigningConfig, keyID string) (string, error) { + var plugin string + // for backwards compatibility, check if there is no plugin specified, and use default + if sc.Plugin == "" { + plugin = defaultSignerID + } else { + plugin = sc.Plugin + } + signer, err := GetSigner(plugin) + if err != nil { + return "", err + } + return signer.GenerateSignedToken(files, sc, keyID) +} + +// DefaultSigner is the default bundle signing implementation. It signs bundles by generating +// a JWT and signing it using a locally-accessible private key. +type DefaultSigner struct{} + +// GenerateSignedToken generates a signed token given the list of files to be +// included in the payload and the bundle signing config. The keyID if non-empty, +// represents the value for the "keyid" claim in the token +func (*DefaultSigner) GenerateSignedToken(files []FileInfo, sc *SigningConfig, keyID string) (string, error) { + payload, err := generatePayload(files, sc, keyID) + if err != nil { + return "", err + } + + privateKey, err := sc.GetPrivateKey() + if err != nil { + return "", err + } + + var headers jws.StandardHeaders + + if err := headers.Set(jws.AlgorithmKey, jwa.SignatureAlgorithm(sc.Algorithm)); err != nil { + return "", err + } + + if keyID != "" { + if err := headers.Set(jws.KeyIDKey, keyID); err != nil { + return "", err + } + } + + hdr, err := json.Marshal(headers) + if err != nil { + return "", err + } + + token, err := jws.SignLiteral(payload, + jwa.SignatureAlgorithm(sc.Algorithm), + privateKey, + hdr, + rand.Reader) + if err != nil { + return "", err + } + return string(token), nil +} + +func generatePayload(files []FileInfo, sc *SigningConfig, keyID string) ([]byte, error) { + payload := make(map[string]interface{}) + payload["files"] = files + + if sc.ClaimsPath != "" { + claims, err := sc.GetClaims() + if err != nil { + return nil, err + } + + for claim, value := range claims { + payload[claim] = value + } + } else { + if keyID != "" { + // keyid claim is deprecated but include it for backwards compatibility. + payload["keyid"] = keyID + } + } + return json.Marshal(payload) +} + +// GetSigner returns the Signer registered under the given id +func GetSigner(id string) (Signer, error) { + signer, ok := signers[id] + if !ok { + return nil, fmt.Errorf("no signer exists under id %s", id) + } + return signer, nil +} + +// RegisterSigner registers a Signer under the given id +func RegisterSigner(id string, s Signer) error { + if id == defaultSignerID { + return fmt.Errorf("signer id %s is reserved, use a different id", id) + } + signers[id] = s + return nil +} + +func init() { + signers = map[string]Signer{ + defaultSignerID: &DefaultSigner{}, + } +} diff --git a/vendor/github.com/open-policy-agent/opa/bundle/store.go b/vendor/github.com/open-policy-agent/opa/bundle/store.go new file mode 100644 index 00000000..0dc77964 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/bundle/store.go @@ -0,0 +1,644 @@ +// Copyright 2019 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package bundle + +import ( + "context" + "encoding/base64" + "encoding/json" + "fmt" + + "github.com/open-policy-agent/opa/ast" + "github.com/open-policy-agent/opa/metrics" + "github.com/open-policy-agent/opa/storage" + "github.com/open-policy-agent/opa/util" +) + +// BundlesBasePath is the storage path used for storing bundle metadata +var BundlesBasePath = storage.MustParsePath("/system/bundles") + +// Note: As needed these helpers could be memoized. + +// ManifestStoragePath is the storage path used for the given named bundle manifest. +func ManifestStoragePath(name string) storage.Path { + return append(BundlesBasePath, name, "manifest") +} + +func namedBundlePath(name string) storage.Path { + return append(BundlesBasePath, name) +} + +func rootsPath(name string) storage.Path { + return append(BundlesBasePath, name, "manifest", "roots") +} + +func revisionPath(name string) storage.Path { + return append(BundlesBasePath, name, "manifest", "revision") +} + +func wasmModulePath(name string) storage.Path { + return append(BundlesBasePath, name, "wasm") +} + +func wasmEntrypointsPath(name string) storage.Path { + return append(BundlesBasePath, name, "manifest", "wasm") +} + +func metadataPath(name string) storage.Path { + return append(BundlesBasePath, name, "manifest", "metadata") +} + +// ReadBundleNamesFromStore will return a list of bundle names which have had their metadata stored. +func ReadBundleNamesFromStore(ctx context.Context, store storage.Store, txn storage.Transaction) ([]string, error) { + value, err := store.Read(ctx, txn, BundlesBasePath) + if err != nil { + return nil, err + } + + bundleMap, ok := value.(map[string]interface{}) + if !ok { + return nil, fmt.Errorf("corrupt manifest roots") + } + + bundles := make([]string, len(bundleMap)) + idx := 0 + for name := range bundleMap { + bundles[idx] = name + idx++ + } + return bundles, nil +} + +// WriteManifestToStore will write the manifest into the storage. This function is called when +// the bundle is activated. +func WriteManifestToStore(ctx context.Context, store storage.Store, txn storage.Transaction, name string, manifest Manifest) error { + return write(ctx, store, txn, ManifestStoragePath(name), manifest) +} + +func write(ctx context.Context, store storage.Store, txn storage.Transaction, path storage.Path, value interface{}) error { + if err := util.RoundTrip(&value); err != nil { + return err + } + + var dir []string + if len(path) > 1 { + dir = path[:len(path)-1] + } + + if err := storage.MakeDir(ctx, store, txn, dir); err != nil { + return err + } + + return store.Write(ctx, txn, storage.AddOp, path, value) +} + +// EraseManifestFromStore will remove the manifest from storage. This function is called +// when the bundle is deactivated. +func EraseManifestFromStore(ctx context.Context, store storage.Store, txn storage.Transaction, name string) error { + path := namedBundlePath(name) + err := store.Write(ctx, txn, storage.RemoveOp, path, nil) + if err != nil && !storage.IsNotFound(err) { + return err + } + return nil +} + +func writeWasmModulesToStore(ctx context.Context, store storage.Store, txn storage.Transaction, name string, b *Bundle) error { + basePath := wasmModulePath(name) + for _, wm := range b.WasmModules { + path := append(basePath, wm.Path) + err := write(ctx, store, txn, path, base64.StdEncoding.EncodeToString(wm.Raw)) + if err != nil { + return err + } + } + return nil +} + +func eraseWasmModulesFromStore(ctx context.Context, store storage.Store, txn storage.Transaction, name string) error { + path := wasmModulePath(name) + + err := store.Write(ctx, txn, storage.RemoveOp, path, nil) + if err != nil && !storage.IsNotFound(err) { + return err + } + return nil +} + +// ReadWasmMetadataFromStore will read Wasm module resolver metadata from the store. +func ReadWasmMetadataFromStore(ctx context.Context, store storage.Store, txn storage.Transaction, name string) ([]WasmResolver, error) { + path := wasmEntrypointsPath(name) + value, err := store.Read(ctx, txn, path) + if err != nil { + return nil, err + } + + bs, err := json.Marshal(value) + if err != nil { + return nil, fmt.Errorf("corrupt wasm manifest data") + } + + var wasmMetadata []WasmResolver + + err = util.UnmarshalJSON(bs, &wasmMetadata) + if err != nil { + return nil, fmt.Errorf("corrupt wasm manifest data") + } + + return wasmMetadata, nil +} + +// ReadWasmModulesFromStore will write Wasm module resolver metadata from the store. +func ReadWasmModulesFromStore(ctx context.Context, store storage.Store, txn storage.Transaction, name string) (map[string][]byte, error) { + path := wasmModulePath(name) + value, err := store.Read(ctx, txn, path) + if err != nil { + return nil, err + } + + encodedModules, ok := value.(map[string]interface{}) + if !ok { + return nil, fmt.Errorf("corrupt wasm modules") + } + + rawModules := map[string][]byte{} + for path, enc := range encodedModules { + encStr, ok := enc.(string) + if !ok { + return nil, fmt.Errorf("corrupt wasm modules") + } + bs, err := base64.StdEncoding.DecodeString(encStr) + if err != nil { + return nil, err + } + rawModules[path] = bs + } + return rawModules, nil +} + +// ReadBundleRootsFromStore returns the roots in the specified bundle. +// If the bundle is not activated, this function will return +// storage NotFound error. +func ReadBundleRootsFromStore(ctx context.Context, store storage.Store, txn storage.Transaction, name string) ([]string, error) { + value, err := store.Read(ctx, txn, rootsPath(name)) + if err != nil { + return nil, err + } + + sl, ok := value.([]interface{}) + if !ok { + return nil, fmt.Errorf("corrupt manifest roots") + } + + roots := make([]string, len(sl)) + + for i := range sl { + roots[i], ok = sl[i].(string) + if !ok { + return nil, fmt.Errorf("corrupt manifest root") + } + } + + return roots, nil +} + +// ReadBundleRevisionFromStore returns the revision in the specified bundle. +// If the bundle is not activated, this function will return +// storage NotFound error. +func ReadBundleRevisionFromStore(ctx context.Context, store storage.Store, txn storage.Transaction, name string) (string, error) { + return readRevisionFromStore(ctx, store, txn, revisionPath(name)) +} + +func readRevisionFromStore(ctx context.Context, store storage.Store, txn storage.Transaction, path storage.Path) (string, error) { + value, err := store.Read(ctx, txn, path) + if err != nil { + return "", err + } + + str, ok := value.(string) + if !ok { + return "", fmt.Errorf("corrupt manifest revision") + } + + return str, nil +} + +// ReadBundleMetadataFromStore returns the metadata in the specified bundle. +// If the bundle is not activated, this function will return +// storage NotFound error. +func ReadBundleMetadataFromStore(ctx context.Context, store storage.Store, txn storage.Transaction, name string) (map[string]interface{}, error) { + return readMetadataFromStore(ctx, store, txn, metadataPath(name)) +} + +func readMetadataFromStore(ctx context.Context, store storage.Store, txn storage.Transaction, path storage.Path) (map[string]interface{}, error) { + value, err := store.Read(ctx, txn, path) + if err != nil { + if storageErr, ok := err.(*storage.Error); ok && storageErr.Code == storage.NotFoundErr { + return nil, nil + } + + return nil, err + } + + data, ok := value.(map[string]interface{}) + if !ok { + return nil, fmt.Errorf("corrupt manifest metadata") + } + + return data, nil +} + +// ActivateOpts defines options for the Activate API call. +type ActivateOpts struct { + Ctx context.Context + Store storage.Store + Txn storage.Transaction + TxnCtx *storage.Context + Compiler *ast.Compiler + Metrics metrics.Metrics + Bundles map[string]*Bundle // Optional + ExtraModules map[string]*ast.Module // Optional + + legacy bool +} + +// Activate the bundle(s) by loading into the given Store. This will load policies, data, and record +// the manifest in storage. The compiler provided will have had the polices compiled on it. +func Activate(opts *ActivateOpts) error { + opts.legacy = false + return activateBundles(opts) +} + +// DeactivateOpts defines options for the Deactivate API call +type DeactivateOpts struct { + Ctx context.Context + Store storage.Store + Txn storage.Transaction + BundleNames map[string]struct{} +} + +// Deactivate the bundle(s). This will erase associated data, policies, and the manifest entry from the store. +func Deactivate(opts *DeactivateOpts) error { + erase := map[string]struct{}{} + for name := range opts.BundleNames { + if roots, err := ReadBundleRootsFromStore(opts.Ctx, opts.Store, opts.Txn, name); err == nil { + for _, root := range roots { + erase[root] = struct{}{} + } + } else if !storage.IsNotFound(err) { + return err + } + } + _, err := eraseBundles(opts.Ctx, opts.Store, opts.Txn, opts.BundleNames, erase) + return err +} + +func activateBundles(opts *ActivateOpts) error { + + // Build collections of bundle names, modules, and roots to erase + erase := map[string]struct{}{} + names := map[string]struct{}{} + + for name, b := range opts.Bundles { + names[name] = struct{}{} + + if roots, err := ReadBundleRootsFromStore(opts.Ctx, opts.Store, opts.Txn, name); err == nil { + for _, root := range roots { + erase[root] = struct{}{} + } + } else if !storage.IsNotFound(err) { + return err + } + + // Erase data at new roots to prepare for writing the new data + for _, root := range *b.Manifest.Roots { + erase[root] = struct{}{} + } + } + + // Before changing anything make sure the roots don't collide with any + // other bundles that already are activated or other bundles being activated. + err := hasRootsOverlap(opts.Ctx, opts.Store, opts.Txn, opts.Bundles) + if err != nil { + return err + } + + // Erase data and policies at new + old roots, and remove the old + // manifests before activating a new bundles. + remaining, err := eraseBundles(opts.Ctx, opts.Store, opts.Txn, names, erase) + if err != nil { + return err + } + + for _, b := range opts.Bundles { + // Write data from each new bundle into the store. Only write under the + // roots contained in their manifest. This should be done *before* the + // policies so that path conflict checks can occur. + if err := writeData(opts.Ctx, opts.Store, opts.Txn, *b.Manifest.Roots, b.Data); err != nil { + return err + } + } + + // Write and compile the modules all at once to avoid having to re-do work. + remainingAndExtra := make(map[string]*ast.Module) + for name, mod := range remaining { + remainingAndExtra[name] = mod + } + for name, mod := range opts.ExtraModules { + remainingAndExtra[name] = mod + } + + err = writeModules(opts.Ctx, opts.Store, opts.Txn, opts.Compiler, opts.Metrics, opts.Bundles, remainingAndExtra, opts.legacy) + if err != nil { + return err + } + + for name, b := range opts.Bundles { + // Always write manifests to the named location. If the plugin is in the older style config + // then also write to the old legacy unnamed location. + if err := WriteManifestToStore(opts.Ctx, opts.Store, opts.Txn, name, b.Manifest); err != nil { + return err + } + if opts.legacy { + if err := LegacyWriteManifestToStore(opts.Ctx, opts.Store, opts.Txn, b.Manifest); err != nil { + return err + } + } + + if err := writeWasmModulesToStore(opts.Ctx, opts.Store, opts.Txn, name, b); err != nil { + return err + } + } + + return nil +} + +// erase bundles by name and roots. This will clear all policies and data at its roots and remove its +// manifest from storage. +func eraseBundles(ctx context.Context, store storage.Store, txn storage.Transaction, names map[string]struct{}, roots map[string]struct{}) (map[string]*ast.Module, error) { + + if err := eraseData(ctx, store, txn, roots); err != nil { + return nil, err + } + + remaining, err := erasePolicies(ctx, store, txn, roots) + if err != nil { + return nil, err + } + + for name := range names { + if err := EraseManifestFromStore(ctx, store, txn, name); err != nil && !storage.IsNotFound(err) { + return nil, err + } + + if err := LegacyEraseManifestFromStore(ctx, store, txn); err != nil && !storage.IsNotFound(err) { + return nil, err + } + + if err := eraseWasmModulesFromStore(ctx, store, txn, name); err != nil && !storage.IsNotFound(err) { + return nil, err + } + } + + return remaining, nil +} + +func eraseData(ctx context.Context, store storage.Store, txn storage.Transaction, roots map[string]struct{}) error { + for root := range roots { + path, ok := storage.ParsePathEscaped("/" + root) + if !ok { + return fmt.Errorf("manifest root path invalid: %v", root) + } + if len(path) > 0 { + if err := store.Write(ctx, txn, storage.RemoveOp, path, nil); err != nil { + if !storage.IsNotFound(err) { + return err + } + } + } + } + return nil +} + +func erasePolicies(ctx context.Context, store storage.Store, txn storage.Transaction, roots map[string]struct{}) (map[string]*ast.Module, error) { + + ids, err := store.ListPolicies(ctx, txn) + if err != nil { + return nil, err + } + + remaining := map[string]*ast.Module{} + + for _, id := range ids { + bs, err := store.GetPolicy(ctx, txn, id) + if err != nil { + return nil, err + } + module, err := ast.ParseModule(id, string(bs)) + if err != nil { + return nil, err + } + path, err := module.Package.Path.Ptr() + if err != nil { + return nil, err + } + deleted := false + for root := range roots { + if RootPathsContain([]string{root}, path) { + if err := store.DeletePolicy(ctx, txn, id); err != nil { + return nil, err + } + deleted = true + break + } + } + if !deleted { + remaining[id] = module + } + } + + return remaining, nil +} + +func writeData(ctx context.Context, store storage.Store, txn storage.Transaction, roots []string, data map[string]interface{}) error { + for _, root := range roots { + path, ok := storage.ParsePathEscaped("/" + root) + if !ok { + return fmt.Errorf("manifest root path invalid: %v", root) + } + if value, ok := lookup(path, data); ok { + if len(path) > 0 { + if err := storage.MakeDir(ctx, store, txn, path[:len(path)-1]); err != nil { + return err + } + } + if err := store.Write(ctx, txn, storage.AddOp, path, value); err != nil { + return err + } + } + } + return nil +} + +func writeModules(ctx context.Context, store storage.Store, txn storage.Transaction, compiler *ast.Compiler, m metrics.Metrics, bundles map[string]*Bundle, extraModules map[string]*ast.Module, legacy bool) error { + + m.Timer(metrics.RegoModuleCompile).Start() + defer m.Timer(metrics.RegoModuleCompile).Stop() + + modules := map[string]*ast.Module{} + + // preserve any modules already on the compiler + for name, module := range compiler.Modules { + modules[name] = module + } + + // preserve any modules passed in from the store + for name, module := range extraModules { + modules[name] = module + } + + // include all the new bundle modules + for bundleName, b := range bundles { + if legacy { + for _, mf := range b.Modules { + modules[mf.Path] = mf.Parsed + } + } else { + for name, module := range b.ParsedModules(bundleName) { + modules[name] = module + } + } + } + + if compiler.Compile(modules); compiler.Failed() { + return compiler.Errors + } + for bundleName, b := range bundles { + for _, mf := range b.Modules { + var path string + + // For backwards compatibility, in legacy mode, upsert policies to + // the unprefixed path. + if legacy { + path = mf.Path + } else { + path = modulePathWithPrefix(bundleName, mf.Path) + } + + if err := store.UpsertPolicy(ctx, txn, path, mf.Raw); err != nil { + return err + } + } + } + return nil +} + +func lookup(path storage.Path, data map[string]interface{}) (interface{}, bool) { + if len(path) == 0 { + return data, true + } + for i := 0; i < len(path)-1; i++ { + value, ok := data[path[i]] + if !ok { + return nil, false + } + obj, ok := value.(map[string]interface{}) + if !ok { + return nil, false + } + data = obj + } + value, ok := data[path[len(path)-1]] + return value, ok +} + +func hasRootsOverlap(ctx context.Context, store storage.Store, txn storage.Transaction, bundles map[string]*Bundle) error { + collisions := map[string][]string{} + allBundles, err := ReadBundleNamesFromStore(ctx, store, txn) + if err != nil && !storage.IsNotFound(err) { + return err + } + + allRoots := map[string][]string{} + + // Build a map of roots for existing bundles already in the system + for _, name := range allBundles { + roots, err := ReadBundleRootsFromStore(ctx, store, txn, name) + if err != nil && !storage.IsNotFound(err) { + return err + } + allRoots[name] = roots + } + + // Add in any bundles that are being activated, overwrite existing roots + // with new ones where bundles are in both groups. + for name, bundle := range bundles { + allRoots[name] = *bundle.Manifest.Roots + } + + // Now check for each new bundle if it conflicts with any of the others + for name, bundle := range bundles { + for otherBundle, otherRoots := range allRoots { + if name == otherBundle { + // Skip the current bundle being checked + continue + } + + // Compare the "new" roots with other existing (or a different bundles new roots) + for _, newRoot := range *bundle.Manifest.Roots { + for _, otherRoot := range otherRoots { + if RootPathsOverlap(newRoot, otherRoot) { + collisions[otherBundle] = append(collisions[otherBundle], newRoot) + } + } + } + } + } + + if len(collisions) > 0 { + var bundleNames []string + for name := range collisions { + bundleNames = append(bundleNames, name) + } + return fmt.Errorf("detected overlapping roots in bundle manifest with: %s", bundleNames) + } + return nil +} + +// Helpers for the older single (unnamed) bundle style manifest storage. + +// LegacyManifestStoragePath is the older unnamed bundle path for manifests to be stored. +// Deprecated: Use ManifestStoragePath and named bundles instead. +var legacyManifestStoragePath = storage.MustParsePath("/system/bundle/manifest") +var legacyRevisionStoragePath = append(legacyManifestStoragePath, "revision") + +// LegacyWriteManifestToStore will write the bundle manifest to the older single (unnamed) bundle manifest location. +// Deprecated: Use WriteManifestToStore and named bundles instead. +func LegacyWriteManifestToStore(ctx context.Context, store storage.Store, txn storage.Transaction, manifest Manifest) error { + return write(ctx, store, txn, legacyManifestStoragePath, manifest) +} + +// LegacyEraseManifestFromStore will erase the bundle manifest from the older single (unnamed) bundle manifest location. +// Deprecated: Use WriteManifestToStore and named bundles instead. +func LegacyEraseManifestFromStore(ctx context.Context, store storage.Store, txn storage.Transaction) error { + err := store.Write(ctx, txn, storage.RemoveOp, legacyManifestStoragePath, nil) + if err != nil { + return err + } + return nil +} + +// LegacyReadRevisionFromStore will read the bundle manifest revision from the older single (unnamed) bundle manifest location. +// Deprecated: Use ReadBundleRevisionFromStore and named bundles instead. +func LegacyReadRevisionFromStore(ctx context.Context, store storage.Store, txn storage.Transaction) (string, error) { + return readRevisionFromStore(ctx, store, txn, legacyRevisionStoragePath) +} + +// ActivateLegacy calls Activate for the bundles but will also write their manifest to the older unnamed store location. +// Deprecated: Use Activate with named bundles instead. +func ActivateLegacy(opts *ActivateOpts) error { + opts.legacy = true + return activateBundles(opts) +} diff --git a/vendor/github.com/open-policy-agent/opa/bundle/verify.go b/vendor/github.com/open-policy-agent/opa/bundle/verify.go new file mode 100644 index 00000000..f76baf2c --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/bundle/verify.go @@ -0,0 +1,233 @@ +// Copyright 2020 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +// Package bundle provide helpers that assist in the bundle signature verification process +package bundle + +import ( + "bytes" + "encoding/base64" + "encoding/hex" + "encoding/json" + "fmt" + + "github.com/open-policy-agent/opa/internal/jwx/jwa" + "github.com/open-policy-agent/opa/internal/jwx/jws" + "github.com/open-policy-agent/opa/internal/jwx/jws/verify" + "github.com/open-policy-agent/opa/util" + + "github.com/pkg/errors" +) + +const defaultVerifierID = "_default" + +var verifiers map[string]Verifier + +// Verifier is the interface expected for implementations that verify bundle signatures. +type Verifier interface { + VerifyBundleSignature(SignaturesConfig, *VerificationConfig) (map[string]FileInfo, error) +} + +// VerifyBundleSignature will retrieve the Verifier implementation based +// on the Plugin specified in SignaturesConfig, and call its implementation +// of VerifyBundleSignature. VerifyBundleSignature verifies the bundle signature +// using the given public keys or secret. If a signature is verified, it keeps +// track of the files specified in the JWT payload +func VerifyBundleSignature(sc SignaturesConfig, bvc *VerificationConfig) (map[string]FileInfo, error) { + // default implementation does not return a nil for map, so don't + // do it here either + files := make(map[string]FileInfo) + var plugin string + // for backwards compatibility, check if there is no plugin specified, and use default + if sc.Plugin == "" { + plugin = defaultVerifierID + } else { + plugin = sc.Plugin + } + verifier, err := GetVerifier(plugin) + if err != nil { + return files, err + } + return verifier.VerifyBundleSignature(sc, bvc) +} + +// DefaultVerifier is the default bundle verification implementation. It verifies bundles by checking +// the JWT signature using a locally-accessible public key. +type DefaultVerifier struct{} + +// VerifyBundleSignature verifies the bundle signature using the given public keys or secret. +// If a signature is verified, it keeps track of the files specified in the JWT payload +func (*DefaultVerifier) VerifyBundleSignature(sc SignaturesConfig, bvc *VerificationConfig) (map[string]FileInfo, error) { + files := make(map[string]FileInfo) + + if len(sc.Signatures) == 0 { + return files, fmt.Errorf(".signatures.json: missing JWT (expected exactly one)") + } + + if len(sc.Signatures) > 1 { + return files, fmt.Errorf(".signatures.json: multiple JWTs not supported (expected exactly one)") + } + + for _, token := range sc.Signatures { + payload, err := verifyJWTSignature(token, bvc) + if err != nil { + return files, err + } + + for _, file := range payload.Files { + files[file.Name] = file + } + } + return files, nil +} + +func verifyJWTSignature(token string, bvc *VerificationConfig) (*DecodedSignature, error) { + // decode JWT to check if the header specifies the key to use and/or if claims have the scope. + + parts, err := jws.SplitCompact(token) + if err != nil { + return nil, err + } + + var decodedHeader []byte + if decodedHeader, err = base64.RawURLEncoding.DecodeString(parts[0]); err != nil { + return nil, errors.Wrap(err, "failed to base64 decode JWT headers") + } + + var hdr jws.StandardHeaders + if err := json.Unmarshal(decodedHeader, &hdr); err != nil { + return nil, errors.Wrap(err, "failed to parse JWT headers") + } + + payload, err := base64.RawURLEncoding.DecodeString(parts[1]) + if err != nil { + return nil, err + } + + var ds DecodedSignature + if err := json.Unmarshal(payload, &ds); err != nil { + return nil, err + } + + // check for the id of the key to use for JWT signature verification + // first in the OPA config. If not found, then check the JWT kid. + keyID := bvc.KeyID + if keyID == "" { + keyID = hdr.KeyID + } + if keyID == "" { + // If header has no key id, check the deprecated key claim. + keyID = ds.KeyID + } + + if keyID == "" { + return nil, fmt.Errorf("verification key ID is empty") + } + + // now that we have the keyID, fetch the actual key + keyConfig, err := bvc.GetPublicKey(keyID) + if err != nil { + return nil, err + } + + // verify JWT signature + alg := jwa.SignatureAlgorithm(keyConfig.Algorithm) + key, err := verify.GetSigningKey(keyConfig.Key, alg) + if err != nil { + return nil, err + } + + _, err = jws.Verify([]byte(token), alg, key) + if err != nil { + return nil, err + } + + // verify the scope + scope := bvc.Scope + if scope == "" { + scope = keyConfig.Scope + } + + if ds.Scope != scope { + return nil, fmt.Errorf("scope mismatch") + } + return &ds, nil +} + +// VerifyBundleFile verifies the hash of a file in the bundle matches to that provided in the bundle's signature +func VerifyBundleFile(path string, data bytes.Buffer, files map[string]FileInfo) error { + var file FileInfo + var ok bool + + if file, ok = files[path]; !ok { + return fmt.Errorf("file %v not included in bundle signature", path) + } + + if file.Algorithm == "" { + return fmt.Errorf("no hashing algorithm provided for file %v", path) + } + + hash, err := NewSignatureHasher(HashingAlgorithm(file.Algorithm)) + if err != nil { + return err + } + + // hash the file content + // For unstructured files, hash the byte stream of the file + // For structured files, read the byte stream and parse into a JSON structure; + // then recursively order the fields of all objects alphabetically and then apply + // the hash function to result to compute the hash. This ensures that the digital signature is + // independent of whitespace and other non-semantic JSON features. + var value interface{} + if IsStructuredDoc(path) { + err := util.Unmarshal(data.Bytes(), &value) + if err != nil { + return err + } + } else { + value = data.Bytes() + } + + bs, err := hash.HashFile(value) + if err != nil { + return err + } + + // compare file hash with same file in the JWT payloads + fb, err := hex.DecodeString(file.Hash) + if err != nil { + return err + } + + if !bytes.Equal(fb, bs) { + return fmt.Errorf("%v: digest mismatch (want: %x, got: %x)", path, fb, bs) + } + + delete(files, path) + return nil +} + +// GetVerifier returns the Verifier registered under the given id +func GetVerifier(id string) (Verifier, error) { + verifier, ok := verifiers[id] + if !ok { + return nil, fmt.Errorf("no verifier exists under id %s", id) + } + return verifier, nil +} + +// RegisterVerifier registers a Verifier under the given id +func RegisterVerifier(id string, v Verifier) error { + if id == defaultVerifierID { + return fmt.Errorf("verifier id %s is reserved, use a different id", id) + } + verifiers[id] = v + return nil +} + +func init() { + verifiers = map[string]Verifier{ + defaultVerifierID: &DefaultVerifier{}, + } +} diff --git a/vendor/github.com/open-policy-agent/opa/format/format.go b/vendor/github.com/open-policy-agent/opa/format/format.go new file mode 100644 index 00000000..dbe165d5 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/format/format.go @@ -0,0 +1,1201 @@ +// Copyright 2017 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +// Package format implements formatting of Rego source files. +package format + +import ( + "bytes" + "fmt" + "regexp" + "sort" + + "github.com/open-policy-agent/opa/ast" +) + +// Source formats a Rego source file. The bytes provided must describe a complete +// Rego module. If they don't, Source will return an error resulting from the attempt +// to parse the bytes. +func Source(filename string, src []byte) ([]byte, error) { + module, err := ast.ParseModule(filename, string(src)) + if err != nil { + return nil, err + } + formatted, err := Ast(module) + if err != nil { + return nil, fmt.Errorf("%s: %v", filename, err) + } + return formatted, nil +} + +// MustAst is a helper function to format a Rego AST element. If any errors +// occurs this function will panic. This is mostly used for test +func MustAst(x interface{}) []byte { + bs, err := Ast(x) + if err != nil { + panic(err) + } + return bs +} + +// Ast formats a Rego AST element. If the passed value is not a valid AST +// element, Ast returns nil and an error. If AST nodes are missing locations +// an arbitrary location will be used. +func Ast(x interface{}) ([]byte, error) { + + // The node has to be deep copied because it may be mutated below. Alternatively, + // we could avoid the copy by checking if mutation will occur first. For now, + // since format is not latency sensitive, just deep copy in all cases. + x = ast.Copy(x) + + wildcards := map[ast.Var]*ast.Term{} + + // NOTE(sr): When the formatter encounters a call to internal.member_2 + // or internal.member_3, it will sugarize them into usage of the `in` + // operator. It has to ensure that the proper future keyword import is + // present. + extraFutureKeywordImports := map[string]bool{} + + // Preprocess the AST. Set any required defaults and calculate + // values required for printing the formatted output. + ast.WalkNodes(x, func(x ast.Node) bool { + switch n := x.(type) { + case ast.Body: + if len(n) == 0 { + return false + } + case *ast.Term: + unmangleWildcardVar(wildcards, n) + + case *ast.Expr: + if n.IsCall() && + ast.Member.Ref().Equal(n.Operator()) || + ast.MemberWithKey.Ref().Equal(n.Operator()) { + extraFutureKeywordImports["in"] = true + } + } + if x.Loc() == nil { + x.SetLoc(defaultLocation(x)) + } + return false + }) + + w := &writer{ + indent: "\t", + } + + switch x := x.(type) { + case *ast.Module: + for kw := range extraFutureKeywordImports { + x.Imports = ensureFutureKeywordImport(x.Imports, kw) + } + w.writeModule(x) + case *ast.Package: + w.writePackage(x, nil) + case *ast.Import: + w.writeImports([]*ast.Import{x}, nil) + case *ast.Rule: + w.writeRule(x, false, nil) + case *ast.Head: + w.writeHead(x, false, false, nil) + case ast.Body: + w.writeBody(x, nil) + case *ast.Expr: + w.writeExpr(x, nil) + case *ast.With: + w.writeWith(x, nil) + case *ast.Term: + w.writeTerm(x, nil) + case ast.Value: + w.writeTerm(&ast.Term{Value: x, Location: &ast.Location{}}, nil) + case *ast.Comment: + w.writeComments([]*ast.Comment{x}) + default: + return nil, fmt.Errorf("not an ast element: %v", x) + } + + return squashTrailingNewlines(w.buf.Bytes()), nil +} + +func unmangleWildcardVar(wildcards map[ast.Var]*ast.Term, n *ast.Term) { + + v, ok := n.Value.(ast.Var) + if !ok || !v.IsWildcard() { + return + } + + first, ok := wildcards[v] + if !ok { + wildcards[v] = n + return + } + + w := v[len(ast.WildcardPrefix):] + + // Prepend an underscore to ensure the variable will parse. + if len(w) == 0 || w[0] != '_' { + w = "_" + w + } + + if first != nil { + first.Value = w + wildcards[v] = nil + } + + n.Value = w +} + +func squashTrailingNewlines(bs []byte) []byte { + if bytes.HasSuffix(bs, []byte("\n")) { + return append(bytes.TrimRight(bs, "\n"), '\n') + } + return bs +} + +func defaultLocation(x ast.Node) *ast.Location { + return ast.NewLocation([]byte(x.String()), "", 1, 1) +} + +type writer struct { + buf bytes.Buffer + + indent string + level int + inline bool + beforeEnd *ast.Comment + delay bool +} + +func (w *writer) writeModule(module *ast.Module) { + var pkg *ast.Package + var others []interface{} + var comments []*ast.Comment + visitor := ast.NewGenericVisitor(func(x interface{}) bool { + switch x := x.(type) { + case *ast.Comment: + comments = append(comments, x) + return true + case *ast.Import, *ast.Rule: + others = append(others, x) + return true + case *ast.Package: + pkg = x + return true + default: + return false + } + }) + visitor.Walk(module) + + sort.Slice(comments, func(i, j int) bool { + return locLess(comments[i], comments[j]) + }) + + // XXX: The parser currently duplicates comments for some reason, so we need + // to remove duplicates here. + comments = dedupComments(comments) + sort.Slice(others, func(i, j int) bool { + return locLess(others[i], others[j]) + }) + + comments = w.writePackage(pkg, comments) + var imports []*ast.Import + var rules []*ast.Rule + for len(others) > 0 { + imports, others = gatherImports(others) + comments = w.writeImports(imports, comments) + rules, others = gatherRules(others) + comments = w.writeRules(rules, comments) + } + + for i, c := range comments { + w.writeLine(c.String()) + if i == len(comments)-1 { + w.write("\n") + } + } +} + +func (w *writer) writePackage(pkg *ast.Package, comments []*ast.Comment) []*ast.Comment { + comments = w.insertComments(comments, pkg.Location) + + w.startLine() + w.write(pkg.String()) + w.blankLine() + + return comments +} + +func (w *writer) writeComments(comments []*ast.Comment) { + for i := 0; i < len(comments); i++ { + if i > 0 && locCmp(comments[i], comments[i-1]) > 1 { + w.blankLine() + } + w.writeLine(comments[i].String()) + } +} + +func (w *writer) writeRules(rules []*ast.Rule, comments []*ast.Comment) []*ast.Comment { + for _, rule := range rules { + comments = w.insertComments(comments, rule.Location) + comments = w.writeRule(rule, false, comments) + w.blankLine() + } + return comments +} + +func (w *writer) writeRule(rule *ast.Rule, isElse bool, comments []*ast.Comment) []*ast.Comment { + if rule == nil { + return comments + } + + if !isElse { + w.startLine() + } + + if rule.Default { + w.write("default ") + } + + // OPA transforms lone bodies like `foo = {"a": "b"}` into rules of the form + // `foo = {"a": "b"} { true }` in the AST. We want to preserve that notation + // in the formatted code instead of expanding the bodies into rules, so we + // pretend that the rule has no body in this case. + isExpandedConst := rule.Body.Equal(ast.NewBody(ast.NewExpr(ast.BooleanTerm(true)))) && rule.Else == nil + + comments = w.writeHead(rule.Head, rule.Default, isExpandedConst, comments) + + if (len(rule.Body) == 0 || isExpandedConst) && !isElse { + w.endLine() + return comments + } + + w.write(" {") + w.endLine() + w.up() + + comments = w.writeBody(rule.Body, comments) + + var closeLoc *ast.Location + + if len(rule.Head.Args) > 0 { + closeLoc = closingLoc('(', ')', '{', '}', rule.Location) + } else { + closeLoc = closingLoc('[', ']', '{', '}', rule.Location) + } + + comments = w.insertComments(comments, closeLoc) + + w.down() + w.startLine() + w.write("}") + if rule.Else != nil { + comments = w.writeElse(rule, comments) + } + return comments +} + +func (w *writer) writeElse(rule *ast.Rule, comments []*ast.Comment) []*ast.Comment { + // If there was nothing else on the line before the "else" starts + // then preserve this style of else block, otherwise it will be + // started as an "inline" else eg: + // + // p { + // ... + // } + // + // else { + // ... + // } + // + // versus + // + // p { + // ... + // } else { + // ... + // } + // + // Note: This doesn't use the `close` as it currently isn't accurate for all + // types of values. Checking the actual line text is the most consistent approach. + wasInline := false + ruleLines := bytes.Split(rule.Location.Text, []byte("\n")) + relativeElseRow := rule.Else.Location.Row - rule.Location.Row + if relativeElseRow > 0 && relativeElseRow < len(ruleLines) { + elseLine := ruleLines[relativeElseRow] + if !bytes.HasPrefix(bytes.TrimSpace(elseLine), []byte("else")) { + wasInline = true + } + } + + // If there are any comments between the closing brace of the previous rule and the start + // of the else block we will always insert a new blank line between them. + hasCommentAbove := len(comments) > 0 && comments[0].Location.Row-rule.Else.Head.Location.Row < 0 || w.beforeEnd != nil + + if !hasCommentAbove && wasInline { + w.write(" ") + } else { + w.blankLine() + w.startLine() + } + + rule.Else.Head.Name = "else" + rule.Else.Head.Args = nil + comments = w.insertComments(comments, rule.Else.Head.Location) + + if hasCommentAbove && !wasInline { + // The comments would have ended the line, be sure to start one again + // before writing the rest of the "else" rule. + w.startLine() + } + + // For backwards compatibility adjust the rule head value location + // TODO: Refactor the logic for inserting comments, or special + // case comments in a rule head value so this can be removed + if rule.Else.Head.Value != nil { + rule.Else.Head.Value.Location = rule.Else.Head.Location + } + + return w.writeRule(rule.Else, true, comments) +} + +func (w *writer) writeHead(head *ast.Head, isDefault bool, isExpandedConst bool, comments []*ast.Comment) []*ast.Comment { + w.write(head.Name.String()) + if len(head.Args) > 0 { + w.write("(") + var args []interface{} + for _, arg := range head.Args { + args = append(args, arg) + } + comments = w.writeIterable(args, head.Location, closingLoc(0, 0, '(', ')', head.Location), comments, w.listWriter()) + w.write(")") + } + if head.Key != nil { + w.write("[") + comments = w.writeTerm(head.Key, comments) + w.write("]") + } + if head.Value != nil && (head.Key != nil || ast.Compare(head.Value, ast.BooleanTerm(true)) != 0 || isExpandedConst || isDefault) { + if head.Assign { + w.write(" := ") + } else { + w.write(" = ") + } + comments = w.writeTerm(head.Value, comments) + } + return comments +} + +func (w *writer) insertComments(comments []*ast.Comment, loc *ast.Location) []*ast.Comment { + before, at, comments := partitionComments(comments, loc) + w.writeComments(before) + if len(before) > 0 && loc.Row-before[len(before)-1].Location.Row > 1 { + w.blankLine() + } + + w.beforeLineEnd(at) + return comments +} + +func (w *writer) writeBody(body ast.Body, comments []*ast.Comment) []*ast.Comment { + comments = w.insertComments(comments, body.Loc()) + offset := 0 + for i, expr := range body { + if i > 0 && expr.Location.Row-body[i-1].Location.Row-offset > 1 { + w.blankLine() + } + w.startLine() + + comments = w.writeExpr(expr, comments) + w.endLine() + } + return comments +} + +func (w *writer) writeExpr(expr *ast.Expr, comments []*ast.Comment) []*ast.Comment { + comments = w.insertComments(comments, expr.Location) + if !w.inline { + w.startLine() + } + + if expr.Negated { + w.write("not ") + } + + switch t := expr.Terms.(type) { + case *ast.SomeDecl: + comments = w.writeSomeDecl(t, comments) + case []*ast.Term: + comments = w.writeFunctionCall(expr, comments) + case *ast.Term: + comments = w.writeTerm(t, comments) + } + + var indented bool + for i, with := range expr.With { + if i > 0 && with.Location.Row-expr.With[i-1].Location.Row > 0 { + if !indented { + indented = true + + w.up() + defer w.down() + } + w.endLine() + w.startLine() + } + comments = w.writeWith(with, comments) + } + + return comments +} + +func (w *writer) writeSomeDecl(decl *ast.SomeDecl, comments []*ast.Comment) []*ast.Comment { + comments = w.insertComments(comments, decl.Location) + w.write("some ") + + row := decl.Location.Row + + for i, term := range decl.Symbols { + switch val := term.Value.(type) { + case ast.Var: + if term.Location.Row > row { + w.endLine() + w.startLine() + w.write(w.indent) + row = term.Location.Row + } else if i > 0 { + w.write(" ") + } + + comments = w.writeTerm(term, comments) + + if i < len(decl.Symbols)-1 { + w.write(",") + } + case ast.Call: + comments = w.writeInOperator(false, val[1:], comments) + } + } + + return comments +} + +func (w *writer) writeFunctionCall(expr *ast.Expr, comments []*ast.Comment) []*ast.Comment { + + terms := expr.Terms.([]*ast.Term) + operator := terms[0].Value.String() + + switch operator { + case ast.Member.Name, ast.MemberWithKey.Name: + return w.writeInOperator(false, terms[1:], comments) + } + + bi, ok := ast.BuiltinMap[operator] + if !ok || bi.Infix == "" { + return w.writeFunctionCallPlain(terms, comments) + } + + numDeclArgs := len(bi.Decl.Args()) + numCallArgs := len(terms) - 1 + + switch numCallArgs { + case numDeclArgs: // Print infix where result is unassigned (e.g., x != y) + comments = w.writeTerm(terms[1], comments) + w.write(" " + bi.Infix + " ") + return w.writeTerm(terms[2], comments) + + case numDeclArgs + 1: // Print infix where result is assigned (e.g., z = x + y) + comments = w.writeTerm(terms[3], comments) + w.write(" " + ast.Equality.Infix + " ") + comments = w.writeTerm(terms[1], comments) + w.write(" " + bi.Infix + " ") + comments = w.writeTerm(terms[2], comments) + return comments + } + return w.writeFunctionCallPlain(terms, comments) +} + +func (w *writer) writeFunctionCallPlain(terms []*ast.Term, comments []*ast.Comment) []*ast.Comment { + w.write(terms[0].String() + "(") + defer w.write(")") + args := make([]interface{}, len(terms)-1) + for i, t := range terms[1:] { + args[i] = t + } + loc := terms[0].Location + return w.writeIterable(args, loc, closingLoc(0, 0, '(', ')', loc), comments, w.listWriter()) +} + +func (w *writer) writeWith(with *ast.With, comments []*ast.Comment) []*ast.Comment { + comments = w.insertComments(comments, with.Location) + w.write(" with ") + comments = w.writeTerm(with.Target, comments) + w.write(" as ") + return w.writeTerm(with.Value, comments) +} + +func (w *writer) writeTerm(term *ast.Term, comments []*ast.Comment) []*ast.Comment { + return w.writeTermParens(false, term, comments) +} + +func (w *writer) writeTermParens(parens bool, term *ast.Term, comments []*ast.Comment) []*ast.Comment { + comments = w.insertComments(comments, term.Location) + if !w.inline { + w.startLine() + } + + switch x := term.Value.(type) { + case ast.Ref: + w.writeRef(x) + case ast.Object: + comments = w.writeObject(x, term.Location, comments) + case *ast.Array: + comments = w.writeArray(x, term.Location, comments) + case ast.Set: + comments = w.writeSet(x, term.Location, comments) + case *ast.ArrayComprehension: + comments = w.writeArrayComprehension(x, term.Location, comments) + case *ast.ObjectComprehension: + comments = w.writeObjectComprehension(x, term.Location, comments) + case *ast.SetComprehension: + comments = w.writeSetComprehension(x, term.Location, comments) + case ast.String: + if term.Location.Text[0] == '`' { + // To preserve raw strings, we need to output the original text, + // not what x.String() would give us. + w.write(string(term.Location.Text)) + } else { + w.write(x.String()) + } + case ast.Var: + w.write(w.formatVar(x)) + case ast.Call: + comments = w.writeCall(parens, x, comments) + case fmt.Stringer: + w.write(x.String()) + } + + if !w.inline { + w.startLine() + } + return comments +} + +func (w *writer) writeRef(x ast.Ref) { + if len(x) > 0 { + w.writeTerm(x[0], nil) + path := x[1:] + for _, t := range path { + switch p := t.Value.(type) { + case ast.String: + w.writeRefStringPath(p) + case ast.Var: + w.writeBracketed(w.formatVar(p)) + default: + w.write("[") + w.writeTerm(t, nil) + w.write("]") + } + } + } +} + +func (w *writer) writeBracketed(str string) { + w.write("[" + str + "]") +} + +var varRegexp = regexp.MustCompile("^[[:alpha:]_][[:alpha:][:digit:]_]*$") + +func (w *writer) writeRefStringPath(s ast.String) { + str := string(s) + if varRegexp.MatchString(str) && !ast.IsKeyword(str) { + w.write("." + str) + } else { + w.writeBracketed(s.String()) + } +} + +func (w *writer) formatVar(v ast.Var) string { + if v.IsWildcard() { + return ast.Wildcard.String() + } + return v.String() +} + +func (w *writer) writeCall(parens bool, x ast.Call, comments []*ast.Comment) []*ast.Comment { + bi, ok := ast.BuiltinMap[x[0].String()] + if !ok || bi.Infix == "" { + return w.writeFunctionCallPlain(x, comments) + } + + if bi.Infix == "in" { + // NOTE(sr): `in` requires special handling, mirroring what happens in the parser, + // since there can be one or two lhs arguments. + return w.writeInOperator(true, x[1:], comments) + } + + // TODO(tsandall): improve to consider precedence? + if parens { + w.write("(") + } + comments = w.writeTermParens(true, x[1], comments) + w.write(" " + bi.Infix + " ") + comments = w.writeTermParens(true, x[2], comments) + if parens { + w.write(")") + } + + return comments +} + +func (w *writer) writeInOperator(parens bool, operands []*ast.Term, comments []*ast.Comment) []*ast.Comment { + kw := "in" + switch len(operands) { + case 2: + comments = w.writeTermParens(true, operands[0], comments) + w.write(" ") + w.write(kw) + w.write(" ") + comments = w.writeTermParens(true, operands[1], comments) + case 3: + if parens { + w.write("(") + defer w.write(")") + } + comments = w.writeTermParens(true, operands[0], comments) + w.write(", ") + comments = w.writeTermParens(true, operands[1], comments) + w.write(" ") + w.write(kw) + w.write(" ") + comments = w.writeTermParens(true, operands[2], comments) + } + return comments +} + +func (w *writer) writeObject(obj ast.Object, loc *ast.Location, comments []*ast.Comment) []*ast.Comment { + w.write("{") + defer w.write("}") + + var s []interface{} + obj.Foreach(func(k, v *ast.Term) { + s = append(s, ast.Item(k, v)) + }) + return w.writeIterable(s, loc, closingLoc(0, 0, '{', '}', loc), comments, w.objectWriter()) +} + +func (w *writer) writeArray(arr *ast.Array, loc *ast.Location, comments []*ast.Comment) []*ast.Comment { + w.write("[") + defer w.write("]") + + var s []interface{} + arr.Foreach(func(t *ast.Term) { + s = append(s, t) + }) + return w.writeIterable(s, loc, closingLoc(0, 0, '[', ']', loc), comments, w.listWriter()) +} + +func (w *writer) writeSet(set ast.Set, loc *ast.Location, comments []*ast.Comment) []*ast.Comment { + + if set.Len() == 0 { + w.write("set()") + return w.insertComments(comments, closingLoc(0, 0, '(', ')', loc)) + } + + w.write("{") + defer w.write("}") + + var s []interface{} + set.Foreach(func(t *ast.Term) { + s = append(s, t) + }) + return w.writeIterable(s, loc, closingLoc(0, 0, '{', '}', loc), comments, w.listWriter()) +} + +func (w *writer) writeArrayComprehension(arr *ast.ArrayComprehension, loc *ast.Location, comments []*ast.Comment) []*ast.Comment { + w.write("[") + defer w.write("]") + + return w.writeComprehension('[', ']', arr.Term, arr.Body, loc, comments) +} + +func (w *writer) writeSetComprehension(set *ast.SetComprehension, loc *ast.Location, comments []*ast.Comment) []*ast.Comment { + w.write("{") + defer w.write("}") + + return w.writeComprehension('{', '}', set.Term, set.Body, loc, comments) +} + +func (w *writer) writeObjectComprehension(object *ast.ObjectComprehension, loc *ast.Location, comments []*ast.Comment) []*ast.Comment { + w.write("{") + defer w.write("}") + + object.Value.Location = object.Key.Location // Ensure the value is not written on the next line. + if object.Key.Location.Row-loc.Row > 1 { + w.endLine() + w.startLine() + } + + comments = w.writeTerm(object.Key, comments) + w.write(": ") + return w.writeComprehension('{', '}', object.Value, object.Body, loc, comments) +} + +func (w *writer) writeComprehension(open, close byte, term *ast.Term, body ast.Body, loc *ast.Location, comments []*ast.Comment) []*ast.Comment { + if term.Location.Row-loc.Row > 1 { + w.endLine() + w.startLine() + } + + comments = w.writeTerm(term, comments) + w.write(" |") + + return w.writeComprehensionBody(open, close, body, term.Location, loc, comments) +} + +func (w *writer) writeComprehensionBody(open, close byte, body ast.Body, term, compr *ast.Location, comments []*ast.Comment) []*ast.Comment { + var exprs []interface{} + for _, expr := range body { + exprs = append(exprs, expr) + } + lines := groupIterable(exprs, term) + + if body.Loc().Row-term.Row > 0 || len(lines) > 1 { + w.endLine() + w.up() + defer w.startLine() + defer w.down() + + comments = w.writeBody(body, comments) + } else { + w.write(" ") + i := 0 + for ; i < len(body)-1; i++ { + comments = w.writeExpr(body[i], comments) + w.write("; ") + } + comments = w.writeExpr(body[i], comments) + } + + return w.insertComments(comments, closingLoc(0, 0, open, close, compr)) +} + +func (w *writer) writeImports(imports []*ast.Import, comments []*ast.Comment) []*ast.Comment { + m, comments := mapImportsToComments(imports, comments) + + groups := groupImports(imports) + for _, group := range groups { + comments = w.insertComments(comments, group[0].Loc()) + + // Sort imports within a newline grouping. + sort.Slice(group, func(i, j int) bool { + a := group[i] + b := group[j] + return a.Compare(b) < 0 + }) + for _, i := range group { + w.startLine() + w.write(i.String()) + if c, ok := m[i]; ok { + w.write(" " + c.String()) + } + w.endLine() + } + w.blankLine() + } + + return comments +} + +type entryWriter func(interface{}, []*ast.Comment) []*ast.Comment + +func (w *writer) writeIterable(elements []interface{}, last *ast.Location, close *ast.Location, comments []*ast.Comment, fn entryWriter) []*ast.Comment { + lines := groupIterable(elements, last) + if len(lines) > 1 { + w.delayBeforeEnd() + w.startMultilineSeq() + } + + i := 0 + for ; i < len(lines)-1; i++ { + comments = w.writeIterableLine(lines[i], comments, fn) + w.write(",") + + w.endLine() + w.startLine() + } + + comments = w.writeIterableLine(lines[i], comments, fn) + + if len(lines) > 1 { + w.write(",") + w.endLine() + comments = w.insertComments(comments, close) + w.down() + w.startLine() + } + + return comments +} + +func (w *writer) writeIterableLine(elements []interface{}, comments []*ast.Comment, fn entryWriter) []*ast.Comment { + if len(elements) == 0 { + return comments + } + + i := 0 + for ; i < len(elements)-1; i++ { + comments = fn(elements[i], comments) + w.write(", ") + } + + return fn(elements[i], comments) +} + +func (w *writer) objectWriter() entryWriter { + return func(x interface{}, comments []*ast.Comment) []*ast.Comment { + entry := x.([2]*ast.Term) + comments = w.writeTerm(entry[0], comments) + w.write(": ") + return w.writeTerm(entry[1], comments) + } +} + +func (w *writer) listWriter() entryWriter { + return func(x interface{}, comments []*ast.Comment) []*ast.Comment { + return w.writeTerm(x.(*ast.Term), comments) + } +} + +// groupIterable will group the `elements` slice into slices according to their +// location: anything on the same line will be put into a slice. +func groupIterable(elements []interface{}, last *ast.Location) [][]interface{} { + // Generated vars occur in the AST when we're rendering the result of + // partial evaluation in a bundle build with optimization. For those vars, + // there is no location, and the grouping based on source location will + // yield a bad result. So if there's a generated variable among elements, + // we'll render the elements all in one line. + vis := ast.NewVarVisitor() + for _, elem := range elements { + vis.Walk(elem) + } + for v := range vis.Vars() { + if v.IsGenerated() { + return [][]interface{}{elements} + } + } + sort.Slice(elements, func(i, j int) bool { + return locLess(elements[i], elements[j]) + }) + var lines [][]interface{} + var cur []interface{} + for i, t := range elements { + elem := t + loc := getLoc(elem) + lineDiff := loc.Row - last.Row + if lineDiff > 0 && i > 0 { + lines = append(lines, cur) + cur = nil + } + + last = loc + cur = append(cur, elem) + } + return append(lines, cur) +} + +func mapImportsToComments(imports []*ast.Import, comments []*ast.Comment) (map[*ast.Import]*ast.Comment, []*ast.Comment) { + var leftovers []*ast.Comment + m := map[*ast.Import]*ast.Comment{} + + for _, c := range comments { + matched := false + for _, i := range imports { + if c.Loc().Row == i.Loc().Row { + m[i] = c + matched = true + break + } + } + if !matched { + leftovers = append(leftovers, c) + } + } + + return m, leftovers +} + +func groupImports(imports []*ast.Import) [][]*ast.Import { + switch len(imports) { // shortcuts + case 0: + return nil + case 1: + return [][]*ast.Import{imports} + } + // there are >=2 imports to group + + var groups [][]*ast.Import + group := []*ast.Import{imports[0]} + + for _, i := range imports[1:] { + last := group[len(group)-1] + + // nil-location imports have been sorted up to come first + if i.Loc() != nil && last.Loc() != nil && // first import with a location, or + i.Loc().Row-last.Loc().Row > 1 { // more than one row apart from previous import + + // start a new group + groups = append(groups, group) + group = []*ast.Import{} + } + group = append(group, i) + } + if len(group) > 0 { + groups = append(groups, group) + } + + return groups +} + +func partitionComments(comments []*ast.Comment, l *ast.Location) (before []*ast.Comment, at *ast.Comment, after []*ast.Comment) { + for _, c := range comments { + switch cmp := c.Location.Row - l.Row; { + case cmp < 0: + before = append(before, c) + case cmp > 0: + after = append(after, c) + case cmp == 0: + at = c + } + } + + return before, at, after +} + +func gatherImports(others []interface{}) (imports []*ast.Import, rest []interface{}) { + i := 0 +loop: + for ; i < len(others); i++ { + switch x := others[i].(type) { + case *ast.Import: + imports = append(imports, x) + case *ast.Rule: + break loop + } + } + return imports, others[i:] +} + +func gatherRules(others []interface{}) (rules []*ast.Rule, rest []interface{}) { + i := 0 +loop: + for ; i < len(others); i++ { + switch x := others[i].(type) { + case *ast.Rule: + rules = append(rules, x) + case *ast.Import: + break loop + } + } + return rules, others[i:] +} + +func locLess(a, b interface{}) bool { + return locCmp(a, b) < 0 +} + +func locCmp(a, b interface{}) int { + al := getLoc(a) + bl := getLoc(b) + switch { + case al == nil && bl == nil: + return 0 + case al == nil: + return -1 + case bl == nil: + return 1 + } + + if cmp := al.Row - bl.Row; cmp != 0 { + return cmp + + } + return al.Col - bl.Col +} + +func getLoc(x interface{}) *ast.Location { + switch x := x.(type) { + case ast.Node: // *ast.Head, *ast.Expr, *ast.With, *ast.Term + return x.Loc() + case *ast.Location: + return x + case [2]*ast.Term: // Special case to allow for easy printing of objects. + return x[0].Location + default: + panic("Not reached") + } +} + +func closingLoc(skipOpen, skipClose, open, close byte, loc *ast.Location) *ast.Location { + i, offset := 0, 0 + + // Skip past parens/brackets/braces in rule heads. + if skipOpen > 0 { + i, offset = skipPast(skipOpen, skipClose, loc) + } + + for ; i < len(loc.Text) && loc.Text[i] != open; i++ { + } + + if i >= len(loc.Text) { + return &ast.Location{Row: -1} + } + + state := 1 + for state > 0 { + i++ + if i >= len(loc.Text) { + return &ast.Location{Row: -1} + } + + switch loc.Text[i] { + case open: + state++ + case close: + state-- + case '\n': + offset++ + } + } + + return &ast.Location{Row: loc.Row + offset} +} + +func skipPast(open, close byte, loc *ast.Location) (int, int) { + i := 0 + for ; i < len(loc.Text) && loc.Text[i] != open; i++ { + } + + state := 1 + offset := 0 + for state > 0 { + i++ + if i >= len(loc.Text) { + return i, offset + } + + switch loc.Text[i] { + case open: + state++ + case close: + state-- + case '\n': + offset++ + } + } + + return i, offset +} + +func dedupComments(comments []*ast.Comment) []*ast.Comment { + if len(comments) == 0 { + return nil + } + + filtered := []*ast.Comment{comments[0]} + for i := 1; i < len(comments); i++ { + if comments[i].Location.Equal(comments[i-1].Location) { + continue + } + filtered = append(filtered, comments[i]) + } + return filtered +} + +// startLine begins a line with the current indentation level. +func (w *writer) startLine() { + w.inline = true + for i := 0; i < w.level; i++ { + w.write(w.indent) + } +} + +// endLine ends a line with a newline. +func (w *writer) endLine() { + w.inline = false + if w.beforeEnd != nil && !w.delay { + w.write(" " + w.beforeEnd.String()) + w.beforeEnd = nil + } + w.delay = false + w.write("\n") +} + +// beforeLineEnd registers a comment to be printed at the end of the current line. +func (w *writer) beforeLineEnd(c *ast.Comment) { + if w.beforeEnd != nil { + if c == nil { + return + } + panic("overwriting non-nil beforeEnd") + } + w.beforeEnd = c +} + +func (w *writer) delayBeforeEnd() { + w.delay = true +} + +// line prints a blank line. If the writer is currently in the middle of a line, +// line ends it and then prints a blank one. +func (w *writer) blankLine() { + if w.inline { + w.endLine() + } + w.write("\n") +} + +// write the input string and writes it to the buffer. +func (w *writer) write(s string) { + w.buf.WriteString(s) +} + +// writeLine writes the string on a newly started line, then terminate the line. +func (w *writer) writeLine(s string) { + if !w.inline { + w.startLine() + } + w.write(s) + w.endLine() +} + +func (w *writer) startMultilineSeq() { + w.endLine() + w.up() + w.startLine() +} + +// up increases the indentation level +func (w *writer) up() { + w.level++ +} + +// down decreases the indentation level +func (w *writer) down() { + if w.level == 0 { + panic("negative indentation level") + } + w.level-- +} + +func ensureFutureKeywordImport(imps []*ast.Import, kw string) []*ast.Import { + allKeywords := ast.MustParseTerm("future.keywords") + kwPath := ast.MustParseTerm("future.keywords." + kw) + for _, imp := range imps { + if allKeywords.Equal(imp.Path) || imp.Path.Equal(kwPath) { + return imps + } + } + return append(imps, &ast.Import{Path: kwPath}) +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/bundle/utils.go b/vendor/github.com/open-policy-agent/opa/internal/bundle/utils.go new file mode 100644 index 00000000..0e0775b1 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/bundle/utils.go @@ -0,0 +1,85 @@ +// Copyright 2020 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package bundle + +import ( + "context" + "fmt" + + "github.com/open-policy-agent/opa/ast" + "github.com/open-policy-agent/opa/bundle" + "github.com/open-policy-agent/opa/resolver/wasm" + "github.com/open-policy-agent/opa/storage" +) + +// LoadWasmResolversFromStore will lookup all Wasm modules from the store along with the +// associated bundle manifest configuration and instantiate the respective resolvers. +func LoadWasmResolversFromStore(ctx context.Context, store storage.Store, txn storage.Transaction, otherBundles map[string]*bundle.Bundle) ([]*wasm.Resolver, error) { + bundleNames, err := bundle.ReadBundleNamesFromStore(ctx, store, txn) + if err != nil && !storage.IsNotFound(err) { + return nil, err + } + + var resolversToLoad []*bundle.WasmModuleFile + for _, bundleName := range bundleNames { + var wasmResolverConfigs []bundle.WasmResolver + rawModules := map[string][]byte{} + + // Save round-tripping the bundle that was just activated + if _, ok := otherBundles[bundleName]; ok { + wasmResolverConfigs = otherBundles[bundleName].Manifest.WasmResolvers + for _, wmf := range otherBundles[bundleName].WasmModules { + rawModules[wmf.Path] = wmf.Raw + } + } else { + wasmResolverConfigs, err = bundle.ReadWasmMetadataFromStore(ctx, store, txn, bundleName) + if err != nil && !storage.IsNotFound(err) { + return nil, fmt.Errorf("failed to read wasm module manifest from store: %s", err) + } + rawModules, err = bundle.ReadWasmModulesFromStore(ctx, store, txn, bundleName) + if err != nil && !storage.IsNotFound(err) { + return nil, fmt.Errorf("failed to read wasm modules from store: %s", err) + } + } + + for path, raw := range rawModules { + wmf := &bundle.WasmModuleFile{ + URL: path, + Path: path, + Raw: raw, + } + for _, resolverConf := range wasmResolverConfigs { + if resolverConf.Module == path { + ref, err := ast.PtrRef(ast.DefaultRootDocument, resolverConf.Entrypoint) + if err != nil { + return nil, fmt.Errorf("failed to parse wasm module entrypoint '%s': %s", resolverConf.Entrypoint, err) + } + wmf.Entrypoints = append(wmf.Entrypoints, ref) + } + } + if len(wmf.Entrypoints) > 0 { + resolversToLoad = append(resolversToLoad, wmf) + } + } + } + + var resolvers []*wasm.Resolver + if len(resolversToLoad) > 0 { + // Get a full snapshot of the current data (including any from "outside" the bundles) + data, err := store.Read(ctx, txn, storage.Path{}) + if err != nil { + return nil, fmt.Errorf("failed to initialize wasm runtime: %s", err) + } + + for _, wmf := range resolversToLoad { + resolver, err := wasm.New(wmf.Entrypoints, wmf.Raw, data) + if err != nil { + return nil, fmt.Errorf("failed to initialize wasm module for entrypoints '%s': %s", wmf.Entrypoints, err) + } + resolvers = append(resolvers, resolver) + } + } + return resolvers, nil +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/cidr/merge/merge.go b/vendor/github.com/open-policy-agent/opa/internal/cidr/merge/merge.go new file mode 100644 index 00000000..a019cde1 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/cidr/merge/merge.go @@ -0,0 +1,367 @@ +// Copyright 2017-2020 Authors of Cilium +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. + +// Package merge provides helper functions for merging a list of +// IP addresses and subnets into the smallest possible list of CIDRs. +// Original Implementation: https://github.com/cilium/cilium +package merge + +import ( + "bytes" + "encoding/binary" + "math/big" + "net" +) + +const ( + ipv4BitLen = 8 * net.IPv4len + ipv6BitLen = 8 * net.IPv6len +) + +var ( + v4Mappedv6Prefix = []byte{0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0xff, 0xff} + defaultIPv4 = []byte{0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0xff, 0xff, 0x0, 0x0, 0x0, 0x0} + defaultIPv6 = []byte{0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0} + upperIPv4 = []byte{0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0xff, 0xff, 255, 255, 255, 255} + upperIPv6 = []byte{0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff, 0xff} + ipv4LeadingZeroes = []byte{0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0, 0x0} +) + +// RangeToCIDRs converts the range of IPs covered by firstIP and lastIP to +// a list of CIDRs that contains all of the IPs covered by the range. +func RangeToCIDRs(firstIP, lastIP net.IP) []*net.IPNet { + // First, create a CIDR that spans both IPs. + spanningCIDR := createSpanningCIDR(&firstIP, &lastIP) + firstIPSpanning, lastIPSpanning := GetAddressRange(spanningCIDR) + + cidrList := []*net.IPNet{} + + // If the first IP of the spanning CIDR passes the lower bound (firstIP), + // we need to split the spanning CIDR and only take the IPs that are + // greater than the value which we split on, as we do not want the lesser + // values since they are less than the lower-bound (firstIP). + if bytes.Compare(firstIPSpanning, firstIP) < 0 { + // Split on the previous IP of the first IP so that the right list of IPs + // of the partition includes the firstIP. + prevFirstRangeIP := GetPreviousIP(firstIP) + var bitLen int + if prevFirstRangeIP.To4() != nil { + bitLen = ipv4BitLen + } else { + bitLen = ipv6BitLen + } + _, _, right := partitionCIDR(spanningCIDR, net.IPNet{IP: prevFirstRangeIP, Mask: net.CIDRMask(bitLen, bitLen)}) + + // Append all CIDRs but the first, as this CIDR includes the upper + // bound of the spanning CIDR, which we still need to partition on. + cidrList = append(cidrList, right...) + spanningCIDR = *right[0] + cidrList = cidrList[1:] + } + + // Conversely, if the last IP of the spanning CIDR passes the upper bound + // (lastIP), we need to split the spanning CIDR and only take the IPs that + // are greater than the value which we split on, as we do not want the greater + // values since they are greater than the upper-bound (lastIP). + if bytes.Compare(lastIPSpanning, lastIP) > 0 { + // Split on the next IP of the last IP so that the left list of IPs + // of the partition include the lastIP. + nextFirstRangeIP := getNextIP(lastIP) + var bitLen int + if nextFirstRangeIP.To4() != nil { + bitLen = ipv4BitLen + } else { + bitLen = ipv6BitLen + } + left, _, _ := partitionCIDR(spanningCIDR, net.IPNet{IP: nextFirstRangeIP, Mask: net.CIDRMask(bitLen, bitLen)}) + cidrList = append(cidrList, left...) + } else { + // Otherwise, there is no need to partition; just use add the spanning + // CIDR to the list of networks. + cidrList = append(cidrList, &spanningCIDR) + } + return cidrList +} + +// GetAddressRange returns the first and last addresses in the given CIDR range. +func GetAddressRange(ipNet net.IPNet) (net.IP, net.IP) { + firstIP := make(net.IP, len(ipNet.IP)) + lastIP := make(net.IP, len(ipNet.IP)) + + copy(firstIP, ipNet.IP) + copy(lastIP, ipNet.IP) + + firstIP = firstIP.Mask(ipNet.Mask) + lastIP = lastIP.Mask(ipNet.Mask) + + if firstIP.To4() != nil { + firstIP = append(v4Mappedv6Prefix, firstIP...) + lastIP = append(v4Mappedv6Prefix, lastIP...) + } + + lastIPMask := make(net.IPMask, len(ipNet.Mask)) + copy(lastIPMask, ipNet.Mask) + for i := range lastIPMask { + lastIPMask[len(lastIPMask)-i-1] = ^lastIPMask[len(lastIPMask)-i-1] + lastIP[net.IPv6len-i-1] = lastIP[net.IPv6len-i-1] | lastIPMask[len(lastIPMask)-i-1] + } + + return firstIP, lastIP +} + +// GetPreviousIP returns the previous IP from the given IP address. +func GetPreviousIP(ip net.IP) net.IP { + // Cannot go lower than zero! + if ip.Equal(net.IP(defaultIPv4)) || ip.Equal(net.IP(defaultIPv6)) { + return ip + } + + previousIP := make(net.IP, len(ip)) + copy(previousIP, ip) + + var overflow bool + var lowerByteBound int + if ip.To4() != nil { + lowerByteBound = net.IPv6len - net.IPv4len + } else { + lowerByteBound = 0 + } + for i := len(ip) - 1; i >= lowerByteBound; i-- { + if overflow || i == len(ip)-1 { + previousIP[i]-- + } + // Track if we have overflowed and thus need to continue subtracting. + if ip[i] == 0 && previousIP[i] == 255 { + overflow = true + } else { + overflow = false + } + } + return previousIP +} + +// createSpanningCIDR returns a single IP network spanning the +// the lower and upper bound IP addresses. +func createSpanningCIDR(firstIP, lastIP *net.IP) net.IPNet { + // Don't want to modify the values of the provided range, so make copies. + lowest := *firstIP + highest := *lastIP + + var isIPv4 bool + var spanningMaskSize, bitLen, byteLen int + if lowest.To4() != nil { + isIPv4 = true + bitLen = ipv4BitLen + byteLen = net.IPv4len + } else { + bitLen = ipv6BitLen + byteLen = net.IPv6len + } + + if isIPv4 { + spanningMaskSize = ipv4BitLen + } else { + spanningMaskSize = ipv6BitLen + } + + // Convert to big Int so we can easily do bitshifting on the IP addresses, + // since golang only provides up to 64-bit unsigned integers. + lowestBig := big.NewInt(0).SetBytes(lowest) + highestBig := big.NewInt(0).SetBytes(highest) + + // Starting from largest mask / smallest range possible, apply a mask one bit + // larger in each iteration to the upper bound in the range until we have + // masked enough to pass the lower bound in the range. This + // gives us the size of the prefix for the spanning CIDR to return as + // well as the IP for the CIDR prefix of the spanning CIDR. + for spanningMaskSize > 0 && lowestBig.Cmp(highestBig) < 0 { + spanningMaskSize-- + mask := big.NewInt(1) + mask = mask.Lsh(mask, uint(bitLen-spanningMaskSize)) + mask = mask.Mul(mask, big.NewInt(-1)) + highestBig = highestBig.And(highestBig, mask) + } + + // If ipv4, need to append 0s because math.Big gets rid of preceding zeroes. + if isIPv4 { + highest = append(ipv4LeadingZeroes, highestBig.Bytes()...) + } else { + highest = highestBig.Bytes() + } + + // Int does not store leading zeroes. + if len(highest) == 0 { + highest = make([]byte, byteLen) + } + + newNet := net.IPNet{IP: highest, Mask: net.CIDRMask(spanningMaskSize, bitLen)} + return newNet +} + +// partitionCIDR returns a list of IP Networks partitioned upon excludeCIDR. +// The first list contains the networks to the left of the excludeCIDR in the +// partition, the second is a list containing the excludeCIDR itself if it is +// contained within the targetCIDR (nil otherwise), and the +// third is a list containing the networks to the right of the excludeCIDR in +// the partition. +func partitionCIDR(targetCIDR net.IPNet, excludeCIDR net.IPNet) ([]*net.IPNet, []*net.IPNet, []*net.IPNet) { + var targetIsIPv4 bool + if targetCIDR.IP.To4() != nil { + targetIsIPv4 = true + } + + targetFirstIP, targetLastIP := GetAddressRange(targetCIDR) + excludeFirstIP, excludeLastIP := GetAddressRange(excludeCIDR) + + targetMaskSize, _ := targetCIDR.Mask.Size() + excludeMaskSize, _ := excludeCIDR.Mask.Size() + + if bytes.Compare(excludeLastIP, targetFirstIP) < 0 { + return nil, nil, []*net.IPNet{&targetCIDR} + } else if bytes.Compare(targetLastIP, excludeFirstIP) < 0 { + return []*net.IPNet{&targetCIDR}, nil, nil + } + + if targetMaskSize >= excludeMaskSize { + return nil, []*net.IPNet{&targetCIDR}, nil + } + + left := []*net.IPNet{} + right := []*net.IPNet{} + + newPrefixLen := targetMaskSize + 1 + + targetFirstCopy := make(net.IP, len(targetFirstIP)) + copy(targetFirstCopy, targetFirstIP) + + iLowerOld := make(net.IP, len(targetFirstCopy)) + copy(iLowerOld, targetFirstCopy) + + // Since golang only supports up to unsigned 64-bit integers, and we need + // to perform addition on addresses, use math/big library, which allows + // for manipulation of large integers. + + // Used to track the current lower and upper bounds of the ranges to compare + // to excludeCIDR. + iLower := big.NewInt(0) + iUpper := big.NewInt(0) + iLower = iLower.SetBytes(targetFirstCopy) + + var bitLen int + + if targetIsIPv4 { + bitLen = ipv4BitLen + } else { + bitLen = ipv6BitLen + } + shiftAmount := (uint)(bitLen - newPrefixLen) + + targetIPInt := big.NewInt(0) + targetIPInt.SetBytes(targetFirstIP.To16()) + + exp := big.NewInt(0) + + // Use left shift for exponentiation + exp = exp.Lsh(big.NewInt(1), shiftAmount) + iUpper = iUpper.Add(targetIPInt, exp) + + matched := big.NewInt(0) + + for excludeMaskSize >= newPrefixLen { + // Append leading zeros to IPv4 addresses, as math.Big.Int does not + // append them when the IP address is copied from a byte array to + // math.Big.Int. Leading zeroes are required for parsing IPv4 addresses + // for use with net.IP / net.IPNet. + var iUpperBytes, iLowerBytes []byte + if targetIsIPv4 { + iUpperBytes = append(ipv4LeadingZeroes, iUpper.Bytes()...) + iLowerBytes = append(ipv4LeadingZeroes, iLower.Bytes()...) + } else { + iUpperBytesLen := len(iUpper.Bytes()) + // Make sure that the number of bytes in the array matches what net + // package expects, as big package doesn't append leading zeroes. + if iUpperBytesLen != net.IPv6len { + numZeroesToAppend := net.IPv6len - iUpperBytesLen + zeroBytes := make([]byte, numZeroesToAppend) + iUpperBytes = append(zeroBytes, iUpper.Bytes()...) + } else { + iUpperBytes = iUpper.Bytes() + + } + + iLowerBytesLen := len(iLower.Bytes()) + if iLowerBytesLen != net.IPv6len { + numZeroesToAppend := net.IPv6len - iLowerBytesLen + zeroBytes := make([]byte, numZeroesToAppend) + iLowerBytes = append(zeroBytes, iLower.Bytes()...) + } else { + iLowerBytes = iLower.Bytes() + + } + } + // If the IP we are excluding over is of a higher value than the current + // CIDR prefix we are generating, add the CIDR prefix to the set of IPs + // to the left of the exclude CIDR + if bytes.Compare(excludeFirstIP, iUpperBytes) >= 0 { + left = append(left, &net.IPNet{IP: iLowerBytes, Mask: net.CIDRMask(newPrefixLen, bitLen)}) + matched = matched.Set(iUpper) + } else { + // Same as above, but opposite. + right = append(right, &net.IPNet{IP: iUpperBytes, Mask: net.CIDRMask(newPrefixLen, bitLen)}) + matched = matched.Set(iLower) + } + + newPrefixLen++ + + if newPrefixLen > bitLen { + break + } + + iLower = iLower.Set(matched) + iUpper = iUpper.Add(matched, big.NewInt(0).Lsh(big.NewInt(1), uint(bitLen-newPrefixLen))) + + } + excludeList := []*net.IPNet{&excludeCIDR} + + return left, excludeList, right +} + +func getNextIP(ip net.IP) net.IP { + if ip.Equal(upperIPv4) || ip.Equal(upperIPv6) { + return ip + } + + nextIP := make(net.IP, len(ip)) + switch len(ip) { + case net.IPv4len: + ipU32 := binary.BigEndian.Uint32(ip) + ipU32++ + binary.BigEndian.PutUint32(nextIP, ipU32) + return nextIP + case net.IPv6len: + ipU64 := binary.BigEndian.Uint64(ip[net.IPv6len/2:]) + ipU64++ + binary.BigEndian.PutUint64(nextIP[net.IPv6len/2:], ipU64) + if ipU64 == 0 { + ipU64 = binary.BigEndian.Uint64(ip[:net.IPv6len/2]) + ipU64++ + binary.BigEndian.PutUint64(nextIP[:net.IPv6len/2], ipU64) + } else { + copy(nextIP[:net.IPv6len/2], ip[:net.IPv6len/2]) + } + return nextIP + default: + return ip + } +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/compiler/wasm/opa/callgraph.csv b/vendor/github.com/open-policy-agent/opa/internal/compiler/wasm/opa/callgraph.csv new file mode 100644 index 00000000..2af55db7 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/compiler/wasm/opa/callgraph.csv @@ -0,0 +1,2401 @@ +opa_agg_count,opa_value_type +opa_agg_count,chartorune +opa_agg_count,opa_number_int +opa_agg_sum,opa_value_type +opa_agg_sum,mpd_qnew +opa_agg_sum,mpd_max_ctx +opa_agg_sum,mpd_qset_i32 +opa_agg_sum,opa_abort +opa_agg_sum,mpd_del +opa_agg_sum,opa_number_to_bf +opa_agg_sum,qadd +opa_agg_sum,opa_bf_to_number +opa_agg_product,opa_value_type +opa_agg_product,mpd_qnew +opa_agg_product,mpd_max_ctx +opa_agg_product,mpd_qset_i32 +opa_agg_product,opa_abort +opa_agg_product,mpd_del +opa_agg_product,opa_number_to_bf +opa_agg_product,qmul +opa_agg_product,opa_bf_to_number +opa_agg_max,opa_value_type +opa_agg_max,opa_value_compare +opa_agg_min,opa_value_type +opa_agg_min,opa_value_compare +opa_agg_sort,opa_value_type +opa_agg_sort,opa_array_with_cap +opa_agg_sort,opa_array_append +opa_agg_sort,opa_value_shallow_copy +opa_agg_sort,opa_array_sort +opa_agg_all,opa_value_type +opa_agg_all,opa_boolean +opa_agg_any,opa_value_type +opa_agg_any,opa_boolean +opa_agg_any,opa_set_get +builtin_member,opa_value_iter +builtin_member,opa_value_get +builtin_member,opa_value_compare +builtin_member,opa_boolean +builtin_member3,opa_value_type +builtin_member3,opa_value_get +builtin_member3,opa_value_compare +builtin_member3,opa_boolean +opa_arith_abs,opa_number_to_bf +opa_arith_abs,mpd_qnew +opa_arith_abs,mpd_max_ctx +opa_arith_abs,mpd_qabs +opa_arith_abs,mpd_del +opa_arith_abs,opa_abort +opa_arith_abs,opa_bf_to_number +opa_arith_round,opa_number_to_bf +opa_arith_round,mpd_qnew +opa_arith_round,mpd_max_ctx +opa_arith_round,mpd_qround_to_int +opa_arith_round,mpd_del +opa_arith_round,opa_abort +opa_arith_round,opa_bf_to_number +opa_arith_ceil,opa_number_to_bf +opa_arith_ceil,mpd_qnew +opa_arith_ceil,mpd_max_ctx +opa_arith_ceil,mpd_qceil +opa_arith_ceil,mpd_del +opa_arith_ceil,opa_bf_to_number +opa_arith_floor,opa_number_to_bf +opa_arith_floor,mpd_qnew +opa_arith_floor,mpd_max_ctx +opa_arith_floor,mpd_qfloor +opa_arith_floor,mpd_del +opa_arith_floor,opa_bf_to_number +opa_arith_plus,opa_number_to_bf +opa_arith_plus,opa_mpd_del +opa_arith_plus,mpd_qnew +opa_arith_plus,mpd_max_ctx +opa_arith_plus,mpd_qadd +opa_arith_plus,mpd_del +opa_arith_plus,opa_abort +opa_arith_plus,opa_bf_to_number +opa_arith_minus,opa_number_to_bf +opa_arith_minus,mpd_qnew +opa_arith_minus,mpd_max_ctx +opa_arith_minus,mpd_qsub +opa_arith_minus,mpd_del +opa_arith_minus,opa_abort +opa_arith_minus,opa_bf_to_number +opa_arith_minus,opa_mpd_del +opa_arith_minus,opa_set_diff +opa_arith_multiply,opa_number_to_bf +opa_arith_multiply,opa_mpd_del +opa_arith_multiply,mpd_qnew +opa_arith_multiply,mpd_max_ctx +opa_arith_multiply,mpd_qmul +opa_arith_multiply,mpd_del +opa_arith_multiply,opa_abort +opa_arith_multiply,opa_bf_to_number +opa_arith_divide,opa_number_to_bf +opa_arith_divide,opa_mpd_del +opa_arith_divide,mpd_qnew +opa_arith_divide,mpd_default_ctx +opa_arith_divide,mpd_qdiv +opa_arith_divide,mpd_del +opa_arith_divide,opa_abort +opa_arith_divide,opa_bf_to_number +opa_arith_rem,opa_number_to_bf +opa_arith_rem,mpd_isinteger +opa_arith_rem,opa_mpd_del +opa_arith_rem,mpd_qnew +opa_arith_rem,mpd_max_ctx +opa_arith_rem,mpd_qrem +opa_arith_rem,mpd_del +opa_arith_rem,opa_bf_to_number +opa_array_concat,opa_value_type +opa_array_concat,opa_array_with_cap +opa_array_concat,opa_array_append +opa_array_slice,opa_value_type +opa_array_slice,opa_number_try_int +opa_array_slice,opa_array_with_cap +opa_array_slice,opa_array_append +opa_array_reverse,opa_value_type +opa_array_reverse,opa_array_with_cap +opa_array_reverse,opa_array_append +opa_bits_or,opa_number_to_bf +opa_bits_or,mpd_isinteger +opa_bits_or,opa_mpd_del +opa_bits_or,mpd_sign +opa_bits_or,qabs +opa_bits_or,qsub_one +opa_bits_or,qand +opa_bits_or,qadd_one +opa_bits_or,qneg +opa_bits_or,opa_bf_to_number +opa_bits_or,qor +opa_bits_or,qand_not +opa_bits_and,opa_number_to_bf +opa_bits_and,mpd_isinteger +opa_bits_and,opa_mpd_del +opa_bits_and,mpd_sign +opa_bits_and,qabs +opa_bits_and,qsub_one +opa_bits_and,qor +opa_bits_and,qadd_one +opa_bits_and,qneg +opa_bits_and,opa_bf_to_number +opa_bits_and,qand +opa_bits_and,qand_not +opa_bits_negate,opa_number_to_bf +opa_bits_negate,mpd_isinteger +opa_bits_negate,mpd_sign +opa_bits_negate,qabs +opa_bits_negate,opa_bf_to_bf_bits +opa_bits_negate,opa_bf_bits_to_bf +opa_bits_negate,qsub_one +opa_bits_negate,qadd_one +opa_bits_negate,qneg +opa_bits_negate,opa_bf_to_number +opa_bits_xor,opa_number_to_bf +opa_bits_xor,mpd_isinteger +opa_bits_xor,opa_mpd_del +opa_bits_xor,mpd_sign +opa_bits_xor,qabs +opa_bits_xor,qsub_one +opa_bits_xor,qxor +opa_bits_xor,opa_bf_to_number +opa_bits_xor,qadd_one +opa_bits_xor,qneg +opa_bits_shiftleft,opa_number_to_bf +opa_bits_shiftleft,opa_bf_to_bf_bits +opa_bits_shiftleft,opa_mpd_del +opa_bits_shiftleft,opa_value_type +opa_bits_shiftleft,opa_number_try_int +opa_bits_shiftleft,mpd_qnew +opa_bits_shiftleft,mpd_max_ctx +opa_bits_shiftleft,mpd_qshiftn +opa_bits_shiftleft,mpd_del +opa_bits_shiftleft,opa_abort +opa_bits_shiftleft,opa_bf_bits_to_bf +opa_bits_shiftleft,opa_bf_to_number +opa_bits_shiftright,opa_number_to_bf +opa_bits_shiftright,mpd_isinteger +opa_bits_shiftright,opa_value_type +opa_bits_shiftright,mpd_del +opa_bits_shiftright,opa_number_try_int +opa_bits_shiftright,mpd_sign +opa_bits_shiftright,qabs +opa_bits_shiftright,qsub_one +opa_bits_shiftright,opa_bf_to_bf_bits +opa_bits_shiftright,mpd_qshiftr_inplace +opa_bits_shiftright,opa_bf_bits_to_bf +opa_bits_shiftright,qadd_one +opa_bits_shiftright,qneg +opa_bits_shiftright,opa_bf_to_number +opa_cidr_contains,opa_value_type +opa_cidr_contains,parse_cidr +opa_cidr_contains,parse_ip +opa_cidr_contains,opa_boolean +parse_cidr,parse_ip +parse_cidr,opa_atoi64 +parse_ip,memchr +opa_cidr_intersects,opa_value_type +opa_cidr_intersects,parse_cidr +opa_cidr_intersects,opa_boolean +opa_cmp_eq,opa_value_compare +opa_cmp_eq,opa_boolean +opa_cmp_neq,opa_value_compare +opa_cmp_neq,opa_boolean +opa_cmp_gt,opa_value_compare +opa_cmp_gt,opa_boolean +opa_cmp_gte,opa_value_compare +opa_cmp_gte,opa_boolean +opa_cmp_lt,opa_value_compare +opa_cmp_lt,opa_boolean +opa_cmp_lte,opa_value_compare +opa_cmp_lte,opa_boolean +opa_eval_ctx_new,opa_malloc +opa_eval,opa_abort +opa_eval,opa_heap_ptr_set +opa_eval,opa_value_parse +opa_eval,eval +opa_eval,opa_value_dump +opa_eval,opa_json_dump +__force_import_opa_builtins,opa_builtin0 +__force_import_opa_builtins,opa_builtin1 +__force_import_opa_builtins,opa_builtin2 +__force_import_opa_builtins,opa_builtin3 +__force_import_opa_builtins,opa_builtin4 +opa_to_number,opa_value_type +opa_to_number,opa_number_int +opa_to_number,opa_atof64 +opa_to_number,opa_number_ref +opa_base64_is_valid,opa_value_type +opa_base64_is_valid,base64_gen_decode +opa_base64_is_valid,free +opa_base64_is_valid,opa_boolean +base64_gen_decode,memset +base64_gen_decode,malloc +base64_gen_decode,free +opa_base64_decode,opa_value_type +opa_base64_decode,base64_gen_decode +opa_base64_decode,opa_string_allocated +opa_base64_encode,opa_value_type +opa_base64_encode,base64_gen_encode +opa_base64_encode,opa_string_allocated +base64_gen_encode,malloc +opa_base64_url_decode,opa_value_type +opa_base64_url_decode,base64_gen_decode +opa_base64_url_decode,opa_string_allocated +opa_base64_url_encode,opa_value_type +opa_base64_url_encode,base64_gen_encode +opa_base64_url_encode,opa_string_allocated +opa_json_unmarshal,opa_value_type +opa_json_unmarshal,opa_json_parse +opa_json_marshal,opa_json_dump +opa_json_marshal,strlen +opa_json_marshal,opa_string_allocated +opa_runtime_error,opa_itoa +opa_runtime_error,opa_strlen +opa_runtime_error,opa_malloc +opa_runtime_error,snprintf_ +opa_runtime_error,opa_abort +builtin_graph_reachable,opa_value_type +builtin_graph_reachable,opa_set +builtin_graph_reachable,opa_array +builtin_graph_reachable,opa_array_append +builtin_graph_reachable,opa_value_get +builtin_graph_reachable,opa_set_get +builtin_graph_reachable,opa_set_add +opa_json_lex_read_number,opa_isdigit +opa_json_lex_read_string,opa_ishex +opa_json_lex_read,opa_strncmp +opa_json_lex_read,opa_isspace +opa_json_lex_read,opa_json_lex_read_string +opa_json_lex_read,opa_isdigit +opa_json_lex_read,opa_json_lex_read_number +opa_json_parse_string,opa_malloc +opa_json_parse_string,opa_string_allocated +opa_json_parse_string,opa_unicode_decode_unit +opa_json_parse_string,opa_unicode_surrogate +opa_json_parse_string,opa_abort +opa_json_parse_string,opa_unicode_decode_utf8 +opa_json_parse_string,opa_unicode_encode_utf8 +opa_json_parse_string,opa_unicode_decode_surrogate +opa_json_parse_token,opa_null +opa_json_parse_token,opa_boolean +opa_json_parse_token,opa_malloc +opa_json_parse_token,opa_number_ref_allocated +opa_json_parse_token,opa_json_parse_string +opa_json_parse_token,opa_array +opa_json_parse_token,opa_json_lex_read +opa_json_parse_token,opa_json_parse_token +opa_json_parse_token,opa_array_append +opa_json_parse_token,opa_object +opa_json_parse_token,opa_json_parse_set +opa_json_parse_token,opa_json_parse_object +opa_json_parse_token,opa_set +opa_json_parse_set,opa_set +opa_json_parse_set,opa_set_add +opa_json_parse_set,opa_json_lex_read +opa_json_parse_set,opa_json_parse_token +opa_json_parse_object,opa_json_lex_read +opa_json_parse_object,opa_json_parse_token +opa_json_parse_object,opa_object +opa_json_parse_object,opa_object_insert +opa_json_parse,opa_json_lex_read +opa_json_parse,opa_json_parse_token +opa_value_parse,opa_json_lex_read +opa_value_parse,opa_json_parse_token +opa_json_writer_emit_boolean,opa_malloc +opa_json_writer_emit_integer,opa_itoa +opa_json_writer_emit_integer,opa_strlen +opa_json_writer_emit_integer,opa_malloc +opa_json_writer_emit_number,opa_json_writer_emit_integer +opa_json_writer_emit_number,opa_malloc +opa_json_writer_emit_number,opa_abort +opa_json_writer_emit_string,opa_malloc +opa_json_writer_emit_string,snprintf_ +opa_json_writer_emit_array_element,opa_value_get +opa_json_writer_emit_array_element,opa_json_writer_emit_value +opa_json_writer_emit_value,opa_value_type +opa_json_writer_emit_value,opa_malloc +opa_json_writer_emit_value,opa_json_writer_emit_boolean +opa_json_writer_emit_value,opa_json_writer_emit_string +opa_json_writer_emit_value,opa_json_writer_emit_number +opa_json_writer_emit_value,opa_json_writer_emit_collection +opa_json_writer_emit_value,opa_json_writer_emit_set_literal +opa_json_writer_emit_collection,opa_malloc +opa_json_writer_emit_collection,opa_value_iter +opa_json_writer_emit_set_element,opa_json_writer_emit_value +opa_json_writer_emit_set_literal,opa_value_length +opa_json_writer_emit_set_literal,opa_malloc +opa_json_writer_emit_set_literal,opa_json_writer_emit_collection +opa_json_writer_emit_object_element,opa_value_type +opa_json_writer_emit_object_element,opa_json_writer_emit_value +opa_json_writer_emit_object_element,opa_json_writer_write +opa_json_writer_emit_object_element,opa_string_terminated +opa_json_writer_emit_object_element,opa_value_free +opa_json_writer_emit_object_element,opa_free +opa_json_writer_emit_object_element,opa_malloc +opa_json_writer_emit_object_element,opa_value_get +opa_json_writer_write,opa_malloc +opa_json_writer_write,opa_json_writer_emit_value +opa_json_writer_write,opa_free +opa_json_dump,opa_json_writer_write +opa_value_dump,opa_json_writer_write +opa_malloc,opa_abort +opa_realloc,opa_malloc +opa_realloc,memcpy +opa_realloc,opa_free +opa_builtin_cache_get,opa_abort +opa_builtin_cache_set,opa_abort +opa_memoize_init,opa_malloc +opa_memoize_init,opa_object +opa_memoize_push,opa_malloc +opa_memoize_push,opa_object +opa_memoize_insert,opa_number_int +opa_memoize_insert,opa_object_insert +opa_memoize_get,opa_number_init_int +opa_memoize_get,opa_object_get +opa_mpd_init,mpd_defaultcontext +opa_mpd_init,mpd_maxcontext +opa_mpd_init,mpd_qnew +opa_mpd_init,mpd_qset_i32 +opa_mpd_init,opa_abort +opa_mpd_del,mpd_del +opa_number_to_bf,opa_value_type +opa_number_to_bf,mpd_qnew +opa_number_to_bf,malloc +opa_number_to_bf,memcpy +opa_number_to_bf,mpd_qset_string +opa_number_to_bf,opa_abort +opa_number_to_bf,free +opa_number_to_bf,mpd_qset_i32 +opa_number_to_bf,snprintf_ +opa_bf_to_number,mpd_qget_i32 +opa_bf_to_number,mpd_del +opa_bf_to_number,opa_number_int +opa_bf_to_number,mpd_to_sci +opa_bf_to_number,opa_strlen +opa_bf_to_number,opa_number_ref +opa_bf_to_number_no_free,mpd_qget_i32 +opa_bf_to_number_no_free,opa_number_int +opa_bf_to_number_no_free,mpd_to_sci +opa_bf_to_number_no_free,opa_strlen +opa_bf_to_number_no_free,opa_number_ref +opa_bf_to_bf_bits,mpd_qnew +opa_bf_to_bf_bits,mpd_qround_to_intx +opa_bf_to_bf_bits,mpd_del +opa_bf_to_bf_bits,mpd_qcmp +opa_bf_to_bf_bits,opa_abort +opa_bf_to_bf_bits,mpd_sign +opa_bf_to_bf_bits,mpd_qabs +opa_bf_to_bf_bits,mpd_sizeinbase +opa_bf_to_bf_bits,malloc +opa_bf_to_bf_bits,mpd_qexport_u16 +opa_bf_to_bf_bits,mpd_qimport_u16 +opa_bf_to_bf_bits,free +opa_bf_bits_to_bf,mpd_sign +opa_bf_bits_to_bf,mpd_qnew +opa_bf_bits_to_bf,mpd_qabs +opa_bf_bits_to_bf,opa_abort +opa_bf_bits_to_bf,mpd_del +opa_bf_bits_to_bf,mpd_sizeinbase +opa_bf_bits_to_bf,malloc +opa_bf_bits_to_bf,mpd_qexport_u16 +opa_bf_bits_to_bf,mpd_qimport_u16 +opa_bf_bits_to_bf,free +qabs,mpd_qnew +qabs,mpd_qabs +qabs,opa_abort +qabs,mpd_del +qadd_one,mpd_qnew +qadd_one,mpd_qadd +qadd_one,opa_abort +qadd_one,mpd_del +qadd,mpd_qnew +qadd,mpd_qadd +qadd,opa_abort +qadd,mpd_del +qsub_one,mpd_qnew +qsub_one,mpd_qsub +qsub_one,opa_abort +qsub_one,mpd_del +qmul,mpd_qnew +qmul,mpd_qmul +qmul,opa_abort +qmul,mpd_del +qand,opa_bf_to_bf_bits +qand,mpd_del +qand,mpd_qnew +qand,mpd_qand +qand,opa_abort +qand,opa_bf_bits_to_bf +qand_not,opa_bf_to_bf_bits +qand_not,mpd_del +qand_not,mpd_sizeinbase +qand_not,malloc +qand_not,mpd_qexport_u16 +qand_not,opa_abort +qand_not,mpd_qnew +qand_not,mpd_qimport_u16 +qand_not,free +qand_not,mpd_qxor +qand_not,mpd_qand +qand_not,opa_bf_bits_to_bf +qor,opa_bf_to_bf_bits +qor,mpd_del +qor,mpd_qnew +qor,mpd_qor +qor,opa_abort +qor,opa_bf_bits_to_bf +qxor,opa_bf_to_bf_bits +qxor,mpd_del +qxor,mpd_qnew +qxor,mpd_qxor +qxor,opa_abort +qxor,opa_bf_bits_to_bf +qneg,opa_bf_to_bf_bits +qneg,mpd_qnew +qneg,mpd_qminus +qneg,mpd_del +qneg,opa_abort +qneg,opa_bf_bits_to_bf +opa_numbers_range,opa_number_to_bf +opa_numbers_range,mpd_isinteger +opa_numbers_range,mpd_qcmp +opa_numbers_range,opa_abort +opa_numbers_range,opa_array +opa_numbers_range,opa_bf_to_number_no_free +opa_numbers_range,opa_array_append +opa_numbers_range,qsub_one +opa_numbers_range,qadd_one +opa_numbers_range,opa_mpd_del +__paths_to_object,opa_object +__paths_to_object,opa_value_get +__paths_to_object,opa_object_insert +__paths_to_object,opa_value_type +__paths_to_object,opa_null +__parse_path,opa_array +__parse_path,opa_value_type +__parse_path,opa_array_append +__parse_path,opa_string_terminated +__parse_path,opa_strings_trim_left +__parse_path,opa_strings_split +__parse_path,opa_strings_replace +__get_json_paths,opa_array +__get_json_paths,opa_value_iter +__get_json_paths,opa_value_type +__get_json_paths,opa_value_get +__get_json_paths,__parse_path +__get_json_paths,opa_array_append +__json_remove,opa_value_type +__json_remove,opa_object +__json_remove,opa_value_iter +__json_remove,opa_value_get +__json_remove,__json_remove +__json_remove,opa_object_insert +__json_remove,opa_set +__json_remove,opa_set_add +__json_remove,opa_array +__json_remove,opa_number_int +__json_remove,opa_strings_format_int +__json_remove,opa_array_append +__json_filter,opa_null +__json_filter,opa_value_compare +__json_filter,opa_value_type +__json_filter,opa_object +__json_filter,opa_value_iter +__json_filter,opa_value_get +__json_filter,__json_filter +__json_filter,opa_object_insert +__json_filter,opa_set +__json_filter,opa_set_add +__json_filter,opa_array +__json_filter,opa_number_int +__json_filter,opa_strings_format_int +__json_filter,opa_array_append +builtin_object_filter,opa_value_type +builtin_object_filter,opa_object +builtin_object_filter,opa_value_iter +builtin_object_filter,opa_value_get +builtin_object_filter,opa_object_get +builtin_object_filter,opa_object_insert +builtin_object_get,opa_value_type +builtin_object_get,opa_object_get +builtin_object_remove,opa_value_type +builtin_object_remove,opa_set +builtin_object_remove,opa_value_iter +builtin_object_remove,opa_value_get +builtin_object_remove,opa_set_add +builtin_object_remove,opa_object +builtin_object_remove,opa_set_get +builtin_object_remove,opa_object_get +builtin_object_remove,opa_object_insert +builtin_object_union,opa_value_type +builtin_object_union,__merge +__merge,opa_object +__merge,opa_value_iter +__merge,opa_object_get +__merge,opa_value_type +__merge,__merge +__merge,opa_object_insert +__merge,opa_value_get +builtin_json_remove,opa_value_type +builtin_json_remove,__get_json_paths +builtin_json_remove,__paths_to_object +builtin_json_remove,__json_remove +builtin_json_filter,opa_value_type +builtin_json_filter,__get_json_paths +builtin_json_filter,__paths_to_object +builtin_json_filter,__json_filter +opa_set_diff,opa_value_type +opa_set_diff,opa_set +opa_set_diff,opa_set_get +opa_set_diff,opa_set_add +opa_set_intersection,opa_value_type +opa_set_intersection,opa_set_with_cap +opa_set_intersection,opa_set_get +opa_set_intersection,opa_set_add +opa_sets_intersection,opa_value_type +opa_sets_intersection,opa_set +opa_sets_intersection,opa_set_union +opa_sets_intersection,opa_set_intersection +opa_sets_intersection,opa_value_free +opa_set_union,opa_value_type +opa_set_union,opa_set +opa_set_union,opa_set_add +opa_sets_union,opa_value_type +opa_sets_union,opa_set +opa_sets_union,opa_set_add +opa_sets_union,opa_value_free +opa_strings_concat,opa_value_type +opa_strings_concat,opa_malloc +opa_strings_concat,memcpy +opa_strings_concat,opa_string_allocated +opa_strings_contains,opa_value_type +opa_strings_contains,opa_strncmp +opa_strings_contains,opa_boolean +opa_strings_endswith,opa_value_type +opa_strings_endswith,opa_boolean +opa_strings_format_int,opa_value_type +opa_strings_format_int,opa_number_try_int +opa_strings_format_int,opa_number_to_bf +opa_strings_format_int,mpd_qnew +opa_strings_format_int,mpd_max_ctx +opa_strings_format_int,mpd_qtrunc +opa_strings_format_int,opa_abort +opa_strings_format_int,mpd_qget_i32 +opa_strings_format_int,opa_malloc +opa_strings_format_int,snprintf_ +opa_strings_format_int,opa_strlen +opa_strings_format_int,opa_string_allocated +opa_strings_indexof,opa_value_type +opa_strings_indexof,opa_strncmp +opa_strings_indexof,opa_number_int +opa_strings_indexof,opa_unicode_decode_utf8 +opa_strings_indexof,opa_abort +opa_strings_replace,opa_value_type +opa_strings_replace,opa_malloc +opa_strings_replace,opa_strncmp +opa_strings_replace,opa_realloc +opa_strings_replace,memcpy +opa_strings_replace,opa_string_allocated +opa_strings_replace_n,opa_value_type +opa_strings_replace_n,opa_malloc +opa_strings_replace_n,memcpy +opa_strings_replace_n,opa_string_allocated +opa_strings_replace_n,opa_strings_replace +opa_strings_replace_n,opa_value_free +opa_strings_reverse,opa_value_type +opa_strings_reverse,opa_malloc +opa_strings_reverse,opa_unicode_decode_utf8 +opa_strings_reverse,opa_abort +opa_strings_reverse,memcpy +opa_strings_reverse,opa_string_allocated +opa_strings_split,opa_value_type +opa_strings_split,opa_array +opa_strings_split,opa_strncmp +opa_strings_split,opa_malloc +opa_strings_split,memcpy +opa_strings_split,opa_string_allocated +opa_strings_split,opa_array_append +opa_strings_split,opa_unicode_decode_utf8 +opa_strings_split,opa_abort +opa_strings_startswith,opa_value_type +opa_strings_startswith,opa_strncmp +opa_strings_startswith,opa_boolean +opa_strings_substring,opa_value_type +opa_strings_substring,opa_number_try_int +opa_strings_substring,opa_string_terminated +opa_strings_substring,opa_unicode_decode_utf8 +opa_strings_substring,opa_abort +opa_strings_substring,opa_malloc +opa_strings_substring,memcpy +opa_strings_substring,opa_string_allocated +opa_strings_trim,opa_value_type +opa_strings_trim,opa_strings_trim_left +opa_strings_trim,opa_strings_trim_right +opa_strings_trim,opa_value_free +opa_strings_trim_left,opa_value_type +opa_strings_trim_left,opa_unicode_decode_utf8 +opa_strings_trim_left,opa_abort +opa_strings_trim_left,opa_strncmp +opa_strings_trim_left,opa_malloc +opa_strings_trim_left,memcpy +opa_strings_trim_left,opa_string_allocated +opa_strings_trim_right,opa_value_type +opa_strings_trim_right,opa_unicode_last_utf8 +opa_strings_trim_right,opa_abort +opa_strings_trim_right,opa_unicode_decode_utf8 +opa_strings_trim_right,opa_strncmp +opa_strings_trim_right,opa_malloc +opa_strings_trim_right,memcpy +opa_strings_trim_right,opa_string_allocated +opa_strings_trim_prefix,opa_value_type +opa_strings_trim_prefix,opa_strncmp +opa_strings_trim_prefix,opa_malloc +opa_strings_trim_prefix,memcpy +opa_strings_trim_prefix,opa_string_allocated +opa_strings_trim_suffix,opa_value_type +opa_strings_trim_suffix,opa_strncmp +opa_strings_trim_suffix,opa_malloc +opa_strings_trim_suffix,memcpy +opa_strings_trim_suffix,opa_string_allocated +opa_strings_trim_space,opa_value_type +opa_strings_trim_space,trim_space +opa_strings_trim_space,opa_malloc +opa_strings_trim_space,memcpy +opa_strings_trim_space,opa_string_allocated +trim_space,opa_unicode_decode_utf8 +trim_space,opa_abort +trim_space,opa_unicode_is_space +trim_space,opa_unicode_last_utf8 +trim_space,opa_malloc +trim_space,memcpy +trim_space,opa_string_allocated +opa_strings_lower,opa_value_type +opa_strings_lower,opa_malloc +opa_strings_lower,opa_string_allocated +opa_strings_lower,malloc +opa_strings_lower,opa_unicode_decode_utf8 +opa_strings_lower,opa_abort +opa_strings_lower,opa_unicode_to_lower +opa_strings_lower,opa_realloc +opa_strings_lower,opa_unicode_encode_utf8 +opa_strings_upper,opa_value_type +opa_strings_upper,opa_malloc +opa_strings_upper,opa_string_allocated +opa_strings_upper,malloc +opa_strings_upper,opa_unicode_decode_utf8 +opa_strings_upper,opa_abort +opa_strings_upper,opa_unicode_to_upper +opa_strings_upper,opa_realloc +opa_strings_upper,opa_unicode_encode_utf8 +opa_types_is_number,opa_value_type +opa_types_is_number,opa_boolean +opa_types_is_string,opa_value_type +opa_types_is_string,opa_boolean +opa_types_is_boolean,opa_value_type +opa_types_is_boolean,opa_boolean +opa_types_is_array,opa_value_type +opa_types_is_array,opa_boolean +opa_types_is_set,opa_value_type +opa_types_is_set,opa_boolean +opa_types_is_object,opa_value_type +opa_types_is_object,opa_boolean +opa_types_is_null,opa_value_type +opa_types_is_null,opa_boolean +opa_types_name,opa_value_type +opa_types_name,opa_string +opa_value_hash,opa_value_hash +opa_value_hash,opa_number_hash +opa_value_compare,opa_value_compare_number +opa_value_compare,opa_value_compare_string +opa_value_compare,opa_value_compare_array +opa_value_compare,opa_value_compare_object +opa_value_compare,opa_value_compare_set +opa_value_compare,opa_abort +opa_object_get,opa_value_hash +opa_object_get,opa_value_compare +opa_set_get,opa_value_hash +opa_set_get,opa_value_compare +opa_number_try_int,opa_atoi64 +opa_number_try_int,opa_abort +opa_value_get,opa_abort +opa_value_get,opa_atoi64 +opa_value_get,opa_value_hash +opa_value_get,opa_value_compare +opa_value_compare_number,opa_atoi64 +opa_value_compare_number,opa_abort +opa_value_compare_number,opa_number_to_bf +opa_value_compare_number,mpd_qcmp +opa_value_compare_number,mpd_del +opa_value_compare_string,opa_strncmp +opa_value_compare_array,opa_value_compare +opa_value_compare_object,opa_object_keys +opa_value_compare_object,opa_value_compare +opa_value_compare_object,opa_value_hash +opa_value_compare_object,opa_value_compare_number +opa_value_compare_object,opa_strncmp +opa_value_compare_object,opa_value_compare_object +opa_value_compare_object,opa_value_compare_set +opa_value_compare_object,opa_abort +opa_value_compare_object,opa_free +opa_value_compare_set,opa_malloc +opa_value_compare_set,opa_free +opa_value_compare_set,opa_value_compare +opa_value_compare_set,opa_value_compare_number +opa_value_compare_set,opa_strncmp +opa_value_compare_set,opa_value_compare_object +opa_value_compare_set,opa_value_compare_set +opa_value_compare_set,opa_abort +opa_number_hash,opa_atof64 +opa_number_hash,opa_abort +opa_value_iter,opa_abort +opa_value_iter,opa_atoi64 +opa_value_iter,opa_value_hash +opa_value_iter,opa_value_compare +opa_object_keys,opa_malloc +opa_object_keys,opa_free +opa_object_keys,opa_value_compare +opa_object_keys,opa_value_compare_number +opa_object_keys,opa_strncmp +opa_object_keys,opa_value_compare_object +opa_object_keys,opa_abort +opa_object_keys,opa_value_compare_set +opa_value_free,opa_free +opa_value_merge,opa_malloc +opa_value_merge,opa_value_get +opa_value_merge,opa_object_insert +opa_value_merge,opa_value_merge +opa_value_merge,opa_abort +opa_value_merge,opa_atoi64 +opa_value_merge,opa_value_hash +opa_value_merge,opa_value_compare_number +opa_value_merge,opa_strncmp +opa_value_merge,opa_value_compare +opa_value_merge,opa_value_compare_object +opa_value_merge,opa_value_compare_set +opa_object_insert,opa_value_hash +opa_object_insert,opa_value_compare +opa_object_insert,__opa_object_grow +opa_object_insert,opa_malloc +__opa_object_grow,opa_malloc +__opa_object_grow,opa_value_hash +__opa_object_grow,opa_value_compare_number +__opa_object_grow,opa_strncmp +__opa_object_grow,opa_value_compare +__opa_object_grow,opa_value_compare_object +__opa_object_grow,opa_value_compare_set +__opa_object_grow,opa_abort +__opa_object_grow,opa_free +opa_boolean,opa_malloc +opa_number_ref,opa_malloc +opa_number_int,opa_malloc +opa_string,opa_malloc +opa_value_shallow_copy_object,opa_malloc +opa_value_shallow_copy_object,opa_value_iter +opa_value_shallow_copy_object,opa_value_get +opa_value_shallow_copy_object,opa_object_insert +opa_value_shallow_copy_set,opa_malloc +opa_value_shallow_copy_set,opa_value_iter +opa_value_shallow_copy_set,opa_set_add +opa_set_add,opa_value_hash +opa_set_add,opa_value_compare +opa_set_add,__opa_set_grow +opa_set_add,opa_malloc +__opa_set_grow,opa_malloc +__opa_set_grow,opa_value_hash +__opa_set_grow,opa_value_compare_number +__opa_set_grow,opa_strncmp +__opa_set_grow,opa_value_compare +__opa_set_grow,opa_value_compare_object +__opa_set_grow,opa_value_compare_set +__opa_set_grow,opa_abort +__opa_set_grow,opa_free +opa_value_shallow_copy,opa_malloc +opa_value_shallow_copy,opa_value_shallow_copy_object +opa_value_shallow_copy,opa_value_shallow_copy_set +opa_value_shallow_copy,opa_abort +opa_value_transitive_closure,opa_malloc +opa_value_transitive_closure,__opa_value_transitive_closure +__opa_value_transitive_closure,opa_malloc +__opa_value_transitive_closure,opa_free +__opa_value_transitive_closure,opa_array_append +__opa_value_transitive_closure,opa_value_iter +__opa_value_transitive_closure,opa_value_get +__opa_value_transitive_closure,__opa_value_transitive_closure +opa_array_append,opa_malloc +opa_array_append,opa_free +opa_null,opa_malloc +opa_number_size,opa_malloc +opa_number_ref_allocated,opa_malloc +opa_string_terminated,opa_malloc +opa_string_terminated,opa_strlen +opa_string_allocated,opa_malloc +opa_array,opa_malloc +opa_array_with_cap,opa_malloc +opa_array_with_cap,opa_free +opa_object,opa_malloc +opa_set,opa_malloc +opa_set_with_cap,opa_malloc +opa_value_add_path,opa_value_get +opa_value_add_path,opa_malloc +opa_value_add_path,opa_object_insert +opa_value_add_path,opa_value_free +opa_value_remove_path,opa_value_get +opa_value_remove_path,opa_value_hash +opa_value_remove_path,opa_value_compare +opa_value_remove_path,opa_value_free +opa_value_remove_path,opa_free +opa_lookup,opa_value_get +opa_lookup,opa_value_iter +opa_lookup,opa_atoi64 +opa_lookup,opa_abort +opa_mapping_init,opa_json_parse +opa_mapping_lookup,opa_lookup +node::re2\28std::__1::basic_string\2c\20std::__1::allocator\20>\20const&\29,std::__1::basic_string\2c\20std::__1::allocator\20>::assign\28char\20const*\29 +node::re2\28std::__1::basic_string\2c\20std::__1::allocator\20>\20const&\29,node::re2\28std::__1::basic_string\2c\20std::__1::allocator\20>\20const&\29 +node::re2\28std::__1::basic_string\2c\20std::__1::allocator\20>\20const&\29,std::__1::basic_string\2c\20std::__1::allocator\20>::append\28char\20const*\2c\20unsigned\20long\29 +node::re2\28std::__1::basic_string\2c\20std::__1::allocator\20>\20const&\29,operator\20delete\28void*\29 +node::re2\28std::__1::basic_string\2c\20std::__1::allocator\20>\20const&\29,std::__1::basic_string\2c\20std::__1::allocator\20>::append\28char\20const*\29 +node::re2\28std::__1::basic_string\2c\20std::__1::allocator\20>\20const&\29,escape\28std::__1::basic_string\2c\20std::__1::allocator\20>\20const&\29 +node::re2\28std::__1::basic_string\2c\20std::__1::allocator\20>\20const&\29,operator\20new\28unsigned\20long\29 +node::re2\28std::__1::basic_string\2c\20std::__1::allocator\20>\20const&\29,memcpy +node::re2\28std::__1::basic_string\2c\20std::__1::allocator\20>\20const&\29,abort +escape\28std::__1::basic_string\2c\20std::__1::allocator\20>\20const&\29,std::__1::basic_string\2c\20std::__1::allocator\20>::push_back\28char\29 +glob_translate\28char\20const*\2c\20unsigned\20long\2c\20std::__1::vector\2c\20std::__1::allocator\20>\2c\20std::__1::allocator\2c\20std::__1::allocator\20>\20>\20>\20const&\2c\20std::__1::basic_string\2c\20std::__1::allocator\20>*\29,operator\20new\28unsigned\20long\29 +glob_translate\28char\20const*\2c\20unsigned\20long\2c\20std::__1::vector\2c\20std::__1::allocator\20>\2c\20std::__1::allocator\2c\20std::__1::allocator\20>\20>\20>\20const&\2c\20std::__1::basic_string\2c\20std::__1::allocator\20>*\29,lexer::lexer\28char\20const*\2c\20unsigned\20long\29 +glob_translate\28char\20const*\2c\20unsigned\20long\2c\20std::__1::vector\2c\20std::__1::allocator\20>\2c\20std::__1::allocator\2c\20std::__1::allocator\20>\20>\20>\20const&\2c\20std::__1::basic_string\2c\20std::__1::allocator\20>*\29,glob_parse\28lexer*\2c\20node**\29 +glob_translate\28char\20const*\2c\20unsigned\20long\2c\20std::__1::vector\2c\20std::__1::allocator\20>\2c\20std::__1::allocator\2c\20std::__1::allocator\20>\20>\20>\20const&\2c\20std::__1::basic_string\2c\20std::__1::allocator\20>*\29,std::__1::basic_string\2c\20std::__1::allocator\20>::compare\28unsigned\20long\2c\20unsigned\20long\2c\20char\20const*\2c\20unsigned\20long\29\20const +glob_translate\28char\20const*\2c\20unsigned\20long\2c\20std::__1::vector\2c\20std::__1::allocator\20>\2c\20std::__1::allocator\2c\20std::__1::allocator\20>\20>\20>\20const&\2c\20std::__1::basic_string\2c\20std::__1::allocator\20>*\29,lexer::~lexer\28\29 +glob_translate\28char\20const*\2c\20unsigned\20long\2c\20std::__1::vector\2c\20std::__1::allocator\20>\2c\20std::__1::allocator\2c\20std::__1::allocator\20>\20>\20>\20const&\2c\20std::__1::basic_string\2c\20std::__1::allocator\20>*\29,operator\20delete\28void*\29 +glob_translate\28char\20const*\2c\20unsigned\20long\2c\20std::__1::vector\2c\20std::__1::allocator\20>\2c\20std::__1::allocator\2c\20std::__1::allocator\20>\20>\20>\20const&\2c\20std::__1::basic_string\2c\20std::__1::allocator\20>*\29,std::__1::basic_string\2c\20std::__1::allocator\20>::assign\28char\20const*\29 +glob_translate\28char\20const*\2c\20unsigned\20long\2c\20std::__1::vector\2c\20std::__1::allocator\20>\2c\20std::__1::allocator\2c\20std::__1::allocator\20>\20>\20>\20const&\2c\20std::__1::basic_string\2c\20std::__1::allocator\20>*\29,opa_unicode_decode_utf8 +glob_translate\28char\20const*\2c\20unsigned\20long\2c\20std::__1::vector\2c\20std::__1::allocator\20>\2c\20std::__1::allocator\2c\20std::__1::allocator\20>\20>\20>\20const&\2c\20std::__1::basic_string\2c\20std::__1::allocator\20>*\29,escape\28std::__1::basic_string\2c\20std::__1::allocator\20>\20const&\29 +glob_translate\28char\20const*\2c\20unsigned\20long\2c\20std::__1::vector\2c\20std::__1::allocator\20>\2c\20std::__1::allocator\2c\20std::__1::allocator\20>\20>\20>\20const&\2c\20std::__1::basic_string\2c\20std::__1::allocator\20>*\29,std::__1::basic_string\2c\20std::__1::allocator\20>::append\28char\20const*\2c\20unsigned\20long\29 +glob_translate\28char\20const*\2c\20unsigned\20long\2c\20std::__1::vector\2c\20std::__1::allocator\20>\2c\20std::__1::allocator\2c\20std::__1::allocator\20>\20>\20>\20const&\2c\20std::__1::basic_string\2c\20std::__1::allocator\20>*\29,std::__1::basic_string\2c\20std::__1::allocator\20>::append\28char\20const*\29 +glob_translate\28char\20const*\2c\20unsigned\20long\2c\20std::__1::vector\2c\20std::__1::allocator\20>\2c\20std::__1::allocator\2c\20std::__1::allocator\20>\20>\20>\20const&\2c\20std::__1::basic_string\2c\20std::__1::allocator\20>*\29,node::re2\28std::__1::basic_string\2c\20std::__1::allocator\20>\20const&\29 +glob_translate\28char\20const*\2c\20unsigned\20long\2c\20std::__1::vector\2c\20std::__1::allocator\20>\2c\20std::__1::allocator\2c\20std::__1::allocator\20>\20>\20>\20const&\2c\20std::__1::basic_string\2c\20std::__1::allocator\20>*\29,node::~node\28\29 +lexer::~lexer\28\29,operator\20delete\28void*\29 +lexer::next\28token*\29,strlen +lexer::next\28token*\29,operator\20new\28unsigned\20long\29 +lexer::next\28token*\29,memcpy +lexer::next\28token*\29,operator\20delete\28void*\29 +lexer::next\28token*\29,std::__1::basic_string\2c\20std::__1::allocator\20>::operator=\28std::__1::basic_string\2c\20std::__1::allocator\20>\20const&\29 +lexer::next\28token*\29,abort +lexer::next\28token*\29,lexer::fetch_item\28\29 +lexer::fetch_item\28\29,opa_unicode_decode_utf8 +lexer::fetch_item\28\29,operator\20new\28unsigned\20long\29 +lexer::fetch_item\28\29,memcpy +lexer::fetch_item\28\29,operator\20delete\28void*\29 +lexer::fetch_item\28\29,lexer::fetch_range\28\29 +lexer::fetch_item\28\29,lexer::fetch_text\28int\20const*\29 +lexer::fetch_item\28\29,abort +lexer::fetch_range\28\29,opa_unicode_decode_utf8 +lexer::fetch_range\28\29,operator\20new\28unsigned\20long\29 +lexer::fetch_range\28\29,memcpy +lexer::fetch_range\28\29,operator\20delete\28void*\29 +lexer::fetch_range\28\29,lexer::fetch_text\28int\20const*\29 +lexer::fetch_range\28\29,abort +lexer::fetch_text\28int\20const*\29,opa_unicode_decode_utf8 +lexer::fetch_text\28int\20const*\29,operator\20new\28unsigned\20long\29 +lexer::fetch_text\28int\20const*\29,memcpy +lexer::fetch_text\28int\20const*\29,operator\20delete\28void*\29 +lexer::fetch_text\28int\20const*\29,opa_malloc +lexer::fetch_text\28int\20const*\29,opa_free +lexer::fetch_text\28int\20const*\29,abort +node::~node\28\29,node::~node\28\29 +node::~node\28\29,operator\20delete\28void*\29 +node::insert\28node*\29,operator\20new\28unsigned\20long\29 +node::insert\28node*\29,memcpy +node::insert\28node*\29,operator\20delete\28void*\29 +node::insert\28node*\29,abort +glob_parse\28lexer*\2c\20node**\29,operator\20new\28unsigned\20long\29 +glob_parse\28lexer*\2c\20node**\29,std::__1::basic_string\2c\20std::__1::allocator\20>::compare\28unsigned\20long\2c\20unsigned\20long\2c\20char\20const*\2c\20unsigned\20long\29\20const +glob_parse\28lexer*\2c\20node**\29,node::~node\28\29 +glob_parse\28lexer*\2c\20node**\29,operator\20delete\28void*\29 +glob_parse\28lexer*\2c\20node**\29,std::__1::basic_string\2c\20std::__1::allocator\20>::basic_string\28std::__1::basic_string\2c\20std::__1::allocator\20>\20const&\29 +parser_main\28state*\2c\20lexer*\29,lexer::next\28token*\29 +parser_main\28state*\2c\20lexer*\29,std::__1::basic_string\2c\20std::__1::allocator\20>::operator=\28std::__1::basic_string\2c\20std::__1::allocator\20>\20const&\29 +parser_main\28state*\2c\20lexer*\29,operator\20new\28unsigned\20long\29 +parser_main\28state*\2c\20lexer*\29,std::__1::basic_string\2c\20std::__1::allocator\20>::basic_string\28std::__1::basic_string\2c\20std::__1::allocator\20>\20const&\29 +parser_main\28state*\2c\20lexer*\29,node::insert\28node*\29 +parser_main\28state*\2c\20lexer*\29,std::__1::basic_string\2c\20std::__1::allocator\20>::assign\28char\20const*\29 +parser_main\28state*\2c\20lexer*\29,operator\20delete\28void*\29 +parser_range\28state*\2c\20lexer*\29,lexer::next\28token*\29 +parser_range\28state*\2c\20lexer*\29,std::__1::basic_string\2c\20std::__1::allocator\20>::assign\28char\20const*\29 +parser_range\28state*\2c\20lexer*\29,std::__1::basic_string\2c\20std::__1::allocator\20>::operator=\28std::__1::basic_string\2c\20std::__1::allocator\20>\20const&\29 +parser_range\28state*\2c\20lexer*\29,opa_unicode_decode_utf8 +parser_range\28state*\2c\20lexer*\29,memcmp +parser_range\28state*\2c\20lexer*\29,std::__1::basic_string\2c\20std::__1::allocator\20>::compare\28unsigned\20long\2c\20unsigned\20long\2c\20char\20const*\2c\20unsigned\20long\29\20const +parser_range\28state*\2c\20lexer*\29,operator\20new\28unsigned\20long\29 +parser_range\28state*\2c\20lexer*\29,std::__1::basic_string\2c\20std::__1::allocator\20>::basic_string\28std::__1::basic_string\2c\20std::__1::allocator\20>\20const&\29 +parser_range\28state*\2c\20lexer*\29,node::insert\28node*\29 +parser_range\28state*\2c\20lexer*\29,operator\20delete\28void*\29 +opa_glob_match,opa_value_type +opa_glob_match,opa_value_iter +opa_glob_match,opa_value_get +opa_glob_match,operator\20new\28unsigned\20long\29 +opa_glob_match,memcpy +opa_glob_match,operator\20delete\28void*\29 +opa_glob_match,opa_builtin_cache_get +opa_glob_match,opa_builtin_cache_set +opa_glob_match,std::__1::basic_string\2c\20std::__1::allocator\20>::basic_string\28std::__1::basic_string\2c\20std::__1::allocator\20>\20const&\29 +opa_glob_match,std::__1::__hash_iterator\2c\20std::__1::allocator\20>\20>\2c\20void*>*>\20std::__1::__hash_table\2c\20std::__1::allocator\20>\20>\2c\20std::__1::__unordered_map_hasher\2c\20std::__1::allocator\20>\20>\2c\20std::__1::hash\2c\20std::__1::equal_to\2c\20true>\2c\20std::__1::__unordered_map_equal\2c\20std::__1::allocator\20>\20>\2c\20std::__1::equal_to\2c\20std::__1::hash\2c\20true>\2c\20std::__1::allocator\2c\20std::__1::allocator\20>\20>\20>\20>::find\28cache_key\20const&\29 +opa_glob_match,std::__1::basic_string\2c\20std::__1::allocator\20>::operator=\28std::__1::basic_string\2c\20std::__1::allocator\20>\20const&\29 +opa_glob_match,glob_translate\28char\20const*\2c\20unsigned\20long\2c\20std::__1::vector\2c\20std::__1::allocator\20>\2c\20std::__1::allocator\2c\20std::__1::allocator\20>\20>\20>\20const&\2c\20std::__1::basic_string\2c\20std::__1::allocator\20>*\29 +opa_glob_match,cache\28\29 +opa_glob_match,std::__1::pair\2c\20std::__1::allocator\20>\20>::pair\2c\20std::__1::allocator\20>&\2c\20false>\28cache_key&\2c\20std::__1::basic_string\2c\20std::__1::allocator\20>&\29 +opa_glob_match,std::__1::pair\2c\20std::__1::allocator\20>\20>\2c\20void*>*>\2c\20bool>\20std::__1::__hash_table\2c\20std::__1::allocator\20>\20>\2c\20std::__1::__unordered_map_hasher\2c\20std::__1::allocator\20>\20>\2c\20std::__1::hash\2c\20std::__1::equal_to\2c\20true>\2c\20std::__1::__unordered_map_equal\2c\20std::__1::allocator\20>\20>\2c\20std::__1::equal_to\2c\20std::__1::hash\2c\20true>\2c\20std::__1::allocator\2c\20std::__1::allocator\20>\20>\20>\20>::__emplace_unique_key_args\2c\20std::__1::allocator\20>\20>\20>\28cache_key\20const&\2c\20std::__1::pair\2c\20std::__1::allocator\20>\20>&&\29 +opa_glob_match,std::__1::pair\2c\20std::__1::allocator\20>\20>::~pair\28\29 +opa_glob_match,opa_string +opa_glob_match,opa_regex_match +opa_glob_match,abort +std::__1::__hash_iterator\2c\20std::__1::allocator\20>\20>\2c\20void*>*>\20std::__1::__hash_table\2c\20std::__1::allocator\20>\20>\2c\20std::__1::__unordered_map_hasher\2c\20std::__1::allocator\20>\20>\2c\20std::__1::hash\2c\20std::__1::equal_to\2c\20true>\2c\20std::__1::__unordered_map_equal\2c\20std::__1::allocator\20>\20>\2c\20std::__1::equal_to\2c\20std::__1::hash\2c\20true>\2c\20std::__1::allocator\2c\20std::__1::allocator\20>\20>\20>\20>::find\28cache_key\20const&\29,std::__1::__unordered_map_hasher\2c\20std::__1::allocator\20>\20>\2c\20std::__1::hash\2c\20std::__1::equal_to\2c\20true>::operator\28\29\28cache_key\20const&\29\20const +std::__1::__hash_iterator\2c\20std::__1::allocator\20>\20>\2c\20void*>*>\20std::__1::__hash_table\2c\20std::__1::allocator\20>\20>\2c\20std::__1::__unordered_map_hasher\2c\20std::__1::allocator\20>\20>\2c\20std::__1::hash\2c\20std::__1::equal_to\2c\20true>\2c\20std::__1::__unordered_map_equal\2c\20std::__1::allocator\20>\20>\2c\20std::__1::equal_to\2c\20std::__1::hash\2c\20true>\2c\20std::__1::allocator\2c\20std::__1::allocator\20>\20>\20>\20>::find\28cache_key\20const&\29,std::__1::equal_to::operator\28\29\28cache_key\20const&\2c\20cache_key\20const&\29\20const +cache\28\29,opa_builtin_cache_get +cache\28\29,operator\20new\28unsigned\20long\29 +cache\28\29,opa_builtin_cache_set +std::__1::pair\2c\20std::__1::allocator\20>\20>::pair\2c\20std::__1::allocator\20>&\2c\20false>\28cache_key&\2c\20std::__1::basic_string\2c\20std::__1::allocator\20>&\29,std::__1::basic_string\2c\20std::__1::allocator\20>::basic_string\28std::__1::basic_string\2c\20std::__1::allocator\20>\20const&\29 +std::__1::pair\2c\20std::__1::allocator\20>\20>::pair\2c\20std::__1::allocator\20>&\2c\20false>\28cache_key&\2c\20std::__1::basic_string\2c\20std::__1::allocator\20>&\29,operator\20new\28unsigned\20long\29 +std::__1::pair\2c\20std::__1::allocator\20>\20>::pair\2c\20std::__1::allocator\20>&\2c\20false>\28cache_key&\2c\20std::__1::basic_string\2c\20std::__1::allocator\20>&\29,abort +std::__1::pair\2c\20std::__1::allocator\20>\20>\2c\20void*>*>\2c\20bool>\20std::__1::__hash_table\2c\20std::__1::allocator\20>\20>\2c\20std::__1::__unordered_map_hasher\2c\20std::__1::allocator\20>\20>\2c\20std::__1::hash\2c\20std::__1::equal_to\2c\20true>\2c\20std::__1::__unordered_map_equal\2c\20std::__1::allocator\20>\20>\2c\20std::__1::equal_to\2c\20std::__1::hash\2c\20true>\2c\20std::__1::allocator\2c\20std::__1::allocator\20>\20>\20>\20>::__emplace_unique_key_args\2c\20std::__1::allocator\20>\20>\20>\28cache_key\20const&\2c\20std::__1::pair\2c\20std::__1::allocator\20>\20>&&\29,std::__1::__unordered_map_hasher\2c\20std::__1::allocator\20>\20>\2c\20std::__1::hash\2c\20std::__1::equal_to\2c\20true>::operator\28\29\28cache_key\20const&\29\20const +std::__1::pair\2c\20std::__1::allocator\20>\20>\2c\20void*>*>\2c\20bool>\20std::__1::__hash_table\2c\20std::__1::allocator\20>\20>\2c\20std::__1::__unordered_map_hasher\2c\20std::__1::allocator\20>\20>\2c\20std::__1::hash\2c\20std::__1::equal_to\2c\20true>\2c\20std::__1::__unordered_map_equal\2c\20std::__1::allocator\20>\20>\2c\20std::__1::equal_to\2c\20std::__1::hash\2c\20true>\2c\20std::__1::allocator\2c\20std::__1::allocator\20>\20>\20>\20>::__emplace_unique_key_args\2c\20std::__1::allocator\20>\20>\20>\28cache_key\20const&\2c\20std::__1::pair\2c\20std::__1::allocator\20>\20>&&\29,std::__1::equal_to::operator\28\29\28cache_key\20const&\2c\20cache_key\20const&\29\20const +std::__1::pair\2c\20std::__1::allocator\20>\20>\2c\20void*>*>\2c\20bool>\20std::__1::__hash_table\2c\20std::__1::allocator\20>\20>\2c\20std::__1::__unordered_map_hasher\2c\20std::__1::allocator\20>\20>\2c\20std::__1::hash\2c\20std::__1::equal_to\2c\20true>\2c\20std::__1::__unordered_map_equal\2c\20std::__1::allocator\20>\20>\2c\20std::__1::equal_to\2c\20std::__1::hash\2c\20true>\2c\20std::__1::allocator\2c\20std::__1::allocator\20>\20>\20>\20>::__emplace_unique_key_args\2c\20std::__1::allocator\20>\20>\20>\28cache_key\20const&\2c\20std::__1::pair\2c\20std::__1::allocator\20>\20>&&\29,operator\20new\28unsigned\20long\29 +std::__1::pair\2c\20std::__1::allocator\20>\20>\2c\20void*>*>\2c\20bool>\20std::__1::__hash_table\2c\20std::__1::allocator\20>\20>\2c\20std::__1::__unordered_map_hasher\2c\20std::__1::allocator\20>\20>\2c\20std::__1::hash\2c\20std::__1::equal_to\2c\20true>\2c\20std::__1::__unordered_map_equal\2c\20std::__1::allocator\20>\20>\2c\20std::__1::equal_to\2c\20std::__1::hash\2c\20true>\2c\20std::__1::allocator\2c\20std::__1::allocator\20>\20>\20>\20>::__emplace_unique_key_args\2c\20std::__1::allocator\20>\20>\20>\28cache_key\20const&\2c\20std::__1::pair\2c\20std::__1::allocator\20>\20>&&\29,std::__1::__next_prime\28unsigned\20long\29 +std::__1::pair\2c\20std::__1::allocator\20>\20>\2c\20void*>*>\2c\20bool>\20std::__1::__hash_table\2c\20std::__1::allocator\20>\20>\2c\20std::__1::__unordered_map_hasher\2c\20std::__1::allocator\20>\20>\2c\20std::__1::hash\2c\20std::__1::equal_to\2c\20true>\2c\20std::__1::__unordered_map_equal\2c\20std::__1::allocator\20>\20>\2c\20std::__1::equal_to\2c\20std::__1::hash\2c\20true>\2c\20std::__1::allocator\2c\20std::__1::allocator\20>\20>\20>\20>::__emplace_unique_key_args\2c\20std::__1::allocator\20>\20>\20>\28cache_key\20const&\2c\20std::__1::pair\2c\20std::__1::allocator\20>\20>&&\29,std::__1::__hash_table\2c\20std::__1::allocator\20>\20>\2c\20std::__1::__unordered_map_hasher\2c\20std::__1::allocator\20>\20>\2c\20std::__1::hash\2c\20std::__1::equal_to\2c\20true>\2c\20std::__1::__unordered_map_equal\2c\20std::__1::allocator\20>\20>\2c\20std::__1::equal_to\2c\20std::__1::hash\2c\20true>\2c\20std::__1::allocator\2c\20std::__1::allocator\20>\20>\20>\20>::__rehash\28unsigned\20long\29 +std::__1::pair\2c\20std::__1::allocator\20>\20>::~pair\28\29,operator\20delete\28void*\29 +std::__1::equal_to::operator\28\29\28cache_key\20const&\2c\20cache_key\20const&\29\20const,memcmp +std::__1::__hash_table\2c\20std::__1::allocator\20>\20>\2c\20std::__1::__unordered_map_hasher\2c\20std::__1::allocator\20>\20>\2c\20std::__1::hash\2c\20std::__1::equal_to\2c\20true>\2c\20std::__1::__unordered_map_equal\2c\20std::__1::allocator\20>\20>\2c\20std::__1::equal_to\2c\20std::__1::hash\2c\20true>\2c\20std::__1::allocator\2c\20std::__1::allocator\20>\20>\20>\20>::__rehash\28unsigned\20long\29,operator\20new\28unsigned\20long\29 +std::__1::__hash_table\2c\20std::__1::allocator\20>\20>\2c\20std::__1::__unordered_map_hasher\2c\20std::__1::allocator\20>\20>\2c\20std::__1::hash\2c\20std::__1::equal_to\2c\20true>\2c\20std::__1::__unordered_map_equal\2c\20std::__1::allocator\20>\20>\2c\20std::__1::equal_to\2c\20std::__1::hash\2c\20true>\2c\20std::__1::allocator\2c\20std::__1::allocator\20>\20>\20>\20>::__rehash\28unsigned\20long\29,operator\20delete\28void*\29 +std::__1::__hash_table\2c\20std::__1::allocator\20>\20>\2c\20std::__1::__unordered_map_hasher\2c\20std::__1::allocator\20>\20>\2c\20std::__1::hash\2c\20std::__1::equal_to\2c\20true>\2c\20std::__1::__unordered_map_equal\2c\20std::__1::allocator\20>\20>\2c\20std::__1::equal_to\2c\20std::__1::hash\2c\20true>\2c\20std::__1::allocator\2c\20std::__1::allocator\20>\20>\20>\20>::__rehash\28unsigned\20long\29,memcmp +std::__1::__hash_table\2c\20std::__1::allocator\20>\20>\2c\20std::__1::__unordered_map_hasher\2c\20std::__1::allocator\20>\20>\2c\20std::__1::hash\2c\20std::__1::equal_to\2c\20true>\2c\20std::__1::__unordered_map_equal\2c\20std::__1::allocator\20>\20>\2c\20std::__1::equal_to\2c\20std::__1::hash\2c\20true>\2c\20std::__1::allocator\2c\20std::__1::allocator\20>\20>\20>\20>::__rehash\28unsigned\20long\29,abort +opa_regex_is_valid,opa_value_type +opa_regex_is_valid,opa_boolean +opa_regex_is_valid,operator\20new\28unsigned\20long\29 +opa_regex_is_valid,memcpy +opa_regex_is_valid,re2::RE2::RE2\28re2::StringPiece\20const&\2c\20re2::RE2::Options\20const&\29 +opa_regex_is_valid,re2::RE2::~RE2\28\29 +opa_regex_is_valid,operator\20delete\28void*\29 +opa_regex_is_valid,abort +opa_regex_match,opa_value_type +opa_regex_match,operator\20new\28unsigned\20long\29 +opa_regex_match,memcpy +opa_regex_match,compile\28char\20const*\29 +opa_regex_match,re2::RE2::PartialMatchN\28re2::StringPiece\20const&\2c\20re2::RE2\20const&\2c\20re2::RE2::Arg\20const*\20const*\2c\20int\29 +opa_regex_match,reuse\28re2::RE2*\29 +opa_regex_match,opa_boolean +opa_regex_match,operator\20delete\28void*\29 +opa_regex_match,abort +compile\28char\20const*\29,opa_builtin_cache_get +compile\28char\20const*\29,operator\20new\28unsigned\20long\29 +compile\28char\20const*\29,opa_builtin_cache_set +compile\28char\20const*\29,strlen +compile\28char\20const*\29,memcpy +compile\28char\20const*\29,std::__1::__hash_iterator\2c\20std::__1::allocator\20>\2c\20re2::RE2*>\2c\20void*>*>\20std::__1::__hash_table\2c\20std::__1::allocator\20>\2c\20re2::RE2*>\2c\20std::__1::__unordered_map_hasher\2c\20std::__1::allocator\20>\2c\20std::__1::__hash_value_type\2c\20std::__1::allocator\20>\2c\20re2::RE2*>\2c\20std::__1::hash\2c\20std::__1::allocator\20>\20>\2c\20std::__1::equal_to\2c\20std::__1::allocator\20>\20>\2c\20true>\2c\20std::__1::__unordered_map_equal\2c\20std::__1::allocator\20>\2c\20std::__1::__hash_value_type\2c\20std::__1::allocator\20>\2c\20re2::RE2*>\2c\20std::__1::equal_to\2c\20std::__1::allocator\20>\20>\2c\20std::__1::hash\2c\20std::__1::allocator\20>\20>\2c\20true>\2c\20std::__1::allocator\2c\20std::__1::allocator\20>\2c\20re2::RE2*>\20>\20>::find\2c\20std::__1::allocator\20>\20>\28std::__1::basic_string\2c\20std::__1::allocator\20>\20const&\29 +compile\28char\20const*\29,operator\20delete\28void*\29 +compile\28char\20const*\29,re2::RE2::RE2\28re2::StringPiece\20const&\2c\20re2::RE2::Options\20const&\29 +compile\28char\20const*\29,re2::RE2::~RE2\28\29 +compile\28char\20const*\29,abort +reuse\28re2::RE2*\29,opa_builtin_cache_get +reuse\28re2::RE2*\29,operator\20new\28unsigned\20long\29 +reuse\28re2::RE2*\29,opa_builtin_cache_set +reuse\28re2::RE2*\29,std::__1::basic_string\2c\20std::__1::allocator\20>::basic_string\28std::__1::basic_string\2c\20std::__1::allocator\20>\20const&\29 +reuse\28re2::RE2*\29,std::__1::pair\2c\20std::__1::allocator\20>\2c\20re2::RE2*>\2c\20void*>*>\2c\20bool>\20std::__1::__hash_table\2c\20std::__1::allocator\20>\2c\20re2::RE2*>\2c\20std::__1::__unordered_map_hasher\2c\20std::__1::allocator\20>\2c\20std::__1::__hash_value_type\2c\20std::__1::allocator\20>\2c\20re2::RE2*>\2c\20std::__1::hash\2c\20std::__1::allocator\20>\20>\2c\20std::__1::equal_to\2c\20std::__1::allocator\20>\20>\2c\20true>\2c\20std::__1::__unordered_map_equal\2c\20std::__1::allocator\20>\2c\20std::__1::__hash_value_type\2c\20std::__1::allocator\20>\2c\20re2::RE2*>\2c\20std::__1::equal_to\2c\20std::__1::allocator\20>\20>\2c\20std::__1::hash\2c\20std::__1::allocator\20>\20>\2c\20true>\2c\20std::__1::allocator\2c\20std::__1::allocator\20>\2c\20re2::RE2*>\20>\20>::__emplace_unique_key_args\2c\20std::__1::allocator\20>\2c\20std::__1::pair\2c\20std::__1::allocator\20>\2c\20re2::RE2*>\20>\28std::__1::basic_string\2c\20std::__1::allocator\20>\20const&\2c\20std::__1::pair\2c\20std::__1::allocator\20>\2c\20re2::RE2*>&&\29 +reuse\28re2::RE2*\29,operator\20delete\28void*\29 +std::__1::__hash_iterator\2c\20std::__1::allocator\20>\2c\20re2::RE2*>\2c\20void*>*>\20std::__1::__hash_table\2c\20std::__1::allocator\20>\2c\20re2::RE2*>\2c\20std::__1::__unordered_map_hasher\2c\20std::__1::allocator\20>\2c\20std::__1::__hash_value_type\2c\20std::__1::allocator\20>\2c\20re2::RE2*>\2c\20std::__1::hash\2c\20std::__1::allocator\20>\20>\2c\20std::__1::equal_to\2c\20std::__1::allocator\20>\20>\2c\20true>\2c\20std::__1::__unordered_map_equal\2c\20std::__1::allocator\20>\2c\20std::__1::__hash_value_type\2c\20std::__1::allocator\20>\2c\20re2::RE2*>\2c\20std::__1::equal_to\2c\20std::__1::allocator\20>\20>\2c\20std::__1::hash\2c\20std::__1::allocator\20>\20>\2c\20true>\2c\20std::__1::allocator\2c\20std::__1::allocator\20>\2c\20re2::RE2*>\20>\20>::find\2c\20std::__1::allocator\20>\20>\28std::__1::basic_string\2c\20std::__1::allocator\20>\20const&\29,memcmp +std::__1::pair\2c\20std::__1::allocator\20>\2c\20re2::RE2*>\2c\20void*>*>\2c\20bool>\20std::__1::__hash_table\2c\20std::__1::allocator\20>\2c\20re2::RE2*>\2c\20std::__1::__unordered_map_hasher\2c\20std::__1::allocator\20>\2c\20std::__1::__hash_value_type\2c\20std::__1::allocator\20>\2c\20re2::RE2*>\2c\20std::__1::hash\2c\20std::__1::allocator\20>\20>\2c\20std::__1::equal_to\2c\20std::__1::allocator\20>\20>\2c\20true>\2c\20std::__1::__unordered_map_equal\2c\20std::__1::allocator\20>\2c\20std::__1::__hash_value_type\2c\20std::__1::allocator\20>\2c\20re2::RE2*>\2c\20std::__1::equal_to\2c\20std::__1::allocator\20>\20>\2c\20std::__1::hash\2c\20std::__1::allocator\20>\20>\2c\20true>\2c\20std::__1::allocator\2c\20std::__1::allocator\20>\2c\20re2::RE2*>\20>\20>::__emplace_unique_key_args\2c\20std::__1::allocator\20>\2c\20std::__1::pair\2c\20std::__1::allocator\20>\2c\20re2::RE2*>\20>\28std::__1::basic_string\2c\20std::__1::allocator\20>\20const&\2c\20std::__1::pair\2c\20std::__1::allocator\20>\2c\20re2::RE2*>&&\29,memcmp +std::__1::pair\2c\20std::__1::allocator\20>\2c\20re2::RE2*>\2c\20void*>*>\2c\20bool>\20std::__1::__hash_table\2c\20std::__1::allocator\20>\2c\20re2::RE2*>\2c\20std::__1::__unordered_map_hasher\2c\20std::__1::allocator\20>\2c\20std::__1::__hash_value_type\2c\20std::__1::allocator\20>\2c\20re2::RE2*>\2c\20std::__1::hash\2c\20std::__1::allocator\20>\20>\2c\20std::__1::equal_to\2c\20std::__1::allocator\20>\20>\2c\20true>\2c\20std::__1::__unordered_map_equal\2c\20std::__1::allocator\20>\2c\20std::__1::__hash_value_type\2c\20std::__1::allocator\20>\2c\20re2::RE2*>\2c\20std::__1::equal_to\2c\20std::__1::allocator\20>\20>\2c\20std::__1::hash\2c\20std::__1::allocator\20>\20>\2c\20true>\2c\20std::__1::allocator\2c\20std::__1::allocator\20>\2c\20re2::RE2*>\20>\20>::__emplace_unique_key_args\2c\20std::__1::allocator\20>\2c\20std::__1::pair\2c\20std::__1::allocator\20>\2c\20re2::RE2*>\20>\28std::__1::basic_string\2c\20std::__1::allocator\20>\20const&\2c\20std::__1::pair\2c\20std::__1::allocator\20>\2c\20re2::RE2*>&&\29,operator\20new\28unsigned\20long\29 +std::__1::pair\2c\20std::__1::allocator\20>\2c\20re2::RE2*>\2c\20void*>*>\2c\20bool>\20std::__1::__hash_table\2c\20std::__1::allocator\20>\2c\20re2::RE2*>\2c\20std::__1::__unordered_map_hasher\2c\20std::__1::allocator\20>\2c\20std::__1::__hash_value_type\2c\20std::__1::allocator\20>\2c\20re2::RE2*>\2c\20std::__1::hash\2c\20std::__1::allocator\20>\20>\2c\20std::__1::equal_to\2c\20std::__1::allocator\20>\20>\2c\20true>\2c\20std::__1::__unordered_map_equal\2c\20std::__1::allocator\20>\2c\20std::__1::__hash_value_type\2c\20std::__1::allocator\20>\2c\20re2::RE2*>\2c\20std::__1::equal_to\2c\20std::__1::allocator\20>\20>\2c\20std::__1::hash\2c\20std::__1::allocator\20>\20>\2c\20true>\2c\20std::__1::allocator\2c\20std::__1::allocator\20>\2c\20re2::RE2*>\20>\20>::__emplace_unique_key_args\2c\20std::__1::allocator\20>\2c\20std::__1::pair\2c\20std::__1::allocator\20>\2c\20re2::RE2*>\20>\28std::__1::basic_string\2c\20std::__1::allocator\20>\20const&\2c\20std::__1::pair\2c\20std::__1::allocator\20>\2c\20re2::RE2*>&&\29,std::__1::__next_prime\28unsigned\20long\29 +std::__1::pair\2c\20std::__1::allocator\20>\2c\20re2::RE2*>\2c\20void*>*>\2c\20bool>\20std::__1::__hash_table\2c\20std::__1::allocator\20>\2c\20re2::RE2*>\2c\20std::__1::__unordered_map_hasher\2c\20std::__1::allocator\20>\2c\20std::__1::__hash_value_type\2c\20std::__1::allocator\20>\2c\20re2::RE2*>\2c\20std::__1::hash\2c\20std::__1::allocator\20>\20>\2c\20std::__1::equal_to\2c\20std::__1::allocator\20>\20>\2c\20true>\2c\20std::__1::__unordered_map_equal\2c\20std::__1::allocator\20>\2c\20std::__1::__hash_value_type\2c\20std::__1::allocator\20>\2c\20re2::RE2*>\2c\20std::__1::equal_to\2c\20std::__1::allocator\20>\20>\2c\20std::__1::hash\2c\20std::__1::allocator\20>\20>\2c\20true>\2c\20std::__1::allocator\2c\20std::__1::allocator\20>\2c\20re2::RE2*>\20>\20>::__emplace_unique_key_args\2c\20std::__1::allocator\20>\2c\20std::__1::pair\2c\20std::__1::allocator\20>\2c\20re2::RE2*>\20>\28std::__1::basic_string\2c\20std::__1::allocator\20>\20const&\2c\20std::__1::pair\2c\20std::__1::allocator\20>\2c\20re2::RE2*>&&\29,std::__1::__hash_table\2c\20std::__1::allocator\20>\2c\20re2::RE2*>\2c\20std::__1::__unordered_map_hasher\2c\20std::__1::allocator\20>\2c\20std::__1::__hash_value_type\2c\20std::__1::allocator\20>\2c\20re2::RE2*>\2c\20std::__1::hash\2c\20std::__1::allocator\20>\20>\2c\20std::__1::equal_to\2c\20std::__1::allocator\20>\20>\2c\20true>\2c\20std::__1::__unordered_map_equal\2c\20std::__1::allocator\20>\2c\20std::__1::__hash_value_type\2c\20std::__1::allocator\20>\2c\20re2::RE2*>\2c\20std::__1::equal_to\2c\20std::__1::allocator\20>\20>\2c\20std::__1::hash\2c\20std::__1::allocator\20>\20>\2c\20true>\2c\20std::__1::allocator\2c\20std::__1::allocator\20>\2c\20re2::RE2*>\20>\20>::__rehash\28unsigned\20long\29 +opa_regex_find_all_string_submatch,opa_value_type +opa_regex_find_all_string_submatch,opa_number_try_int +opa_regex_find_all_string_submatch,operator\20new\28unsigned\20long\29 +opa_regex_find_all_string_submatch,memcpy +opa_regex_find_all_string_submatch,compile\28char\20const*\29 +opa_regex_find_all_string_submatch,opa_array +opa_regex_find_all_string_submatch,memset +opa_regex_find_all_string_submatch,strlen +opa_regex_find_all_string_submatch,re2::RE2::Match\28re2::StringPiece\20const&\2c\20unsigned\20long\2c\20unsigned\20long\2c\20re2::RE2::Anchor\2c\20re2::StringPiece*\2c\20int\29\20const +opa_regex_find_all_string_submatch,fullrune +opa_regex_find_all_string_submatch,chartorune +opa_regex_find_all_string_submatch,opa_array_with_cap +opa_regex_find_all_string_submatch,opa_malloc +opa_regex_find_all_string_submatch,opa_string_allocated +opa_regex_find_all_string_submatch,opa_array_append +opa_regex_find_all_string_submatch,reuse\28re2::RE2*\29 +opa_regex_find_all_string_submatch,operator\20delete\28void*\29 +opa_regex_find_all_string_submatch,abort +std::__1::__hash_table\2c\20std::__1::allocator\20>\2c\20re2::RE2*>\2c\20std::__1::__unordered_map_hasher\2c\20std::__1::allocator\20>\2c\20std::__1::__hash_value_type\2c\20std::__1::allocator\20>\2c\20re2::RE2*>\2c\20std::__1::hash\2c\20std::__1::allocator\20>\20>\2c\20std::__1::equal_to\2c\20std::__1::allocator\20>\20>\2c\20true>\2c\20std::__1::__unordered_map_equal\2c\20std::__1::allocator\20>\2c\20std::__1::__hash_value_type\2c\20std::__1::allocator\20>\2c\20re2::RE2*>\2c\20std::__1::equal_to\2c\20std::__1::allocator\20>\20>\2c\20std::__1::hash\2c\20std::__1::allocator\20>\20>\2c\20true>\2c\20std::__1::allocator\2c\20std::__1::allocator\20>\2c\20re2::RE2*>\20>\20>::__rehash\28unsigned\20long\29,operator\20new\28unsigned\20long\29 +std::__1::__hash_table\2c\20std::__1::allocator\20>\2c\20re2::RE2*>\2c\20std::__1::__unordered_map_hasher\2c\20std::__1::allocator\20>\2c\20std::__1::__hash_value_type\2c\20std::__1::allocator\20>\2c\20re2::RE2*>\2c\20std::__1::hash\2c\20std::__1::allocator\20>\20>\2c\20std::__1::equal_to\2c\20std::__1::allocator\20>\20>\2c\20true>\2c\20std::__1::__unordered_map_equal\2c\20std::__1::allocator\20>\2c\20std::__1::__hash_value_type\2c\20std::__1::allocator\20>\2c\20re2::RE2*>\2c\20std::__1::equal_to\2c\20std::__1::allocator\20>\20>\2c\20std::__1::hash\2c\20std::__1::allocator\20>\20>\2c\20true>\2c\20std::__1::allocator\2c\20std::__1::allocator\20>\2c\20re2::RE2*>\20>\20>::__rehash\28unsigned\20long\29,operator\20delete\28void*\29 +std::__1::__hash_table\2c\20std::__1::allocator\20>\2c\20re2::RE2*>\2c\20std::__1::__unordered_map_hasher\2c\20std::__1::allocator\20>\2c\20std::__1::__hash_value_type\2c\20std::__1::allocator\20>\2c\20re2::RE2*>\2c\20std::__1::hash\2c\20std::__1::allocator\20>\20>\2c\20std::__1::equal_to\2c\20std::__1::allocator\20>\20>\2c\20true>\2c\20std::__1::__unordered_map_equal\2c\20std::__1::allocator\20>\2c\20std::__1::__hash_value_type\2c\20std::__1::allocator\20>\2c\20re2::RE2*>\2c\20std::__1::equal_to\2c\20std::__1::allocator\20>\20>\2c\20std::__1::hash\2c\20std::__1::allocator\20>\20>\2c\20true>\2c\20std::__1::allocator\2c\20std::__1::allocator\20>\2c\20re2::RE2*>\20>\20>::__rehash\28unsigned\20long\29,memcmp +std::__1::__hash_table\2c\20std::__1::allocator\20>\2c\20re2::RE2*>\2c\20std::__1::__unordered_map_hasher\2c\20std::__1::allocator\20>\2c\20std::__1::__hash_value_type\2c\20std::__1::allocator\20>\2c\20re2::RE2*>\2c\20std::__1::hash\2c\20std::__1::allocator\20>\20>\2c\20std::__1::equal_to\2c\20std::__1::allocator\20>\20>\2c\20true>\2c\20std::__1::__unordered_map_equal\2c\20std::__1::allocator\20>\2c\20std::__1::__hash_value_type\2c\20std::__1::allocator\20>\2c\20re2::RE2*>\2c\20std::__1::equal_to\2c\20std::__1::allocator\20>\20>\2c\20std::__1::hash\2c\20std::__1::allocator\20>\20>\2c\20true>\2c\20std::__1::allocator\2c\20std::__1::allocator\20>\2c\20re2::RE2*>\20>\20>::__rehash\28unsigned\20long\29,abort +snprintf_,_vsnprintf +_vsnprintf,_ntoa_format +_vsnprintf,_ftoa +_vsnprintf,_etoa +_ntoa_format,memset +_ftoa,_out_rev +_ftoa,_etoa +_ftoa,memset +_etoa,_ftoa +_etoa,memset +fprintf,opa_abort +fwrite,opa_abort +fputc,opa_abort +abort,opa_abort_ +opa_abort,opa_abort_ +malloc,opa_malloc +free,opa_free +calloc,opa_malloc +calloc,memset +realloc,opa_realloc +strtol,isspace +strtol,isalpha +strtol,isupper +memmove,opa_malloc +memmove,opa_free +_mpd_baseadd,opa_abort +_mpd_shortadd,opa_abort +_mpd_baseincr,opa_abort +_mpd_basesub,opa_abort +_mpd_shortmul,opa_abort +_mpd_basemul,opa_abort +_mpd_basemul,memset +_mpd_shortdiv,opa_abort +_mpd_basedivmod,opa_abort +_mpd_basedivmod,mpd_alloc +_mpd_basedivmod,fprintf +_mpd_basedivmod,fwrite +_mpd_basedivmod,fputc +_mpd_basedivmod,abort +_mpd_baseshiftl,opa_abort +_mpd_baseshiftl,mpd_uint_zero +_mpd_baseshiftr,opa_abort +_mpd_shortmul_c,opa_abort +crt3,opa_abort +fnt_dif2,opa_abort +std_fnt,opa_abort +std_fnt,_mpd_init_fnt_params +std_fnt,fnt_dif2 +std_inv_fnt,opa_abort +std_inv_fnt,_mpd_init_fnt_params +std_inv_fnt,fnt_dif2 +four_step_fnt,opa_abort +four_step_fnt,_mpd_init_w3table +four_step_fnt,_mpd_getkernel +four_step_fnt,six_step_fnt +inv_four_step_fnt,opa_abort +inv_four_step_fnt,inv_six_step_fnt +inv_four_step_fnt,_mpd_getkernel +inv_four_step_fnt,_mpd_init_w3table +mpd_qset_string,mpd_set_flags +mpd_qset_string,mpd_set_negative +mpd_qset_string,mpd_setspecial +mpd_qset_string,strtol +mpd_qset_string,mpd_qresize +mpd_qset_string,mpd_seterror +mpd_qset_string,mpd_setdigits +mpd_qset_string,mpd_qfinalize +mpd_to_sci,_mpd_to_string +_mpd_to_string,mpd_isspecial +_mpd_to_string,mpd_isnan +_mpd_to_string,mpd_alloc +_mpd_to_string,mpd_isnegative +_mpd_to_string,mpd_isqnan +_mpd_to_string,mpd_msword +_mpd_to_string,mpd_word_digits +_mpd_to_string,word_to_string +_mpd_to_string,mpd_isinfinite +_mpd_to_string,abort +_mpd_to_string,opa_abort +_mpd_to_string,mpd_iszero +_mpd_to_string,memset +mpd_switch_to_dyn,opa_abort +mpd_switch_to_dyn,mpd_set_qnan +mpd_switch_to_dyn,mpd_set_positive +mpd_switch_to_dyn,memcpy +mpd_switch_to_dyn,mpd_set_dynamic_data +mpd_realloc_dyn,mpd_set_qnan +mpd_realloc_dyn,mpd_set_positive +mpd_msword,opa_abort +mpd_iszero,opa_abort +mpd_uint_zero,memset +mpd_qresize,opa_abort +mpd_qresize,mpd_switch_to_dyn +mpd_qresize,mpd_realloc_dyn +mpd_setdigits,opa_abort +mpd_zerocoeff,opa_abort +mpd_zerocoeff,mpd_realloc +mpd_qmaxcoeff,opa_abort +mpd_qmaxcoeff,mpd_switch_to_dyn +mpd_qmaxcoeff,mpd_realloc_dyn +mpd_isinteger,opa_abort +_mpd_isint,opa_abort +mpd_setspecial,opa_abort +mpd_setspecial,mpd_realloc +mpd_seterror,opa_abort +mpd_seterror,mpd_realloc +mpd_qsset_ssize,mpd_qfinalize +mpd_qfinalize,_mpd_fix_nan +mpd_qfinalize,_mpd_check_exp +mpd_qfinalize,mpd_qshiftr_inplace +mpd_qfinalize,_mpd_baseincr +mpd_qfinalize,opa_abort +_mpd_fix_nan,opa_abort +_mpd_fix_nan,mpd_realloc +_mpd_fix_nan,mpd_switch_to_dyn +_mpd_fix_nan,mpd_realloc_dyn +_mpd_check_exp,opa_abort +_mpd_check_exp,mpd_realloc +_mpd_check_exp,mpd_qmaxcoeff +_mpd_check_exp,mpd_setspecial +_mpd_check_exp,abort +_mpd_check_exp,mpd_qshiftl +_mpd_check_exp,mpd_qshiftr_inplace +_mpd_check_exp,_mpd_apply_round_excess +_mpd_check_exp,mpd_zerocoeff +mpd_qshiftr_inplace,opa_abort +mpd_qshiftr_inplace,_mpd_get_rnd +mpd_qshiftr_inplace,mpd_realloc +mpd_qshiftr_inplace,_mpd_baseshiftr +mpd_qshiftr_inplace,mpd_switch_to_dyn +mpd_qshiftr_inplace,mpd_realloc_dyn +_settriple,opa_abort +_settriple,mpd_realloc +mpd_qset_i32,opa_abort +mpd_qset_i32,mpd_realloc +mpd_qset_i32,mpd_qsset_ssize +_mpd_qget_uint,opa_abort +_mpd_qget_uint,_mpd_isint +_mpd_qget_uint,mpd_qsshiftr +mpd_qsshiftr,opa_abort +mpd_qsshiftr,memcpy +mpd_qsshiftr,_mpd_get_rnd +mpd_qsshiftr,mpd_realloc +mpd_qsshiftr,_mpd_baseshiftr +mpd_qget_i32,_mpd_qget_uint +mpd_qcopy,opa_abort +mpd_qcopy,mpd_switch_to_dyn +mpd_qcopy,mpd_realloc_dyn +mpd_qcopy,memcpy +mpd_qshiftl,opa_abort +mpd_qshiftl,mpd_qcopy +mpd_qshiftl,mpd_switch_to_dyn +mpd_qshiftl,mpd_realloc_dyn +mpd_qshiftl,_mpd_baseshiftl +_mpd_apply_round_excess,_mpd_baseincr +_mpd_apply_round_excess,opa_abort +_mpd_apply_round_excess,mpd_switch_to_dyn +_mpd_apply_round_excess,mpd_realloc_dyn +mpd_qcmp,_mpd_cmp +_mpd_cmp,opa_abort +_mpd_cmp,_mpd_cmp_same_adjexp +_mpd_cmp_same_adjexp,_mpd_basecmp +mpd_qshiftr,opa_abort +mpd_qshiftr,mpd_qcopy +mpd_qshiftr,_mpd_get_rnd +mpd_qshiftr,mpd_realloc +mpd_qshiftr,_mpd_baseshiftr +mpd_qshiftr,mpd_switch_to_dyn +mpd_qshiftr,mpd_realloc_dyn +mpd_qand,opa_abort +mpd_qand,mpd_realloc +mpd_qand,mpd_switch_to_dyn +mpd_qand,mpd_realloc_dyn +mpd_qand,mpd_setdigits +mpd_qand,_mpd_cap +mpd_qand,mpd_seterror +_mpd_cap,opa_abort +_mpd_cap,mpd_switch_to_dyn +_mpd_cap,mpd_realloc_dyn +_mpd_cap,_settriple +mpd_qor,opa_abort +mpd_qor,mpd_realloc +mpd_qor,mpd_switch_to_dyn +mpd_qor,mpd_realloc_dyn +mpd_qor,mpd_setdigits +mpd_qor,_mpd_cap +mpd_qor,mpd_seterror +_mpd_qaddsub,opa_abort +_mpd_qaddsub,mpd_qshiftl +_mpd_qaddsub,mpd_realloc +_mpd_qaddsub,mpd_switch_to_dyn +_mpd_qaddsub,mpd_realloc_dyn +_mpd_qaddsub,_mpd_baseadd +_mpd_qaddsub,_mpd_basesub +mpd_qshiftn,mpd_qcopy +mpd_qshiftn,_mpd_fix_nan +mpd_qshiftn,mpd_qshiftl +mpd_qshiftn,_mpd_cap +mpd_qshiftn,mpd_qshiftr_inplace +mpd_qshiftn,opa_abort +mpd_qshiftn,mpd_realloc +mpd_qxor,opa_abort +mpd_qxor,mpd_realloc +mpd_qxor,mpd_switch_to_dyn +mpd_qxor,mpd_realloc_dyn +mpd_qxor,mpd_setdigits +mpd_qxor,_mpd_cap +mpd_qxor,mpd_seterror +mpd_qabs,mpd_qcopy +mpd_qabs,_mpd_fix_nan +mpd_qabs,mpd_qminus +mpd_qabs,mpd_qplus +mpd_qminus,mpd_qcopy +mpd_qminus,_mpd_fix_nan +mpd_qminus,opa_abort +mpd_qminus,mpd_qfinalize +mpd_qplus,mpd_qcopy +mpd_qplus,_mpd_fix_nan +mpd_qplus,opa_abort +mpd_qplus,mpd_qfinalize +mpd_qadd,mpd_qcopy +mpd_qadd,_mpd_fix_nan +mpd_qadd,opa_abort +mpd_qadd,mpd_realloc +mpd_qadd,_mpd_qaddsub +mpd_qadd,mpd_qfinalize +mpd_qsub,mpd_qcopy +mpd_qsub,_mpd_fix_nan +mpd_qsub,opa_abort +mpd_qsub,mpd_realloc +mpd_qsub,_mpd_qaddsub +mpd_qsub,mpd_qfinalize +mpd_qdiv,_mpd_qdiv +_mpd_qdiv,mpd_qcopy +_mpd_qdiv,_mpd_fix_nan +_mpd_qdiv,opa_abort +_mpd_qdiv,mpd_realloc +_mpd_qdiv,_settriple +_mpd_qdiv,mpd_qshiftl +_mpd_qdiv,mpd_seterror +_mpd_qdiv,mpd_switch_to_dyn +_mpd_qdiv,mpd_realloc_dyn +_mpd_qdiv,_mpd_shortdiv +_mpd_qdiv,_mpd_basedivmod +_mpd_qdiv,_mpd_base_ndivmod +_mpd_qdiv,mpd_setspecial +_mpd_qdiv,mpd_setdigits +_mpd_qdiv,mpd_trail_zeros +_mpd_qdiv,mpd_qshiftr_inplace +_mpd_qdiv,mpd_qfinalize +_mpd_base_ndivmod,mpd_qnew +_mpd_base_ndivmod,mpd_maxcontext +_mpd_base_ndivmod,opa_abort +_mpd_base_ndivmod,mpd_switch_to_dyn +_mpd_base_ndivmod,mpd_realloc_dyn +_mpd_base_ndivmod,_mpd_shortdiv +_mpd_base_ndivmod,_mpd_qmul_exact +_mpd_base_ndivmod,mpd_qshiftr +_mpd_base_ndivmod,_mpd_qmul +_mpd_base_ndivmod,mpd_qfinalize +_mpd_base_ndivmod,mpd_qsub +_mpd_base_ndivmod,mpd_realloc +_mpd_base_ndivmod,_mpd_qround_to_integral +_mpd_base_ndivmod,fprintf +_mpd_base_ndivmod,fwrite +_mpd_base_ndivmod,fputc +_mpd_base_ndivmod,_mpd_cmp +_mpd_base_ndivmod,mpd_qadd +_mpd_base_ndivmod,mpd_qcopy +_mpd_qdivmod,opa_abort +_mpd_qdivmod,mpd_qcopy +_mpd_qdivmod,_settriple +_mpd_qdivmod,mpd_qshiftl +_mpd_qdivmod,mpd_switch_to_dyn +_mpd_qdivmod,mpd_realloc_dyn +_mpd_qdivmod,_mpd_shortdiv +_mpd_qdivmod,_mpd_basedivmod +_mpd_qdivmod,_mpd_base_ndivmod +_mpd_qdivmod,mpd_setdigits +_mpd_qdivmod,mpd_realloc +_mpd_qmul,mpd_qcopy +_mpd_qmul,_mpd_fix_nan +_mpd_qmul,opa_abort +_mpd_qmul,mpd_realloc +_mpd_qmul,_mpd_mul_2_le2 +_mpd_qmul,memset +_mpd_qmul,_mpd_shortmul +_mpd_qmul,_mpd_basemul +_mpd_qmul,mpd_switch_to_dyn +_mpd_qmul,mpd_realloc_dyn +_mpd_qmul,mpd_calloc +_mpd_qmul,_mpd_kmul +_mpd_qmul,_mpd_fntmul +_mpd_qmul,_mpd_kmul_fnt +_mpd_qmul,mpd_seterror +_mpd_kmul,opa_abort +_mpd_kmul,mpd_calloc +_mpd_kmul,_kmul_worksize +_mpd_kmul,_karatsuba_rec +_mpd_kmul,fprintf +_mpd_kmul,fwrite +_mpd_kmul,fputc +_mpd_kmul,abort +_mpd_fntmul,opa_abort +_mpd_fntmul,fprintf +_mpd_fntmul,fwrite +_mpd_fntmul,fputc +_mpd_fntmul,abort +_mpd_fntmul,mpd_calloc +_mpd_fntmul,memcpy +_mpd_fntmul,fnt_autoconvolute +_mpd_fntmul,fnt_convolute +_mpd_fntmul,memset +_mpd_fntmul,crt3 +_mpd_kmul_fnt,opa_abort +_mpd_kmul_fnt,mpd_calloc +_mpd_kmul_fnt,_kmul_worksize +_mpd_kmul_fnt,fprintf +_mpd_kmul_fnt,fwrite +_mpd_kmul_fnt,fputc +_mpd_kmul_fnt,abort +_mpd_kmul_fnt,_karatsuba_rec_fnt +mpd_qmul,_mpd_qmul +mpd_qmul,mpd_qfinalize +_mpd_qround_to_integral,mpd_qcopy +_mpd_qround_to_integral,_mpd_fix_nan +_mpd_qround_to_integral,opa_abort +_mpd_qround_to_integral,_settriple +_mpd_qround_to_integral,mpd_qshiftr +_mpd_qround_to_integral,_mpd_apply_round_excess +mpd_qrem,mpd_qcopy +mpd_qrem,_mpd_fix_nan +mpd_qrem,opa_abort +mpd_qrem,mpd_realloc +mpd_qrem,abort +mpd_qrem,_mpd_qdivmod +mpd_qrem,mpd_qfinalize +_mpd_qmul_exact,_mpd_qmul +_mpd_qmul_exact,mpd_qfinalize +_mpd_qmul_exact,opa_abort +_mpd_qmul_exact,mpd_realloc +mpd_qround_to_int,_mpd_qround_to_integral +mpd_qround_to_intx,_mpd_qround_to_integral +mpd_qtrunc,_mpd_qround_to_integral +mpd_qfloor,_mpd_qround_to_integral +mpd_qceil,_mpd_qround_to_integral +mpd_sizeinbase,opa_abort +mpd_sizeinbase,log10 +mpd_qexport_u16,opa_abort +mpd_qexport_u16,mpd_sizeinbase +mpd_qexport_u16,mpd_alloc +mpd_qexport_u16,mpd_qshiftl +mpd_qexport_u16,mpd_qshiftr +mpd_qexport_u16,mpd_realloc +mpd_qexport_u16,_mpd_shortdiv +mpd_qimport_u16,opa_abort +mpd_qimport_u16,log10 +mpd_qimport_u16,mpd_realloc +mpd_qimport_u16,mpd_alloc +mpd_qimport_u16,mpd_switch_to_dyn +mpd_qimport_u16,mpd_realloc_dyn +mpd_qimport_u16,_mpd_shortmul_c +mpd_qimport_u16,_mpd_shortadd +mpd_qimport_u16,mpd_qfinalize +_mpd_basecmp,opa_abort +_kmul_worksize,_kmul_worksize +_kmul_worksize,fprintf +_kmul_worksize,fwrite +_kmul_worksize,fputc +_kmul_worksize,abort +_karatsuba_rec,opa_abort +_karatsuba_rec,_mpd_basemul +_karatsuba_rec,memset +_karatsuba_rec,_karatsuba_rec +_karatsuba_rec,_mpd_baseaddto +_karatsuba_rec,memcpy +_karatsuba_rec,_mpd_basesubfrom +_karatsuba_rec_fnt,opa_abort +_karatsuba_rec_fnt,_mpd_basemul +_karatsuba_rec_fnt,_mpd_fntmul +_karatsuba_rec_fnt,memcpy +_karatsuba_rec_fnt,memset +_karatsuba_rec_fnt,_karatsuba_rec_fnt +_karatsuba_rec_fnt,_mpd_baseaddto +_karatsuba_rec_fnt,_mpd_basesubfrom +_mpd_init_fnt_params,opa_abort +_mpd_init_fnt_params,mpd_sh_alloc +six_step_fnt,opa_abort +six_step_fnt,transpose_pow2 +six_step_fnt,_mpd_init_fnt_params +six_step_fnt,fnt_dif2 +six_step_fnt,_mpd_getkernel +inv_six_step_fnt,opa_abort +inv_six_step_fnt,_mpd_init_fnt_params +inv_six_step_fnt,fnt_dif2 +inv_six_step_fnt,_mpd_getkernel +inv_six_step_fnt,transpose_pow2 +transpose_pow2,opa_abort +transpose_pow2,squaretrans_pow2 +transpose_pow2,swap_halfrows_pow2 +transpose_pow2,abort +transpose_pow2,fprintf +transpose_pow2,fwrite +transpose_pow2,fputc +squaretrans_pow2,memcpy +swap_halfrows_pow2,opa_abort +swap_halfrows_pow2,mpd_calloc +swap_halfrows_pow2,memcpy +swap_halfrows_pow2,memmove +swap_halfrows_pow2,fprintf +swap_halfrows_pow2,fwrite +swap_halfrows_pow2,fputc +swap_halfrows_pow2,abort +std::__1::__next_prime\28unsigned\20long\29,abort +operator\20new\28unsigned\20long\29,opa_malloc +operator\20delete\28void*\29,opa_free +operator\20new\5b\5d\28unsigned\20long\29,opa_malloc +operator\20delete\5b\5d\28void*\29,opa_free +__cxa_pure_virtual,opa_abort +std::__1::basic_string\2c\20std::__1::allocator\20>::basic_string\28std::__1::basic_string\2c\20std::__1::allocator\20>\20const&\29,opa_malloc +std::__1::basic_string\2c\20std::__1::allocator\20>::basic_string\28std::__1::basic_string\2c\20std::__1::allocator\20>\20const&\29,memcpy +std::__1::basic_string\2c\20std::__1::allocator\20>::basic_string\28std::__1::basic_string\2c\20std::__1::allocator\20>\20const&\29,abort +std::__1::basic_string\2c\20std::__1::allocator\20>::operator=\28std::__1::basic_string\2c\20std::__1::allocator\20>\20const&\29,memcpy +std::__1::basic_string\2c\20std::__1::allocator\20>::operator=\28std::__1::basic_string\2c\20std::__1::allocator\20>\20const&\29,opa_malloc +std::__1::basic_string\2c\20std::__1::allocator\20>::operator=\28std::__1::basic_string\2c\20std::__1::allocator\20>\20const&\29,opa_free +std::__1::basic_string\2c\20std::__1::allocator\20>::operator=\28std::__1::basic_string\2c\20std::__1::allocator\20>\20const&\29,abort +std::__1::basic_string\2c\20std::__1::allocator\20>::resize\28unsigned\20long\2c\20char\29,std::__1::basic_string\2c\20std::__1::allocator\20>::append\28unsigned\20long\2c\20char\29 +std::__1::basic_string\2c\20std::__1::allocator\20>::append\28unsigned\20long\2c\20char\29,opa_malloc +std::__1::basic_string\2c\20std::__1::allocator\20>::append\28unsigned\20long\2c\20char\29,memcpy +std::__1::basic_string\2c\20std::__1::allocator\20>::append\28unsigned\20long\2c\20char\29,opa_free +std::__1::basic_string\2c\20std::__1::allocator\20>::append\28unsigned\20long\2c\20char\29,memset +std::__1::basic_string\2c\20std::__1::allocator\20>::append\28unsigned\20long\2c\20char\29,abort +std::__1::basic_string\2c\20std::__1::allocator\20>::append\28char\20const*\2c\20unsigned\20long\29,memcpy +std::__1::basic_string\2c\20std::__1::allocator\20>::append\28char\20const*\2c\20unsigned\20long\29,opa_malloc +std::__1::basic_string\2c\20std::__1::allocator\20>::append\28char\20const*\2c\20unsigned\20long\29,opa_free +std::__1::basic_string\2c\20std::__1::allocator\20>::append\28char\20const*\2c\20unsigned\20long\29,abort +std::__1::basic_string\2c\20std::__1::allocator\20>::append\28char\20const*\29,strlen +std::__1::basic_string\2c\20std::__1::allocator\20>::append\28char\20const*\29,std::__1::basic_string\2c\20std::__1::allocator\20>::append\28char\20const*\2c\20unsigned\20long\29 +std::__1::basic_string\2c\20std::__1::allocator\20>::push_back\28char\29,opa_malloc +std::__1::basic_string\2c\20std::__1::allocator\20>::push_back\28char\29,memcpy +std::__1::basic_string\2c\20std::__1::allocator\20>::push_back\28char\29,opa_free +std::__1::basic_string\2c\20std::__1::allocator\20>::push_back\28char\29,abort +std::__1::basic_string\2c\20std::__1::allocator\20>::__assign_external\28char\20const*\2c\20unsigned\20long\29,memmove +std::__1::basic_string\2c\20std::__1::allocator\20>::__assign_external\28char\20const*\2c\20unsigned\20long\29,opa_malloc +std::__1::basic_string\2c\20std::__1::allocator\20>::__assign_external\28char\20const*\2c\20unsigned\20long\29,memcpy +std::__1::basic_string\2c\20std::__1::allocator\20>::__assign_external\28char\20const*\2c\20unsigned\20long\29,opa_free +std::__1::basic_string\2c\20std::__1::allocator\20>::__assign_external\28char\20const*\2c\20unsigned\20long\29,abort +std::__1::basic_string\2c\20std::__1::allocator\20>::assign\28char\20const*\2c\20unsigned\20long\29,std::__1::basic_string\2c\20std::__1::allocator\20>::__assign_external\28char\20const*\2c\20unsigned\20long\29 +std::__1::basic_string\2c\20std::__1::allocator\20>::assign\28char\20const*\29,strlen +std::__1::basic_string\2c\20std::__1::allocator\20>::assign\28char\20const*\29,std::__1::basic_string\2c\20std::__1::allocator\20>::__assign_external\28char\20const*\2c\20unsigned\20long\29 +std::__1::basic_string\2c\20std::__1::allocator\20>::compare\28unsigned\20long\2c\20unsigned\20long\2c\20char\20const*\2c\20unsigned\20long\29\20const,memcmp +std::__1::basic_string\2c\20std::__1::allocator\20>::compare\28unsigned\20long\2c\20unsigned\20long\2c\20char\20const*\2c\20unsigned\20long\29\20const,abort +void\20std::__1::__sort&\2c\20int*>\28int*\2c\20int*\2c\20std::__1::__less&\29,unsigned\20int\20std::__1::__sort5&\2c\20int*>\28int*\2c\20int*\2c\20int*\2c\20int*\2c\20int*\2c\20std::__1::__less&\29 +void\20std::__1::__sort&\2c\20int*>\28int*\2c\20int*\2c\20std::__1::__less&\29,bool\20std::__1::__insertion_sort_incomplete&\2c\20int*>\28int*\2c\20int*\2c\20std::__1::__less&\29 +void\20std::__1::__sort&\2c\20int*>\28int*\2c\20int*\2c\20std::__1::__less&\29,void\20std::__1::__sort&\2c\20int*>\28int*\2c\20int*\2c\20std::__1::__less&\29 +bool\20std::__1::__insertion_sort_incomplete&\2c\20int*>\28int*\2c\20int*\2c\20std::__1::__less&\29,unsigned\20int\20std::__1::__sort5&\2c\20int*>\28int*\2c\20int*\2c\20int*\2c\20int*\2c\20int*\2c\20std::__1::__less&\29 +re2::BitState::Push\28int\2c\20char\20const*\29,operator\20new\28unsigned\20long\29 +re2::BitState::Push\28int\2c\20char\20const*\29,memmove +re2::BitState::Push\28int\2c\20char\20const*\29,operator\20delete\28void*\29 +re2::BitState::Push\28int\2c\20char\20const*\29,abort +re2::BitState::TrySearch\28int\2c\20char\20const*\29,re2::BitState::Push\28int\2c\20char\20const*\29 +re2::BitState::TrySearch\28int\2c\20char\20const*\29,opa_abort +re2::BitState::TrySearch\28int\2c\20char\20const*\29,re2::Prog::EmptyFlags\28re2::StringPiece\20const&\2c\20char\20const*\29 +re2::BitState::Search\28re2::StringPiece\20const&\2c\20re2::StringPiece\20const&\2c\20bool\2c\20bool\2c\20re2::StringPiece*\2c\20int\29,operator\20new\28unsigned\20long\29 +re2::BitState::Search\28re2::StringPiece\20const&\2c\20re2::StringPiece\20const&\2c\20bool\2c\20bool\2c\20re2::StringPiece*\2c\20int\29,operator\20delete\28void*\29 +re2::BitState::Search\28re2::StringPiece\20const&\2c\20re2::StringPiece\20const&\2c\20bool\2c\20bool\2c\20re2::StringPiece*\2c\20int\29,memset +re2::BitState::Search\28re2::StringPiece\20const&\2c\20re2::StringPiece\20const&\2c\20bool\2c\20bool\2c\20re2::StringPiece*\2c\20int\29,re2::BitState::TrySearch\28int\2c\20char\20const*\29 +re2::BitState::Search\28re2::StringPiece\20const&\2c\20re2::StringPiece\20const&\2c\20bool\2c\20bool\2c\20re2::StringPiece*\2c\20int\29,memchr +re2::BitState::Search\28re2::StringPiece\20const&\2c\20re2::StringPiece\20const&\2c\20bool\2c\20bool\2c\20re2::StringPiece*\2c\20int\29,re2::Prog::PrefixAccel_FrontAndBack\28void\20const*\2c\20unsigned\20long\29 +re2::BitState::Search\28re2::StringPiece\20const&\2c\20re2::StringPiece\20const&\2c\20bool\2c\20bool\2c\20re2::StringPiece*\2c\20int\29,abort +re2::Prog::SearchBitState\28re2::StringPiece\20const&\2c\20re2::StringPiece\20const&\2c\20re2::Prog::Anchor\2c\20re2::Prog::MatchKind\2c\20re2::StringPiece*\2c\20int\29,re2::BitState::Search\28re2::StringPiece\20const&\2c\20re2::StringPiece\20const&\2c\20bool\2c\20bool\2c\20re2::StringPiece*\2c\20int\29 +re2::Prog::SearchBitState\28re2::StringPiece\20const&\2c\20re2::StringPiece\20const&\2c\20re2::Prog::Anchor\2c\20re2::Prog::MatchKind\2c\20re2::StringPiece*\2c\20int\29,operator\20delete\28void*\29 +re2::Compiler::AllocInst\28int\29,operator\20new\28unsigned\20long\29 +re2::Compiler::AllocInst\28int\29,memset +re2::Compiler::AllocInst\28int\29,memmove +re2::Compiler::AllocInst\28int\29,operator\20delete\28void*\29 +re2::Compiler::AllocInst\28int\29,abort +re2::Compiler::~Compiler\28\29,re2::Prog::~Prog\28\29 +re2::Compiler::~Compiler\28\29,operator\20delete\28void*\29 +re2::Compiler::~Compiler\28\29,re2::Regexp::Walker::~Walker\28\29 +re2::Regexp::Walker::~Walker\28\29,operator\20delete\5b\5d\28void*\29 +re2::Regexp::Walker::~Walker\28\29,operator\20delete\28void*\29 +re2::Regexp::Walker::~Walker\28\29,std::__1::__deque_base\2c\20std::__1::allocator\20>\20>::~__deque_base\28\29 +re2::Compiler::~Compiler\28\29.1,re2::Compiler::~Compiler\28\29 +re2::Compiler::~Compiler\28\29.1,operator\20delete\28void*\29 +re2::Compiler::Cat\28re2::Frag\2c\20re2::Frag\29,opa_abort +re2::Compiler::Star\28re2::Frag\2c\20bool\29,re2::Compiler::AllocInst\28int\29 +re2::Compiler::Star\28re2::Frag\2c\20bool\29,re2::Prog::Inst::InitAlt\28unsigned\20int\2c\20unsigned\20int\29 +re2::Compiler::Star\28re2::Frag\2c\20bool\29,opa_abort +re2::Compiler::Quest\28re2::Frag\2c\20bool\29,re2::Compiler::AllocInst\28int\29 +re2::Compiler::Quest\28re2::Frag\2c\20bool\29,re2::Prog::Inst::InitNop\28unsigned\20int\29 +re2::Compiler::Quest\28re2::Frag\2c\20bool\29,re2::Prog::Inst::InitAlt\28unsigned\20int\2c\20unsigned\20int\29 +re2::Compiler::Nop\28\29,re2::Compiler::AllocInst\28int\29 +re2::Compiler::Nop\28\29,re2::Prog::Inst::InitNop\28unsigned\20int\29 +re2::Compiler::ByteRange\28int\2c\20int\2c\20bool\29,re2::Compiler::AllocInst\28int\29 +re2::Compiler::ByteRange\28int\2c\20int\2c\20bool\29,re2::Prog::Inst::InitByteRange\28int\2c\20int\2c\20int\2c\20unsigned\20int\29 +re2::Compiler::Match\28int\29,re2::Compiler::AllocInst\28int\29 +re2::Compiler::Match\28int\29,re2::Prog::Inst::InitMatch\28int\29 +re2::Compiler::EmptyWidth\28re2::EmptyOp\29,re2::Compiler::AllocInst\28int\29 +re2::Compiler::EmptyWidth\28re2::EmptyOp\29,re2::Prog::Inst::InitEmptyWidth\28re2::EmptyOp\2c\20unsigned\20int\29 +re2::Compiler::Capture\28re2::Frag\2c\20int\29,re2::Compiler::AllocInst\28int\29 +re2::Compiler::Capture\28re2::Frag\2c\20int\29,re2::Prog::Inst::InitCapture\28int\2c\20unsigned\20int\29 +re2::Compiler::Capture\28re2::Frag\2c\20int\29,opa_abort +re2::Compiler::BeginRange\28\29,operator\20delete\28void*\29 +re2::Compiler::UncachedRuneByteSuffix\28unsigned\20char\2c\20unsigned\20char\2c\20bool\2c\20int\29,re2::Compiler::AllocInst\28int\29 +re2::Compiler::UncachedRuneByteSuffix\28unsigned\20char\2c\20unsigned\20char\2c\20bool\2c\20int\29,re2::Prog::Inst::InitByteRange\28int\2c\20int\2c\20int\2c\20unsigned\20int\29 +re2::Compiler::UncachedRuneByteSuffix\28unsigned\20char\2c\20unsigned\20char\2c\20bool\2c\20int\29,opa_abort +std::__1::pair\2c\20void*>*>\2c\20bool>\20std::__1::__hash_table\2c\20std::__1::__unordered_map_hasher\2c\20std::__1::hash\2c\20std::__1::equal_to\2c\20true>\2c\20std::__1::__unordered_map_equal\2c\20std::__1::equal_to\2c\20std::__1::hash\2c\20true>\2c\20std::__1::allocator\20>\20>::__emplace_unique_key_args\2c\20std::__1::tuple<>\20>\28unsigned\20long\20long\20const&\2c\20std::__1::piecewise_construct_t\20const&\2c\20std::__1::tuple&&\2c\20std::__1::tuple<>&&\29,operator\20new\28unsigned\20long\29 +std::__1::pair\2c\20void*>*>\2c\20bool>\20std::__1::__hash_table\2c\20std::__1::__unordered_map_hasher\2c\20std::__1::hash\2c\20std::__1::equal_to\2c\20true>\2c\20std::__1::__unordered_map_equal\2c\20std::__1::equal_to\2c\20std::__1::hash\2c\20true>\2c\20std::__1::allocator\20>\20>::__emplace_unique_key_args\2c\20std::__1::tuple<>\20>\28unsigned\20long\20long\20const&\2c\20std::__1::piecewise_construct_t\20const&\2c\20std::__1::tuple&&\2c\20std::__1::tuple<>&&\29,std::__1::__next_prime\28unsigned\20long\29 +std::__1::pair\2c\20void*>*>\2c\20bool>\20std::__1::__hash_table\2c\20std::__1::__unordered_map_hasher\2c\20std::__1::hash\2c\20std::__1::equal_to\2c\20true>\2c\20std::__1::__unordered_map_equal\2c\20std::__1::equal_to\2c\20std::__1::hash\2c\20true>\2c\20std::__1::allocator\20>\20>::__emplace_unique_key_args\2c\20std::__1::tuple<>\20>\28unsigned\20long\20long\20const&\2c\20std::__1::piecewise_construct_t\20const&\2c\20std::__1::tuple&&\2c\20std::__1::tuple<>&&\29,std::__1::__hash_table\2c\20std::__1::__unordered_map_hasher\2c\20std::__1::hash\2c\20std::__1::equal_to\2c\20true>\2c\20std::__1::__unordered_map_equal\2c\20std::__1::equal_to\2c\20std::__1::hash\2c\20true>\2c\20std::__1::allocator\20>\20>::__rehash\28unsigned\20long\29 +re2::Compiler::AddSuffixRecursive\28int\2c\20int\29,opa_abort +re2::Compiler::AddSuffixRecursive\28int\2c\20int\29,re2::Compiler::FindByteRange\28int\2c\20int\29 +re2::Compiler::AddSuffixRecursive\28int\2c\20int\29,re2::Compiler::AllocInst\28int\29 +re2::Compiler::AddSuffixRecursive\28int\2c\20int\29,re2::Prog::Inst::InitAlt\28unsigned\20int\2c\20unsigned\20int\29 +re2::Compiler::AddSuffixRecursive\28int\2c\20int\29,std::__1::__hash_iterator\2c\20void*>*>\20std::__1::__hash_table\2c\20std::__1::__unordered_map_hasher\2c\20std::__1::hash\2c\20std::__1::equal_to\2c\20true>\2c\20std::__1::__unordered_map_equal\2c\20std::__1::equal_to\2c\20std::__1::hash\2c\20true>\2c\20std::__1::allocator\20>\20>::find\28unsigned\20long\20long\20const&\29 +re2::Compiler::AddSuffixRecursive\28int\2c\20int\29,re2::Prog::Inst::InitByteRange\28int\2c\20int\2c\20int\2c\20unsigned\20int\29 +re2::Compiler::AddSuffixRecursive\28int\2c\20int\29,re2::Compiler::AddSuffixRecursive\28int\2c\20int\29 +re2::Compiler::FindByteRange\28int\2c\20int\29,opa_abort +re2::Compiler::FindByteRange\28int\2c\20int\29,re2::Compiler::ByteRangeEqual\28int\2c\20int\29 +re2::Compiler::ByteRangeEqual\28int\2c\20int\29,opa_abort +re2::Compiler::AddRuneRange\28int\2c\20int\2c\20bool\29,re2::Compiler::AddRuneRangeUTF8\28int\2c\20int\2c\20bool\29 +re2::Compiler::AddRuneRange\28int\2c\20int\2c\20bool\29,re2::Compiler::AddRuneRangeLatin1\28int\2c\20int\2c\20bool\29 +re2::Compiler::AddRuneRangeUTF8\28int\2c\20int\2c\20bool\29,re2::Compiler::Add_80_10ffff\28\29 +re2::Compiler::AddRuneRangeUTF8\28int\2c\20int\2c\20bool\29,runetochar +re2::Compiler::AddRuneRangeUTF8\28int\2c\20int\2c\20bool\29,re2::Compiler::AddRuneRangeUTF8\28int\2c\20int\2c\20bool\29 +re2::Compiler::AddRuneRangeUTF8\28int\2c\20int\2c\20bool\29,opa_abort +re2::Compiler::AddRuneRangeUTF8\28int\2c\20int\2c\20bool\29,std::__1::__hash_iterator\2c\20void*>*>\20std::__1::__hash_table\2c\20std::__1::__unordered_map_hasher\2c\20std::__1::hash\2c\20std::__1::equal_to\2c\20true>\2c\20std::__1::__unordered_map_equal\2c\20std::__1::equal_to\2c\20std::__1::hash\2c\20true>\2c\20std::__1::allocator\20>\20>::find\28unsigned\20long\20long\20const&\29 +re2::Compiler::AddRuneRangeUTF8\28int\2c\20int\2c\20bool\29,re2::Compiler::UncachedRuneByteSuffix\28unsigned\20char\2c\20unsigned\20char\2c\20bool\2c\20int\29 +re2::Compiler::AddRuneRangeUTF8\28int\2c\20int\2c\20bool\29,std::__1::pair\2c\20void*>*>\2c\20bool>\20std::__1::__hash_table\2c\20std::__1::__unordered_map_hasher\2c\20std::__1::hash\2c\20std::__1::equal_to\2c\20true>\2c\20std::__1::__unordered_map_equal\2c\20std::__1::equal_to\2c\20std::__1::hash\2c\20true>\2c\20std::__1::allocator\20>\20>::__emplace_unique_key_args\2c\20std::__1::tuple<>\20>\28unsigned\20long\20long\20const&\2c\20std::__1::piecewise_construct_t\20const&\2c\20std::__1::tuple&&\2c\20std::__1::tuple<>&&\29 +re2::Compiler::AddRuneRangeUTF8\28int\2c\20int\2c\20bool\29,re2::Compiler::AddSuffixRecursive\28int\2c\20int\29 +re2::Compiler::AddRuneRangeUTF8\28int\2c\20int\2c\20bool\29,re2::Compiler::AllocInst\28int\29 +re2::Compiler::AddRuneRangeUTF8\28int\2c\20int\2c\20bool\29,re2::Prog::Inst::InitAlt\28unsigned\20int\2c\20unsigned\20int\29 +re2::Compiler::AddRuneRangeUTF8\28int\2c\20int\2c\20bool\29,re2::Prog::Inst::InitByteRange\28int\2c\20int\2c\20int\2c\20unsigned\20int\29 +re2::Compiler::AddRuneRangeLatin1\28int\2c\20int\2c\20bool\29,re2::Compiler::AllocInst\28int\29 +re2::Compiler::AddRuneRangeLatin1\28int\2c\20int\2c\20bool\29,re2::Prog::Inst::InitByteRange\28int\2c\20int\2c\20int\2c\20unsigned\20int\29 +re2::Compiler::AddRuneRangeLatin1\28int\2c\20int\2c\20bool\29,re2::Compiler::AddSuffixRecursive\28int\2c\20int\29 +re2::Compiler::AddRuneRangeLatin1\28int\2c\20int\2c\20bool\29,re2::Prog::Inst::InitAlt\28unsigned\20int\2c\20unsigned\20int\29 +re2::Compiler::Add_80_10ffff\28\29,re2::Compiler::AllocInst\28int\29 +re2::Compiler::Add_80_10ffff\28\29,re2::Prog::Inst::InitByteRange\28int\2c\20int\2c\20int\2c\20unsigned\20int\29 +re2::Compiler::Add_80_10ffff\28\29,re2::Compiler::UncachedRuneByteSuffix\28unsigned\20char\2c\20unsigned\20char\2c\20bool\2c\20int\29 +re2::Compiler::Add_80_10ffff\28\29,re2::Compiler::AddSuffixRecursive\28int\2c\20int\29 +re2::Compiler::Add_80_10ffff\28\29,re2::Prog::Inst::InitAlt\28unsigned\20int\2c\20unsigned\20int\29 +re2::Compiler::Literal\28int\2c\20bool\29,re2::Compiler::AllocInst\28int\29 +re2::Compiler::Literal\28int\2c\20bool\29,re2::Prog::Inst::InitByteRange\28int\2c\20int\2c\20int\2c\20unsigned\20int\29 +re2::Compiler::Literal\28int\2c\20bool\29,runetochar +re2::Compiler::Literal\28int\2c\20bool\29,re2::Compiler::Cat\28re2::Frag\2c\20re2::Frag\29 +re2::Compiler::PostVisit\28re2::Regexp*\2c\20re2::Frag\2c\20re2::Frag\2c\20re2::Frag*\2c\20int\29,re2::Compiler::Nop\28\29 +re2::Compiler::PostVisit\28re2::Regexp*\2c\20re2::Frag\2c\20re2::Frag\2c\20re2::Frag*\2c\20int\29,re2::Compiler::Match\28int\29 +re2::Compiler::PostVisit\28re2::Regexp*\2c\20re2::Frag\2c\20re2::Frag\2c\20re2::Frag*\2c\20int\29,re2::Compiler::EmptyWidth\28re2::EmptyOp\29 +re2::Compiler::PostVisit\28re2::Regexp*\2c\20re2::Frag\2c\20re2::Frag\2c\20re2::Frag*\2c\20int\29,re2::Compiler::Cat\28re2::Frag\2c\20re2::Frag\29 +re2::Compiler::PostVisit\28re2::Regexp*\2c\20re2::Frag\2c\20re2::Frag\2c\20re2::Frag*\2c\20int\29,re2::Compiler::AllocInst\28int\29 +re2::Compiler::PostVisit\28re2::Regexp*\2c\20re2::Frag\2c\20re2::Frag\2c\20re2::Frag*\2c\20int\29,re2::Prog::Inst::InitAlt\28unsigned\20int\2c\20unsigned\20int\29 +re2::Compiler::PostVisit\28re2::Regexp*\2c\20re2::Frag\2c\20re2::Frag\2c\20re2::Frag*\2c\20int\29,re2::Compiler::Star\28re2::Frag\2c\20bool\29 +re2::Compiler::PostVisit\28re2::Regexp*\2c\20re2::Frag\2c\20re2::Frag\2c\20re2::Frag*\2c\20int\29,re2::Compiler::Quest\28re2::Frag\2c\20bool\29 +re2::Compiler::PostVisit\28re2::Regexp*\2c\20re2::Frag\2c\20re2::Frag\2c\20re2::Frag*\2c\20int\29,re2::Compiler::Literal\28int\2c\20bool\29 +re2::Compiler::PostVisit\28re2::Regexp*\2c\20re2::Frag\2c\20re2::Frag\2c\20re2::Frag*\2c\20int\29,opa_abort +re2::Compiler::PostVisit\28re2::Regexp*\2c\20re2::Frag\2c\20re2::Frag\2c\20re2::Frag*\2c\20int\29,re2::Compiler::BeginRange\28\29 +re2::Compiler::PostVisit\28re2::Regexp*\2c\20re2::Frag\2c\20re2::Frag\2c\20re2::Frag*\2c\20int\29,re2::Compiler::AddRuneRange\28int\2c\20int\2c\20bool\29 +re2::Compiler::PostVisit\28re2::Regexp*\2c\20re2::Frag\2c\20re2::Frag\2c\20re2::Frag*\2c\20int\29,re2::Compiler::ByteRange\28int\2c\20int\2c\20bool\29 +re2::Compiler::PostVisit\28re2::Regexp*\2c\20re2::Frag\2c\20re2::Frag\2c\20re2::Frag*\2c\20int\29,re2::Compiler::AddRuneRangeUTF8\28int\2c\20int\2c\20bool\29 +re2::Compiler::PostVisit\28re2::Regexp*\2c\20re2::Frag\2c\20re2::Frag\2c\20re2::Frag*\2c\20int\29,re2::Compiler::AddRuneRangeLatin1\28int\2c\20int\2c\20bool\29 +re2::Compiler::PostVisit\28re2::Regexp*\2c\20re2::Frag\2c\20re2::Frag\2c\20re2::Frag*\2c\20int\29,re2::Compiler::Capture\28re2::Frag\2c\20int\29 +re2::Compiler::Compile\28re2::Regexp*\2c\20bool\2c\20long\20long\29,operator\20new\28unsigned\20long\29 +re2::Compiler::Compile\28re2::Regexp*\2c\20bool\2c\20long\20long\29,re2::Prog::Prog\28\29 +re2::Compiler::Compile\28re2::Regexp*\2c\20bool\2c\20long\20long\29,re2::Prog::Inst::InitFail\28\29 +re2::Compiler::Compile\28re2::Regexp*\2c\20bool\2c\20long\20long\29,re2::Regexp::Simplify\28\29 +re2::Compiler::Compile\28re2::Regexp*\2c\20bool\2c\20long\20long\29,re2::IsAnchorStart\28re2::Regexp**\2c\20int\29 +re2::Compiler::Compile\28re2::Regexp*\2c\20bool\2c\20long\20long\29,re2::IsAnchorEnd\28re2::Regexp**\2c\20int\29 +re2::Compiler::Compile\28re2::Regexp*\2c\20bool\2c\20long\20long\29,re2::Regexp::Walker::WalkInternal\28re2::Regexp*\2c\20re2::Frag\2c\20bool\29 +re2::Compiler::Compile\28re2::Regexp*\2c\20bool\2c\20long\20long\29,re2::Regexp::Decref\28\29 +re2::Compiler::Compile\28re2::Regexp*\2c\20bool\2c\20long\20long\29,memset +re2::Compiler::Compile\28re2::Regexp*\2c\20bool\2c\20long\20long\29,memmove +re2::Compiler::Compile\28re2::Regexp*\2c\20bool\2c\20long\20long\29,operator\20delete\28void*\29 +re2::Compiler::Compile\28re2::Regexp*\2c\20bool\2c\20long\20long\29,re2::Prog::Inst::InitMatch\28int\29 +re2::Compiler::Compile\28re2::Regexp*\2c\20bool\2c\20long\20long\29,re2::Compiler::Cat\28re2::Frag\2c\20re2::Frag\29 +re2::Compiler::Compile\28re2::Regexp*\2c\20bool\2c\20long\20long\29,re2::Compiler::DotStar\28\29 +re2::Compiler::Compile\28re2::Regexp*\2c\20bool\2c\20long\20long\29,re2::Compiler::Finish\28re2::Regexp*\29 +re2::Compiler::Compile\28re2::Regexp*\2c\20bool\2c\20long\20long\29,re2::Compiler::~Compiler\28\29 +re2::Compiler::Compile\28re2::Regexp*\2c\20bool\2c\20long\20long\29,abort +re2::IsAnchorStart\28re2::Regexp**\2c\20int\29,re2::Regexp::Incref\28\29 +re2::IsAnchorStart\28re2::Regexp**\2c\20int\29,re2::IsAnchorStart\28re2::Regexp**\2c\20int\29 +re2::IsAnchorStart\28re2::Regexp**\2c\20int\29,operator\20new\28unsigned\20long\29 +re2::IsAnchorStart\28re2::Regexp**\2c\20int\29,re2::Regexp::Concat\28re2::Regexp**\2c\20int\2c\20re2::Regexp::ParseFlags\29 +re2::IsAnchorStart\28re2::Regexp**\2c\20int\29,re2::Regexp::Decref\28\29 +re2::IsAnchorStart\28re2::Regexp**\2c\20int\29,operator\20delete\28void*\29 +re2::IsAnchorStart\28re2::Regexp**\2c\20int\29,opa_abort +re2::IsAnchorStart\28re2::Regexp**\2c\20int\29,re2::Regexp::Capture\28re2::Regexp*\2c\20re2::Regexp::ParseFlags\2c\20int\29 +re2::IsAnchorStart\28re2::Regexp**\2c\20int\29,re2::Regexp::LiteralString\28int*\2c\20int\2c\20re2::Regexp::ParseFlags\29 +re2::IsAnchorEnd\28re2::Regexp**\2c\20int\29,re2::Regexp::Incref\28\29 +re2::IsAnchorEnd\28re2::Regexp**\2c\20int\29,re2::IsAnchorEnd\28re2::Regexp**\2c\20int\29 +re2::IsAnchorEnd\28re2::Regexp**\2c\20int\29,operator\20new\28unsigned\20long\29 +re2::IsAnchorEnd\28re2::Regexp**\2c\20int\29,re2::Regexp::Concat\28re2::Regexp**\2c\20int\2c\20re2::Regexp::ParseFlags\29 +re2::IsAnchorEnd\28re2::Regexp**\2c\20int\29,re2::Regexp::Decref\28\29 +re2::IsAnchorEnd\28re2::Regexp**\2c\20int\29,operator\20delete\28void*\29 +re2::IsAnchorEnd\28re2::Regexp**\2c\20int\29,opa_abort +re2::IsAnchorEnd\28re2::Regexp**\2c\20int\29,re2::Regexp::Capture\28re2::Regexp*\2c\20re2::Regexp::ParseFlags\2c\20int\29 +re2::IsAnchorEnd\28re2::Regexp**\2c\20int\29,re2::Regexp::LiteralString\28int*\2c\20int\2c\20re2::Regexp::ParseFlags\29 +re2::Regexp::Walker::WalkInternal\28re2::Regexp*\2c\20re2::Frag\2c\20bool\29,operator\20delete\5b\5d\28void*\29 +re2::Regexp::Walker::WalkInternal\28re2::Regexp*\2c\20re2::Frag\2c\20bool\29,operator\20delete\28void*\29 +re2::Regexp::Walker::WalkInternal\28re2::Regexp*\2c\20re2::Frag\2c\20bool\29,std::__1::deque\2c\20std::__1::allocator\20>\20>::__add_back_capacity\28\29 +re2::Regexp::Walker::WalkInternal\28re2::Regexp*\2c\20re2::Frag\2c\20bool\29,operator\20new\5b\5d\28unsigned\20long\29 +re2::Compiler::DotStar\28\29,re2::Compiler::AllocInst\28int\29 +re2::Compiler::DotStar\28\29,re2::Prog::Inst::InitByteRange\28int\2c\20int\2c\20int\2c\20unsigned\20int\29 +re2::Compiler::DotStar\28\29,re2::Prog::Inst::InitAlt\28unsigned\20int\2c\20unsigned\20int\29 +re2::Compiler::DotStar\28\29,opa_abort +re2::Compiler::Finish\28re2::Regexp*\29,operator\20delete\28void*\29 +re2::Compiler::Finish\28re2::Regexp*\29,re2::Prog::Optimize\28\29 +re2::Compiler::Finish\28re2::Regexp*\29,re2::Prog::Flatten\28\29 +re2::Compiler::Finish\28re2::Regexp*\29,re2::Prog::ComputeByteMap\28\29 +re2::Compiler::Finish\28re2::Regexp*\29,re2::Regexp::RequiredPrefixForAccel\28std::__1::basic_string\2c\20std::__1::allocator\20>*\2c\20bool*\29 +re2::Regexp::CompileToProg\28long\20long\29,re2::Compiler::Compile\28re2::Regexp*\2c\20bool\2c\20long\20long\29 +re2::Regexp::CompileToReverseProg\28long\20long\29,re2::Compiler::Compile\28re2::Regexp*\2c\20bool\2c\20long\20long\29 +std::__1::__deque_base\2c\20std::__1::allocator\20>\20>::~__deque_base\28\29,operator\20delete\28void*\29 +std::__1::__hash_table\2c\20std::__1::__unordered_map_hasher\2c\20std::__1::hash\2c\20std::__1::equal_to\2c\20true>\2c\20std::__1::__unordered_map_equal\2c\20std::__1::equal_to\2c\20std::__1::hash\2c\20true>\2c\20std::__1::allocator\20>\20>::__rehash\28unsigned\20long\29,operator\20new\28unsigned\20long\29 +std::__1::__hash_table\2c\20std::__1::__unordered_map_hasher\2c\20std::__1::hash\2c\20std::__1::equal_to\2c\20true>\2c\20std::__1::__unordered_map_equal\2c\20std::__1::equal_to\2c\20std::__1::hash\2c\20true>\2c\20std::__1::allocator\20>\20>::__rehash\28unsigned\20long\29,operator\20delete\28void*\29 +std::__1::__hash_table\2c\20std::__1::__unordered_map_hasher\2c\20std::__1::hash\2c\20std::__1::equal_to\2c\20true>\2c\20std::__1::__unordered_map_equal\2c\20std::__1::equal_to\2c\20std::__1::hash\2c\20true>\2c\20std::__1::allocator\20>\20>::__rehash\28unsigned\20long\29,abort +std::__1::deque\2c\20std::__1::allocator\20>\20>::__add_back_capacity\28\29,memmove +std::__1::deque\2c\20std::__1::allocator\20>\20>::__add_back_capacity\28\29,operator\20new\28unsigned\20long\29 +std::__1::deque\2c\20std::__1::allocator\20>\20>::__add_back_capacity\28\29,operator\20delete\28void*\29 +std::__1::deque\2c\20std::__1::allocator\20>\20>::__add_back_capacity\28\29,std::__1::__split_buffer*\2c\20std::__1::allocator*>\20>::push_back\28re2::WalkState*&&\29 +std::__1::deque\2c\20std::__1::allocator\20>\20>::__add_back_capacity\28\29,std::__1::__split_buffer*\2c\20std::__1::allocator*>\20>::push_front\28re2::WalkState*&&\29 +std::__1::deque\2c\20std::__1::allocator\20>\20>::__add_back_capacity\28\29,std::__1::__split_buffer*\2c\20std::__1::allocator*>&>::push_back\28re2::WalkState*&&\29 +std::__1::deque\2c\20std::__1::allocator\20>\20>::__add_back_capacity\28\29,std::__1::__split_buffer*\2c\20std::__1::allocator*>&>::push_front\28re2::WalkState*\20const&\29 +std::__1::deque\2c\20std::__1::allocator\20>\20>::__add_back_capacity\28\29,abort +std::__1::__split_buffer*\2c\20std::__1::allocator*>\20>::push_back\28re2::WalkState*&&\29,memmove +std::__1::__split_buffer*\2c\20std::__1::allocator*>\20>::push_back\28re2::WalkState*&&\29,operator\20new\28unsigned\20long\29 +std::__1::__split_buffer*\2c\20std::__1::allocator*>\20>::push_back\28re2::WalkState*&&\29,operator\20delete\28void*\29 +std::__1::__split_buffer*\2c\20std::__1::allocator*>\20>::push_back\28re2::WalkState*&&\29,abort +std::__1::__split_buffer*\2c\20std::__1::allocator*>\20>::push_front\28re2::WalkState*&&\29,memmove +std::__1::__split_buffer*\2c\20std::__1::allocator*>\20>::push_front\28re2::WalkState*&&\29,operator\20new\28unsigned\20long\29 +std::__1::__split_buffer*\2c\20std::__1::allocator*>\20>::push_front\28re2::WalkState*&&\29,operator\20delete\28void*\29 +std::__1::__split_buffer*\2c\20std::__1::allocator*>\20>::push_front\28re2::WalkState*&&\29,abort +std::__1::__split_buffer*\2c\20std::__1::allocator*>&>::push_back\28re2::WalkState*&&\29,memmove +std::__1::__split_buffer*\2c\20std::__1::allocator*>&>::push_back\28re2::WalkState*&&\29,operator\20new\28unsigned\20long\29 +std::__1::__split_buffer*\2c\20std::__1::allocator*>&>::push_back\28re2::WalkState*&&\29,operator\20delete\28void*\29 +std::__1::__split_buffer*\2c\20std::__1::allocator*>&>::push_back\28re2::WalkState*&&\29,abort +std::__1::__split_buffer*\2c\20std::__1::allocator*>&>::push_front\28re2::WalkState*\20const&\29,memmove +std::__1::__split_buffer*\2c\20std::__1::allocator*>&>::push_front\28re2::WalkState*\20const&\29,operator\20new\28unsigned\20long\29 +std::__1::__split_buffer*\2c\20std::__1::allocator*>&>::push_front\28re2::WalkState*\20const&\29,operator\20delete\28void*\29 +std::__1::__split_buffer*\2c\20std::__1::allocator*>&>::push_front\28re2::WalkState*\20const&\29,abort +re2::DFA::DFA\28re2::Prog*\2c\20re2::Prog::MatchKind\2c\20long\20long\29,operator\20new\28unsigned\20long\29 +re2::DFA::DFA\28re2::Prog*\2c\20re2::Prog::MatchKind\2c\20long\20long\29,abort +re2::DFA::~DFA\28\29,opa_abort +re2::DFA::~DFA\28\29,operator\20delete\28void*\29 +re2::DFA::WorkqToCachedState\28re2::DFA::Workq*\2c\20re2::DFA::Workq*\2c\20unsigned\20int\29,operator\20new\28unsigned\20long\29 +re2::DFA::WorkqToCachedState\28re2::DFA::Workq*\2c\20re2::DFA::Workq*\2c\20unsigned\20int\29,opa_abort +re2::DFA::WorkqToCachedState\28re2::DFA::Workq*\2c\20re2::DFA::Workq*\2c\20unsigned\20int\29,void\20std::__1::__sort&\2c\20int*>\28int*\2c\20int*\2c\20std::__1::__less&\29 +re2::DFA::WorkqToCachedState\28re2::DFA::Workq*\2c\20re2::DFA::Workq*\2c\20unsigned\20int\29,abort +re2::DFA::WorkqToCachedState\28re2::DFA::Workq*\2c\20re2::DFA::Workq*\2c\20unsigned\20int\29,re2::DFA::CachedState\28int*\2c\20int\2c\20unsigned\20int\29 +re2::DFA::WorkqToCachedState\28re2::DFA::Workq*\2c\20re2::DFA::Workq*\2c\20unsigned\20int\29,operator\20delete\28void*\29 +re2::DFA::CachedState\28int*\2c\20int\2c\20unsigned\20int\29,std::__1::__hash_iterator*>\20std::__1::__hash_table\20>::find\28re2::DFA::State*\20const&\29 +re2::DFA::CachedState\28int*\2c\20int\2c\20unsigned\20int\29,operator\20new\28unsigned\20long\29 +re2::DFA::CachedState\28int*\2c\20int\2c\20unsigned\20int\29,memset +re2::DFA::CachedState\28int*\2c\20int\2c\20unsigned\20int\29,memmove +re2::DFA::CachedState\28int*\2c\20int\2c\20unsigned\20int\29,std::__1::pair*>\2c\20bool>\20std::__1::__hash_table\20>::__emplace_unique_key_args\28re2::DFA::State*\20const&\2c\20re2::DFA::State*\20const&\29 +std::__1::__hash_iterator*>\20std::__1::__hash_table\20>::find\28re2::DFA::State*\20const&\29,opa_abort +std::__1::pair*>\2c\20bool>\20std::__1::__hash_table\20>::__emplace_unique_key_args\28re2::DFA::State*\20const&\2c\20re2::DFA::State*\20const&\29,opa_abort +std::__1::pair*>\2c\20bool>\20std::__1::__hash_table\20>::__emplace_unique_key_args\28re2::DFA::State*\20const&\2c\20re2::DFA::State*\20const&\29,operator\20new\28unsigned\20long\29 +std::__1::pair*>\2c\20bool>\20std::__1::__hash_table\20>::__emplace_unique_key_args\28re2::DFA::State*\20const&\2c\20re2::DFA::State*\20const&\29,std::__1::__next_prime\28unsigned\20long\29 +std::__1::pair*>\2c\20bool>\20std::__1::__hash_table\20>::__emplace_unique_key_args\28re2::DFA::State*\20const&\2c\20re2::DFA::State*\20const&\29,std::__1::__hash_table\20>::__rehash\28unsigned\20long\29 +re2::SparseSetT::InsertInternal\28bool\2c\20int\29,opa_abort +re2::SparseSetT::InsertInternal\28bool\2c\20int\29,re2::SparseSetT::create_index\28int\29 +re2::DFA::AddToQueue\28re2::DFA::Workq*\2c\20int\2c\20unsigned\20int\29,opa_abort +re2::DFA::AddToQueue\28re2::DFA::Workq*\2c\20int\2c\20unsigned\20int\29,re2::SparseSetT::InsertInternal\28bool\2c\20int\29 +re2::DFA::RunWorkqOnByte\28re2::DFA::Workq*\2c\20re2::DFA::Workq*\2c\20int\2c\20unsigned\20int\2c\20bool*\29,re2::SparseSetT::InsertInternal\28bool\2c\20int\29 +re2::DFA::RunWorkqOnByte\28re2::DFA::Workq*\2c\20re2::DFA::Workq*\2c\20int\2c\20unsigned\20int\2c\20bool*\29,re2::DFA::AddToQueue\28re2::DFA::Workq*\2c\20int\2c\20unsigned\20int\29 +re2::DFA::RunWorkqOnByte\28re2::DFA::Workq*\2c\20re2::DFA::Workq*\2c\20int\2c\20unsigned\20int\2c\20bool*\29,opa_abort +re2::DFA::RunStateOnByte\28re2::DFA::State*\2c\20int\29,re2::SparseSetT::InsertInternal\28bool\2c\20int\29 +re2::DFA::RunStateOnByte\28re2::DFA::State*\2c\20int\29,re2::DFA::AddToQueue\28re2::DFA::Workq*\2c\20int\2c\20unsigned\20int\29 +re2::DFA::RunStateOnByte\28re2::DFA::State*\2c\20int\29,re2::DFA::RunWorkqOnByte\28re2::DFA::Workq*\2c\20re2::DFA::Workq*\2c\20int\2c\20unsigned\20int\2c\20bool*\29 +re2::DFA::RunStateOnByte\28re2::DFA::State*\2c\20int\29,re2::DFA::WorkqToCachedState\28re2::DFA::Workq*\2c\20re2::DFA::Workq*\2c\20unsigned\20int\29 +re2::DFA::ResetCache\28re2::DFA::RWLocker*\29,re2::hooks::GetDFAStateCacheResetHook\28\29 +re2::DFA::ResetCache\28re2::DFA::RWLocker*\29,operator\20delete\28void*\29 +re2::DFA::SearchFFF\28re2::DFA::SearchParams*\29,bool\20re2::DFA::InlinedSearchLoop\28re2::DFA::SearchParams*\29 +bool\20re2::DFA::InlinedSearchLoop\28re2::DFA::SearchParams*\29,re2::SparseSetT::InsertInternal\28bool\2c\20int\29 +bool\20re2::DFA::InlinedSearchLoop\28re2::DFA::SearchParams*\29,re2::DFA::RunStateOnByte\28re2::DFA::State*\2c\20int\29 +bool\20re2::DFA::InlinedSearchLoop\28re2::DFA::SearchParams*\29,operator\20new\5b\5d\28unsigned\20long\29 +bool\20re2::DFA::InlinedSearchLoop\28re2::DFA::SearchParams*\29,memmove +bool\20re2::DFA::InlinedSearchLoop\28re2::DFA::SearchParams*\29,re2::DFA::ResetCache\28re2::DFA::RWLocker*\29 +bool\20re2::DFA::InlinedSearchLoop\28re2::DFA::SearchParams*\29,re2::DFA::CachedState\28int*\2c\20int\2c\20unsigned\20int\29 +bool\20re2::DFA::InlinedSearchLoop\28re2::DFA::SearchParams*\29,operator\20delete\5b\5d\28void*\29 +re2::DFA::SearchFFT\28re2::DFA::SearchParams*\29,bool\20re2::DFA::InlinedSearchLoop\28re2::DFA::SearchParams*\29 +bool\20re2::DFA::InlinedSearchLoop\28re2::DFA::SearchParams*\29,re2::SparseSetT::InsertInternal\28bool\2c\20int\29 +bool\20re2::DFA::InlinedSearchLoop\28re2::DFA::SearchParams*\29,re2::DFA::RunStateOnByte\28re2::DFA::State*\2c\20int\29 +bool\20re2::DFA::InlinedSearchLoop\28re2::DFA::SearchParams*\29,operator\20new\5b\5d\28unsigned\20long\29 +bool\20re2::DFA::InlinedSearchLoop\28re2::DFA::SearchParams*\29,memmove +bool\20re2::DFA::InlinedSearchLoop\28re2::DFA::SearchParams*\29,re2::DFA::ResetCache\28re2::DFA::RWLocker*\29 +bool\20re2::DFA::InlinedSearchLoop\28re2::DFA::SearchParams*\29,re2::DFA::CachedState\28int*\2c\20int\2c\20unsigned\20int\29 +bool\20re2::DFA::InlinedSearchLoop\28re2::DFA::SearchParams*\29,operator\20delete\5b\5d\28void*\29 +re2::DFA::SearchFTF\28re2::DFA::SearchParams*\29,bool\20re2::DFA::InlinedSearchLoop\28re2::DFA::SearchParams*\29 +bool\20re2::DFA::InlinedSearchLoop\28re2::DFA::SearchParams*\29,re2::DFA::RunStateOnByte\28re2::DFA::State*\2c\20int\29 +bool\20re2::DFA::InlinedSearchLoop\28re2::DFA::SearchParams*\29,operator\20new\5b\5d\28unsigned\20long\29 +bool\20re2::DFA::InlinedSearchLoop\28re2::DFA::SearchParams*\29,memmove +bool\20re2::DFA::InlinedSearchLoop\28re2::DFA::SearchParams*\29,re2::DFA::ResetCache\28re2::DFA::RWLocker*\29 +bool\20re2::DFA::InlinedSearchLoop\28re2::DFA::SearchParams*\29,re2::DFA::CachedState\28int*\2c\20int\2c\20unsigned\20int\29 +bool\20re2::DFA::InlinedSearchLoop\28re2::DFA::SearchParams*\29,operator\20delete\5b\5d\28void*\29 +bool\20re2::DFA::InlinedSearchLoop\28re2::DFA::SearchParams*\29,re2::SparseSetT::InsertInternal\28bool\2c\20int\29 +re2::DFA::SearchFTT\28re2::DFA::SearchParams*\29,bool\20re2::DFA::InlinedSearchLoop\28re2::DFA::SearchParams*\29 +bool\20re2::DFA::InlinedSearchLoop\28re2::DFA::SearchParams*\29,re2::DFA::RunStateOnByte\28re2::DFA::State*\2c\20int\29 +bool\20re2::DFA::InlinedSearchLoop\28re2::DFA::SearchParams*\29,operator\20new\5b\5d\28unsigned\20long\29 +bool\20re2::DFA::InlinedSearchLoop\28re2::DFA::SearchParams*\29,memmove +bool\20re2::DFA::InlinedSearchLoop\28re2::DFA::SearchParams*\29,re2::DFA::ResetCache\28re2::DFA::RWLocker*\29 +bool\20re2::DFA::InlinedSearchLoop\28re2::DFA::SearchParams*\29,re2::DFA::CachedState\28int*\2c\20int\2c\20unsigned\20int\29 +bool\20re2::DFA::InlinedSearchLoop\28re2::DFA::SearchParams*\29,operator\20delete\5b\5d\28void*\29 +bool\20re2::DFA::InlinedSearchLoop\28re2::DFA::SearchParams*\29,re2::SparseSetT::InsertInternal\28bool\2c\20int\29 +re2::DFA::SearchTFF\28re2::DFA::SearchParams*\29,bool\20re2::DFA::InlinedSearchLoop\28re2::DFA::SearchParams*\29 +bool\20re2::DFA::InlinedSearchLoop\28re2::DFA::SearchParams*\29,re2::SparseSetT::InsertInternal\28bool\2c\20int\29 +bool\20re2::DFA::InlinedSearchLoop\28re2::DFA::SearchParams*\29,opa_abort +bool\20re2::DFA::InlinedSearchLoop\28re2::DFA::SearchParams*\29,memchr +bool\20re2::DFA::InlinedSearchLoop\28re2::DFA::SearchParams*\29,re2::Prog::PrefixAccel_FrontAndBack\28void\20const*\2c\20unsigned\20long\29 +bool\20re2::DFA::InlinedSearchLoop\28re2::DFA::SearchParams*\29,re2::DFA::RunStateOnByte\28re2::DFA::State*\2c\20int\29 +bool\20re2::DFA::InlinedSearchLoop\28re2::DFA::SearchParams*\29,operator\20new\5b\5d\28unsigned\20long\29 +bool\20re2::DFA::InlinedSearchLoop\28re2::DFA::SearchParams*\29,memmove +bool\20re2::DFA::InlinedSearchLoop\28re2::DFA::SearchParams*\29,re2::DFA::ResetCache\28re2::DFA::RWLocker*\29 +bool\20re2::DFA::InlinedSearchLoop\28re2::DFA::SearchParams*\29,re2::DFA::CachedState\28int*\2c\20int\2c\20unsigned\20int\29 +bool\20re2::DFA::InlinedSearchLoop\28re2::DFA::SearchParams*\29,operator\20delete\5b\5d\28void*\29 +re2::DFA::SearchTFT\28re2::DFA::SearchParams*\29,bool\20re2::DFA::InlinedSearchLoop\28re2::DFA::SearchParams*\29 +bool\20re2::DFA::InlinedSearchLoop\28re2::DFA::SearchParams*\29,re2::SparseSetT::InsertInternal\28bool\2c\20int\29 +bool\20re2::DFA::InlinedSearchLoop\28re2::DFA::SearchParams*\29,opa_abort +bool\20re2::DFA::InlinedSearchLoop\28re2::DFA::SearchParams*\29,memchr +bool\20re2::DFA::InlinedSearchLoop\28re2::DFA::SearchParams*\29,re2::Prog::PrefixAccel_FrontAndBack\28void\20const*\2c\20unsigned\20long\29 +bool\20re2::DFA::InlinedSearchLoop\28re2::DFA::SearchParams*\29,re2::DFA::RunStateOnByte\28re2::DFA::State*\2c\20int\29 +bool\20re2::DFA::InlinedSearchLoop\28re2::DFA::SearchParams*\29,operator\20new\5b\5d\28unsigned\20long\29 +bool\20re2::DFA::InlinedSearchLoop\28re2::DFA::SearchParams*\29,memmove +bool\20re2::DFA::InlinedSearchLoop\28re2::DFA::SearchParams*\29,re2::DFA::ResetCache\28re2::DFA::RWLocker*\29 +bool\20re2::DFA::InlinedSearchLoop\28re2::DFA::SearchParams*\29,re2::DFA::CachedState\28int*\2c\20int\2c\20unsigned\20int\29 +bool\20re2::DFA::InlinedSearchLoop\28re2::DFA::SearchParams*\29,operator\20delete\5b\5d\28void*\29 +re2::DFA::SearchTTF\28re2::DFA::SearchParams*\29,bool\20re2::DFA::InlinedSearchLoop\28re2::DFA::SearchParams*\29 +bool\20re2::DFA::InlinedSearchLoop\28re2::DFA::SearchParams*\29,opa_abort +bool\20re2::DFA::InlinedSearchLoop\28re2::DFA::SearchParams*\29,memchr +bool\20re2::DFA::InlinedSearchLoop\28re2::DFA::SearchParams*\29,re2::Prog::PrefixAccel_FrontAndBack\28void\20const*\2c\20unsigned\20long\29 +bool\20re2::DFA::InlinedSearchLoop\28re2::DFA::SearchParams*\29,re2::DFA::RunStateOnByte\28re2::DFA::State*\2c\20int\29 +bool\20re2::DFA::InlinedSearchLoop\28re2::DFA::SearchParams*\29,operator\20new\5b\5d\28unsigned\20long\29 +bool\20re2::DFA::InlinedSearchLoop\28re2::DFA::SearchParams*\29,memmove +bool\20re2::DFA::InlinedSearchLoop\28re2::DFA::SearchParams*\29,re2::DFA::ResetCache\28re2::DFA::RWLocker*\29 +bool\20re2::DFA::InlinedSearchLoop\28re2::DFA::SearchParams*\29,re2::DFA::CachedState\28int*\2c\20int\2c\20unsigned\20int\29 +bool\20re2::DFA::InlinedSearchLoop\28re2::DFA::SearchParams*\29,operator\20delete\5b\5d\28void*\29 +bool\20re2::DFA::InlinedSearchLoop\28re2::DFA::SearchParams*\29,re2::SparseSetT::InsertInternal\28bool\2c\20int\29 +re2::DFA::SearchTTT\28re2::DFA::SearchParams*\29,bool\20re2::DFA::InlinedSearchLoop\28re2::DFA::SearchParams*\29 +bool\20re2::DFA::InlinedSearchLoop\28re2::DFA::SearchParams*\29,opa_abort +bool\20re2::DFA::InlinedSearchLoop\28re2::DFA::SearchParams*\29,memchr +bool\20re2::DFA::InlinedSearchLoop\28re2::DFA::SearchParams*\29,re2::Prog::PrefixAccel_FrontAndBack\28void\20const*\2c\20unsigned\20long\29 +bool\20re2::DFA::InlinedSearchLoop\28re2::DFA::SearchParams*\29,re2::DFA::RunStateOnByte\28re2::DFA::State*\2c\20int\29 +bool\20re2::DFA::InlinedSearchLoop\28re2::DFA::SearchParams*\29,operator\20new\5b\5d\28unsigned\20long\29 +bool\20re2::DFA::InlinedSearchLoop\28re2::DFA::SearchParams*\29,memmove +bool\20re2::DFA::InlinedSearchLoop\28re2::DFA::SearchParams*\29,re2::DFA::ResetCache\28re2::DFA::RWLocker*\29 +bool\20re2::DFA::InlinedSearchLoop\28re2::DFA::SearchParams*\29,re2::DFA::CachedState\28int*\2c\20int\2c\20unsigned\20int\29 +bool\20re2::DFA::InlinedSearchLoop\28re2::DFA::SearchParams*\29,operator\20delete\5b\5d\28void*\29 +bool\20re2::DFA::InlinedSearchLoop\28re2::DFA::SearchParams*\29,re2::SparseSetT::InsertInternal\28bool\2c\20int\29 +re2::DFA::AnalyzeSearch\28re2::DFA::SearchParams*\29,re2::DFA::AddToQueue\28re2::DFA::Workq*\2c\20int\2c\20unsigned\20int\29 +re2::DFA::AnalyzeSearch\28re2::DFA::SearchParams*\29,re2::DFA::WorkqToCachedState\28re2::DFA::Workq*\2c\20re2::DFA::Workq*\2c\20unsigned\20int\29 +re2::DFA::AnalyzeSearch\28re2::DFA::SearchParams*\29,re2::DFA::ResetCache\28re2::DFA::RWLocker*\29 +re2::DFA::Search\28re2::StringPiece\20const&\2c\20re2::StringPiece\20const&\2c\20bool\2c\20bool\2c\20bool\2c\20bool*\2c\20char\20const**\2c\20re2::SparseSetT*\29,re2::DFA::AnalyzeSearch\28re2::DFA::SearchParams*\29 +re2::Prog::GetDFA\28re2::Prog::MatchKind\29,std::__1::__call_once\28unsigned\20long\20volatile&\2c\20void*\2c\20void\20\28*\29\28void*\29\29 +void\20std::__1::__call_once_proxy\20>\28void*\29,operator\20new\28unsigned\20long\29 +void\20std::__1::__call_once_proxy\20>\28void*\29,re2::DFA::DFA\28re2::Prog*\2c\20re2::Prog::MatchKind\2c\20long\20long\29 +void\20std::__1::__call_once_proxy\20>\28void*\29,operator\20new\28unsigned\20long\29 +void\20std::__1::__call_once_proxy\20>\28void*\29,re2::DFA::DFA\28re2::Prog*\2c\20re2::Prog::MatchKind\2c\20long\20long\29 +void\20std::__1::__call_once_proxy\20>\28void*\29,operator\20new\28unsigned\20long\29 +void\20std::__1::__call_once_proxy\20>\28void*\29,re2::DFA::DFA\28re2::Prog*\2c\20re2::Prog::MatchKind\2c\20long\20long\29 +re2::Prog::DeleteDFA\28re2::DFA*\29,re2::DFA::~DFA\28\29 +re2::Prog::DeleteDFA\28re2::DFA*\29,operator\20delete\28void*\29 +re2::Prog::SearchDFA\28re2::StringPiece\20const&\2c\20re2::StringPiece\20const&\2c\20re2::Prog::Anchor\2c\20re2::Prog::MatchKind\2c\20re2::StringPiece*\2c\20bool*\2c\20re2::SparseSetT*\29,re2::Prog::GetDFA\28re2::Prog::MatchKind\29 +re2::Prog::SearchDFA\28re2::StringPiece\20const&\2c\20re2::StringPiece\20const&\2c\20re2::Prog::Anchor\2c\20re2::Prog::MatchKind\2c\20re2::StringPiece*\2c\20bool*\2c\20re2::SparseSetT*\29,re2::DFA::Search\28re2::StringPiece\20const&\2c\20re2::StringPiece\20const&\2c\20bool\2c\20bool\2c\20bool\2c\20bool*\2c\20char\20const**\2c\20re2::SparseSetT*\29 +re2::Prog::SearchDFA\28re2::StringPiece\20const&\2c\20re2::StringPiece\20const&\2c\20re2::Prog::Anchor\2c\20re2::Prog::MatchKind\2c\20re2::StringPiece*\2c\20bool*\2c\20re2::SparseSetT*\29,re2::hooks::GetDFASearchFailureHook\28\29 +re2::SparseSetT::create_index\28int\29,opa_abort +std::__1::__hash_table\20>::__rehash\28unsigned\20long\29,operator\20new\28unsigned\20long\29 +std::__1::__hash_table\20>::__rehash\28unsigned\20long\29,operator\20delete\28void*\29 +std::__1::__hash_table\20>::__rehash\28unsigned\20long\29,opa_abort +std::__1::__hash_table\20>::__rehash\28unsigned\20long\29,abort +re2::NFA::NFA\28re2::Prog*\29,memset +re2::NFA::NFA\28re2::Prog*\29,re2::SparseArray::resize\28int\29 +re2::NFA::NFA\28re2::Prog*\29,operator\20new\28unsigned\20long\29 +re2::NFA::NFA\28re2::Prog*\29,operator\20delete\28void*\29 +re2::NFA::NFA\28re2::Prog*\29,abort +re2::SparseArray::resize\28int\29,opa_abort +re2::SparseArray::resize\28int\29,operator\20new\28unsigned\20long\29 +re2::SparseArray::resize\28int\29,memmove +re2::SparseArray::resize\28int\29,operator\20delete\28void*\29 +re2::SparseArray::resize\28int\29,abort +re2::NFA::~NFA\28\29,operator\20delete\5b\5d\28void*\29 +re2::NFA::~NFA\28\29,std::__1::__deque_base\20>::~__deque_base\28\29 +re2::NFA::~NFA\28\29,operator\20delete\28void*\29 +re2::NFA::~NFA\28\29,opa_abort +std::__1::__deque_base\20>::~__deque_base\28\29,operator\20delete\28void*\29 +re2::NFA::AddToThreadq\28re2::SparseArray*\2c\20int\2c\20int\2c\20re2::StringPiece\20const&\2c\20char\20const*\2c\20re2::NFA::Thread*\29,opa_abort +re2::NFA::AddToThreadq\28re2::SparseArray*\2c\20int\2c\20int\2c\20re2::StringPiece\20const&\2c\20char\20const*\2c\20re2::NFA::Thread*\29,std::__1::deque\20>::__add_back_capacity\28\29 +re2::NFA::AddToThreadq\28re2::SparseArray*\2c\20int\2c\20int\2c\20re2::StringPiece\20const&\2c\20char\20const*\2c\20re2::NFA::Thread*\29,operator\20new\5b\5d\28unsigned\20long\29 +re2::NFA::AddToThreadq\28re2::SparseArray*\2c\20int\2c\20int\2c\20re2::StringPiece\20const&\2c\20char\20const*\2c\20re2::NFA::Thread*\29,memmove +re2::NFA::AddToThreadq\28re2::SparseArray*\2c\20int\2c\20int\2c\20re2::StringPiece\20const&\2c\20char\20const*\2c\20re2::NFA::Thread*\29,re2::Prog::EmptyFlags\28re2::StringPiece\20const&\2c\20char\20const*\29 +std::__1::deque\20>::__add_back_capacity\28\29,memmove +std::__1::deque\20>::__add_back_capacity\28\29,operator\20new\28unsigned\20long\29 +std::__1::deque\20>::__add_back_capacity\28\29,operator\20delete\28void*\29 +std::__1::deque\20>::__add_back_capacity\28\29,std::__1::__split_buffer\20>::push_back\28re2::NFA::Thread*&&\29 +std::__1::deque\20>::__add_back_capacity\28\29,std::__1::__split_buffer\20>::push_front\28re2::NFA::Thread*&&\29 +std::__1::deque\20>::__add_back_capacity\28\29,std::__1::__split_buffer&>::push_back\28re2::NFA::Thread*&&\29 +std::__1::deque\20>::__add_back_capacity\28\29,std::__1::__split_buffer&>::push_front\28re2::NFA::Thread*\20const&\29 +std::__1::deque\20>::__add_back_capacity\28\29,abort +re2::NFA::Step\28re2::SparseArray*\2c\20re2::SparseArray*\2c\20int\2c\20re2::StringPiece\20const&\2c\20char\20const*\29,memmove +re2::NFA::Step\28re2::SparseArray*\2c\20re2::SparseArray*\2c\20int\2c\20re2::StringPiece\20const&\2c\20char\20const*\29,opa_abort +re2::NFA::Step\28re2::SparseArray*\2c\20re2::SparseArray*\2c\20int\2c\20re2::StringPiece\20const&\2c\20char\20const*\29,re2::NFA::AddToThreadq\28re2::SparseArray*\2c\20int\2c\20int\2c\20re2::StringPiece\20const&\2c\20char\20const*\2c\20re2::NFA::Thread*\29 +re2::NFA::Search\28re2::StringPiece\20const&\2c\20re2::StringPiece\20const&\2c\20bool\2c\20bool\2c\20re2::StringPiece*\2c\20int\29,operator\20new\5b\5d\28unsigned\20long\29 +re2::NFA::Search\28re2::StringPiece\20const&\2c\20re2::StringPiece\20const&\2c\20bool\2c\20bool\2c\20re2::StringPiece*\2c\20int\29,memset +re2::NFA::Search\28re2::StringPiece\20const&\2c\20re2::StringPiece\20const&\2c\20bool\2c\20bool\2c\20re2::StringPiece*\2c\20int\29,re2::NFA::Step\28re2::SparseArray*\2c\20re2::SparseArray*\2c\20int\2c\20re2::StringPiece\20const&\2c\20char\20const*\29 +re2::NFA::Search\28re2::StringPiece\20const&\2c\20re2::StringPiece\20const&\2c\20bool\2c\20bool\2c\20re2::StringPiece*\2c\20int\29,opa_abort +re2::NFA::Search\28re2::StringPiece\20const&\2c\20re2::StringPiece\20const&\2c\20bool\2c\20bool\2c\20re2::StringPiece*\2c\20int\29,re2::Prog::PrefixAccel\28void\20const*\2c\20unsigned\20long\29 +re2::NFA::Search\28re2::StringPiece\20const&\2c\20re2::StringPiece\20const&\2c\20bool\2c\20bool\2c\20re2::StringPiece*\2c\20int\29,std::__1::deque\20>::__add_back_capacity\28\29 +re2::NFA::Search\28re2::StringPiece\20const&\2c\20re2::StringPiece\20const&\2c\20bool\2c\20bool\2c\20re2::StringPiece*\2c\20int\29,memmove +re2::NFA::Search\28re2::StringPiece\20const&\2c\20re2::StringPiece\20const&\2c\20bool\2c\20bool\2c\20re2::StringPiece*\2c\20int\29,re2::NFA::AddToThreadq\28re2::SparseArray*\2c\20int\2c\20int\2c\20re2::StringPiece\20const&\2c\20char\20const*\2c\20re2::NFA::Thread*\29 +re2::Prog::PrefixAccel\28void\20const*\2c\20unsigned\20long\29,opa_abort +re2::Prog::PrefixAccel\28void\20const*\2c\20unsigned\20long\29,memchr +re2::Prog::PrefixAccel\28void\20const*\2c\20unsigned\20long\29,re2::Prog::PrefixAccel_FrontAndBack\28void\20const*\2c\20unsigned\20long\29 +re2::Prog::SearchNFA\28re2::StringPiece\20const&\2c\20re2::StringPiece\20const&\2c\20re2::Prog::Anchor\2c\20re2::Prog::MatchKind\2c\20re2::StringPiece*\2c\20int\29,re2::NFA::NFA\28re2::Prog*\29 +re2::Prog::SearchNFA\28re2::StringPiece\20const&\2c\20re2::StringPiece\20const&\2c\20re2::Prog::Anchor\2c\20re2::Prog::MatchKind\2c\20re2::StringPiece*\2c\20int\29,re2::NFA::Search\28re2::StringPiece\20const&\2c\20re2::StringPiece\20const&\2c\20bool\2c\20bool\2c\20re2::StringPiece*\2c\20int\29 +re2::Prog::SearchNFA\28re2::StringPiece\20const&\2c\20re2::StringPiece\20const&\2c\20re2::Prog::Anchor\2c\20re2::Prog::MatchKind\2c\20re2::StringPiece*\2c\20int\29,re2::NFA::~NFA\28\29 +re2::SparseArray::SetInternal\28bool\2c\20int\2c\20int\20const&\29,opa_abort +re2::SparseArray::SetInternal\28bool\2c\20int\2c\20int\20const&\29,re2::SparseArray::create_index\28int\29 +re2::SparseArray::SetInternal\28bool\2c\20int\2c\20int\20const&\29,re2::SparseArray::SetExistingInternal\28int\2c\20int\20const&\29 +re2::SparseArray::create_index\28int\29,opa_abort +re2::SparseArray::SetExistingInternal\28int\2c\20int\20const&\29,opa_abort +std::__1::__split_buffer\20>::push_back\28re2::NFA::Thread*&&\29,memmove +std::__1::__split_buffer\20>::push_back\28re2::NFA::Thread*&&\29,operator\20new\28unsigned\20long\29 +std::__1::__split_buffer\20>::push_back\28re2::NFA::Thread*&&\29,operator\20delete\28void*\29 +std::__1::__split_buffer\20>::push_back\28re2::NFA::Thread*&&\29,abort +std::__1::__split_buffer\20>::push_front\28re2::NFA::Thread*&&\29,memmove +std::__1::__split_buffer\20>::push_front\28re2::NFA::Thread*&&\29,operator\20new\28unsigned\20long\29 +std::__1::__split_buffer\20>::push_front\28re2::NFA::Thread*&&\29,operator\20delete\28void*\29 +std::__1::__split_buffer\20>::push_front\28re2::NFA::Thread*&&\29,abort +std::__1::__split_buffer&>::push_back\28re2::NFA::Thread*&&\29,memmove +std::__1::__split_buffer&>::push_back\28re2::NFA::Thread*&&\29,operator\20new\28unsigned\20long\29 +std::__1::__split_buffer&>::push_back\28re2::NFA::Thread*&&\29,operator\20delete\28void*\29 +std::__1::__split_buffer&>::push_back\28re2::NFA::Thread*&&\29,abort +std::__1::__split_buffer&>::push_front\28re2::NFA::Thread*\20const&\29,memmove +std::__1::__split_buffer&>::push_front\28re2::NFA::Thread*\20const&\29,operator\20new\28unsigned\20long\29 +std::__1::__split_buffer&>::push_front\28re2::NFA::Thread*\20const&\29,operator\20delete\28void*\29 +std::__1::__split_buffer&>::push_front\28re2::NFA::Thread*\20const&\29,abort +re2::Prog::SearchOnePass\28re2::StringPiece\20const&\2c\20re2::StringPiece\20const&\2c\20re2::Prog::Anchor\2c\20re2::Prog::MatchKind\2c\20re2::StringPiece*\2c\20int\29,memset +re2::Prog::SearchOnePass\28re2::StringPiece\20const&\2c\20re2::StringPiece\20const&\2c\20re2::Prog::Anchor\2c\20re2::Prog::MatchKind\2c\20re2::StringPiece*\2c\20int\29,re2::Prog::EmptyFlags\28re2::StringPiece\20const&\2c\20char\20const*\29 +re2::Prog::SearchOnePass\28re2::StringPiece\20const&\2c\20re2::StringPiece\20const&\2c\20re2::Prog::Anchor\2c\20re2::Prog::MatchKind\2c\20re2::StringPiece*\2c\20int\29,memcpy +re2::Prog::IsOnePass\28\29,operator\20new\28unsigned\20long\29 +re2::Prog::IsOnePass\28\29,memset +re2::Prog::IsOnePass\28\29,opa_abort +re2::Prog::IsOnePass\28\29,re2::SparseSetT::InsertInternal\28bool\2c\20int\29 +re2::Prog::IsOnePass\28\29,std::__1::vector\20>::insert\28std::__1::__wrap_iter\2c\20unsigned\20long\2c\20unsigned\20char\20const&\29 +re2::Prog::IsOnePass\28\29,operator\20delete\28void*\29 +re2::Prog::IsOnePass\28\29,memmove +re2::Prog::IsOnePass\28\29,abort +std::__1::vector\20>::insert\28std::__1::__wrap_iter\2c\20unsigned\20long\2c\20unsigned\20char\20const&\29,memmove +std::__1::vector\20>::insert\28std::__1::__wrap_iter\2c\20unsigned\20long\2c\20unsigned\20char\20const&\29,operator\20new\28unsigned\20long\29 +std::__1::vector\20>::insert\28std::__1::__wrap_iter\2c\20unsigned\20long\2c\20unsigned\20char\20const&\29,memcpy +std::__1::vector\20>::insert\28std::__1::__wrap_iter\2c\20unsigned\20long\2c\20unsigned\20char\20const&\29,operator\20delete\28void*\29 +std::__1::vector\20>::insert\28std::__1::__wrap_iter\2c\20unsigned\20long\2c\20unsigned\20char\20const&\29,abort +std::__1::__tree\20>::destroy\28std::__1::__tree_node*\29,std::__1::__tree\20>::destroy\28std::__1::__tree_node*\29 +std::__1::__tree\20>::destroy\28std::__1::__tree_node*\29,operator\20delete\28void*\29 +re2::Regexp::ParseState::PushRegexp\28re2::Regexp*\29,re2::Regexp::ParseState::MaybeConcatString\28int\2c\20re2::Regexp::ParseFlags\29 +re2::Regexp::ParseState::PushRegexp\28re2::Regexp*\29,re2::CharClassBuilder::RemoveAbove\28int\29 +re2::Regexp::ParseState::PushRegexp\28re2::Regexp*\29,re2::Regexp::Decref\28\29 +re2::Regexp::ParseState::PushRegexp\28re2::Regexp*\29,operator\20new\28unsigned\20long\29 +re2::Regexp::ParseState::PushRegexp\28re2::Regexp*\29,re2::Regexp::Regexp\28re2::RegexpOp\2c\20re2::Regexp::ParseFlags\29 +re2::Regexp::ParseState::PushRegexp\28re2::Regexp*\29,re2::CharClassBuilder::Contains\28int\29 +re2::Regexp::ParseState::PushRegexp\28re2::Regexp*\29,re2::Regexp::ComputeSimple\28\29 +re2::Regexp::ParseState::MaybeConcatString\28int\2c\20re2::Regexp::ParseFlags\29,re2::Regexp::AddRuneToString\28int\29 +re2::Regexp::ParseState::MaybeConcatString\28int\2c\20re2::Regexp::ParseFlags\29,operator\20delete\5b\5d\28void*\29 +re2::Regexp::ParseState::MaybeConcatString\28int\2c\20re2::Regexp::ParseFlags\29,re2::Regexp::Decref\28\29 +re2::Regexp::ParseState::PushLiteral\28int\29,re2::CycleFoldRune\28int\29 +re2::Regexp::ParseState::PushLiteral\28int\29,operator\20new\28unsigned\20long\29 +re2::Regexp::ParseState::PushLiteral\28int\29,re2::Regexp::Regexp\28re2::RegexpOp\2c\20re2::Regexp::ParseFlags\29 +re2::Regexp::ParseState::PushLiteral\28int\29,re2::CharClassBuilder::CharClassBuilder\28\29 +re2::Regexp::ParseState::PushLiteral\28int\29,re2::CharClassBuilder::AddRange\28int\2c\20int\29 +re2::Regexp::ParseState::PushLiteral\28int\29,re2::Regexp::ParseState::MaybeConcatString\28int\2c\20re2::Regexp::ParseFlags\29 +re2::Regexp::ParseState::PushLiteral\28int\29,re2::Regexp::ParseState::PushRegexp\28re2::Regexp*\29 +re2::Regexp::ParseState::PushSimpleOp\28re2::RegexpOp\29,operator\20new\28unsigned\20long\29 +re2::Regexp::ParseState::PushSimpleOp\28re2::RegexpOp\29,re2::Regexp::Regexp\28re2::RegexpOp\2c\20re2::Regexp::ParseFlags\29 +re2::Regexp::ParseState::PushSimpleOp\28re2::RegexpOp\29,re2::Regexp::ParseState::PushRegexp\28re2::Regexp*\29 +re2::Regexp::ParseState::PushDot\28\29,operator\20new\28unsigned\20long\29 +re2::Regexp::ParseState::PushDot\28\29,re2::Regexp::Regexp\28re2::RegexpOp\2c\20re2::Regexp::ParseFlags\29 +re2::Regexp::ParseState::PushDot\28\29,re2::Regexp::ParseState::PushRegexp\28re2::Regexp*\29 +re2::Regexp::ParseState::PushDot\28\29,re2::CharClassBuilder::CharClassBuilder\28\29 +re2::Regexp::ParseState::PushDot\28\29,re2::CharClassBuilder::AddRange\28int\2c\20int\29 +re2::Regexp::ParseState::PushRepeatOp\28re2::RegexpOp\2c\20re2::StringPiece\20const&\2c\20bool\29,operator\20new\28unsigned\20long\29 +re2::Regexp::ParseState::PushRepeatOp\28re2::RegexpOp\2c\20re2::StringPiece\20const&\2c\20bool\29,re2::Regexp::Regexp\28re2::RegexpOp\2c\20re2::Regexp::ParseFlags\29 +re2::Regexp::ParseState::PushRepeatOp\28re2::RegexpOp\2c\20re2::StringPiece\20const&\2c\20bool\29,re2::CharClassBuilder::GetCharClass\28\29 +re2::Regexp::ParseState::PushRepeatOp\28re2::RegexpOp\2c\20re2::StringPiece\20const&\2c\20bool\29,std::__1::__tree\20>::destroy\28std::__1::__tree_node*\29 +re2::Regexp::ParseState::PushRepeatOp\28re2::RegexpOp\2c\20re2::StringPiece\20const&\2c\20bool\29,operator\20delete\28void*\29 +re2::Regexp::ParseState::PushRepeatOp\28re2::RegexpOp\2c\20re2::StringPiece\20const&\2c\20bool\29,re2::Regexp::ComputeSimple\28\29 +re2::Regexp::ParseState::PushRepetition\28int\2c\20int\2c\20re2::StringPiece\20const&\2c\20bool\29,operator\20new\28unsigned\20long\29 +re2::Regexp::ParseState::PushRepetition\28int\2c\20int\2c\20re2::StringPiece\20const&\2c\20bool\29,re2::Regexp::Regexp\28re2::RegexpOp\2c\20re2::Regexp::ParseFlags\29 +re2::Regexp::ParseState::PushRepetition\28int\2c\20int\2c\20re2::StringPiece\20const&\2c\20bool\29,re2::CharClassBuilder::GetCharClass\28\29 +re2::Regexp::ParseState::PushRepetition\28int\2c\20int\2c\20re2::StringPiece\20const&\2c\20bool\29,std::__1::__tree\20>::destroy\28std::__1::__tree_node*\29 +re2::Regexp::ParseState::PushRepetition\28int\2c\20int\2c\20re2::StringPiece\20const&\2c\20bool\29,operator\20delete\28void*\29 +re2::Regexp::ParseState::PushRepetition\28int\2c\20int\2c\20re2::StringPiece\20const&\2c\20bool\29,re2::Regexp::ComputeSimple\28\29 +re2::Regexp::ParseState::PushRepetition\28int\2c\20int\2c\20re2::StringPiece\20const&\2c\20bool\29,re2::Regexp::Walker::WalkInternal\28re2::Regexp*\2c\20int\2c\20bool\29 +re2::Regexp::ParseState::PushRepetition\28int\2c\20int\2c\20re2::StringPiece\20const&\2c\20bool\29,re2::Regexp::Walker::~Walker\28\29 +re2::Regexp::Walker::WalkInternal\28re2::Regexp*\2c\20int\2c\20bool\29,operator\20delete\5b\5d\28void*\29 +re2::Regexp::Walker::WalkInternal\28re2::Regexp*\2c\20int\2c\20bool\29,operator\20delete\28void*\29 +re2::Regexp::Walker::WalkInternal\28re2::Regexp*\2c\20int\2c\20bool\29,std::__1::deque\2c\20std::__1::allocator\20>\20>::__add_back_capacity\28\29 +re2::Regexp::Walker::WalkInternal\28re2::Regexp*\2c\20int\2c\20bool\29,operator\20new\5b\5d\28unsigned\20long\29 +re2::Regexp::Walker::~Walker\28\29,operator\20delete\5b\5d\28void*\29 +re2::Regexp::Walker::~Walker\28\29,operator\20delete\28void*\29 +re2::Regexp::Walker::~Walker\28\29,std::__1::__deque_base\2c\20std::__1::allocator\20>\20>::~__deque_base\28\29 +std::__1::__deque_base\2c\20std::__1::allocator\20>\20>::~__deque_base\28\29,operator\20delete\28void*\29 +re2::Regexp::ParseState::DoLeftParen\28re2::StringPiece\20const&\29,operator\20new\28unsigned\20long\29 +re2::Regexp::ParseState::DoLeftParen\28re2::StringPiece\20const&\29,re2::Regexp::Regexp\28re2::RegexpOp\2c\20re2::Regexp::ParseFlags\29 +re2::Regexp::ParseState::DoLeftParen\28re2::StringPiece\20const&\29,memcpy +re2::Regexp::ParseState::DoLeftParen\28re2::StringPiece\20const&\29,re2::Regexp::ParseState::PushRegexp\28re2::Regexp*\29 +re2::Regexp::ParseState::DoLeftParen\28re2::StringPiece\20const&\29,abort +re2::Regexp::ParseState::DoVerticalBar\28\29,re2::Regexp::ParseState::MaybeConcatString\28int\2c\20re2::Regexp::ParseFlags\29 +re2::Regexp::ParseState::DoVerticalBar\28\29,operator\20new\28unsigned\20long\29 +re2::Regexp::ParseState::DoVerticalBar\28\29,re2::Regexp::Regexp\28re2::RegexpOp\2c\20re2::Regexp::ParseFlags\29 +re2::Regexp::ParseState::DoVerticalBar\28\29,re2::Regexp::ParseState::PushRegexp\28re2::Regexp*\29 +re2::Regexp::ParseState::DoVerticalBar\28\29,re2::Regexp::ParseState::DoCollapse\28re2::RegexpOp\29 +re2::Regexp::ParseState::DoVerticalBar\28\29,re2::Regexp::Decref\28\29 +re2::Regexp::ParseState::DoCollapse\28re2::RegexpOp\29,operator\20new\28unsigned\20long\29 +re2::Regexp::ParseState::DoCollapse\28re2::RegexpOp\29,re2::Regexp::Incref\28\29 +re2::Regexp::ParseState::DoCollapse\28re2::RegexpOp\29,re2::Regexp::Decref\28\29 +re2::Regexp::ParseState::DoCollapse\28re2::RegexpOp\29,re2::CharClassBuilder::GetCharClass\28\29 +re2::Regexp::ParseState::DoCollapse\28re2::RegexpOp\29,std::__1::__tree\20>::destroy\28std::__1::__tree_node*\29 +re2::Regexp::ParseState::DoCollapse\28re2::RegexpOp\29,operator\20delete\28void*\29 +re2::Regexp::ParseState::DoCollapse\28re2::RegexpOp\29,re2::Regexp::ConcatOrAlternate\28re2::RegexpOp\2c\20re2::Regexp**\2c\20int\2c\20re2::Regexp::ParseFlags\2c\20bool\29 +re2::Regexp::ParseState::DoCollapse\28re2::RegexpOp\29,re2::Regexp::ComputeSimple\28\29 +re2::Regexp::ParseState::DoCollapse\28re2::RegexpOp\29,abort +re2::Regexp::ParseState::DoRightParen\28\29,re2::Regexp::ParseState::DoVerticalBar\28\29 +re2::Regexp::ParseState::DoRightParen\28\29,re2::Regexp::Decref\28\29 +re2::Regexp::ParseState::DoRightParen\28\29,re2::Regexp::ParseState::DoCollapse\28re2::RegexpOp\29 +re2::Regexp::ParseState::DoRightParen\28\29,re2::CharClassBuilder::GetCharClass\28\29 +re2::Regexp::ParseState::DoRightParen\28\29,std::__1::__tree\20>::destroy\28std::__1::__tree_node*\29 +re2::Regexp::ParseState::DoRightParen\28\29,operator\20delete\28void*\29 +re2::Regexp::ParseState::DoRightParen\28\29,re2::Regexp::ComputeSimple\28\29 +re2::Regexp::ParseState::DoRightParen\28\29,re2::Regexp::ParseState::PushRegexp\28re2::Regexp*\29 +re2::Regexp::ParseState::DoFinish\28\29,re2::Regexp::ParseState::DoVerticalBar\28\29 +re2::Regexp::ParseState::DoFinish\28\29,re2::Regexp::Decref\28\29 +re2::Regexp::ParseState::DoFinish\28\29,re2::Regexp::ParseState::DoCollapse\28re2::RegexpOp\29 +re2::Regexp::ParseState::DoFinish\28\29,re2::CharClassBuilder::GetCharClass\28\29 +re2::Regexp::ParseState::DoFinish\28\29,std::__1::__tree\20>::destroy\28std::__1::__tree_node*\29 +re2::Regexp::ParseState::DoFinish\28\29,operator\20delete\28void*\29 +re2::Regexp::RemoveLeadingString\28re2::Regexp*\2c\20int\29,operator\20delete\5b\5d\28void*\29 +re2::Regexp::RemoveLeadingString\28re2::Regexp*\2c\20int\29,memmove +re2::Regexp::RemoveLeadingString\28re2::Regexp*\2c\20int\29,re2::Regexp::Decref\28\29 +re2::Regexp::RemoveLeadingString\28re2::Regexp*\2c\20int\29,re2::Regexp::Swap\28re2::Regexp*\29 +re2::Regexp::FactorAlternation\28re2::Regexp**\2c\20int\2c\20re2::Regexp::ParseFlags\29,void\20std::__1::vector\20>::__emplace_back_slow_path\28re2::Regexp**&\2c\20int&\29 +re2::Regexp::FactorAlternation\28re2::Regexp**\2c\20int\2c\20re2::Regexp::ParseFlags\29,re2::Regexp::AlternateNoFactor\28re2::Regexp**\2c\20int\2c\20re2::Regexp::ParseFlags\29 +re2::Regexp::FactorAlternation\28re2::Regexp**\2c\20int\2c\20re2::Regexp::ParseFlags\29,re2::Regexp::Concat\28re2::Regexp**\2c\20int\2c\20re2::Regexp::ParseFlags\29 +re2::Regexp::FactorAlternation\28re2::Regexp**\2c\20int\2c\20re2::Regexp::ParseFlags\29,re2::FactorAlternationImpl::Round2\28re2::Regexp**\2c\20int\2c\20re2::Regexp::ParseFlags\2c\20std::__1::vector\20>*\29 +re2::Regexp::FactorAlternation\28re2::Regexp**\2c\20int\2c\20re2::Regexp::ParseFlags\29,re2::FactorAlternationImpl::Round3\28re2::Regexp**\2c\20int\2c\20re2::Regexp::ParseFlags\2c\20std::__1::vector\20>*\29 +re2::Regexp::FactorAlternation\28re2::Regexp**\2c\20int\2c\20re2::Regexp::ParseFlags\29,operator\20delete\28void*\29 +re2::Regexp::FactorAlternation\28re2::Regexp**\2c\20int\2c\20re2::Regexp::ParseFlags\29,re2::FactorAlternationImpl::Round1\28re2::Regexp**\2c\20int\2c\20re2::Regexp::ParseFlags\2c\20std::__1::vector\20>*\29 +void\20std::__1::vector\20>::__emplace_back_slow_path\28re2::Regexp**&\2c\20int&\29,operator\20new\28unsigned\20long\29 +void\20std::__1::vector\20>::__emplace_back_slow_path\28re2::Regexp**&\2c\20int&\29,operator\20delete\28void*\29 +void\20std::__1::vector\20>::__emplace_back_slow_path\28re2::Regexp**&\2c\20int&\29,abort +re2::FactorAlternationImpl::Round2\28re2::Regexp**\2c\20int\2c\20re2::Regexp::ParseFlags\2c\20std::__1::vector\20>*\29,re2::Regexp::Equal\28re2::Regexp*\2c\20re2::Regexp*\29 +re2::FactorAlternationImpl::Round2\28re2::Regexp**\2c\20int\2c\20re2::Regexp::ParseFlags\2c\20std::__1::vector\20>*\29,re2::Regexp::Incref\28\29 +re2::FactorAlternationImpl::Round2\28re2::Regexp**\2c\20int\2c\20re2::Regexp::ParseFlags\2c\20std::__1::vector\20>*\29,re2::Regexp::Decref\28\29 +re2::FactorAlternationImpl::Round2\28re2::Regexp**\2c\20int\2c\20re2::Regexp::ParseFlags\2c\20std::__1::vector\20>*\29,memmove +re2::FactorAlternationImpl::Round2\28re2::Regexp**\2c\20int\2c\20re2::Regexp::ParseFlags\2c\20std::__1::vector\20>*\29,operator\20new\28unsigned\20long\29 +re2::FactorAlternationImpl::Round2\28re2::Regexp**\2c\20int\2c\20re2::Regexp::ParseFlags\2c\20std::__1::vector\20>*\29,re2::Regexp::Regexp\28re2::RegexpOp\2c\20re2::Regexp::ParseFlags\29 +re2::FactorAlternationImpl::Round2\28re2::Regexp**\2c\20int\2c\20re2::Regexp::ParseFlags\2c\20std::__1::vector\20>*\29,memcpy +re2::FactorAlternationImpl::Round2\28re2::Regexp**\2c\20int\2c\20re2::Regexp::ParseFlags\2c\20std::__1::vector\20>*\29,operator\20delete\28void*\29 +re2::FactorAlternationImpl::Round2\28re2::Regexp**\2c\20int\2c\20re2::Regexp::ParseFlags\2c\20std::__1::vector\20>*\29,abort +re2::FactorAlternationImpl::Round3\28re2::Regexp**\2c\20int\2c\20re2::Regexp::ParseFlags\2c\20std::__1::vector\20>*\29,re2::CharClassBuilder::CharClassBuilder\28\29 +re2::FactorAlternationImpl::Round3\28re2::Regexp**\2c\20int\2c\20re2::Regexp::ParseFlags\2c\20std::__1::vector\20>*\29,re2::CharClassBuilder::AddRange\28int\2c\20int\29 +re2::FactorAlternationImpl::Round3\28re2::Regexp**\2c\20int\2c\20re2::Regexp::ParseFlags\2c\20std::__1::vector\20>*\29,re2::AddFoldedRange\28re2::CharClassBuilder*\2c\20int\2c\20int\2c\20int\29 +re2::FactorAlternationImpl::Round3\28re2::Regexp**\2c\20int\2c\20re2::Regexp::ParseFlags\2c\20std::__1::vector\20>*\29,re2::Regexp::Decref\28\29 +re2::FactorAlternationImpl::Round3\28re2::Regexp**\2c\20int\2c\20re2::Regexp::ParseFlags\2c\20std::__1::vector\20>*\29,re2::CharClassBuilder::GetCharClass\28\29 +re2::FactorAlternationImpl::Round3\28re2::Regexp**\2c\20int\2c\20re2::Regexp::ParseFlags\2c\20std::__1::vector\20>*\29,re2::Regexp::NewCharClass\28re2::CharClass*\2c\20re2::Regexp::ParseFlags\29 +re2::FactorAlternationImpl::Round3\28re2::Regexp**\2c\20int\2c\20re2::Regexp::ParseFlags\2c\20std::__1::vector\20>*\29,operator\20new\28unsigned\20long\29 +re2::FactorAlternationImpl::Round3\28re2::Regexp**\2c\20int\2c\20re2::Regexp::ParseFlags\2c\20std::__1::vector\20>*\29,memcpy +re2::FactorAlternationImpl::Round3\28re2::Regexp**\2c\20int\2c\20re2::Regexp::ParseFlags\2c\20std::__1::vector\20>*\29,operator\20delete\28void*\29 +re2::FactorAlternationImpl::Round3\28re2::Regexp**\2c\20int\2c\20re2::Regexp::ParseFlags\2c\20std::__1::vector\20>*\29,std::__1::__tree\20>::destroy\28std::__1::__tree_node*\29 +re2::FactorAlternationImpl::Round3\28re2::Regexp**\2c\20int\2c\20re2::Regexp::ParseFlags\2c\20std::__1::vector\20>*\29,abort +re2::FactorAlternationImpl::Round1\28re2::Regexp**\2c\20int\2c\20re2::Regexp::ParseFlags\2c\20std::__1::vector\20>*\29,re2::Regexp::LiteralString\28int*\2c\20int\2c\20re2::Regexp::ParseFlags\29 +re2::FactorAlternationImpl::Round1\28re2::Regexp**\2c\20int\2c\20re2::Regexp::ParseFlags\2c\20std::__1::vector\20>*\29,re2::Regexp::RemoveLeadingString\28re2::Regexp*\2c\20int\29 +re2::FactorAlternationImpl::Round1\28re2::Regexp**\2c\20int\2c\20re2::Regexp::ParseFlags\2c\20std::__1::vector\20>*\29,operator\20new\28unsigned\20long\29 +re2::FactorAlternationImpl::Round1\28re2::Regexp**\2c\20int\2c\20re2::Regexp::ParseFlags\2c\20std::__1::vector\20>*\29,memcpy +re2::FactorAlternationImpl::Round1\28re2::Regexp**\2c\20int\2c\20re2::Regexp::ParseFlags\2c\20std::__1::vector\20>*\29,operator\20delete\28void*\29 +re2::FactorAlternationImpl::Round1\28re2::Regexp**\2c\20int\2c\20re2::Regexp::ParseFlags\2c\20std::__1::vector\20>*\29,abort +re2::AddFoldedRange\28re2::CharClassBuilder*\2c\20int\2c\20int\2c\20int\29,re2::CharClassBuilder::AddRange\28int\2c\20int\29 +re2::AddFoldedRange\28re2::CharClassBuilder*\2c\20int\2c\20int\2c\20int\29,re2::AddFoldedRange\28re2::CharClassBuilder*\2c\20int\2c\20int\2c\20int\29 +re2::CharClassBuilder::AddRangeFlags\28int\2c\20int\2c\20re2::Regexp::ParseFlags\29,re2::AddFoldedRange\28re2::CharClassBuilder*\2c\20int\2c\20int\2c\20int\29 +re2::CharClassBuilder::AddRangeFlags\28int\2c\20int\2c\20re2::Regexp::ParseFlags\29,re2::CharClassBuilder::AddRange\28int\2c\20int\29 +re2::ParseUnicodeGroup\28re2::StringPiece*\2c\20re2::Regexp::ParseFlags\2c\20re2::CharClassBuilder*\2c\20re2::RegexpStatus*\29,fullrune +re2::ParseUnicodeGroup\28re2::StringPiece*\2c\20re2::Regexp::ParseFlags\2c\20re2::CharClassBuilder*\2c\20re2::RegexpStatus*\29,chartorune +re2::ParseUnicodeGroup\28re2::StringPiece*\2c\20re2::Regexp::ParseFlags\2c\20re2::CharClassBuilder*\2c\20re2::RegexpStatus*\29,re2::StringPiece::find\28char\2c\20unsigned\20long\29\20const +re2::ParseUnicodeGroup\28re2::StringPiece*\2c\20re2::Regexp::ParseFlags\2c\20re2::CharClassBuilder*\2c\20re2::RegexpStatus*\29,re2::IsValidUTF8\28re2::StringPiece\20const&\2c\20re2::RegexpStatus*\29 +re2::ParseUnicodeGroup\28re2::StringPiece*\2c\20re2::Regexp::ParseFlags\2c\20re2::CharClassBuilder*\2c\20re2::RegexpStatus*\29,memcmp +re2::ParseUnicodeGroup\28re2::StringPiece*\2c\20re2::Regexp::ParseFlags\2c\20re2::CharClassBuilder*\2c\20re2::RegexpStatus*\29,strlen +re2::ParseUnicodeGroup\28re2::StringPiece*\2c\20re2::Regexp::ParseFlags\2c\20re2::CharClassBuilder*\2c\20re2::RegexpStatus*\29,re2::AddUGroup\28re2::CharClassBuilder*\2c\20re2::UGroup\20const*\2c\20int\2c\20re2::Regexp::ParseFlags\29 +re2::IsValidUTF8\28re2::StringPiece\20const&\2c\20re2::RegexpStatus*\29,fullrune +re2::IsValidUTF8\28re2::StringPiece\20const&\2c\20re2::RegexpStatus*\29,chartorune +re2::AddUGroup\28re2::CharClassBuilder*\2c\20re2::UGroup\20const*\2c\20int\2c\20re2::Regexp::ParseFlags\29,re2::CharClassBuilder::AddRangeFlags\28int\2c\20int\2c\20re2::Regexp::ParseFlags\29 +re2::AddUGroup\28re2::CharClassBuilder*\2c\20re2::UGroup\20const*\2c\20int\2c\20re2::Regexp::ParseFlags\29,re2::CharClassBuilder::CharClassBuilder\28\29 +re2::AddUGroup\28re2::CharClassBuilder*\2c\20re2::UGroup\20const*\2c\20int\2c\20re2::Regexp::ParseFlags\29,re2::CharClassBuilder::AddRange\28int\2c\20int\29 +re2::AddUGroup\28re2::CharClassBuilder*\2c\20re2::UGroup\20const*\2c\20int\2c\20re2::Regexp::ParseFlags\29,re2::CharClassBuilder::Negate\28\29 +re2::AddUGroup\28re2::CharClassBuilder*\2c\20re2::UGroup\20const*\2c\20int\2c\20re2::Regexp::ParseFlags\29,re2::CharClassBuilder::AddCharClass\28re2::CharClassBuilder*\29 +re2::AddUGroup\28re2::CharClassBuilder*\2c\20re2::UGroup\20const*\2c\20int\2c\20re2::Regexp::ParseFlags\29,std::__1::__tree\20>::destroy\28std::__1::__tree_node*\29 +re2::StringPieceToRune\28int*\2c\20re2::StringPiece*\2c\20re2::RegexpStatus*\29,fullrune +re2::StringPieceToRune\28int*\2c\20re2::StringPiece*\2c\20re2::RegexpStatus*\29,chartorune +re2::Regexp::ParseState::ParseCCCharacter\28re2::StringPiece*\2c\20int*\2c\20re2::StringPiece\20const&\2c\20re2::RegexpStatus*\29,re2::ParseEscape\28re2::StringPiece*\2c\20int*\2c\20re2::RegexpStatus*\2c\20int\29 +re2::Regexp::ParseState::ParseCCCharacter\28re2::StringPiece*\2c\20int*\2c\20re2::StringPiece\20const&\2c\20re2::RegexpStatus*\29,fullrune +re2::Regexp::ParseState::ParseCCCharacter\28re2::StringPiece*\2c\20int*\2c\20re2::StringPiece\20const&\2c\20re2::RegexpStatus*\29,chartorune +re2::ParseEscape\28re2::StringPiece*\2c\20int*\2c\20re2::RegexpStatus*\2c\20int\29,fullrune +re2::ParseEscape\28re2::StringPiece*\2c\20int*\2c\20re2::RegexpStatus*\2c\20int\29,chartorune +re2::ParseEscape\28re2::StringPiece*\2c\20int*\2c\20re2::RegexpStatus*\2c\20int\29,isalpha +re2::ParseEscape\28re2::StringPiece*\2c\20int*\2c\20re2::RegexpStatus*\2c\20int\29,re2::StringPieceToRune\28int*\2c\20re2::StringPiece*\2c\20re2::RegexpStatus*\29 +re2::Regexp::ParseState::ParseCharClass\28re2::StringPiece*\2c\20re2::Regexp**\2c\20re2::RegexpStatus*\29,operator\20new\28unsigned\20long\29 +re2::Regexp::ParseState::ParseCharClass\28re2::StringPiece*\2c\20re2::Regexp**\2c\20re2::RegexpStatus*\29,re2::Regexp::Regexp\28re2::RegexpOp\2c\20re2::Regexp::ParseFlags\29 +re2::Regexp::ParseState::ParseCharClass\28re2::StringPiece*\2c\20re2::Regexp**\2c\20re2::RegexpStatus*\29,re2::CharClassBuilder::CharClassBuilder\28\29 +re2::Regexp::ParseState::ParseCharClass\28re2::StringPiece*\2c\20re2::Regexp**\2c\20re2::RegexpStatus*\29,re2::CharClassBuilder::AddRange\28int\2c\20int\29 +re2::Regexp::ParseState::ParseCharClass\28re2::StringPiece*\2c\20re2::Regexp**\2c\20re2::RegexpStatus*\29,fullrune +re2::Regexp::ParseState::ParseCharClass\28re2::StringPiece*\2c\20re2::Regexp**\2c\20re2::RegexpStatus*\29,chartorune +re2::Regexp::ParseState::ParseCharClass\28re2::StringPiece*\2c\20re2::Regexp**\2c\20re2::RegexpStatus*\29,re2::ParseUnicodeGroup\28re2::StringPiece*\2c\20re2::Regexp::ParseFlags\2c\20re2::CharClassBuilder*\2c\20re2::RegexpStatus*\29 +re2::Regexp::ParseState::ParseCharClass\28re2::StringPiece*\2c\20re2::Regexp**\2c\20re2::RegexpStatus*\29,strlen +re2::Regexp::ParseState::ParseCharClass\28re2::StringPiece*\2c\20re2::Regexp**\2c\20re2::RegexpStatus*\29,memcmp +re2::Regexp::ParseState::ParseCharClass\28re2::StringPiece*\2c\20re2::Regexp**\2c\20re2::RegexpStatus*\29,re2::Regexp::ParseState::ParseCCCharacter\28re2::StringPiece*\2c\20int*\2c\20re2::StringPiece\20const&\2c\20re2::RegexpStatus*\29 +re2::Regexp::ParseState::ParseCharClass\28re2::StringPiece*\2c\20re2::Regexp**\2c\20re2::RegexpStatus*\29,re2::CharClassBuilder::AddRangeFlags\28int\2c\20int\2c\20re2::Regexp::ParseFlags\29 +re2::Regexp::ParseState::ParseCharClass\28re2::StringPiece*\2c\20re2::Regexp**\2c\20re2::RegexpStatus*\29,re2::CharClassBuilder::Negate\28\29 +re2::Regexp::ParseState::ParseCharClass\28re2::StringPiece*\2c\20re2::Regexp**\2c\20re2::RegexpStatus*\29,re2::AddUGroup\28re2::CharClassBuilder*\2c\20re2::UGroup\20const*\2c\20int\2c\20re2::Regexp::ParseFlags\29 +re2::Regexp::ParseState::ParseCharClass\28re2::StringPiece*\2c\20re2::Regexp**\2c\20re2::RegexpStatus*\29,re2::Regexp::Decref\28\29 +re2::Regexp::ParseState::ParsePerlFlags\28re2::StringPiece*\29,re2::StringPiece::find\28char\2c\20unsigned\20long\29\20const +re2::Regexp::ParseState::ParsePerlFlags\28re2::StringPiece*\29,re2::IsValidUTF8\28re2::StringPiece\20const&\2c\20re2::RegexpStatus*\29 +re2::Regexp::ParseState::ParsePerlFlags\28re2::StringPiece*\29,re2::IsValidCaptureName\28re2::StringPiece\20const&\29 +re2::Regexp::ParseState::ParsePerlFlags\28re2::StringPiece*\29,re2::Regexp::ParseState::DoLeftParen\28re2::StringPiece\20const&\29 +re2::Regexp::ParseState::ParsePerlFlags\28re2::StringPiece*\29,fullrune +re2::Regexp::ParseState::ParsePerlFlags\28re2::StringPiece*\29,chartorune +re2::Regexp::ParseState::ParsePerlFlags\28re2::StringPiece*\29,operator\20new\28unsigned\20long\29 +re2::Regexp::ParseState::ParsePerlFlags\28re2::StringPiece*\29,re2::Regexp::Regexp\28re2::RegexpOp\2c\20re2::Regexp::ParseFlags\29 +re2::Regexp::ParseState::ParsePerlFlags\28re2::StringPiece*\29,re2::Regexp::ParseState::PushRegexp\28re2::Regexp*\29 +re2::Regexp::Parse\28re2::StringPiece\20const&\2c\20re2::Regexp::ParseFlags\2c\20re2::RegexpStatus*\29,operator\20new\28unsigned\20long\29 +re2::Regexp::Parse\28re2::StringPiece\20const&\2c\20re2::Regexp::ParseFlags\2c\20re2::RegexpStatus*\29,runetochar +re2::Regexp::Parse\28re2::StringPiece\20const&\2c\20re2::Regexp::ParseFlags\2c\20re2::RegexpStatus*\29,std::__1::basic_string\2c\20std::__1::allocator\20>::append\28char\20const*\2c\20unsigned\20long\29 +re2::Regexp::Parse\28re2::StringPiece\20const&\2c\20re2::Regexp::ParseFlags\2c\20re2::RegexpStatus*\29,operator\20delete\28void*\29 +re2::Regexp::Parse\28re2::StringPiece\20const&\2c\20re2::Regexp::ParseFlags\2c\20re2::RegexpStatus*\29,fullrune +re2::Regexp::Parse\28re2::StringPiece\20const&\2c\20re2::Regexp::ParseFlags\2c\20re2::RegexpStatus*\29,chartorune +re2::Regexp::Parse\28re2::StringPiece\20const&\2c\20re2::Regexp::ParseFlags\2c\20re2::RegexpStatus*\29,re2::Regexp::ParseState::PushLiteral\28int\29 +re2::Regexp::Parse\28re2::StringPiece\20const&\2c\20re2::Regexp::ParseFlags\2c\20re2::RegexpStatus*\29,re2::Regexp::ParseState::DoFinish\28\29 +re2::Regexp::Parse\28re2::StringPiece\20const&\2c\20re2::Regexp::ParseFlags\2c\20re2::RegexpStatus*\29,re2::Regexp::ParseState::ParsePerlFlags\28re2::StringPiece*\29 +re2::Regexp::Parse\28re2::StringPiece\20const&\2c\20re2::Regexp::ParseFlags\2c\20re2::RegexpStatus*\29,re2::Regexp::Regexp\28re2::RegexpOp\2c\20re2::Regexp::ParseFlags\29 +re2::Regexp::Parse\28re2::StringPiece\20const&\2c\20re2::Regexp::ParseFlags\2c\20re2::RegexpStatus*\29,re2::Regexp::ParseState::PushRegexp\28re2::Regexp*\29 +re2::Regexp::Parse\28re2::StringPiece\20const&\2c\20re2::Regexp::ParseFlags\2c\20re2::RegexpStatus*\29,re2::Regexp::ParseState::DoLeftParen\28re2::StringPiece\20const&\29 +re2::Regexp::Parse\28re2::StringPiece\20const&\2c\20re2::Regexp::ParseFlags\2c\20re2::RegexpStatus*\29,re2::Regexp::ParseState::DoVerticalBar\28\29 +re2::Regexp::Parse\28re2::StringPiece\20const&\2c\20re2::Regexp::ParseFlags\2c\20re2::RegexpStatus*\29,re2::Regexp::ParseState::DoRightParen\28\29 +re2::Regexp::Parse\28re2::StringPiece\20const&\2c\20re2::Regexp::ParseFlags\2c\20re2::RegexpStatus*\29,re2::Regexp::ParseState::PushDot\28\29 +re2::Regexp::Parse\28re2::StringPiece\20const&\2c\20re2::Regexp::ParseFlags\2c\20re2::RegexpStatus*\29,re2::Regexp::ParseState::ParseCharClass\28re2::StringPiece*\2c\20re2::Regexp**\2c\20re2::RegexpStatus*\29 +re2::Regexp::Parse\28re2::StringPiece\20const&\2c\20re2::Regexp::ParseFlags\2c\20re2::RegexpStatus*\29,re2::Regexp::ParseState::PushRepeatOp\28re2::RegexpOp\2c\20re2::StringPiece\20const&\2c\20bool\29 +re2::Regexp::Parse\28re2::StringPiece\20const&\2c\20re2::Regexp::ParseFlags\2c\20re2::RegexpStatus*\29,re2::ParseInteger\28re2::StringPiece*\2c\20int*\29 +re2::Regexp::Parse\28re2::StringPiece\20const&\2c\20re2::Regexp::ParseFlags\2c\20re2::RegexpStatus*\29,re2::Regexp::ParseState::PushSimpleOp\28re2::RegexpOp\29 +re2::Regexp::Parse\28re2::StringPiece\20const&\2c\20re2::Regexp::ParseFlags\2c\20re2::RegexpStatus*\29,re2::CharClassBuilder::CharClassBuilder\28\29 +re2::Regexp::Parse\28re2::StringPiece\20const&\2c\20re2::Regexp::ParseFlags\2c\20re2::RegexpStatus*\29,re2::ParseUnicodeGroup\28re2::StringPiece*\2c\20re2::Regexp::ParseFlags\2c\20re2::CharClassBuilder*\2c\20re2::RegexpStatus*\29 +re2::Regexp::Parse\28re2::StringPiece\20const&\2c\20re2::Regexp::ParseFlags\2c\20re2::RegexpStatus*\29,re2::Regexp::Decref\28\29 +re2::Regexp::Parse\28re2::StringPiece\20const&\2c\20re2::Regexp::ParseFlags\2c\20re2::RegexpStatus*\29,strlen +re2::Regexp::Parse\28re2::StringPiece\20const&\2c\20re2::Regexp::ParseFlags\2c\20re2::RegexpStatus*\29,re2::ParseEscape\28re2::StringPiece*\2c\20int*\2c\20re2::RegexpStatus*\2c\20int\29 +re2::Regexp::Parse\28re2::StringPiece\20const&\2c\20re2::Regexp::ParseFlags\2c\20re2::RegexpStatus*\29,re2::AddUGroup\28re2::CharClassBuilder*\2c\20re2::UGroup\20const*\2c\20int\2c\20re2::Regexp::ParseFlags\29 +re2::Regexp::Parse\28re2::StringPiece\20const&\2c\20re2::Regexp::ParseFlags\2c\20re2::RegexpStatus*\29,re2::Regexp::ParseState::PushRepetition\28int\2c\20int\2c\20re2::StringPiece\20const&\2c\20bool\29 +re2::RepetitionWalker::~RepetitionWalker\28\29,re2::Regexp::Walker::~Walker\28\29 +re2::RepetitionWalker::~RepetitionWalker\28\29,operator\20delete\28void*\29 +std::__1::deque\2c\20std::__1::allocator\20>\20>::__add_back_capacity\28\29,memmove +std::__1::deque\2c\20std::__1::allocator\20>\20>::__add_back_capacity\28\29,operator\20new\28unsigned\20long\29 +std::__1::deque\2c\20std::__1::allocator\20>\20>::__add_back_capacity\28\29,operator\20delete\28void*\29 +std::__1::deque\2c\20std::__1::allocator\20>\20>::__add_back_capacity\28\29,std::__1::__split_buffer*\2c\20std::__1::allocator*>\20>::push_back\28re2::WalkState*&&\29 +std::__1::deque\2c\20std::__1::allocator\20>\20>::__add_back_capacity\28\29,std::__1::__split_buffer*\2c\20std::__1::allocator*>\20>::push_front\28re2::WalkState*&&\29 +std::__1::deque\2c\20std::__1::allocator\20>\20>::__add_back_capacity\28\29,std::__1::__split_buffer*\2c\20std::__1::allocator*>&>::push_back\28re2::WalkState*&&\29 +std::__1::deque\2c\20std::__1::allocator\20>\20>::__add_back_capacity\28\29,std::__1::__split_buffer*\2c\20std::__1::allocator*>&>::push_front\28re2::WalkState*\20const&\29 +std::__1::deque\2c\20std::__1::allocator\20>\20>::__add_back_capacity\28\29,abort +std::__1::__split_buffer*\2c\20std::__1::allocator*>\20>::push_back\28re2::WalkState*&&\29,memmove +std::__1::__split_buffer*\2c\20std::__1::allocator*>\20>::push_back\28re2::WalkState*&&\29,operator\20new\28unsigned\20long\29 +std::__1::__split_buffer*\2c\20std::__1::allocator*>\20>::push_back\28re2::WalkState*&&\29,operator\20delete\28void*\29 +std::__1::__split_buffer*\2c\20std::__1::allocator*>\20>::push_back\28re2::WalkState*&&\29,abort +std::__1::__split_buffer*\2c\20std::__1::allocator*>\20>::push_front\28re2::WalkState*&&\29,memmove +std::__1::__split_buffer*\2c\20std::__1::allocator*>\20>::push_front\28re2::WalkState*&&\29,operator\20new\28unsigned\20long\29 +std::__1::__split_buffer*\2c\20std::__1::allocator*>\20>::push_front\28re2::WalkState*&&\29,operator\20delete\28void*\29 +std::__1::__split_buffer*\2c\20std::__1::allocator*>\20>::push_front\28re2::WalkState*&&\29,abort +std::__1::__split_buffer*\2c\20std::__1::allocator*>&>::push_back\28re2::WalkState*&&\29,memmove +std::__1::__split_buffer*\2c\20std::__1::allocator*>&>::push_back\28re2::WalkState*&&\29,operator\20new\28unsigned\20long\29 +std::__1::__split_buffer*\2c\20std::__1::allocator*>&>::push_back\28re2::WalkState*&&\29,operator\20delete\28void*\29 +std::__1::__split_buffer*\2c\20std::__1::allocator*>&>::push_back\28re2::WalkState*&&\29,abort +std::__1::__split_buffer*\2c\20std::__1::allocator*>&>::push_front\28re2::WalkState*\20const&\29,memmove +std::__1::__split_buffer*\2c\20std::__1::allocator*>&>::push_front\28re2::WalkState*\20const&\29,operator\20new\28unsigned\20long\29 +std::__1::__split_buffer*\2c\20std::__1::allocator*>&>::push_front\28re2::WalkState*\20const&\29,operator\20delete\28void*\29 +std::__1::__split_buffer*\2c\20std::__1::allocator*>&>::push_front\28re2::WalkState*\20const&\29,abort +re2::Prog::Inst::InitAlt\28unsigned\20int\2c\20unsigned\20int\29,opa_abort +re2::Prog::Inst::InitByteRange\28int\2c\20int\2c\20int\2c\20unsigned\20int\29,opa_abort +re2::Prog::Inst::InitCapture\28int\2c\20unsigned\20int\29,opa_abort +re2::Prog::Inst::InitEmptyWidth\28re2::EmptyOp\2c\20unsigned\20int\29,opa_abort +re2::Prog::Inst::InitMatch\28int\29,opa_abort +re2::Prog::Inst::InitNop\28unsigned\20int\29,opa_abort +re2::Prog::Inst::InitFail\28\29,opa_abort +re2::Prog::~Prog\28\29,re2::Prog::DeleteDFA\28re2::DFA*\29 +re2::Prog::~Prog\28\29,operator\20delete\28void*\29 +re2::Prog::Optimize\28\29,operator\20new\28unsigned\20long\29 +re2::Prog::Optimize\28\29,re2::SparseSetT::InsertInternal\28bool\2c\20int\29 +re2::Prog::Optimize\28\29,opa_abort +re2::Prog::Optimize\28\29,abort +re2::Prog::Optimize\28\29,operator\20delete\28void*\29 +re2::ByteMapBuilder::Mark\28int\2c\20int\29,opa_abort +re2::ByteMapBuilder::Mark\28int\2c\20int\29,operator\20new\28unsigned\20long\29 +re2::ByteMapBuilder::Mark\28int\2c\20int\29,memcpy +re2::ByteMapBuilder::Mark\28int\2c\20int\29,operator\20delete\28void*\29 +re2::ByteMapBuilder::Mark\28int\2c\20int\29,abort +re2::ByteMapBuilder::Merge\28\29,opa_abort +re2::ByteMapBuilder::Merge\28\29,operator\20new\28unsigned\20long\29 +re2::ByteMapBuilder::Merge\28\29,memcpy +re2::ByteMapBuilder::Merge\28\29,operator\20delete\28void*\29 +re2::ByteMapBuilder::Merge\28\29,abort +re2::ByteMapBuilder::Recolor\28int\29,operator\20new\28unsigned\20long\29 +re2::ByteMapBuilder::Recolor\28int\29,memcpy +re2::ByteMapBuilder::Recolor\28int\29,operator\20delete\28void*\29 +re2::ByteMapBuilder::Recolor\28int\29,abort +re2::ByteMapBuilder::Build\28unsigned\20char*\2c\20int*\29,opa_abort +re2::ByteMapBuilder::Build\28unsigned\20char*\2c\20int*\29,re2::ByteMapBuilder::Recolor\28int\29 +re2::ByteMapBuilder::Build\28unsigned\20char*\2c\20int*\29,memset +re2::Prog::ComputeByteMap\28\29,re2::ByteMapBuilder::Mark\28int\2c\20int\29 +re2::Prog::ComputeByteMap\28\29,opa_abort +re2::Prog::ComputeByteMap\28\29,operator\20new\28unsigned\20long\29 +re2::Prog::ComputeByteMap\28\29,memcpy +re2::Prog::ComputeByteMap\28\29,operator\20delete\28void*\29 +re2::Prog::ComputeByteMap\28\29,re2::ByteMapBuilder::Merge\28\29 +re2::Prog::ComputeByteMap\28\29,abort +re2::Prog::ComputeByteMap\28\29,re2::ByteMapBuilder::Build\28unsigned\20char*\2c\20int*\29 +re2::Prog::Flatten\28\29,operator\20new\28unsigned\20long\29 +re2::Prog::Flatten\28\29,re2::Prog::MarkSuccessors\28re2::SparseArray*\2c\20re2::SparseArray*\2c\20std::__1::vector\20>\2c\20std::__1::allocator\20>\20>\20>*\2c\20re2::SparseSetT*\2c\20std::__1::vector\20>*\29 +re2::Prog::Flatten\28\29,memmove +re2::Prog::Flatten\28\29,void\20std::__1::__sort::IndexValue\20const&\2c\20re2::SparseArray::IndexValue\20const&\29\2c\20re2::SparseArray::IndexValue*>\28re2::SparseArray::IndexValue*\2c\20re2::SparseArray::IndexValue*\2c\20bool\20\28*&\29\28re2::SparseArray::IndexValue\20const&\2c\20re2::SparseArray::IndexValue\20const&\29\29 +re2::Prog::Flatten\28\29,re2::Prog::MarkDominator\28int\2c\20re2::SparseArray*\2c\20re2::SparseArray*\2c\20std::__1::vector\20>\2c\20std::__1::allocator\20>\20>\20>*\2c\20re2::SparseSetT*\2c\20std::__1::vector\20>*\29 +re2::Prog::Flatten\28\29,memset +re2::Prog::Flatten\28\29,re2::Prog::EmitList\28int\2c\20re2::SparseArray*\2c\20std::__1::vector\20>*\2c\20re2::SparseSetT*\2c\20std::__1::vector\20>*\29 +re2::Prog::Flatten\28\29,re2::Prog::ComputeHints\28std::__1::vector\20>*\2c\20int\2c\20int\29 +re2::Prog::Flatten\28\29,opa_abort +re2::Prog::Flatten\28\29,operator\20delete\28void*\29 +re2::Prog::Flatten\28\29,abort +re2::Prog::MarkSuccessors\28re2::SparseArray*\2c\20re2::SparseArray*\2c\20std::__1::vector\20>\2c\20std::__1::allocator\20>\20>\20>*\2c\20re2::SparseSetT*\2c\20std::__1::vector\20>*\29,re2::SparseArray::SetInternal\28bool\2c\20int\2c\20int\20const&\29 +re2::Prog::MarkSuccessors\28re2::SparseArray*\2c\20re2::SparseArray*\2c\20std::__1::vector\20>\2c\20std::__1::allocator\20>\20>\20>*\2c\20re2::SparseSetT*\2c\20std::__1::vector\20>*\29,opa_abort +re2::Prog::MarkSuccessors\28re2::SparseArray*\2c\20re2::SparseArray*\2c\20std::__1::vector\20>\2c\20std::__1::allocator\20>\20>\20>*\2c\20re2::SparseSetT*\2c\20std::__1::vector\20>*\29,operator\20new\28unsigned\20long\29 +re2::Prog::MarkSuccessors\28re2::SparseArray*\2c\20re2::SparseArray*\2c\20std::__1::vector\20>\2c\20std::__1::allocator\20>\20>\20>*\2c\20re2::SparseSetT*\2c\20std::__1::vector\20>*\29,operator\20delete\28void*\29 +re2::Prog::MarkSuccessors\28re2::SparseArray*\2c\20re2::SparseArray*\2c\20std::__1::vector\20>\2c\20std::__1::allocator\20>\20>\20>*\2c\20re2::SparseSetT*\2c\20std::__1::vector\20>*\29,re2::SparseSetT::InsertInternal\28bool\2c\20int\29 +re2::Prog::MarkSuccessors\28re2::SparseArray*\2c\20re2::SparseArray*\2c\20std::__1::vector\20>\2c\20std::__1::allocator\20>\20>\20>*\2c\20re2::SparseSetT*\2c\20std::__1::vector\20>*\29,memcpy +re2::Prog::MarkSuccessors\28re2::SparseArray*\2c\20re2::SparseArray*\2c\20std::__1::vector\20>\2c\20std::__1::allocator\20>\20>\20>*\2c\20re2::SparseSetT*\2c\20std::__1::vector\20>*\29,abort +void\20std::__1::__sort::IndexValue\20const&\2c\20re2::SparseArray::IndexValue\20const&\29\2c\20re2::SparseArray::IndexValue*>\28re2::SparseArray::IndexValue*\2c\20re2::SparseArray::IndexValue*\2c\20bool\20\28*&\29\28re2::SparseArray::IndexValue\20const&\2c\20re2::SparseArray::IndexValue\20const&\29\29,unsigned\20int\20std::__1::__sort4::IndexValue\20const&\2c\20re2::SparseArray::IndexValue\20const&\29\2c\20re2::SparseArray::IndexValue*>\28re2::SparseArray::IndexValue*\2c\20re2::SparseArray::IndexValue*\2c\20re2::SparseArray::IndexValue*\2c\20re2::SparseArray::IndexValue*\2c\20bool\20\28*&\29\28re2::SparseArray::IndexValue\20const&\2c\20re2::SparseArray::IndexValue\20const&\29\29 +void\20std::__1::__sort::IndexValue\20const&\2c\20re2::SparseArray::IndexValue\20const&\29\2c\20re2::SparseArray::IndexValue*>\28re2::SparseArray::IndexValue*\2c\20re2::SparseArray::IndexValue*\2c\20bool\20\28*&\29\28re2::SparseArray::IndexValue\20const&\2c\20re2::SparseArray::IndexValue\20const&\29\29,void\20std::__1::__insertion_sort_3::IndexValue\20const&\2c\20re2::SparseArray::IndexValue\20const&\29\2c\20re2::SparseArray::IndexValue*>\28re2::SparseArray::IndexValue*\2c\20re2::SparseArray::IndexValue*\2c\20bool\20\28*&\29\28re2::SparseArray::IndexValue\20const&\2c\20re2::SparseArray::IndexValue\20const&\29\29 +void\20std::__1::__sort::IndexValue\20const&\2c\20re2::SparseArray::IndexValue\20const&\29\2c\20re2::SparseArray::IndexValue*>\28re2::SparseArray::IndexValue*\2c\20re2::SparseArray::IndexValue*\2c\20bool\20\28*&\29\28re2::SparseArray::IndexValue\20const&\2c\20re2::SparseArray::IndexValue\20const&\29\29,bool\20std::__1::__insertion_sort_incomplete::IndexValue\20const&\2c\20re2::SparseArray::IndexValue\20const&\29\2c\20re2::SparseArray::IndexValue*>\28re2::SparseArray::IndexValue*\2c\20re2::SparseArray::IndexValue*\2c\20bool\20\28*&\29\28re2::SparseArray::IndexValue\20const&\2c\20re2::SparseArray::IndexValue\20const&\29\29 +void\20std::__1::__sort::IndexValue\20const&\2c\20re2::SparseArray::IndexValue\20const&\29\2c\20re2::SparseArray::IndexValue*>\28re2::SparseArray::IndexValue*\2c\20re2::SparseArray::IndexValue*\2c\20bool\20\28*&\29\28re2::SparseArray::IndexValue\20const&\2c\20re2::SparseArray::IndexValue\20const&\29\29,void\20std::__1::__sort::IndexValue\20const&\2c\20re2::SparseArray::IndexValue\20const&\29\2c\20re2::SparseArray::IndexValue*>\28re2::SparseArray::IndexValue*\2c\20re2::SparseArray::IndexValue*\2c\20bool\20\28*&\29\28re2::SparseArray::IndexValue\20const&\2c\20re2::SparseArray::IndexValue\20const&\29\29 +re2::Prog::MarkDominator\28int\2c\20re2::SparseArray*\2c\20re2::SparseArray*\2c\20std::__1::vector\20>\2c\20std::__1::allocator\20>\20>\20>*\2c\20re2::SparseSetT*\2c\20std::__1::vector\20>*\29,operator\20new\28unsigned\20long\29 +re2::Prog::MarkDominator\28int\2c\20re2::SparseArray*\2c\20re2::SparseArray*\2c\20std::__1::vector\20>\2c\20std::__1::allocator\20>\20>\20>*\2c\20re2::SparseSetT*\2c\20std::__1::vector\20>*\29,operator\20delete\28void*\29 +re2::Prog::MarkDominator\28int\2c\20re2::SparseArray*\2c\20re2::SparseArray*\2c\20std::__1::vector\20>\2c\20std::__1::allocator\20>\20>\20>*\2c\20re2::SparseSetT*\2c\20std::__1::vector\20>*\29,opa_abort +re2::Prog::MarkDominator\28int\2c\20re2::SparseArray*\2c\20re2::SparseArray*\2c\20std::__1::vector\20>\2c\20std::__1::allocator\20>\20>\20>*\2c\20re2::SparseSetT*\2c\20std::__1::vector\20>*\29,re2::SparseSetT::InsertInternal\28bool\2c\20int\29 +re2::Prog::MarkDominator\28int\2c\20re2::SparseArray*\2c\20re2::SparseArray*\2c\20std::__1::vector\20>\2c\20std::__1::allocator\20>\20>\20>*\2c\20re2::SparseSetT*\2c\20std::__1::vector\20>*\29,memcpy +re2::Prog::MarkDominator\28int\2c\20re2::SparseArray*\2c\20re2::SparseArray*\2c\20std::__1::vector\20>\2c\20std::__1::allocator\20>\20>\20>*\2c\20re2::SparseSetT*\2c\20std::__1::vector\20>*\29,re2::SparseArray::create_index\28int\29 +re2::Prog::MarkDominator\28int\2c\20re2::SparseArray*\2c\20re2::SparseArray*\2c\20std::__1::vector\20>\2c\20std::__1::allocator\20>\20>\20>*\2c\20re2::SparseSetT*\2c\20std::__1::vector\20>*\29,re2::SparseArray::SetExistingInternal\28int\2c\20int\20const&\29 +re2::Prog::MarkDominator\28int\2c\20re2::SparseArray*\2c\20re2::SparseArray*\2c\20std::__1::vector\20>\2c\20std::__1::allocator\20>\20>\20>*\2c\20re2::SparseSetT*\2c\20std::__1::vector\20>*\29,abort +re2::Prog::EmitList\28int\2c\20re2::SparseArray*\2c\20std::__1::vector\20>*\2c\20re2::SparseSetT*\2c\20std::__1::vector\20>*\29,operator\20new\28unsigned\20long\29 +re2::Prog::EmitList\28int\2c\20re2::SparseArray*\2c\20std::__1::vector\20>*\2c\20re2::SparseSetT*\2c\20std::__1::vector\20>*\29,operator\20delete\28void*\29 +re2::Prog::EmitList\28int\2c\20re2::SparseArray*\2c\20std::__1::vector\20>*\2c\20re2::SparseSetT*\2c\20std::__1::vector\20>*\29,opa_abort +re2::Prog::EmitList\28int\2c\20re2::SparseArray*\2c\20std::__1::vector\20>*\2c\20re2::SparseSetT*\2c\20std::__1::vector\20>*\29,re2::SparseSetT::InsertInternal\28bool\2c\20int\29 +re2::Prog::EmitList\28int\2c\20re2::SparseArray*\2c\20std::__1::vector\20>*\2c\20re2::SparseSetT*\2c\20std::__1::vector\20>*\29,memcpy +re2::Prog::EmitList\28int\2c\20re2::SparseArray*\2c\20std::__1::vector\20>*\2c\20re2::SparseSetT*\2c\20std::__1::vector\20>*\29,abort +re2::Prog::ComputeHints\28std::__1::vector\20>*\2c\20int\2c\20int\29,opa_abort +re2::Prog::ComputeHints\28std::__1::vector\20>*\2c\20int\2c\20int\29,re2::Prog::ComputeHints\28std::__1::vector\20>*\2c\20int\2c\20int\29::$_1::operator\28\29\28int\2c\20int\29\20const +re2::Prog::ComputeHints\28std::__1::vector\20>*\2c\20int\2c\20int\29::$_1::operator\28\29\28int\2c\20int\29\20const,opa_abort +re2::Prog::PrefixAccel_FrontAndBack\28void\20const*\2c\20unsigned\20long\29,opa_abort +re2::Prog::PrefixAccel_FrontAndBack\28void\20const*\2c\20unsigned\20long\29,memchr +bool\20std::__1::__insertion_sort_incomplete::IndexValue\20const&\2c\20re2::SparseArray::IndexValue\20const&\29\2c\20re2::SparseArray::IndexValue*>\28re2::SparseArray::IndexValue*\2c\20re2::SparseArray::IndexValue*\2c\20bool\20\28*&\29\28re2::SparseArray::IndexValue\20const&\2c\20re2::SparseArray::IndexValue\20const&\29\29,unsigned\20int\20std::__1::__sort4::IndexValue\20const&\2c\20re2::SparseArray::IndexValue\20const&\29\2c\20re2::SparseArray::IndexValue*>\28re2::SparseArray::IndexValue*\2c\20re2::SparseArray::IndexValue*\2c\20re2::SparseArray::IndexValue*\2c\20re2::SparseArray::IndexValue*\2c\20bool\20\28*&\29\28re2::SparseArray::IndexValue\20const&\2c\20re2::SparseArray::IndexValue\20const&\29\29 +re2::RE2::Init\28re2::StringPiece\20const&\2c\20re2::RE2::Options\20const&\29,std::__1::__call_once\28unsigned\20long\20volatile&\2c\20void*\2c\20void\20\28*\29\28void*\29\29 +re2::RE2::Init\28re2::StringPiece\20const&\2c\20re2::RE2::Options\20const&\29,std::__1::basic_string\2c\20std::__1::allocator\20>::assign\28char\20const*\2c\20unsigned\20long\29 +re2::RE2::Init\28re2::StringPiece\20const&\2c\20re2::RE2::Options\20const&\29,re2::Regexp::Parse\28re2::StringPiece\20const&\2c\20re2::Regexp::ParseFlags\2c\20re2::RegexpStatus*\29 +re2::RE2::Init\28re2::StringPiece\20const&\2c\20re2::RE2::Options\20const&\29,operator\20new\28unsigned\20long\29 +re2::RE2::Init\28re2::StringPiece\20const&\2c\20re2::RE2::Options\20const&\29,re2::RegexpStatus::Text\28\29\20const +re2::RE2::Init\28re2::StringPiece\20const&\2c\20re2::RE2::Options\20const&\29,memcpy +re2::RE2::Init\28re2::StringPiece\20const&\2c\20re2::RE2::Options\20const&\29,operator\20delete\28void*\29 +re2::RE2::Init\28re2::StringPiece\20const&\2c\20re2::RE2::Options\20const&\29,re2::Regexp::RequiredPrefix\28std::__1::basic_string\2c\20std::__1::allocator\20>*\2c\20bool*\2c\20re2::Regexp**\29 +re2::RE2::Init\28re2::StringPiece\20const&\2c\20re2::RE2::Options\20const&\29,re2::Regexp::Incref\28\29 +re2::RE2::Init\28re2::StringPiece\20const&\2c\20re2::RE2::Options\20const&\29,re2::Regexp::CompileToProg\28long\20long\29 +re2::RE2::Init\28re2::StringPiece\20const&\2c\20re2::RE2::Options\20const&\29,re2::Regexp::NumCaptures\28\29 +re2::RE2::Init\28re2::StringPiece\20const&\2c\20re2::RE2::Options\20const&\29,re2::Prog::IsOnePass\28\29 +re2::RE2::Init\28re2::StringPiece\20const&\2c\20re2::RE2::Options\20const&\29,abort +void\20std::__1::__call_once_proxy\20>\28void*\29,operator\20new\28unsigned\20long\29 +re2::RE2::RE2\28re2::StringPiece\20const&\2c\20re2::RE2::Options\20const&\29,re2::RE2::Init\28re2::StringPiece\20const&\2c\20re2::RE2::Options\20const&\29 +re2::RE2::ReverseProg\28\29\20const,std::__1::__call_once\28unsigned\20long\20volatile&\2c\20void*\2c\20void\20\28*\29\28void*\29\29 +void\20std::__1::__call_once_proxy\20>\28void*\29,re2::Regexp::CompileToReverseProg\28long\20long\29 +re2::RE2::~RE2\28\29,re2::Regexp::Decref\28\29 +re2::RE2::~RE2\28\29,re2::Prog::~Prog\28\29 +re2::RE2::~RE2\28\29,operator\20delete\28void*\29 +re2::RE2::~RE2\28\29,std::__1::__tree\2c\20std::__1::allocator\20>\2c\20int>\2c\20std::__1::__map_value_compare\2c\20std::__1::allocator\20>\2c\20std::__1::__value_type\2c\20std::__1::allocator\20>\2c\20int>\2c\20std::__1::less\2c\20std::__1::allocator\20>\20>\2c\20true>\2c\20std::__1::allocator\2c\20std::__1::allocator\20>\2c\20int>\20>\20>::destroy\28std::__1::__tree_node\2c\20std::__1::allocator\20>\2c\20int>\2c\20void*>*\29 +re2::RE2::~RE2\28\29,std::__1::__tree\2c\20std::__1::allocator\20>\20>\2c\20std::__1::__map_value_compare\2c\20std::__1::allocator\20>\20>\2c\20std::__1::less\2c\20true>\2c\20std::__1::allocator\2c\20std::__1::allocator\20>\20>\20>\20>::destroy\28std::__1::__tree_node\2c\20std::__1::allocator\20>\20>\2c\20void*>*\29 +std::__1::__tree\2c\20std::__1::allocator\20>\2c\20int>\2c\20std::__1::__map_value_compare\2c\20std::__1::allocator\20>\2c\20std::__1::__value_type\2c\20std::__1::allocator\20>\2c\20int>\2c\20std::__1::less\2c\20std::__1::allocator\20>\20>\2c\20true>\2c\20std::__1::allocator\2c\20std::__1::allocator\20>\2c\20int>\20>\20>::destroy\28std::__1::__tree_node\2c\20std::__1::allocator\20>\2c\20int>\2c\20void*>*\29,std::__1::__tree\2c\20std::__1::allocator\20>\2c\20int>\2c\20std::__1::__map_value_compare\2c\20std::__1::allocator\20>\2c\20std::__1::__value_type\2c\20std::__1::allocator\20>\2c\20int>\2c\20std::__1::less\2c\20std::__1::allocator\20>\20>\2c\20true>\2c\20std::__1::allocator\2c\20std::__1::allocator\20>\2c\20int>\20>\20>::destroy\28std::__1::__tree_node\2c\20std::__1::allocator\20>\2c\20int>\2c\20void*>*\29 +std::__1::__tree\2c\20std::__1::allocator\20>\2c\20int>\2c\20std::__1::__map_value_compare\2c\20std::__1::allocator\20>\2c\20std::__1::__value_type\2c\20std::__1::allocator\20>\2c\20int>\2c\20std::__1::less\2c\20std::__1::allocator\20>\20>\2c\20true>\2c\20std::__1::allocator\2c\20std::__1::allocator\20>\2c\20int>\20>\20>::destroy\28std::__1::__tree_node\2c\20std::__1::allocator\20>\2c\20int>\2c\20void*>*\29,operator\20delete\28void*\29 +std::__1::__tree\2c\20std::__1::allocator\20>\20>\2c\20std::__1::__map_value_compare\2c\20std::__1::allocator\20>\20>\2c\20std::__1::less\2c\20true>\2c\20std::__1::allocator\2c\20std::__1::allocator\20>\20>\20>\20>::destroy\28std::__1::__tree_node\2c\20std::__1::allocator\20>\20>\2c\20void*>*\29,std::__1::__tree\2c\20std::__1::allocator\20>\20>\2c\20std::__1::__map_value_compare\2c\20std::__1::allocator\20>\20>\2c\20std::__1::less\2c\20true>\2c\20std::__1::allocator\2c\20std::__1::allocator\20>\20>\20>\20>::destroy\28std::__1::__tree_node\2c\20std::__1::allocator\20>\20>\2c\20void*>*\29 +std::__1::__tree\2c\20std::__1::allocator\20>\20>\2c\20std::__1::__map_value_compare\2c\20std::__1::allocator\20>\20>\2c\20std::__1::less\2c\20true>\2c\20std::__1::allocator\2c\20std::__1::allocator\20>\20>\20>\20>::destroy\28std::__1::__tree_node\2c\20std::__1::allocator\20>\20>\2c\20void*>*\29,operator\20delete\28void*\29 +re2::RE2::DoMatch\28re2::StringPiece\20const&\2c\20re2::RE2::Anchor\2c\20unsigned\20long*\2c\20re2::RE2::Arg\20const*\20const*\2c\20int\29\20const,memset +re2::RE2::DoMatch\28re2::StringPiece\20const&\2c\20re2::RE2::Anchor\2c\20unsigned\20long*\2c\20re2::RE2::Arg\20const*\20const*\2c\20int\29\20const,operator\20new\5b\5d\28unsigned\20long\29 +re2::RE2::DoMatch\28re2::StringPiece\20const&\2c\20re2::RE2::Anchor\2c\20unsigned\20long*\2c\20re2::RE2::Arg\20const*\20const*\2c\20int\29\20const,re2::RE2::Match\28re2::StringPiece\20const&\2c\20unsigned\20long\2c\20unsigned\20long\2c\20re2::RE2::Anchor\2c\20re2::StringPiece*\2c\20int\29\20const +re2::RE2::DoMatch\28re2::StringPiece\20const&\2c\20re2::RE2::Anchor\2c\20unsigned\20long*\2c\20re2::RE2::Arg\20const*\20const*\2c\20int\29\20const,operator\20delete\5b\5d\28void*\29 +re2::RE2::Match\28re2::StringPiece\20const&\2c\20unsigned\20long\2c\20unsigned\20long\2c\20re2::RE2::Anchor\2c\20re2::StringPiece*\2c\20int\29\20const,memcmp +re2::RE2::Match\28re2::StringPiece\20const&\2c\20unsigned\20long\2c\20unsigned\20long\2c\20re2::RE2::Anchor\2c\20re2::StringPiece*\2c\20int\29\20const,re2::RE2::ReverseProg\28\29\20const +re2::RE2::Match\28re2::StringPiece\20const&\2c\20unsigned\20long\2c\20unsigned\20long\2c\20re2::RE2::Anchor\2c\20re2::StringPiece*\2c\20int\29\20const,re2::Prog::SearchDFA\28re2::StringPiece\20const&\2c\20re2::StringPiece\20const&\2c\20re2::Prog::Anchor\2c\20re2::Prog::MatchKind\2c\20re2::StringPiece*\2c\20bool*\2c\20re2::SparseSetT*\29 +re2::RE2::Match\28re2::StringPiece\20const&\2c\20unsigned\20long\2c\20unsigned\20long\2c\20re2::RE2::Anchor\2c\20re2::StringPiece*\2c\20int\29\20const,re2::Prog::SearchOnePass\28re2::StringPiece\20const&\2c\20re2::StringPiece\20const&\2c\20re2::Prog::Anchor\2c\20re2::Prog::MatchKind\2c\20re2::StringPiece*\2c\20int\29 +re2::RE2::Match\28re2::StringPiece\20const&\2c\20unsigned\20long\2c\20unsigned\20long\2c\20re2::RE2::Anchor\2c\20re2::StringPiece*\2c\20int\29\20const,re2::Prog::SearchBitState\28re2::StringPiece\20const&\2c\20re2::StringPiece\20const&\2c\20re2::Prog::Anchor\2c\20re2::Prog::MatchKind\2c\20re2::StringPiece*\2c\20int\29 +re2::RE2::Match\28re2::StringPiece\20const&\2c\20unsigned\20long\2c\20unsigned\20long\2c\20re2::RE2::Anchor\2c\20re2::StringPiece*\2c\20int\29\20const,re2::Prog::SearchNFA\28re2::StringPiece\20const&\2c\20re2::StringPiece\20const&\2c\20re2::Prog::Anchor\2c\20re2::Prog::MatchKind\2c\20re2::StringPiece*\2c\20int\29 +re2::RE2::Match\28re2::StringPiece\20const&\2c\20unsigned\20long\2c\20unsigned\20long\2c\20re2::RE2::Anchor\2c\20re2::StringPiece*\2c\20int\29\20const,memset +re2::RE2::PartialMatchN\28re2::StringPiece\20const&\2c\20re2::RE2\20const&\2c\20re2::RE2::Arg\20const*\20const*\2c\20int\29,re2::RE2::DoMatch\28re2::StringPiece\20const&\2c\20re2::RE2::Anchor\2c\20unsigned\20long*\2c\20re2::RE2::Arg\20const*\20const*\2c\20int\29\20const +re2::Regexp::~Regexp\28\29,operator\20delete\28void*\29 +re2::Regexp::~Regexp\28\29,operator\20delete\5b\5d\28void*\29 +re2::Regexp::~Regexp\28\29,std::__1::__tree\20>::destroy\28std::__1::__tree_node*\29 +re2::Regexp::Incref\28\29,std::__1::__call_once\28unsigned\20long\20volatile&\2c\20void*\2c\20void\20\28*\29\28void*\29\29 +re2::Regexp::Incref\28\29,operator\20new\28unsigned\20long\29 +re2::Regexp::Incref\28\29,void\20std::__1::__tree_balance_after_insert*>\28std::__1::__tree_node_base*\2c\20std::__1::__tree_node_base*\29 +void\20std::__1::__call_once_proxy\20>\28void*\29,operator\20new\28unsigned\20long\29 +re2::Regexp::Decref\28\29,operator\20new\28unsigned\20long\29 +re2::Regexp::Decref\28\29,void\20std::__1::__tree_balance_after_insert*>\28std::__1::__tree_node_base*\2c\20std::__1::__tree_node_base*\29 +re2::Regexp::Decref\28\29,void\20std::__1::__tree_remove*>\28std::__1::__tree_node_base*\2c\20std::__1::__tree_node_base*\29 +re2::Regexp::Decref\28\29,operator\20delete\28void*\29 +re2::Regexp::Decref\28\29,re2::Regexp::Destroy\28\29 +re2::Regexp::Destroy\28\29,re2::Regexp::~Regexp\28\29 +re2::Regexp::Destroy\28\29,operator\20delete\28void*\29 +re2::Regexp::Destroy\28\29,re2::Regexp::Decref\28\29 +re2::Regexp::Destroy\28\29,operator\20delete\5b\5d\28void*\29 +re2::Regexp::AddRuneToString\28int\29,opa_abort +re2::Regexp::AddRuneToString\28int\29,operator\20new\5b\5d\28unsigned\20long\29 +re2::Regexp::AddRuneToString\28int\29,operator\20delete\5b\5d\28void*\29 +re2::Regexp::StarPlusOrQuest\28re2::RegexpOp\2c\20re2::Regexp*\2c\20re2::Regexp::ParseFlags\29,operator\20new\28unsigned\20long\29 +re2::Regexp::StarPlusOrQuest\28re2::RegexpOp\2c\20re2::Regexp*\2c\20re2::Regexp::ParseFlags\29,re2::Regexp::Incref\28\29 +re2::Regexp::StarPlusOrQuest\28re2::RegexpOp\2c\20re2::Regexp*\2c\20re2::Regexp::ParseFlags\29,re2::Regexp::Decref\28\29 +re2::Regexp::Plus\28re2::Regexp*\2c\20re2::Regexp::ParseFlags\29,re2::Regexp::StarPlusOrQuest\28re2::RegexpOp\2c\20re2::Regexp*\2c\20re2::Regexp::ParseFlags\29 +re2::Regexp::Star\28re2::Regexp*\2c\20re2::Regexp::ParseFlags\29,re2::Regexp::StarPlusOrQuest\28re2::RegexpOp\2c\20re2::Regexp*\2c\20re2::Regexp::ParseFlags\29 +re2::Regexp::Quest\28re2::Regexp*\2c\20re2::Regexp::ParseFlags\29,re2::Regexp::StarPlusOrQuest\28re2::RegexpOp\2c\20re2::Regexp*\2c\20re2::Regexp::ParseFlags\29 +re2::Regexp::ConcatOrAlternate\28re2::RegexpOp\2c\20re2::Regexp**\2c\20int\2c\20re2::Regexp::ParseFlags\2c\20bool\29,operator\20new\28unsigned\20long\29 +re2::Regexp::ConcatOrAlternate\28re2::RegexpOp\2c\20re2::Regexp**\2c\20int\2c\20re2::Regexp::ParseFlags\2c\20bool\29,memcpy +re2::Regexp::ConcatOrAlternate\28re2::RegexpOp\2c\20re2::Regexp**\2c\20int\2c\20re2::Regexp::ParseFlags\2c\20bool\29,re2::Regexp::FactorAlternation\28re2::Regexp**\2c\20int\2c\20re2::Regexp::ParseFlags\29 +re2::Regexp::ConcatOrAlternate\28re2::RegexpOp\2c\20re2::Regexp**\2c\20int\2c\20re2::Regexp::ParseFlags\2c\20bool\29,operator\20new\5b\5d\28unsigned\20long\29 +re2::Regexp::ConcatOrAlternate\28re2::RegexpOp\2c\20re2::Regexp**\2c\20int\2c\20re2::Regexp::ParseFlags\2c\20bool\29,re2::Regexp::ConcatOrAlternate\28re2::RegexpOp\2c\20re2::Regexp**\2c\20int\2c\20re2::Regexp::ParseFlags\2c\20bool\29 +re2::Regexp::ConcatOrAlternate\28re2::RegexpOp\2c\20re2::Regexp**\2c\20int\2c\20re2::Regexp::ParseFlags\2c\20bool\29,opa_abort +re2::Regexp::ConcatOrAlternate\28re2::RegexpOp\2c\20re2::Regexp**\2c\20int\2c\20re2::Regexp::ParseFlags\2c\20bool\29,operator\20delete\28void*\29 +re2::Regexp::ConcatOrAlternate\28re2::RegexpOp\2c\20re2::Regexp**\2c\20int\2c\20re2::Regexp::ParseFlags\2c\20bool\29,abort +re2::Regexp::Concat\28re2::Regexp**\2c\20int\2c\20re2::Regexp::ParseFlags\29,re2::Regexp::ConcatOrAlternate\28re2::RegexpOp\2c\20re2::Regexp**\2c\20int\2c\20re2::Regexp::ParseFlags\2c\20bool\29 +re2::Regexp::AlternateNoFactor\28re2::Regexp**\2c\20int\2c\20re2::Regexp::ParseFlags\29,re2::Regexp::ConcatOrAlternate\28re2::RegexpOp\2c\20re2::Regexp**\2c\20int\2c\20re2::Regexp::ParseFlags\2c\20bool\29 +re2::Regexp::Capture\28re2::Regexp*\2c\20re2::Regexp::ParseFlags\2c\20int\29,operator\20new\28unsigned\20long\29 +re2::Regexp::Repeat\28re2::Regexp*\2c\20re2::Regexp::ParseFlags\2c\20int\2c\20int\29,operator\20new\28unsigned\20long\29 +re2::Regexp::LiteralString\28int*\2c\20int\2c\20re2::Regexp::ParseFlags\29,operator\20new\28unsigned\20long\29 +re2::Regexp::LiteralString\28int*\2c\20int\2c\20re2::Regexp::ParseFlags\29,re2::Regexp::AddRuneToString\28int\29 +re2::Regexp::NewCharClass\28re2::CharClass*\2c\20re2::Regexp::ParseFlags\29,operator\20new\28unsigned\20long\29 +re2::Regexp::Equal\28re2::Regexp*\2c\20re2::Regexp*\29,re2::TopEqual\28re2::Regexp*\2c\20re2::Regexp*\29 +re2::Regexp::Equal\28re2::Regexp*\2c\20re2::Regexp*\29,operator\20new\28unsigned\20long\29 +re2::Regexp::Equal\28re2::Regexp*\2c\20re2::Regexp*\29,memcpy +re2::Regexp::Equal\28re2::Regexp*\2c\20re2::Regexp*\29,operator\20delete\28void*\29 +re2::Regexp::Equal\28re2::Regexp*\2c\20re2::Regexp*\29,opa_abort +re2::Regexp::Equal\28re2::Regexp*\2c\20re2::Regexp*\29,std::__1::vector\20>::__append\28unsigned\20long\29 +re2::Regexp::Equal\28re2::Regexp*\2c\20re2::Regexp*\29,abort +re2::TopEqual\28re2::Regexp*\2c\20re2::Regexp*\29,memcmp +std::__1::vector\20>::__append\28unsigned\20long\29,memset +std::__1::vector\20>::__append\28unsigned\20long\29,operator\20new\28unsigned\20long\29 +std::__1::vector\20>::__append\28unsigned\20long\29,memcpy +std::__1::vector\20>::__append\28unsigned\20long\29,operator\20delete\28void*\29 +std::__1::vector\20>::__append\28unsigned\20long\29,abort +re2::RegexpStatus::Text\28\29\20const,strlen +re2::RegexpStatus::Text\28\29\20const,operator\20new\28unsigned\20long\29 +re2::RegexpStatus::Text\28\29\20const,memcpy +re2::RegexpStatus::Text\28\29\20const,std::__1::basic_string\2c\20std::__1::allocator\20>::append\28char\20const*\2c\20unsigned\20long\29 +re2::RegexpStatus::Text\28\29\20const,operator\20delete\28void*\29 +re2::RegexpStatus::Text\28\29\20const,std::__1::basic_string\2c\20std::__1::allocator\20>::append\28char\20const*\29 +re2::RegexpStatus::Text\28\29\20const,abort +re2::Regexp::NumCaptures\28\29,re2::Regexp::Walker::WalkInternal\28re2::Regexp*\2c\20int\2c\20bool\29 +re2::Regexp::NumCaptures\28\29,re2::Regexp::Walker::~Walker\28\29 +re2::ConvertRunesToBytes\28bool\2c\20int*\2c\20int\2c\20std::__1::basic_string\2c\20std::__1::allocator\20>*\29,std::__1::basic_string\2c\20std::__1::allocator\20>::resize\28unsigned\20long\2c\20char\29 +re2::ConvertRunesToBytes\28bool\2c\20int*\2c\20int\2c\20std::__1::basic_string\2c\20std::__1::allocator\20>*\29,runetochar +re2::ConvertRunesToBytes\28bool\2c\20int*\2c\20int\2c\20std::__1::basic_string\2c\20std::__1::allocator\20>*\29,operator\20new\28unsigned\20long\29 +re2::ConvertRunesToBytes\28bool\2c\20int*\2c\20int\2c\20std::__1::basic_string\2c\20std::__1::allocator\20>*\29,memcpy +re2::ConvertRunesToBytes\28bool\2c\20int*\2c\20int\2c\20std::__1::basic_string\2c\20std::__1::allocator\20>*\29,operator\20delete\28void*\29 +re2::Regexp::RequiredPrefix\28std::__1::basic_string\2c\20std::__1::allocator\20>*\2c\20bool*\2c\20re2::Regexp**\29,re2::Regexp::Incref\28\29 +re2::Regexp::RequiredPrefix\28std::__1::basic_string\2c\20std::__1::allocator\20>*\2c\20bool*\2c\20re2::Regexp**\29,re2::Regexp::ConcatOrAlternate\28re2::RegexpOp\2c\20re2::Regexp**\2c\20int\2c\20re2::Regexp::ParseFlags\2c\20bool\29 +re2::Regexp::RequiredPrefix\28std::__1::basic_string\2c\20std::__1::allocator\20>*\2c\20bool*\2c\20re2::Regexp**\29,operator\20new\28unsigned\20long\29 +re2::Regexp::RequiredPrefix\28std::__1::basic_string\2c\20std::__1::allocator\20>*\2c\20bool*\2c\20re2::Regexp**\29,re2::ConvertRunesToBytes\28bool\2c\20int*\2c\20int\2c\20std::__1::basic_string\2c\20std::__1::allocator\20>*\29 +re2::Regexp::RequiredPrefixForAccel\28std::__1::basic_string\2c\20std::__1::allocator\20>*\2c\20bool*\29,re2::ConvertRunesToBytes\28bool\2c\20int*\2c\20int\2c\20std::__1::basic_string\2c\20std::__1::allocator\20>*\29 +re2::CharClassBuilder::AddRange\28int\2c\20int\29,void\20std::__1::__tree_remove*>\28std::__1::__tree_node_base*\2c\20std::__1::__tree_node_base*\29 +re2::CharClassBuilder::AddRange\28int\2c\20int\29,operator\20delete\28void*\29 +re2::CharClassBuilder::AddRange\28int\2c\20int\29,operator\20new\28unsigned\20long\29 +re2::CharClassBuilder::AddRange\28int\2c\20int\29,void\20std::__1::__tree_balance_after_insert*>\28std::__1::__tree_node_base*\2c\20std::__1::__tree_node_base*\29 +re2::CharClassBuilder::AddCharClass\28re2::CharClassBuilder*\29,re2::CharClassBuilder::AddRange\28int\2c\20int\29 +re2::CharClassBuilder::RemoveAbove\28int\29,void\20std::__1::__tree_remove*>\28std::__1::__tree_node_base*\2c\20std::__1::__tree_node_base*\29 +re2::CharClassBuilder::RemoveAbove\28int\29,operator\20delete\28void*\29 +re2::CharClassBuilder::RemoveAbove\28int\29,operator\20new\28unsigned\20long\29 +re2::CharClassBuilder::RemoveAbove\28int\29,void\20std::__1::__tree_balance_after_insert*>\28std::__1::__tree_node_base*\2c\20std::__1::__tree_node_base*\29 +re2::CharClassBuilder::Negate\28\29,operator\20new\28unsigned\20long\29 +re2::CharClassBuilder::Negate\28\29,memcpy +re2::CharClassBuilder::Negate\28\29,operator\20delete\28void*\29 +re2::CharClassBuilder::Negate\28\29,std::__1::__tree\20>::destroy\28std::__1::__tree_node*\29 +re2::CharClassBuilder::Negate\28\29,void\20std::__1::__tree_balance_after_insert*>\28std::__1::__tree_node_base*\2c\20std::__1::__tree_node_base*\29 +re2::CharClassBuilder::Negate\28\29,abort +re2::CharClassBuilder::GetCharClass\28\29,operator\20new\5b\5d\28unsigned\20long\29 +re2::CharClassBuilder::GetCharClass\28\29,opa_abort +re2::NumCapturesWalker::~NumCapturesWalker\28\29,re2::Regexp::Walker::~Walker\28\29 +re2::NumCapturesWalker::~NumCapturesWalker\28\29,operator\20delete\28void*\29 +re2::Regexp::Simplify\28\29,re2::Regexp::Walker::WalkInternal\28re2::Regexp*\2c\20re2::Regexp*\2c\20bool\29 +re2::Regexp::Simplify\28\29,re2::Regexp::Decref\28\29 +re2::Regexp::Simplify\28\29,re2::Regexp::Walker::~Walker\28\29 +re2::Regexp::Walker::WalkInternal\28re2::Regexp*\2c\20re2::Regexp*\2c\20bool\29,operator\20delete\5b\5d\28void*\29 +re2::Regexp::Walker::WalkInternal\28re2::Regexp*\2c\20re2::Regexp*\2c\20bool\29,operator\20delete\28void*\29 +re2::Regexp::Walker::WalkInternal\28re2::Regexp*\2c\20re2::Regexp*\2c\20bool\29,std::__1::deque\2c\20std::__1::allocator\20>\20>::__add_back_capacity\28\29 +re2::Regexp::Walker::WalkInternal\28re2::Regexp*\2c\20re2::Regexp*\2c\20bool\29,operator\20new\5b\5d\28unsigned\20long\29 +re2::Regexp::Walker::~Walker\28\29,operator\20delete\5b\5d\28void*\29 +re2::Regexp::Walker::~Walker\28\29,operator\20delete\28void*\29 +re2::Regexp::Walker::~Walker\28\29,std::__1::__deque_base\2c\20std::__1::allocator\20>\20>::~__deque_base\28\29 +re2::CoalesceWalker::Copy\28re2::Regexp*\29,re2::Regexp::Incref\28\29 +re2::CoalesceWalker::ShortVisit\28re2::Regexp*\2c\20re2::Regexp*\29,re2::Regexp::Incref\28\29 +re2::CoalesceWalker::PostVisit\28re2::Regexp*\2c\20re2::Regexp*\2c\20re2::Regexp*\2c\20re2::Regexp**\2c\20int\29,re2::Regexp::Incref\28\29 +re2::CoalesceWalker::PostVisit\28re2::Regexp*\2c\20re2::Regexp*\2c\20re2::Regexp*\2c\20re2::Regexp**\2c\20int\29,re2::CoalesceWalker::CanCoalesce\28re2::Regexp*\2c\20re2::Regexp*\29 +re2::CoalesceWalker::PostVisit\28re2::Regexp*\2c\20re2::Regexp*\2c\20re2::Regexp*\2c\20re2::Regexp**\2c\20int\29,re2::Regexp::Decref\28\29 +re2::CoalesceWalker::PostVisit\28re2::Regexp*\2c\20re2::Regexp*\2c\20re2::Regexp*\2c\20re2::Regexp**\2c\20int\29,re2::CoalesceWalker::DoCoalesce\28re2::Regexp**\2c\20re2::Regexp**\29 +re2::CoalesceWalker::PostVisit\28re2::Regexp*\2c\20re2::Regexp*\2c\20re2::Regexp*\2c\20re2::Regexp**\2c\20int\29,operator\20new\28unsigned\20long\29 +re2::CoalesceWalker::PostVisit\28re2::Regexp*\2c\20re2::Regexp*\2c\20re2::Regexp*\2c\20re2::Regexp**\2c\20int\29,re2::Regexp::Regexp\28re2::RegexpOp\2c\20re2::Regexp::ParseFlags\29 +re2::CoalesceWalker::PostVisit\28re2::Regexp*\2c\20re2::Regexp*\2c\20re2::Regexp*\2c\20re2::Regexp**\2c\20int\29,opa_abort +re2::CoalesceWalker::PostVisit\28re2::Regexp*\2c\20re2::Regexp*\2c\20re2::Regexp*\2c\20re2::Regexp**\2c\20int\29,operator\20new\5b\5d\28unsigned\20long\29 +re2::CoalesceWalker::CanCoalesce\28re2::Regexp*\2c\20re2::Regexp*\29,re2::Regexp::Equal\28re2::Regexp*\2c\20re2::Regexp*\29 +re2::CoalesceWalker::DoCoalesce\28re2::Regexp**\2c\20re2::Regexp**\29,re2::Regexp::Incref\28\29 +re2::CoalesceWalker::DoCoalesce\28re2::Regexp**\2c\20re2::Regexp**\29,re2::Regexp::Repeat\28re2::Regexp*\2c\20re2::Regexp::ParseFlags\2c\20int\2c\20int\29 +re2::CoalesceWalker::DoCoalesce\28re2::Regexp**\2c\20re2::Regexp**\29,opa_abort +re2::CoalesceWalker::DoCoalesce\28re2::Regexp**\2c\20re2::Regexp**\29,re2::Regexp::LiteralString\28int*\2c\20int\2c\20re2::Regexp::ParseFlags\29 +re2::CoalesceWalker::DoCoalesce\28re2::Regexp**\2c\20re2::Regexp**\29,operator\20new\28unsigned\20long\29 +re2::CoalesceWalker::DoCoalesce\28re2::Regexp**\2c\20re2::Regexp**\29,re2::Regexp::Regexp\28re2::RegexpOp\2c\20re2::Regexp::ParseFlags\29 +re2::CoalesceWalker::DoCoalesce\28re2::Regexp**\2c\20re2::Regexp**\29,re2::Regexp::Decref\28\29 +re2::SimplifyWalker::Copy\28re2::Regexp*\29,re2::Regexp::Incref\28\29 +re2::SimplifyWalker::ShortVisit\28re2::Regexp*\2c\20re2::Regexp*\29,re2::Regexp::Incref\28\29 +re2::SimplifyWalker::PreVisit\28re2::Regexp*\2c\20re2::Regexp*\2c\20bool*\29,re2::Regexp::Incref\28\29 +re2::SimplifyWalker::PostVisit\28re2::Regexp*\2c\20re2::Regexp*\2c\20re2::Regexp*\2c\20re2::Regexp**\2c\20int\29,re2::Regexp::Decref\28\29 +re2::SimplifyWalker::PostVisit\28re2::Regexp*\2c\20re2::Regexp*\2c\20re2::Regexp*\2c\20re2::Regexp**\2c\20int\29,re2::Regexp::Incref\28\29 +re2::SimplifyWalker::PostVisit\28re2::Regexp*\2c\20re2::Regexp*\2c\20re2::Regexp*\2c\20re2::Regexp**\2c\20int\29,re2::SimplifyWalker::SimplifyCharClass\28re2::Regexp*\29 +re2::SimplifyWalker::PostVisit\28re2::Regexp*\2c\20re2::Regexp*\2c\20re2::Regexp*\2c\20re2::Regexp**\2c\20int\29,re2::SimplifyWalker::SimplifyRepeat\28re2::Regexp*\2c\20int\2c\20int\2c\20re2::Regexp::ParseFlags\29 +re2::SimplifyWalker::PostVisit\28re2::Regexp*\2c\20re2::Regexp*\2c\20re2::Regexp*\2c\20re2::Regexp**\2c\20int\29,operator\20new\28unsigned\20long\29 +re2::SimplifyWalker::PostVisit\28re2::Regexp*\2c\20re2::Regexp*\2c\20re2::Regexp*\2c\20re2::Regexp**\2c\20int\29,re2::Regexp::Regexp\28re2::RegexpOp\2c\20re2::Regexp::ParseFlags\29 +re2::SimplifyWalker::PostVisit\28re2::Regexp*\2c\20re2::Regexp*\2c\20re2::Regexp*\2c\20re2::Regexp**\2c\20int\29,opa_abort +re2::SimplifyWalker::PostVisit\28re2::Regexp*\2c\20re2::Regexp*\2c\20re2::Regexp*\2c\20re2::Regexp**\2c\20int\29,operator\20new\5b\5d\28unsigned\20long\29 +re2::SimplifyWalker::SimplifyCharClass\28re2::Regexp*\29,opa_abort +re2::SimplifyWalker::SimplifyCharClass\28re2::Regexp*\29,operator\20new\28unsigned\20long\29 +re2::SimplifyWalker::SimplifyCharClass\28re2::Regexp*\29,re2::Regexp::Regexp\28re2::RegexpOp\2c\20re2::Regexp::ParseFlags\29 +re2::SimplifyWalker::SimplifyCharClass\28re2::Regexp*\29,re2::Regexp::Incref\28\29 +re2::SimplifyWalker::SimplifyRepeat\28re2::Regexp*\2c\20int\2c\20int\2c\20re2::Regexp::ParseFlags\29,re2::Regexp::Incref\28\29 +re2::SimplifyWalker::SimplifyRepeat\28re2::Regexp*\2c\20int\2c\20int\2c\20re2::Regexp::ParseFlags\29,re2::Regexp::Star\28re2::Regexp*\2c\20re2::Regexp::ParseFlags\29 +re2::SimplifyWalker::SimplifyRepeat\28re2::Regexp*\2c\20int\2c\20int\2c\20re2::Regexp::ParseFlags\29,re2::Regexp::Plus\28re2::Regexp*\2c\20re2::Regexp::ParseFlags\29 +re2::SimplifyWalker::SimplifyRepeat\28re2::Regexp*\2c\20int\2c\20int\2c\20re2::Regexp::ParseFlags\29,operator\20new\28unsigned\20long\29 +re2::SimplifyWalker::SimplifyRepeat\28re2::Regexp*\2c\20int\2c\20int\2c\20re2::Regexp::ParseFlags\29,re2::Regexp::Concat\28re2::Regexp**\2c\20int\2c\20re2::Regexp::ParseFlags\29 +re2::SimplifyWalker::SimplifyRepeat\28re2::Regexp*\2c\20int\2c\20int\2c\20re2::Regexp::ParseFlags\29,operator\20delete\28void*\29 +re2::SimplifyWalker::SimplifyRepeat\28re2::Regexp*\2c\20int\2c\20int\2c\20re2::Regexp::ParseFlags\29,re2::Regexp::Regexp\28re2::RegexpOp\2c\20re2::Regexp::ParseFlags\29 +re2::SimplifyWalker::SimplifyRepeat\28re2::Regexp*\2c\20int\2c\20int\2c\20re2::Regexp::ParseFlags\29,re2::Regexp::Quest\28re2::Regexp*\2c\20re2::Regexp::ParseFlags\29 +re2::SimplifyWalker::SimplifyRepeat\28re2::Regexp*\2c\20int\2c\20int\2c\20re2::Regexp::ParseFlags\29,operator\20new\5b\5d\28unsigned\20long\29 +re2::SimplifyWalker::SimplifyRepeat\28re2::Regexp*\2c\20int\2c\20int\2c\20re2::Regexp::ParseFlags\29,abort +re2::CoalesceWalker::~CoalesceWalker\28\29,re2::Regexp::Walker::~Walker\28\29 +re2::CoalesceWalker::~CoalesceWalker\28\29,operator\20delete\28void*\29 +re2::SimplifyWalker::~SimplifyWalker\28\29,re2::Regexp::Walker::~Walker\28\29 +re2::SimplifyWalker::~SimplifyWalker\28\29,operator\20delete\28void*\29 +std::__1::__deque_base\2c\20std::__1::allocator\20>\20>::~__deque_base\28\29,operator\20delete\28void*\29 +std::__1::deque\2c\20std::__1::allocator\20>\20>::__add_back_capacity\28\29,memmove +std::__1::deque\2c\20std::__1::allocator\20>\20>::__add_back_capacity\28\29,operator\20new\28unsigned\20long\29 +std::__1::deque\2c\20std::__1::allocator\20>\20>::__add_back_capacity\28\29,operator\20delete\28void*\29 +std::__1::deque\2c\20std::__1::allocator\20>\20>::__add_back_capacity\28\29,std::__1::__split_buffer*\2c\20std::__1::allocator*>\20>::push_back\28re2::WalkState*&&\29 +std::__1::deque\2c\20std::__1::allocator\20>\20>::__add_back_capacity\28\29,std::__1::__split_buffer*\2c\20std::__1::allocator*>\20>::push_front\28re2::WalkState*&&\29 +std::__1::deque\2c\20std::__1::allocator\20>\20>::__add_back_capacity\28\29,std::__1::__split_buffer*\2c\20std::__1::allocator*>&>::push_back\28re2::WalkState*&&\29 +std::__1::deque\2c\20std::__1::allocator\20>\20>::__add_back_capacity\28\29,std::__1::__split_buffer*\2c\20std::__1::allocator*>&>::push_front\28re2::WalkState*\20const&\29 +std::__1::deque\2c\20std::__1::allocator\20>\20>::__add_back_capacity\28\29,abort +std::__1::__split_buffer*\2c\20std::__1::allocator*>\20>::push_back\28re2::WalkState*&&\29,memmove +std::__1::__split_buffer*\2c\20std::__1::allocator*>\20>::push_back\28re2::WalkState*&&\29,operator\20new\28unsigned\20long\29 +std::__1::__split_buffer*\2c\20std::__1::allocator*>\20>::push_back\28re2::WalkState*&&\29,operator\20delete\28void*\29 +std::__1::__split_buffer*\2c\20std::__1::allocator*>\20>::push_back\28re2::WalkState*&&\29,abort +std::__1::__split_buffer*\2c\20std::__1::allocator*>\20>::push_front\28re2::WalkState*&&\29,memmove +std::__1::__split_buffer*\2c\20std::__1::allocator*>\20>::push_front\28re2::WalkState*&&\29,operator\20new\28unsigned\20long\29 +std::__1::__split_buffer*\2c\20std::__1::allocator*>\20>::push_front\28re2::WalkState*&&\29,operator\20delete\28void*\29 +std::__1::__split_buffer*\2c\20std::__1::allocator*>\20>::push_front\28re2::WalkState*&&\29,abort +std::__1::__split_buffer*\2c\20std::__1::allocator*>&>::push_back\28re2::WalkState*&&\29,memmove +std::__1::__split_buffer*\2c\20std::__1::allocator*>&>::push_back\28re2::WalkState*&&\29,operator\20new\28unsigned\20long\29 +std::__1::__split_buffer*\2c\20std::__1::allocator*>&>::push_back\28re2::WalkState*&&\29,operator\20delete\28void*\29 +std::__1::__split_buffer*\2c\20std::__1::allocator*>&>::push_back\28re2::WalkState*&&\29,abort +std::__1::__split_buffer*\2c\20std::__1::allocator*>&>::push_front\28re2::WalkState*\20const&\29,memmove +std::__1::__split_buffer*\2c\20std::__1::allocator*>&>::push_front\28re2::WalkState*\20const&\29,operator\20new\28unsigned\20long\29 +std::__1::__split_buffer*\2c\20std::__1::allocator*>&>::push_front\28re2::WalkState*\20const&\29,operator\20delete\28void*\29 +std::__1::__split_buffer*\2c\20std::__1::allocator*>&>::push_front\28re2::WalkState*\20const&\29,abort diff --git a/vendor/github.com/open-policy-agent/opa/internal/compiler/wasm/opa/opa.go b/vendor/github.com/open-policy-agent/opa/internal/compiler/wasm/opa/opa.go new file mode 100644 index 00000000..8c1c9d90 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/compiler/wasm/opa/opa.go @@ -0,0 +1,51 @@ +// Copyright 2018 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +// THIS FILE IS GENERATED. DO NOT EDIT. + +// Package opa contains bytecode for the OPA-WASM library. +package opa + +import ( + "bytes" + "compress/gzip" + "io/ioutil" + "sync" +) + +var ( + bytesOnce sync.Once + bs []byte + callGraphCSVOnce sync.Once + callGraphCSV []byte +) + +// Bytes returns the OPA-WASM bytecode. +func Bytes() ([]byte, error) { + var err error + bytesOnce.Do(func() { + gr, err := gzip.NewReader(bytes.NewBuffer(gzipped)) + if err != nil { + return + } + bs, err = ioutil.ReadAll(gr) + }) + return bs, err +} + +// CallGraphCSV returns a CSV representation of the +// OPA-WASM bytecode's call graph: 'caller,callee' +func CallGraphCSV() ([]byte, error) { + var err error + callGraphCSVOnce.Do(func() { + cg, err := gzip.NewReader(bytes.NewBuffer(gzippedCallGraphCSV)) + if err != nil { + return + } + callGraphCSV, err = ioutil.ReadAll(cg) + }) + return callGraphCSV, err +} +var gzipped = []byte("\x1F\x8B\x08\x00\x00\x00\x00\x00\x00\xFF\xEC\xBD\x0B\xB4\x65\x57\x55\x20\xBA\xBE\x7B\xEF\x73\xF6\xD9\xF7\x9E\xAA\x54\xA5\x8A\xDC\x20\x6B\x6F\x79\xA3\x2B\x9A\x52\xB4\xB5\xE2\x0B\xF0\xA8\x75\x47\xDF\xAA\xDC\x14\x50\xA8\xA0\xAF\x19\xFD\x1E\x89\x1A\xBB\xDD\x37\x40\x55\xE5\xA6\x8C\xC3\xA2\xCE\x2D\x2A\x90\xF0\x33\xD8\x2A\x1F\x45\x09\x10\x0D\xD8\x44\xA0\x45\x7E\xCA\x4B\x44\x14\x68\x51\x71\x88\x43\x10\xD4\xA0\x68\xD3\x08\x18\x21\xF0\x90\x5F\x7A\xCC\xCF\x5A\x7B\xED\x73\xCE\xBD\x55\x37\x84\x8F\x48\x29\xB9\x67\xAF\xBD\xF6\xFA\xCC\x39\xD7\xFC\xAC\x35\xD7\x9C\xE2\xEA\xEB\x9E\x2C\x85\x10\xF2\x3D\x72\xE9\x2A\x3D\x99\x4C\xE4\xE4\x2A\x15\xFE\x2B\xAE\x32\x93\x09\xFC\x91\xF8\x3F\x39\xB9\xCA\x4E\xA8\x40\xE3\x7F\xF3\xC9\x84\x9F\xB1\x9E\x9C\x5C\x95\x51\x41\xAC\x29\x27\x57\x09\x39\x89\x15\xE1\x51\x5C\xA5\x26\xA7\xE5\xE4\x2A\x7D\x1A\x9F\xE5\x69\xEC\xEB\xB4\xB8\x4A\x9E\x92\xA7\xAE\x2A\x27\xDD\x3F\x39\xB9\xAA\x80\xBF\xA7\x92\xDF\xFD\x7E\x70\x20\xD0\x00\xF4\x0F\x7F\x07\xC9\xC7\xEA\x58\xA6\xAF\x79\xCA\xC9\xD1\x53\x8F\x5D\xFD\xA4\x1F\xB9\xFE\x27\xAE\x5D\xFF\x89\xA7\x3C\x4C\xC8\xE9\xA2\xEF\x12\x62\xBA\xE8\xBB\xC5\x60\xBA\xE8\xDF\x8B\x72\xBA\xE8\x7B\xC4\x10\x8A\x06\x50\x74\xF5\x8F\x3C\xF5\xC4\xBA\x30\xFA\x2D\xF6\x4D\x36\xFC\x93\x02\xFF\x8B\xFF\x84\x95\x12\x7F\x0A\x41\x05\x72\xA4\x2A\xAB\x94\x5D\xB0\x76\x80\xFF\x6F\xAD\x96\xD6\x5A\x21\xB1\xC2\xA2\x94\x42\x96\xF0\x60\xAD\x19\x19\x6B\xA4\x55\x0B\x0B\x0B\xCA\x2E\x8C\x46\x26\xB4\x4F\x6D\x5B\x49\xBD\x84\x86\xE1\x19\x7A\x1E\x73\x57\x02\xEA\xD1\x4F\x2C\x0F\xFF\x84\x1C\xD0\x08\x79\x88\xE1\x5B\x23\x73\x65\xE5\x0E\x69\x61\x78\x36\x57\x23\x2B\x77\x5A\x39\xB2\xA3\x91\xDD\xA9\x84\x94\xCA\xE6\x2A\x13\x56\x19\xA3\xAC\xCC\x95\x12\x72\x24\xB4\x95\x42\x59\x69\x8D\xD4\x42\x59\x6B\x2F\x18\x0C\xB4\xDE\xB5\x7B\xF7\x85\x62\x20\x17\x60\xF4\x40\x61\x42\x08\x5B\xE6\x42\x66\xB9\xCE\x06\xC3\x6C\x30\x30\x66\x20\x74\x0E\xE5\x5A\x0E\x06\x52\x0E\xC4\x48\x74\x23\x54\x46\x18\xF8\xA7\x4C\x6E\xAD\xCD\x73\x9D\xE7\xB9\xD4\x5A\xE4\x52\x0C\xB4\x90\x52\x0C\x32\x95\xED\xC9\x32\xAD\x75\x96\x65\x7B\xB2\x3D\x99\x2E\xCB\x32\xCB\xB0\x44\x6B\x59\x16\xA5\xDC\x33\x14\x02\x3A\x50\x03\x6B\x8D\x35\x0B\x52\xE6\x42\x48\x25\x84\x2C\xF3\xA1\xC8\x73\x51\x55\x30\x63\xAD\xB5\xCA\xF2\x5C\x9B\x52\xEE\x11\x5A\x68\xAD\x4D\xAE\x33\x5D\xEC\x95\x32\x53\x72\x71\xD1\x9A\xAC\xC8\x95\x51\x4A\xA9\x07\xD9\xC1\x40\x6A\xA1\xF7\x08\x25\xA7\xFF\x5D\x24\x61\xB8\x17\x2A\x25\x95\xB5\x85\x19\x56\xA2\x1A\x28\x01\x5F\x55\x76\xA0\x88\x06\x06\x83\xA1\x28\x07\x80\x3C\x65\xAD\x12\x39\x8E\x15\x5A\x2C\x01\x02\x42\x1A\x69\x06\x43\xEC\x29\xCF\xF2\x5C\x29\x83\x08\x81\xFF\x37\x7B\x64\x5E\xEC\xD1\xB9\x28\x73\x91\x1B\x61\x8D\x55\x6A\x58\x0D\xCC\xC8\x8C\x84\x55\x16\x00\x05\xE8\x91\xA5\x10\x62\x20\xA4\x92\x52\x29\xAB\x07\xC2\x0A\x25\x95\xB1\x66\x20\x80\xCC\xA4\x18\x4A\x25\xC5\x60\x68\x07\x66\x60\xAC\x19\x4A\xEC\x0C\x3E\x34\x56\x1E\x93\x57\x5E\x69\xB5\x14\x2A\x5B\x54\x13\xE9\x37\x36\x4C\x39\x11\xFE\xEE\xDB\xF2\x32\x7F\x8F\xDA\x99\x3D\xF9\x9A\x27\x3F\xF5\xC4\x4F\x29\x31\x06\x92\xBF\xE6\xE4\xD5\xD7\x3E\xE9\x47\xD7\x6F\x78\xD2\x53\xAE\xF9\x49\x71\xE9\x10\x8A\x9E\x7C\xF5\xB5\xD7\x3E\xF5\x47\xC5\x7F\xBC\xB0\xF7\xFE\xBA\x6B\xD6\x9F\xF4\x13\x4F\x39\x76\xFD\xBA\xD8\x5F\x84\x17\xE2\x3B\xB0\x8D\xFF\x72\xCD\xD5\xC7\x9E\x74\x6C\xFD\x04\xD4\x11\xFF\xF7\x22\x14\x9D\xBC\xFA\xDA\xEB\xAF\x79\xD2\xB1\xAB\x4F\x5C\x77\x8D\xB8\x62\xA1\x2B\xF9\xB1\xEB\x9F\x7C\x4C\x3C\xA1\x82\x82\xF6\xBA\xA7\x3E\x85\x9E\x1F\xBF\x7B\xA6\xA7\x1F\xBB\x7A\xFD\x6A\xF1\xB0\xA5\x99\xF2\x6B\x9E\xB2\x7E\xE2\xA7\x8E\x3D\xF5\x27\x9E\xB2\x2E\xBE\x6B\x4F\xEF\xED\x7F\xBE\x66\xFD\x49\x27\xAE\xB9\xEE\xFA\x6B\xD7\xC5\x77\x2F\xC4\xF6\x69\x04\x87\x71\xC4\x3F\x7E\xE2\x9A\x6B\xC4\x13\xCB\x27\xD1\x78\x7F\xE4\xEA\xEB\xAE\xD1\xB2\x3F\xFE\xFF\x7C\xCD\xBA\xF8\xE1\x9D\xDD\x68\xAF\xFE\xB1\x1F\x7B\xD2\xB1\xAB\xD7\xFF\x8B\xF8\x5B\xB9\xBB\x2B\x3D\x71\xCD\x93\x9F\x7A\xF2\x1A\x7A\xF1\x77\x72\xF0\x0B\x52\x0A\x2F\xCB\xD5\x37\xCA\xC7\x3C\xEE\x07\x3E\x25\xEF\x95\xCF\x55\xCF\x53\x2F\xD0\xBF\xA5\xDE\xA8\x5E\xA8\xDF\xA0\xDE\xA4\x5E\xA8\x7E\x51\xBD\x58\xBD\x48\xFD\x96\x7E\xA3\xFE\x23\xFD\x27\xFA\x0F\xF5\xBB\xF5\x1B\xF4\xFB\xF5\x5F\xEA\x0F\xE8\x0F\xEA\x5B\xF5\x19\xF3\x74\x73\xD6\x7C\x42\xDF\xA3\x3F\xA9\xEF\xD5\x9F\xD1\x9F\xD5\x9F\xD3\x5F\xD0\xAF\x32\xBF\x67\x6E\x33\xBF\x6A\xDE\x6E\x7E\xCD\xFC\xBE\xF9\x03\xF3\x0E\xF3\x37\xE6\xA3\xE6\xE3\xE6\x5E\xF3\x19\xF3\x05\xF3\x12\xFB\xCB\xF6\x57\xEC\xCB\xEC\x6F\xD8\xD7\xD8\x5F\xB5\xAF\xB0\xB7\xD9\xD7\xDA\x5F\xB7\xFF\xCD\xBE\xD2\xBE\xCA\xFE\x77\xFB\x9B\xF6\xF5\x76\xF8\xC5\x0F\x5D\xF0\x26\xFB\x0E\xA9\xD4\x44\x9E\xFE\xD6\x8D\x8D\x8D\x0D\xE1\xC7\x6B\x8D\x7C\x28\xFD\x14\xB5\x52\x07\xBB\xFF\x73\x62\xFC\x5B\x67\xA0\xFC\x54\xBB\x60\x84\x54\xDA\x96\x50\xB8\x4F\x99\x4A\x2C\x8B\x5A\x8F\x4C\x09\x9F\xC0\x4F\x7D\xD0\xE9\x65\x79\xAA\xD6\x4E\xFA\x51\x0B\x75\x0A\xA7\xDA\xF1\x6F\x3F\x63\x63\x43\x38\xD5\x36\x0A\x3F\x5B\xAD\xC4\xC8\x94\xA5\x13\xDF\xAB\x8A\xF1\x1F\x42\xD3\xB5\x1A\x69\x7C\x1E\xC5\x67\x35\xF5\x2C\x4B\xA7\xC3\x43\xE9\xA4\x1F\xB7\x34\x5A\xA7\xCA\x37\x6B\x69\x27\x5B\x4C\x23\x4E\xE0\xA7\xDB\x05\x0D\x6B\xA5\x1C\xFF\xEE\xD3\xA1\x21\x18\xA6\x38\xA0\x0A\xA7\xBD\x18\xFF\x3D\xB6\x26\x7D\xD1\x8E\xFF\x0E\xDE\xAA\x83\x4E\xEE\x53\xC5\x4A\x25\xFC\xBB\xCE\xDE\xB8\x21\xC6\x2F\x80\x52\x9E\x3A\x16\x8B\xDA\x78\x53\x2B\x7D\x90\xC1\xE1\x54\xBB\x4F\x85\xBE\xF4\xA1\x4A\x38\x3D\xFE\xE3\xA7\xF3\x50\x46\xA6\x74\x3A\xAD\xF6\x3F\xA1\xBF\xF1\x27\x36\x68\x20\xCA\x17\x6D\xAD\x9C\xF1\xB2\x6D\x0C\xF6\xB0\x5A\x89\x12\xE6\xFC\x91\x8D\x00\x80\xFE\xA8\x47\xBD\x51\x8F\x7A\xA3\x1E\xCD\x8E\x9A\xC7\xDD\x58\x1E\x79\x37\x6A\xE3\xD5\x3A\x8C\xA8\x51\x2B\x95\xD0\x07\x9D\x4A\x26\x71\xB8\x32\x4E\x53\xC9\xD4\x78\xF7\x29\xD3\xA8\x4A\x94\xD8\x6A\x6D\xCB\x30\x74\x3B\x3B\xEE\x14\x0C\x5F\x01\xD4\xC9\xAF\x11\xEA\xEE\xF9\xB2\x50\x27\xBF\xEA\xA8\xBB\xE7\x81\x45\xDD\xBF\x48\x69\x26\x5E\xD4\x72\x33\x4C\x31\xB0\x25\x7C\xEB\x4D\xAD\xA1\xAE\xA6\xBA\x72\xA5\x12\x4E\xD2\x1C\x34\x82\xF4\x8D\xF8\xED\xE4\xCA\x4A\x96\x49\x71\x0D\x23\x00\xF0\x02\x94\x25\xF3\x0F\x00\xEF\x48\x95\x58\x0F\xA0\x45\xC0\x31\x04\x1C\x15\x7A\x61\x7C\x05\x00\x69\x04\x50\xDA\xB9\x9E\xEE\x56\x87\x0E\x01\x3E\x3A\xC2\xC7\x94\xA1\x6F\x43\xF0\x91\x5F\x91\xA9\xCB\x2B\xBE\x26\x53\xA7\x6E\xCF\x7F\xEA\x2F\xEB\x4F\x7D\x6A\xF2\xA0\x37\xD1\xE0\xC6\x1F\x44\x56\x2D\x69\x84\x8A\x60\xA1\xBB\xD1\xE9\x30\x3A\x43\x94\x2B\x9D\x81\x41\xBD\x17\xBE\x82\x9F\xA6\x31\x71\x18\x0A\x20\x21\xDB\x46\x3B\x15\xE7\x3F\xFE\x53\x6A\x1F\x88\xF2\x0C\x92\xF7\x87\xA0\x00\x86\xF8\x6C\x35\x35\xC4\xE9\x41\x1A\x61\x4A\x1E\x57\x25\xBD\xAC\xF5\x48\x05\xDE\x50\x09\x7E\xC6\x51\xD6\x06\x06\xED\x4D\x8D\x88\xD4\x07\x9D\x71\x32\xE1\x13\xEA\x70\x45\x02\xAD\xE1\xF2\xFD\x42\xAE\x54\x0A\x19\x51\x2D\xA7\xD0\x56\x62\xC3\xB2\xDC\x64\x01\x4B\x04\x83\x47\x16\xD1\xEB\x81\x9E\xB1\x65\xFC\x69\x1A\x99\x00\x06\x1A\x0D\xCB\x57\x85\xE5\xFB\xAE\x08\x9A\xF2\xC3\x52\xEA\xED\xF1\xD7\x1E\x93\xD4\x9B\x33\x49\x75\x98\xC8\x8F\x4B\x71\x80\xC2\x4B\xEE\x1C\x79\x26\x33\x46\xC6\x5D\x64\x8C\x5E\xC4\x4A\x92\xBB\x1B\x41\x6F\x49\xA9\x93\xFE\xE9\xEA\xE1\x72\xC1\x01\x7F\x5C\x68\xC7\x6F\xC6\x2E\xC5\xE1\x50\x65\x8A\x15\x3D\x42\xCA\x09\xCD\x09\x29\x49\x8D\xDF\x06\xD5\x90\xEA\xA0\x05\xA7\xC6\xBF\x0D\x05\x44\xF2\x08\x24\x15\x1B\x2B\xBF\x8F\x3E\xD6\xEA\xA0\x53\x11\x18\x5E\x1E\xC1\x95\xA2\x9C\x48\xBF\x5D\xA9\x75\x04\x70\x79\x93\x94\x6A\x16\xB6\x0C\x55\x64\xB9\x8D\x40\x28\x8A\x4E\x00\xA8\x20\x00\xA0\xE1\x44\x00\x7C\x1A\xDE\x3A\x41\xCC\x36\x11\x04\x7F\xFC\xEC\x4E\x10\x38\xC5\x8C\x59\xF4\xA6\xFF\xC0\x0E\xE4\xA6\xB3\xF3\x07\xF2\xFC\xE7\x9C\xC7\x40\x7E\x7A\x1E\xB1\xCD\x19\x47\x2A\x11\x7D\x32\x16\xDD\x1F\xCB\x73\xD2\xB1\xE0\x48\x50\x2E\xB3\x74\x9A\xA2\x81\x07\xB8\xEF\x67\x6F\xA7\xEF\xB7\xF5\x11\xA0\x02\x40\xA8\xE3\x5A\x38\xC9\xBF\xA2\xC0\x40\x3E\x8C\x0C\x7F\xFC\x0F\xD4\x29\xFD\xED\x23\x09\x58\x7F\x18\x9C\x0B\x0A\x83\x82\xE1\x7D\xF6\xE9\xBD\xE1\xF1\x5F\x05\xB2\x7D\xE4\xEF\x7C\xD2\xF1\x06\x3E\xE3\x8E\xFC\x6B\x52\x22\xD2\x1D\xEE\x54\x8A\xBB\x77\xF6\x39\xC5\xF4\x1C\x74\x9C\x83\xC1\x19\x80\x50\x71\xC0\xBB\x79\xA8\x22\x0C\x55\x38\xED\x4C\x3A\xD4\x7F\xC1\xA1\xB1\x1E\xE1\xCC\xEC\x50\x35\x0D\x15\x5A\xF4\xB7\xA4\x43\x15\x3C\x54\x4D\x8A\x08\x01\xCA\xF0\x5F\x98\xF5\x59\xE4\x06\xBA\x37\x13\xFD\x35\xC0\xC6\xD9\xB3\xDB\xC0\xC6\xC7\x9E\x75\x1E\xD8\xF8\x93\xAD\xB1\xB1\xBD\x39\xE8\x6E\x0E\x28\xD0\xC2\x3C\x0C\xCE\xE3\xC3\xDD\x3C\x3E\x37\x9F\xAA\xF6\xA9\x51\x23\xBC\x39\x5E\xC1\x9C\xC4\xCC\x84\xFE\x29\xC5\x99\x09\x38\x9B\x42\xCA\x9D\xF7\x6F\x42\xA8\xE3\x8C\xDF\x0F\x6D\xE3\x33\xFD\xBC\x3F\xD3\x4C\xD0\xF5\x8C\xF9\xE8\xA2\x35\xBE\xD9\x04\x7E\x0B\x26\x40\x42\x26\x0A\x4D\x7B\x18\xC7\x94\x3E\x80\xC5\x0B\x92\xAE\x65\x05\x48\xCD\xC8\x53\x03\x7A\x3A\x4B\xFF\x36\xD1\x79\x40\x58\x9A\x19\x61\x19\x0D\x19\x51\x8B\xF8\xB1\xDC\xE4\x63\x01\x1F\xD3\x20\xE8\x63\x55\xBE\x5E\x29\x35\x51\x89\x91\xAF\xA3\x0A\x60\x36\x9B\x89\x86\x07\x95\x3C\x40\x65\x50\x1B\xC1\xC0\x7A\x0B\xC9\x50\xA7\x9C\x0E\xBF\x89\x21\x5C\xA2\x45\x63\x97\x27\x4F\xC0\x4D\x01\xEB\xF4\xB2\xB8\x4C\xC3\xEA\x71\x16\x0D\xFD\x26\xFB\x61\xE0\xA0\x2E\xBB\x4C\x0B\x97\xD5\xB6\x0C\x5F\x15\x4D\x16\xBE\xCA\xE8\xAB\x02\xBE\xCA\x9C\xA5\x0F\xEC\x65\xBA\x70\xB6\xCE\xA0\xA1\xEC\x69\xB7\x33\x5C\x0D\x76\x18\xBE\x77\xCF\x5C\x76\x37\x35\xF8\x41\x76\xBB\xD7\xEB\xDE\x9C\x00\x55\xD8\x19\x82\xB3\x4C\x40\xC5\xCA\x19\x7E\x6D\x97\xE5\xA9\xC6\x3E\x81\xD4\xA6\x0E\xD9\xA6\xFC\x51\xC2\xB5\x9C\x86\x10\xA0\x65\x4A\xB3\x05\x0D\x1A\xE8\x4A\xAF\xFB\x53\x2D\xEE\x80\xF4\x94\x7B\xEA\x54\xFB\x1B\x48\x91\x9F\xB4\x68\x79\x81\x6A\xF6\x59\x20\xA8\xCD\x16\xC0\xF6\xC9\x7F\x11\xCA\xDE\x1D\x64\x3B\xFF\xD2\xB1\x0C\x90\xAD\x1C\x20\x53\x1D\x44\xBE\xED\xC4\xF8\x63\xA8\x37\xFF\x13\xB7\x93\x3C\x8D\xFF\x19\xFF\xFB\x71\xFC\xEF\xA7\xF1\xBF\xB8\x28\xA0\x0F\x27\xC7\x9F\x9A\x2E\x71\x66\xA9\xDF\x98\x13\x50\xF4\xC9\xF9\xAD\x94\x1F\xFB\x3A\x9E\xF9\xA7\xB6\x9C\xF9\x3F\xF7\x4A\x68\x9A\x73\x00\xC0\x33\xA7\xD9\x5E\x4F\xFA\xA5\x4C\x35\x91\xFE\xEC\x58\x4B\x79\x77\x98\x6C\x18\xDF\x47\x43\x65\xD0\xD1\xFE\xB1\x6B\x1E\x85\x92\xEB\x8D\xB2\x4E\xC4\x25\x2A\xFE\x5F\xCF\x30\xBE\x77\x06\xAE\xF7\x9E\x27\x5C\xEF\xDD\x84\xA2\xFE\x42\x4A\x39\x2B\x5D\x52\xED\x2F\x00\x93\x39\x29\x4F\x90\x0C\x90\x1E\xDF\x93\x68\xB6\x74\x6C\xEE\x12\x5D\x2C\x4F\x7E\x38\xEA\x39\x92\xA4\x8A\xC1\x41\x2A\x60\x07\x41\xB0\x98\x76\xFC\xA9\x29\x2D\x5E\xED\x53\x68\x9B\x7F\x31\xD9\xCE\x71\x92\x51\xF9\x91\x14\x95\x61\x93\x65\x5A\x0F\x78\x81\x9A\xA7\xCB\xB0\xB9\xB2\x29\x39\x75\x33\x3A\x54\xC5\xF1\xE0\x5C\xD5\xC1\xF9\x13\x7C\x42\xD5\x1B\x86\x66\xCB\x2C\xD2\x24\x3E\xA4\x98\x20\x70\x6A\x92\x6A\x60\x2F\x6B\x82\xC6\x87\xE0\x83\x8B\x9C\xE6\x49\xCE\xA0\xAA\x6B\xBA\xDF\x82\xDC\xAC\x85\xF9\xD2\xF8\x85\xAA\x53\x27\xDE\x23\x66\x01\x43\xD3\x37\xA9\x54\x33\x24\xC8\x48\xB6\x86\x2D\x92\x7D\xED\xF8\x15\x1B\x1D\xD8\xE0\x0D\x0A\x57\x35\xBE\x6D\x83\xE0\x93\x96\x71\x55\x06\xA4\xDA\x2F\xF6\x35\x12\xFE\x08\x22\x7C\x89\xBB\x07\x06\xA0\x01\x2D\xFB\x1D\x2D\xAA\xC5\x3B\x48\x2C\xE8\xFD\x42\xF8\xC9\x75\x4E\xEC\x17\xA2\x31\xC7\xC1\x34\xF5\xC7\xDA\xFD\x02\xA4\x03\xFD\x35\xC7\x0F\xE3\x7E\x04\x09\xF2\x5A\xA0\x32\x80\x7B\x86\x93\xB6\x91\x2B\x95\x1A\xA1\xF5\x5C\x1B\x50\xB3\xDE\xD5\xE9\xBE\xEF\x11\x1D\x60\xDE\x62\xB4\x99\xC8\xD3\x29\xD1\xE8\x64\x35\x48\xDE\x42\x44\x4B\x59\x38\xD3\x36\x16\x86\xF5\x9D\x87\x2A\x09\xAB\x0D\x37\x12\x0E\xC5\x8E\x32\xA8\x12\x60\x41\x5B\x38\x99\xB3\x5E\xB6\xCE\xC0\x4C\x64\x4B\x2A\xC1\xCF\xD0\x26\x1A\xB4\x81\xF2\x38\x5F\x16\x3F\x58\x49\x97\x23\x68\x9A\xE2\x8E\x65\xFD\xCC\x27\x54\xC0\xFF\x68\x7B\x25\x03\x00\x15\x5E\x1E\xAF\x33\x1C\x53\x01\xD3\x1B\xD0\xB8\x80\x0E\x0A\xFF\x25\x79\xBC\xB6\xB4\x51\xE2\x14\x0C\x52\xFA\x1D\xAD\x9F\xF8\x8D\xD3\x2E\xBF\xFD\x7A\x97\x2F\xE7\x4F\x68\xC4\xD2\xE5\x02\xA4\xF9\xCE\xF0\x62\xF9\x86\x53\xCB\xC2\x89\xA5\x26\x56\x91\x58\x85\x5F\xC8\xA5\x3A\x77\xD6\x19\xAF\xDA\x06\x48\xA1\x04\x2A\xCC\x56\x50\x93\x31\xED\x4C\xFB\xF0\x69\x09\x83\x75\xC5\x4A\x25\x5D\xE1\xF5\x71\x60\x1B\xC8\xC6\x07\x5E\xAF\x56\xC2\x15\xFE\x0B\xF2\x78\x5D\xD0\xAE\x1B\x68\x65\x6D\x83\x00\xB4\xCE\x22\x36\xFD\x0E\xC0\xEA\xF1\xCB\xF1\xB7\x4A\xCA\x77\x26\xE5\x3A\x29\xBF\x20\x29\x37\x49\xF9\xAE\x58\x5E\x38\x09\x6F\x24\x8E\x1F\xB8\x38\x60\x4D\x21\xA5\x01\xAC\x8C\x3F\xD6\x36\xD2\x49\xFC\x8C\xBF\xC1\x39\xC0\xB8\x90\xAC\x26\x2D\xB0\xC0\x91\x22\xF4\xF6\xB4\x9E\xAC\xFC\xB9\xA1\x2C\x03\xDD\xB8\x1E\xDD\x00\xE5\xA0\xA2\x13\x76\xAF\x88\xB5\x3A\x03\x23\xF3\x3F\xD0\x2E\x54\x78\x08\xCA\xFF\x24\x2E\x80\xC9\x01\xB1\x03\xD8\x22\x8C\x01\x15\x42\x2F\x60\x0A\xBA\xCE\xBC\xA8\x73\x2F\xEA\x82\x88\x10\x68\x64\xB0\x52\xA1\x60\xF2\xAF\x7C\xCD\x8D\x1B\xC2\x0D\xFC\xDE\x75\xBF\xF7\x7A\x5F\x8E\x5F\x02\xCC\xA0\x19\x32\xD5\xBA\xDC\x8B\xC3\xCE\xFA\xFB\xE4\xF1\x66\xB0\x72\xBC\x1A\xB8\x21\x7D\xB1\xE6\x06\x7E\x78\x6D\xDB\xC0\xAB\x23\xD5\x00\x94\xC7\xCB\x05\x2C\xC9\x1C\x37\xA5\xEA\x9C\x48\x4E\x5F\x59\x0D\xE0\x09\xA8\xAF\xAD\x0B\xDE\x8D\x83\x4A\x85\x1B\xF8\xEF\x38\x5C\x15\xAE\xF0\xE6\x50\x55\xE0\x80\x33\x18\xB0\x74\x19\xD4\xC5\x31\x97\x61\x5D\xD2\x82\xAC\x78\x35\xC2\x37\x57\x54\x16\x04\xE0\x3E\x25\x0E\x08\x5C\xC2\x23\x5B\x3A\xB5\x3C\xB9\x0C\x61\x30\x86\x89\x2B\xFF\xA0\x16\x0A\x04\x2B\xB9\xAC\x21\x63\x0B\x08\x84\x7C\xA5\x32\x4E\xD4\x40\x97\xB9\xBF\x1C\xD5\xCD\xFD\x42\xC2\x2F\x0B\xD0\x3F\x54\x59\xEA\xBF\xA4\x5D\xD9\x1A\x6C\x6D\x98\x06\x22\xB1\xAE\x5C\x89\x83\x23\x58\xD3\x66\x30\xE0\x18\x80\xEC\xEF\x02\x20\x91\xE5\xAE\xF8\xC9\xE5\x04\xE4\x66\xE8\x77\x30\x98\x61\x0D\xDE\x4D\x35\xE9\xAF\x4B\xDE\x01\xDB\x73\x12\x40\x98\xB9\x81\x33\x6B\x2E\xF3\x66\xFD\x44\x9D\xC1\xD0\x80\xB6\x6C\x3D\x60\x78\xE2\xF8\x01\x45\xB9\xBF\x1C\xE4\x4F\x4E\x90\x05\x09\xD9\x08\x57\x1D\xA9\x0A\xA2\x85\x8B\x5D\xE9\x8A\x43\x55\xE6\xB4\xBF\xB8\xAD\x07\x4C\x15\x60\xBA\x0D\x91\x32\x4A\x00\x8A\xEC\x53\x86\x9C\xA2\x0C\x5C\xC0\x06\xA8\x82\xBA\x94\x2B\xC7\xAB\xA1\xCB\x02\x55\x48\xA4\x8A\x1C\xA9\x62\xE8\x06\x2E\x27\xAA\x30\x2B\x1D\xA7\x1E\xC2\x94\x86\xC8\x75\x87\x30\x93\x21\xB0\x20\xB3\x52\x0D\x9C\x84\x71\x03\x89\x99\x43\x15\x8E\xCE\x0D\x88\x22\x06\x50\x6F\x40\x8C\x18\x80\xDE\x94\xAE\xA0\x15\x39\xF4\xFA\xD1\x55\x46\x94\x70\xF1\x01\x21\x46\xBA\x64\xB9\x00\x58\x56\xF5\x00\xF0\xEB\xEC\x4A\x25\x47\x45\x89\x58\x5F\xA9\x72\xA7\x80\x3B\x48\x80\x4C\xEE\x94\xCB\x7C\xB1\xEE\x32\xBF\xF1\x25\x7D\xDC\x17\x27\x4F\x3C\x5C\xD0\xC7\x96\x3F\x96\x88\xDA\x0C\x65\x28\x80\x0F\xA9\x04\x06\xC7\xB4\x39\xA8\xAD\x33\xB5\x74\x79\x25\x91\xC5\x99\x8A\xCA\x14\xAA\x1A\x34\x0E\xE8\x0F\x31\x61\xE7\xF6\x07\xF5\xE1\x53\x8B\x7A\x85\xAB\x0E\x55\x9A\xA4\x92\xB3\x6B\xCD\x10\xF8\x00\xC2\xEA\x38\xAC\x60\x67\xFD\xE4\x3A\x2F\x9C\x58\xC3\xA1\xF8\xD3\x08\x19\x37\xF4\xA7\x8F\xAF\xE1\xAA\x11\x20\xFA\x9C\x00\x1E\xBE\xD8\xBA\x01\xFC\x42\x2E\x99\xEF\x17\xC8\x0F\x72\x62\x0B\xD2\x2F\xB4\xC8\xB3\x88\xFD\xE1\x9F\x0C\xDB\x6B\x40\xBA\x96\x5E\x3A\xB1\x06\x00\x2F\x60\x95\x54\x40\x80\xD6\x0D\x9D\x59\x6B\x1B\xC1\x5F\x09\xFC\x8A\x4E\xA3\x0D\x8F\x7B\x64\xCA\x28\xD7\x0E\x87\x05\x0B\x18\x1E\xA1\x05\x08\xAB\x71\xE4\x8B\x16\xE5\xD6\x65\xC8\x2C\x0D\x30\x45\xD7\x99\x82\x1F\x52\xD2\x3C\xC0\x9A\xC6\x3C\x9D\x42\x74\x2A\x85\x49\x54\x0A\x83\x84\x2A\xE7\xA9\x14\xCE\xD4\x16\xD7\x49\x4F\xB1\x90\xF3\x14\x0B\xC9\x8A\x85\x9C\x52\x2C\x2C\xF0\x31\xBB\x52\xE9\x11\xED\x89\x73\xEB\xD4\xCF\x8C\xCE\xB2\xAD\xA6\x0D\x34\x6D\x3A\x9D\x05\x98\xC7\x66\x3A\xCB\x05\xB4\x9B\x47\x3B\xEC\xBC\xB9\xBE\x3B\x29\xEB\xF6\xEA\xFB\xA5\x57\x72\xE9\xDE\xB4\x74\x72\x9D\x7F\xC8\xC9\x79\xD5\xE7\x97\xCA\x2B\xB8\xF4\x21\x60\xA6\x8D\xC7\xEF\x27\x8D\x7A\x59\x5C\xA6\x80\xA2\x8A\x96\x7F\x95\x03\xFC\xEA\x80\x12\xE5\x0B\xFB\xC6\x46\x1E\xC4\x24\xED\x86\xBD\x3F\xD5\xF5\xED\xF8\x7D\x34\x47\x67\xC6\xEF\x64\x4B\x2C\x87\x56\x46\x40\xF7\x07\x54\x01\x4A\x12\x58\x13\xB9\x13\x07\x94\x70\xF9\xF8\x35\x1B\xA8\xFD\xE6\xFB\x54\x51\x13\x4D\x64\xA4\xDD\xFF\x79\x6A\x2E\xBC\x37\xD8\x7C\x79\x6A\x2C\xEC\x14\x71\x54\xD0\xDC\xC8\x77\xA3\x36\xF1\xD7\xA8\xCC\x69\x1F\xA3\x7C\x8C\xF0\x13\x2F\xC6\xF8\xC9\x45\xF0\xD3\x8B\xF1\x99\xE4\xC1\x8B\xF1\xD3\x7B\x8F\x5E\x8C\xCF\x4E\x15\x78\x31\xBE\x11\x8B\xCA\x17\x6D\x7A\x08\x31\x75\xD2\x34\x41\x2F\x12\xA3\x74\xB9\x2C\xD8\xA1\x43\xA0\xF7\xC7\x77\x09\x19\x9F\x55\xE9\xBB\xBD\x33\xDE\xC4\x01\x55\xF3\x16\xB6\x5A\x62\xF1\xF8\x7F\x84\x13\xA1\xC8\xE0\xA6\x4F\x25\x9E\x32\x6F\x60\x53\x3B\x79\x86\x96\x5B\xDA\xDB\xA8\xF5\x1B\xAF\x05\x5C\xBE\x21\x35\xB2\x5E\xC4\x5B\xD3\x9B\x1F\x46\xDD\x68\x65\x11\xFA\x7B\xBE\x5A\x6B\x4C\x60\x25\x7E\xDC\xFA\x0D\xE9\x37\xD4\xF8\x56\x34\x71\xBC\xAE\x2D\x2A\x6D\xE3\xD6\x69\x67\xDB\x26\xF3\x4F\x83\x35\xD5\x3A\xEB\x9F\xD6\x12\x1F\x1C\xB7\x2E\xF3\xA7\x43\xE9\xE9\xB4\x74\x12\x4A\x27\x49\x29\x15\xC1\xB3\x45\x15\xD2\xFF\x1E\x71\x4F\x50\x0A\x90\xAB\x3E\x26\x9E\x02\x03\x03\xCF\x83\x6A\x47\x1C\x0B\xF8\xB5\xA5\xE3\xB7\xD3\xA4\xD9\x62\x89\x3E\xE8\x32\x6A\x5F\xC0\x30\x81\x29\xA0\xDE\xB7\x21\x0F\xB7\x3C\x7C\x2F\xDB\xB4\x14\xD4\x66\x67\x51\xE9\x0D\xAA\x36\x68\x62\x69\x33\xFD\xEA\x34\xC2\x0C\xF5\x3C\x97\xAD\x79\x7D\xBC\xD1\x2E\x6B\xBD\x3A\xE9\xF5\xB5\xE3\x17\x06\x15\x04\x44\xD3\x50\x81\x40\x01\xA9\x4B\x5A\x61\x01\x65\x9E\x4E\x73\xAC\x3E\xE8\x1F\x09\x3A\x8F\xB3\x4E\x1E\x45\xE9\x85\x3D\xA1\xBC\x54\x07\x03\xE8\x40\x53\xC8\x5A\xD4\xB9\xFC\x86\x3C\x84\x2A\xC4\xA8\x75\xBA\x45\x05\xC1\x15\x2E\xF3\x8F\x3C\x44\x9A\x95\x6C\x1B\x4D\xDC\x7D\x08\xE2\xA9\xF2\xE6\x24\xFC\x1D\x79\xB5\x7E\x02\xEA\x62\xE9\x82\x57\x27\xF9\x2D\x94\xCA\x50\x9A\xAD\xC3\xDF\xC5\x13\x97\x0B\xE5\x8A\x4A\x82\x7A\xD0\xD6\x43\x57\x00\x0B\x2C\xD1\x90\x02\xF8\x00\x77\x2F\x79\x91\x80\x2D\xB4\xA0\xA4\x12\xA5\x1B\x10\xAD\x21\x4A\x34\xA8\x16\x8A\x95\x13\x52\x53\x4A\xA7\xDC\xD0\x0D\xD6\x80\x6D\x0C\xC8\x89\xE3\xF9\x2A\x50\xA1\x2D\x9F\xFC\xC0\x10\x7D\x3C\x32\x73\x74\x08\x36\xFE\xCB\xB9\x34\xFF\xE5\xF6\xF6\xE6\x6D\xF5\x76\x93\xA6\x03\x7E\x83\xB4\xFB\xA5\xFB\xEE\xBB\x4F\xE3\x69\xAD\x57\xEB\x5E\x3F\xC5\x5B\x30\x9D\x24\x58\x68\x96\x29\x27\x0B\xF2\x55\x7A\xFD\x58\x20\x43\xD2\x01\x32\x32\x9A\x40\x3D\x01\x5A\xB0\x5E\x9D\x6C\xA3\x06\xA2\x61\x51\x0A\xAF\x98\x4C\x1E\x75\xBC\x7B\x25\x71\xBD\x7A\xB3\x8E\x32\x8F\x3E\xDD\xBB\xEE\x2F\x3E\x79\xA2\x57\x4D\x51\x35\xB5\x4E\xAA\xF6\xB7\x4C\xBD\x36\x64\x91\x69\x32\x28\x9E\x06\x0A\x92\xBA\x92\x17\x0B\xAE\xD0\x79\x03\x4B\xD6\x2B\xBE\x87\x51\xF8\x87\x71\xAB\xD2\x3F\x92\x04\x84\x89\xE3\xDB\x2F\x64\x23\xA6\x07\x27\x69\x66\xEB\xFE\x11\x58\x00\x0C\xD3\xF8\x47\x5E\x2E\xB4\x33\x4E\x00\xAD\xD2\xE0\x4A\xD6\xC3\x40\x2F\x61\x3B\xD9\x65\x40\x6F\x08\xB9\xD2\x99\x2F\x03\xED\xEF\xF9\xAA\x12\xD9\x7B\xEE\x07\x91\x5D\x9A\x6E\xDB\x26\x4A\x5E\x27\x79\xDE\xD1\xED\xB6\x3E\x94\x36\xC8\xDE\xDB\xF5\x80\x1B\x9B\x62\xFC\x32\xA0\x3F\xEA\xA1\x7C\x4B\xA2\x37\xDC\x23\x3A\xC1\x20\x5F\xED\x8C\xBF\x4B\xB4\x7E\x38\x7E\xDE\x19\x94\xFE\xEA\xD5\x24\x2D\x62\x81\x18\xDF\xCC\xE7\x3C\x77\x89\x96\x7E\x13\xE7\x0D\xBF\x35\xFF\x00\xE2\x97\xA4\xC4\xD4\x80\x2F\xCD\x07\x60\xC0\x25\x0C\x7E\x01\x4A\x87\xA1\x86\x40\xF1\x50\x4E\xFA\x5B\x9E\x71\x23\x1E\xAC\x3E\x9B\x76\xDE\x14\x69\x2F\xCE\xF8\x7B\x82\x92\x56\xDE\x64\x65\x3E\x49\x40\x91\x1D\xAA\xC4\xF8\xAF\xCF\xE0\x7E\xED\xF8\x03\xF1\xA4\x4B\xC6\x85\x36\xE3\x11\x25\xA3\x47\x94\x3E\xE7\x39\x16\x1F\x82\xC9\xC4\xE1\x49\x26\x0E\x70\xE4\x46\x24\xE7\xB9\x11\xA9\xF9\x6E\x44\x92\x1D\xDD\xB8\x61\x72\x74\xA3\xF1\x23\x86\x55\x3C\x64\xCB\xC8\xDB\x05\xB7\x6E\x5D\xE6\x35\xB4\x6E\x9A\x9C\x9C\x40\xD0\x95\x8A\x39\xB5\x9E\x99\xA1\xA6\x19\x76\xBB\x6C\x96\x26\x8A\x7B\xAE\x3A\x4E\xB5\x29\xC8\x9D\x05\x57\x54\x41\x43\x2D\xC3\xCC\x49\x26\x60\x53\x30\x73\x8D\xBA\xB3\x8D\x00\xD3\x9B\x00\x6C\xEA\x33\x00\x18\xFC\x6E\x0A\xDE\xD7\xE4\xF1\xE0\x58\xB0\x0D\xDB\x87\x9B\x8A\x0E\x58\xDD\xD0\x3A\x40\x96\x09\x24\xA1\xD9\xBA\x08\xC2\xCB\x85\xEE\xB6\x6C\x7D\x33\xAC\xCC\x69\x0B\x30\xED\xF2\xF1\x9F\x50\xB7\x19\xBC\xC8\x10\x3B\xC1\xFF\xEC\x0E\x23\xB3\x89\xC3\x65\x38\x6E\xE4\x01\x55\x10\xB9\x01\x9B\x54\x7E\x3F\x2E\x50\xDC\x91\x80\x77\x63\x72\xA7\x82\xBA\x62\x0D\xD7\xEC\xD1\x8A\x76\xC7\x6A\xC5\x5B\xB8\xA0\x0A\xF8\x87\x4D\x7D\x46\x3B\xD4\x8A\xB6\x1A\xC6\xCF\x26\x88\xD0\x92\xC6\x5E\x7B\x0D\xA2\xF3\xDA\xA5\x42\x50\xBD\x1A\x4F\x76\xC7\xB5\x74\xC1\xF9\x28\x34\xDA\x1B\xC7\x2A\xF1\x79\xC5\x87\x85\xA2\x31\x54\xDE\x00\x65\x82\x89\xCF\x07\x13\x34\xAD\xEF\x98\x1A\x1F\x61\x6B\xAD\x51\xCE\x6E\xDA\x3F\x4C\x20\x38\x1A\x42\xF3\xF4\xC2\xD4\x16\x7D\xDB\x66\x86\xD5\x60\x73\x71\x08\x28\x84\x94\x8B\xA3\x81\xF5\x1B\xC0\xE6\x94\x77\x27\x10\x6E\x7F\x2F\x0E\xC5\xD1\x86\x36\x55\x00\xBB\x82\x11\x66\x30\x42\x19\x4C\x5B\xE1\x9F\x00\x8B\x45\x0A\xAE\xAE\xA8\x7A\x52\x55\xE1\xB9\x2F\x4D\x4A\x75\x93\x92\x34\x29\x05\x84\x8C\xE7\x2E\xA1\x27\x3A\x48\x4F\x81\x0A\x6F\xC1\xF4\xB1\xE8\xA9\x55\xBE\x4D\x4B\x1B\xD9\x38\x20\x0F\x87\xD1\x10\x40\x43\x33\x05\x7D\x14\x5A\x6B\x34\x23\x03\xB0\x4B\xDF\xC6\x2D\x4F\x45\xF2\xF8\x83\x38\x73\xEB\x9B\x43\x15\x50\xBB\x3B\x0A\x8C\x0D\x59\x3E\xB6\x69\x09\x08\xD2\x59\xA7\xD7\x9C\x39\x5A\x91\xCF\x0C\x6E\xB5\xFF\xBF\xED\xC2\xE3\x45\x96\xFC\xEB\x3D\x9C\xEB\x9F\x88\xFF\x15\x49\x89\xC8\x84\xCC\xA8\x77\x86\x69\xD8\x34\x48\x8A\x9C\x71\x6A\xCD\xE9\xD6\x9B\x2B\x2A\x43\xD0\x7D\x2E\x5A\xDC\x15\x9E\x92\x8F\x2F\x15\x72\xBA\x40\x4D\x17\xE8\xA4\x00\x8B\xC0\xC8\xC0\xC6\xC9\x5D\x12\xE1\x57\x6B\xEC\x5D\x96\x29\xA6\xBC\x74\xD2\x59\xFF\x25\x71\x64\x09\x44\xA6\x42\xC0\xA0\x5B\x24\xC3\x8D\x7C\x87\xA0\x3E\xD4\xCE\x7D\xE6\xE4\x12\x09\xD7\xB3\x16\xD4\x3D\x99\xE0\x70\x16\x4D\xFA\xA0\xDA\xE4\xFF\x70\xA6\x8D\x04\xB8\xFF\x27\x53\x86\x7F\x83\x32\xFD\x37\x2C\xCF\xF1\x2F\x2F\x8B\xF0\x53\xF1\x5F\x01\xFF\xD1\x92\x1E\x6C\x99\xD1\x2E\x47\x0A\xE4\x91\x7F\xDF\xCF\xDE\x48\x47\x83\xE3\x67\x21\x13\x19\x75\x50\x83\x49\xAA\xC5\x99\x6F\x4A\xFF\xB9\x17\xF4\xBF\x29\xFB\xDF\xE8\xA9\x6F\xEC\x15\xD5\xD0\xBF\xEA\x85\xF4\x8D\xE5\x6F\x86\xE1\x1B\x8B\xDF\x18\xFE\x46\xEC\x53\xBB\x56\xAA\x41\xDA\xDF\xC0\x7F\xE4\x37\xFA\xFD\x0D\x02\x73\x55\xD4\x23\x41\xD0\x8C\x9F\x43\x88\x17\x78\x6E\xC4\x15\x10\x57\xF0\x5A\xD2\x6B\x5A\x80\x71\x99\x5D\x72\x18\x5B\x0B\x38\x5D\xC0\x43\xE4\xDF\x8D\xC7\xBE\xA1\xBC\xE8\x3D\x0D\x7A\x4F\xC3\xDE\x53\xD9\x7B\x1A\xF5\x9E\xAA\x45\x66\xA8\x81\x61\x3B\x09\xF2\x00\xB7\x4B\xEE\xA2\x3E\xF1\xED\x73\x3A\x76\x4E\xDB\x56\x04\xA9\x3E\x43\xE9\x68\x2B\xC8\x9D\x3B\x73\x99\x6D\x72\xA6\x26\x7C\x46\x0E\x33\xA4\x76\xE1\x59\x78\xF4\x0A\xA1\x83\x2C\xD4\xA1\xFD\xA4\xA5\x83\x23\xE5\x4F\x1D\x8F\x7B\xA3\xC6\x89\xB6\xC9\x81\x0D\xB7\x4D\x11\xF7\x44\x65\x8B\x67\x14\xF1\x59\xC1\xB3\xEA\x9E\x35\x3C\xEB\xF0\x9C\x39\x81\x9B\xEB\xB4\x09\x6D\xD9\xC3\xBC\x45\x32\x11\x6D\x4D\x38\x93\x5C\x19\xF7\xEB\xE3\x76\x1D\x6D\x01\xE2\x77\xC6\x29\xD6\x85\xC3\x59\x31\x6B\xFE\xBC\x33\x1F\xF4\x04\xB2\xDB\x3F\x18\x0E\x3D\x01\x07\xE3\x3B\x50\x3D\x2A\xFC\x04\x39\x2B\x35\x0D\xB3\xA8\x73\xE4\x03\xDE\xB6\x08\x83\x62\xFC\x6A\x46\x0E\x4C\x89\x5E\xE6\x60\x6C\xF0\xD9\x4A\x23\xC8\x8B\xD9\x29\x97\xAF\x05\x78\x12\x34\xA5\x33\xDD\x28\x78\x1C\x78\x78\xD9\x14\xC4\x8E\xE3\x82\x2F\xBC\xA3\xE3\xB9\x06\xD0\xEF\x5F\xD7\xF9\xDB\xD1\xE9\x22\x9D\x20\x08\x74\x7C\xCF\x5D\x41\x5B\xC7\xE9\x88\x51\xCA\xE1\xBC\x9C\xF6\xA3\x76\xFC\x9A\x64\x72\xFE\x35\xAF\x48\xDD\xF7\xF6\x01\xD7\xC2\x4F\x5D\xE1\xF2\xF1\x6B\x51\x09\xC7\x79\xCD\xE1\x44\xA0\xE7\x5C\x8A\x03\x46\x21\x90\x27\xFF\x7A\x0F\xE7\xFA\x47\x95\x65\x9E\xE7\x8A\x4B\x74\x9E\xE7\x26\xB7\x59\x1E\x66\x34\x82\x5F\x1E\x7F\x65\xF0\x6B\x04\xBF\x2C\xFC\x1A\xD2\x1E\xB9\xCB\x7D\x75\x39\x1E\x50\xB8\xDC\x0F\xE0\x57\x37\xEB\x1E\x36\xFD\x1F\xDF\x9C\xBA\x64\xFA\xAC\xAD\x2D\x81\x39\xE2\xD2\xD6\xC1\x91\xC2\xD9\xF8\xB5\x9D\xFB\xF5\x08\x60\x55\x38\x3B\xFE\x0D\x24\xB4\xA2\x9C\x06\x9C\x2A\xD3\x4F\x88\x78\x54\x87\x1B\x5C\xF1\x57\x90\x66\xDC\xAD\xB8\x12\x68\x03\x08\x38\x77\x66\x2D\xD0\x70\xEF\x70\x53\x94\xCF\x07\xC5\x71\xFA\x6A\x5B\x3C\xDB\x3C\xDD\x2E\x54\x42\x2A\x6D\x4C\x56\xD8\xA2\x28\xF2\xA2\x1C\xFF\x39\x99\x37\xC1\xA7\x7D\x31\x3A\xAE\x93\x22\x38\x6A\xD8\x05\x4C\xAF\x35\x26\x98\x5C\xB8\xC7\x84\x6B\xDF\xA4\x6B\x5F\xD3\x71\x39\x2D\x7F\x93\x2E\x7F\x85\x47\x23\x74\x42\x92\x27\x2B\x14\xE7\x1B\x9F\x55\x8B\x0B\x26\x3E\xEB\x16\x79\xC0\xE6\xCB\x5F\xD3\xF2\x57\xDB\x58\xFE\xCA\x99\xF1\x5F\x9C\x89\x1E\x38\x30\xB3\x1C\xE7\x09\x4B\xF1\xF7\x36\x18\x16\xB5\x0C\xE6\x9E\x0E\xDB\xBA\x6F\x43\xAB\x37\xF7\xD7\xB7\x0B\xCA\x0A\xDC\x3D\xC4\xF3\x7B\xE0\xAE\x96\xBC\x25\x9C\x70\xF9\xF8\xED\x58\x4F\xB2\x2A\xAD\x9C\x24\x83\x00\x44\xD3\x88\x34\xCE\xD0\x96\xF4\x83\xC3\x95\x18\xFF\xD5\x99\xC0\xB6\x85\x93\xFD\xCF\x03\x22\x93\xEE\x7F\xB2\x5D\xB0\x42\x6B\x21\x35\xF9\xA4\xE5\xE3\xDF\x4F\x3C\x8C\xFE\x80\x7E\xB3\xCD\xA7\x4A\xA7\xCA\x0D\xB0\xC7\xC3\xED\xAA\x5D\x2B\xC1\xA0\x05\xC3\x93\x0C\x10\xA7\xFC\xE0\x10\x2A\x9C\xDC\x4B\x18\x03\x89\xC1\x58\xAF\x37\xF2\x11\xEA\xC4\x30\x01\xBA\xD3\x38\xFD\x2D\x1F\x42\xA1\x76\xFA\xBB\x52\x2A\x1E\xC1\x54\x35\x60\xBB\x34\x7F\x18\x8E\x53\xE3\x77\xF4\x3B\x82\xD9\x1A\x25\x59\x95\x9E\x3B\xBC\x50\xB3\x3A\x3C\x67\x0A\x68\x9D\xCC\x6D\x79\x9B\x53\xF8\x7F\xBA\x3D\x0D\xD7\xB9\xF5\x2A\x27\x0E\xA8\xBD\x80\x63\xBE\x62\x00\x4A\x36\x3A\x4E\x5D\xEC\xD4\xB2\xB8\x4C\x8F\xD1\x09\x89\x3C\x91\x92\xB6\xF1\xF8\xC9\x75\x8B\x76\x5B\xAD\xCB\xED\xB6\xFE\xD7\x16\x6D\x49\xBC\x0E\x47\x76\x44\xAD\x83\x39\x09\x74\x66\x51\x96\x6B\x6F\x83\x47\x6E\x23\x1F\x53\xD1\x72\x96\xEB\x8D\xE5\xB3\xA2\xAC\x12\x7E\xB2\x58\x76\xFB\xEA\xFA\x38\xDD\x2C\x40\xC6\xC2\x32\xDF\xC3\x8B\x53\xC7\xD7\xEA\xDC\x4B\xBC\x49\x94\x39\xD5\x36\xF0\x1A\x3B\x76\xAA\xA5\xFD\x7C\x5C\xA1\xA1\x28\x65\x07\xA1\x56\x9F\x29\x84\xD2\x4E\x33\xA0\x63\x76\xA8\x78\x18\xE5\xA8\x9F\xB4\x78\x7F\x8A\x68\x02\x7A\x9D\x6A\x1D\x5D\x09\x40\xD0\x4D\x5A\xBC\xF9\x05\xF8\xB6\x07\x68\x8F\x0B\xCD\x49\x97\x1D\x50\x60\x2D\x65\x7C\x92\xB9\x86\x9A\x12\x5E\xEF\xF9\x07\x3A\x77\xDD\xA7\x8C\xFF\x10\xEE\x3F\x9A\x5A\xF9\x4F\x89\x5A\x7A\x5D\x6B\xFF\x31\x51\x67\x5E\xA1\x30\xF4\x86\x41\x69\xBE\x09\xCA\xF9\xA0\xBC\x17\x5A\xF0\xBA\x56\xFE\xD3\x00\x3F\x55\x6B\xFF\x49\x80\x9F\x44\x99\x07\xB5\xF3\xD6\x65\x01\xDC\x4E\xB7\x4E\xC6\x07\xD5\xFA\xBF\x27\x4C\x04\xB7\xF2\x03\xCA\x78\x51\xFE\x8E\xE9\x4E\xA2\x92\xA3\x6B\xF4\x00\x0C\x5B\x8B\xA4\xE4\xD1\xEE\x21\xA1\xC6\xD0\x25\x56\xB1\xD6\x36\x96\x50\x95\x11\xAA\x2C\xA0\x2A\x67\x54\x15\x80\x2A\xDC\x4D\x08\x47\x94\x19\x60\x6B\xE0\x05\xA8\x07\x78\x3C\x15\xB0\x95\x21\xB6\x60\x2A\x56\x1F\x04\xE1\xDF\x36\x26\x62\xCB\x26\xD8\x32\xA1\x28\xDD\x69\x0F\xB5\x64\x6F\x63\x3D\x94\xAA\x4E\x06\xD2\x81\x57\xC0\x16\x08\xB6\xDA\x96\x6E\x40\xD8\x82\x5E\xA7\x5A\x07\x6D\xAC\xB6\x6E\xC0\xDE\x67\x88\xAD\x3C\x62\xCB\x3A\xE1\x0A\xC2\x16\xEF\x24\xDB\xB5\xB6\x31\x80\xAD\x70\xFB\x45\xE3\x84\x69\x33\x10\xA7\x4B\xDB\x05\x78\x80\xA6\x41\xC6\x17\x7C\x78\x66\xA0\x73\x85\xF3\x0E\xAE\x0A\xD8\x60\x4B\x9E\x74\x71\x66\xA1\x54\xB5\x38\xC9\xA9\x52\xDD\xE2\xA9\x44\x52\x5A\x1B\x3C\x65\x33\xE1\x94\x2D\x4C\x36\xF6\xD7\x9F\x29\x7F\xD2\x9B\x30\x50\x12\x12\x0B\xC0\x2A\x3D\x95\xB7\xE5\xEF\x19\xDC\xF2\xED\x94\xA4\xFD\x42\xE2\xE9\x94\x90\xB4\x85\x74\x89\x2E\xC6\xFF\x83\x45\x2C\x6E\xB0\xF2\x79\xAD\x21\x3A\x42\xF7\x9B\x11\x6D\x03\xB6\x41\x57\xB2\x44\x4A\x12\x48\x29\x63\x52\xCA\xE3\xAA\x47\xD5\xC5\x02\x58\x0B\xDE\xB3\xB1\x09\x1D\x59\xA4\x23\xD4\x2B\xF4\x41\x97\x3B\x3C\x9A\x0B\x74\x24\x13\x3A\xD2\xA1\x28\x3C\x47\x3A\x92\x09\x1D\xE9\x48\x47\x32\xA1\x23\xCB\x1E\x76\x81\x8E\x60\xCA\xB5\x44\xB7\x13\xC1\xBD\x4E\xB5\xCE\x0A\x15\x7A\x31\x16\x0C\xD6\x2C\xD2\x91\x24\xAA\x02\x3A\xCA\xF9\x44\x62\xAD\x6D\x34\xAE\x7A\xB3\x52\x29\xD2\x13\x0B\xD4\x1B\x3B\x1A\xC2\xBD\x1B\xD4\x13\x73\x76\x2F\xD4\xD0\x31\xFA\x18\xEA\x94\x2E\x70\x42\xE1\xF8\x35\x2D\x55\x2D\x4E\x70\xAA\x54\x43\x69\x9F\x86\x34\xCC\x29\x38\x15\xE2\x56\xEE\x73\x9E\x19\xF5\x6F\xC0\x4B\x98\x7A\x1C\xC1\x94\x22\x49\x8D\xF4\xA6\xAF\x23\x0B\x7A\xE3\x45\x72\xB8\x99\x63\x76\x42\x29\xDD\xD5\xF9\x40\x21\x66\x86\x42\x6A\x36\x51\x3A\x22\x31\xEC\x84\x33\x4B\x24\x85\x97\xE8\xBF\x92\x3B\xDD\x82\x25\xC2\x38\xD3\x09\x91\xD8\x50\x94\x2C\x92\x58\x4B\x76\xA5\xAA\x2B\x8D\xE0\x2C\x58\x92\x05\x22\x01\x32\x04\x75\xC8\x04\x22\xD1\xED\x54\xEB\x3D\xAF\x98\x19\x22\xD1\x33\x44\xA2\x23\x91\x68\xDF\x74\xAC\x1D\x8F\x2F\x0E\x28\xDA\x4E\x94\xEC\x5D\xCE\x87\x16\xC1\x7F\x99\x0E\xB6\x07\x78\xC8\x3D\x24\x53\x78\xE8\x9B\x43\xF8\x87\x0D\x65\xB7\x35\xD8\x87\x53\x60\xCF\xBF\xA1\xC0\x3E\x3C\x3F\xB0\xBB\x41\xD4\x6D\xA0\x0D\x32\x70\xB7\x86\x5B\x39\x05\xB7\xEC\x1B\x0A\x6E\xE5\x79\x92\xEB\x07\xC5\x7C\x7A\x8D\x82\x64\x08\xC6\xCB\x45\x46\x59\xAB\xED\x9C\x7F\x52\x94\x48\xA9\x87\x2B\x73\x6E\x90\x4F\x93\xEA\xF0\x1B\x0A\xE4\xF7\x93\x54\xEF\x07\xDC\x06\xFF\x16\xE1\xE6\x3F\x21\xFA\x80\xD3\xDB\x07\x5C\xF1\x6F\x12\x70\x9F\x9C\x02\x9C\xDA\x3E\xE0\xFE\x4D\x0A\x15\x32\xF3\x36\x93\x2A\xFA\xBC\x00\xF7\x8D\x25\x55\xCE\x17\x70\x9F\xEE\x00\xF7\xB0\xCB\x85\x8C\xBF\x54\x90\x34\x9A\xDD\x59\x06\xD0\x84\xF2\x55\xEB\xB5\xBF\x73\x03\xCF\xB6\xD8\xB1\x65\x5A\xE7\x54\x3D\x70\x4F\x01\x7B\xA5\x32\xFF\x06\xC1\xEC\xD4\x7E\x51\x75\xFB\x0A\xFB\xC5\x42\x84\xB5\xC2\x5E\x85\x83\x9A\xC1\x31\x43\x76\xAE\x0C\xE7\x58\xFD\xD3\x9A\xBC\xFE\x86\x22\xE2\xFB\xA7\xC9\x93\xDD\x8D\x4E\xCB\x93\x99\x8B\x9F\xBB\x44\x2F\x72\xCD\x1F\xA3\x17\xD6\x7F\x35\xD2\x4E\xFC\xE9\xD9\x33\x8C\xE8\x6F\x35\x69\x17\x72\x21\xB5\x32\x99\xCD\x43\x94\xBE\xE0\xB0\x92\x6C\xBE\xA9\x64\xF3\x8D\xCF\x2E\x1A\x1B\xCD\x70\xF2\xF6\x0B\x9B\x6F\x32\x3D\xBB\xF4\x74\x7A\x49\xDB\x39\x60\x97\x5A\x30\x49\xD5\x5C\x33\x5C\x4D\x19\xCA\x6A\xAE\x19\xAE\xE6\x9A\xE1\xD9\x26\x66\x38\x6F\xBE\xD9\x4D\xCD\xF0\xDE\xE6\x9B\xE9\x99\xE1\x96\xF0\x62\x3B\x33\x1C\x1D\xE3\x9D\x4C\xF5\x80\x4F\x8B\x48\xF1\x1F\x43\x5F\xC8\xF8\x53\x07\xCC\xB1\x6D\xCB\x07\x0B\xEF\x4A\x0E\x19\xDE\x9D\xFC\xFE\xC3\xEE\xB7\xFF\x80\xF0\x7F\x25\x3C\x3A\xD3\x8F\xFF\x24\x9E\x45\x53\xDC\xA3\xA4\xC2\xD9\xB4\x02\x36\xF2\xA7\x49\x23\x9F\x17\xFE\x8B\xC2\xDF\xD8\xD5\xA1\x53\x8C\xCF\xE4\x32\x9B\x04\x4C\xDB\x6E\x11\x66\x84\xE9\x9C\x30\x9D\x01\xA6\x0B\xC6\xF4\x20\x62\x1A\x9D\xB2\x73\xC0\x74\xC6\x1B\x77\x79\x82\xE9\x1C\x31\x3D\xE4\x8D\xBB\x81\xB3\x6D\x93\xCF\xDD\xB8\xCB\x5D\x7F\x6B\x2D\x9F\xBB\x71\x97\xCF\xDD\xB8\x1B\x6E\xB2\x71\x97\x11\xA6\x07\x9B\x6E\xDC\x65\x80\xE9\x8C\x31\x5D\xF4\x36\xEE\x06\x84\xE9\x81\x8B\x1B\x77\x16\x31\x6D\x9D\x9A\xB3\x02\xF1\x38\x13\xE3\x36\xD8\x24\x70\x14\xF3\xA7\x6F\x42\xF6\xFC\x20\xEB\x2F\x9D\x86\x6C\xE9\x14\x7E\x22\x9D\x72\x66\x07\xF2\x35\x94\xAB\x82\xDC\xDD\xCF\x07\xAC\xEA\x1B\x17\xAC\xEA\x3C\x09\x56\xCF\x21\xD8\xBC\x74\x79\x39\x42\x0F\x36\x16\x0C\xB7\x68\x69\x27\x28\x06\x7E\x2F\x84\x64\xE9\xB3\x7E\x1B\x22\xD1\x35\x9A\xE0\xAB\xE6\xB3\xFE\x6E\x63\x5B\x31\xEB\xD7\x09\x7C\xF5\x2C\xEB\x7F\xE0\x77\x60\x37\x63\xFD\xEA\x5C\xAC\x3F\x89\xAE\x72\xFE\xAC\xFF\x53\x1D\xEB\xFF\xFB\x8E\xF5\xDF\xDB\xB1\xFE\x7D\xC8\xF9\xE1\xD7\x25\x97\x0B\x13\x50\x61\x53\x19\xC0\xAC\x39\x61\xDF\xE5\x19\x23\xF3\xCD\xAE\x52\xB3\x1B\x5B\x12\x63\xC7\x90\x9F\x52\x40\x67\x63\x56\xF0\x52\x88\xD3\x7E\x6F\xEB\xD1\x77\x5B\xFB\x71\x8B\xD7\x85\xF9\xF2\x30\x22\xA6\x68\x9D\x1A\xFF\x59\xC4\xE0\xD3\xE8\x42\xAB\x70\x76\xFC\x7E\x64\x63\x19\x35\x57\x1B\x97\x8D\x7F\x1F\x0F\xB4\xED\xF8\x2F\xE9\x2E\x52\x25\xE3\x12\x34\x33\x4B\xD0\x26\x4B\xB0\x7F\xDE\x63\xA6\xB7\x60\x33\xBE\x86\x31\xAB\xB8\x0D\xBC\x44\xA7\xE5\x02\x83\x0C\x44\x2A\x31\xB3\x8A\x9B\x99\x55\xDC\xCC\x5C\xC5\xCD\x24\x54\x32\xA0\x3B\xF2\x26\x50\x09\xE8\x63\xB5\x89\xAB\x10\x7A\x9D\x6A\x9D\xEF\xC0\xF7\x56\x61\x77\xDE\x63\x66\xCE\x7B\x4C\x38\xEF\x71\xC6\x5F\x3E\xBD\x0A\x67\x95\xB3\x99\x8B\xA3\xBC\x26\xFD\x46\x10\x0E\xA8\x56\x09\xBF\x51\xF4\xB4\x45\x45\x9D\xAA\xA8\x2D\x72\xEB\xD4\x6A\xB7\x92\x55\x87\xA6\xE4\x04\x75\x4A\x85\xEB\xDC\xCF\xCF\x75\x78\x9A\xF1\xE1\xA9\xFD\xCA\x1C\x9E\x66\x5B\x1F\x9E\xDA\xF3\x3B\x3C\x35\xBD\xC3\xD3\xFE\x22\x4E\x0F\x4F\xC5\x2C\x7A\xF6\x29\x76\xD3\x11\x44\xF2\x5E\x94\x8F\xEE\x39\x13\xC8\x70\x18\x1A\xD7\x98\x8C\x6B\x4C\xD2\x05\x7D\x09\x2B\x4C\xD0\x0A\xC3\x4B\x3A\x89\xEF\xC0\x63\x36\x69\x6C\xDC\x2E\xC3\xCF\x71\xAF\x19\x79\x40\xED\xDD\xB2\xB5\x9B\x35\x60\x46\x1C\x50\xB7\xBE\xE2\x26\x18\xEB\x41\x8C\x51\xE3\x3F\x72\xEB\x4D\x1B\xE2\x80\xFA\xEC\xAD\x58\xE8\xEF\xA5\xC7\x7B\xE8\x11\x1A\xFF\x08\xBF\x79\x27\xBD\x79\x1F\x3F\xBE\x97\x1E\xDF\xD3\x55\x7C\x27\xBF\x79\x25\xBD\x79\x13\x3F\xBE\x9E\x1E\x5F\xDB\x55\x7C\x25\xBF\xB9\x99\xDE\xBC\x88\x1F\x7F\x9E\x1E\x9F\xDF\x55\xBC\x99\xDF\x7C\xE4\xA5\x34\xC6\x97\xF2\x18\xE9\xF1\x9E\x97\x76\x63\xA4\x9F\x4E\xF8\xF1\xFA\x01\x75\x1B\x4D\x10\x20\xFE\x85\x5B\xE3\xCF\x7B\xBB\x9F\x1F\xEC\x7E\xBE\xB7\xFB\xF9\x3B\xDD\xCF\xD7\x77\x3F\x5F\xD2\xFD\xFC\xF9\xEE\xE7\x17\x5E\xDA\xB5\xDB\x0D\xE3\x9D\xAF\x88\x3F\xEF\xEA\x7E\xBE\xA9\xFB\xF9\x5A\xF8\x59\x96\xC2\x8B\x7D\x84\x86\xF2\x5E\xF5\x8D\x87\x84\x48\x62\x5F\xB7\x18\x78\x1E\x3B\xEA\x51\xD8\x07\xB1\x4F\x7D\x18\x9A\x70\xE2\x28\x5D\x85\x87\x92\x1B\x6F\x8D\x25\x8A\x4A\x6E\xEB\x4A\x74\x2D\xFD\xDD\xF0\x08\x2A\xCC\x3E\xF5\x76\x7A\xB3\x4A\xE1\x2B\xEC\xBA\xBF\x1B\x9A\x43\x66\xA4\xFC\xAE\x96\x3C\x7E\x12\xEF\x7E\xB2\x7D\xC7\x14\x94\xD7\x69\x0C\x8C\x43\xFC\x57\xEE\x53\xC8\xBA\xF1\x94\x0D\x6F\x6D\x63\xE8\x50\x33\x32\xEC\xD9\x68\xA6\x3F\x34\x78\x8D\xA8\xE5\xAB\x31\x12\xDE\x64\x7C\xA9\xC2\x09\x30\x9B\x77\xE1\x55\x45\x6C\x97\x62\x13\x36\xDA\x65\x4E\xAC\xF9\xF5\x16\x39\x12\x74\x9C\x39\x4D\xFE\x4D\xE3\x96\xFC\x47\x24\x5D\xE7\xB2\xB8\x2D\x41\xC3\x30\x2D\x45\xDC\x8D\xC7\xF0\x19\xCC\x98\x1B\x60\x65\x93\x42\x13\xF6\xEB\x2B\xF4\x11\xD4\x7C\xF2\x4B\x91\xC1\x69\x2A\xA5\xEF\x16\x42\x23\x1D\xB9\xF9\xE3\x13\x2A\x8F\x00\x72\x78\x58\xE5\x2D\x9E\xF1\x49\xF4\xA0\x3E\x28\xC8\xEB\xF3\x1D\xBF\xD4\x79\x7D\x52\x43\x58\x1B\x26\x01\x33\xC3\x87\x92\xC1\xC7\xF3\x19\xC9\x92\x3C\xC4\xC2\x8B\x02\xDD\x07\xCA\xCF\x1B\x39\x98\x47\x0E\x7E\xBD\x6D\x14\xBA\x6A\xCF\x10\x86\x9E\x21\x0C\xDD\x27\x0C\xD3\x11\x86\x9E\x21\x0C\x14\xF8\xE3\x16\x2F\x4C\xEF\x6A\xEB\x0C\xA3\x52\xD4\x39\x5E\xB5\x6A\x0A\xBA\x0D\xE2\x64\x3D\x40\x04\x34\xD2\x29\xBC\x54\xE4\x0B\xF8\x42\xBA\xEC\x70\xB8\xDA\x13\xC2\xA0\xE0\xAD\x65\xC0\x8B\xA4\xDD\xDE\x01\x3C\x09\xD0\x6F\x47\xAD\xC3\x28\xCA\x03\x3C\x5B\xF7\x23\xC4\xB9\xF1\x7B\x99\x84\x8A\x43\x95\x8A\x38\x40\x8B\x46\x12\x0D\x0D\xF0\x53\x7B\xB8\xD2\xCE\x70\x48\x68\x14\x30\xB8\x0F\x05\x0A\x98\x1B\x00\x05\x81\x0D\x81\xDF\x3A\x89\xF1\x49\xD0\x67\x76\x80\x9A\x83\x13\xB8\x9D\x46\xCE\xDA\xA7\xDA\x46\xC2\x64\x90\x08\x07\x6C\x99\xF8\x1B\x5A\xB2\x4A\x24\x4A\x69\xDA\x5B\xD5\x38\xE7\xB9\x23\x95\x5B\x8F\x54\x6D\x6F\xA4\x18\xE2\x1B\x06\x66\x31\xE4\xC2\xEC\xA8\x6C\x1C\x55\x6F\x14\x62\xEB\x51\xC8\x6D\x8D\xA2\x2C\xCB\xEF\x96\x72\xE2\xF8\x3E\xA5\x23\xA2\xC3\x68\xFB\x68\xD4\xCA\xD5\xA5\xF1\xAF\x70\x50\x38\xD6\x32\x9C\x2C\xFF\x1D\x6A\x6C\xBE\x00\x65\xEB\x85\x1B\x3D\xE7\x67\xB5\xEE\x91\xC5\x41\x1B\xE5\x25\xE7\x51\x8F\xE2\x4D\xEC\x97\x6A\xE2\x83\x12\x89\x4C\x95\x5D\x43\x81\xC2\x61\xED\xBC\x07\x07\xCD\x71\x1E\xBE\x0F\x6A\x8B\x7D\x54\x58\x8B\xF8\x21\xAD\x32\xFE\x10\xF5\xC0\xF0\x21\xBE\x31\xE5\x1E\x41\x2B\x15\x0B\xF7\xA9\xD0\x6E\xF9\xAD\xC2\x89\x57\x87\xC0\x0C\x49\x05\xD2\x52\xD1\x7B\xB4\x7C\x62\x77\x35\xB7\xA7\x0D\x81\xC6\xF3\xEA\xF1\xFB\xD8\xCD\x57\xD0\x02\x8E\xDF\x63\x44\x87\x37\x9D\x89\xBA\x31\xB0\x1F\x8A\xDE\x90\xA8\x46\x6F\xE8\xC7\x95\x10\xC1\x92\xF2\x62\xBF\x78\x2F\xB4\x54\x09\xFF\x3E\xF8\x3B\x7E\x0D\x5D\x97\x07\x00\x7D\x18\x85\xCA\x17\xB0\xF8\xB5\x5C\x6C\x0E\xA8\x9F\xBF\x2D\xCA\xA5\x9B\xE9\x27\x45\xF5\x03\x7A\xB8\xF5\x36\x62\x2B\x18\x32\x56\x7A\x49\x5F\x23\x63\x8F\xE9\x19\x38\xEA\xBE\xFF\xD4\xD9\x1E\x7B\x93\x97\xD3\x48\x00\x77\x61\x0B\xB7\x2C\x78\x58\xF0\x03\x9B\x2A\x77\x86\x18\x21\x21\xD0\x5E\x59\x7E\x46\x29\xDD\xA5\x60\xB9\x4B\x6C\x76\xAD\x99\x83\xB0\x72\xB0\xEA\x47\x74\x76\x64\xF0\xC9\x92\x42\x77\x91\xBD\x61\x98\xB5\x76\x1A\xCD\x6E\xBA\x02\x8F\x16\x83\xD3\x44\xAF\x8D\x68\x7D\xD8\xA1\xE5\x69\x4A\xFF\x88\x76\xFC\xE6\x2E\xE6\xF7\x23\x66\x82\x8F\x73\xE8\x8B\x24\x82\x38\xF4\x7F\x89\x2E\x1A\x83\x1A\x6F\x71\x6A\xF9\xBE\xFB\xEE\xBB\x6F\xF1\x87\xE8\xF2\xE8\xED\x34\x7B\x6A\xF8\xEF\x82\x2F\xBF\x74\xE6\x32\x4D\x0E\x60\xE3\xD6\x3B\xFF\xCF\xBF\x84\x41\x9D\xE8\x80\xC6\x3B\x98\xE5\x47\x93\xB5\xC0\xD8\x51\x54\xFD\x7D\x53\x23\x2D\x93\xA1\xBE\x23\x5D\x41\x0A\x6F\x24\xFA\xDF\x7A\x66\x1A\x75\xD1\xDF\x25\xBA\xCB\xDD\xCF\xDB\x22\x62\x7A\x8C\x94\x1E\x22\x83\x73\x44\xF0\x7F\x88\xB3\xE6\x68\xE0\x21\x80\xB0\x8A\xAB\x83\xAE\x2B\x88\xF1\x5B\x02\x22\xBA\x48\xD0\xE4\x61\xC9\xD1\x4A\xA6\x23\x94\x3C\xED\x01\x18\xCD\x66\xA3\x40\xBE\xB5\x65\xEF\xB7\xEA\x2E\xC2\xD1\xDC\xEE\xB7\x08\x99\x1E\xC9\x67\xD4\x8E\x6F\x3E\xDB\x0B\x19\x1F\x42\x7E\x53\x68\x47\xAE\xF3\xD1\x18\x2E\x34\x04\x96\x7F\xC7\x73\x7B\x44\x16\x42\x67\x9A\x7E\x0B\x7C\xE5\x1A\xC3\x51\x72\x34\x2D\x9D\xC4\xE6\x6E\xC0\x10\x4E\x86\xF2\xE9\xA9\xE8\xF5\xBD\x4E\x42\x4A\x13\x8C\xF5\x24\x81\xDF\x8E\x9F\x7B\x16\xDD\x5F\xBD\x5C\xA7\xE5\x42\xDA\x47\x81\x3E\x74\x61\xE8\xCF\x3B\x8B\x00\x38\xC7\xD0\x3B\x92\xC5\x05\xE7\x8C\x1F\xA6\x03\xFB\x99\xB3\x5B\x0C\x0C\x3E\x09\x01\x66\x7A\xB1\x08\x5E\xA9\xCE\x81\x21\x8E\x7C\x13\x30\x93\x06\x94\x05\x42\x4D\x42\x63\x37\x53\x48\xDB\x12\x52\x22\x42\x4A\x95\x18\xFF\x79\xC8\x90\x12\xB3\x90\x12\x7E\xE8\x74\x0A\xA9\x4D\x91\xAC\x7B\x90\xE2\x68\x58\x18\x95\xEC\xCB\x85\x94\x28\xAF\x3F\xC7\x4A\x52\xF3\x33\x20\x6C\x05\x91\x0F\xCD\x81\xC8\x14\x7E\x4E\xDF\xCF\x5E\x41\x59\xB9\x2D\x4A\x2A\xEA\xFE\xB3\x53\xDD\xDF\xF2\xA2\x73\x76\xBF\x31\x37\xE0\x7D\x88\x50\x16\x42\xFD\xCE\x8D\x62\xCF\x3D\xAB\xB4\x67\xC5\x3D\xFF\xE5\x2F\xCD\xF6\x1C\xA2\xA5\x4F\x85\x63\x7D\x20\x01\xF0\x2F\x53\x00\xF8\xFC\x0B\xBF\x5A\x00\x38\x7B\xB6\x0F\x80\xDB\x7E\xF6\xBC\x01\xB0\x69\xCE\x81\x8F\x86\xED\x1E\xFE\x35\x13\xAF\x7F\x5A\x27\xF0\x58\x1B\x23\x7B\x46\xEE\x77\xCE\x91\xFF\xD3\xD3\xE7\x44\xB4\x4F\x66\xF2\xBC\x5F\xEC\xAD\xBF\x7F\xDC\x98\x97\x73\xE0\xCE\x64\xCF\x7A\xCE\x1C\x74\x9C\x83\x09\x21\x30\x30\x4A\x0D\xCD\x41\x27\xFC\x9A\x6F\x28\xC3\x1C\x4C\x27\x03\xC2\xD8\x03\x13\x41\xD3\xD3\x74\x2C\x05\xB4\x75\x27\x56\x97\x1A\xDB\x63\x2E\x78\x93\xC7\xFA\x21\x72\x88\xC0\x5C\x2E\x4A\xE6\x96\x32\x86\xB0\x37\xAD\xC0\xDE\xCA\x9C\xF5\x39\x45\x6D\xCD\xC3\x16\x75\xBE\x8A\x37\x15\x30\x04\xBC\xF5\x37\x90\xA3\x79\x0E\xA6\xC2\xF2\x99\x8D\x1B\x37\x9E\xBF\x71\x97\xB8\x4C\x02\x70\x6F\x68\xA7\x0A\xC6\xE8\xE4\xED\x72\xB0\x02\xF3\x18\x7F\x55\xB8\xCC\xE5\x5E\xAE\xB7\x78\x22\x22\xBD\x7C\xB8\xA4\x2D\x51\xBC\x62\x47\xD1\x56\xCB\xC0\x7A\x25\x8E\xCB\x59\x2F\xA2\x50\x50\x29\xAB\x9B\x37\x23\x8A\x89\xFC\xA2\x1E\xC3\x34\x7D\xE4\xDF\x3B\x85\xEC\x17\xBF\xB0\x97\x8F\xA1\x47\x13\x71\x24\x9D\x18\x48\x08\x28\x24\xE9\x10\x33\x04\xF4\xDA\x39\x11\xAC\xF1\x50\xA6\x23\x20\xF9\xB5\x5E\x04\xFF\x7C\x8E\x45\xF0\x47\x37\x9D\xC7\x22\xF8\x1A\xCF\xE1\xDE\x73\xCC\xE1\xBD\xCF\x3C\x8F\x39\x6C\x99\x0F\xA8\x17\x07\xFD\xDC\xD2\xF0\x33\x53\x04\x11\xB8\xF2\x73\x5F\xD0\xCB\x07\x14\x07\xD2\x93\xC6\x2F\xD6\xF3\x92\x7E\x24\x11\x01\xD3\xC4\x3C\xE8\x6B\x42\xE1\x0D\x39\x92\xB9\x0C\x51\xEB\x39\x0A\x6C\x8C\x50\x4F\x21\xC3\x75\xD9\xE5\x9C\x91\x04\xBE\x4E\xBD\x0C\x00\x7B\x79\xB2\x96\xF8\xE6\xAA\xE6\x50\x70\xF2\x0A\x0E\x7D\x23\xC6\xFF\x8B\x8E\xDD\x2A\xE1\x3F\xD7\xD3\x54\x9C\xE1\xA0\x35\x1C\xF6\xBC\x39\xCF\x9E\x9C\xF1\x93\x2B\x69\x1F\xE3\xFC\x7B\xF8\xF8\x36\x7B\x80\xF1\x97\x31\x56\x7F\xD9\xA9\x81\x9A\xE2\x6D\x4E\x05\xF1\xEF\x0B\xAC\x8F\x4B\x59\x4C\xC2\x36\x42\x2F\x47\x89\x0A\x69\xC8\x38\xD3\x0F\xC7\x2F\xE5\x48\x48\x76\x9F\x2A\xBC\x3C\xC4\xE1\x02\x4D\x9D\x3B\x19\x22\x42\x3B\xE3\x2C\x5D\x54\xA3\x28\x47\x1C\x32\x09\x23\x95\x15\xE1\x32\x2B\x6D\x18\x38\x8D\x11\xD8\x92\x30\x4A\x93\x76\x21\x33\x94\xCC\xB8\x84\xB7\x78\x7D\xBD\xAD\xF3\xE0\x6A\x86\xFD\x4E\xDA\x46\xAF\x72\xAC\x02\x4B\xE1\x9F\x68\x58\x74\x57\x9B\x37\x23\xA6\x13\xAF\x95\x4E\x96\x2F\x50\x52\x4F\x18\xFF\x32\xBD\x05\x9E\x26\x37\x45\x2B\xDA\x89\x5E\xA0\x2A\xD5\x25\xF4\x50\x49\xDC\xA5\xB9\xA9\xCD\x42\x66\x3E\x4A\x80\xE0\x9F\xF7\x6B\x80\x2E\x3C\x8A\x1D\xBF\x04\xFB\x48\x4B\x5E\x84\xF0\x98\x49\xB5\xD6\xEB\xCC\xDF\x72\x5B\xFC\xA0\xF7\xF1\x2F\x60\xC9\xAF\x27\x6F\x37\x36\xA6\xDE\x6E\x35\xD0\x12\xC3\x08\x97\x4E\x95\x3F\x25\x4D\x04\x0B\xEE\x47\x89\xCE\x05\x46\xE8\xB9\xD9\xF3\x94\xC0\x60\x9B\x9A\x03\xBC\x10\x9A\x6B\x72\x9D\x1E\x7F\x36\x92\x39\x5D\x23\x8F\xD4\xDD\x25\x65\x63\x8C\xFC\x9A\x56\x18\xA8\x1E\xC3\x23\xBB\xE4\x4A\xF7\x7C\xAF\xB6\x2C\x0B\xC4\x41\x86\x75\xE7\xEE\x46\x89\xB5\x95\x29\xA1\x11\xA6\x68\x95\xCC\x44\xAF\x54\x9C\x77\xD0\x69\x1A\x2C\x98\x32\x7C\x74\xFB\xB9\x70\xDE\x8E\xC1\x1E\x9C\x09\x24\x04\x75\xDF\x16\xE8\x77\x64\x62\xB0\x30\x76\xA2\x30\x2B\x95\x8A\x21\xAF\x44\x2F\xE4\x15\x9F\x70\x62\x46\x45\x83\x8B\x02\xA0\x30\xA7\x37\x13\xA2\x5C\xCD\x26\x5D\xE4\x80\x58\x86\x43\x80\x85\x1B\xF7\x81\x34\x97\x69\xE9\xA5\xD9\x14\x5B\x74\x88\x70\x19\xED\x01\x2C\x0F\xE9\xEF\xF8\xE7\xF0\xBF\x73\x3B\xE7\xC8\x5E\x9C\x62\x32\x5B\x96\xA7\x9A\x0C\x13\xF5\x3C\x3E\x46\xF4\x42\x0A\xF9\x13\xAD\x2C\xE3\x89\x57\xD9\x1B\xCF\xCC\xE4\x99\xC8\x0E\xC7\xB8\x68\x29\x6A\x30\xB9\xB7\x90\x3A\xA2\x85\x23\x4C\x60\x54\x01\x90\x24\xAB\x8D\x5E\x6A\x4C\x44\x95\x85\xB1\x71\x4A\x19\xBD\xC4\x13\xD4\xCE\xD2\x0C\x28\x1A\xB5\x8D\x38\xE4\x5F\xE3\xCF\x6F\xC4\x48\xE0\xCA\x59\x97\x05\x1C\x9A\xE8\xCF\x65\xF1\x08\x3A\xDE\xFE\xD7\x21\xA1\xAA\x01\x9D\x15\x29\x2F\xEB\x60\x99\x05\x4C\x5A\x3E\xAB\x9E\x87\xC9\xCF\x77\xC0\xD4\x1D\x26\x2D\x72\xC8\x14\x93\xCC\xBD\xD0\x4B\x16\xBA\xEA\x44\x90\xA3\x74\x93\x50\xB8\x2C\xEA\xDC\x9B\x1E\x35\x31\x3E\xF3\x2D\xF0\x39\x35\x04\xC6\xA7\xA5\x83\x88\x1C\xF0\x99\x07\x7C\x96\x3C\x4B\x18\x58\xF9\x66\xCE\xE0\x99\xEE\x2F\x66\x87\xFB\xE8\x3C\x94\xE6\xB5\x4A\x1F\xF2\xC3\x95\xEC\x16\x99\x4C\x17\x59\xC7\x2E\xD2\xE0\x7F\x52\x00\xBB\x20\x76\xDF\xAD\xBB\xDA\x50\x54\x07\xC3\xDB\xBF\x36\xC4\x97\xB4\x18\x58\x91\xA6\x22\xD3\x35\x88\xC9\xAC\xBE\xA7\x4B\xD5\x98\x8E\x1C\xB8\x11\x6E\x46\x53\x63\x82\x13\xA3\x1A\x8A\x50\xF0\xB9\x39\x89\xBB\xB6\x39\xDF\x8E\x6C\xBA\x19\x9B\xAD\x67\x6C\x38\x8F\x65\xA0\x98\x9A\x52\xA4\x70\x60\x09\x28\xE6\x66\xD0\x3E\x98\xE5\x5A\x31\x2E\x1F\xE0\x36\x04\xE5\xEB\x00\x26\x03\xC0\x64\x02\x30\xD1\x6F\xD5\xA9\xF2\xDF\xC7\xC4\x98\x7D\x80\xA5\xEB\x16\x21\xB7\xD1\x85\xC5\x78\xBE\x02\x99\xD0\x2D\x58\x31\x85\x64\xE4\x8D\x6F\x22\x2C\x22\x8E\xE8\xB7\xC5\xC0\x7E\x75\xE7\x35\x8F\xD1\xE6\x60\xD9\xA4\x40\xB6\x41\xEE\xDB\xD9\x52\x5C\xB0\x1B\x4C\x1E\x29\x3B\xEE\x73\xE3\x92\xF7\x7D\xA7\xF2\x21\xCB\x5E\x3A\xD5\x40\x8F\xF0\x29\x48\xED\xDF\x3E\x13\xB3\x96\xE1\xB1\x69\x02\x35\x0C\x94\xA8\xD3\x28\x91\xFD\x74\xC8\xAA\xFC\xA1\xF9\x50\x4C\xF0\x3E\x4B\x33\x4E\x8E\xFF\x65\x23\x62\x0A\x5E\xFE\xFF\x1B\x91\x0B\x13\xA0\x1F\xC8\x66\x3F\xDF\x35\xFB\xF3\x73\xD6\x77\x7E\xB8\xDF\x90\xE8\x8B\x33\xBD\x79\x62\x69\xDB\xC5\x87\xB4\xFD\xF8\x90\xF8\x38\x9F\xF1\xE9\x08\x43\x1D\x60\xF8\xF6\xF3\x19\x15\x0A\x06\xDA\x3B\x1D\x35\x1A\x90\xB5\xBA\x34\xFE\x9B\xC0\x6C\x38\x5E\x2B\x56\x22\xC1\x11\x88\x20\x08\x8C\x2E\xC7\xB0\xE9\x05\x07\x15\xDD\x24\x44\x9C\x04\xAD\x20\x91\xCC\x02\xFD\xAC\x44\x45\xF1\x2E\x67\xE3\x85\x3A\x55\xFE\xA9\x8C\xE1\x04\xA7\xE7\x41\x67\x43\x5D\xBE\xE3\x68\xD2\x6C\x92\xF3\xD8\xA1\x93\x2A\xED\x94\x74\x59\x75\xF3\xC3\x15\x99\x27\x36\xE0\x88\xDE\x3E\x93\xD6\x18\xA5\xD3\xC3\x92\x1B\x19\x2C\x96\x5D\xFB\x54\x6D\x9D\x5A\xA9\x4C\x39\x93\x3D\xD9\x59\xCA\x25\x1D\x33\x28\x83\xFE\x15\x84\x61\xF9\xBE\xED\x90\x4B\x10\x9C\xE7\x24\x98\x1E\x7D\x9C\x93\x3C\x70\x35\x8B\xCD\x57\xF3\xD6\x6D\xE2\xB2\x15\xB1\x4D\x11\x90\xF5\x45\x95\x78\x0E\xF4\xE6\x36\x33\x1D\xD2\x3B\xB6\x9C\x11\xC5\x73\xEC\x63\x8A\x74\x83\x2C\xFA\x71\xE6\x87\x2B\xE3\xB2\xE4\x81\x3B\xCA\x39\xC4\x6C\xC3\xBA\xFE\x00\x7B\xC1\x10\xB3\x83\x29\xBD\x31\x27\x7E\x45\x73\xEC\x34\x43\xC5\xA1\x52\x07\x30\xC7\x01\x87\x4A\x55\x07\x5D\x36\xD3\x6A\xB6\x8D\x56\xB3\x79\xAD\x3A\xC5\x14\x95\xAF\x54\xDA\xE5\xB5\x9A\x41\x1F\x14\x92\xEF\xC8\x14\x1A\x17\xE3\xC7\x5E\x94\xFF\x8E\x73\x28\x52\xF4\xA9\x16\xD5\x05\x20\xC1\x9A\x52\x66\x85\xDB\x36\xE5\xA3\x31\x06\x92\x53\x91\x63\xEF\x17\x18\xF4\x08\xD3\x64\x1D\xED\x3C\x93\x9D\x79\x4C\x25\x3C\xF4\xD1\x0F\x61\xD5\xB9\xFD\x7A\x51\x2E\x80\xE1\xF5\xB8\x16\xC3\x93\x0E\x57\xCB\x87\xC5\xF8\x4A\xFE\x64\xDB\x48\x7F\xE1\x11\xF4\x15\x51\x5E\x3A\xB9\xEE\x9F\xB5\xB1\xA1\x8E\xA3\xE7\xAD\x1F\x1C\x22\xE6\xF9\x6D\xBD\xEF\x9D\xF0\x77\x4E\xE8\x21\x5B\x3D\xE1\x84\x7F\x69\xF7\x54\xFE\xAD\x54\xFA\xB4\xC1\x58\xB4\xCB\x8F\xBA\xA9\xD1\xA7\x9C\x7E\x46\xAD\x9D\x7A\x35\x88\x45\x34\xE5\xD0\x7B\x63\xE3\x65\xE8\x66\x02\xE3\x9F\x80\x4E\x7F\xFA\x69\xB7\x4F\xB9\x3C\xDA\x5A\x3B\xBB\x2C\x9E\x50\xE1\xFE\xA5\xC0\x64\x98\x4E\xF9\xFD\x5D\x15\xCA\xC5\x45\x7E\x8E\x68\x22\x66\xD1\x32\x56\x75\xE6\x72\x02\x27\x3A\xE6\xA8\x43\x21\x22\x00\x56\xC3\x98\x62\x75\xE1\x72\xCA\x13\x75\xBA\x6D\x06\xD1\x81\xB6\xA0\x0E\x9E\x06\x6D\x91\x7F\x66\x50\x57\xF3\x60\x36\x3F\x5B\x2A\x79\x1A\xD1\x23\xBD\x7C\x2C\xE1\x62\x99\x15\xB0\xFD\x42\xF8\xFD\x87\x1A\xE3\xE0\x05\xE6\x82\xAA\x51\xB1\x59\xAB\xC1\x40\xD1\x1D\x2E\x6D\x00\xE8\x00\x10\x78\x7A\x31\xA6\x51\xD2\xCB\xC3\xD3\xCE\xDE\xB1\x7C\x9F\x3C\x7B\x6A\xF9\x71\xA7\x62\xA6\x33\xA2\xC2\x65\xE1\xF4\xD3\x00\x6C\x4B\x98\x25\xA9\xFC\x98\xB1\x72\xA2\x4E\xC1\xFF\x74\x7F\x44\xFF\x41\xE0\xBF\x7B\xEE\x0C\x3F\x1E\x15\x47\xA7\x97\x6A\xC3\xA5\x82\x6F\x7A\x50\xFA\x80\xC7\x56\x22\x29\x27\xAD\x06\xE3\x1A\x3E\xAE\x6D\x32\x1A\xED\x91\x4A\x3A\xCB\x95\x1E\x7A\xF0\xE5\x2E\x7B\xE3\xAD\xE8\xE2\x43\x9C\x94\xBC\x8E\x9D\x7D\x39\x45\x31\xA4\x5F\xD0\x32\x45\x58\xE0\xE6\x9A\x0C\xA3\xF7\x76\x5D\x91\xE7\x36\x85\xB7\x86\x31\xFC\xE2\x8B\xE1\xDF\x9B\x1F\x55\xDB\xA4\xCE\xA6\xC3\x29\xC0\xBE\x79\xE3\xCB\x6F\xAD\x8B\x64\x64\xAF\x98\x1E\x55\xEE\x8C\x2B\x5E\x7E\x2B\x65\xBD\x8A\x0F\x34\x36\x19\x5A\xAE\xB3\xD2\x9F\xAE\x07\x2E\xEB\xA2\xF9\x52\x26\x97\x61\x30\xE4\xA8\x49\x0C\x27\x81\x81\x7B\xC9\xF2\xD7\x5E\xB5\x75\xE6\x27\x94\x8B\x80\x9E\x80\x3E\xE9\x88\x36\xC3\x59\xC1\x23\xCE\x81\x62\x51\xE8\x48\x00\x30\x87\xD2\x0F\xAF\x75\xBA\xF5\x8F\x6B\x6B\xD0\x8F\x89\xE4\x82\xD3\x4F\xE6\xE4\xE1\x8A\xD2\xDA\xAB\x83\xAE\xA4\x50\x7A\xA5\xCF\x8F\x33\x17\x2F\x71\x13\xFF\x28\xA7\xE9\x97\xA5\x87\xB7\x37\xB0\x3F\x33\x66\xAC\xF1\x1B\x6F\xFB\x8B\xEF\xBC\x16\x4C\x5A\x5F\x04\x47\xEF\x10\xB0\xCD\x0F\xAF\xED\x13\x19\x00\x65\x78\xED\x1B\x5F\x5E\xE7\x18\x72\xF2\xFF\xC4\x74\x5C\x83\xD2\x0D\xCA\xDF\xB7\x32\x9F\x16\x8B\xE6\x70\x12\x65\xB2\xA7\xF2\x1B\xC1\xCE\x77\x8D\x46\x18\xAA\x11\x07\x98\x96\x68\xE4\x75\xA9\xFF\x95\x37\xB5\x81\x47\x0E\x0D\xD4\x45\x40\x6F\xB2\xAE\x17\x13\xE2\x7A\x67\x64\xDC\x63\xAA\x1E\x19\x7C\x05\x9B\x0C\xAD\xC9\x2C\x98\x39\xE8\xD2\x7E\x2D\x56\x2B\x9D\xE6\x48\x84\x39\x77\x8E\xF7\x65\xB0\x1B\xD4\x0A\xF0\xA8\x15\x94\x5F\x72\x5C\xC2\x2B\x9C\x85\x57\x18\x96\x10\xD5\x77\x3F\xC2\x08\x3B\x14\x57\x8B\x7C\xE8\x39\xDC\x3D\x7E\x71\x51\x88\x5D\x65\xDA\xA6\x68\x69\xF8\x96\x74\x9A\x7E\xD3\x17\x91\x0A\xC6\xD1\xC9\xD1\x26\xE5\x6D\xB9\x2E\x3B\x71\x4E\xDE\xDE\x86\xBD\x51\x72\x67\x28\x76\xE3\x22\x05\x3D\x97\x14\x63\x90\x3B\x99\x09\x58\x6E\x50\xBE\x05\xA0\x1D\xE2\x2D\x27\x4D\x9B\x67\xB5\x9A\x8E\xFE\xAE\xE3\xB6\x8A\x5E\xED\x8C\x88\x11\xE9\x62\x2C\xF0\x01\x8C\x8A\x5C\x43\x19\x8C\x59\xB0\x35\x30\xD2\xCA\x04\xF1\x56\x74\xA3\x2A\x66\x74\x04\x0A\x73\x26\xAE\xC0\x93\x21\x3D\x07\x7E\xBC\x5F\x83\x9B\x03\x00\x60\xF4\x22\xD4\xE8\x2C\x90\xD4\xE2\x40\xEF\xB5\x9E\xD6\x73\x0C\xA6\x4E\x6C\x9B\xC2\x71\x68\xDA\x8C\x04\x60\x00\x62\x16\x80\x48\xF2\xEC\x79\x73\x8C\xDF\xD9\x5C\x55\x93\x18\xBA\x89\xDC\xE3\xF4\x15\x61\x27\x98\xC6\xAF\x5A\x76\x67\xA0\x18\xC1\xA0\xE6\xAA\xB4\xFA\xDA\x15\x14\xA5\x15\xC1\x5E\x89\x12\x1A\x44\x9F\xFF\x2E\x17\x13\x0D\xE7\x57\xE7\x28\x9B\x73\x53\x67\x75\x97\x53\xC8\xF9\x96\xFC\x4C\x55\xBA\xEE\x15\xC7\x34\x2F\x00\x4E\x96\xB5\x40\xBE\x5B\x80\xE9\xAA\x38\xD3\xE6\x21\x8C\x01\x6F\x30\x0A\x62\xD4\x18\x0E\xE3\x7D\x21\xD2\x4A\x28\xFD\x73\x3A\xCC\x7F\x54\x4A\x75\x6E\x59\x6E\xB3\xA4\x5F\x7A\x3A\x69\x36\xC7\x9B\x57\x7E\x6F\x2F\xBF\xEA\xDE\xE5\xD3\xA7\x96\x1F\xF5\xAC\xC6\x2C\xE7\x3F\x54\x09\xFF\x62\xE9\xCC\xED\x8D\x04\x8E\x78\xD2\x93\x7F\x68\x2F\x93\xAB\xF4\x6A\xBD\x96\xFD\xA3\xA2\x5D\x4E\x3B\x11\x53\xCD\xEE\x6A\xC7\xCF\xEA\xCE\x0C\x77\x01\x69\x3E\xE7\x97\x7A\xCE\x4E\x3F\xFF\xBA\x1B\x37\x30\x02\xA8\xA6\xEA\xE4\x2E\x94\x7E\xF0\xC9\x5F\xEC\xFB\x53\x88\x5A\xFA\xDD\x4C\xF5\x9A\x75\xA6\xC9\x95\x78\xC3\x84\x35\x11\xE1\xC4\x1A\xDD\x45\x92\xAD\xDF\x8D\xF7\x5F\xE8\x62\x3D\x9E\xEE\x3A\x71\x40\x8D\x9D\xA6\x17\x7E\xDC\xF2\xBB\x12\x54\x1F\x76\x3F\x22\xBA\xC4\x63\x1B\xD7\x9D\x44\xDC\xA4\x70\x23\x78\xF3\x24\xB7\x9B\x90\x49\x77\x38\x81\x24\x62\xAE\x08\xB9\x21\xD8\xD7\x8D\x69\xD6\x04\x9A\x55\x74\x2B\x2D\x54\x5F\xBB\x02\x73\x52\x52\x1E\x34\x20\x81\xE5\x09\x3B\x50\x31\xAB\xE6\x4C\xBF\xA3\x65\xE2\x02\xDD\x8E\x02\x26\x6F\x0B\xC7\x22\x21\x3C\x30\xF9\x3D\xFF\xEE\x8B\x13\xA0\x66\xCB\xF2\x54\x8D\xD9\x0B\x46\x4E\x62\x90\x57\x3A\xB0\xC9\x42\x3F\x53\x3E\x04\xF7\xB1\xBD\xB2\xE5\xAC\xD3\x3B\x63\x5D\x8C\x9D\x10\x7F\xD6\x76\x91\xB5\x32\x8A\xDB\x4C\x73\xE1\x53\x6E\x2F\xEB\x01\x09\x69\xA8\x31\x5C\x73\x05\xF0\xA9\x02\x7A\x9C\x06\xDB\x30\x2E\x75\xF8\x88\x41\xD7\xC4\x2F\xAF\xA8\xCB\x00\xBC\x32\xEE\x17\x0F\x9C\xAE\x33\xE0\xE9\x61\x68\x28\x95\x31\xB4\x8C\xE1\x84\x1A\xAE\x58\x6B\x86\x2E\xA7\x1B\xF7\x6D\x53\x82\xEA\x68\x5D\x39\xFE\x00\x0D\xDF\x95\xB8\xFD\xED\x72\x66\x9C\x45\xEB\x86\x81\x25\x22\xC3\xCD\xDA\x9A\x32\x3E\x60\x32\x28\x4D\xBA\x97\x85\x51\xE3\xD1\xBA\x4A\x58\xA8\x62\x46\x9B\x97\x7C\xA3\x8B\xE4\x26\x35\xEF\xF9\x02\x5A\xDE\x11\x25\x9E\x89\x99\x49\x12\x59\x76\xCE\x86\x99\xE1\x24\xFF\x00\x75\x76\x0B\x8E\xC9\xFC\xA0\xE8\x57\x9E\xCE\x45\x91\x0B\x9F\xD3\x60\x66\xE5\x44\xE2\x7E\x19\x39\x2C\xEB\x54\x15\xB0\xE9\x03\xFA\x60\xD3\x01\x0F\x48\xD4\x70\xAD\xAF\x26\xB1\x8A\x5A\x4D\x30\xD2\x6C\xAD\x66\xB6\x61\x7A\x06\xDA\xDF\xCA\x79\x5E\x65\xF3\xB8\xB3\x48\x27\x8C\x32\xD0\xC4\x6B\x69\xB8\x8D\x82\x01\x1C\xF0\xF4\x8A\xFD\xCF\x58\x6E\x38\x4A\xC0\xB3\xE9\x12\xE1\x15\xC9\x3B\x3E\xA6\x5D\x6B\xA3\xC0\x31\x09\xDE\x47\x4E\xB5\x21\x3D\x05\x8B\x3D\xED\x82\xC4\xA3\x36\x36\xC9\xE2\xF3\x11\xDD\x0B\xDB\x7E\x9E\xAC\x25\xD9\xA5\x27\x66\xC1\xF9\x53\xE2\xDA\x42\xE1\xCB\x61\xD1\xAD\x8E\x77\x3A\x0B\x20\xDC\x94\xEB\xA0\xF2\x65\xD7\x9A\x84\x5C\xB0\x9A\x8D\xF3\x83\xB6\xE2\x44\xA2\xF0\xE6\x53\x33\xEA\x1C\x96\x8B\x0D\xB1\xD2\x25\x58\x6B\x48\x2B\xA6\x4C\x06\x23\x5D\x0E\x5F\xAF\x3D\x86\x75\x02\xBB\x36\xDB\x23\xDF\x2D\xE6\x7E\xF3\x78\xD7\x38\xF4\x1E\x4B\xD2\x31\xE0\x29\x5F\x02\x83\x9C\xEC\xDA\x04\xC9\xB9\x3B\x07\x1F\x04\x06\x38\x35\x9A\x1C\xD7\xEB\x88\x87\x82\xCC\x61\x14\xC7\x41\x8F\x3D\x40\x20\x0F\xCD\x5B\x1E\x74\xA0\x81\x1E\xF7\xFC\x4F\xE8\xF7\x7E\x4E\xCD\x62\x4A\xD1\x59\x0D\x19\x9D\x7A\xFA\x0D\x69\x80\x89\x42\xF0\x1C\x9D\x81\x42\x20\xF9\x7F\x33\x17\x7B\xC3\x8A\x98\xD7\xB1\x4E\xB9\x36\x6B\x09\xB8\x71\xDE\xD3\x12\xF0\x86\x6F\x7C\xD4\x97\xE8\xBD\x8D\x5D\x16\x3F\x48\x61\x3A\x2E\xD1\xE3\x65\xF1\x03\x95\xF0\xB7\xBF\x26\x9E\xF2\xD2\x96\x39\x71\x59\xD0\x38\x31\xDF\x23\xEE\x80\x1A\x8E\xC8\x0A\x3F\x97\x45\x9D\xBB\x0C\x2C\x7B\x0E\x9A\x69\xBF\x1F\x4D\x50\xC1\x3E\x97\x49\x80\xFB\x39\x98\x83\x61\x16\x4B\x6C\x7A\xE5\x3C\xAA\x1F\xA4\xF1\x8D\x9B\x41\x48\xF3\x6D\x9F\xE6\x06\xDF\x5F\xE9\x12\x4F\x9D\xEA\x9C\xA2\xE2\x1B\x8E\x03\x83\xCA\x9B\x2C\x43\xCE\x30\x27\x67\x57\x83\xEC\x56\x83\x88\xAB\x41\x44\x45\x76\xFA\x62\xED\xCA\xEC\xBE\xF9\xAC\x98\x84\x47\x3C\x74\x6F\xE0\xD7\x2F\x07\x5E\x4C\xCC\x0F\x50\xFA\x46\x25\xED\xF6\x39\x43\xCC\x9D\xCE\x73\xD3\x47\xC9\x31\x05\x95\xF4\xA9\xEC\x54\x05\xF0\xEA\x2D\x97\x86\xE2\x6D\xC0\x51\x63\xD9\xD6\x32\x47\x42\x7A\xD8\x60\x3B\x80\x89\x40\x64\xC9\xBA\x44\x6D\x5D\x86\x52\x13\x13\xC8\x91\xD8\x08\x52\x90\x92\x66\x61\x0E\x1F\x89\x4C\xD2\x1E\xC6\x1F\x7C\x1A\xE6\x31\x6D\x10\x54\x4C\xC1\x0F\x3D\x60\x8D\x20\xBE\xF8\xFA\x73\xAA\xAF\x25\x0B\xED\x8F\xD4\xFD\x61\xA9\x51\xA1\x4F\xDC\x6E\xF6\xE1\xED\x1B\x33\x7E\xDD\x99\x24\x61\x40\x8F\x00\xE9\x02\x1C\x7B\x63\xA4\x70\xCD\xCE\x13\xAE\x59\x60\x7C\x33\x70\xB5\xAD\xCB\x18\xAE\x39\xC1\x35\x73\x39\xFA\xAA\x64\x6B\x01\xAE\x19\x66\x09\x4C\xE0\x9A\x05\xB8\x66\x78\xBD\xCF\x8B\x2B\xC9\x1B\x65\x8A\xA0\x03\x35\xA7\x42\xCA\x6C\x02\x4E\x74\xD6\x7A\x80\x18\x17\xEB\x26\xC0\xB3\xF8\x8E\x74\x43\x7B\xAE\xE9\xE0\x54\x1B\xEE\x81\x38\x39\x65\x2A\x9E\x7B\x20\xDD\x6D\xF4\xA9\x61\x28\xA7\xD7\xDA\xE9\xC1\x50\xAD\x35\x32\x06\xFB\x20\x52\x71\xC1\xAB\xB8\xE0\x55\x3A\x9C\xDB\x15\x29\x63\xD3\x09\xF8\x7C\x74\xE0\xE1\x15\xC8\xF6\x7B\x81\x3A\x88\xA4\x58\x2E\x74\x5A\x8F\xC9\x36\x14\x59\x31\x74\x54\xF9\xB2\x33\x21\xF8\x0E\x66\xFE\x6A\x94\xFF\xC9\xD6\x5B\xCA\xC9\xE2\x0E\x57\x78\x46\x4B\xA9\xB1\x28\x1A\x14\x39\xA4\x69\x27\x1F\x1D\x3B\xE1\xC6\x31\x00\xD5\xFD\x69\x1F\x9D\x98\x9C\xBC\x12\x2D\x7A\x3A\x24\xD6\x38\x63\xE0\x8C\xD3\x50\x92\x6D\x04\x94\x8C\x80\x92\x11\x50\x98\x1E\xE9\xB9\xAA\xE7\x64\x17\x43\x4D\x80\x36\x06\xA2\xF5\xB1\x95\x64\xF9\x40\x47\xEE\xFD\xAC\x26\x72\xDE\xAA\x1B\xFF\x77\x56\x30\x99\x93\xD7\x92\x4C\x24\x6C\x54\x35\xD2\x99\x47\x23\xDB\xC3\x69\xF3\xF2\x55\xF3\x96\x21\x06\x36\x90\xE8\x64\xC2\xFD\xCD\x5B\xE5\x96\xFA\xE3\xEC\x6F\xA6\x0F\x06\x60\x4F\xAA\x53\x14\x18\x06\x5D\xAE\x1A\x93\xA6\xF9\x30\xE5\xDB\x75\xCF\x89\x79\xAB\x2C\x91\xF3\x28\x48\x42\x3D\x64\xEF\x8A\x11\x5C\x63\x62\xFF\xA3\x55\xDC\x2F\xB0\x88\xF0\x12\x03\xBF\x3C\xBA\xE2\x93\xB5\x44\x35\x0E\x5A\x62\xF4\xA6\x03\x1B\x3C\xEA\xB5\xB4\x21\xEE\x80\xE5\xDA\x78\xB8\x70\xD1\xEA\x52\xD8\xEC\xEF\x67\x61\xC3\xBC\xC0\xF3\xD4\x5C\x32\xEB\x6C\x72\xC3\x0B\x6F\xC4\x64\x9E\x13\x73\x30\x8F\x0B\x37\x32\x08\xF6\x7A\x2E\x8A\xF4\xF8\x37\xB7\x78\xCB\x81\x1C\xF2\xC7\x22\x8D\xE7\x14\x14\x24\x73\x99\xCB\xAF\x58\x6A\x32\xE8\x9D\x4C\x37\x8C\x1E\x07\x50\xA3\x54\x2E\xB4\x05\x87\xAA\xBB\xE5\x38\xF2\xC4\x2D\x4A\x37\x0F\x89\xD3\x6A\xFB\xBF\x0E\x2C\x5E\x45\x58\x7C\xE9\xD7\x07\x16\x5F\xFF\x75\x87\xC5\x87\xCE\xA6\x50\x45\xFD\x37\x24\x82\xA7\xD4\xA9\xF3\xF3\xAC\x9E\xB3\x92\x3A\x9F\x4A\xF6\x7C\x2A\xE5\xE7\x53\x29\x3B\x9F\x4A\x72\xA6\xD2\xCB\x64\xAC\x35\xE5\xDB\x28\x7A\x11\xFB\x94\x36\x36\xCB\x4B\xCA\x18\xE7\xCD\xF8\xDD\xBC\x0F\xFE\xAE\x5B\xE0\x39\x8F\xCF\xAF\x7A\x0E\x3C\x67\xF1\xF9\xCF\x7F\xA1\xFF\xFC\xB3\x1B\xF0\x6C\xE3\xF3\xCB\x6E\xEC\xBF\x7F\x2E\x3E\x6B\x7A\xA6\x21\x52\x50\x9A\x63\xC7\xFD\xC6\x6B\xF5\xA1\xF2\x16\x2D\xF5\xC4\x4F\xD8\xD9\x29\x6B\x9D\xBA\x92\xB4\xE9\xB6\x91\x31\xED\x17\xCA\x48\x78\xFA\x74\x48\x02\xC6\x79\x92\x95\x7F\x5C\xDB\x08\x3E\x3D\xE5\x38\x53\x61\x79\xD8\x23\x28\x06\x5F\x35\x69\x6B\xBE\x0F\x72\x67\xF7\x06\xD6\xE5\x6A\x5B\xD3\x9D\x4E\x8A\x50\x4F\x8D\x99\xF9\x8D\x65\x24\x53\x93\x16\x14\xB5\x60\xA8\xE9\x57\x61\x34\x20\x6C\xCC\xB4\xE1\x10\xB2\x51\x69\x63\xB6\xDF\x98\xED\x37\x66\xA1\x31\x94\xCE\x16\x1B\xA3\xEC\x58\xDE\x76\x8D\xC9\x6D\x36\x26\xBB\xC6\xF0\xC8\x19\xDE\x9A\x75\x87\x13\x6C\xE1\x97\x8A\xBF\xB0\x69\x3A\xCE\x29\xBF\x17\xD0\x33\x5C\xF7\x1B\x1B\x1B\xD7\xB4\x4E\xFA\x8D\x77\x8A\xF6\x84\xFF\xE2\x7D\xC0\x13\x02\xD6\x96\xF0\xB9\xC3\xE2\x52\xF9\xB6\x8C\xD0\x88\x97\xF0\x40\x07\xA0\x33\x49\xC9\x29\xD3\x32\xBF\x21\x8F\x53\xF4\xA0\x03\x4A\xB8\x0C\x55\x96\xCC\xDF\x2D\x8F\xFB\xBB\x64\x40\x70\x23\x49\x79\xC8\xFC\x97\xB0\xFC\x50\x25\xA1\x45\x6C\x86\x72\x2A\x4E\xAE\xC4\xD0\x07\x30\x93\x3B\xE5\x11\xD4\x33\x30\x9C\x51\xE6\xB3\x75\x7F\xD7\xE2\x71\x27\xFD\xA3\x8E\x9F\xE0\xC6\xEF\x91\xC7\xFD\xDD\xDC\xB8\x8A\x8D\x73\x18\x2F\xBC\x8E\x43\xBF\x68\xF7\x2D\xE3\xBA\xA4\x47\x09\x7F\x2B\x26\xF4\x16\xD8\x0F\x06\xED\xC1\x24\x6E\xA8\x7A\xD1\x00\xEE\x92\xAB\x74\x09\x23\xF3\x0F\x21\xD0\x97\x44\x70\x49\x4D\x15\x87\x7A\xAE\x26\x3E\xCE\x9D\xCF\x7C\xFC\xD2\xAD\x3F\x66\x60\x7D\x42\x1E\x46\x82\x8C\xB5\x54\xD2\xB7\x4A\x3F\x57\x09\xFC\x50\x3D\xD4\x14\x0A\xE4\x51\xC7\x7D\xB6\xEE\x32\x3F\x5A\xF7\x1B\x77\xEB\xE3\x27\x02\x28\x5D\xE6\x3F\x2B\x8F\xFB\x7B\x18\x90\x9A\x01\x29\x22\x20\x15\x06\xCC\xEA\x83\x54\x30\x48\xE9\x2B\x24\xBD\xC6\xFA\xE7\x4B\x26\xD5\xCD\xA6\x44\x2F\xE6\x00\x21\x01\xD4\x22\xC1\x5A\x1D\xC1\x86\x93\x9A\x62\x6B\x58\x9F\xA3\x6D\x97\xF9\x7B\x25\x1D\xD9\xC4\x1A\x92\xDB\xBC\x05\x49\x4D\xA6\x2F\x52\x1A\x54\xE9\x8B\x0E\xEA\x28\xE2\x0D\x86\x90\x02\xE0\x8E\x00\xB8\x3B\x61\x59\xDD\x23\x18\xBA\x3E\x5B\x3F\x81\x90\x3F\x41\xD9\xCD\xDF\x2A\x29\x44\xC3\x7D\xE2\x08\x45\x6C\xB9\x5C\xA0\xFF\x09\x15\x2E\x72\x21\x7C\xB8\x21\x31\xEB\x3F\x3C\x65\x27\xFD\x5D\xF0\x24\xBC\x0A\x35\x39\x3B\x7D\x52\x15\x9D\xD9\xFC\xE8\xA4\xBF\x1B\xAB\x86\x0F\x63\x4B\x5E\x53\x0E\xB6\x58\x82\x59\x8C\xFD\xCE\x93\xFE\x9E\xB9\x1F\xC4\xF6\xBA\x16\x4C\xF9\xB3\x32\xE1\xE4\x44\x23\x14\x81\x8C\x0E\x0C\x35\xAC\x61\x04\x3C\x59\xCB\xFE\x74\xF0\x49\x00\x7B\x84\xB8\xE8\xA9\x36\xEC\x58\xB3\x37\xC2\x5D\x12\x3A\x38\xC4\x75\xD8\x9C\xA9\x15\x59\x48\x12\x74\x87\x2B\x41\xD3\x0A\x47\x2A\x18\x26\x43\x39\x79\xC5\x12\xF1\xB2\x5F\x07\x53\x25\xC8\x3F\x7F\x1F\x18\x41\x94\x83\xD3\xFF\x24\x30\xE7\x3D\x47\x2A\xE9\xA5\x53\xEB\xFE\xA5\x1B\x1B\xE6\x38\x05\xC1\xA3\xCC\xA5\x7E\xE3\x6E\x71\x25\x67\x12\x46\x90\x82\x6D\xB5\xB1\xFF\x48\x25\xFC\xDD\xA7\x6B\xED\x55\x18\xF2\xC6\x5D\x62\xB5\x52\x69\xAD\xE7\xDC\x85\x61\x81\x36\x0E\xD6\xDA\xC7\x99\xBD\x72\xA6\xDA\xAF\x53\xB5\xF7\xDD\x39\xA1\xAB\x18\x54\xEF\x35\x33\xF5\xEE\xA6\x7A\x7F\x8A\xF5\x6C\xEC\xF6\x6E\x71\xB8\x52\x7E\xE3\x56\x28\xCD\x6A\xF4\x60\x90\xFE\xFD\xC2\xA9\x93\x5E\x1E\xAF\x28\xC1\x9F\x53\x3E\xBB\xD6\xBF\x1D\x8F\xE6\xBE\x53\x8A\xA7\xAE\xF0\x7D\x27\xFF\x0C\x04\x29\x30\xB9\x70\x8C\x2D\xCB\x17\x4B\x99\x4F\xFC\x5B\x15\x59\xBB\xE8\xCE\xA3\xD6\xBC\x7A\xB2\x53\x6D\xA3\xF9\x98\x79\xD7\xB5\x8D\xF5\xF7\x60\x7B\x14\x5B\x48\x1C\x69\x72\x74\x2D\x34\xCE\xFA\x7B\xC3\x0B\xC7\x61\x7A\xE0\x03\xFF\x85\x58\x5D\xF9\x8D\x8D\xB7\xF3\x49\xB5\x70\xD9\x9A\x3F\x7D\xBC\xF5\xB2\xC5\xDC\x7C\x02\xFE\x18\xCA\x33\x4B\xD1\x1B\xF3\xA5\x46\xA1\xD1\x2A\xCA\x17\x7C\x85\x86\xF6\xD9\x39\x43\xE3\x81\xB9\xEC\x3C\xC6\x75\x09\xE8\x5A\xA6\xBB\x99\x07\x92\x4E\xF8\x1B\xDA\x05\xA5\x84\x2C\x3D\xDE\x2C\x20\x9D\xE7\xD3\x46\x99\x89\x3A\x3D\x5F\x31\x03\x42\x3F\xDD\x2E\x14\xA0\x93\x09\x69\x14\x25\xE0\x42\xA7\xCE\xEC\x1C\x4E\x9D\x9A\xDC\x12\xC6\x6F\xA0\xED\xFB\x96\x1C\x22\xF0\xA9\x25\x47\xCF\x22\xF5\xCC\xEC\x79\x65\x8E\xF2\xB2\xEB\xC8\x6E\xB3\x23\x6A\xDC\xCC\x6F\xFC\x68\x95\x8D\xF0\xA6\xD8\x7E\x21\xBD\xBC\x8E\xD2\xF0\x9E\x49\xC2\x10\x1B\x4C\xAB\xF8\xFB\xBF\xFD\xC9\x9B\x6F\x58\x0C\x77\x17\x9C\xC4\xCB\xB3\x21\x78\x20\xFA\xDC\x60\x8D\x9A\x56\xA5\x93\xE8\x77\xA3\x43\x21\x1E\xFE\xFC\xDC\xD9\x8D\xE2\x5A\x0C\x98\x74\x5D\xF8\x4D\x22\xA8\x7B\x54\xFD\x47\xDD\x7F\x34\xFD\x47\xDB\x7F\xCC\xFA\x8F\x39\x3E\xA2\x1B\x22\xFA\x99\x68\x5F\x60\xEC\xC0\x70\x0F\xAB\xA0\x34\x87\xEA\x8E\xE5\xC9\x29\x4C\xC5\x99\xDC\x9C\x09\xFB\xE4\x4E\xDD\xF1\xC4\x4A\x24\xB7\x68\x90\xF7\xA7\x70\xB5\x2E\xFB\xFE\x4A\x85\xD3\x5C\xBE\x2F\x43\x20\x1E\x85\x5B\xC0\xD3\x93\x67\xD7\x48\x50\x96\xE2\x25\x60\x27\xCB\xFB\x52\xF7\x07\xF4\x14\x03\x53\x95\xD2\xF3\x4F\x6A\x85\x4D\xC1\xF0\x28\x5D\x9E\x4A\x3C\xE8\x3A\x22\x26\x0D\xD7\xA2\x76\xAB\x62\xEE\xFB\x5E\x0D\x4D\x35\x34\x58\x69\x18\xF4\x51\x22\x8B\xE6\x33\xAB\xE9\x4B\x56\x06\x4D\x90\x60\x83\x10\x95\x60\xB2\xAF\x35\x0E\x42\xFD\x3B\x67\xBA\x80\xD4\x6F\x4D\x7E\xFF\x7F\xC9\xEF\x3B\x93\xDF\x77\xB1\xAD\x71\x5F\x77\x45\x34\x5C\x26\x7A\x0C\xC5\x53\x7A\xC3\x99\xC4\xFD\xD6\x90\x2F\x49\x71\x6C\xC6\x5B\x9B\xA2\xBC\xD2\x7D\x23\xBA\x23\x46\x58\x15\xEC\xEC\xF9\x40\xB4\x66\x62\x6B\xAB\x62\x3A\xA8\x0E\x26\x3A\x93\x18\xE4\x06\x5D\x10\xB9\x7F\x8C\x09\x44\x8D\xFD\x0C\x4F\xF5\x25\xCF\x48\x52\x66\x95\x9F\x50\x9A\x4E\x8B\xB6\x0A\x49\x90\x72\x9B\x9F\xC6\xB4\xF8\x4A\x53\x96\x7D\x30\x92\xBB\xFB\x89\x5D\xDA\x35\x85\x41\x60\x8A\x96\x16\x60\xDA\xA5\xC0\xA4\x2F\xE4\x7D\xC2\x7B\xC2\x23\xCA\x7E\x89\x03\xAC\x34\x5D\x03\x94\x5C\xE5\x12\x2D\x1A\xB3\x2C\x7E\x10\xA3\x72\x89\xEF\x55\xC5\x7F\xAC\xD8\x9D\xC7\xDC\xCE\x17\x2A\x31\xA7\x47\x84\x2D\x87\xDC\xB4\x33\x90\x95\xC1\x71\x20\x42\x36\xBA\xC0\x0B\x0E\x38\x46\x47\x0F\x78\x17\xE6\x5C\xAD\x89\xCD\x5A\x43\x24\x85\x55\x3D\xBB\x85\xFE\x46\xA5\xD4\x44\xCD\xBA\xEB\x4C\xB3\xF5\x80\x55\x8D\x92\x04\xB1\xBA\x97\x0F\xD1\x3A\xA4\xA2\xF3\x0E\x81\x4D\x95\x5B\x61\x41\x4C\x61\x61\xA4\x4B\x86\x7E\x6C\x69\x1C\x5B\xC2\x87\x1A\x3D\xA3\x34\xBA\x04\x35\x06\x46\x20\x1A\x4B\x58\xB0\x4F\x40\x88\xA7\xED\x95\xC1\x2B\x08\x2F\x9D\xD3\x2F\xB9\xC9\x8D\x67\x9D\xDC\x12\xFE\xB3\x67\x6D\x1D\xB5\x22\x71\xBD\x79\x0C\xB9\x5E\x77\xE7\x08\x32\x46\x9A\x35\xF1\xFE\xC5\xB3\xCE\xF0\xB5\x4D\x62\x51\x14\x51\x90\xA2\xA8\xAD\x92\xDD\x71\x84\x16\xF7\x07\x64\x66\xF8\x80\x54\xE3\x8D\xC1\xE0\x3B\x17\x7C\x20\xB0\xC1\xC6\x2C\xD1\x9D\x0D\x7B\x07\xA5\xB7\xF4\xA6\xB6\xCB\xB2\x2E\x9C\xAE\x07\x98\x0B\x01\xB8\x96\x53\xAB\x4B\x8B\x25\xB2\xE9\x21\x39\x97\xBA\x62\x79\x72\xAA\x19\xB9\x41\xC7\xAF\xD1\xF1\xB0\x46\x57\xBE\x91\x93\xDF\xAB\x8A\x27\xF2\x01\x08\xBF\xC1\x64\x50\xA5\x1B\x12\x2D\x35\xC3\x4A\xB8\xC2\x65\x8F\xAF\x73\x97\xB9\xE2\xFB\xD1\xF6\x41\xC7\xC4\x02\x58\x7B\x41\x3C\x1D\x86\x50\xBA\x61\x37\x0A\x97\x7B\x79\x7C\xA9\x7C\xD7\xC8\xEC\x9C\x28\x9C\x9C\x13\xE3\xB7\x07\x96\x83\x8E\xA0\x6F\xA7\x9D\x37\x14\xEE\xE4\xE7\x6F\x01\xA5\xAB\x4B\x4D\x96\x1C\x9A\xE3\xD5\x68\xAF\xD7\x9B\xA2\x45\x61\x43\xEE\x91\xA0\x1F\xD0\xF0\x06\x40\x25\xB1\xB0\x19\xF2\x6A\x19\xC0\xF4\xBD\xA8\x47\x81\xBF\x0D\xFA\xEB\x66\xC0\xBC\xC4\x0D\x29\xFC\xF8\x3E\x85\xBE\x4E\x18\xE7\x6F\x50\xE1\xB5\xF0\x11\x7A\xA3\xF2\xCB\xA6\x72\xC3\x43\xA8\xD2\x56\x24\x6D\x16\xBC\xA9\x17\x99\xCC\x2B\x90\x25\xE3\x54\x96\x2C\xC2\x02\x19\xD7\x8B\x41\xDA\x0C\x67\x6A\x2C\x50\x8D\x85\xD2\x2D\xB8\xC5\xC4\x43\x36\xAE\x9D\x45\x14\x30\x65\x46\x12\xA6\x74\x95\x1B\x92\x4C\xA9\x77\x8C\xB2\xD2\x55\xE8\x6E\xC4\xFF\x31\xCD\x22\x14\x98\x66\xC1\x2D\xBA\x85\x40\x7D\x55\x46\x85\x15\x57\x59\x85\x02\xB7\x78\xA4\xDE\x31\xB2\x74\xD8\x07\xDF\x37\x3B\xB1\xAD\xE6\x02\xB7\xD3\x5D\xB0\xDA\xEC\x58\x6A\x16\x81\xCE\x16\xEF\xA8\x77\x79\x59\x8F\x96\x65\xBD\x1B\xA6\xEA\x2E\xA8\x2F\x84\x21\x5F\xE0\x76\x1E\xAD\xEC\x28\x23\x2A\x5B\xF0\xA2\x1E\xAB\x83\x6E\x37\x50\xD9\x1E\x77\x21\x52\x19\x74\xEA\x16\x91\x96\xC6\x40\x4B\x7B\xDC\x90\xA9\x6C\xD8\xBD\x59\x80\x37\x63\xB7\xC0\x68\x5C\xA8\x84\xDB\xED\x76\x3D\xBE\x1E\xB9\x5D\x6E\x37\x50\xD9\x22\x50\xD9\xA2\xDB\x0D\x54\xB6\x1B\x87\x88\x03\x28\xDD\x82\x9F\xB8\x1D\x30\x8E\xD5\x25\x37\x02\x2A\xAB\x77\x8C\x0C\xC1\xE7\x4E\x86\x8F\xA6\xC7\xBB\xF8\x51\xF5\xA4\xEA\x08\x80\x09\xD2\x7A\x08\xD2\xBA\xDE\x51\xBA\x1D\x2B\x95\x2A\x67\x70\x0F\x7F\xDC\x60\x05\x4C\xE0\x01\x70\xE3\x51\x99\x50\x5F\x47\x68\xC1\xA9\x75\x80\xAB\xF6\x7C\x09\x0C\x98\x7C\x8F\xB4\xC8\x48\xFB\xDA\x13\x57\xF1\xE5\x12\x57\xB1\x05\x71\x15\x7D\xE2\xDA\xF1\xD5\x26\xAE\x1D\xE7\x45\x5C\x05\x13\xD7\x0E\x24\xAE\xA2\x4F\x5C\x45\x9F\xB8\x8A\xAD\x89\xAB\xC0\x3C\xA0\x32\x12\x97\xA4\x68\xA2\x1D\x51\x95\xA5\x1B\xB9\x32\xE1\x65\x39\x58\x1F\x39\x07\x44\x45\xEF\x40\xE3\x2C\xA8\xBC\x13\x0F\xE6\x9B\x3D\xB2\x44\x07\xA5\x08\x7B\xBA\x4E\xAE\xD8\x61\xBD\x8A\x61\xBB\xB9\x10\x3D\xD4\x7D\x51\x57\x5E\xD6\xC3\xC0\x4C\xAB\x36\x56\x73\x15\x80\xA1\x72\x43\xE8\x73\x88\x9F\xB0\x8B\x10\x0C\xAD\x2A\x5D\x45\xF5\x4A\xA7\x62\xBB\x3A\xE9\x58\xCF\xEB\x58\xCF\xE9\x58\x6F\xDD\xB1\x8E\x1D\xEB\xE9\x8E\x35\x7F\x30\x28\x6F\xDF\x95\x95\x89\x33\x91\xC3\x70\x8D\xCA\x8F\x39\x50\xA7\x76\x8A\xE2\x24\x59\x3C\x30\x5A\x16\x97\x71\xBE\x4F\x45\xEE\x27\x74\x22\xAD\xBC\x5C\xC7\x70\x2D\xDE\xAC\xF3\xB1\x95\xE9\x26\x82\x37\x0D\x42\xF2\x61\xD5\x06\xAF\xF8\x4B\x94\xB8\x4C\xC5\x98\x10\xC1\x1D\x3E\x8C\x39\xC0\x84\x5C\xFA\xE2\xB8\x5D\x48\x7A\x9C\xDE\x6C\x9C\x77\x5F\x3E\xE7\xBB\x66\x40\xB7\x05\x8F\xA5\x19\xE0\x19\x32\x3A\x70\xF3\xA8\xFD\xD0\xA9\xA5\x46\xD1\xE0\x75\x32\x78\xFE\x84\xEE\xB2\x85\xEB\x12\xDB\x98\x44\x33\xD8\x74\x1A\xF8\xBE\x1E\xC4\xD9\x38\x8D\x77\xDE\x0E\xA8\xC2\xEF\x0D\xD1\xF9\xDD\xE0\x8E\xCB\x74\xE1\x94\x3F\xAB\x1E\x2E\xF9\x96\x9C\x71\x03\x50\x88\x1B\xEB\x8A\x03\xA8\xB2\xE2\xF6\x76\x0E\xAF\xF2\xD4\xD2\xE6\x7B\x53\x2A\x19\x00\x93\x0E\x91\xCD\x90\x6E\x42\x83\x5E\xD1\x36\x19\x28\x01\x39\x2A\x01\x50\xA2\xD7\x1B\x85\x8C\x78\x40\x8B\x87\xEE\xBC\xD0\x17\x83\x8A\x07\x62\x9D\x6A\x31\x6A\x33\x7A\xAF\x52\x22\xE7\x21\x82\x46\x39\xDC\xD4\xA4\x2C\x24\x4E\xF9\x63\x14\x87\xFA\x06\xBE\x02\x81\x77\xC1\x29\x0E\x0D\x83\xC5\x59\x98\x92\x53\x2E\x6B\x2C\x6B\xFB\x14\xDD\xD7\x86\xC1\xB5\x54\x6F\x70\x08\x8C\xE7\x95\x4A\x7A\x83\x5A\x5A\x30\x49\x15\x70\xE9\x62\xC6\x24\x2D\x3A\x93\x74\x30\x53\x23\xA7\x1A\x39\x98\xA4\xF9\x2A\x3A\x15\x17\xE8\x53\x66\x56\xA7\x99\x3A\x1B\xA4\x83\xC8\xD2\x95\x1B\xA4\x2C\x1D\x1D\x7D\x07\xFC\x1F\x43\x1A\x97\x69\x72\x6C\x38\x68\xAB\x45\x65\xA9\x58\x71\xA5\xD5\x2A\x07\x48\x25\x4C\x1D\xF9\x18\xB5\xD6\x8C\x5C\xE9\x46\xC8\xD4\xF1\x36\x88\xB9\x03\xD7\xFB\xC2\xB2\xAC\x17\x61\xDE\x6E\x54\x8F\x61\xFC\x23\x57\x02\x53\xCF\x89\xA9\xE7\x14\xD1\xD6\x2D\x02\x53\xDF\xE1\xC6\xC8\xD4\x55\xBC\xCF\x43\x67\x59\x3B\xDC\x80\x99\xFA\xA0\x7B\x43\x99\xB9\x5C\xCE\xCC\x32\x07\x36\xEE\xAA\xC7\xD7\x0B\x20\x77\xBE\x1F\x53\xD1\xE1\x7E\xC6\x22\x30\xF5\x45\xBA\xFB\x39\x46\xA6\x9E\x23\x53\x1F\xB9\x72\x75\xC9\x2D\x74\x4C\x1D\x20\x94\x30\x75\x78\xDC\x84\xA9\xC3\x5B\x60\xEA\x83\xC8\xD4\x83\x0F\x13\xD3\x3B\x11\xC7\x00\x28\x62\x86\xA7\xC9\x3E\xA3\xB2\xC4\xA8\x2C\x31\x2A\x3B\xCD\xA8\xD8\x35\x7B\x0E\xA3\xC2\x68\x6C\x3D\x46\x35\x40\xDF\x44\xD5\x52\x54\x9F\xE9\x35\xCE\xD7\x53\x6D\xC2\xA8\xEC\x2C\xA3\xB2\x2E\x67\x46\x25\x13\x46\x25\xBA\xCB\x3A\x22\x30\x2A\x4E\x68\x5D\xE0\xE5\x5D\x1E\x4B\x83\xB1\x27\x98\x51\xD9\x2D\x18\x55\x4E\xD7\xB3\x03\xA3\x32\x1D\xA3\x3A\xEF\x49\x34\x66\xD3\x69\xE0\x7B\xF6\x81\xE7\x0C\x40\x66\x86\x51\x99\x1E\xA3\xB2\x4C\xFE\xC8\xA8\x06\x2E\x3B\x80\x2A\x1C\x32\xAA\x02\x5E\x15\xF1\xEE\x90\xA2\x9B\x97\x8D\xE8\x18\x15\x0E\x47\x05\x56\xB5\xA0\x0F\xBA\x05\x54\x04\x73\xD0\x1F\xDA\x46\x74\xF6\x0A\x94\x44\x56\x65\x52\x56\x45\x5F\x18\x8E\xF9\xD0\x0C\x22\xAB\x1A\x90\x86\x59\xE7\x50\x05\x59\x15\xA6\xC6\x1B\x00\xAB\x1A\x10\xAB\xCA\x03\xAB\xCA\x51\x82\x44\x56\x65\x79\xF1\x32\xAB\x12\xCD\xA0\xC7\xAA\x06\x61\x70\xCC\xAA\x4C\xC2\xAA\x0A\x60\x57\x29\xAB\xCA\x66\x18\x51\x86\x4B\x90\x98\xCD\x4C\x8D\x82\x6A\x14\xA5\xCB\x5D\x41\xAC\x0A\x88\xA4\xC0\x5B\xB1\x53\xAC\x2A\x9F\x61\x55\x86\x59\x55\x16\x59\x55\xBC\x18\xD7\xE4\xC4\x93\x0A\x6C\x38\xB0\xAA\xAC\x63\x55\x54\x89\x58\x55\x7E\xA4\xCE\x02\xAB\x32\xB3\xAC\x2A\x5B\x6A\x28\x15\x16\xB1\x2A\xC9\xAC\x2A\x77\x23\xBA\x97\xD9\x67\x55\x05\xEB\x9F\xCC\xAA\x86\x1D\xAB\xCA\x3B\xFD\x73\x87\x33\xCC\xAA\x4C\xF7\xA6\x20\xFD\xB3\x60\x56\x55\x04\x56\x25\x03\xAB\xE2\x28\x5D\x29\xAB\x1A\x22\xAB\x2A\xFC\xC4\x65\xCC\xAA\x24\xB2\xAA\x8C\x58\x95\x61\x56\x95\x11\xAB\x32\xCC\xAA\xB2\xCD\x58\x95\x41\x56\x95\x81\x90\x65\x56\xC5\x14\x4F\xC4\x81\x14\xB1\x00\x64\xBD\xC0\xD4\xBC\x9A\x44\xE0\x47\xBD\x20\x8F\xFC\x88\x98\x93\xCB\xBD\x38\xB2\x58\x3A\x75\x47\xBD\xC3\x9B\x7A\x00\xC0\x43\x38\x29\xDE\x15\x46\x38\x55\x2E\x47\x38\xA1\xDA\x31\xE0\xAD\x61\x30\x63\xDC\x8E\x27\x32\xAD\x73\xB1\x42\x77\x18\x06\x91\xA9\x14\x2D\x62\xE5\x16\xB9\xFB\x3B\x9E\x58\x49\x37\x00\x40\x0D\xDC\xA2\x53\x77\x34\x3B\x9E\xD8\x71\x77\x90\x38\x65\x89\x07\xEB\x39\xAE\x4B\xF8\x03\xA3\x33\xE5\x2D\x4A\xCB\x89\x3C\x95\xDE\x8D\x92\xD3\x21\xF4\x7A\x7B\x95\xE2\x12\x5D\xBC\x99\x2F\x9F\x4E\x6D\x55\xFA\xA2\x1D\xDF\x42\x5B\x6A\xC2\xBF\x38\x0D\xDE\x27\xBF\x5D\x17\xE4\x8A\xF4\xD6\x6E\xF3\xA9\xBB\x8C\x3C\xE5\xD4\xF3\xD6\x46\xDF\xCE\x1E\x1E\x2F\x7D\xCB\xEB\xF6\x85\xCD\x78\xE1\x8B\x93\x50\xD8\x3D\x8F\xA7\x9E\xF7\x9E\x0C\xBF\xF5\xB2\xBB\xF9\xF6\xF4\xA5\x5E\xDE\x37\x5D\xF0\xB0\xE9\x82\xEF\xBB\xF9\xF6\xEB\xCA\xCF\x18\xDC\x58\x9D\x13\x53\xF0\x5C\x1B\xAB\x21\xCC\x18\x4A\x1D\x72\xC2\xC2\x9D\xC4\xD1\x57\x77\xCF\x75\x82\xBB\x7D\xCB\xF2\x54\xB3\xC9\xC6\xAB\xE0\x64\xCA\xC9\x78\x1B\x1B\x46\x4C\x87\x2C\x78\x9C\x86\x7E\x03\xE4\x98\x09\xAC\xD0\xAE\x54\x06\x0F\x23\xE6\xEF\xB1\x36\x39\x88\x3E\x54\x17\xF1\xBA\x69\x46\xFB\x5E\x59\x6F\x47\xBC\xE4\x77\xB1\x5B\xB2\xE0\xE4\x63\x78\x8C\x35\xC5\xFE\x91\x41\x8A\x66\x78\xEE\x8E\x6E\x22\xF6\x50\xA5\xB1\xFB\x8C\x66\x20\x67\x67\xA0\xCE\x3D\x03\xBD\xAD\x19\x98\xED\xCC\x40\x9E\xC7\x0C\x54\x32\x83\xA9\x0D\xE6\x47\x92\x67\xB4\x0C\x64\x75\x2A\xB8\x12\xA1\x5B\x79\xE7\xC6\xA0\xD6\xFD\xEB\xFF\x8E\xCF\x30\x11\x99\x74\xD8\xF8\x7F\xE4\xD5\x44\x9F\x1E\x4E\xE4\x69\x3A\x16\x20\x8B\x4F\x46\x45\x4A\x39\x49\x4A\x06\x2A\x52\xAA\x53\xA4\x24\x47\xBA\xC4\xFB\x95\x92\x15\x29\xD9\x29\x52\xBA\xB3\x99\xC3\xDD\x0F\x74\x54\x94\x2D\x71\x77\x19\x74\x10\x22\xE1\x70\x1B\x4C\x25\x8A\x14\x8A\x1E\x4A\x18\x16\x4D\x64\xCC\x2A\x13\x2D\x3E\xC9\x97\x9F\xE6\xC4\xBA\xE3\x64\x6B\x08\xB3\x2E\xC6\x09\x79\xDE\xE3\xE6\x30\x8D\xDA\x0F\x9D\x5C\x6A\x24\x0D\x5E\xAF\xF7\x2E\x99\xAA\xA0\x48\x15\x7C\xEB\x72\x3B\x93\x68\x36\x9F\x46\x8C\x98\xC2\xB3\x71\x0A\xEF\x4D\xA7\x8A\x94\x74\xC5\x1D\x94\x9F\x8B\x14\x29\x45\xBB\xDE\x05\x2A\x52\x06\xF5\x2F\x67\x28\xFF\x08\x86\x59\xC9\xA2\xC5\x27\xE3\x5D\x72\x99\x0C\x20\x5A\x7C\xA0\xEB\x0C\xEA\x21\x06\x9A\x40\x4D\xA5\x04\xCD\xC9\xD0\xB6\x2F\x8A\xAD\x46\xA2\xE2\x32\x4A\xD5\x28\xFA\x62\xC4\x92\xB9\x29\x9C\x6C\x69\x7D\x14\x68\x36\x80\x1D\x32\x40\x63\x52\x92\xC5\x07\xB6\x64\x3D\x74\x92\x2C\x3E\x09\x6A\x94\x9C\xB6\xF8\x08\x2C\x68\x64\xB6\x4E\xBA\xB2\x19\x26\x6A\x94\x24\x73\xB4\x0C\x6A\x94\x70\x23\x8C\x0F\x41\x6A\x94\x9C\x3E\x84\x04\x7B\xAE\xD1\x9B\xDA\x7C\x23\xA8\x93\x43\x9D\x2C\xD6\x41\xF7\x86\xBC\xC6\xF0\x18\x72\x75\xA6\xD9\xAC\xDF\x58\xDE\x35\xA6\xFB\x4D\x00\xC4\xBD\xAC\x2B\xC5\x8E\xFD\x53\x8A\x18\xE8\x1A\xED\xC2\x60\x60\x51\x0D\x93\x36\x2F\x9D\x70\x23\x56\xC4\xAA\x91\xE5\x13\x9E\x11\xFF\x87\xB3\x62\xE2\xE5\x72\x67\xA2\x22\x56\x55\x7C\xA7\x47\x52\x25\xB4\x19\x25\xD8\x8C\xD5\x88\x06\x86\x2D\x90\x1B\x6C\xD1\xE4\xA0\xA2\xAF\x36\xD5\x12\x93\xDD\x1D\xF5\x02\x68\x8A\xAC\x88\x49\x97\xD3\xBA\xCD\x9D\x3D\x5A\x99\xA0\x88\x19\x4F\x5B\x23\xA8\x60\x8C\x9D\xEE\xCE\x32\x64\xAA\x6E\x8D\x58\x11\x1B\x75\x6F\x70\x3D\x16\xCE\x44\x2D\x03\x14\xB1\x85\xC7\xD7\x99\x5B\x20\x45\x8C\x57\x07\x6B\x15\x74\x07\x14\x15\x31\xE3\x27\xAE\x82\x71\xAC\x2E\xB9\x0C\x15\xB1\x6A\xA4\x09\x42\x77\x32\x84\xE8\xBE\x48\xCA\x92\x0A\x62\x49\x05\xB1\xA4\x62\x9A\x25\xE1\xC5\x98\x62\x1E\x4B\x42\x55\x79\x86\x25\x65\xB0\x9A\x8B\x2D\x56\x73\x91\xB0\xA4\x62\x96\x25\x15\x68\xD6\x4C\xB3\xA4\x1D\x1D\x4B\xDA\x31\x6F\x13\xCA\xF2\x58\x60\x51\x47\x96\x54\x6C\xC1\x92\x32\xA2\xA6\xC0\x92\x74\x9F\x25\x9D\xD7\x24\x1A\xBD\xE9\x34\xF0\x3D\xC8\x16\x9E\x0D\x26\x1F\x9F\x66\x49\xBA\xC7\x92\x0A\x4E\x00\xC6\x2C\xC9\x26\x2C\x69\x6A\x13\x4A\x96\x6E\x07\x8C\x62\x47\xC7\x92\x8A\x84\x25\xED\xD0\x07\xDD\x0E\xF6\x78\xD8\x11\x2D\x3B\xDA\x84\x82\x92\xC8\x92\x74\xCA\x92\xE8\x0B\x5D\xF1\x40\x4C\x64\x49\x86\x37\xA1\x32\xA8\x42\xA1\x86\x28\x23\xD4\x84\x78\xD1\xB1\xB6\xCE\x02\x4B\xCA\x58\x26\x33\x4B\x22\xB0\x90\xB9\xEA\xA4\x13\x8D\xE9\xB1\x24\xD3\xB3\xEC\x42\x02\xB6\x6E\x13\x2A\xDB\xD4\xEB\x21\x23\xAF\x87\x2C\xF2\x90\x99\x1A\xB4\x23\x51\xE7\xA5\xCB\xC2\x26\x14\x66\x10\x76\xD9\xAC\x65\x97\x4D\x59\x76\xD2\x69\x66\x28\x16\x2C\x3B\xC9\x01\x3B\x28\x6A\x47\xB8\x17\x9F\x63\xC3\x81\xA1\xD8\xCA\xF2\x39\x29\x57\x22\x86\x92\x1D\xA9\x6D\xB0\xEC\x34\x9D\x2C\xC8\xE4\x64\xC1\xD2\xA9\xA1\xCB\xEE\xA8\xC7\xCB\xA2\x5E\xF4\xB2\xDE\xE5\x2E\xA0\xD9\xF1\xA9\x42\x5E\xD2\x1D\x27\x3A\x00\x71\xD9\x1D\xF1\xDC\x73\xF1\xF6\x26\x0F\x27\xF8\x19\xD8\x1F\x18\xFF\x53\xBA\x6C\xA5\x52\x5E\xD6\xBB\xC1\x02\xA3\x52\x30\x49\x16\x6F\xC7\x4D\xAC\x0C\xEF\xB8\x85\xE3\x49\x84\x7B\x4E\xAE\x29\xBB\x5D\xBE\x82\x47\x85\x19\x9E\xFE\xEE\x1E\xE9\xD2\x9B\xFA\x42\x6F\xEA\x3D\x01\x4C\x00\xE4\xBD\x29\x90\xF7\xC0\x40\xF7\xD6\x7B\x62\x0E\xB4\x99\x1A\x17\x52\x8D\x0B\xE9\xD0\xE3\xC2\xA3\xB1\x6D\x77\xA1\xDB\xB3\x5A\xCD\x26\xB7\xDF\x83\xB8\x28\x32\x44\x85\xE8\x5B\x89\x84\x4C\x46\xCD\x6E\xE0\xF5\x19\x5E\x30\xE6\xFF\x98\x66\x0F\x06\xD1\x69\x2E\x84\xAE\x22\x6A\x76\x57\x39\x87\x4A\x6B\x32\x5E\x49\x61\x18\x05\x36\x78\x04\x61\x45\x13\x00\xE4\x3C\x88\x54\x80\x8B\xDC\x83\xDC\x45\xAB\xCD\xEE\xA5\x66\x0F\xE0\x68\xCF\x1D\xF5\x92\x97\xF5\xC5\x00\x90\x65\x59\x2F\xB8\x8B\xEA\x07\xC3\xE4\x2E\x72\x0F\x3A\x5A\x19\x6C\x8E\x19\xFF\x85\x5E\xD4\x7B\xD5\x41\xB7\x00\x8C\xFF\x5B\xDC\x83\x11\x65\x18\xDD\x67\x0F\xB2\xF7\xBD\x00\x8C\x6F\x71\x39\x33\xFE\xBC\x7B\x83\x60\xDA\xEB\x2E\x64\xC6\x7F\x61\x25\xDC\x82\x5B\x7A\x7C\x7D\xB1\x5B\x72\x0B\xC0\xF8\x17\x80\xE3\x2F\xB8\x3D\xC0\x8A\xF6\x50\x48\xB7\x07\x23\xE3\xBF\xD0\x4F\xDC\x6E\x18\xCB\xEA\x92\xBB\x18\x19\x3F\x02\x19\xC0\x75\x27\x83\x4B\xD1\xE3\x5D\xFC\x28\x4B\x40\xA8\x04\x9C\xC9\xB5\x7A\x77\xE9\x76\x57\xAA\x44\x91\xD2\x2C\xBA\xF1\xE3\xEB\x5D\x48\x6F\xE3\x1F\x40\xEE\x88\x5F\xE3\xBE\x12\x79\x97\xED\xF6\x13\x67\xF9\xC0\x69\x17\x76\x67\x31\x25\xA1\xD3\xDC\x9D\x45\x63\xCB\x69\xEE\xCE\xCE\x31\xF8\xC1\xF0\x82\x85\x2B\xD7\x6A\x0B\x3A\x16\x1B\xFC\xCC\x06\x89\x67\x60\xCA\x43\xE6\x75\x51\x6C\x80\xA8\x4C\xE5\x97\x21\xF9\x65\x48\x7E\x99\x69\xF9\xC5\x21\xBB\xE6\xC8\xAF\x9C\xF7\x5B\x3A\xF9\xA5\x71\x5B\x8F\x7D\xF9\x66\x58\x3F\xFB\xD8\x99\x44\x7E\x99\x59\xF9\x65\xE2\xDE\xE4\x68\x46\x7E\x8D\x7A\xF2\x2B\x0D\x3A\x2C\xC2\xDE\x4F\x06\x7F\x58\x7E\x99\x2D\xE4\x57\x1E\xB7\x8B\xA2\x94\xDE\xEE\x24\x38\x48\xC6\xBC\x69\x50\x54\xCC\x2C\xCE\xC6\x19\x3C\xF7\xE8\xCB\xAF\xAC\x27\xBF\x78\xDB\x0B\x0F\x15\x1A\x4D\xD9\x27\x35\xC9\xAF\x2E\xAE\xD1\x68\x13\xF9\x85\xC3\x91\x41\x82\xED\xD1\x07\xDD\x1E\xB4\x78\x72\x64\x08\x0D\x0A\x2D\xCB\x7B\x93\x7B\x3A\x09\x96\xA5\x12\x8C\xBE\xC8\x78\x53\xAB\xD1\x51\x82\x69\xF4\x11\x31\x75\x0E\x55\x50\x82\xE5\x14\xDB\x66\xD2\x62\x88\x30\xDC\x9B\x64\x09\x36\xB5\x37\x69\x98\xA7\xB3\x04\xDB\xD1\xE8\x9E\x04\xD3\x61\x70\x2C\xC1\xB2\x44\x82\xD9\x64\x6F\x52\x4E\xBB\xA7\x12\x55\x8A\x6E\x6F\x32\x9B\xA9\x61\xA9\x86\x45\xBD\x92\x24\x18\xDE\x3E\x39\x9F\xBD\x49\xE9\x32\x66\x93\x22\x4A\xB0\x8C\xFF\x03\x48\xA2\x8B\xE3\xA4\x28\x32\x9B\x14\x9D\x04\xCB\xE2\xDE\xA4\x04\xEE\x28\x82\x04\x43\xB6\x78\x31\x49\xB0\x9D\xEE\x62\xB7\x73\xB5\x11\xDD\xDE\x64\x90\x60\x17\xBA\x9D\x34\xBB\x9D\xEE\xE2\x39\x12\x2C\xEF\x49\xB0\x70\x2A\x55\xE7\x20\xDC\x58\x82\xE5\x24\xC1\xF6\x92\x04\xCB\x3A\x09\x66\x39\x21\x3E\x71\x53\x1B\xF3\x14\x90\x04\xDB\xEB\x2C\x49\xB0\x1C\xD9\xFB\x5E\x92\x60\xBB\xBD\x01\x5E\xD6\xC9\xA7\x07\xA7\x40\xDE\x05\x03\x7D\x70\xBD\x2B\xA0\xC1\xCE\xD4\xD8\x4D\x35\x76\x03\x95\xEE\x72\xBB\x8F\xC6\xB6\xDD\x6E\xB7\x6B\x9E\x04\xDB\xB5\xB5\x04\xCB\x9D\x65\xD4\xEC\x05\x09\x86\xC2\xCB\xF2\x7F\x4C\xB3\x8B\x24\xD4\x6E\xE8\x2A\xA2\x66\x2F\x4A\x30\x12\x5D\xBC\x96\xC2\x30\x70\xE7\xDA\x1E\x41\x58\xC5\xC3\x8A\xE6\x21\x41\x90\x3D\xC4\x3D\x68\xB5\xD9\xBB\xD4\xEC\x02\x1C\xED\x22\x09\x76\x11\x00\x04\x25\xD8\x83\xEA\x0B\x60\x72\x0F\x72\x0F\x21\x09\xB6\x37\x48\xB0\xDD\x5E\xD4\x0F\xEE\x24\xD8\x05\x77\x44\x39\xB5\x0B\xE5\xD4\x83\x49\x82\x59\x96\x60\xB6\x7B\x83\x60\x7A\xB0\xDB\xCD\x12\x6C\x77\x90\x60\x17\x4D\x49\xB0\x5D\xC0\x8C\x70\xB2\x05\x0E\x82\x44\xCA\x5E\x18\xCB\xEA\x92\xBB\x08\x45\x0A\x02\x19\xC0\x75\x27\x83\x4B\xD1\xE3\x5D\xFC\x08\x16\x17\xC8\x10\x8B\x32\x64\x6F\xE9\xF6\xF6\x25\xD8\x85\xA9\x04\xDB\x1B\x25\x58\x8E\x12\x6C\xAF\x9F\x38\x01\x24\xBA\xBA\xE4\x2E\xC4\xEE\x04\x49\xB0\x8C\xBB\x23\x19\xE5\x32\xEE\x4E\x6C\x26\xC1\x32\xEC\x1D\xF3\x41\x86\x08\x01\x84\x28\xE2\x19\x78\x44\xB3\x07\xB8\xDD\x1E\x66\x72\xAB\x41\xCC\x15\xB0\x74\x0C\xB1\x0B\x81\xC6\x32\xF9\x2A\x67\xC8\xF7\x30\xE3\x7B\xD1\xB9\x2A\xA3\xE6\x66\xE2\x33\x03\xB8\xAA\x14\xB3\x6A\x8A\x92\xD9\x36\x92\x9B\x3E\x8A\x57\xBE\xD0\x81\x59\x3A\x8A\x65\x00\x66\x35\x2C\x9E\x23\x95\xC4\x3C\xC6\x30\xFC\x11\x0E\xBF\x2A\x5D\x05\xC3\x97\x18\xDD\xE9\xBE\x17\xF4\x23\xBA\x00\x2F\xA5\x6D\x85\x11\x1E\x38\xD1\xE9\x56\xE2\x78\x51\xFE\xAA\x96\x66\x32\xBD\x12\xC2\x0E\x1A\xBA\x1C\x93\xCF\x71\xC1\x8E\xE9\xEA\x70\x95\xC1\xAF\xF1\x4A\x95\xB1\x4D\x83\xA6\xFB\x7E\x21\x57\x2A\x1B\x4A\x42\xD4\x1B\xB9\x42\xE6\x3A\x6E\xE7\xCA\xCE\x1F\x43\xB0\xD1\xA3\x7C\x51\x4B\x0E\x64\xD8\x19\xD8\x21\x4B\x2B\x89\xBD\x7E\x9E\x90\xE0\xAD\x8A\xE2\xB9\x0B\x38\x31\xEB\x7B\xCF\x97\x1E\xF0\xA2\x81\xEB\x9F\x41\xCA\x10\x88\x89\xE2\x0E\x26\x63\x4A\x42\x30\xCB\xD5\xD4\x45\x5D\xA6\x51\x43\xEE\x4F\x67\xE6\x9C\x9D\x05\x2F\x5C\x59\x76\x1A\x09\x83\xA2\x2C\x6F\x5F\xC8\xF2\x89\x3C\x9D\xF1\xEE\xE4\xDC\x6C\x3D\x0E\x23\x4B\xF0\xC6\x2E\x87\xD5\xA3\xC8\xED\x61\xB7\xD7\x19\xBC\xF7\x56\x1C\x4E\xCB\xE4\x79\x55\xEC\x1C\x5F\x7C\x86\x1E\x2F\xDE\x05\x25\xEE\x80\x32\x78\x3F\xDA\x50\x82\x7E\xB0\x51\x28\x65\x74\xF8\x55\xC4\x5F\xA3\xF8\x6B\x1C\x7F\xED\x8A\xBF\xF6\x3A\xBD\x5C\x5C\x96\x54\xBF\x38\x6C\x25\x64\x89\x7B\xF8\x9C\x28\x93\x78\x77\xCA\xC4\x08\xF2\xB8\xB7\x8C\x8A\x12\x29\x14\x14\x43\x9E\x27\x86\xB6\xF4\xF8\x0F\x42\x3D\xDC\x40\xC8\xCA\x50\xBD\x4B\x4B\xD1\xBF\x04\x92\xC5\xBD\xC4\x2C\x09\x86\xDD\x45\x5B\x0A\xA7\xD7\xF9\xBC\xD3\x6B\x33\xBB\xB2\xF8\xC8\x03\xD4\x33\x3E\xD7\xD0\x9D\x67\x45\xEF\x5C\x83\xBC\x0C\x66\xCF\x35\x32\x3A\xD7\xC8\xE2\xC9\xE5\xBC\x73\x8D\x50\xE5\x12\x2D\x28\x10\x90\x72\x83\xFE\x91\xC6\x20\xFA\x92\x67\xC0\x67\xF9\x4E\x45\xCD\xF1\x43\xB3\xBE\x17\x6B\xB6\x42\x52\xD3\x99\xB8\xA3\xBC\x52\xA9\xB8\xEF\x3A\xB2\xA8\x79\xE1\x56\x82\xC5\xD3\x5E\x4E\x33\xE0\x4D\x3D\xF4\xA6\xEE\x89\xEA\x51\x2A\xAA\x4B\x3C\x0D\xAD\xCB\xDE\x79\x6F\xAF\x06\x9D\x97\xD6\xC3\xD2\x0D\x5D\x39\xC7\xC5\x90\xA3\xCE\x2E\xE4\x25\xED\x2F\xE6\x78\x28\x4F\xF2\xBA\x2A\x46\xC3\x92\x44\x75\x77\xCC\xCB\x07\x94\x43\x57\xBA\x61\x74\x32\x0C\x42\x9A\xAB\x80\x22\x65\x5D\x79\xA4\xCA\x47\x03\x26\x89\x8A\xE4\xF4\x82\xAB\xDC\xC2\x6A\xB3\xB8\xD4\x94\xC0\xDA\xCA\x3B\xEA\xB1\x97\x78\x4E\x59\x2E\xCB\x7A\xE0\x16\xEA\x9D\xD1\x2D\x7A\xA4\x0E\xBA\x01\xC8\xE3\x0B\xDC\xCE\x3B\xA2\xD4\x2D\x51\x36\x8C\x80\x26\x2F\xE8\x9F\xE9\x96\x9D\x5B\xF4\xA8\xE7\x16\x3D\x00\xD9\xB8\xC3\x8D\xDD\x00\xE4\xF1\x00\xE4\xF1\xC0\x95\x80\x67\x9A\x5D\xBD\x33\xBA\x45\x2F\xBA\x05\x57\x05\x9F\xC2\x2A\x03\x6D\xC3\x86\x33\x5D\xD4\xEC\xF0\x11\x05\x64\x65\x80\xFE\x6D\x77\x88\xBB\x08\x84\xB0\xE0\xAA\xA3\x74\xB3\x6F\x4A\x74\x2E\xA2\xF3\x6A\x44\xB9\xA1\xD3\x18\x8E\xC2\x73\x4E\xF2\x91\xF3\xC9\xC7\x84\xB6\xBE\x49\x3E\xFF\x56\xC8\x87\x50\xAE\xC3\x61\x1E\x28\x49\x78\xFB\x17\x59\x26\xAA\x4F\x81\x15\xA3\x56\x54\xA4\xB1\x75\xD9\x69\x75\x5E\x80\xB9\x9B\x94\xD4\xDD\x75\x23\x1D\xA5\x75\x9F\x12\x7B\xE1\x89\x93\xEB\x46\x06\xB3\xD5\x2F\x26\xEC\xBF\xE4\x44\x48\x5E\xB6\xE3\x77\xA2\xC5\x3A\x4A\x1D\xAB\x8C\xE3\xDC\xF7\x16\x05\x9D\x3A\x38\xA7\xC3\x46\xF3\x71\x4F\xF0\x0B\x65\x59\x45\x06\xB0\xB8\x82\xA8\x11\xD4\x87\x26\x48\x15\xCA\x61\x30\x55\x0D\x8F\x3E\x3D\x5E\x7D\x43\xD5\x48\x63\xC8\x12\xEC\xDC\x19\x4C\xBC\x1F\x46\x7A\x40\x8D\xCA\x4F\xE6\xD9\x42\xE2\xBF\x1A\x3D\x43\xDF\xF4\x1F\x7E\x1C\xFF\xFD\xC3\xA3\x5E\xEE\xE4\x9B\x7E\xBC\x12\xD3\x92\x3D\x75\xF7\x0C\x21\xEB\x64\x6D\xD9\x2F\xB4\x91\x07\x28\x63\xF2\x01\x85\x51\x0B\xD0\x13\x8F\x04\x3F\x08\x2D\xD6\x66\x4F\xB7\x5E\x53\x0E\xCF\x53\xC7\x6B\xE3\xC7\xF1\x0E\xAA\x26\x25\xCF\xAF\xB7\x9D\xB2\x00\xCF\x37\x4C\x3D\x9F\x9A\x7A\xE6\x0E\xC6\xB4\xAD\x8D\x61\x52\x14\x06\x7F\xC2\x40\x31\xB6\x74\x99\xD7\xC7\xE9\x56\xFD\x3A\x1E\x5F\x4F\x7D\x68\x48\x93\x0C\x39\x08\xA6\x8E\x28\xF2\x4E\xB7\x98\x11\xE0\xCD\x00\x8F\x28\xD1\x75\x9E\xA9\x0A\x26\x33\x54\xA1\x0F\xDC\x1E\x26\x44\x97\x5D\xA4\x50\xC5\xCE\xF6\x86\x0E\xD2\xD5\x21\xBE\xD8\xCD\x3E\x58\xB2\x1E\x92\x91\x9B\x79\x13\x03\xDE\xCF\xB2\x2D\x4B\x6C\xCB\x06\xC6\xA6\x66\x6A\x64\x54\x23\x43\x37\x31\x97\x1D\x8D\x6D\xBB\x0C\x13\xA2\xCC\xB0\x39\x4B\x46\x6E\x64\x73\xC6\xA9\xB8\x65\x6E\x89\xF0\xBB\x28\xAD\x81\x95\x65\xD0\x76\xB2\x65\x6E\x88\xD5\x40\x95\xA3\x55\x86\x3D\x16\x31\x9E\x1B\x70\x39\xD3\xE7\x72\xE1\x2A\x51\xE0\x72\x76\x59\xD6\x3B\xDD\x02\x59\xB1\xC8\x2B\x42\x23\x64\x47\x31\xDB\xDB\x09\x6C\x6F\x17\x5B\xB1\xA6\xBB\x32\x34\x22\x93\x5E\x31\xDB\x53\xDD\x9B\x8C\xD8\x1E\xEF\x2E\xA1\x22\xB8\x33\xB0\xBD\x9D\xC0\xF6\x76\x02\xDB\xDB\x19\xEE\x14\x99\xCE\x8A\xCD\xA6\xD9\x1E\xED\x84\x02\x78\xEE\x8C\x3B\xA1\xF8\x18\x36\x46\x25\x2A\x6F\x98\x27\x23\xEE\x84\x3E\xB6\x92\x01\xB7\x3D\x5B\x4C\x12\x9B\x8D\xE1\x5F\x4B\x57\xBA\x01\xBA\xD9\xB9\x21\x2D\x28\x1D\x97\xAE\x2B\xD2\xF3\x22\xE6\x75\x89\x85\x10\x4C\x08\xA7\x2F\x51\xE6\x32\xE0\x35\xC1\x4A\xB8\x58\xCA\x89\x57\x71\x97\x4F\x60\xB8\x03\xAF\x30\x16\x41\xF9\xED\xF0\x32\xBA\x27\x7A\x71\xB9\x18\xBB\xE0\xD3\xE0\x38\x89\xEF\x59\x83\x47\xEC\xE5\x83\xD3\xBA\xD2\x89\x74\xB7\x50\x96\xDF\x0A\x6F\x47\xD1\x2D\x82\xBE\x85\x96\x70\x65\x52\x0B\x6F\x55\xD2\x4E\x78\x12\x9D\x0B\x05\x71\x18\xE9\x55\x74\x2F\x08\x57\xE5\xF0\xFB\x82\xFD\x00\x02\x63\x41\xEE\x83\x29\xE0\xC1\x42\xC6\x34\x4B\xE8\x85\x02\x6B\x5D\xF7\x18\x8D\x05\x46\x63\x99\xD1\x98\x70\xF7\x20\x61\x34\x6A\x8A\xD1\xA8\x29\x46\xA3\xE6\x30\x1A\x8B\x57\x9D\x1B\x43\x49\x24\xF0\xA6\x70\xC9\xE9\x94\x03\x9B\x51\x73\xD8\x0C\x05\x8F\x0B\x6C\x26\xE6\xE5\x92\x21\x3D\x0E\x86\x53\xE1\xF4\x70\x21\x73\x96\x0C\x95\xD8\x2C\xFF\xCD\xF9\xE0\xCB\xBF\x09\x3E\x27\x39\x41\xCE\x2C\xD0\xFE\x45\xA6\xB2\x5F\x75\xC2\x78\xEA\xB2\x71\x92\xFC\xB1\xBB\xC2\xAA\xBA\x8C\x3B\xA9\xCC\x7F\x0F\x4A\xDF\x22\x3D\xB0\x60\x81\x67\xE6\xB6\xDF\x28\xEC\xA1\x73\x09\x92\x94\xD4\x67\x46\xD8\xEB\x46\x72\x40\x34\x52\x07\xA6\xAA\x05\x61\xAF\x4B\x82\x0B\xC6\x1E\x44\x8F\x12\x43\x67\x2B\x3D\x61\x7F\xBF\xA4\x7D\xFE\x4D\x69\x3F\x57\xDA\x9B\x6F\x4A\xFB\x6F\x4A\xFB\x2F\x53\xDA\x1B\x96\xF6\x66\x8E\xB4\x37\x0F\x90\xB4\xFF\x12\x3A\x1E\x9F\xD6\x5B\x6D\xDF\x0E\x0A\x3A\x58\x2A\x8A\x9C\x36\x69\x6B\xB1\x99\x82\x40\xF7\x3B\x3B\x5F\x65\x29\x32\xF2\x55\xAE\x55\xA7\x0A\x90\xDA\x20\x9D\x62\x8D\x00\xA5\xBD\x5C\x9C\xAD\x18\x6B\xA8\xAE\x86\x32\xB5\x8A\x7A\x03\xD4\x80\x05\xE5\xBB\x26\x00\x26\x53\x8E\x87\x50\x96\xDE\xE0\x90\x7C\xD5\x0C\x1D\x0F\xC5\xBC\x53\x52\xDE\xA6\xC3\x1C\x85\x26\x82\xB1\xA8\x3B\xCE\x87\x9E\x79\x23\x10\xC3\x07\x30\x0E\x80\xA5\x58\x4D\xC8\x55\x71\x1C\xE3\x3F\x0A\x71\x32\xC6\x7F\xCC\xA1\x23\xFE\x26\xB9\xF1\x1F\x52\xB4\x3E\x5A\xAA\x49\xD7\x28\xC9\xE1\xD0\x1A\x1E\x65\x27\x82\x5B\xCC\xB8\x8E\x62\xD4\x25\x31\xFE\xB3\x33\x94\x95\xFF\x7F\x81\xC0\x4F\x98\x33\x5D\x2B\xF4\x38\x52\xBA\x5D\x18\x36\x5C\xA7\x6E\x10\x5A\xCE\x23\x06\x6B\x82\x78\x94\x0D\x10\x61\xB2\xE7\x6B\x6F\x33\x37\x08\x69\x99\xF6\x6F\x10\x02\x60\x39\x9A\xBB\x76\x2A\xA6\x85\xD6\xF4\x0B\x23\x50\x05\x81\x9C\xC7\x8C\x08\xA3\x69\xDF\x2C\x39\x85\x35\xCB\x67\xDB\x18\xE7\x59\xCE\x1B\x63\xD8\x07\x4F\xB0\x46\xF9\x80\x92\x7B\x4C\x08\x0B\x0B\x78\x2A\xE2\x75\x26\x3A\xA5\xB1\x2E\x8F\x43\xB5\x4E\xB9\x9C\xB5\x2B\x02\x2E\x3C\xBF\x2D\xDC\xE1\x2A\xCB\xF2\xAF\x24\x1E\x79\xB0\x84\x24\x4A\xA1\x10\xC9\x30\x8C\x75\x3F\x74\x7A\xA9\xD1\x74\xBB\x73\xEA\xE6\xA3\x08\x87\xF6\xAA\xCB\xE1\x4E\x37\x1F\x75\x1B\xA2\xD3\xF0\xC4\xF8\x04\xA7\x23\xC7\xEE\xCE\x83\xEB\x42\xA1\xEA\xF4\x3C\x40\x61\x72\x58\x3C\xB4\xA7\x00\xA5\xE9\xA1\xBD\x76\x0A\x0F\xED\x35\x2F\xA9\x2E\xE1\x7B\x23\x1C\x86\xA3\x13\xA8\x82\x97\xBB\x41\x3B\x97\xFC\x95\xF0\x12\xA3\x36\x97\xDF\x32\xA5\xD1\xDF\xD1\x57\xE9\xA7\x8C\x03\xB9\x95\x71\xB0\x03\xBB\x82\xEF\x05\x7D\x5F\xEE\x87\x95\x10\x17\x36\xAD\x68\x41\xE9\x64\x28\x25\x28\x2E\x34\xDA\x2F\xE1\x90\x9C\x9B\x99\x0F\x37\x92\x93\x2F\xD9\x18\xE3\x38\x0B\x0A\xA7\x41\x48\x17\xB4\x7E\x44\xF9\xCB\x18\x3B\xAD\x5B\xD6\xB3\x2B\x90\x2F\x1B\x60\xE0\x35\x72\x22\x11\x9D\x13\x89\xEA\x92\x6F\xC1\x92\x66\x1A\x55\x4E\x30\x8D\x8A\x80\x4A\x81\x81\x44\x12\x1A\x2D\x92\xDC\x54\x86\x33\x19\xC7\xF3\x1E\x9C\x66\xE9\x64\x79\x43\xCA\x1F\x04\x99\x3E\x22\x1E\x9B\xE0\x7C\x25\x69\x2C\x82\x8E\x3D\x44\x3C\x39\x11\xF1\xE4\x44\xC4\x93\x13\x11\x4F\x4E\x44\x3C\x39\x11\xF1\xE4\x24\xFC\xBA\xD8\x89\xE9\x8E\xF3\xAF\x5A\xC7\x1B\x4A\xC9\x53\x66\xE2\xC4\x9B\x6A\xE9\x0B\xCA\xAE\x08\x24\xB8\x5E\x2B\x27\x7A\xAA\xE8\x8F\x56\xA2\x9C\xB6\x6B\xC4\x5C\xBB\x46\xF4\xEC\x1A\xD1\xD9\x35\x4E\xF8\x53\x6D\xAD\x3D\xE5\x93\x52\x84\xB1\xBE\x41\x22\xA6\x1E\xCC\xD4\x73\xD1\x86\x96\xC6\x84\xE0\x53\x2D\xA7\x78\x87\xA6\x85\x7F\x1A\x51\x44\xDA\x94\xF0\x26\xD2\x82\x71\xFA\x28\xA8\x15\x31\x90\x9E\x2A\xC7\xC9\xDA\x90\x97\x0B\x59\xBE\x55\xCA\x2C\x30\x1B\xCA\xE0\x86\x87\x7C\xCA\x8F\x6A\x0D\x96\x19\x9E\x1A\x5A\x85\x37\x84\x71\x31\x2B\x7F\x8A\xE2\xD2\xB0\xC7\xE2\x8E\x33\x00\x11\x76\x94\xD1\xB5\x72\x86\x12\x21\xD2\xED\xC5\x26\x77\xB9\xBF\x01\x3E\xA0\x80\x94\x37\x70\xD4\x64\xCC\x2E\xEE\x8F\x61\x4B\x49\x23\x7C\xCE\x2D\xD2\x2B\xC6\xC8\x41\x99\x5B\x85\xE4\xAC\x81\xC4\xCB\xF7\x6A\x99\x4D\xBC\x26\x73\x11\x57\x12\x9D\x16\xDA\xC3\x68\x09\xE0\x21\xE1\x21\x0E\xDA\x8F\x02\x05\x75\x4E\x75\x45\xC5\x77\x9D\xC9\xE5\x12\x60\x66\xE2\x7E\x2B\xDD\xCD\x58\xB0\x42\x6B\x2D\x74\x19\xB3\xE8\xE1\xC5\x66\x8E\xF6\x2C\x63\x0B\x7C\x93\x93\xA3\xF6\xD6\xD6\xB1\x54\x31\x78\x5F\x2E\x43\x0D\x52\x93\x3A\xFC\x98\xE0\x9E\x52\xD0\x1D\x1D\xBA\xC0\xCB\xC9\xCC\x1B\x43\xD1\x55\xA1\xF7\xEC\x50\x85\x91\x1F\x3B\x47\x34\x5A\x9D\x51\xC6\x36\xE8\xC3\x6B\x09\xE5\x86\x16\x89\x89\x8B\xC4\xC4\x45\x62\xE2\x22\x31\x71\x91\x98\xB8\x48\x4C\x5C\x24\xA6\x5B\x24\x5D\xC2\xEC\xD2\x15\x30\xF3\x82\x33\x5A\x3B\x53\x0B\x97\x77\x99\x16\xD1\xFC\xCB\xC8\x6F\x28\xCE\xAC\xF3\x1B\x45\xC7\x2E\xCE\x5A\x2E\xBD\x02\xE4\xE1\xD4\x0E\x53\x76\x04\xA7\xA8\x13\xBA\xD7\x44\xFE\x61\x9C\x55\x43\x97\xAF\xD2\xA0\x6C\x68\xE2\x80\x11\xAB\x9A\xB0\xAA\xA7\xB0\x6A\xD8\x8E\x21\xAC\x1A\xA2\xAD\x80\x55\x4E\xFE\x3F\x07\xAB\x51\xFE\x4D\xDA\xC6\x86\xFC\x0B\xB1\x05\xD2\x13\x4C\xC0\xAA\x09\xE2\xD4\x00\xD3\xE0\xC0\xDE\xC6\xE9\x38\x6F\x85\x18\xB5\x84\x51\xCB\x18\xA5\xE4\xD0\x74\x71\x97\x8E\x06\x5C\xC6\x5C\xFA\x50\x05\x12\xBA\xEC\x24\xB5\x8A\x8D\x76\x64\x12\x6E\xD1\xA3\x36\x1F\x4E\x0C\x66\xB7\xEF\x65\xD8\x33\xE0\x99\x3A\x1B\xF6\x0C\x44\xB0\xE3\x31\x75\x94\xEA\xBD\xAD\xC2\xAE\x5F\xD8\x03\x28\x48\x46\xA3\x96\x3B\x69\xF9\x2C\x9C\x53\x9D\x50\x56\x41\xFE\x49\x92\x04\x03\x99\xF1\x46\xD0\x66\x17\xE9\xCE\x73\x45\x76\x6B\x32\x2E\x46\x2F\x39\x38\x44\x6F\x41\x1A\x83\x29\x32\x79\x41\xDA\x84\x16\x61\x01\x3E\x16\x0F\x74\x68\x29\xCA\x20\xFD\x92\x50\x87\xA2\xCB\xA3\xDF\x21\x48\x4D\x67\xD1\xA7\x15\xA8\x0F\x57\xF3\x03\xB9\xF1\x95\x08\x15\xE2\x7D\x4D\x9D\x6B\x4B\xBA\x31\x84\x1E\x09\xBD\x28\x5C\xBD\x43\xA0\x6F\x15\xEA\xA0\x17\xFB\xD4\x6D\xB7\xDD\x04\x88\xF0\xB4\x59\xF4\x0E\x78\x7B\x80\x0A\xCB\xF2\x02\x11\x6B\x38\x31\xFE\x30\x2E\x8D\x37\x0C\x7B\x49\x53\x38\x49\x03\x29\x1C\xA4\x06\x20\x47\xE0\xA9\x63\x30\xD0\x67\xBD\x14\x06\xF1\xEB\x67\x31\x8B\xDE\xAC\x21\x46\x55\xD1\x14\x23\x4B\x2C\xB7\x59\x51\xC6\xE0\x63\xB4\xC0\x0E\x55\x39\x90\xED\x09\xD2\xE5\x39\x12\x95\x06\x52\x25\x49\x30\xFE\x9F\x41\x0D\xE7\x40\xF3\x84\x71\x79\xBC\x29\x96\x38\x07\xA5\x97\x27\x5D\xB1\x34\xFE\x55\x1C\x07\xBB\xD5\x73\xD6\x42\xF8\x68\xFC\xCB\x67\x29\xD9\x1D\x20\x3F\x8B\x56\x26\xE6\xBC\xC3\x21\xAC\x79\x75\xFD\x6A\x25\x46\x05\x20\xC8\xBF\xFB\x65\x30\xA9\x5F\x0B\x8D\xC9\xFD\xE2\xFB\x56\x92\xD9\xD2\x0B\xCA\xC2\xB1\xAB\x1D\x7F\x24\x1D\x1D\xCC\x83\x06\x98\xC1\x00\x15\x0F\x30\x83\x01\xAA\x73\x0F\x50\xF8\x3B\x93\xBE\x47\xD9\x36\x46\x03\x16\xE0\x3E\xBE\x46\x5B\xB7\x50\xCD\xC5\x31\x48\xFF\x50\x12\x85\xBD\x61\x38\xE1\x6F\xF9\xB5\xAE\x09\x6C\xE1\x7B\xB8\x85\xFD\xD8\xC2\xA5\x49\x0B\x0F\x9B\xDF\x42\x6F\xBC\xF6\xCB\x00\x4A\x07\x12\x93\x80\x64\x64\x88\x95\xF2\x06\xD9\xB2\xB8\x4C\xB3\xFE\xC2\xB7\xE6\xE5\x49\xBC\x2F\xD4\xE4\x94\x22\xA0\x31\xFE\xD8\xD1\xCA\xF0\xC5\xBA\xB0\xD5\xE4\x4B\xDA\x9F\xDB\xD1\xFA\x63\xC7\x1B\x3B\x7E\xC9\x59\xDE\xF0\x63\xE9\x0F\x8B\xFA\x04\x76\x80\x27\x89\x14\xBC\x26\xF3\x72\x9D\xAC\x50\x79\x02\x58\x2B\x46\x11\x32\xAE\x40\xD3\xC4\xE5\x4B\x2E\xE3\xE4\x6F\x28\x8A\x42\xEA\x04\xFF\x42\x00\xA9\x97\x11\x42\xF7\x0B\x04\x3A\x05\x01\xDE\x83\xA2\x5D\x2A\xE8\x59\xC5\x56\xCC\x12\x27\x2C\x84\x56\x4C\x68\x05\x99\x87\xFF\xC7\xDF\xE8\x53\x4D\xBA\xE2\xD0\xD7\x49\xD3\xBD\xD4\xA9\xD5\x05\xE3\x6C\x6C\x32\x52\x93\x8E\xD4\x9C\x9B\x82\xE3\xA2\x82\x25\x45\x6E\xD1\x94\x61\x11\x54\x3D\xE1\x9F\xBE\x91\x10\x1C\x5F\x4A\xDE\x6A\x99\xE7\xE9\x32\xCF\xB7\xB7\xCC\x59\x9B\xEB\x8D\x88\x77\x44\xFC\x3F\x24\xE0\xA1\xF0\xC6\x2F\x48\x96\x42\x99\xB2\xB8\x8F\xFF\x46\x6F\xD1\x47\x46\xBB\x58\x8E\x7F\x01\xF0\x2F\xCA\xFB\x72\x3C\xAD\x40\x36\xE9\x31\x83\xD7\x14\xAB\x0C\x0A\x86\x3C\x09\x00\x06\xA0\xAE\x70\xF8\x7A\xD0\x7B\x56\x70\x03\x83\xD3\x26\x1B\xBA\xFE\x46\xE9\x3B\x50\xFF\x01\x49\x84\x5F\x69\x2E\xC5\x64\xDD\x4E\xF8\x0F\x8A\xF1\xED\x67\x71\xFB\x84\xC0\xC0\x97\xBC\xE9\x16\x1E\x7E\x15\x3E\xB8\x74\xBB\x1F\x5C\xBE\xDD\x0F\x9A\xED\x7E\xF0\xC8\xED\x7E\xF0\x88\xED\x7E\xF0\x7F\x6D\xF7\x83\x0F\x88\xED\x7E\xF1\x57\xDB\xFE\xE2\xAF\xB7\xFD\xC5\x77\x6E\xF7\x83\x0F\x6E\xBB\x8B\xCF\x6F\xFB\x8B\x2F\x6E\xFB\x8B\x2F\x6C\xFB\x8B\x6F\xDB\xEE\x07\xDF\xBE\xDD\x0F\x1E\xD5\xFF\x00\xD8\x6C\x46\x97\xE4\x93\x16\x1A\x5A\xB1\x2B\x75\x96\xE4\x06\x07\x8B\x11\xF9\x21\x99\x89\xBF\x9D\x75\x99\x80\x1E\xB6\xD6\x58\xD6\x52\x1F\x46\x82\x06\xCC\x58\xA7\xC6\x1F\x25\x5B\x85\xCC\xAA\x4B\x1D\xE6\x57\x82\xA7\x4B\xDB\xF1\x27\xCF\xC4\x70\x19\xCE\xEE\x53\x0F\x75\x16\xA4\xB7\x42\xE1\x80\x2E\x48\x02\xAB\x7B\xE1\x27\x94\x5D\xD1\x8B\xF1\x7F\x3B\x4B\xA1\x67\x4B\x27\xC7\xFF\x78\x86\xB2\x9E\x13\x03\x44\xF6\x43\x1F\x14\xC1\x34\x16\xCE\x5E\xA2\xDD\x65\x4A\x50\x7E\x8F\xBD\x2C\x50\x2D\x08\xD4\xB1\x8A\x61\x48\x48\x13\x3F\x8C\xFD\x8D\x5B\xFF\xBA\x97\x75\xDA\x1D\x7D\x38\x6E\xFD\xCD\x89\xCA\x17\xCE\x85\x74\x3C\xF6\x01\x49\x8A\x17\xF6\x0A\x0A\x36\x44\xE7\x24\xAE\x00\xCB\xBE\xE0\x88\xB3\x41\x88\x0D\x01\xB2\xE5\x92\x17\x78\xC6\xB5\x0F\xAF\x8B\xCB\x93\xAE\x5C\x72\xB6\xCB\x9B\x87\x01\xD6\xED\x3E\xB6\x17\xA8\x21\xAA\x1C\xDA\x38\x09\x9F\x1D\x5F\x3A\x54\xC9\x32\x80\x1C\xEC\x98\xE5\xD7\x6D\x6C\x6C\xBC\xD6\xA0\xEA\x4A\xBB\xA6\x1A\x64\xF4\xB7\xC2\x9F\x4B\xC4\x27\x6E\xBE\x71\x43\x5C\x26\x04\x6B\x17\x97\x88\xCF\x74\x05\x63\x2C\xF8\x52\x57\xB0\x17\x0B\x9E\xF9\xAC\x58\xF0\x90\xD6\x8B\x7D\xE2\x79\x50\x70\x40\xA0\x92\x62\x9D\x62\xA5\x07\x80\x44\xEE\x72\x39\xF9\xEA\x01\x0A\x9B\xE1\x12\x47\x5C\xC2\xF1\x26\x32\xCC\x76\x32\xCC\x26\x32\xAC\xF0\x23\xB0\x8D\xD9\x65\x3D\x01\xF2\x9A\x1F\x3D\x99\x53\x04\xC3\x40\x53\xFD\x0B\x46\x61\xF7\x21\x6D\x8D\x5B\x12\xA7\x7C\xD0\xC6\x1D\x98\xA4\x03\x03\x14\x81\x8A\xBD\x21\x72\x09\xB4\x42\x34\x78\x69\xD8\xA0\xBD\x07\x0D\x9A\x20\xDD\x67\x49\x4D\x3C\x5C\x62\x10\x31\xBB\x5F\x8C\xBB\x79\xEC\x0D\x5F\x00\x25\x77\xC5\xFB\x62\xB1\x7F\x18\x8B\xD1\x72\x95\x2D\x8C\x11\x68\x39\xB8\xF5\xC6\x7B\x71\x74\xD8\xCC\xF9\x44\x0F\xA8\x87\x3A\xE1\x77\xB5\xC1\x16\x79\x10\xFC\xC2\x5D\xB3\x7D\xA1\xAC\x7C\x85\x94\x76\x42\x46\x95\xDF\x45\x69\x8B\xC1\x82\x1A\xE3\xE9\x29\xE7\xB1\xE3\x5D\x6B\xDA\x8E\x51\x74\x25\x21\xEE\xDC\xEC\x17\x26\x8C\x95\xF4\xA5\x38\xDC\x30\xDD\x7D\x6A\x4C\xF1\x85\x77\x91\xFB\xBA\x49\xB6\x8D\x28\xC1\xC0\xF5\x31\xD8\x02\xE7\xB1\x38\xA0\x76\x39\x15\x15\xFC\xF2\xE3\x7A\x9E\x45\xAB\x93\xB3\x2B\xDC\x90\x27\xE0\xBF\x8C\x32\xA2\x03\xD5\x10\x2C\x8E\x1D\xC5\x30\xCD\xE6\x44\x17\xF6\x80\x94\x5B\x01\xC4\x90\x28\xB7\x19\x9A\x8D\x8E\xF7\xC5\x46\x2D\xDD\x7D\x56\xAC\xE9\x1A\xCE\x86\x8E\x2A\xAE\x21\xEF\x84\xCC\x11\x3B\xC9\x30\x8B\x7B\x50\x6D\x33\xDC\xFA\xE3\x8C\x7E\xCC\x4C\xF0\x7C\x4F\x90\x2D\x12\x00\x26\xFB\x00\x13\xCE\x24\xBB\xC7\x48\x62\xF4\x31\x87\xB2\x1E\x37\xB8\xC5\xB0\xEB\x10\x5A\xC8\x48\xD9\x3A\x69\x1D\x21\x3B\x7E\x79\x58\x23\xF8\x41\x6F\x1F\xA2\xEB\x58\xF7\x3B\xD6\x29\xA6\x24\x9F\x36\xEC\x02\x16\xDC\x29\xA4\x22\xEE\x3D\x4A\x87\xB9\x69\x24\x71\x14\xBA\x5B\x67\xFA\x3B\x81\xF0\x2D\xF5\x9F\xE8\x8F\x2E\xF3\xA7\xDA\x03\x6A\x57\x6A\x78\x77\xFA\xA0\x13\xE3\x8F\xA3\xCB\xBD\x28\xCB\xDF\x79\xB0\x2C\x02\xBA\xEF\x12\xBD\xD8\x48\x60\xFA\x3B\x32\xE5\xC4\xE5\xC2\x39\x3E\xD4\x10\x64\xC6\x35\xAA\x6D\xC8\xD3\xC1\x5F\x4A\xC1\x15\xE0\xF5\x43\xF1\x28\x36\xBA\x5E\x10\x14\x0B\x22\x70\x8C\x49\xAE\x5B\x34\xA3\x55\x3C\x7D\xE2\x62\xF4\x1C\xA1\xDC\x6C\x40\x40\xE2\x8A\x46\x2D\xD5\xC6\xA3\x45\x20\x9C\x5A\x22\xAA\x16\xDC\x2A\x7A\x19\x08\x67\x69\xD5\x5D\xDA\x3A\xCE\x49\xB3\xAF\xA5\x98\x6D\x98\x0B\x98\xC6\x37\xD7\xEA\x47\xA3\xEB\x2F\x05\xE6\x9C\xF4\x3F\xD4\x2E\x5C\x68\x73\x99\xCF\xFC\x33\x19\xDD\xB5\x78\xC6\xA4\x5D\xD0\x22\x57\x9A\x3D\x4F\x2E\xC6\xB3\x9F\x8B\xFD\x98\xE5\xA7\xF1\x0B\x44\xC6\xC7\x8E\x56\x83\xE0\x1C\x40\x64\xAF\x98\xEC\xB3\xC8\xF7\x91\x9B\x29\xF2\x14\xC9\x88\xD8\x9D\x81\x05\xA0\x99\xFA\xD0\x2B\x87\xFC\x3B\xD8\xBC\x03\x84\xAB\x15\x72\xBF\xB5\x21\x8F\x23\x1A\xF8\x9C\x3E\x2E\xE1\x28\xB0\x68\xF6\x92\x1B\xCD\x51\x8C\x1D\x63\x18\x1E\xDE\x00\x41\x8C\x06\x25\xB9\xC3\x8C\x1B\xBB\xD6\x64\x5E\x5D\xDF\xE0\x6E\x83\xF2\x80\x76\xC3\xE3\xA7\x0C\xE9\x6B\xA0\x5A\x5C\xDF\x14\x60\x52\xAE\x2E\xF9\xFB\xEE\xBB\xEF\x3E\xED\x34\x50\x29\xFC\x94\xAB\x4B\xEC\x47\x43\xE9\x37\x43\x03\x43\xBC\x7A\xC3\xD3\x45\xB7\x18\xF4\xE2\x68\x72\x1A\x89\xA2\x9B\x39\x35\x58\x4B\x40\xC9\x18\x25\x00\x37\x1E\x95\xB3\x89\xC9\x8A\x27\x6D\x7B\x49\x32\xEE\x22\x36\x35\x76\x76\xA5\x2A\x02\xAF\x1B\x15\x25\xE3\xE2\x8A\xCA\x26\xA8\x28\x22\x2A\x8A\xAF\x73\x54\x14\xE7\x40\x45\xF1\xE5\xA2\x62\xF0\x95\x45\x45\x1E\x51\x91\x77\xA8\x30\x09\x2A\x16\x23\x2A\xF2\xAF\x0D\x2A\xB6\x81\x0C\xF9\xAF\x1C\x19\x22\x20\x23\xF2\xA8\x09\xF0\xA8\x51\x06\xA0\x03\xCB\x51\x27\x88\x19\x44\xC4\x64\x5F\xDF\x6B\xC4\x89\xF1\x27\x50\x52\x65\xE7\x40\x4F\xF6\xE5\xA2\x27\xFF\xCA\xA1\x87\xAF\xC9\x24\x08\xE2\x49\xD9\x32\xC1\x49\x16\x71\x62\xBF\xBE\x71\x32\xB2\xE7\xC0\x85\xFD\x72\x71\x91\x7D\x65\x97\x8A\x89\x7C\x2B\x86\xCA\x9A\xD1\x74\x00\x2E\xE8\xF1\x80\x3E\x7D\x64\x7E\x92\x2A\x63\x52\x55\x26\xE7\x13\x81\xBE\x2A\x63\xA2\x2A\x03\x78\x11\x57\x34\x3A\x51\x65\xF4\x12\x5D\xBF\xEA\x54\x19\x8C\xB4\x95\xCF\x51\x65\x32\x82\x7B\xE6\x74\x4B\x28\xFE\xB6\xC3\x95\x4C\x69\x06\x6F\x97\x64\x4E\x01\xF0\xBF\x49\x38\x5F\x75\xC2\x49\xC8\xE6\xEF\x5F\x88\x36\xE0\x88\xF6\x6A\x25\x13\x11\xA8\x9F\x09\xBA\x4C\x5C\xE2\xE6\xEB\x1C\x53\xE6\x1C\x98\x32\x5F\x2E\xA6\xEC\x57\x16\x53\x3A\x62\x4A\x97\xCE\xD0\x2E\x94\x77\x2D\xEE\x9C\xEC\x53\x77\xFF\x1D\x22\x0B\x4C\x39\x71\x89\x7E\x1F\x3C\x5D\xA6\xF7\xD1\xD3\x7B\xE8\xC9\x39\x89\xBB\x2A\xFB\xD4\xCD\x1F\x8E\x95\x71\xDF\x45\x6F\x7C\x18\x6B\xF0\xD7\x9F\xA5\xFA\x05\x3D\xDD\x43\x4F\xD3\xE6\xD3\x7C\xEA\x70\x6A\x8D\x9D\xC9\x24\x8E\x8E\xC4\xB6\xB8\x72\x69\xFC\x4F\x7C\x21\x3A\xD2\x8E\xF2\xE2\xE1\xD2\x90\xD7\x4D\x8A\x47\xD3\xC7\xA3\x09\xD6\x8F\xE9\xB4\x1A\x33\x07\x8F\x26\xA0\x81\xEF\x7B\x4E\xE1\xD1\x6C\x85\x47\xD3\xC3\xA3\x4A\xF1\x88\xD1\x6C\x02\x1E\x99\x78\x09\x8F\x6A\x06\x8F\x66\x0A\x8F\xAA\x87\x47\x33\xAB\xD5\x48\x7F\x97\x98\xB1\x64\x6F\xBA\x40\x8E\xE6\xA7\x38\x0F\xFB\x28\x21\x44\xCA\x66\xC6\xEC\x3E\xB5\xAF\xB1\x6D\x93\xB1\x77\xD5\x43\xEB\x1C\xFE\x5C\x8A\x91\x34\x3B\x33\x96\x6E\xCA\x36\x36\xF2\xFE\x8C\x78\xBF\x8D\x31\x37\xB9\xD8\x59\x8A\x12\xCC\x79\xFA\x9D\x05\xF6\x6F\x97\xEA\x82\xD8\x7F\xE1\xEC\x12\x05\x65\x65\xF6\x9F\x31\xFB\x2F\x0E\xA8\x4B\xC9\xA0\xDD\x17\x85\x01\x40\x88\x46\x86\x81\xE8\x70\xEC\x77\x3D\x9D\x29\x89\xE3\xDB\x6A\xDA\x65\xC0\xF0\x8D\x7F\x25\x28\x01\xE6\x7F\x7D\x16\xD7\x49\x98\x4F\xE1\x87\x74\x0D\xF2\x18\x23\x1D\x9D\x0E\xCA\xD5\xB0\x93\x7A\xEC\x78\xA3\x22\xF3\x29\x70\xC3\x0D\x63\x9F\x03\x83\x41\xE6\x53\xA4\xCC\xA7\x20\xE6\x63\xFB\xCC\xC7\x32\xF3\x01\xBB\xA0\x63\x3E\xB6\x63\x3E\xFB\xD4\x2E\xF2\xB5\xDC\x4B\x51\x34\x30\x21\x04\xEE\xFD\x76\x6C\xC7\xF2\xEE\xCB\x5A\xA3\x80\x5C\xF1\xCC\xD4\xF6\xC9\xD5\xBA\xDC\xE9\x35\xA0\xE4\xEB\x41\xC0\x3A\x1B\xC9\x35\x4F\xC9\x95\x02\x56\xA3\x67\x78\xDE\x91\x6B\xDE\x91\xAB\x2D\xD1\x87\x02\xC8\x35\xA3\x91\x58\xA2\xDE\xBA\xA0\x23\x3A\x8C\xC6\xCA\xBE\x4E\xE8\x6F\xDB\x91\x6B\x41\xE4\x9A\x13\xB9\x5A\x0C\x98\x80\xC7\xA2\x91\xED\xD0\xB7\x80\x9C\x04\x09\xA3\xAF\x2B\x24\xE8\x79\x48\xF0\xB2\x56\x5E\x52\x50\x67\x46\x46\xBE\x25\x32\xB4\xCB\x99\x77\x0C\xDC\x20\x41\xC6\x1C\xDE\x61\x37\xE7\x1D\xF3\x90\xA1\x53\x64\xE8\x14\x19\xF9\x3C\x64\xE8\x0E\x19\x61\x12\x18\xF1\xC8\xE5\x84\x14\x9C\x95\xA2\x2D\x1D\x4C\xDC\x5D\xF8\xFA\x70\x0F\x3D\xF9\xBF\x0A\xF4\x98\xC0\xDF\xEE\x07\x8A\xFA\xEB\xE5\x6B\x8C\xA2\x38\x91\x3E\x9A\x42\xF8\x6E\x8E\x4F\xD0\xA0\xBF\x14\xB3\x57\x4A\x0C\x48\xEC\x95\xB2\xCD\x99\x3A\x73\x85\xDF\x1F\x38\x73\x3D\x40\x8D\x19\x3D\xD5\x0B\x50\x9B\x23\x7A\xB5\x2F\xCF\x85\xDE\x61\x50\x48\xF0\xD4\x85\xD0\xAB\xDD\x90\xD1\xAB\x01\xBD\x8A\xD1\xAB\xF1\xD6\x53\x44\x2F\xDE\x7C\x52\x8C\x5E\xD5\x43\xAF\xEA\xD0\x3B\x85\x60\x45\x08\xD6\x7D\x26\x28\x53\xA4\x0E\x01\xA9\x65\x0F\xA9\x8A\x91\xAA\x1C\x22\x5D\x5E\xDF\x8C\xDC\x28\x41\xAA\x4A\x91\xAA\x3A\xA4\x46\xE5\x0D\xAB\xF4\x90\x5A\xD2\x85\x29\x1A\x89\xC5\xD7\x40\xAC\x25\xDE\x6C\x53\x07\x01\x02\x01\xA9\xC3\x04\xA9\x9A\x90\xAA\x12\x26\x48\x31\xCB\x58\x66\x47\xD0\x5B\x5F\xE1\x4A\x49\x41\x5F\x10\xE8\x0B\x06\x7D\x1E\x57\x96\x05\xD0\x47\x76\x4C\xA0\xB7\xE9\xCA\xB2\xB4\xB2\x8A\xFE\xCA\x2A\x78\x65\x0D\x5C\xD6\xBA\x22\x59\x5C\x45\x7F\x71\x15\xC9\xE2\x2A\x08\xC5\x02\x17\x6B\xE0\x7D\xBC\xB0\x8A\x39\x0B\xAB\xE8\x2F\xAC\x62\x8A\xF7\x15\x5B\x2D\xAC\x62\xF3\x85\x55\x60\x34\x5B\x5E\x58\xCC\x53\x68\x61\xD9\x99\x85\x55\x4C\x2D\x2C\xDB\x5B\x58\x05\x2C\x2C\x3A\x3E\x50\xBD\x45\x15\xF4\x85\x73\xE8\xA4\xB8\xE2\xF6\xB1\x5A\xEA\x9F\x0B\x9A\x2E\x29\xA3\xC0\x3B\x59\x89\xEF\x8E\x11\xBA\x94\x06\xE5\x5F\x2C\xF4\x14\x31\x75\xDE\xA7\x08\xBA\xC5\x08\x45\xF1\x14\xC1\x92\x16\x96\xF5\xD4\xAF\x9C\x19\x41\x54\xBF\xF2\xB6\x0E\x59\xA4\x6D\xA2\x7E\xE5\xC8\x8C\xA3\xFA\x95\xBB\x1C\xD4\xAF\x7C\x09\xF4\x34\xF2\x50\xCA\x97\xC8\x85\x94\xD5\x2F\x68\x07\x8F\x4B\xA2\xF5\x6D\xA3\xF5\xCD\xCB\x52\xBB\x9C\x86\xC8\x61\xE0\xE8\x30\xC9\xB4\x98\x53\x32\x0F\xA9\x35\x31\x3B\x60\x56\x57\x51\xDD\xCC\x59\xF6\x87\x94\x19\xC8\x6C\x3E\x88\x1A\xDA\x80\x7C\x8A\x45\x93\x63\x84\xAC\x3C\x30\xBB\x6A\x0D\x2F\xED\x50\xBA\x4D\x20\xCF\x92\xC2\xA7\x97\x0E\x57\x0F\x85\x02\x01\x74\x95\x6E\x04\x54\xA7\xAF\x6F\x6C\x47\x95\x0A\xCC\xDF\xC2\x0D\xE1\x5D\x09\x14\x0B\xAC\x3E\x50\xA4\x74\xA5\xD7\x44\x91\x62\x75\xA9\xF7\x85\xD7\xEB\xFF\x9B\xBD\x77\x81\xB6\x2C\xAB\x0A\x43\xF7\xFA\xEC\xEF\xDA\xFB\x9C\x7D\x3F\xD5\x14\x7D\xDA\xD7\x73\xEF\x81\xE3\x5D\x42\x97\xAF\xC6\x78\x52\x4D\xDA\x4F\xD7\x2A\xBB\xEA\xD6\xED\x02\xE9\x97\x11\xC7\x4B\xC6\x4B\xC6\x23\x0F\x93\xA7\xFB\x16\x4D\x9D\x5B\x65\xD3\x86\xA2\xEE\xED\xA2\x90\x4E\x14\xC5\x80\x1F\x90\x3C\xAB\x0B\xA4\x10\xBD\x09\x6A\x7C\xE8\x43\xA4\xA2\x4F\x41\xFC\x80\x22\x7E\x22\x1A\xE4\x35\x8A\x62\x0C\x6A\x34\x28\x86\x7A\x63\xCE\xB9\xD6\xFE\x9C\x73\x6E\xDD\xEA\xA6\x9A\x88\xA3\x6B\x40\xDF\xB3\xF7\x5E\xFF\x35\xE7\x5C\x73\xCE\x35\x3F\xB5\xF1\x27\x20\xDB\x9C\xD7\x9C\x37\x07\x42\xEA\x98\xB9\x4C\x86\xBD\x1C\x54\x0B\x7B\x09\x98\xA6\xCA\xD8\x43\x88\x1C\xFD\x70\x65\xF2\xF6\x7E\xB8\xCA\x8D\x15\x15\x1B\x4C\x57\x05\xAD\x01\x85\xCF\x0C\x4F\x14\x82\xE3\x2E\xC5\x84\x76\x55\x02\x31\x67\x59\x8C\xDB\xC8\x58\xB3\x20\xAA\x7A\x20\xAA\x41\x21\x7C\xCE\x03\xF4\x1C\x5B\xDF\x82\x16\x01\x14\xB3\xF5\xC3\xDB\xA9\x64\x96\xAD\x4F\x16\xC3\x55\x42\x44\xD1\x9F\x3B\x28\x23\x21\x5C\x25\x2D\x5C\xE5\x90\x0C\xE1\x2A\x71\x70\x15\x32\x5B\xAF\x98\xAD\x67\x28\x23\x92\xC4\x10\xC5\x7E\x09\x4F\x62\xF3\x43\xDC\x7C\x35\xD8\x7C\xE9\x37\x9F\xC4\x3C\x2C\xD7\x6D\x7E\xB4\x70\xF3\x25\x6D\x7E\x6B\x03\xC2\xF1\xB1\x13\xEE\x36\xF1\x5B\xEF\x6C\xFD\x23\x3C\x62\xBA\xAD\x5F\xBC\xE7\x11\xFB\x82\xE6\x60\x4E\x38\xBB\x66\x1A\xAC\x61\xFA\x67\x03\xEF\x84\x46\xD3\xC0\x19\xC4\x56\x20\xA2\x9C\x9A\xD4\xDA\xAA\x29\xBB\xDE\x72\xF0\xEA\xC2\x22\x81\x76\xDA\x35\x22\x9D\xB9\x5D\x6D\xAA\x18\xB4\x3D\x3F\xA5\xFD\xA1\x0F\x08\x51\xCE\x6C\x23\xA6\xFF\xB0\xB3\x45\xEB\x5B\x91\x34\x4D\x43\x10\x66\x01\x2B\x2B\x22\xA7\xCE\x7C\xDF\x23\x73\xCE\x86\xE6\xC4\xF0\xBB\xA0\xE7\x90\x70\x25\xCA\xCA\xD3\x33\x91\x4F\xFC\x0C\xC0\xB0\xAB\xB5\x98\x56\x2E\x53\x67\xEB\xEC\xA7\x58\x3B\x73\x61\x5A\x71\xD8\xB5\x9C\x88\x44\xEB\x5E\xE7\x86\xA9\xD9\x3F\xD9\xAD\x28\x44\x60\xC8\x36\xBD\xA9\xB1\x88\x9F\x91\xCD\x87\xA5\x5C\x19\xF2\x1D\x29\x69\x31\xF0\x98\x68\x6A\xB5\xCE\x1E\x79\x14\xF1\x1A\x3B\xC9\xDD\x66\xC6\x6E\x61\xB4\xDF\xB9\xF6\x44\x8E\xAD\x6C\x4F\x64\xB9\xF8\x44\x56\x9E\x43\xA0\x01\xF1\x89\xCC\x86\xD0\x47\xC8\xD6\x31\x67\x62\xC9\xF2\x95\xA6\xF4\x7E\xEE\x44\x0E\xD9\x70\xB0\x4B\x49\x81\x7C\x63\xEF\x4C\x0E\xBB\x33\xB9\xA7\xCB\x48\x86\xBA\x8C\x84\x63\xC9\xB6\x67\x33\x07\xCB\xF7\xBA\x0C\xA6\x70\x83\x33\x59\xB5\x67\xB2\xD7\x65\x14\x50\x3C\x91\x33\x59\xCF\x9D\xC9\x9A\xCE\x64\xCD\x23\xF1\x67\x72\x0C\xDA\x9F\xC9\xC6\xD3\xC5\x10\x4C\xEF\x4C\x8E\xE7\xCE\xE4\x19\x5D\x46\xE4\xBD\xD1\x86\xF8\xB3\xE8\xAE\xFE\xB7\xD9\x66\xC4\xE7\x20\xAA\x5D\x62\x70\x79\xC2\xD9\x09\x2C\x32\x15\x81\xB0\x6F\x15\xE3\x9C\x17\x81\x93\x13\xEC\x63\x14\x12\x1C\x0A\xEE\x72\x86\x0B\x01\x1B\x5B\xBB\xC1\x39\x52\xDB\x7E\x5B\x9B\xFB\xB6\xDA\x7E\xBB\x63\xE6\x9B\x73\xC0\x70\x76\x4E\xB9\x17\x92\x0D\x04\xE6\x35\x72\x38\x3D\x49\x86\x22\x6B\x92\x0E\x4A\xE9\x2C\x6C\x9C\x5A\xD2\xF9\x04\x6B\x08\x38\xE7\xA9\x6B\x78\xB3\xD6\x08\x0C\xE1\x40\x41\x19\x38\x05\xA5\x46\x38\x88\x20\xEA\xE9\x26\xC3\x16\x08\xA2\x4E\x31\x19\xB5\x9C\x1D\x31\x62\x7D\xC5\x24\x85\x33\xAC\x5D\x1C\x29\xC9\x7C\x5A\x15\x11\x17\xCA\x2E\x2B\x5E\x31\xA9\x40\xF7\x80\x20\x22\x8F\x3C\x44\x11\x6F\x38\x84\x07\x29\x9B\xCD\x78\x95\x56\x3B\x1D\xBF\xDB\xEF\x95\x42\x6F\x77\x29\xD8\x9D\x87\x83\xFD\x72\xAF\x42\x66\xD7\x4A\x6D\x7D\x98\x8B\x7B\x82\xE7\xE1\x9F\x2F\x13\x77\xF1\x1F\x20\x25\xF1\x31\x67\x36\xB5\xDA\x90\xF5\x93\x33\x10\x6B\x0D\xD4\x4B\xFB\xAA\x1D\x46\x6C\xC5\xBF\x68\x4F\x38\x46\x95\x42\xC2\xE2\x2E\x05\xD5\x51\x16\x83\x04\x84\x4B\x17\xC9\x41\x0D\xD4\x9A\x5C\x05\x75\x28\x28\x71\xB9\x1F\x42\x44\x9F\x4E\x68\xC9\xE6\x2C\xFF\x0A\xC1\x39\x4B\xC3\xE3\x2E\x27\x90\x66\x98\xBC\x1D\xB4\x07\x4B\x88\xCA\xC7\x5A\xBB\x72\xB6\xCD\x12\x34\xD6\x16\x7A\x98\x21\x50\x9D\xE5\xD6\xE0\xAB\x71\xD6\x5D\x4E\x7D\x8D\x90\xED\xD3\xB2\x9B\x57\xC5\x62\x41\x82\x2B\x18\xE8\x51\xC9\xEA\xF1\x8F\xD8\x07\xD9\xEA\xAD\x4A\x2C\xB6\x14\xE1\x38\x0A\xC1\xA8\x0C\x84\xCC\x94\x0E\xB3\x24\xCA\xB2\x2C\x8B\xD3\xCC\x38\x53\xB0\xBC\xFD\xC5\xD6\x44\xC2\x59\x06\xE5\xA9\xCF\xB5\x3F\xBF\x83\x1C\x2B\x64\xB0\x59\x20\xFC\x7A\x2C\xDC\x59\x04\xAF\x3F\x79\xC4\xB9\x35\xB8\x7D\xD3\xCE\x0A\x22\xA8\x74\xDB\x85\x70\xFE\xEB\x7C\x3B\x24\xB8\x29\xC1\x4D\x09\x6E\x4A\xB4\xFD\x8A\x16\x48\x34\x88\x05\xCD\xC7\x7B\x37\x1F\xDD\x82\xE6\xA3\xBD\x9B\x8F\x6F\x41\xF3\x21\xEE\xCB\x37\xF9\x27\x6D\xE6\x7B\x49\x9E\x50\x2F\x01\x87\xA4\xE1\x8E\x3E\x57\xAC\x14\x8B\xF6\x93\x75\xDF\x41\xAE\x16\x8C\x75\xBF\xD1\x09\x1E\x80\x9B\x0D\x8E\x95\x4C\xFB\x17\x76\x82\x25\xD9\x11\xA3\xF7\xD2\x55\xA1\x20\xEC\x6C\x08\xD7\x01\xB5\x7D\xE7\x6B\x06\x9E\x54\x20\x3B\x13\x37\x39\x34\x71\xEB\x8E\x31\xF3\x86\x54\x8C\x3C\x1A\x7E\x30\xE8\xE3\xE1\xB5\x9E\x7F\xAF\x44\xAA\x84\x12\xEE\xE1\x99\x77\x87\x41\xB2\x95\x70\xEF\x1D\xB0\xBF\xB7\xE2\xE2\xE4\xFA\xC2\xAD\x89\x2D\x4A\xDD\x5B\x36\x36\xDC\x22\xF3\xC2\xB2\x41\xAC\x6E\x6D\x7D\x51\x54\xA3\x5E\x82\x2F\x13\xAB\x4C\x02\x28\x54\x86\x2D\x1B\xA6\x02\x56\x54\x0B\x8C\xC5\x10\xF9\xCB\x51\x11\x88\x30\x49\x92\x44\x26\x49\x44\xD1\x96\x95\xFD\xEC\x1B\xBA\x05\x41\x11\xD0\x13\x83\xD8\xF4\x9E\x50\x7C\xF4\xD4\x20\x42\x81\x08\xFF\x9F\xF9\xB8\xEF\x72\x4D\xDE\x81\x67\xD3\xA1\x60\xB5\x36\x56\x4C\xEB\x74\xC2\x02\xC5\x41\x64\x2D\x1E\x82\x74\xD2\x17\x3B\x52\xB6\x35\x26\x79\x83\xCA\xF8\x7A\x0F\x61\x69\x67\x60\xDC\xEF\xDA\xFE\xCB\x47\x7B\xF6\xD1\xA1\xE1\x85\x6A\x07\x84\xA3\xF1\x37\x99\x73\x23\x49\x86\x23\x49\x06\x23\x49\x9E\xF0\x48\x7E\xB0\x3F\x12\x6D\x68\xF3\xDA\x81\x74\x0B\x7D\x1F\xB6\x74\xB4\x6B\xA9\xCE\x27\x9C\xB3\xF3\x4B\xF1\xC3\xE1\xBA\xC0\x0F\x05\x7E\x18\x4D\xEA\x02\x85\xA3\x0D\x97\x43\x5E\xAE\xC9\xE7\x81\x86\x11\xB9\x1C\x9D\x84\x10\xF2\x09\x8C\xCB\x7F\x8D\xC7\x6A\x9D\x17\xC2\x40\x01\x66\x83\xC3\x73\xE7\x76\xFB\xFE\x42\xCD\x0C\xF0\x37\x5F\xDD\x1B\x20\xA5\xAE\x83\x99\x95\x92\x06\x52\x1B\xAC\xA3\xEC\xB7\x3E\xED\x43\x0A\xF6\xF8\x2C\x1C\x1F\x74\xE3\x23\x07\x1A\x43\x09\xDE\x81\x1B\x9B\x3F\x22\x6D\xB0\x7E\x76\x6E\xA1\x7E\xF9\xBB\x87\x0B\x45\x82\x06\xD1\xC8\xA2\xA5\x08\x86\x29\x82\xB1\x8A\x45\x43\x4F\x0A\x4C\x4B\x0A\x0C\x8E\x9E\x76\xDB\x1F\x29\xC6\xDE\xD5\xD0\xAA\xB7\x2F\xC8\x2A\x1D\x29\x45\x01\xC6\xD1\x08\x67\xFC\xBA\xB0\x2F\x39\xDF\x17\xCD\x6B\xBE\x3D\xC3\x94\xCA\x30\xA5\xEA\x5A\xEF\x53\x1B\x9C\x74\x4E\xF9\xC8\x8C\x23\x26\x9E\x7D\x44\x48\xF4\x94\xC4\x14\x81\xFB\x0A\xDD\xD7\xB5\x8E\xB3\x44\x98\xE8\x3E\x3C\x6F\xF0\xE1\x68\xF7\xE1\x64\x47\x99\x3E\xE8\xEF\x10\xCD\xBF\x28\x44\xE6\x49\xD3\x8E\xE8\x78\xAD\x36\x96\x48\xF9\xA3\x44\x14\xC8\xDE\x43\xB8\x07\xD2\x14\xC8\xFE\x17\x0A\x76\x70\x86\x19\xAB\xAF\x63\x87\xD2\x36\x04\x4B\x78\xDC\x85\x8A\x47\xDA\xC4\xFE\xBC\xA2\xF5\x8E\x8E\x7D\x2B\x27\xDC\xB5\xAA\x32\x9C\x89\x20\x61\x0F\x41\x8A\xE0\xAF\x17\x8B\x66\xA9\xE3\x4F\x51\xA2\x4D\x98\x7B\x4B\x49\x2E\x23\x8D\xE4\x11\x59\xB2\x87\x20\xCB\x63\x09\x44\x55\xEC\xD4\xA3\x28\x32\x76\x72\x58\x3C\x90\xC3\xD4\x9A\x3C\xCD\x99\xAA\xCF\xD4\x09\x4A\x60\x19\xA8\x67\xAB\xE4\x6E\x72\x7B\x48\x1A\x9E\x48\xEB\xB0\xA1\x20\xB3\x79\x73\x44\x9E\xC6\xAE\xA8\xD6\xD7\xD5\x7A\xD3\xE6\x2F\x61\x6D\x75\x6C\x7F\xED\xED\x1F\x7E\xBB\xF0\x36\x1D\x31\x0A\x53\xF4\x15\xC7\x44\xCA\xD2\x78\x63\x62\x3F\x8C\x65\x20\xB1\x6F\xFF\xF0\xDB\x3F\x4C\xAA\x89\x36\x6D\x3D\x24\xBE\x81\x08\x12\x9B\x9F\x76\x13\x4E\x0D\xA4\x08\x84\xA7\x9B\x3A\x6E\x47\x17\xCF\x8F\x2E\xA5\x4A\x4D\x65\x48\x4E\xAD\x72\x1A\x44\x06\xB8\xD6\x90\x39\xB5\x01\xDD\x52\x27\xF8\x27\x71\xA6\xE0\x2C\xE7\xDA\xA4\xA9\x53\xD7\x4E\xC2\xC0\x9E\xF2\xF9\x93\x00\x07\xCB\x53\x60\x68\xCF\x21\x3F\x22\x4F\xF3\x8A\x7F\x1D\x64\xF6\x5C\x83\x2D\xE3\x6E\x75\x1E\x0D\x09\x8F\xA9\xD5\x8A\x81\x5E\xAF\xB8\x57\xA7\xF7\x02\xCD\xC1\x8F\x17\xB5\x69\x20\xC3\x36\xB2\x4E\x5E\x08\x3D\x64\xB1\xB8\x66\x45\xF9\x91\x1D\x17\x3F\xD8\xAE\x3A\x8C\xCD\x78\xCC\x19\x4A\x34\x7F\xAD\xF0\x90\xC3\xED\x63\xE7\x3F\x01\x59\xF9\xDB\x2E\x76\x7D\x0B\x6B\x2C\x48\xDD\x22\x58\xF3\xE0\xBE\x0F\xC4\x81\x42\x54\xA4\xD2\xAD\x10\x60\x1F\x0F\x7A\x62\x8A\xFC\x47\x0C\x91\x9A\x61\x2B\xDE\xAC\x13\x9B\xBF\x84\xB2\x64\x27\xB8\x2A\xA9\x07\x10\x85\xBD\x7B\x05\xE6\x11\xAC\x86\xE3\x7C\x11\xFE\x81\x14\xA1\xE0\x88\xFC\x5A\xCE\x92\x7F\x82\x83\xC8\x41\xCC\x5D\x56\xA9\xF7\x62\x21\x4D\x4B\xEC\x76\x37\xA5\x77\xC6\x35\xE3\x04\x90\xA4\x73\xE3\x2C\xFD\x76\x64\x3C\x89\xF2\x2F\xDC\x65\x94\xB2\xEF\xEB\x4F\xA0\xE7\x32\x91\x30\x21\xC0\xC3\x37\x41\x92\xE9\x18\x02\x12\xC6\x0F\x37\xAD\x2B\x3D\xD2\xFF\x4F\xD0\xAA\x20\xE1\xFE\x03\x1F\x79\xE0\x4B\x71\x00\x87\x11\x73\x1E\xC2\x65\xC6\xB3\x99\x04\xA5\x4F\x07\x4D\xF9\x5F\xB9\x10\x63\xC0\x07\xDB\xBA\x9F\xF6\x7A\xCA\xF2\x2F\xA9\x40\x52\xFE\xD5\x23\x5E\xD2\x3A\xDC\x4D\xE5\x79\xAD\x57\x02\x9E\x99\xCA\xB1\x51\x58\xE8\x68\x9D\xB0\xD7\x32\x7E\xBA\x0F\xA8\xEF\x78\x52\xFE\x02\x4B\x4B\xE5\x75\x9A\xB2\x66\xF9\x2C\x99\x69\xCD\xBA\x44\x8B\xAA\x23\xC2\xAA\x23\xC2\x4C\x69\xFE\x51\x9D\xFA\xD8\x07\x6A\x4D\xBE\xA8\x4E\x20\x5D\x67\xD5\x38\x23\xFF\x4D\xE1\x52\x3A\x8B\x4B\xD8\x30\x6F\x5D\x8A\x10\xD0\xFA\x12\xA9\x43\xC1\x03\xDD\x48\xFE\x41\x6F\xD6\x5F\x37\x18\xC7\xE9\x5B\x37\x8E\xAF\x6B\xC7\x71\xBA\x37\x0E\xBB\x23\x7C\x18\x04\xDD\x49\xFD\xDF\x2D\x45\xBC\x4D\x2A\x10\x10\xE5\x67\x38\x2C\x93\xED\xA2\x63\xBB\x10\x1F\xF2\x28\xE8\xAF\x47\xB9\xFB\x94\x8B\xD2\x01\x72\xCA\xBA\x54\x89\xEC\x2E\x85\xDB\xC6\x3F\x67\x2A\x27\x55\xB1\xAA\xC2\xE7\x73\x74\x09\x98\x8E\x3B\x5F\x1C\x76\x3E\x87\x90\x3C\x4E\xFC\x9D\x05\x27\x30\x95\x27\x38\x14\x0B\xB7\x18\x60\x8B\xEC\x7B\xB8\x5E\x84\xAC\x85\x8B\x59\xCA\xFD\x6B\xCE\x65\x60\xBA\xDC\x4E\x8A\x53\x65\xBA\x70\x32\xC3\x26\xB1\xA5\x29\x36\xA2\x17\x34\x22\xCD\x20\x41\x14\xC4\x14\x3E\xDB\x7C\x95\x10\xDB\x3D\x32\x17\xF4\xC8\x5C\x1B\xF5\xA1\x25\x73\x41\x8F\xCC\x05\x2D\x99\x33\xEF\xA7\xB0\xA3\x41\x27\x5D\x7B\x6F\x2E\xE2\x96\xBC\x65\x97\x73\x3E\xA7\x8B\x95\xFC\x25\x8E\x07\x70\xD1\x50\x1D\xA1\x11\x14\xAD\xC6\x5D\x26\xB3\x76\x47\xA1\x64\x77\x9A\x2D\x95\x49\xB7\xE5\xA2\xAB\xB1\x5B\x77\x59\x0B\x26\x39\x28\xBA\x3F\xF6\x4A\xAF\x73\xC9\x59\xFD\x96\x93\xFA\xCD\x45\xEF\xC8\x29\x5E\x38\xB5\x51\x1A\x52\x84\x82\x6C\xC7\xDB\x41\xCA\x2F\xC4\x2A\xDE\xD6\x17\xE4\x2B\x48\x00\x04\xE9\xA0\xC5\x93\x6E\xE1\xD2\x81\xB5\x37\x4E\x5F\x5F\x47\x0C\x2F\x21\xC2\x8B\x9E\xBA\x9C\x6D\xA4\x3A\x0E\x39\x11\x5A\x78\xA6\x8A\x0D\x9B\xA6\x0E\xC2\x6D\x72\x80\xCF\x94\x2C\xB6\x18\x46\xE8\xD6\x06\xBF\x49\x9F\x98\x33\x22\x9A\x4A\xF7\x17\xD4\x5A\x84\xAD\x51\x3E\x9D\x78\x1D\xE1\x22\x65\x33\x09\xC9\x9B\x7C\x9C\xAF\x79\x42\x8E\x52\xB2\xDD\x90\x38\xE6\xDB\x14\xC3\x36\x21\x82\x6C\x8A\xAD\xC8\x3D\x5B\x11\xC6\xDE\xE5\x95\xB5\xA0\x9E\x2D\x91\x8D\x56\xC7\xDA\xE0\x6E\xB5\xC4\x57\x39\x87\xF3\x63\xAF\x49\x7E\x55\x74\xAF\x80\xE3\xC1\x92\xB4\xC8\x74\xFB\xD9\xF2\x60\x35\xE2\x66\x0E\xE2\x69\xCD\x66\x1D\x7C\xFA\xDB\xB2\x81\xFC\x6E\xE2\x03\xCC\xDD\x32\x81\x04\x77\xA9\xA0\x17\xC8\x6E\x8F\xF8\x97\xB3\xDB\xE5\xFD\xF8\x3B\xB2\xAC\xC7\x10\xFE\xC8\xEB\x39\x0E\x9F\x68\x7E\xA4\x2E\xFF\x31\x6E\x47\xC1\xD9\x8F\xCF\x41\x68\xD5\x06\x84\xB4\x1E\x53\x1B\xAC\x6F\x6D\xB9\xC8\x8A\x25\x8C\xBF\xE3\x35\xF5\xF8\xAB\x82\x60\xE7\x85\xFF\x18\xF0\x6F\x10\xBC\x88\xC8\xCF\xF8\xFB\xDB\xA0\x39\x14\x95\xD4\x19\x62\x2A\x90\x1B\x93\x5A\x51\x20\x61\xB2\x56\x56\xD8\x24\x05\xA5\x62\x9B\x8E\xF2\x7B\xDC\x85\xBF\xB3\x1F\xE0\xF8\xB6\xA7\xD8\x04\xE1\x85\x74\xA2\xA8\x8D\x5E\x84\xC0\x1F\xA1\xE1\xDF\xC4\x20\xA8\x06\xE5\x74\xF9\x7A\x06\x36\x69\x85\x05\x0A\xF0\xFA\x7F\x6E\x9E\x03\x69\xE5\xC6\x84\xB5\xAC\xD2\x8D\x81\xF5\xAA\x10\xD2\x90\xB9\x77\xE6\x6E\xFE\xDB\x23\xEE\x88\x70\x79\xD4\xE8\xEE\x45\x4D\x39\x5F\xCC\xD4\xDF\x10\x6A\x08\xC9\xDA\xD1\x67\x20\x24\x10\xF6\xE3\x76\x60\xDC\x46\xF5\x07\x0E\xE0\xC4\xFA\xBE\x84\xD9\x16\x61\x93\xE6\x08\xA7\x31\x08\x38\x9E\xBF\xA6\xDC\xD0\x6E\x65\xBA\x5E\x35\xA8\x29\x5B\x3C\xF6\xF1\x45\x3B\x7C\xA1\xEB\x01\xBA\x18\x61\x8A\xA6\x1D\xD7\xA8\xE9\x9D\xA1\x70\x58\xC2\xEF\xFC\x11\x99\xD3\x8D\xE0\x3D\x81\xE6\x4B\x81\xC0\xFC\x10\x52\x26\x56\x9E\x1F\x6C\x95\xE7\x30\xA7\x3C\xCF\x29\x8A\xAD\xBB\x2B\x29\x9D\x9A\x5C\x74\x2E\xA3\x20\xF1\xE8\x91\xCA\xC7\xA4\x6D\xB5\x32\x83\x63\x0A\xAB\x90\xD6\x84\xCC\x31\x95\x8B\x6A\x9A\x73\x54\x43\x81\xE4\x4C\x0E\xD4\xF9\xBE\x99\xA0\x1F\xFE\xC5\x5C\x8C\xF0\xA8\xE2\x5C\xFD\x56\x34\xEC\x81\x2E\x39\xFA\xC3\x80\x0C\xB1\x23\xB1\x42\x88\xD4\x2F\xE4\xD0\x68\xAD\x21\xBD\x3D\xDF\xD4\xA1\xD5\x7C\x1E\xF3\x14\xF4\x5A\x10\xD8\xEF\xF8\xD0\x7B\x7F\x27\x3C\x5D\x4B\x7B\xF0\x21\x90\x67\xDD\x23\x28\xF7\xE3\x6C\x25\xF9\xE6\x45\x43\x88\xC4\x32\xB4\x7A\xA3\x10\xEA\x28\xBD\xEC\x55\x0F\xB1\x7A\xD8\x56\xD7\x7B\x7F\x92\xBE\xE5\x41\x0F\x14\x5C\x49\x51\x9E\x6E\x8B\xE8\x61\x0C\xA2\x50\x2B\xEF\x73\xCC\x5E\x25\x05\xC5\x46\x3B\x14\x48\x5B\x9E\x03\x79\x16\xD7\x91\x32\x03\x25\xFE\x49\x92\x67\xB3\x6F\x9B\xDE\xD8\xA2\x37\xB1\x5A\xDB\xF1\x43\xA0\xCF\xBA\x28\xCD\x65\xD3\x46\x28\xCE\x29\x75\x2C\x68\x88\x49\x64\xD2\x96\x92\x23\x9F\x22\x2B\x52\x8A\xA7\x1C\x23\xF9\x3F\x4D\xC4\xAB\x1F\x74\x85\x22\x01\x84\xDD\x3E\xC8\x36\x10\x40\xF8\xF4\x3E\xF4\xF6\x21\xB2\xD1\x39\x3C\xD5\x1E\x6A\xF6\xD8\x92\xC6\xFE\xF8\x9F\xFD\xF6\x9F\x7C\x63\x03\xD1\x59\x4A\xC1\xCC\xD9\xDF\xF8\x96\x35\x32\x3F\xA2\xF8\xCE\x6A\x18\x39\xA5\xD6\xE0\xD6\xBF\x8E\x7A\x11\x80\xBA\xA0\x2A\x6A\x42\x92\xFE\xD8\xF4\x62\xE5\xA8\x09\xE5\xC9\x14\x6D\x1C\x3D\x3C\xF9\x8E\xE3\x6A\xBA\xA4\x8A\x14\x42\xC6\xD7\x73\x96\x4E\x31\x95\xE7\x7B\x72\x97\xA5\x1E\x28\xDB\xB1\x70\x4E\xFD\x13\x50\x10\xB1\xEE\xEB\x38\x57\xED\x41\x98\xB3\x8D\x8B\x37\x49\xF3\xE9\x42\x19\xE6\xB5\xDE\x3C\xD1\x16\x8D\x91\x4B\x0B\xAC\x18\x1B\xE5\x79\x38\xCE\xF0\x48\xF3\x14\x7E\x3A\x2E\x1E\x35\x47\xCA\x6E\x83\x41\xD4\xB2\x9D\x27\xA7\xE3\x21\xF0\x9C\x54\xFE\x96\x59\xE1\x81\xC3\xB9\xE1\xC2\x36\xA7\xAA\x6A\xAC\x68\x70\xBA\x12\x14\xFE\x5D\x2F\x14\xF8\x8C\xAA\xEB\x6E\x8A\xA1\x63\x62\x13\x5A\x00\x8A\x92\x01\xE1\x60\x96\xC4\x54\x51\xF0\xB4\xD0\x8B\x74\xD1\x7A\x21\xB0\xAE\xF9\xBD\x58\xAC\xB0\xEF\x3D\x13\x53\x10\xED\x55\x21\x47\xEC\xF7\x96\x7F\x7C\x61\xC3\x57\x76\xC1\xDC\x95\x1D\x07\x71\x05\x41\x01\xCE\x59\x57\x44\xA1\xD6\x39\xC2\xBA\xE0\x38\xFD\x89\xBF\x97\xF5\x7E\x42\x6D\x44\x75\xFF\x7C\x7E\xE6\x79\xF0\xE0\xC3\x60\xD2\x5D\x9C\x1A\xC4\xE9\x6F\xFD\xF6\x29\x44\xBF\xEA\xF5\xD0\x77\x0E\xEF\x42\xAB\x83\x4F\xCF\x26\xBD\xD6\x9C\x92\x41\x31\xB0\x89\xAF\xC7\x5D\xC3\x03\x3B\xA6\x39\x38\xEE\x90\xC3\x5B\xE3\x1F\xE2\x0C\x83\xF6\x48\x65\x8B\xA8\xC8\x9D\x99\x92\x43\xCC\x13\x20\x9F\xAA\x52\x97\xD2\xD5\x71\x2F\x29\x9D\xD8\xC9\x94\xAD\x21\x38\x81\x1E\xFE\x39\x43\x51\x1F\x89\x3D\x89\x29\x20\x5E\x6B\xE8\xE5\x8D\x72\x02\xBA\x65\xAF\x33\xE6\xCC\x5C\x50\x88\xA0\x56\x78\x46\xB3\x12\x56\xBA\x34\xE9\x55\x04\xAA\x8A\xC9\xC1\xC3\xA6\x4D\x95\x73\x92\x35\x7D\x28\x48\x08\x18\xEB\x82\xC1\x71\x34\xA9\xC6\x58\x55\x1D\x85\x31\xA8\xBA\xC4\x42\x25\x16\xEA\x63\xE6\x7A\x21\xA1\x5C\x93\x25\x94\xA4\xBE\x60\xC4\xA4\x21\x8D\x8A\x00\xF2\x4A\x42\x51\x85\x30\x26\xCC\x24\xBC\x26\xBC\x5C\x2F\xC2\xD6\xF1\xC6\xD9\x7F\x3A\x5B\x11\x02\xD8\xB1\x1B\x7D\x09\x39\x8C\x10\x27\x9D\x3E\x1A\x39\x63\xBD\x26\x0F\xD6\x4B\xF8\x67\xB5\x96\x9B\xD8\xF7\x41\xFC\xCF\x6A\xBD\xBC\xB9\x4E\xFC\xB5\x84\x25\x96\x1D\x86\xA4\xA4\x5E\x69\xB1\x6F\x15\x96\xF1\xD3\x32\x67\x68\x77\xB7\xB1\x75\x38\x59\x2F\x34\x7E\x49\x60\x19\x27\x12\x4E\x9C\x59\x0D\xE2\xDD\x6A\x9B\x93\x91\x10\xCD\xE1\x5C\xE8\x71\x2E\x86\x95\x19\x9C\x5B\x75\x1A\xD4\x96\x5C\x41\x08\xAB\x6E\x16\xA1\x81\x65\xC4\xB7\x65\xBA\x19\xA8\x25\x2C\x31\x81\x2C\x2B\x83\x53\x09\x38\x54\x6B\x59\x51\x98\xC7\x12\x85\x82\x92\x54\xC8\xBC\xCB\x59\x43\x97\x4D\x1D\xE0\xD3\x0B\x77\x91\x1B\x39\xDE\x0D\xE5\xC9\x2E\x68\x81\x43\x4C\xEF\x03\x33\x83\x98\xA4\x40\x37\xE6\x33\xBD\x1B\xF4\x77\x88\x41\x3C\x83\xBE\x52\x97\x48\x54\xF9\x73\x8F\x70\x06\x52\xE1\x63\xAE\x92\x81\xAA\x0B\x72\xB8\xD8\x45\x4B\x3B\x42\x11\x70\x24\xE1\xB7\x08\x4A\xB7\x2F\xB6\x8E\xC8\xCB\x82\x2F\xF4\xDF\x26\x38\xAF\x04\xBB\x66\xE1\x5B\x7B\x59\x90\x4E\x2C\x70\xFE\x59\xC1\x8C\x33\x81\xF4\x11\x2C\x8E\xE1\x58\xCB\xBB\xD5\x77\x0B\xFE\xAD\xEF\x56\xAF\xC5\xFA\xE2\x9E\xE0\x5B\x05\x5F\xFD\x7D\x8B\xE0\x7B\xBE\x47\x05\xB3\x96\x6F\xEB\xDA\xC7\xB3\xE8\xB2\xA8\x03\x26\xC2\x1C\xD3\x59\xAC\xE1\x10\x03\x0E\xE3\xC6\xF1\x97\xCB\x86\xF8\x5F\x10\xF6\x51\xD1\x94\x9F\xBC\x44\xFE\x45\x6B\xF2\xF0\x71\xBF\x1C\x20\xCB\x3F\xBE\x44\x92\x27\x35\xE8\x03\x68\x60\x5F\x9D\x33\xCE\x3B\x5A\x45\x46\x4F\x3C\xFD\x2B\xD5\x2D\xFE\xE1\x99\x68\x98\x7B\xA9\xD4\xBD\x3F\xA5\xE6\xD0\x78\x33\xAB\xEF\x42\xE3\xE9\x99\xE8\x21\xEE\xCE\xFF\x8E\x36\x60\xC8\x41\x0E\x45\x0C\xEC\xC7\xAD\x69\xE9\x0F\x82\xB4\x07\xDD\xC2\xEB\x76\xE1\x3B\x5B\x28\x26\xF3\xBA\x35\x0F\x96\x6B\x12\x7C\x15\xC4\x38\xE4\xC9\x27\xE5\x0E\x5D\x21\x69\xB7\x04\xBA\x76\x31\x4E\x28\x64\x93\xB7\x65\xDE\x23\xC6\x89\xE2\x41\xE5\xED\x28\x5D\xA4\xE4\xB2\x1F\xDD\x84\x2D\x56\x71\x94\xCA\x05\x38\xC1\xA3\x2F\xE8\x19\x30\xFB\x00\x27\x72\x4D\x96\x34\x73\xBE\x77\x9B\xA5\x61\x47\xE4\x5D\xD8\xA8\x6F\x0E\xF0\xCD\x1A\x48\x14\x48\xB5\x45\x68\xFF\xB3\x4B\xBC\x74\xBA\x7C\xE4\x22\x6D\x9C\xDB\x70\x45\x6D\x75\xD7\x25\xAD\x82\x53\x76\x72\x09\xAE\x4D\x77\x8B\x72\xB8\x8D\xE5\xD9\xED\xFD\x5B\x7B\x71\x63\xA0\x87\x77\x36\x70\xCA\x1B\xD9\x53\xDE\x38\x09\x5C\x76\xCA\x1B\xD9\x2A\x6F\x10\xFE\x58\x79\x43\x20\x43\x21\x66\x6A\x85\xAB\xEE\xB1\x58\xF1\x72\xAB\x19\x94\x0C\x39\x4B\x40\xDE\x62\x24\x22\x40\xE8\x96\x9B\xB8\x09\x5A\x6E\xE1\x96\x3B\x44\x0A\xC2\xC6\xF6\x3D\xAB\xD6\x90\x0E\x53\x5A\x6E\xF6\xE4\xBA\x78\xD1\x67\xAA\x13\xDD\x2A\x89\x6E\x95\x3A\x45\x91\x5A\x93\xAB\x2E\xA9\x8C\x47\xE4\x83\x2D\x1E\x97\x1C\xE9\x17\x51\x38\x67\x0C\x4E\xF8\xA6\x7E\x64\x3F\x1D\xB4\xCC\x07\x11\x90\xC0\x07\xD2\x76\x93\xAF\x38\xE3\x3F\xB9\x23\x93\xAB\x60\xE2\xD1\x56\x21\xDA\x12\x15\x74\x08\x2B\x3B\x17\x0A\xF6\x78\x9C\xDF\xA7\xCB\xA2\xB3\x8C\xB9\x45\xFB\xC4\xF4\x24\x70\x2A\x2E\xC5\xB3\x58\x25\x0F\x4A\x52\x4D\xD0\x88\x5F\x79\x91\xC3\x3D\x2D\x5E\xC5\xDE\x60\xCD\x5B\xBD\x8C\xA9\x5B\xF1\xD2\xB1\x96\x6A\x52\x6B\x17\x4C\x98\xC5\x4F\x64\x42\x2B\xE4\x0C\x28\x10\x21\x0A\x22\x4E\x51\x8A\xA2\x4D\xE4\x44\x1B\x0A\x00\x4E\xA5\x48\xC4\x09\x7B\x82\x88\x40\x41\x44\x74\x32\x4A\x4F\x00\x09\x39\x4D\x4D\x84\x2B\x19\x51\xCB\xEA\x28\x07\xE2\xE9\xAA\x47\x58\x3D\x6A\xAB\x8B\xBD\x3F\x2D\x12\x71\x5C\x46\x8C\xBE\x88\x03\x8A\x92\xA6\x38\x38\x8F\x7A\x12\x8E\x18\x48\x38\x62\x46\xC2\x11\x43\x09\xC7\x06\xE4\xBB\xC6\x1E\x86\xAC\x0B\x18\x88\x39\x02\xC5\x1C\x71\xD6\x91\xE5\xA8\xE5\x03\x15\x75\x39\x75\x76\x91\x55\x82\x2F\x98\x73\x8B\xCE\x54\xCE\xD2\x08\x12\xAF\x25\x14\x6E\x0F\x83\x4E\x3F\x21\xFB\xEA\x64\xC1\x61\xCC\x15\x2B\xA4\x24\x37\x25\xB1\x29\x1A\x73\xE2\xD4\xC9\x1C\xFF\xB2\x65\xDB\x02\x2F\x45\xD0\x75\x0C\x56\x2C\x88\x7C\x6B\x97\xB4\x48\xE0\x71\x99\x3A\xC6\x45\x32\xE3\x22\x5D\x42\x06\x8A\x33\x94\x87\x86\x7C\x7E\x29\xAC\x5E\xE9\xD0\x1D\x45\x10\x4A\x8A\xE9\xE4\x05\x65\xDC\xC8\x7D\x92\x4B\xE4\x65\x17\x0C\x1C\x29\xEA\x14\x92\x13\x6C\x39\x30\x33\xD4\x68\x6E\xA8\xC7\x79\xAC\x37\xE8\x9A\x4D\x02\x90\x7F\xBE\xC1\x7C\x04\x44\xBD\xF9\x28\xD3\x1B\x2C\x72\x65\x82\x72\xF5\x9B\x7F\x63\x54\xBA\x2D\x2F\xC8\x57\xF4\x78\x44\x8A\x34\xAC\xBD\x1C\x5A\xE9\x1E\xF3\x16\x4E\xEA\xB8\x92\x10\x55\x09\x81\x56\x8B\x2B\x11\xE2\x4A\xE2\x70\x25\x42\xD1\x89\x4A\x91\xD6\x2B\xDE\x5B\x1D\x10\xB5\x90\xEC\x9C\xB4\x25\x24\x55\x48\x7E\x6F\x8C\x2B\x72\x88\x2B\x09\x56\x4F\x3A\x84\xD8\xFB\x93\x9E\xC3\x15\xED\xC3\x88\x87\x88\x2B\xA1\xC3\x95\xB0\x4A\x3A\x5C\x49\x7A\xB8\x22\x1D\xAE\xE8\xB3\x95\xA6\x27\xC2\x15\x7E\xE2\x75\x6A\x71\x85\xC2\x69\x17\x28\xF3\x77\xD3\x1C\xE3\x34\xDB\xE0\xBC\x0B\x74\xEA\x9A\xB1\x25\x61\xAD\xAD\xBB\xAF\xAE\x52\x7C\xB1\xC1\xEF\xCF\x54\xA9\xD3\x11\xA6\x1E\x5B\xA4\x3B\x46\xF9\x17\x47\x6C\x77\xB8\xC2\xC9\x04\x34\x82\x59\xC0\xEA\xCD\xBE\xAA\x51\x43\xBA\x5E\xB0\x3F\x69\x7B\xEE\xE3\x36\xD7\x19\x6F\xB4\x99\x90\xB8\xCE\xF6\xF6\xB8\x85\xC4\xE3\xEB\x3C\xA2\x8B\x38\x88\x8E\x17\x21\x9D\xEA\x8E\x31\x68\xC0\x4C\x50\xC2\xF7\x98\xC0\x7E\x0E\x9C\x55\x0E\x99\x21\x06\xC5\x18\xB4\x07\xC5\xCC\x99\x0A\xAF\xB3\xE2\x5D\x93\xF9\x8E\x9B\x07\x9B\x76\x92\x32\x3F\x9F\x9D\x09\x68\xC8\xA7\x38\x76\x79\x13\x63\x27\x3B\x17\x3F\x5E\x7D\xA3\xF1\xBA\x4B\xC7\xC8\x59\x4D\xE8\x27\x38\x78\xE5\x07\x4F\x11\xFC\x0F\x76\x0E\x7F\x7C\x35\xD0\xBF\x40\xD0\xF8\x6A\xE4\x13\xAA\x49\x48\x38\x05\x8A\x0B\x59\x57\x3A\x8D\xBF\x84\xE2\x6E\x97\x59\xF6\x49\xA9\xFB\xC3\xFD\xD5\xFD\x31\x6F\x51\x6C\xAC\x24\x85\x43\x4C\xBA\xF3\x78\x63\x82\x72\x1F\xC3\x8C\x6B\x12\xFB\x62\xCC\x0D\x9D\xAA\xDD\xD9\x77\x69\x07\xC7\xA7\x18\xB0\x90\xCF\x0E\x9F\x98\xBA\xBF\x1D\x44\xAB\xEE\x0F\x9D\xBA\x3F\x26\x75\x7F\xEC\xD4\xFD\x31\xA9\xFB\x59\x93\xE0\xC6\x10\xB3\x19\x80\xA6\x21\x73\xEF\x94\x65\xB7\x7C\xD5\xC5\x05\xEA\x7E\xCD\x8A\xF7\x84\x11\x8B\x4C\xFA\x49\xDD\x9F\x38\x44\x63\xEC\xEA\xD4\xFD\x84\x61\x75\xE2\xD5\xFD\xB2\xA7\xEE\xA7\x70\x8D\x49\xAB\xEE\x97\x37\x50\xF7\x27\x6D\xAF\xFB\xA8\xFB\xBD\x03\xA8\xEC\xA9\xFB\xA5\x53\xF7\xEB\x39\x75\xBF\x66\x75\x3F\x15\xF8\xFE\x48\x99\x6D\x71\x41\xF6\x0D\xBC\x2B\x0D\x9D\xE1\x51\xB8\x8F\x94\xE4\x0D\x8F\x50\x3C\x27\xDF\x12\x94\x74\x0E\x36\xE5\xFF\xE3\xD2\x23\xCF\xC9\xB0\xFB\x04\x84\x71\x43\xCF\xDB\x18\x23\x09\x68\x64\xDD\x4A\x5E\x01\xE9\x18\x66\xED\x18\xE6\x4E\x7C\xED\x85\x17\xA1\x71\x04\xBD\x10\x23\xA4\xF0\x70\x55\x48\x1B\xD3\x93\xA2\xE2\xE3\x45\x50\xFE\xD6\x23\xEE\x22\x94\xF8\x8B\x78\x4D\xFE\x83\x9A\xA2\x88\x66\x56\x9D\xB3\x63\x1C\x60\xCF\x7F\x10\x32\x48\x79\xE8\xB8\xD1\xEA\x1C\x72\x92\x97\x2F\xF6\x63\x60\xD7\x2E\x04\x2B\xA7\x79\x80\xA0\xA9\x0C\x45\x25\xAD\x72\x1B\x54\x85\x3A\x0A\x8A\x42\x41\xEA\x67\xAB\x83\xF5\xE8\xD8\xF6\xFF\x52\x04\x30\x82\xF0\xB9\x32\xF9\x5F\x0B\x4A\x21\xC7\xF5\xFC\x8D\x1D\xFB\x85\x44\x7C\x61\xC6\xDC\x37\xA7\xF2\x20\x9F\x17\x5C\x1E\x50\x9B\x9C\xA7\x01\x41\x2D\x2B\xFF\xF4\x12\x87\xB9\x75\x56\xE2\x50\xAC\xBB\x1C\x78\x8E\x22\xC7\x6B\xB2\xA4\x34\x1A\x40\xAA\x8F\xCD\x5A\x59\x0D\xCA\xEA\x93\x93\xF2\xDD\xAF\x62\xD5\x25\x0A\x82\xD8\x72\xF9\xAE\x57\x91\x61\x00\x42\xE9\xD8\x5E\xBF\xFE\xD3\xC1\xFD\x75\x39\xA9\x23\x52\xDB\x51\x56\x07\x24\x13\xF6\xBF\x5D\x57\x27\xB6\x78\x23\x48\xF7\x44\xB6\x43\x2F\xE0\xE4\x0F\x59\xF9\x11\x5A\xDD\x31\xC8\x4A\x41\x4E\x6C\x31\x9B\x8D\x92\x31\x3F\x45\x91\xAC\x28\xCD\x85\x15\x0D\x3B\x33\x71\xD8\x5A\x27\x6F\xD6\x85\x13\x7E\xC6\x50\x40\x50\xFE\x07\xF2\x22\xF9\xD5\xD6\xC6\xA4\x52\x83\x34\x30\x30\x76\xDF\xA4\xCF\x88\xD3\xD4\x0A\x89\x28\x8E\x07\x4B\xC4\x4E\xD2\x4C\x6E\x37\x1E\x14\xBC\x4E\xAC\x0D\xF3\xDA\x09\x29\x61\x27\xA4\xFC\x9B\x44\x8C\x9F\xD6\xD6\xFE\xCD\xD6\xD6\xEE\xA1\xAF\x75\x49\x4B\x6A\x45\x15\x0C\x72\x0B\x33\x8A\xD9\xDC\xB3\xCD\x55\xE1\xD8\xFB\x02\x59\x66\xD2\xE2\x8E\x58\x55\x2B\x8F\x42\x01\xAA\x1E\x63\xC5\xF1\xAC\xA2\x03\x79\x8B\x71\xA5\xF2\xD8\xC0\x18\x89\xCD\xB8\xA7\xB3\x85\x51\x25\x21\xAF\xC2\xF6\x32\xD4\x5F\xA0\x50\x8D\xC4\xEC\xA1\xAF\x35\x30\xAE\xA8\x35\xA7\xB5\x1C\xF3\xC0\x15\x85\x75\xF6\x03\xE7\x61\x1B\xBA\xCA\xC0\x21\x07\x34\x4A\x32\x6D\xC1\x89\x46\x2E\xA9\x4E\x08\xE6\x78\x35\x26\x20\xAF\x4A\xAF\xB4\xD6\x95\x01\x59\x68\x4A\x48\x63\x5C\xA0\x58\xBD\xA8\xFD\x49\x11\x12\x4F\x35\xA3\xE5\xD6\xBC\x28\x25\xF9\xD6\x81\xAA\x25\x4F\x5C\x4C\x27\x50\xB4\x77\x20\x88\xF4\x64\xD0\xCD\x27\x9D\xF2\xE6\xA7\x55\xC8\x9D\xE4\xA4\x0F\x3F\x4E\xD4\xC1\xE0\x02\x63\xF7\x39\xF6\x3C\x9A\xB0\x59\x8E\xC1\x2F\x6C\x22\xB2\x70\xF6\x13\x52\x80\x77\x53\x10\x8B\x0B\xF5\xB2\xE3\x46\x0C\x87\x4F\xB1\xD6\x77\x84\x14\xED\x7F\x9F\xDA\xF7\x6C\x37\xF6\xF6\x0D\x63\xF0\xD1\xFD\x3E\x2C\xA4\x8F\xB9\xFC\xB2\xA6\x16\xF6\x19\xA7\x28\xA5\x91\xB4\x02\xC4\x39\xFB\x3D\x3B\x3B\x7A\xCA\xB4\xD4\x9C\x20\xB9\xD4\xFC\xB9\x52\xE2\x82\xDA\x8E\xCE\x43\xF0\xEE\x5A\x5C\x6D\x4D\xB1\xC4\x31\x78\xF4\x6A\xAD\xED\xF5\xEB\xF7\xDE\xEF\x18\xF8\xEB\xD7\xAF\x5F\x8F\xA7\x20\xB7\x8A\xE0\xBE\x80\xFE\x3D\xF0\xD3\x36\x78\x8E\x7A\xC7\x5B\x5E\xBD\x13\xBC\x99\xAF\xDD\x8E\x6D\x7F\x0D\x11\xFF\xE0\x31\x57\x24\xC0\x2F\xD0\x56\xF8\xAA\x2B\x75\xF0\x6E\x6C\xBA\xD2\xF6\x7E\x4E\x8A\xA5\xED\xCE\xCE\xB5\xEB\xF1\x49\xAE\x77\xD9\x65\xFE\xE4\xB7\x78\xF6\xB7\x0D\x21\x0B\xC0\x69\x3F\xEC\xEA\x37\x80\x6A\x68\x48\xF7\x4E\x6B\x6D\x1F\xFF\xC9\x2F\x6E\x6A\x69\x57\x1F\x6A\xEC\x23\x0F\x37\xEF\xAC\xA3\xFB\x82\x17\x7D\xEF\x03\x2B\xF7\x7D\xE8\xDE\x2B\x75\x4C\xFA\x9C\x6B\xC1\x14\xF4\x16\xB7\x79\x76\xF7\x18\x7C\x13\x04\xEF\x3E\x86\x13\x1A\xBF\xF2\xD2\x7B\x5C\x0F\x9F\x7A\xCF\xE5\x9A\xC6\xE0\x9E\x3F\x7A\xEF\x95\x2B\x75\xF8\xD8\xBB\x49\xBD\x75\xE6\x95\xEF\xA9\x93\xFB\x82\x00\x0E\xBC\xFC\xE7\x7F\xEB\xDE\x2B\x75\x7A\xB9\xCE\x20\x85\x18\xB2\xC7\x2E\x43\x00\xC9\x63\x10\x3E\xD6\xAB\x1B\x1C\xBD\xFC\xE6\x3A\x20\x6D\x5B\x80\x63\x88\xAF\xB8\xA6\xBF\xF7\x67\x1F\xFE\x60\xFA\xC6\x9F\xBE\xF7\xCA\x7D\xFF\xF6\xE1\x6F\xFD\xA2\x9F\x99\xFE\xC2\xBD\x97\xAF\xDC\xA7\xFF\xEA\xBB\xDE\xF0\x86\x37\xFC\xE6\xBD\x97\xAF\x40\x0C\x5C\xEE\xBE\xBF\xF7\x95\xFF\x71\xF9\x4F\x7E\xEA\xDE\x2B\xF7\xFD\x8E\xFA\xF9\xEF\xFC\xDA\x13\xEF\xC5\x72\xFF\xF0\x75\xF5\xEB\x9F\xB5\xF1\x2B\xF8\xF3\x75\x5F\x83\xFF\x3E\x7E\xEF\xE5\x2B\x97\x2F\x5F\xB9\x5C\x07\xDD\xD0\x20\xBA\xEF\xC8\x73\xFE\x64\xE9\xCF\x3E\xFB\x0F\xBF\xE2\x0A\x8E\xEC\xF2\x7D\x1F\xDE\x7D\xE3\xFB\x9F\xF7\xFA\x77\x7D\xC5\x95\xCB\x97\x2F\x5F\xA6\x18\x18\xE6\xEF\x0A\xD1\x73\x18\xD3\xDE\x4A\x8F\x35\x8A\xC4\x4C\x82\x2A\xBF\xF9\xA2\x0F\xCF\xD6\x4B\x67\xF4\xA1\x2F\x0A\x33\x4A\xEC\xAC\x2E\xF4\x35\x6B\xAE\x09\xFB\x6A\xFA\xEF\xA3\xAC\x2B\x9F\xB8\x84\xA0\xDB\x20\xAD\x6C\xAA\x98\xB3\xC3\xB2\xEC\x9E\xB0\x71\xE1\x17\xAF\x93\x00\x97\xC8\xA3\xEA\x28\x02\x6A\x8F\xDE\x22\x8F\x5D\xA7\x77\x51\x58\xF9\x17\x35\xA3\x25\x19\x45\x2A\xE2\x7F\x22\x0A\xA2\x48\x47\x28\x22\x48\x4E\x66\x2C\x73\x69\x6C\xC2\x66\x3A\x25\x82\xB6\xBB\x37\x4E\x40\x6E\x55\x49\x4E\xC7\x13\xC5\xF6\x78\xA0\xB1\xD7\xC5\xD4\xA6\xA7\xD8\x1D\x92\x44\xF7\xEC\x34\x64\xF8\xB6\xB1\x0F\x20\x9D\xE4\x30\xEF\x1C\xCF\xDE\x15\xCF\x36\xC8\x58\xD6\xCB\x05\x1C\x26\xE4\xEF\x70\x7C\x83\x98\x8D\x27\xE5\x16\x24\x2E\x2A\xBC\x0D\x4E\x4E\xF0\xF4\x01\x69\xEF\xFC\x86\x3A\x6B\x20\x3B\x4B\x51\x56\xE8\x04\x8C\x0F\x05\x41\x95\x91\x23\x84\x1B\x12\x76\xF0\x25\xD8\x56\xCC\x4C\x3F\x0D\x3D\x85\xC4\xEE\x24\x5B\xA4\xBF\x88\x0F\x05\xA2\x9E\x1D\xBA\x51\x47\xC1\x0C\x87\x6E\xF6\x1A\x3A\x29\x52\xB8\x23\x4A\xE4\x40\xC3\xCC\x6C\x00\x99\x0D\xEE\x9F\x90\xE1\xB8\xC4\x2E\xBB\x21\x4A\x1C\xA2\xA8\x72\xBB\x23\xAB\xB8\xB7\x29\x19\xA7\x64\xB0\xDF\xBD\xDD\xD8\x3B\x5F\x36\xCA\x84\x0C\xB4\xD6\x0A\xFF\x87\xC8\x9C\xF2\x48\xFF\x88\x43\x3F\xB4\xF3\xA0\x3D\xD0\xB8\x5D\xB9\xDD\x89\x7C\x10\x08\x57\xF8\x13\x33\x85\x05\x16\x56\x5C\xF8\x9A\xE0\x25\xDF\xD1\x78\xDE\x27\x10\x6F\x55\x09\xA4\x90\xF3\x24\xAB\x6C\x71\x96\x2B\x37\xC8\x3A\xB6\xFF\x5B\x33\xFA\xFB\xD1\x7C\xAC\xEB\xC1\x3F\x29\xE4\xEC\xAB\xA0\xFB\xA5\x02\xFC\xCE\x2F\x82\x30\x8E\x35\xFE\x0E\x62\x04\xB6\xDC\x96\x33\x6B\x73\x5D\x4C\xEB\xCC\xBE\x6D\xBB\x19\x55\xC2\x93\xAD\x40\x06\xBD\x7F\xCA\xFF\x10\x94\x56\xE0\xA5\xD3\x2A\xB1\x19\xCD\xCD\x02\x02\x50\x66\x7F\x23\x38\x81\xC0\x13\x57\x39\x64\xF6\x4D\xDB\xCD\x28\x92\x42\x08\x21\x29\xE7\x76\x8E\x05\xCF\x4E\xAB\x04\x72\xAE\x74\x61\x0A\x09\x41\xCA\x74\xE2\x8C\x5C\x5D\xA5\x00\x2B\x05\x2C\x52\xDA\x1D\x3D\xF5\x4A\xCB\x84\x7A\x05\x65\xE3\xC6\x3E\x3C\xAD\x8B\x67\xAB\xA0\x1E\x1D\x0B\xFE\xDE\xA4\x1E\x63\x2B\xAC\x98\xC8\x60\xF4\x00\xD9\xB3\xEF\x56\x25\x8C\x60\x74\xEC\xDE\x57\xD7\x4B\xE7\x61\xE9\x55\xD5\x12\x8C\x2D\x4C\xED\xEF\x06\x67\xED\x5F\x38\xB1\x23\xA1\x9C\xB3\x49\x63\x0F\x43\x0E\x4B\xB0\x04\xE5\x4E\xBD\x0C\xE5\x85\x57\x5C\xAD\x33\x4A\x01\x97\x6D\x4C\x20\x6B\x90\xD1\x4E\x70\x93\x33\x2C\xF2\xF7\x0B\x01\x89\xBD\x73\xA3\x52\x90\x55\x09\x2C\x57\x4B\xC0\xD9\xEE\x0B\x97\xAA\x8F\x03\x1B\x81\x86\x0C\xBB\x7F\xF4\x2A\xC4\x80\x7C\xC2\xB8\xFC\x56\x96\x53\x18\x60\x64\x3E\x72\x33\x94\x73\x33\x44\x18\x5F\x99\xD4\xAB\xDD\xB4\x72\x58\xA1\x03\x64\x05\x56\x10\x31\x93\x06\x92\xB3\x55\x02\xAB\xFD\x19\x1D\x70\x9E\x73\x1A\x32\x9C\xD1\x01\x6C\x0E\xE2\x07\xEB\x31\xC4\xA7\x37\xEB\xBC\x9D\x50\x4E\x13\xCA\x48\xF6\xC2\x12\x1B\x85\x80\x0C\x27\x54\x40\x5E\x65\x30\xAE\x12\x28\xD8\x8C\x87\x71\xAA\x9B\x50\x8E\xBD\x3F\xE4\xE6\xB3\xBA\x68\x3E\x34\xA3\x6B\x01\x9B\x2A\xDD\x15\x04\x9C\xAC\x1C\x27\x54\xAD\x10\x86\xD0\x97\x15\x5B\x9E\xB3\xE5\x37\x54\x2B\xA6\x3F\xED\x2F\xBC\x19\x17\x37\x05\xA3\x4B\xF3\x30\xBA\xD4\xC1\xE8\xE7\x1D\x28\x6D\xB0\x17\x40\x16\x37\x04\xC8\x6C\x52\xAF\xF4\xB7\x27\xE3\xED\x99\xDD\x8F\xDE\xF0\x0F\x40\x06\xD9\xDE\xFB\x91\xF0\x7E\x64\xBC\x1F\x89\xDF\x8F\x04\xC6\x55\xB6\xE7\x7E\xB4\xA3\x5F\x59\x34\xFA\x01\xF8\x11\x99\xF5\xE0\x57\x67\xF6\xFA\x75\x35\x85\x8C\xA1\x70\xE2\x08\xB0\x9F\xE1\xDF\xF4\xC9\xE5\xA6\x85\xA9\x78\xB8\xA3\xF1\x73\x54\x40\x75\x5A\x32\x8C\x67\xE5\xCF\x06\x27\x26\xE5\xB7\x0D\xDB\x20\x39\x37\xB3\xB0\x45\x25\x7E\x9F\x12\x29\x25\x76\xA7\xDC\x72\xFA\x79\x57\xF5\x5D\x94\x45\x01\x09\x30\x35\x49\xFE\x11\xFB\x75\x5D\xBE\x76\xD8\x15\x9D\xC0\x31\xED\x86\x9C\xD6\x79\x11\x58\x59\xC5\xEE\xD6\x89\x05\xE9\x18\x65\xFD\x8C\x74\x5E\x17\x9C\x9C\x1F\xA3\x9C\x6F\x78\x95\x2D\x47\xB9\x4A\x28\xBF\x22\x84\x4B\xAF\xE4\xC4\xBB\xF8\x96\xAC\xCA\xE6\xDE\xC9\x05\xEF\x54\xFF\x1D\xB8\x50\x7F\x64\x1E\x48\x4D\x57\x91\x61\xF9\x32\x26\xFF\xAD\xB6\xD3\x5E\x9D\x88\x13\x9A\x64\x38\xE2\x8C\x37\xED\xAE\x20\xD8\xA3\x94\xDB\xCF\x9C\x14\xF9\x10\xBF\x80\xFF\xD8\xED\xB3\x0D\x4F\x14\xE2\x4D\xAB\xA6\xC4\x3E\xDE\xA0\x2F\xCF\xFD\x91\xB2\x93\xBA\xCC\x70\x79\x38\x8E\x5C\x15\x2F\xA8\xC9\x2F\x16\xAC\x4A\xB4\x60\x55\xA2\x99\x55\x69\x53\x59\x9E\x6F\xC8\xA9\x23\x32\x0E\x5F\x0A\x1A\x74\x81\x0C\x19\x45\x3F\x31\x76\x1B\xCC\xC4\x6E\x37\xD5\x18\x0A\xA2\xBA\x63\x5C\x38\xD2\x08\x1E\x0A\x02\x94\x31\x63\xFE\x1A\xF3\xA8\x29\x54\x12\x18\x84\xE8\x8D\x09\xC4\x7C\xD4\xD7\xE3\x09\xB5\x4B\x70\x71\xA0\x70\x44\x79\x85\x96\x27\x3B\x85\x47\x03\xA3\xAC\xC4\x35\x3F\x4B\x1E\x81\xF8\xBB\x5D\x38\xCE\xEA\xE9\x96\x20\x1E\xCC\x24\x71\xF9\x38\xDD\xFE\xC6\x38\xAD\x15\xFC\xB3\x42\xC1\x8E\x18\xC6\x24\xAC\x6C\xD2\x02\x3F\xC5\xEB\x68\x1C\x68\x65\xB4\x82\x2E\xDE\x12\xA2\x17\x2E\x65\xE1\x46\x20\x8F\xB2\xED\x99\x39\x5E\x48\x5C\xDF\xA6\x32\x06\x72\xE6\x00\xF7\x82\x0E\x6A\xCD\x03\x49\xDE\xA7\x23\x07\xE8\x42\x06\xB2\x17\xF0\x1F\x5A\xBE\xA4\xB7\x7C\xF1\x7E\x30\x4E\xC3\x25\xD1\x85\x27\x90\xE0\x92\x51\x5B\x9F\x07\xB8\x0B\x8D\x27\xDF\xD5\x16\x05\x84\xF3\xC7\x4E\x3C\xA9\xF3\x8E\x32\x27\x10\x1F\x67\x5F\x6C\x49\xA4\x58\x36\xF6\xB0\xBD\x1B\x62\x3B\x62\xE2\x1B\xDB\xF1\xB4\x69\x03\x3D\x22\x48\x94\x1B\x04\x99\xFA\x21\x22\x40\x77\x6E\x54\x19\x24\x95\x84\x6C\x31\x09\x4E\x6C\x60\x4B\x30\x36\x81\x7C\x96\x04\xC7\xC6\x7E\xF1\xCC\x0A\xB0\x62\x6F\xF8\xCE\xB8\x05\x35\xBE\x9E\xA2\xF0\x64\x58\xC6\x6E\x37\xF8\x67\x63\xD2\x5F\x88\x9E\x62\x35\x1A\xE3\xA2\x2F\xDC\x7E\xB7\xE1\x4E\x1D\xC6\x3D\xA0\x68\x68\x6E\xE3\x3B\x7B\x0A\xC5\x24\x40\x36\x10\xDC\x13\xA0\xC8\x18\x98\x4F\xC5\xEC\xB5\x90\x21\xAA\x19\x77\xA0\xA7\x9D\xE7\x6E\xE6\xB4\xBC\x29\xE0\xE6\xE7\x74\xCD\xB5\x49\x37\x36\x10\x02\x39\x16\x87\xF6\x4E\x77\x0F\x85\x07\x5E\x68\xB7\xCF\x22\x41\xCE\xED\x9D\x10\x6E\xD6\x05\xE4\xE4\xD5\x9F\x5B\xE1\x55\xFD\x90\x63\x49\x41\x49\x7F\x7C\xEB\x21\xA4\xF3\x4D\xA5\xD4\xD8\x0D\x9B\x0A\x21\x77\x4D\xB9\xA3\xA9\x74\xF1\x93\x33\x04\x06\xB6\x4D\xE6\x0B\x35\xBA\xC3\x0C\x21\x5D\xE7\xCB\xBB\x0B\x76\x1B\x77\xFD\xC4\xC4\x6E\x53\x8E\xE5\x49\xD3\x1A\xA3\xE3\xFB\x75\xD7\x08\x4C\x29\xF0\xF0\xA7\x11\xA4\x42\x0B\x1B\xE4\xD9\x76\xBC\x90\xF6\x37\xDA\x37\xA4\xA9\x8F\xAD\x44\x39\xFF\x63\xFC\xF6\xCE\x53\x9C\xBC\x35\x6C\x20\xE9\x22\x68\x19\xF7\xC5\x4D\x6F\xEE\x43\x60\x0F\xF1\x65\x7E\x11\xD8\xE7\x54\x78\x44\xEB\x69\x11\x58\xA0\x9F\x08\xD9\x0B\xDB\x94\x54\x27\xB3\xE4\xE0\x22\x49\x78\x0C\x5F\x40\xFB\x45\x8B\xE7\xAE\x04\x53\x08\x19\xBB\xE9\x4A\x31\xC3\x82\x74\xAD\xC8\x3F\x09\x6B\x05\x05\xDB\x0A\x3C\x28\x39\x44\x4F\x3C\x82\xB4\xF8\x1E\x31\x89\x4C\x11\xDF\xB3\x05\x35\xF9\x05\xE9\x35\x67\xDF\xC9\x05\xEF\x54\xFF\x9D\x0F\x21\x98\x21\xBE\x67\xBD\xB0\x6E\xCE\x45\x40\xF0\x95\x68\xE6\xEE\xF4\xE8\xBC\x6D\xEE\xA2\x6C\x61\xC9\xB0\x19\x41\x4E\xBF\x7C\xA1\xEA\x58\xFF\x1A\x27\xB1\xA0\x70\x6C\x2F\xF8\xD7\xC3\x51\xD3\x88\x69\x34\x17\xBA\xD1\x50\x04\x87\x04\xE4\x26\x43\x6C\x0A\xB2\xA9\x71\x99\x70\xAD\x63\xBA\xE1\x4A\x68\xA5\x5B\x5E\x61\xE1\xD0\xFA\xBC\x42\x4C\x0B\x6A\x03\x90\x4F\xC1\x82\x66\x8E\x9D\x69\xE8\xE2\x0D\xC9\xB5\xF9\xEC\x01\x15\x6F\xAB\xF3\xFD\x8B\xCB\xA4\x17\xE0\x53\x83\xFE\x27\xA4\xAA\x24\x1D\x99\xBD\xF8\xAD\x97\x76\x02\xAB\xF0\xA8\x2C\xBF\x9D\x48\x5D\xE8\xDC\xB6\xEE\xBB\x4E\xFF\xFE\xF3\xF5\x17\x93\x29\xBF\x95\xD3\x2A\x05\x49\xE7\x76\xEC\xC1\x31\x83\xC8\xB2\x2F\x21\x45\x04\x74\x9F\x23\xFB\xF2\xC6\x4F\x3A\xB2\xE7\x1B\x7B\x7E\xBA\x59\x21\x28\xFB\xE9\x67\xB3\xD3\xCA\x16\x4C\x3F\x5B\x30\xFD\x6C\x76\xFA\x08\x4F\x19\x85\xBE\x45\xE2\xE6\xFC\x0E\x16\xF4\xE2\xE1\xBD\x0B\x0F\x68\x0F\xCD\x0E\xE4\x0F\x82\x05\x23\xF9\xE3\x60\xC1\x50\x7E\x2F\xD8\x6B\x2C\x21\xA4\xC8\x03\x85\x08\x40\xD1\x0B\xD9\x34\xA8\x81\x6C\xB3\xCE\xEC\xCB\x39\x08\x63\x36\x03\x3D\x61\xBF\x0D\x17\x8B\xAF\x07\x3D\x06\x17\x52\x0C\xA1\x27\x9C\x5D\x96\x70\xC1\xF2\x85\x0B\x96\x2F\x9C\x83\x1E\x8A\xCE\x1B\x36\x64\xED\x95\x4B\x33\xD8\xF8\xED\xAF\xC5\xD3\x56\x5B\x85\x07\xE7\xB4\xCE\x27\x8B\xF6\x3F\xC5\x69\x06\x90\xE2\xB1\x10\x35\x55\xC1\xF6\x06\x90\x12\x92\x18\xD2\x2F\x12\x58\x20\x4A\xBB\x9F\x8B\xB6\xA7\x9D\xB8\xE0\xFD\xDC\xF6\xFB\x59\x30\x39\x8A\x20\xDC\xAC\xC2\xA7\x08\x7C\x42\x72\xAD\x22\xA0\xF8\x76\xCA\xB7\xFD\xA6\xD7\x5D\xDA\x09\x20\x27\x63\x7C\x3C\xC0\xFD\x19\x46\x0A\x5D\x47\x4C\x16\x40\x97\x61\x2C\x29\x38\xC4\x2C\xA4\x9B\x55\x8E\x94\xB9\xB1\x17\x28\x36\x30\x92\x92\x11\x92\xD5\x54\x11\xB1\x6B\x1B\x32\x14\x1F\xA3\xB7\x18\xFC\x21\x9E\x21\x57\x21\xB7\x93\xB2\xAA\x33\x07\x63\x65\x53\x9B\xA6\xA0\xE0\x06\x06\x3F\x15\xC7\xC9\x88\x46\x6E\x82\x69\x18\xFC\xB0\x3F\x30\x9B\x44\x1B\x89\x8D\xC6\xAD\xA3\x30\x15\x1C\x0F\x36\x45\xBC\x25\xB3\x80\x76\x6D\xC3\x41\xA7\x44\x64\x1B\x0A\xBA\x6A\xE8\x23\x1E\x4A\xF9\xE7\x1B\x28\x09\x2C\x83\x20\x08\xFE\xE9\x2F\xFE\xA2\xFD\x5A\x64\x21\xFC\xD3\xBF\x7F\xB1\xB3\x72\x10\x1C\xD3\x8F\x82\x57\xC6\x4E\x12\x45\x52\xD6\x5E\xBF\x80\x7E\x0C\x89\x5F\xFB\xFC\xE2\x49\x35\x76\xE7\x0E\x72\xA0\x3B\xC9\x14\x79\x8F\xEC\x85\x74\xE1\x54\xF8\x38\xDC\x89\x3D\x0C\xB9\x7D\xA8\xA9\x43\xE4\x52\xEC\x9D\x1B\x93\x9A\xA4\x9D\xB0\x65\x51\x6C\x3C\x6D\x35\xE1\xF1\x0B\x3B\xCE\x2A\xB4\xCF\x9B\x52\xB8\x61\xDC\x6E\x5C\xE9\xA4\xA9\x53\x5A\xC6\xDC\x81\x06\x79\x55\xB7\x87\x4E\xEA\x9D\xAC\x1D\x12\xE4\xB4\x65\x7D\x5C\x60\x65\x15\xF7\x35\x7E\x83\xBF\xE5\xB1\x44\x9F\xC7\x6F\xA7\x5E\xE9\xE2\xFD\x61\x6F\x71\x33\x86\xF4\x9D\x8F\xE1\x6E\x9D\xB3\x97\x1F\xBF\xB4\x13\x34\xCF\x51\x41\x5D\x5E\xA9\x97\xFC\x8D\x91\x7D\x31\x2C\xB5\x0B\xF2\xCF\x08\x30\x96\x7E\x80\x90\xD5\x7A\xE5\x34\x2C\x41\xF6\x63\x8F\xB5\x55\x3E\x7A\x2F\x52\x05\x28\x9D\xCC\xF7\x63\xFF\x14\x41\x8E\x75\xDF\xD6\x29\x5A\xBA\xA2\x2F\x66\xD1\x6E\x0A\xD9\xF1\x2D\xC8\x9A\x56\xDD\x8D\x22\x0B\xA4\x90\xF2\x00\xBB\xE2\x56\x9C\x9D\x36\x3C\x0F\xA8\x0C\x83\x73\x82\x30\xB7\xCC\xB7\x0D\xEA\x28\x2C\x23\x68\x67\x90\xD9\xEC\xC1\x7A\x64\xB3\xD3\x9B\xF6\xF0\x16\xF2\x4B\x86\x02\x17\xE3\xFB\x0D\xA2\xBF\xA6\xA9\x56\xA0\xA8\x0C\x8C\xAA\x0C\x56\xEC\x9D\x2E\xC9\x2D\x14\x4D\x4D\x2B\x4A\x6B\xB9\x42\x9B\xF6\x3F\x9C\xA2\xB8\xB2\x05\x73\x90\x45\x53\x8D\x60\x19\xFF\xAC\x40\x81\x54\x2D\x6F\x18\x1A\xA8\xFB\x95\x21\x6F\x97\xC1\x88\x99\x5D\x64\x62\x91\xF7\x2B\x20\x5C\xAF\x0C\x64\x55\x08\xBC\x67\xB9\x85\x8D\x6A\x05\xF1\x90\xE4\xBA\x15\x9E\x92\x69\xEC\x97\xB4\xA3\x0E\x79\xB6\xCC\x24\x52\xC8\x93\xB0\xC1\xD5\xB1\xD9\x4B\xEA\x8C\xA7\xD8\xB4\x3D\xA2\x40\x91\x36\x76\x85\x66\x89\x73\xC8\xC1\x20\xE1\x44\x8A\xE0\x58\x98\x98\x4E\x18\x32\x63\xF1\xD6\x5F\xDE\x4A\x26\xEA\xC3\x3F\xC4\x2D\xBF\x1F\x19\x30\x4C\xC8\x0D\xB3\xE8\x34\xC4\xC3\x10\x21\xF8\x9D\x65\xE0\x37\x9B\x75\x8A\x44\x90\x0D\xD9\x5A\x16\xDD\x38\x6E\x9F\xB9\x27\xCF\xE8\x86\x2D\xC6\x52\x57\xC4\xF2\x86\x74\x8E\x10\xCB\x4B\x51\x6A\x99\xE5\xC5\x9E\x20\x6C\x17\x83\x2E\xB5\xB1\x76\x46\x07\x0C\x62\xB3\x79\x01\x1D\x00\xC6\xAB\x02\xE8\x80\x31\x74\xC0\xA4\x38\x62\x62\x72\x09\x59\xDD\xCF\x45\x27\x6B\x7B\xFE\x0B\xA6\xA1\x48\xC9\xDD\x06\xB9\x03\x26\x7B\x0A\x68\x19\x87\x1D\xED\xF1\xBB\x8E\x8F\x72\x38\xCF\x56\xA0\xCC\xF6\x1A\x46\x9E\x84\x11\xDE\xB3\xBD\x0B\xB8\x03\xE4\x09\x84\x0F\x8E\x93\x31\x38\x2F\x28\x6C\x5A\xB6\x37\x9C\x3B\x47\x24\x0F\xCA\xB3\xBD\xC8\xB3\x04\x2D\xCF\xE2\x72\x94\x65\x4E\xC0\x20\x9E\xC5\x89\x17\x4F\x90\x6D\x09\x3E\x2F\x27\x04\xE9\x28\x3A\xB3\x24\x73\x31\x8B\xC5\xB6\x38\x2F\x2E\x88\xF3\xF8\x37\x5E\xCC\x07\x0F\x39\x5C\x12\x4A\xF5\xFF\x41\x07\x4B\x8F\xFB\x59\x70\xB0\xB0\x36\xB5\x8D\x12\xAF\xDF\x38\x73\xA6\xD4\xE9\xBB\xEB\xEC\x18\xB7\x71\x3D\x7E\x25\xDD\xBD\xEF\xEC\xEC\x7C\xFA\xDE\xF6\x9A\xFE\xD3\xEF\xB9\x7C\xDF\x3F\xF9\xAA\x97\xBE\xF8\xEA\xAB\x7F\xE5\xDE\x2B\x90\x1D\xFB\xD2\x47\xAF\xDA\xEB\xE3\x29\xDD\xFC\xDF\xF7\x99\x6F\x74\xF7\xFE\xF7\xFD\xC8\xFB\x5E\xF4\xCD\x6B\xFF\xF2\x67\xEF\xBD\x7C\xB9\x36\x33\xB4\xDE\xBC\xBD\xCA\x3B\x5A\x9F\x1B\xC8\xDF\x59\x9B\xFB\x6E\xFB\x9A\x1F\x7D\xD7\x0F\x6D\xCA\xA3\x57\x1C\x13\x66\xEE\x9B\xBE\xF9\x1B\xD3\x17\xBE\x2E\x3B\x7A\xA5\xA5\xB0\x0B\xDB\x1A\x75\x6D\x8D\x0C\x9F\x84\x78\x32\x42\x0A\x05\x8C\xDE\x79\xDF\x7F\xFE\xBB\x7F\xF5\xD9\x63\x5F\xF2\x7B\xEF\xB9\x72\xB9\x36\x60\x2E\xB7\x06\x02\x60\x1E\x03\x03\xE6\x0A\x45\x2F\xF7\xC7\xEC\x5D\x47\xDF\xEC\x7E\x3D\xEB\xE8\x65\xFF\xF3\xE0\xD1\xCB\x6F\x6E\x1F\x3E\x75\xEF\x65\x18\xD9\xEB\x71\xB3\x7B\xEC\x4B\xBF\xE9\x3D\x57\x6A\xC3\xC3\x68\x6B\xBD\xB9\x32\xEE\xE8\x33\x74\x32\x8F\x27\x55\x68\xB5\x0D\x21\xB7\xFF\x5F\xD0\xD8\xF7\x8A\x8D\x49\x55\x12\x95\xDB\xF1\xCA\x80\xF4\xBE\x43\x5F\x75\xC7\x27\x3F\x76\xE4\xF6\x7B\xFF\x19\x1D\xA4\xD4\xD6\xCE\xA5\x2F\xE1\x19\x86\x90\xDF\x5F\x8D\xB0\xCD\xB3\xD5\x98\xD4\xFA\x21\x8C\x1B\x1B\xC0\x68\x42\x74\x68\x27\xE1\xB0\x93\x25\xCB\x3D\x61\x97\x83\x23\x84\xB1\x0D\xD6\x37\x91\x40\x07\xD5\xA8\x05\x85\x14\xCC\x9B\x21\x85\x7C\x52\x9B\x37\x82\x19\xEC\x3E\x84\x96\x84\x1E\x28\x37\xEB\x31\x8C\x21\x3A\x35\xA9\xC7\x50\x4E\x60\x4C\x88\x5F\x2F\x59\xF1\xD0\x04\x62\x7B\xFD\xA5\x53\x07\x49\xCB\xF2\x28\x94\x45\x00\xCB\xCE\x40\xCC\xC2\xD4\x7E\x3C\x38\x0B\x02\x96\xFB\x20\x9F\x43\x6E\xEF\xFC\x86\x3A\x6E\x20\x3E\x5B\xC5\x50\xDA\x57\x34\xD5\xD8\xDE\x89\x27\x18\xD1\x81\x11\x85\x22\x87\x18\x8F\xC5\xD5\xDE\xB1\xB8\x82\x2B\xB9\x02\x63\xD6\xC9\x12\x9E\x8E\xB0\x14\x1E\x1F\xAB\x14\x2E\xFD\x4E\x1F\x4C\x3A\xE4\x83\x70\x04\xA5\xBD\xD0\x70\x4C\xEF\x11\x9E\x01\x25\x8C\x36\xED\x2B\x50\xBC\xBE\x13\x46\x9B\x75\x08\x31\xA5\x2F\xE9\x4E\x80\x31\xAC\xE0\xFF\x37\x26\x58\x1E\x3B\xC0\xD6\x46\xFE\x1C\x19\x35\xF6\x90\x7D\x0E\xE4\x36\x38\x39\xC1\x11\x8D\xB8\x08\xF9\xBB\x3A\x63\xB9\x11\x53\xC0\x91\xE3\xB8\x60\x44\x42\x9A\xA3\x5E\x03\xCE\x78\x99\xCC\x30\xB8\xFD\x96\x06\x86\x4D\x3D\x6A\x69\xE0\xA0\xF8\xA8\xA5\x81\xB3\xBC\x34\xD9\x0B\xC4\xC4\x52\xBB\x68\xF1\x4B\x24\xE5\x0E\x69\x60\xE8\x14\xF7\x23\xCE\x63\x45\x34\xB0\x27\x60\xC6\xC3\x16\xC5\x4C\xFC\xF9\x51\x8F\x06\x86\x0B\xAA\xF0\x8B\x05\x34\x30\x5E\x40\x03\xE3\xD9\x93\x85\xC3\xB8\xC7\x8B\x68\x60\x6C\xDE\xAD\x85\xDE\x76\xBA\xA3\xB8\xD3\x1D\x45\xAC\x3B\x0A\x9D\xB0\x96\xBA\xB3\xD4\x11\x77\xD2\xBD\x98\x4E\x77\x64\xF6\xD2\x1D\xF1\x41\xBD\x40\xD5\x91\xFA\xB3\xD4\x6C\x56\xE6\x29\xD2\x1D\x19\x3C\x4B\x4D\x77\x96\xC6\x0B\x55\x48\xA6\xA7\x42\x32\x37\xA9\x42\x32\xC8\x15\x2E\x54\x21\x65\xFB\xA9\x90\xF0\xB0\xE5\x41\xE1\x82\xB7\x2A\xA4\x0E\x92\x8C\x53\x21\x65\x94\x66\x81\x55\x48\xE6\xC6\x2A\xA4\x01\xE3\xDF\x41\xD2\x2D\x5F\x57\xD3\xAA\x90\x3C\x24\x99\x32\xB0\x97\xDF\x78\x69\x27\x28\xBF\xF3\x22\xD9\xBA\xE3\x8B\x1F\x9F\x7D\xF1\xCB\x83\x17\xE3\x80\xC3\xA5\x96\xAF\x62\xFD\xB7\x09\xA0\xFD\x9D\xE1\x6F\x32\x5F\xE6\x9F\x1C\x4E\x7F\x35\x00\xC1\x6F\x2D\xFE\x22\x62\x62\x72\xD2\x4C\x95\xBF\x45\x25\x1E\xD1\xC2\x6C\x03\x7B\xC0\xD3\x3D\xAE\x66\x87\xF7\xB0\x52\xA0\xCB\x47\xC9\x41\x8B\x1C\x45\x22\xD9\xE5\x89\xFF\x9A\x66\xA4\x44\xC0\x16\xFE\x11\x84\x64\xAB\x83\x3F\x59\x90\x45\x0C\xA5\x96\x3A\x5F\x24\x69\x5F\x4A\xDE\x04\x74\xED\x79\x98\x35\xC3\xF1\x21\x5C\xAC\x2D\xFB\x69\x8E\x6C\x7F\x9C\xC3\x60\xC5\x87\x02\x51\x69\x5B\x92\xC2\x14\xE9\x07\xF9\xA7\x1C\xAF\x42\x9B\x91\x95\x99\x6B\x40\x18\xB2\x52\x3E\x5C\x69\x4A\xE7\x09\xE1\xA4\x4A\x0C\x5B\x1F\xF2\x31\x0B\xD1\xA4\x4E\x21\x79\x90\x93\xD1\x78\x73\x41\x9F\x81\x86\x72\xFC\xA4\x90\x41\x72\x7A\xB3\x2A\xB8\x59\x32\x7A\xCE\x5B\x21\x28\x46\xF1\x82\x6D\xCD\x40\xD9\x07\x1A\x32\xA9\x02\x6D\x1F\x68\x9C\x51\x70\xF9\x4D\x6C\x64\x2A\xED\x86\xFD\xFE\x6D\x50\xE5\xAB\xF1\x79\x02\x9A\x2C\xD0\x13\x50\x64\xE9\xA3\x9F\x5F\x08\x22\xCC\x95\xB2\xDB\x8E\xB3\x56\x6C\x99\x6A\xF8\x30\xA4\x8B\xA4\xE1\xF3\x3A\x05\xE5\x34\x50\x80\x3E\xC9\xE9\xB3\x92\xD3\xD8\x2E\x59\x5B\x0D\x16\x99\xFD\x25\xDC\xC4\x38\x28\x48\x15\x63\x5B\xC6\x59\x8A\x03\x5F\x16\x42\x00\xF9\x84\x6C\x4A\x53\x30\x36\x00\xB3\x09\xD1\x84\x0F\x0C\x73\x77\xEB\xA8\x27\x68\x1D\x38\xD6\x4A\x70\x08\x91\x91\xCC\xF4\xC7\xCE\xB0\x3E\xE8\x59\x66\xDB\xC0\x6C\x90\x1D\x29\xC2\x12\x6F\xA0\x73\xDD\x0A\x68\x58\xEC\xA8\x4A\x81\xCB\x14\xE8\xCD\xB1\xE1\xA0\x2E\x62\x61\x4B\xEF\x12\x42\x6F\xFB\x41\x48\xAB\xA6\x3E\x53\x25\x15\x73\xF7\x89\x6C\x9B\xEE\xEC\xD2\x81\x72\x8D\x09\xFC\x13\x1D\x0A\x02\x7F\x0F\x45\xF7\x35\xED\xB3\x6C\xE8\x82\xAC\x7D\x56\x0D\x5D\x8E\xB9\xE7\x81\x5D\x3A\xEF\x09\xB6\x57\x49\x6A\x9D\xBA\x61\xD7\xB8\x5E\x36\x33\x8D\x7F\x34\x05\x12\xAB\xD9\xEA\x23\x30\x8F\x28\x11\x6F\x83\x74\x89\x50\xD4\x60\x1A\x9A\x85\x5A\x9E\x6D\xD4\x9B\x48\xCC\xEA\x20\xCA\x74\x57\x93\x57\x36\x5B\xAB\x79\x83\x06\x12\xF7\xDB\x67\xD9\x90\x46\xAA\x7D\x56\xF8\xDC\x4E\x24\x76\xAC\xBA\x37\xB0\x67\xCF\x7C\x14\xE9\x29\xF5\x47\x43\x1D\x85\x14\xAF\xF1\x9E\x8E\xF4\x39\x79\xC2\xC7\xCC\x41\x99\x70\x6E\xD8\xFD\x11\x0B\x37\xE2\x80\x47\xAC\x3E\x97\x11\x8B\xFE\x88\x79\xBC\x8A\xC7\x1B\x3C\x81\xF1\x2A\x26\x72\x10\x98\xFF\xDB\x85\x0D\x73\x33\xD8\x6E\x9C\x19\xBF\xB4\xF1\x94\x5C\xA0\x1C\xB3\x19\xB8\xA6\x45\x2F\x27\x0B\x15\x97\x83\x76\x6D\xCC\xFE\x9A\x6D\xC1\xB8\xF1\xBF\xA2\xF6\x57\xD8\xFE\xD2\xED\x2F\xD5\xFE\x92\x4D\xD7\x89\xFF\x95\x70\x77\x0F\xB7\xBE\x0D\xE6\x7F\x14\x6A\xDB\x05\xB0\x0A\x80\x00\x8C\x22\x2B\xA9\x4A\x10\xE0\x91\xFB\xB3\xDD\x6E\xCC\xAF\x2B\x11\xB1\x59\xAC\x5B\x2B\xFD\x42\xA4\x80\x17\x5F\xDB\x9E\x0E\xA6\x75\x96\xD2\x5D\x4E\x5A\x24\xCE\x41\x15\x83\xE6\xF8\xBE\x6C\x4E\x9E\x42\xD8\x90\x35\x3C\x9E\xC4\x3B\x7F\xF4\xFA\x37\x9F\x27\x5D\x0F\xA4\x1B\x90\xD9\xEB\xAF\xFB\xE4\x47\xD4\xA9\xAD\x3A\x9C\x90\xAB\x04\xDF\x1C\x4B\xF6\x65\x25\x97\xD6\x9E\xB9\x04\x22\x10\xA8\x8D\x29\xFB\xF2\x08\x72\xE9\xA8\xC3\xA6\xCA\x78\x13\xA5\x15\xB4\xDA\xD2\x06\x90\xB9\x54\x3E\x11\x44\x76\xE7\xF5\x9F\xFC\x88\x3A\xE1\x7A\x70\xF8\xE4\xDE\x92\x7B\x22\xAB\x8B\x5D\x9F\xD8\x81\x87\x6A\xBE\x79\xD5\xC8\x57\x29\xBE\x27\x42\xC2\x82\x87\x7C\x74\xBC\x1B\x80\xE4\x01\xC8\x86\xCD\x76\xA9\xEB\x23\x32\x98\x6D\xD7\xF7\xDB\xBA\xEE\xF0\x0D\x12\xFB\xA5\x44\xD4\x32\xE5\x3E\x0A\x20\x42\x31\x5D\xE0\x1F\xD9\x6B\x8A\x7D\x79\xFD\x73\xD2\xF8\xD8\x72\xFC\x9C\x37\x3E\x03\x80\x0F\xA6\x49\x84\xAF\x44\xC2\xD7\xB3\x5B\x08\xCD\xC7\xA4\x48\x7B\x80\x8B\x87\x90\x03\x5C\xC1\x1E\x6D\xBA\xCB\x55\x78\x61\xCA\x89\xD5\x41\x54\x31\x70\x04\xFF\x90\x78\xF3\xA4\xDD\xD6\xF8\xA6\xB6\x95\x58\x91\x8C\x56\xC6\x55\xE3\xB9\xCC\xD5\xAC\xD3\xB6\x6E\x6D\x7C\x6D\x97\x03\x8A\x12\x5E\x21\xB7\xAB\xAD\xEC\x39\xF7\xB8\xA4\x36\xB4\x15\x64\x52\xC4\x31\x4C\xB1\x17\x01\x09\xF7\x92\xB8\x5E\x70\x0C\x64\x15\xE6\xBB\x98\x90\xC5\x60\x9D\xD2\x41\xE5\x6E\x05\x48\x5D\x26\xA6\x4E\x4F\xC0\x79\x18\x11\x9A\x13\xCB\xC1\x0A\xC9\x10\x05\x25\x4C\x06\xAB\xB8\x9D\xA1\x8B\x3C\x6B\xCC\x9B\x05\x9F\x53\xA2\x08\xEC\x1B\xBF\xAF\xC3\x18\x97\xDE\xA1\x56\x9C\x2D\xDE\xCD\x5C\xA2\xF8\xB9\x01\xB2\x5D\x33\x3D\x61\x37\x0B\xC4\x3C\x2B\x29\x62\x85\xE4\xD0\x26\xDA\xC1\x98\x75\xEE\x5E\xBA\xA9\x29\xFA\x2A\x8F\x84\xEA\xF9\x47\x84\x6C\x07\x7A\x0A\xC4\x46\xD5\xE6\x68\x0C\x98\x02\x9A\xD3\x4C\xC0\xF0\xC4\xF6\x99\x5A\x89\x0B\x09\xDA\x1C\xFA\x0D\xA5\xCE\x77\x8D\x2B\x87\x3D\x6D\xE3\x34\x1A\x72\xF5\xDC\x20\xDF\x70\x3E\xAD\x98\x9C\x8D\x8D\x7D\x7D\x37\x6F\x2B\xCC\xC7\x35\x41\x5D\xC7\x8B\x68\xA4\x26\xDF\xDC\x95\xE9\x13\x11\x4A\xE9\xE5\xF3\x90\x39\xA8\x8C\x5C\x6E\x3A\xCE\xEA\x85\x90\x88\x84\x86\x6E\xAE\x40\xD0\x55\x07\xE2\x4E\x4A\xE9\xB9\xEA\x9C\xD1\x3F\x6E\xEA\x78\xB3\x2E\x78\xC8\x0D\x14\x90\xE3\x5E\xF3\x7E\xA5\x88\x51\x86\xA1\x10\xCB\xF3\xAF\x7D\xAA\x24\x4D\x95\x12\xEE\xD1\x65\x56\x42\xD2\x12\x9D\xF7\x33\xD9\xCC\x38\xEA\x6F\x9D\x22\x04\xA6\xD4\x43\x06\x92\x7F\x61\x0F\xE9\x66\x6D\x7C\x0F\x86\xC1\x3E\x66\x4E\x29\xF6\xCE\x79\xDE\xE6\x43\x73\x3B\x44\x60\x52\xE4\xC6\xB6\x69\x9E\x06\x62\x87\x4C\x0D\x43\x0D\xA4\x7D\xE2\x96\x16\xC2\x06\x76\x1B\xD2\x09\x59\x71\xB5\x97\xA9\x95\xB9\x49\xF2\x66\x7A\xE4\x2D\x6B\x28\x9D\x6B\xE6\x6E\x31\xC0\x87\x65\x4F\x29\xFF\xA2\x6F\xDE\xF5\xDC\x93\x6E\xFA\xE4\x2D\x63\xF2\x66\x88\xBC\x65\x9C\xF2\x2A\xEB\x22\xBC\x23\x57\x44\xED\xB5\xCF\x49\x43\xCB\xDD\x3E\xE7\x0D\x85\x04\x5F\x48\xDE\x3A\xD1\xD1\xFC\xB0\x14\xC9\xAD\x22\x6E\xB1\x4F\x06\xB7\x89\x44\x8A\x77\x2B\xA3\xE4\xCA\x7D\xA2\x96\xD2\x7A\x60\x71\x07\x41\x37\xAA\xF1\xE4\x08\x99\xA3\x62\xC4\x04\x6D\xD6\xDA\xB7\xAC\x21\x86\xC4\xB5\x4C\x0C\x3A\x47\x0A\xED\x13\x2C\x9E\x0A\x32\xE3\x0E\x4A\x86\xB4\xEA\xB8\x23\x56\x5A\x5C\x90\xE4\x3B\xC3\x6B\x55\x04\x03\xE4\x0D\x2A\x8A\xA5\xA5\x76\x69\xF1\x14\x90\xC3\x53\x15\x29\xCE\x69\x2C\x9E\x8B\xFB\x73\xA1\x8E\xAF\x36\x94\xB6\x5E\x6D\x90\x1B\x53\x7C\xDE\xB9\x27\x41\xB2\x7B\xE9\x18\x8D\x78\xA7\x8E\xAF\xD6\x8A\x89\xDE\xE9\x86\xB7\x91\xA2\xBB\x38\x6D\x7A\x7B\x32\x42\x7C\xB5\xEA\x48\x2F\x65\xC9\x31\x7F\x21\x54\xD2\x0E\xF1\x46\x54\x84\xDC\xE1\x8E\x17\xD2\xF9\x44\xD3\x38\x5D\xFA\xB0\x2A\x06\x81\xCB\x5E\xA5\xA0\x58\x46\x23\x8A\x91\x32\x05\x88\x71\x22\xC9\x73\x65\x70\xA1\x2E\xAE\x36\xF5\x08\x0C\x8A\xF7\xC5\x31\x78\x14\x46\x90\x6F\xEC\x9E\x07\x03\xA3\x8D\xDD\xF3\x38\x3B\xD3\xCD\xA9\xB8\x8A\xB8\xDC\xCD\xC9\xAD\x6D\x8B\x1C\x4E\x89\x02\x1C\xE1\x8D\x26\x54\x7C\x65\xC7\xE8\xF0\x2D\x2C\x07\xBD\x37\x24\xF4\x30\x00\x78\xFF\xEA\xC7\x84\x16\x34\x6F\x71\x61\xD1\xD6\x8C\x39\xC6\x99\x3C\x87\x28\x40\x8B\x49\x82\x44\x80\xBB\xD5\x09\x2E\x82\x43\x4E\xEA\x5D\x1E\x34\x6F\x96\x86\x68\xB0\x55\x9F\xA2\xD4\x94\xAF\x04\xBD\x7B\x09\xC2\x9D\xAB\x35\x7B\x3E\x42\x04\xEA\xF4\x26\x22\x89\x3D\xCF\x28\x77\x7E\x20\x4A\x81\x36\x3F\x5E\x84\xD1\xB6\xB8\xA0\xB6\xC5\x85\xC2\x2B\xE5\x5F\xAB\x37\xEB\xA8\x67\x9D\xC2\x11\xCE\x5D\xD0\x03\x63\xFF\xD3\xB7\x75\x67\x62\x1B\x6B\x03\x49\x1B\xC3\x36\x72\xE6\xC4\xF2\xB9\xB0\x1D\x56\x34\x0F\x56\x29\x44\xF6\xB5\x92\x54\x2D\x74\x2E\x5C\x0B\x98\xBA\x88\xC6\xEA\xF2\xDD\xE4\x37\xCD\x76\xDD\x11\x52\x05\x9F\xEC\x95\x4B\x85\xFD\x52\x86\xBB\x42\x69\xB7\x55\xF7\xCF\x81\x3C\x05\x29\x4A\x77\xD9\x7F\x1D\x32\x72\x89\x1D\xAB\xA3\x30\x82\x02\x24\x42\x4A\x7E\xA1\x2E\xAF\x12\x92\x42\xC1\xEB\x58\x0E\xD7\x31\xEE\x60\xA4\xBC\x8A\x27\x4A\x07\x23\xEE\x58\x1E\xE1\x9F\x11\x29\x71\xEB\x31\x85\xE9\xBB\x4A\x81\x53\x21\xDC\xC4\xE1\xF5\x42\x1D\x48\x17\x96\x74\x66\x90\xAA\x37\x48\x09\xA6\x2A\x20\xA4\x41\x16\xC8\x50\xF4\x06\x39\x82\x11\xC8\x85\x83\x1C\x0D\x06\x29\xFB\x83\x74\xD6\x93\x05\xFE\x29\x86\x83\xA4\x74\x3A\x88\x52\xF5\x12\x7B\x38\xB3\x57\xDC\x89\x82\xB5\x4E\xF2\x5C\xBD\xDC\x54\x2B\x54\xE6\x42\x53\xAF\x12\xEE\x1D\xC0\x87\x69\x75\x1B\x11\xAA\xEA\x19\x90\xB9\x2C\xFF\x07\xE9\x4C\x89\xEB\x67\x62\xB5\xDB\x9B\x7A\x02\x4B\x4D\x7D\x07\x33\x49\x2B\xE4\xB5\xF6\xA0\x3F\x8E\x4F\xC3\x04\x96\xE9\x34\x1D\x91\x02\xA6\x38\xBD\xD9\x42\x34\x4E\x53\x21\x7E\x2E\x98\xA4\xDA\xBD\x04\x72\x77\xE7\x6A\x5D\x70\xFA\x51\xAA\x59\x29\xC8\x60\x15\x9E\x49\x28\x59\x8D\x38\x65\x30\xC4\x8E\x07\x23\x89\xE7\x00\x2E\x61\xBC\x5B\xE3\x2A\xBA\x55\xBA\x5A\x17\xEC\x96\x5D\xAC\x17\x12\xFC\xA6\x42\x79\xB5\x81\xD1\x9A\x0C\x5E\x40\x86\x86\x28\xDD\xC5\x2E\x1B\x24\xB2\x7A\x1C\x26\x8D\x1A\xD8\xA0\x40\xA8\xF1\x6E\x95\x23\x7D\x1A\xD3\x9E\x1D\xE4\xA4\xCE\x24\x99\x73\xBC\x6C\x90\x90\x43\x41\xA4\x88\xB7\x6F\xB8\x79\x67\xFA\xFB\xD6\xED\x1A\x8C\x9A\x7A\xB4\x59\x7F\x91\x3F\x1F\xBE\x08\x34\x8C\x36\xEA\x11\x91\xFC\x11\x89\x27\x7E\x37\xDD\xDE\x8E\x91\xF8\x8C\x49\x79\x8C\x04\xEA\x76\x1C\x6D\xCC\xDB\x29\x59\xE8\xC3\xB9\xB4\xF0\xC7\xB7\x32\x9C\x1E\x26\xB6\x41\x35\xCA\x89\xCA\x2A\xFF\xA8\x8E\xC2\x41\x9C\x72\x01\x05\x45\x4F\x80\x11\xB2\x37\xAA\xA9\x47\xC4\x83\x0F\xF8\xDD\x71\x9F\xDF\x1D\xD3\x10\x0B\x4E\xE6\x4E\x5E\xD3\x05\x8C\x1B\xC2\x8F\xF9\x9A\x45\xBF\x26\x4F\xCE\x45\x45\xB8\x8D\xD4\x77\x64\x23\x9E\x4B\x33\x90\x55\x59\xCD\xF5\x0C\x9C\xD4\x84\x00\xAF\x56\x6D\x5F\xD2\x8F\x93\xFA\x52\xBE\x2F\x0E\xE6\x0E\x6A\xD8\x97\xA1\x5B\x0D\x6C\xE7\x0E\x1B\xC0\x1D\x4E\x1A\x50\x7E\x7F\x4F\x70\x99\x83\x48\x25\x0F\xC2\x33\x09\x12\xE0\x99\x85\x77\xF3\x14\x8E\x16\x55\x7C\x22\x41\xD6\x60\x41\x66\xBB\x99\xA4\x5A\x71\x92\xFD\xB9\x29\x0E\x0E\x7D\x25\x2F\x77\x32\x39\x0F\xEE\xF7\xC1\xD7\xE7\x69\x41\xCC\x29\x80\x42\x77\x16\xC8\x86\x42\xC0\xD1\xA6\x3B\x0A\x11\xBB\x50\x1C\x92\xA1\x6C\x04\xC5\x00\x7B\x8A\x3D\xB0\xA7\xD8\xBD\x04\xF9\xCE\xD5\x7A\x44\x7B\x04\x29\x8C\x48\x7F\xE9\xC6\x26\xF1\x4F\xCF\x85\x9F\xAF\x95\x89\x52\x53\x9A\x1F\x1B\xAC\xC9\x4F\x5C\x79\xF5\x4E\xB0\x74\xC9\x19\x50\x33\x81\x3E\x51\x50\xFA\xE1\xB9\xAF\x91\x7D\xAD\x6E\xEF\x3B\xC6\xF8\x7C\xE9\x3B\x28\xFF\x34\x9E\x98\x1F\xD1\x94\xB4\x6A\x4D\x5E\x7A\xEC\xD5\x3B\x81\x7D\xF4\xDF\x5E\xDA\x09\x20\x2A\xFF\x15\x1D\x95\xF6\x3B\x2F\x5E\xDA\x09\xEC\xB3\xAD\x68\x4B\x94\xAF\xE3\x2F\x59\xF7\xE6\xF5\xF4\xC6\x3B\xB6\x7F\x3C\x23\x76\xB1\xD5\x8F\x72\x10\x7C\xFB\xFA\x6F\xEF\x1F\x4F\x74\xDA\xA4\x0F\xD6\xA1\x7D\xD9\x69\x52\xFE\x21\x66\x70\x96\x4C\x87\xE8\x0A\x59\x66\xC6\x18\xDC\x55\x41\xBB\xAA\x49\xBF\x84\xCF\x01\x3D\x47\x94\x7C\xB8\x0D\x57\xC3\x45\x22\xFA\xC2\x2D\x0D\x82\x50\x10\xF3\x2C\xDA\xF6\xCE\xCC\xB4\x77\xC6\xB5\x87\x7C\xB1\xEE\xF8\xE2\x08\xF9\x66\xDD\xF1\xCD\xC8\x64\x70\x44\x70\xF7\xDC\xEB\xFF\x0C\xF7\x7F\x86\xFB\x3F\x4F\xCA\x45\xD2\xE6\x2A\x7B\x81\x63\x54\xE0\xEC\xEC\x3B\x70\x7B\xDA\x24\x26\xE4\x8B\xEB\xA7\x48\x39\x23\x3B\x3F\x53\x9B\x42\x84\x8C\x6F\x86\x67\xF9\xFF\xF5\xE6\x3E\xEE\xA1\x90\xC0\xDE\xC1\x51\xCF\x2B\x75\xBB\x19\xE9\x40\x48\x15\xA2\x50\x97\x3D\x58\x1B\xFB\x10\xA9\x9C\x23\x30\x1C\xE0\x4A\xDB\xC7\x03\x7C\xFD\xA6\xED\xDE\x7B\x0A\x35\xFC\x89\x18\xDF\x7F\xF7\xC3\xBD\xF7\x78\x72\xDA\xD7\xFE\x12\x55\xF8\xD4\x0F\xF5\x6B\xF4\x62\x1B\x67\xF6\xE5\xFD\x4E\x2F\xBF\x26\xC2\xE2\x1F\xFD\xD3\x6F\x9C\xED\xE0\xDA\xA5\xAF\xC0\x2F\xD7\x3E\x73\x6C\xB6\x8B\x9D\xDD\x8F\x69\xFC\xB4\xF3\xA1\x7F\xFD\xF2\x41\x27\xF8\xED\xA7\x7E\xFD\x7F\xA2\x6F\x3F\xF9\xB6\x07\xBA\x6F\xA6\x2F\xDB\xFA\xC0\x20\x9A\x19\x81\x0B\x8D\x0F\xE6\x94\xD8\xED\x75\x8A\xFD\xA6\x18\xC5\x79\x21\x19\x34\x0C\xDF\xF9\x9C\x63\x4D\x31\x1F\x9C\x82\x26\xC3\xD9\x14\x29\x91\x7B\x7F\xCD\x87\x6E\xC7\xA3\x10\x67\xAC\x43\x04\x2B\x9A\xB2\xE6\x29\xC7\x0D\x85\xAA\x80\x98\xA7\xAB\x79\xBA\xF4\x56\x91\xC5\x00\x4D\x55\xBB\xA9\xD2\x7B\x3A\xDF\x78\x9A\xDA\x4D\x33\x76\xB7\x13\x8E\x79\xA3\x0F\x7C\x20\xC5\x0D\x47\xCD\x44\x2E\x32\xF5\xAC\xCF\x79\xD6\x39\x12\x37\xA9\xAA\xA8\x95\x18\x9D\x99\x6A\xC6\x86\xC7\x37\x3F\xED\xD9\xA9\x6A\x25\x45\x10\xEE\x35\x1C\x9E\xEE\x82\x09\xDC\x70\xC2\x73\xCB\x23\x16\x2D\xE5\x13\x9A\x2A\xCF\x30\x6F\x67\x28\xDA\x19\xA2\x7C\x8F\x1C\x7B\x25\x91\x45\xDD\x6B\x7A\x92\x61\x5D\x33\xAC\x4B\x12\xED\xC9\x1F\x05\x24\x61\x87\x26\xEC\x68\xDF\x53\xE2\x83\xC7\xA9\xC2\x9B\xFA\xE5\x49\x12\xCE\xF0\xF5\x43\xDD\x5B\x03\x12\x45\x79\x9C\x51\xDC\x9F\x91\x9B\x4A\x8C\x7F\x62\x9A\x51\x8E\x33\x62\x47\xA5\x80\x85\x2C\x50\x90\x9E\xA6\x43\x30\x80\xB0\xFC\x45\xA2\xA1\xBF\x6A\x44\xB1\xED\xC4\x97\x3F\x7C\xCB\x90\xB6\xBA\xCB\xAE\x14\x87\xF0\xB2\xD3\xC8\x2D\x84\x9D\xEE\x81\x48\x4C\xC4\x34\xC8\x92\x33\x84\x7B\x47\xB7\xC3\xF3\x94\x86\x72\xDA\xE8\x53\x3E\x74\x83\x1C\x64\x29\x20\xB4\xD7\x78\xA4\x64\x0F\xD6\x29\xCE\x37\x6A\x58\x0C\xCC\x23\x7C\x8B\xAB\x93\xD2\xEA\x44\x4D\x4D\x85\x32\x57\xC8\xE4\x21\x16\xC0\x65\x4D\x69\x59\xA9\x00\x96\xCF\x5C\x79\x8E\x54\x13\xF1\x9E\xE4\xBC\x27\x51\x43\xEE\xF0\x11\xB2\xC4\x58\x35\xA3\xAA\xA4\x34\x8A\xAA\x94\x79\x27\x77\x53\x88\x0B\xE0\x75\x25\x20\xB0\xF1\xF3\x9C\x83\xDD\x41\x6A\xE6\x20\x55\x11\xF8\x40\x64\x1F\xA6\xEC\x45\x48\x4D\xD5\x51\x27\x38\xF7\x38\x87\xC4\x9F\xCE\xC9\x71\xE7\xFB\x11\xDC\x4F\x16\x78\xA0\x37\x6B\x14\x8E\xE9\xC4\x22\xA6\x20\xE5\x5C\xFF\x67\x41\xBA\x3B\x2F\x65\xCF\x4F\x29\x06\x2E\xAD\x3F\x99\x48\x2A\x67\xEC\x4A\x1A\xC1\x08\x0A\xFC\x93\xF4\x4F\x98\xA4\x77\xC2\xF8\xEC\x7D\xED\x73\xDE\x50\x82\x37\x9F\xB1\x90\x03\x34\x49\x12\x90\xD9\x6E\x96\x12\xB0\x23\x82\x53\x0C\x39\x62\x5B\x52\xD0\xCC\x97\x47\xC4\x15\x47\xBD\xCE\x7A\xAE\x98\x0E\xF6\x72\x65\x16\x65\x8C\x0D\x91\xD4\x87\x41\xE8\x37\x9D\xB0\x35\x65\x6C\xA5\xFD\xA3\xBD\xCA\xDA\xBD\x32\x79\x42\x19\xC8\x89\x04\xA4\x4C\x02\xAA\x9C\xB7\x95\x77\x21\xE5\x5D\xA8\xF2\x9C\xA2\xD1\x3A\xF2\x92\x3A\xF2\x52\xE5\x3E\xA5\x52\x9D\xD8\xEC\x85\x14\x5A\x94\xA1\x8B\x9A\x20\x12\x92\x12\x09\xE1\xFB\x54\x0A\x04\x95\x73\xCF\xC2\xF4\xC1\x18\x2B\x11\x70\xC8\x2E\xAE\xE6\xCB\x3B\xF8\x35\x3C\x95\x8C\xA7\x62\xB0\x3E\x0A\x64\xDC\x43\xC6\x44\x8A\xDE\x4A\x7C\xCB\xF3\xC9\x1C\x49\x33\xDC\x1B\xBE\x1F\x00\x96\xF1\x86\x9E\x04\x01\xA9\x97\xAC\xAB\xC8\x83\x64\xC7\xA7\xDE\x2C\xB4\xA1\x44\xC0\x0A\x5C\x12\x82\x58\x9D\xA9\xDA\xBC\xED\xD9\x06\x6B\x7E\x4E\x91\xFD\xD2\xCB\x9B\xAA\xF4\x70\x58\x8F\xAA\x1C\xD8\x1B\x3B\x45\x16\x85\x62\x0F\xB5\x4B\x21\xE6\x99\x8B\x31\x99\x86\xC8\x6E\xE7\x47\x21\x1D\x78\x14\x0B\xCA\xE3\xBA\x22\x4E\x98\x0F\x00\xD5\xC3\xF5\xF6\xBD\xC2\xF7\x1D\x8A\xB7\xEF\x25\xBE\x27\x58\x49\x19\x56\xDA\x2F\x83\x28\x5C\xA5\x3B\x64\xA9\xD3\x3E\xAC\x0D\x3B\xEE\x81\xC1\x4C\xD7\x03\xC0\x9B\xE9\x7E\x00\x7F\xFD\x01\x78\xB9\xA3\x03\xC3\xF6\xAB\x01\x45\xEA\xF1\xC2\x40\x02\x05\xC4\xC8\x8F\xF4\x14\x54\x0E\x97\x7A\x04\x3C\x65\xF5\xE1\x88\xA8\x78\xCA\xF7\x21\x74\xE6\xB4\x31\x3B\x20\x43\x96\x7D\x9D\x5C\x22\x32\x73\x51\x48\x49\x0A\xB9\x85\x6A\x29\xAF\x2E\x64\xBA\xD1\xAA\x09\xC3\xAB\x74\x25\xE7\xD5\x84\x61\x2B\xD1\x46\x9D\x26\x22\x5C\xAC\x26\xF4\xF7\x0D\xAD\xEA\x29\xBC\x6A\xBE\x3C\x80\xC0\x5E\x7B\xF5\x0F\xFE\xE8\x85\x23\x32\x81\xE0\xD8\xBF\xC2\xD6\x1E\x7D\xCF\xDB\x9F\x49\x59\xE9\x5C\x00\x70\x79\x87\xFB\x79\x2D\xB8\x5B\xAE\x42\x70\xEC\x27\xBF\x4B\xDC\x2D\xF3\xB9\xCA\xD7\x3E\xFD\xA1\xF7\x3F\xB2\x67\xE5\x8F\x0E\x2B\xFF\xA6\x0A\x49\x29\xB7\x2D\x2E\xE8\x6D\xFB\x28\x0A\x17\x0C\xDD\x24\xC1\xEF\x56\xA1\x15\x2E\x1E\xF6\x85\x86\x02\xD7\x93\xDD\x8A\x64\xCF\x60\xC6\x8B\x69\x11\x40\xBC\xCB\x66\x59\xF1\x6E\x9D\x40\xB4\x7B\x01\xC2\x8B\x57\x29\xD8\x21\x45\x88\x83\x04\x12\x7E\x15\xE3\x9B\x87\x48\xB5\x1F\x78\xE2\x00\x24\xB1\xD8\x7F\x41\x1A\xB7\x7F\xB9\xC3\x29\xBD\x77\xE0\x54\x1D\x4F\xEC\x37\xF3\xE3\xD7\x5B\x71\xA2\x4E\x27\xB5\x5E\x22\x91\x2A\xE0\xA3\x95\x0C\x5F\xBB\x37\xF6\x5B\xA8\x85\xD7\xB0\xD0\x35\xB1\xDF\x4A\x3F\xD2\x09\x9D\x4C\xB4\xDC\xB9\x73\xCE\x06\xD6\x28\xE2\x86\x3C\x97\x30\x5F\xBB\x8D\x7D\xAE\x0C\x70\x9C\x5F\xE9\x74\xF9\x35\x8E\x3C\x1D\xBC\xA4\x44\x42\x2E\xD8\x36\x8B\xE9\x90\x6F\x38\xC6\x01\xC7\x63\x7A\xE3\x11\x74\x6B\xFA\x8A\x86\xD2\x42\x42\xB4\x4B\xBA\x79\x56\x65\x06\xA4\xA5\xF5\xCD\x06\x9C\x5F\x26\x9E\x79\x99\x2C\x7A\xA9\x17\xBD\xA4\xFB\x03\xB6\xC9\xF0\x9E\x76\x99\xF9\xD1\x05\x5B\x2B\xFD\xD6\x12\x5C\x0B\x32\x55\x40\x41\x89\xE2\xBB\x71\x8E\x31\x4E\x2E\x4C\xEF\x69\x6B\xA3\x5D\x16\xCA\xA3\xDD\x3A\x86\x70\xF7\x02\xE8\x8B\x57\xAB\x90\x82\xDF\x9D\xAA\x12\x88\x21\xE6\x57\x11\x65\x87\x42\xFA\xE9\xB6\x36\x65\x0B\x17\x90\x83\xAD\x15\xB4\xB5\xCA\x6F\xAD\xA0\xAD\x8D\x26\x7B\x6C\xA3\xF2\xDB\x18\xB1\x4B\xBE\x20\xD6\x93\xB7\x31\xF2\xB6\x57\xA4\x24\xAC\xDD\x40\xBE\xD2\xE9\xFC\xEA\x84\xB4\xE0\xC3\xD7\x2E\xEA\xA4\x44\xE4\x35\x7E\xDB\x10\x90\xB2\xC1\xB6\xA5\x20\x70\xDB\x28\x86\x56\xB8\x4B\x1A\x07\xD5\x6D\x5B\xDB\x1E\x6D\x5B\x04\xD1\xCC\xCB\x64\xD1\x4B\xBD\xE8\x25\x6D\x1B\x0D\x56\x41\xC2\xC3\x49\xCD\x9F\x29\x8D\x5B\x16\x6D\xCB\x0B\x6D\xFE\x68\xC5\x9B\x43\x8E\x2F\xCF\x95\xB4\x95\x7C\xE9\xE3\xF4\x21\x89\xC7\x46\x56\x05\xA5\x7C\x3D\x9B\xB1\xC6\xDC\xD8\x80\x77\x9F\xE3\xE2\x1A\x08\x37\x26\x9B\xB5\xA1\x20\x2B\x21\x64\xFC\x98\x6F\x42\x68\x03\x30\x90\x6F\x4C\x9A\x5D\xEC\xE5\x32\x56\xB9\x00\xE1\xEE\x45\x88\xF8\x2E\x21\xF4\xCD\x24\xB5\xC1\x9A\x1B\x74\xAD\x70\xB5\xA9\x33\x30\x67\x98\x0E\xEE\x5E\x02\xB3\x5B\x17\x3B\x57\x4F\x53\x93\xE4\x67\xD6\x35\xF9\x16\x6E\xB2\xB8\x58\x17\xF8\xF8\x83\xF4\x58\x8F\xA8\x11\x18\x5D\xAD\x0D\xD0\xFB\xB7\xF1\x7B\xD7\xB8\x06\xB3\xD1\x90\x0B\x2D\x84\x80\xC3\xC0\x09\x34\x35\xF6\xDF\x50\x24\x92\x84\x06\xB0\xD1\x40\x88\xDF\x62\xFC\xC6\xA7\xC9\x0B\x3B\x71\x35\x9C\x93\xA2\xD8\x55\x0C\xF7\x77\xFE\x26\x24\xF6\x85\x4C\x43\x5F\xB3\xEE\xEB\xE8\xAA\xBF\x84\x3F\x0D\x59\x73\xC4\xED\x6D\xD5\x52\x77\x7F\x55\xED\xCD\x9C\x60\x04\xC5\xA5\xAB\xC7\xDB\x4C\x89\x2E\xA8\xBB\x30\xF6\xB7\x7B\x17\xEA\xE6\x67\x93\x98\xF1\x94\xF1\x95\xC4\x0C\x42\x8D\x22\xB0\xBF\x7A\x75\x70\xCB\x20\x28\x62\xB9\xFD\xB5\x37\xF7\xEE\xE3\x05\x61\x9E\xC7\xF1\x35\x19\xF4\x29\x38\xD5\x71\x41\x4D\x72\x66\x3C\xE5\xB9\xF6\xAE\x30\x25\x5E\x88\x09\x61\x06\x09\xC4\x0D\x05\x64\xAC\x72\x82\xD6\xAA\xE0\xEB\xC1\x11\x3B\xB1\x52\x50\xCB\x92\x74\xC4\xF5\xA8\xA9\x97\x40\xDB\x00\x96\xA0\x44\x08\x5A\xB2\x01\x68\x58\x02\xBD\x31\xD9\xE4\x04\xD7\x39\xDF\xDB\x2D\x35\xF5\x32\x15\x5C\x46\x48\xD8\xAC\x97\xA9\xE0\x72\x5B\x70\x0C\x19\x8C\xA0\xDC\xA4\x32\x23\x6C\xAC\xD9\xF5\xA4\xCD\x40\x01\x4B\x90\x6F\xBA\x8E\xF2\xFE\x37\xE4\xD1\xAB\xC8\xDF\x42\xA6\x88\xD5\x69\x77\x2B\x2C\xAC\x6E\x6D\x0D\x56\xAC\xAC\x56\x29\x68\xAD\x78\xA8\x3A\x20\x7D\x28\x58\xCA\xE2\x51\x2D\x73\xD0\xD1\x18\x0E\xE0\x9F\xDB\x9A\xEA\x19\x28\x56\x9C\xAB\x0E\xB2\x55\xBD\xBB\x81\x7C\x46\x53\x2F\xF1\xB2\xD0\x12\xA5\x9C\xE9\x33\x81\x04\x6E\x6B\x97\x85\x66\x6B\x9A\xFA\x99\x34\xDA\x67\xF2\xB2\x3C\x93\x66\xFB\xCC\x76\xB6\x29\x31\xCB\x6D\x99\x7C\x51\x99\x31\x18\xBF\x1E\x86\xD6\x03\x5F\x2E\x41\xE4\x17\x22\xA2\x85\xE0\x45\x3E\x88\x93\x1F\xC1\x32\xEE\x91\xD8\x60\xBB\x27\xF5\x82\x42\xE1\x0C\x05\xCE\x09\xFF\x7F\x6A\x52\xDD\x0E\x0A\x0E\xA0\x34\x52\x3D\xD3\x8A\xEA\x19\xB0\xE2\x32\x37\x3D\x03\x56\x4F\x23\xB4\xD8\x9C\x41\x80\x66\xAD\xFC\xBD\x2B\xCE\x6E\x89\x2F\x16\x69\x76\x25\x92\xB1\x76\x08\x9B\x75\x64\xF9\x72\xAD\xDB\xF4\x98\xAC\x89\xE1\x99\xBC\x46\x48\x67\xDD\xAA\x8C\xA9\xDE\x18\x22\xAC\x37\xA6\x7A\xE3\xB6\xDE\x12\x94\x7E\x72\x25\xEF\x32\x64\x7E\x9F\x53\x30\x10\xF9\xD5\x88\x86\xDF\x16\xCF\x9F\xE4\xAA\x67\x20\xC7\xFE\x0C\xB8\x7D\xBD\x55\x54\x28\xBA\xD7\x58\xC5\x4D\x5F\x85\x03\x95\xA2\xD8\xA8\xC2\x52\x12\xA4\x53\x93\x6A\x44\xA1\x3B\xD8\x90\x55\x73\x9E\x66\xD0\x2F\x28\x02\xC8\x79\xB3\x73\x70\xF9\x2A\xEA\xD2\x09\x72\x25\xC5\x55\x35\x94\x47\x65\x6C\xEF\x74\xE1\xE1\x29\x86\x53\x1D\x4D\xEA\xD2\x3E\xDC\x40\x09\x1A\xB4\x2D\x1F\x82\x68\x52\x47\x64\xD7\x6A\xF0\xCB\x79\xFC\x12\x21\x0C\x3F\x04\x06\xBF\x8C\xDD\x87\x0B\xFE\x83\x76\x1F\x14\x47\xB6\xA7\x1C\x83\x60\x26\x56\x4C\x37\xAD\x68\x1E\xDA\xE4\x50\x63\x39\x4E\x36\x87\x91\xBB\x7A\xDD\x11\x94\xE4\x78\x21\xED\x60\x92\xE1\x2C\x09\x18\x3D\xEC\x5F\x3D\x76\x89\xB3\xAC\xD8\x47\x76\x76\x0E\x6F\x50\xC6\xA0\x3E\xA9\xD9\x06\x59\xBE\x9E\x93\x14\xBB\xB4\x78\x24\xFE\x94\xFF\xEE\x22\x87\x71\x9E\x51\x7C\x5B\x71\xCB\x47\x20\x9E\xF0\x08\xDE\x17\x45\x9A\x2E\x72\xE5\x85\x78\x5B\x5E\xE8\xE2\x70\xBA\xE0\xE8\x48\x41\x1F\xAC\xB4\xBD\x4A\x83\x09\xDD\x60\x0E\x77\x83\x09\xFB\x83\x09\xFD\xD8\x87\x0C\x54\x15\xDA\x6D\x27\xC3\xD3\x22\x7D\xC7\x2B\xC9\xBD\x6B\xD7\x2B\xF2\x10\x5C\xEF\x27\xBB\x60\xFC\xFA\x3A\xCE\x2C\xCC\x8A\xD3\xCE\x16\x46\x3D\x57\xE6\x95\xC1\x3F\x49\x95\x83\x46\xAA\x53\x00\xDF\x40\x8D\x60\x04\xE9\x90\xE0\x46\xC8\xD3\xE0\x21\x9D\x30\x65\x49\xE8\xFC\x4E\xF8\xFC\x46\x1C\x28\x1C\xAE\x2E\x37\xF5\x0A\x15\x5C\x81\x64\xB6\xA0\xA7\x24\xE5\x6E\xBD\x0A\xE6\x02\xC4\x17\xAF\x36\x75\x49\xA5\x4B\x46\xCD\x92\x4A\x97\xDC\x6C\x09\xCB\xBB\xF5\x01\xC8\x5D\xC1\x41\xFF\xFD\x82\x4C\xB5\x22\x58\x6D\x8B\x0E\xDA\x8C\xA8\x68\xC4\x6D\x46\x70\x60\x71\xCF\xFD\x52\x74\xD9\xC7\x27\x0F\x07\x07\x1D\x2C\x24\xDB\x9C\x46\xCE\xB4\x8D\x55\x3C\x23\x8A\xE6\x5F\x87\xBB\xB5\xE1\xE6\x2B\x73\x4C\x10\x32\x97\xB4\x9E\xB9\x0B\x48\x5B\xE7\x30\x22\x96\x3E\x26\x0A\x42\x9D\x8C\x61\x8C\xAC\x59\xB8\x5B\xAF\x76\xEF\x13\x6C\x33\x77\x6D\xE5\x60\x80\xBE\x5D\xAD\x42\x28\xF1\xAC\x29\xD9\x21\x81\xFA\x57\xE7\x9A\x2A\x84\xA4\xED\xBA\x1D\x42\x7E\x4C\x54\xAB\xEE\x52\x31\x84\x55\x3F\x84\x55\x08\x7B\x43\x20\x8B\xDE\x12\x8C\xFF\x6A\xA0\x1C\x7C\x25\xEB\x57\x3F\x10\xE3\xD7\xB8\x5A\x45\x19\x17\x17\x88\x1D\x23\x9C\xA0\x26\xAC\xE0\x58\xBF\x82\x56\x25\x64\xC3\xA5\x11\xDB\x97\x23\x83\x58\x7E\xE7\x2B\xC9\x33\x81\x9C\x6E\x29\x61\xEF\xA8\xA9\x03\x32\x5D\x20\x2A\xD9\x06\xA3\x8D\xCC\x8F\x46\x49\xC4\xD6\x2A\x64\xB9\xA1\xB6\xC5\x4D\x61\xD3\xCD\x61\x14\x8F\x94\xEE\x1B\xC8\x70\xD4\x8F\x38\x66\x5B\x63\xF2\x69\xC2\xE1\x7E\x57\x3B\xDC\x24\x57\xC8\x77\x46\xE4\xF9\xC4\x73\x9E\x91\x6A\xC2\xDD\x2A\xB5\xA2\x4A\x1C\xDD\x60\x68\x61\xB9\x41\x6D\xB4\x56\x49\x14\x49\x0A\x61\x25\x76\x1B\x96\x76\xB0\x22\xE8\x7A\xAB\x07\x2B\x11\x6D\x45\x7A\xD1\x1B\xBB\xD4\x05\x5D\x69\xA3\xA4\x3B\xEA\xDE\x13\x27\x92\xBB\xB6\x10\x56\xE8\x1B\xCA\xBB\x02\xF7\x48\x50\xFA\xFB\x0E\x56\x22\xC8\xDA\xAE\xDB\x21\x20\xAC\x8C\x9C\x01\x60\x04\x23\x3F\x84\xD1\xFC\x10\x44\x1F\x56\xC4\xE0\xEB\x60\x20\xC8\x7D\xE6\xF4\x6B\xE4\x85\x55\xBD\xD1\xC6\x94\xA2\x25\x62\xBA\xA5\x3D\xD0\xF0\xF2\x8C\xE7\x28\x93\x3C\x47\x8A\x2D\x75\xAE\x5A\x62\x41\x08\xCA\xA6\x8E\xF9\x96\x3F\x70\xC4\x49\x10\x2E\x0B\x3A\xD3\x6B\x41\xB8\x2C\x18\xE3\x05\x04\xB0\xD4\xD4\x74\x9D\xEE\x7C\x77\x03\xC8\x40\xCC\x16\xE4\xAB\xD9\x08\xF4\x2E\x32\x9C\xB4\x82\xC8\xFF\x87\xCC\x64\x60\x69\x4D\xA5\x35\x37\xAB\x41\xEE\xD6\xCB\x3C\xC3\xAB\x33\xFD\xF7\x0B\x92\xC9\x1A\x2E\x69\x5B\x74\xD0\xE6\x0C\x71\x5A\x5E\xDC\xF3\x2C\x71\x22\x91\x2E\x80\xF1\x1C\xDA\x24\xE6\x03\x63\x91\xCD\x87\x7F\x66\x43\x2E\x1B\x94\x1F\xE6\x84\x79\x14\x6D\x9C\xFF\xE4\x1E\x67\x0E\x05\x41\x1D\xB1\x8F\x90\x0C\x10\x37\xCB\x5F\x61\xC3\x53\x94\x55\xC4\xA1\x40\xB8\x0C\xC2\x95\x58\x18\x99\x97\x1D\xE6\x88\xC9\x88\xED\x3B\xB7\x9B\xD1\x73\xC2\x28\x8A\xA2\x00\xFF\xA3\xA3\x45\xFF\x7A\xDF\x85\xCB\x04\x06\x5B\xF6\x77\x83\x13\x2E\xB0\x54\xA1\xF3\xC4\xE5\x0B\x83\x2D\xFB\xC7\x01\xE7\x9E\x08\xAD\x2E\x7F\x8B\x71\xF8\x50\xA0\x90\x0E\x23\xC6\x29\xC4\xDD\xC8\x1E\x3E\x51\xE8\x3C\x34\x33\x23\x31\x52\xF9\x7F\x22\xA0\x28\x56\x11\xC4\xF6\x0F\x82\xF5\x82\x53\x48\xB7\x7D\x73\x1F\xBE\xC7\xDF\xF5\x4F\x6A\xD8\x7F\xD2\xF5\x4F\x59\xC9\x62\x96\xBF\x62\xEA\xF5\xF0\x3A\xF9\x20\x70\xA4\x3D\x5E\xD2\xC3\x27\xF8\x3A\xBA\x12\x54\xC2\xFB\x24\xC5\x84\xEF\x9C\x93\x99\x0B\x3E\xD0\x70\x30\x6A\x03\x51\x11\x51\xA8\x4B\xE4\xDB\x28\x76\x20\x09\xDA\x20\x20\xDA\xAC\xD3\x36\x4B\xC5\x1D\x9B\x2F\x28\x42\xBE\x78\xA1\xC9\x1E\x2F\x42\x88\xED\xCF\x91\x63\x16\xCF\xAF\xB7\xAE\xC3\xB9\xFD\x5E\xC0\xDE\x5C\x7E\x0D\x03\x1A\x1A\x57\x8D\xA8\x6A\x44\x55\xB5\xAF\x1A\xE2\x53\x88\x4F\x7F\xE0\x9F\x22\x7C\xFA\x2F\xFE\x29\xC6\xA7\xBF\xF4\x4F\x49\x11\xBA\xEB\x2C\x5E\xAC\x9C\x1C\x85\x13\x97\x2B\xD9\x29\xF0\x06\x40\x93\xDA\x8F\xBB\xDE\x39\xC0\x73\x64\x7F\x86\x27\x10\xE2\x06\xB3\x72\x88\x27\x43\x20\xAA\x04\x5D\x20\x7C\xC9\x3A\xC7\xC1\xD8\xA2\x18\x1B\x21\x19\x27\x62\xC1\xF4\x78\x21\xF3\xD8\x79\x2F\x45\xC0\xFA\x0B\x48\xDD\xE2\x87\xDE\x15\x51\x50\xEC\x07\x97\x65\x2E\x1C\xBC\x61\xDE\x2C\xA5\x0D\xE5\xA6\x0F\x05\xA2\xDD\x24\xC1\x7D\xBB\x05\x75\xEF\xD8\x75\x83\xC3\x5A\xE3\x4F\xCA\x32\xC4\xB1\xE8\x04\x44\x54\xBD\x60\xF8\x53\xAE\x3B\x4B\x28\xF8\xB6\xB7\xE0\x41\x42\x66\x8C\x29\x99\x22\xD8\xAC\xFC\x1E\x8E\xAD\x27\x8F\x22\x93\x49\x05\x6A\xC1\x1E\xDD\x87\x02\x9F\x4D\x38\x27\x67\x13\x08\x38\xA9\x4B\x58\x09\xBA\x10\xB3\xB7\xF9\x16\xF9\x43\x4E\x3B\x11\x71\xFE\x90\x7A\xBD\xA0\x92\x10\xE1\x99\xB8\xF3\x70\x63\xC5\xA9\x42\xA3\x14\x11\x6F\x92\xD2\x2C\x61\xF5\x26\xB9\x64\xD6\xC2\x5E\xFB\xF4\x87\xDE\x8F\x25\x20\x70\xFE\x79\x9C\x8A\x0F\xC4\x26\xC5\xEB\xA3\x77\x5B\x27\x27\xD8\x78\x8A\x02\xD8\xA9\xCD\x2A\xC5\x75\x76\xF5\x94\x7D\xE4\xF5\x9F\xFC\x88\x22\xC8\xE7\x9A\xFC\xE2\xFE\x22\xE0\xFE\x69\x24\xD4\xCA\x7A\xA1\x7C\x9E\x93\x5C\x9A\x5B\x81\x42\x62\x88\x42\x72\x31\x0A\x51\xC0\xBA\xD4\xA6\x2F\xA9\x33\xC8\xEC\xCB\x4E\x23\xE7\x4B\x1A\x7A\x8F\x16\x28\x6F\xAA\xF2\x03\x17\x5D\x6A\x22\xBB\xA3\x41\x95\x1F\xF1\x77\xA5\xB4\xB2\x39\x4E\x6E\x95\x82\x08\x19\x3A\xCD\xF0\x30\x47\x59\x9B\xEF\x79\x50\x6C\x0B\x39\x55\x8A\xB1\x62\xEA\x82\x10\x60\x67\x56\x40\xBA\x89\x00\xF6\x00\x19\x08\x5B\x8A\x4D\x31\x75\x61\x90\x40\xD8\xEC\x34\xC4\x10\x43\x72\xA2\xA9\xC5\x5D\x41\xD0\xD8\x07\x9A\x3A\xC6\x76\x10\xD1\xB3\xD3\x3E\xA3\xBB\xA0\x12\xB1\x2F\x21\xD8\x4E\x9A\xBC\x0D\x59\x35\xC1\x4C\x6C\x76\xDA\x3E\x80\xE4\x9B\xAC\xA1\xE8\xE0\xF3\x8D\x53\xD5\xAE\x16\x11\x50\xD2\x1B\x5B\x79\x0E\x72\xBA\xB9\xE2\x95\x66\xD3\xF1\xB6\x5E\x78\x57\x10\xB4\x63\x12\x3C\x26\x67\x8E\x1B\xB6\x05\x9A\x5E\x89\xB9\x8F\x54\xE7\x6F\x5B\x09\xDA\x21\x5E\x7F\x77\xDF\xDF\x73\x59\x0A\xCA\x0F\xF2\x21\x4C\x89\x18\x7E\xC7\xDF\xC0\x05\x56\x7A\xB8\x32\xBD\xB4\x0C\xC6\x0E\xD2\x39\xB8\x7C\x52\xA4\x4D\x23\xAE\x35\x58\xB7\x72\xCB\x8A\xF2\x5D\x1C\x70\x40\xAE\xC9\xBC\xE2\x04\x59\x3E\xAB\xB1\xF9\xC3\x55\x61\xBC\xDF\x88\x28\xDF\xEF\xF3\xCC\x1B\xCE\x22\x54\xBE\xD7\x27\x5D\xE1\x84\x5F\x3E\x05\x95\x35\x0D\x27\x15\x14\xCE\xE6\x37\x44\x14\x3A\xE4\x13\xAD\x95\xEF\x63\x84\xB0\x40\xAE\x66\xD7\x82\x29\x05\xB2\x89\x20\x24\x23\xAC\x1D\xE7\xFB\x13\x42\xD4\x3A\xED\x51\x4A\x90\xAE\x3F\xD7\xCE\xCF\xF9\xEE\x95\xFD\xA5\x9F\xFA\x71\x79\x24\x08\xAC\x62\x1A\xAC\xBD\x8C\x1B\x1C\x0A\xFE\x12\x59\x79\x6C\x09\x85\xEE\xE0\xCF\xF1\xE9\x48\x10\x18\x50\x10\x11\x7D\x77\x43\x97\x20\xCA\x9F\x71\xA9\x2D\x22\x50\xA0\xCA\xFF\x97\x6F\xD3\xCA\x9F\xE8\xB2\xDF\xAE\xC9\x1C\x91\xF2\xAB\xB1\xCF\x2A\xE2\x8B\xE8\x6D\xE6\x1F\xE4\x39\xFB\x30\xDB\xFD\x8B\x35\xB9\xCA\x3E\x03\x64\xAF\x95\x22\x1E\xF3\x00\x20\x6D\x2F\x7E\x39\xF8\x90\x22\x37\x4A\xC8\xE8\xF5\xE9\xCD\x3A\x6D\x2F\x8C\x39\xD0\x84\x22\xB7\x4A\xFC\xBE\xFB\x31\x4D\xDF\xDD\x35\xB3\xAF\xAE\xF8\xF3\xB5\x4B\x5F\x41\x5F\xDD\xD5\xB4\xAF\xAC\xF9\xEB\xE5\xD7\x44\xF4\xD5\x5D\xB9\xFB\xAF\x21\x7F\x7D\xED\x2F\x05\xF4\xF5\xFA\x75\x35\x75\x96\x12\xBE\xF5\x88\x4B\x7C\x22\xEE\x0A\x3C\x3E\x68\x22\xE6\x02\x8F\xBB\x16\xC4\x94\xED\x36\xFC\xE7\x84\x3F\x77\x81\x33\x12\x04\xE9\x04\xA2\x4A\xB9\xE0\x98\xA4\x6D\x60\x13\x02\x10\xE5\xCF\x76\x1B\x9A\xD0\xAA\x1D\x0A\xBE\x65\xA7\xDB\xBD\x67\x07\x97\xF0\xE9\xEE\x00\xC9\xA7\x37\x22\x74\xDB\x42\x0D\xD9\xCF\xBE\xA5\xAF\x5D\x76\xE7\xCD\x9A\x4C\xBC\x45\xBA\xB4\xC9\x94\x43\x2E\x2B\x3E\x6A\x0D\x05\x7E\x2D\xA7\x8E\x8B\x21\xDD\x07\xBE\x33\xF6\x9F\xDF\x5F\xB0\x73\x36\x45\x57\x72\xE5\x69\x94\x3F\xDF\x83\x7A\x6D\x65\x63\xD5\x4B\x6B\xC5\xEE\xFF\xCA\x06\x27\x29\x8E\x33\x37\x4E\xAE\xCC\x2F\xAD\x29\x3E\x6F\x84\xCD\x9F\x9C\x90\xAB\x8F\xF1\x99\xDA\xE2\x4D\x0E\x25\x10\xE3\xFF\xEF\x9F\x20\x0D\x3D\x39\xB1\x4B\x4D\x3D\xC0\x1D\x61\x0F\x79\xE5\x53\x0F\x77\x54\x0F\x77\x14\x84\x84\x4B\x1D\xEE\xA8\x16\x77\x38\x99\x2E\xF0\x52\x43\x64\xDF\xF1\x91\xE0\xCB\x02\x0E\x24\x8C\xAD\xE2\x49\xB7\x59\x27\x8C\xBD\xCA\x1E\x86\xC4\x47\x20\x89\x78\x78\x64\x59\xE5\x50\x03\x11\x23\x82\xE8\x26\x50\x43\x3A\xD4\xC8\x9E\x28\x6A\x14\x43\xD4\x28\x66\x51\xA3\x18\xA2\x46\x31\x83\x1A\xC5\x10\x35\x8A\x21\x6A\x14\x43\xD4\x28\x86\xA8\x51\x0C\x51\xA3\x98\x47\x8D\x62\x88\x1A\xC5\x1C\x6A\x14\x43\xD4\x28\x66\x50\xA3\x18\xA2\x46\xB1\x18\x35\xD8\x6D\x85\x51\x83\xCD\xF2\x62\x02\x18\xA2\xAF\xFB\x11\xA9\x05\x7B\x21\x9E\xDE\x8B\x5B\xB1\x17\xC6\xB8\x7D\x78\x3E\x59\xB8\x29\xD2\x77\x28\x17\xB8\x4E\xB1\xA1\x0D\x17\xD8\x3C\xC9\x71\xD2\x55\xC3\xAB\xEF\x37\x2D\x01\xBA\x7B\xE1\x4D\xA3\x38\xEA\xB5\x72\x3B\xC7\x07\x51\x9D\x5A\xC9\x78\x28\xF8\x72\x68\xBB\xA9\x0A\xB2\xCC\xC2\x5D\x4B\xFD\xAE\x91\x2B\x23\x07\x85\x05\x45\x99\x7B\x38\xF0\x5D\x52\xA1\x44\xE7\x8E\x93\x07\xEB\x91\x5B\xA1\x0C\x46\x6E\xE7\x5C\x20\x59\xD1\xD4\x7B\x55\xA4\x7D\x1D\xB9\x75\xA0\x8A\xB4\xA5\xFB\x56\xA4\x0D\xEF\x77\x48\x7B\xBD\x6F\x35\x82\x84\x7E\x77\x04\x04\xFB\x56\x23\x10\xE9\x57\x23\xE8\xD8\xB7\x5A\x0B\x3B\xFD\x81\x22\xD8\xDC\x5C\xCD\xC7\x67\x3A\x7D\xFC\xA6\xFA\x64\x68\xEB\xD7\xCB\xF6\xAB\x86\x78\x28\xA8\x36\x57\x4A\x11\x1E\x53\x1F\x4D\xBE\x40\x90\x2C\x1C\x48\x52\xE4\x1B\xCA\x4C\x98\x3B\xF7\xAB\x9F\x09\xEC\xC7\xD9\xFB\x92\x22\x43\x45\x14\x28\xCA\x20\x91\x57\x1C\x2D\x4A\xD0\x41\x75\xE7\x37\xD4\xA2\xA1\x04\xE8\x1C\x53\x7E\x9E\x98\x44\x7C\xFC\x01\x27\xC5\xB6\x5F\x7C\x4F\x1B\xD0\xDE\x70\x80\x23\xD0\xCD\x46\x11\xD8\x0F\xBC\x76\x70\xD1\x1B\x41\x88\x5D\x51\x64\x16\x94\x4F\x3E\xF8\x58\xEF\x4C\x76\x24\x26\x80\x90\xCC\x11\xC7\x2E\x7D\x5D\x60\xB7\xCD\x67\x74\xB0\x50\x31\x24\xED\x85\x66\x94\x26\x71\xC4\xC6\xB7\x14\x44\x3D\xE0\x45\x0B\x78\xD1\x38\xC0\x08\x89\x80\xFE\xBE\x5C\xBA\x95\x73\xB1\x47\x04\x9E\x8D\xE4\xC4\xB4\x49\x3A\x28\xDF\x80\x5B\x75\xD5\x6F\x80\x50\x47\x3A\x00\xE9\xD7\x67\x0C\xDA\xAF\x3A\x21\xD0\xA2\xEA\x8C\x47\xFB\x54\x27\x34\x5A\x50\x9B\xB1\x69\x9F\xCA\x84\x4C\x0B\x2A\x33\x4E\xED\x53\x99\x50\x6A\x41\x65\xC6\xAC\x7D\x2A\x23\x4E\x2D\xA8\x4B\xA8\xB5\x4F\xD5\xC7\x17\x77\xFB\xF8\x4D\xF4\x9A\x2D\xAC\x99\xED\x53\x91\x91\x0B\xEB\x77\x55\xCD\xFF\x2C\xC5\xB6\xB8\xE0\x72\xB8\x89\x5D\x08\x76\x2F\xD4\xEA\x18\x3C\x7A\x15\x1B\xB8\x6A\x83\x35\xF9\x11\xBA\xAD\xE3\xD8\x4B\x9C\x24\xF1\xB0\x10\xDB\x83\x1A\xAE\x3C\xC9\x56\x6B\xF2\x71\xAA\xF0\x48\xAF\xC2\xBD\x42\x5C\x60\x90\xDE\x05\xB1\x7B\xA1\xD6\x6D\x05\x4D\x5D\x7C\xB4\x57\xA3\x96\x74\xA5\x68\x29\x92\x88\xAB\x6F\xFD\x20\xD5\x7C\x1B\xFA\x6A\x2D\x21\x40\x89\x5E\x72\x5E\xD7\xD9\x21\x2B\x03\xCA\xBC\x45\x70\xFC\x91\x00\x3F\xFF\x06\x7E\xAE\x84\x3C\x6A\x0F\xCE\x96\x76\xF9\xD9\xDD\x8D\x08\xA2\xB2\x9A\xBA\xA0\x3F\x88\xAE\xAB\x9C\x46\x44\xE0\xB9\x34\x5B\x13\x3F\x43\x40\x31\x8C\xE6\x2E\x39\x83\x31\x55\x3B\x42\x89\xCD\x83\x23\x32\x07\x79\x2C\xB8\x9B\x33\x8A\x53\xCC\x1D\xDA\x20\xF3\xC3\x42\xC8\x6D\xD6\x8B\x28\xD6\xFE\x94\x20\x9E\x5F\x04\xF6\xBF\x7C\xCF\x8C\x67\x50\x37\xB6\xF5\xF6\xF2\x68\x6E\x44\x1A\x47\xA4\xF9\xCE\x4D\xD1\xE8\xCA\x5F\x66\x31\xBA\xFC\x50\xA7\xD3\xCE\x21\x70\x63\x61\x4B\x94\x1D\xBD\x85\x74\x09\x87\xAC\x41\xD1\x20\xAC\x3C\xE7\x52\x3E\xB3\xB6\xA9\x84\xA0\xFC\x55\x17\x61\xE1\x75\xC3\x31\x0F\x07\xA6\xDA\x81\x0D\xB6\x58\xB3\xC9\x27\xB7\x04\x38\x4C\x2B\xC6\x7E\x94\x1C\x10\xC2\x4D\x1E\x39\xB9\x99\x51\xDB\x36\x25\xE8\xE2\x91\xE3\x76\x7F\xAF\x08\xE4\x51\xB6\xDF\xC7\x66\x82\x17\x14\xBD\xE7\x84\x9E\xAD\xB0\xD2\xBF\xD1\x10\x9C\x9A\x8C\x8D\x55\x56\xFB\x57\x39\xBF\xEA\x6A\x1D\xE4\x5A\xA1\x8D\xFC\x9B\xD5\xD9\x22\xC0\x45\x62\x9B\xF8\x37\x77\xB8\x86\x53\x9B\xF9\x57\xCF\xC2\x57\xE6\x6E\xBE\x74\xC7\x8E\x6A\x31\x27\xC2\xD1\xFB\x0A\xF9\x27\xF2\xBB\x5A\x93\xAB\xCE\xEF\x89\x02\x9E\x1D\x0A\x02\x2B\xA7\xED\xCF\xBC\xFB\x29\xBA\x9F\xBA\xFB\x39\x9A\x9A\x17\x30\xC6\x0A\x17\xB3\xCA\x8E\x58\xEE\x7B\xA2\xDD\x1F\xA7\x97\xFD\xEE\x6E\x0F\xBC\x6D\x0E\x85\x5D\xEB\x79\x90\x9B\x53\x6E\x8A\x87\x82\xA0\x16\x1C\xBE\x11\x9B\x9A\xC5\x0D\x2A\xC0\x74\x4B\x90\x3C\x0A\xC1\x9C\x7B\x9E\xF9\x49\x17\xF5\x28\xB8\x2B\x08\x6A\x65\xB7\x71\xC0\xEF\x78\x6B\x6F\xC0\x87\x08\xD1\xC9\x08\x92\x33\x75\xD9\xF7\xBF\xB5\x8F\x32\x2D\xDA\xD7\x0C\xD1\xB5\x46\xB4\xFA\x96\x37\x0D\xE6\x5C\x56\xBA\x23\x0F\xCA\x38\x73\x4B\xE2\x1A\x40\x9C\x9C\xD4\x0A\x34\x27\xDC\xA7\xD9\x33\x7B\xA0\x80\xC2\x9E\xE1\x0F\x59\xFE\x14\xB6\xC4\x80\x0C\xB2\xFC\x69\x0E\x1A\x6E\x20\x34\x7F\xEE\x02\xB3\xEC\xB3\xE0\xE4\xCE\xD6\x42\xED\xEC\xEA\xD7\x72\x06\x8C\xE5\x1C\x18\xCB\x53\x13\x4E\xFD\x39\x80\x64\x7E\x2B\x06\xC0\x2C\xE7\x80\x79\xBE\x14\x70\xA9\x01\x3C\xFB\x52\x43\x90\xA6\xB7\x6C\x52\x2B\x6C\x7A\xBA\xB1\x2F\x6B\x8E\xC8\xC4\xAC\x50\xF2\x56\x5C\xAC\x3F\x11\x53\xAB\x91\x31\x31\xE3\xF6\x9D\xA0\xE7\xB2\x7D\xFE\xAC\x98\x0E\x0B\xDC\x49\xCF\x5D\x23\x9F\x12\x53\xE0\x4A\xEF\xC3\xF5\xEC\x34\x7C\xA2\xCD\x18\x4F\x00\x22\xF7\x00\x10\xC9\x7C\xDD\x3C\x80\x0C\x3E\xC3\x94\x21\xB5\xEC\x60\x46\x3E\x9F\x4F\x96\x7B\x82\x31\x87\xD2\x59\x05\x69\x35\x08\x3B\x6E\xCA\xF7\x60\x03\x48\x5C\xC5\xA1\x60\xEC\xA1\x97\xAA\x1D\x91\xA5\x81\xE0\xD8\x23\x6C\xCF\x9D\x38\xE8\xF7\x71\x44\x9C\xAA\xF2\xAF\x95\xD0\xDB\x6C\xF7\xA9\x6C\xFA\x12\xEF\x8D\x44\x90\xB7\x0F\xB4\x43\x68\x83\x75\xB2\x21\x58\x08\xF4\xA0\x21\x72\x8A\x0E\x3F\x0F\xCD\xB0\x1F\x2D\x84\xFD\xA8\x83\x7D\xED\x33\x7F\x6B\x3C\x02\x4E\x4E\x6A\x0D\xD1\x09\xA7\x7D\xEC\xC1\xBE\x86\xE8\xF9\xE4\x41\xA3\x3D\xEC\x7B\x5F\x43\xDD\x82\xBF\x36\x6C\x3E\xC9\xC4\x3D\x77\x51\x21\x8E\xC8\x84\x1D\xD1\x04\x47\xD9\x3C\xE9\x88\x03\x9B\x14\x38\xE7\xCF\xD0\x87\x61\xD8\x76\x81\x01\x2E\x34\x4C\x94\x02\x5F\x9C\x2E\xAF\xF8\xD4\x11\x0D\x05\x40\x73\x4A\x7A\xBA\x15\x0F\xF0\x23\x05\x16\x22\x57\x6A\xBE\x04\x3E\xCF\x26\xBC\xE4\xF7\xDA\x0B\x84\x26\x28\x10\x1A\x5B\x1C\x6B\xF6\x35\x0D\x1B\xFB\xB5\x4D\x15\x60\x03\xC7\xB0\x81\x3F\x7B\xCF\x0F\xFF\xE2\x97\x91\x89\xBE\x3D\xD8\xCC\xBE\x29\xE7\xDE\x24\x73\x6F\x5E\xC4\x46\xDF\x0F\x37\xB5\xB6\x31\xC9\x30\xE6\x41\xE1\x72\x00\x3B\xC2\xD0\x2E\x44\xE5\xB2\xF1\xBA\x3C\xEB\x3E\x9D\x77\x76\xA6\x20\x2B\x40\xF6\xE5\x26\x2F\x13\x7C\x79\xBC\x08\x72\x6D\x4C\xE7\xAA\x90\xB2\x07\xE0\x76\x53\x0B\x67\xDC\x4D\x7C\xD4\x35\x21\xF4\x9E\xE7\x80\xDC\x83\x2C\x39\xB4\x70\xB4\xA8\xF6\x2E\xB8\x48\x7A\xC6\x5E\xB9\xE8\xEE\x6F\x94\x72\x5A\x18\x77\xC4\x6B\x3F\x60\x17\x02\x4A\xB3\xCB\x5F\x37\x60\x67\x84\xAB\x70\xC0\x6A\x10\x28\x11\xE1\x9B\x48\x9A\x6A\xEC\xF6\x59\x7B\xE7\x43\x7C\xF4\xFC\x80\x0B\x9F\xB8\x2F\x21\xED\xA8\x67\x2D\xE7\x47\x2C\x86\x23\x96\x37\x3F\xE2\xD6\x5E\x98\x46\x3C\x58\xE2\xD9\x11\x9B\x8F\xE0\xA9\x35\x67\x80\xE3\x11\x5B\xEF\x81\xD8\xDA\xB8\x38\x23\x8B\xA9\x94\xFB\xBC\x80\x4A\x85\xCF\x27\x18\xE9\x53\xA9\xD0\x22\x30\xF7\xA8\x94\xA4\xD0\xA7\x8B\xE8\x14\xB7\x3D\xC3\x58\x11\x41\xDF\x02\x8D\xA4\xD7\xE9\x8C\x5A\xCA\xF5\x7B\x7F\x0B\x66\x47\x13\xE3\x83\xC9\x31\x91\x20\xB6\xBC\xBF\xA0\x9B\xE7\xCF\x69\x8F\xA4\x82\xE6\xD4\x5A\x82\x20\x6C\x69\xE4\x7A\x36\xEB\xD6\x5C\x9D\xAE\x8C\x29\xCA\x1F\x9B\x9C\x0B\x67\x7B\x22\xAC\xF4\x36\xED\x1B\x93\x5A\x30\x25\xF4\x87\x1A\x9F\x84\xFD\xA3\x5F\x3B\x42\xC8\xE7\xBE\x9E\x39\xF7\xF5\xDC\xB9\xAF\x4F\x4D\xC8\x83\x7C\x78\xEE\xF3\x5B\x39\x38\xF7\xF5\xDC\xB9\x3F\x5F\x0A\xB8\xD4\xE0\xDC\xF7\xA5\x86\xE7\x3E\xBF\x15\x66\xB8\x26\x9D\x36\x22\xE4\xE5\x08\xDB\xE5\x68\xBA\xE5\x10\x7B\x2D\xC7\x2D\x59\x8B\x59\x1E\xC8\x8F\xF4\x66\xD6\x42\xDC\xD4\x5A\xCC\xF2\x40\xF4\x96\xE5\xF3\xF4\x34\x68\xC7\x03\xF5\xAF\x2C\xCD\x07\xC3\x96\xDE\x53\xE4\x58\x3B\x62\x1E\xD2\xE6\xD3\xE3\xC4\x44\x0A\x82\x40\x7B\x07\x4E\xB5\xFC\x8F\x2D\x37\x29\x40\x96\x1F\x6D\x91\x84\xA8\x36\x04\x7C\x91\xC2\xC4\xF6\xF9\x24\x4F\x29\xD0\x9B\xB5\x2E\x7F\xD7\xE9\xC9\x03\x26\x47\xBA\x39\x22\xF5\x9E\xD6\x47\xC2\x1E\xC4\xCE\x46\x49\x98\x4A\x15\x68\x11\xA5\x64\x6C\x76\xAA\xD2\x39\xB9\x14\x86\xA7\x8A\x08\xFF\xAC\x17\x31\xED\x2E\x45\x1E\x99\x56\x64\x21\xA1\x8E\xE3\x4B\x1C\xD1\xF6\x59\x7E\xA9\xE9\x65\xE4\xC5\x03\x8E\x6A\x80\x8D\x70\x14\x03\x85\x0C\x8B\x66\xFF\xC1\x60\xBD\x6D\x2F\x3B\x53\xEB\xE3\xA0\x6D\x78\x62\x6B\xDA\x32\x0C\xFE\x58\x5C\x25\x9A\x5E\x7E\x9F\xBF\x7B\xC2\xA7\x3E\x63\xDC\x6E\x14\x41\x16\x4D\xD9\x22\x98\xF1\x36\xF2\x41\xA1\xF7\x38\x28\xF4\x0C\xC7\xAD\x67\x39\xEE\x70\x06\xDA\xC2\x39\x68\x0B\x4F\x4D\x28\x9A\xF3\x10\xDA\xF8\xED\x10\xDA\xC2\x39\x68\x9B\x2F\x05\x5C\x6A\x00\x6D\xBE\xD4\x10\xDA\xE8\x2D\x1B\x28\xA6\xA7\x91\x57\x79\x59\x83\xF2\x3E\xB9\xC6\xAD\xC9\x80\x64\x12\x2B\x18\x16\x6E\xEF\x71\x5D\xFD\x15\x1A\x80\x96\x71\x04\xB0\x0E\xEC\x0E\x30\x0D\x24\xF7\x68\x7B\x0D\xB6\x2A\xBA\x6D\x67\x19\x1B\x9F\xB0\x2C\x32\xB2\xC6\x7C\x34\xDA\xEB\x10\x68\x2F\xC6\x69\xE1\x04\xC8\xCD\x5A\x3C\xDF\xDF\x60\x17\xED\x39\x21\xF6\x38\x27\xDC\x19\xBE\xD7\x39\x41\x7C\x20\x4C\x71\x96\x83\x53\x42\x3C\x9F\xA2\x97\xF4\x4F\x09\x31\x7B\x4A\xF0\x19\x21\x86\x67\x04\xDD\x56\x20\x03\x2E\x41\x22\x0B\x2E\x1A\x64\x88\xD6\x89\x43\x65\x57\x53\xE6\xCB\x38\xDA\x6C\xCB\x5B\xE8\x86\x92\x7D\x41\x88\x6B\xC3\x6C\xAB\x37\x6E\x3D\xC3\xAE\x06\xEC\xAE\x70\x72\x52\x31\x5B\x0A\x54\xA3\x92\xE4\xD3\x2C\xAC\x3C\x59\x08\xE2\x22\x2A\xC1\x71\x25\xB4\x8F\xDF\x80\xEC\x88\xB3\xC5\x22\xE3\xBC\x27\xB7\x5A\x03\xD1\x97\x17\x8B\xA4\x99\x79\xF6\x5F\xF6\x34\x63\x9C\xAE\x53\x80\x80\xF0\xE4\xA4\x16\x20\xE7\xC4\x5E\x01\x92\xC5\x5E\x41\x3C\x33\x73\xFF\xB7\xE7\x2E\xC3\x09\xBD\xFA\x69\xD6\x03\xB0\xCA\x3B\x27\x3E\x7F\x31\x16\x86\x33\x58\xC8\x4C\x1B\xBB\xB4\xFB\x68\x0C\xB5\x98\xC1\x44\x31\x87\x89\xE2\xD4\x84\xAD\xB9\x06\x98\xC8\x6F\x87\x98\x28\xE6\x30\x71\xBE\x14\x70\xA9\x01\x26\xFA\x52\x43\x4C\xA4\xB7\x6C\xB8\x97\x22\xDC\x30\xDD\xDF\x7B\xBE\xB8\xD9\x7C\x52\x12\x53\xDE\x06\x9C\x20\x21\x09\x19\x93\xA4\x6F\x9E\x6A\x7E\x2D\x13\x7A\x11\x86\x39\xFA\x4D\x16\x62\xD4\x5C\xD2\x50\x30\x24\x77\xFB\xFF\xFC\x8E\xB1\xBF\x11\xF5\x9B\x23\x79\xAC\x98\xA3\x46\xE9\xD8\xC0\xDF\x77\xB8\x00\x91\x44\x44\x36\x1D\xED\xF8\x9C\x11\x98\x3E\x2F\x60\xF4\x44\xCB\xE8\x15\x43\x14\x2E\x86\x28\x5C\xDC\x9C\xB0\x4D\x9C\x4D\x77\xDC\xAD\xC9\x83\xA3\x34\x10\x52\x05\x41\x20\x02\x9E\x06\x85\x6F\x77\x26\x94\x4F\x9E\x1E\x51\xF4\x58\xBD\xD7\xAC\x3A\xF6\x75\x34\x64\x5F\x47\xB3\xEC\xEB\x68\x0F\xF6\x55\x2C\x60\x5F\x41\x10\x67\x6E\x25\x32\x90\xF6\xDA\xE1\x2A\xC0\x53\x9B\xA9\xFA\xAF\x3B\x8B\x29\x41\xC8\xD1\xED\x1C\x17\x0B\xFD\xCC\xDB\xF8\x9D\x37\x53\x07\x47\xE0\xCD\x53\xF9\x9D\xEE\xB5\x73\x93\xCD\x10\x1F\x2E\x86\xCD\xA8\x19\xB3\x96\x3B\x9C\x7E\xA1\xAD\x4D\x10\x0D\x0C\x86\x11\x99\x92\x97\x9F\x60\x96\xA0\x63\x74\x36\x8F\x74\xBA\x60\x62\xE3\xF7\xC7\x80\x39\xF8\x3F\x5E\x28\xBB\x73\x2D\xA8\x48\x8D\x40\xA6\x07\x92\xBD\x18\xC8\x70\x45\x7F\x75\x21\x3D\x1E\xD0\x5E\x53\xF3\x7B\x13\x36\x17\x9E\x96\x23\x51\xCE\xE1\x19\x0D\xFA\xAB\x69\x02\x0E\xD9\x9E\x6A\x8C\xDA\xEF\x50\xBC\x39\x8C\x12\xA6\xD5\xB8\x5F\x0B\xB6\x38\x5E\x27\x3B\x2B\x7B\x75\x50\xB8\xE9\xB8\xD0\xD0\xCD\x0D\x58\x2D\x43\xBC\x26\xC8\xF2\x0F\x68\xD8\x8E\xE1\x10\x9E\xE1\x08\x91\xE1\x40\xC1\xE0\xDA\xE5\xC1\xF6\x3D\x01\x0D\x35\xF6\x5F\xFE\x1A\x3B\x80\x89\xAA\x65\x6A\x20\x70\x37\x03\x1D\x5D\xFD\x98\x16\xE1\xBC\x91\x60\xA7\x18\xC1\x35\x7E\xC3\x5B\x67\x3C\x79\x69\x63\x3E\xF0\x7D\x83\x58\xF4\xEA\x89\xD1\xD9\x3A\x6C\x21\x2D\x20\xCD\x39\x57\x27\x5E\xFE\xFE\x82\x9D\x20\x14\x88\x13\xE5\x7F\x6A\x4D\x8C\x9E\x42\xA8\x90\x73\x50\x21\x07\x50\x21\x9F\x10\x54\x84\x34\x7A\x51\xEE\xF6\x25\x90\x04\x05\x64\x71\x44\x26\x24\x2A\x53\x34\x86\xF4\x25\x14\xD5\x8E\x74\x97\xFB\x4C\x0E\x22\xD0\xEB\x8E\xAC\x2E\x52\x72\x86\xA0\x89\x47\x9B\x53\x72\x86\x0B\xB9\x9C\x70\xA8\xE4\xA4\xC8\x59\x9A\x94\xFB\x1A\xC2\x39\x2E\x47\x43\xC8\x5C\x8E\xA6\xA0\xF7\x03\x2E\xC7\xBD\xEA\xB8\x1C\x94\x62\x4D\xDF\x8A\x54\x99\xDF\x54\x03\xEE\x58\x0F\x55\x24\xE1\x1E\x93\x0E\x8D\x0B\x21\xBB\x78\x47\xDD\xE7\x05\x3B\x4A\x24\x52\x0F\x76\x34\xB2\x1A\xF4\x8C\x8A\x44\xEF\xAD\x22\x09\x9D\x9E\xD6\x49\xF2\xCA\x5B\x0F\x38\xC5\x86\xEA\x14\x1B\xB2\x27\xC9\xCB\x56\x92\x97\x2C\xC9\x87\x9D\xAA\x7E\xC8\xDA\x29\xE6\x8E\xBF\x70\xA4\x2B\xD9\x4A\x57\x24\x5B\x75\xE4\xE3\xDF\x0F\x6F\x20\x9E\xEA\xAD\x0D\xE7\xB7\x36\x9C\xDD\xDA\xBD\x36\x96\x23\xAF\x94\xBF\x7D\x91\x3D\xC7\xDA\x29\xFC\xB0\x92\x61\xE7\x21\x08\x03\xDE\x92\x5C\x9B\x48\xD1\xE0\xB2\xD3\x88\x3D\x0E\x3A\xE1\x0E\x3A\x3A\xDF\x04\x02\x4D\xE4\xB7\x37\x3E\x5E\x90\xA2\xF9\x38\xBB\x7C\x10\x87\x20\x9C\x75\xAA\xE6\xF3\x34\x68\x6C\x76\x3F\x61\x1D\xEB\x0B\x41\x94\xFF\x81\x4F\x75\x14\x22\x0E\xDF\x13\x24\x54\x9C\xB4\xE1\x08\x7E\xEA\x88\xBC\x03\x94\x95\x47\xE4\x41\x36\x5F\x13\x36\x80\x60\xB3\xFC\xF8\x45\xE7\x5C\x8C\xF0\xA7\x88\xCA\xE2\x78\xD4\x9A\xBC\xA3\x1D\x4F\x15\xE3\x73\xC9\xA6\x10\xE9\x19\x77\xF5\x40\x56\x67\x54\x38\x6A\xEC\xC3\x0D\xEB\x5F\x52\x10\x93\x9A\x6E\xAE\x09\x22\x5C\x03\x0F\xD6\x4E\xC4\xB3\x19\x08\x0A\x09\xC7\x7E\x85\x17\xEA\x84\x22\x32\x07\x1B\x85\x80\xE4\x18\x3C\x7A\xD5\x92\x17\x0E\x47\xDB\xF9\x6A\xF2\xB9\x20\x85\x14\xCB\x87\x7A\x50\xD3\x59\x31\x24\x57\x59\xB2\xF4\xE7\xAA\xA4\x3B\xF8\x6D\xCA\xF9\xD4\x65\x62\xD3\xE6\xBA\x5A\x2C\x0D\xF8\xFD\x42\xB0\xBA\xF4\xF6\xC1\xA9\x25\xE7\x4F\x2D\xBF\x9F\x0B\x4E\x2C\xD1\xE9\x45\xFC\xA9\x25\x66\x4F\x2D\xE9\x32\x59\x9E\x60\xF1\x7B\x15\xD8\x69\xB5\x67\x22\xB0\x26\xDD\xA1\x2F\x9E\x2D\x93\xBB\x59\xED\xD0\xDE\xE0\xE1\x8F\x31\xA9\x93\x7D\x1C\x6C\x82\x85\x08\xA4\x3F\xFF\x22\x90\xEE\xFC\xBB\xFD\xA9\xBD\xD3\x53\x73\x77\x7A\x9F\x0B\x53\x44\xC9\xD9\x36\x91\x54\x26\xED\xBA\x80\xE4\xD3\xF0\x76\x4A\xCE\xD0\xEA\x17\x6C\x00\xD1\xE6\x7A\x43\x47\x46\x7F\xBD\x16\xAD\xD2\x80\x6D\xF9\x46\xB2\x63\xA1\xAC\x66\x8F\xD3\x29\x2B\xD9\xE0\xCC\xDB\x22\xA3\x24\xB3\x23\x69\xBA\x4E\x07\x47\x61\x95\xE9\xCE\x8C\x32\xBE\x81\x84\x80\xAC\x16\x08\x31\x5D\xA2\x4E\x22\xF4\xF8\x8B\xE2\xE3\x0A\xD0\x1E\x02\x5D\xDB\x06\x94\x79\x93\x74\x77\xF0\x6C\x0F\x43\x80\xA2\x9E\xB4\xBA\xDF\x3C\xE9\x23\xBB\x7F\x2B\x19\x2E\xBC\x95\x0C\x6F\xEA\x56\xB2\x8B\x4E\xD0\xC2\xE8\x8D\xF7\xC1\xF1\x48\xEE\x3F\x79\x07\xF1\xC6\x0A\xF3\x2E\xC5\x17\x53\x4F\x01\x1E\x76\x58\xB8\x00\x07\x41\x95\xBF\xCF\x7A\x62\xCB\x3E\x28\x20\x9B\x5A\x6F\x92\x1F\x79\xFA\x92\x3A\x42\x36\x2B\xDE\x7F\x97\x20\x86\x70\x9D\xD3\x2C\x2F\x64\xB3\x22\x08\x49\x8E\xE9\x9D\x48\x37\x79\x97\x1C\xBA\x48\xFB\x88\x0C\x27\x27\x75\xB8\xF0\x26\x39\xF4\x37\xC9\x21\x28\xB7\x67\x9C\xC5\x8A\x5E\x74\x86\x14\x7E\xCF\xDA\x8D\xA0\x73\x24\x07\x59\xFE\xA0\x13\x39\x75\x7B\xC9\xEC\x29\x10\xAF\xC9\x91\x3D\x89\x90\x15\xE6\x4A\x48\xE6\x19\x8B\x6C\x30\x07\x8A\x71\x61\xF5\xA9\x4A\xA2\x88\x2D\x58\x31\x2E\xE6\x14\xE3\x12\xA5\x64\xD1\xEA\xC0\x59\x31\x2E\x51\xE6\x15\xC7\x8B\xB0\x45\x4A\x12\x2A\xB1\x11\x8E\x91\x2A\x6C\xB0\xCE\x51\x51\xC5\x8C\x62\x5C\x1E\x07\xC9\x8A\x71\x49\x69\xCB\x84\x99\x55\x8A\x57\xC2\xDD\xD3\x3A\x4B\x9C\x27\xAF\x2B\xE4\x7B\x65\xBD\x40\x65\xB8\x2F\x66\x0A\xCF\x4C\x0B\x90\xF7\x4F\x6A\xB9\x10\x33\xA5\xC7\x4C\xD9\xEE\xB2\xE4\x5D\x96\xED\x2E\xF3\x2C\x65\x6F\x96\xC4\x30\x8A\x56\xBB\x9F\x5B\xD1\x10\x9F\xC9\x38\xB5\xF7\x9D\x75\x9F\xEF\x94\xB3\x8A\xFD\x2F\x00\x75\xA2\xEC\xAB\x13\x8D\xF9\x1A\x21\x66\xAE\x8E\x3C\x5B\x35\x9A\xF2\x7D\x09\xDD\x23\xF9\x97\x39\xFB\xC3\x0C\xF9\x09\xA2\xE6\x4C\xF3\xCA\x4F\xD2\x96\xBF\x4D\xB5\x44\x7D\xDD\x21\x17\x79\x85\x4A\x2B\xA7\x4C\xDD\x59\x39\x26\x5D\x6E\x64\xE5\x13\xDC\x90\xBA\x6B\x73\x6C\xAC\xA0\x5F\x56\x9C\xDB\x1C\x9B\x7E\x01\x2B\xCE\xD9\xED\x66\x6C\xF6\x37\x2F\x68\x8F\x92\xBE\x9D\x41\x77\x67\x1F\xD0\x09\xA6\xF6\x20\x91\x74\xBF\x34\x4B\x1E\x85\xE7\x37\x7B\x83\xB8\x41\x03\x2D\x52\x52\xB1\x61\x63\x08\x16\x38\x1F\x37\x3D\xE9\x5B\xAE\x15\xE7\x7B\x94\x9B\x7E\x8A\x4E\x09\xCB\x77\x72\xCC\xDF\x36\x94\x15\xD2\x06\xDC\x80\x9B\xC7\x76\x43\x86\x27\x64\xD9\x68\xB7\xCF\x72\xC6\xC9\x2D\xB7\x2B\x7F\x44\x2A\x31\xE1\x87\x84\xD5\x4E\xB3\x6D\xE8\xFB\x3A\xCB\x07\x5D\x4B\xE6\xA2\x15\x4B\xAC\x12\xD4\x66\xED\x8D\x30\x98\x2E\x7A\x0A\x49\x5E\xB6\x48\x1F\xBF\xED\x95\x3B\x3B\xC1\xE6\x80\x76\x32\x40\x93\x0B\x19\xC8\x4D\x2E\xC1\x05\x70\xB7\x04\x5B\x64\x50\x58\x7F\x67\x89\xC2\x41\xD2\x88\xA5\x65\x48\x21\xA6\xC9\x6B\xB3\x41\x72\xD2\x3F\x0E\xCC\x4F\xF2\x1D\x85\x7D\xA7\x35\x30\x76\xDB\x0A\x3C\x02\x36\x26\x48\x6D\xF5\xB6\xD3\xDC\xF8\x00\xAF\xED\x25\x0B\x47\x4B\x70\xB7\x23\xDB\x0D\xDB\xC4\xB8\x88\xAF\xB2\x8D\xFE\x7F\xDC\x99\x5C\x68\xA4\x97\xF7\x3B\xB3\xB5\x81\xA0\x49\x46\x33\x33\x48\xAE\xE6\x90\x5C\x2D\x94\x2F\xD5\x02\xF9\x52\xCD\x21\xF9\x7C\x29\xE0\x52\x03\x24\x57\x0B\xE5\x4B\x7A\x6B\x64\x17\x25\xBA\x8E\x7C\xA4\x7A\x69\x7B\xD1\xED\x21\x9A\x89\x29\xAB\xDB\x98\xB2\xBA\x1F\x53\x56\x77\x31\x65\x75\x3F\xA6\xAC\xEE\x62\xCA\xEA\x41\x4C\x59\xDD\x8B\x29\xAB\x29\xA4\xAB\x8B\x4E\x28\x65\x3F\x94\xF1\x3F\x6F\xE6\xC2\xCB\xEA\x5E\x78\x59\x3D\x08\x2F\xAB\x7B\xE1\x65\xF5\x30\xBC\xAC\xEE\x87\x97\xD5\xC3\xF0\xB2\xBA\x1F\x5E\x56\xCF\x84\x97\xD5\x83\xF0\xB2\x9A\xC3\xCB\x6A\x67\xDF\x73\x92\xF8\xA0\xA0\x05\xA5\x87\x7D\xAC\xE0\xFD\x20\xC7\x41\x8D\x01\xDD\x0B\x32\x4B\xC6\xD8\xEB\xEC\xC0\x6B\x3E\x15\x0D\xB4\x83\xFA\x73\x93\xB3\xF6\x96\x9B\xDB\x95\x9E\x95\x9E\x3D\x8F\x47\xF9\xD4\x7A\x7C\xDE\xA4\x05\x76\x96\x9A\x62\x96\x9A\x22\x08\x21\x6E\xA5\x26\xA2\x64\x7F\xF3\x14\x11\x7B\x8A\x4D\x31\x8A\x4D\x31\xB2\x68\x31\x32\xAD\x09\x24\x2C\x22\xC5\x9B\xEB\x3E\xA9\xBB\x4B\xD0\x1B\xD1\x4C\x9D\x4C\xC5\x2C\xCF\x1E\xD3\x14\x37\x9C\xA6\x19\xB2\xB2\x62\x21\x2B\x2B\xE6\x58\x59\x64\x93\x71\xA5\x3D\x23\x2B\x3C\x44\x0C\xD8\x58\xD1\xB1\xB1\xB7\x13\x77\xD7\x71\xB1\xB7\x3B\xAB\x84\x27\xB5\x37\xE6\x49\xF3\xDF\x33\xC3\xFE\x5C\xF8\xEF\x6D\xE6\x5D\x87\xB2\x50\xB7\x23\xB4\x9D\x37\x29\xD6\x76\xBA\x28\x90\xE6\xAD\xCB\x62\xB9\x87\x71\x61\xFF\x9E\xD3\x07\xDA\x21\x9E\x47\xE2\xEF\x98\x7F\x47\x2E\x9A\x31\xFF\xC1\xFE\x0A\x0A\x51\xE1\x2C\x48\x9E\x22\xCD\x39\xA9\x0E\xC2\x39\xD5\x41\x38\xA3\x67\x0D\xF7\xD6\xB3\x2E\xB8\xCB\x13\x3D\x53\x34\x4D\x07\xA7\xE3\xDA\x9C\xCB\x09\x4E\x8B\xB4\x61\xC9\xFD\x75\x32\xA9\x53\xA4\x1F\xAD\x60\x17\xED\x31\x45\x76\x58\xDB\x0F\x94\x22\x06\xA5\x6C\x21\x28\x65\x1D\x28\x45\x0C\x4A\x88\x83\x08\x4A\x11\x64\x0B\x40\x29\x82\x8C\x41\x29\x02\xED\x40\x29\xF2\xCA\x11\xED\x40\x29\x32\x10\x79\x1B\x1F\x92\x1B\x24\x24\x93\xDA\x10\x73\x8F\xD4\x2C\xA5\xC8\x20\x60\xF0\x4F\x41\x19\x9F\x84\x99\x7B\x19\xB9\x30\x6D\x39\x44\x6C\x1D\x6F\xB3\x07\xAB\x18\x0A\xF7\x28\xD8\x4F\x8A\x9C\xF0\xEB\xC4\x8A\x53\x74\xB7\x14\xD3\xE3\x98\x1E\xA9\xEF\x31\x24\xAC\xE4\x2C\xE9\x0C\x5C\x93\x74\x46\x57\xA5\xCB\x5B\x41\xB9\xD9\xB3\x33\x14\x94\x02\x1F\xC7\x90\xA1\x54\x96\x71\x7B\x71\xD7\x4E\x0C\x19\xB7\xB3\x84\xED\x78\x06\xA3\x5A\x72\x79\x31\xAA\x0C\x12\xDF\x0E\x3E\x26\x30\xC6\x76\xC6\x0B\xDA\x19\x73\x3B\xCB\xBD\x76\x92\x6A\xD9\x67\xDE\x70\x23\xA0\x86\xE8\x99\x1B\xF6\x33\x1C\xB4\xE4\x66\xB6\xD2\x6B\x29\xAF\x56\x0C\x3B\xC6\xF9\x31\x50\x4B\xF4\xBC\xFF\xDC\x56\x7B\x2D\x95\xD5\xAA\x61\xFF\xBC\xC1\xE4\xE8\x79\xFF\xD9\x1D\xE8\xB5\xB4\x5A\x1D\x30\xCE\x4F\x70\x30\x3D\x7E\xB1\xFF\xFC\x6E\xEB\xB5\x75\xB0\xBA\xCD\x38\x97\x45\x04\x06\x9A\xA0\xF4\xCE\x8D\x55\xC2\x33\x14\x0E\x1A\xBA\xB6\x24\x08\x6E\x2B\xEB\xB5\x75\x47\x95\x19\xDE\x7D\xC1\x63\xE0\x0A\xE0\xAA\x67\x70\x1B\x1C\x80\x55\x58\x81\x65\x58\x82\xB2\xE1\x7F\xEE\xB8\x12\x20\xB9\x41\xD1\x6B\x10\x2A\x2F\xBD\x47\x0D\x08\x88\x9B\x5E\x72\x89\x11\x8A\xB5\x23\x30\x6C\x20\x72\x92\xF3\x53\x50\x50\xFC\x5A\x34\xCC\x95\x75\xDC\x6D\x01\xE2\x09\x48\xB1\xE5\x42\x29\xB6\xBC\x29\x29\xB6\xBC\x29\x29\xB6\x5C\x28\xC5\x96\x6C\x32\x56\x49\x28\xED\x76\x53\x2D\x51\xEC\x6B\x46\xAF\x08\x4A\x4E\xA5\x53\x27\x88\xB5\x12\x1C\xB6\x8E\x09\x3D\x69\x81\x05\x24\xEC\xD3\x58\x17\xF4\x48\xCB\x5A\xF8\x7D\xA2\xD4\xA0\x94\xA2\x81\x5C\x7F\x69\x09\x71\x45\x71\xCF\x29\x57\x4F\x9B\xFF\x12\x99\xAA\xB1\xCB\x2E\xE0\x92\xCC\x2D\xD9\xE4\x85\x24\xA5\x66\x67\xB8\xA7\xD2\x26\x28\xCD\x25\x88\xE2\xBD\x97\x31\xBF\xFC\x44\xDC\x7F\x19\xF1\x4B\xC4\xBD\xDE\xDB\x90\xDF\x22\x1E\xF5\xDE\x6A\x7E\x8B\x38\xD1\x7B\xAB\xF8\x2D\x81\x77\xEF\xB5\x74\xAF\x11\x52\xFD\x6B\x24\xB8\x23\x8A\x41\x40\xF7\xE0\x28\x33\x8E\xEC\xF6\x59\x10\x4D\x95\xD1\x9C\x20\x6F\xDA\xE4\xCC\x7C\x97\x9E\x3D\x58\x47\x0E\x83\x20\x22\xD7\xFE\x1E\xE2\x30\x15\x73\xAD\x4B\xA6\x45\xBD\xA7\xDE\xA4\xA4\xA3\x0A\xBD\xC7\xDE\x34\xA4\x47\xCF\xFE\x73\x6F\xE0\xD2\xDB\xC9\xB7\x1B\xD1\x99\x08\xD3\x89\x65\xCF\x4C\xEB\x0C\x8F\x3A\xE3\x1C\x1F\x9C\x91\x5C\xCC\xCC\xFC\xF8\xC6\x46\x72\x91\x17\x33\xA3\xCE\x48\x2E\xF6\x9C\x07\xE7\xF8\x5A\x78\x1C\x66\x9C\xE9\xE9\x56\x9A\xC9\xC5\x64\x26\x17\xEF\x6F\x26\xA7\x67\x6C\xE4\x74\x77\x75\xDC\x06\x37\x52\xE5\xA7\x2E\xB6\xA1\x8D\x76\x24\x68\x1F\xDB\x28\xEC\x2E\x00\x7E\x2B\x1C\xDC\x00\xC9\x81\xC5\x25\x4A\xE7\x9D\xCD\x25\x01\x04\xBB\x2E\xA5\x2F\xA1\x9C\xDA\x6C\xD8\xA8\x7B\x86\x8D\xBA\x75\xA5\x19\x1A\x36\xD2\x7D\x4F\xCD\x99\x21\xFB\x17\x57\x0B\x0C\x1B\xD9\xC9\x13\x42\x4E\x3F\xB7\x60\xCF\x34\xC5\xDF\xF5\xB9\xE7\xC2\x6E\xCF\xD4\xAD\x35\x6C\xDC\xD7\xA7\x8F\xB5\x64\xB4\x63\x8A\x76\x6C\xDE\x9F\x4F\x78\x7F\x3E\x41\xF1\xA2\x66\x36\x8D\x5E\x75\x57\xFE\x8A\xC1\x6D\x81\x6A\x29\x60\xD5\xD2\x8C\x52\xE2\x0B\x50\x0B\xA9\x06\x5A\xC8\x19\xBF\xA1\xBD\x27\x3E\x37\xD7\xA2\x73\xFB\x9B\x72\xA2\x77\x5D\x7E\xCC\x6B\x9D\x5B\xE0\xBE\xB2\xFA\xB4\x10\xF0\xB4\x10\xF0\xDF\x43\x08\xD8\xBA\x45\x42\xC0\xD6\x2D\x12\x02\xB6\x6E\x99\x10\xB0\x75\xCB\x84\x80\xAD\x5B\x26\x04\x6C\xDD\x42\x21\x60\xEB\x16\x0A\x01\x5B\xB7\x5A\x08\xD8\xFA\x9B\x22\x04\x14\x0B\x0F\x91\xE2\xA6\x0E\x91\xE2\xA6\x0E\x91\x62\xE1\x21\x52\xB4\x42\x40\x81\xCC\x40\x39\x10\x02\x8A\x9B\x17\x02\x4C\x27\x04\x18\xBF\x4F\x37\x2B\x04\xC8\xA3\xC8\xE1\x7B\x96\xDF\xB5\xEC\x98\x9A\xC2\x31\x35\x20\x4E\x73\xCD\x82\xC2\x83\xDA\xB4\x2F\x10\xF0\xA6\xFB\x2A\xC2\x57\x91\x6D\x15\xD9\xAB\xC2\x2C\xF5\xBE\x55\x54\xAF\x8A\xE3\xBB\xF7\xAD\xA3\x7B\x75\x1C\x73\xBE\x6F\x9D\xB0\x57\xC7\x71\xF0\xFB\xD6\x89\x7A\x75\x3C\x9B\xBF\x6F\xA5\xB8\x5F\xC9\xC9\x02\x37\xAE\x34\x90\xCE\x6A\xD9\xCA\x67\x84\x06\x71\x4F\xE6\x19\x93\x38\x51\x0B\x8E\x95\x2E\x66\xE5\x1D\x22\x16\x09\x08\x17\x71\x91\x10\x33\xF9\x3C\xCB\x3B\x11\x88\x1E\x22\x3B\xE9\x27\xEE\x4F\xE4\xE4\x0D\xA4\xA0\xA7\x05\xA0\x9B\x17\x80\xDE\x5B\x76\xD1\xDA\x3F\x28\x37\xEB\xC8\xA7\x3F\xA0\x74\x79\x3B\xC9\xDD\xF2\x4B\x21\x42\x86\xEA\x2E\x88\xEC\xE1\x7B\x82\x35\x8A\x40\x79\x2D\x68\x8E\xC8\x2F\x47\x5E\xF1\x88\x7C\x16\x44\xFE\x2A\x62\xD5\x55\x2B\xEF\x96\x39\x44\xF6\xB5\xE2\x9E\x20\x21\xDE\xB1\x8A\xD9\xF2\x30\xA1\xDA\xCF\x6A\x8E\xC8\x3B\x38\x45\x9A\xE4\x44\x33\x14\x79\x4C\x50\x8A\x10\x3E\x3A\xF0\x28\xBA\xBF\x4E\x27\x75\x8C\xBB\x9D\xCC\x73\xC6\xF8\xBE\x4A\x0C\x31\x3A\xE9\x84\xEE\x2E\x12\xDC\xEA\x78\xC0\x24\xC7\xAE\x49\xF6\x4A\xD7\x75\x46\x2C\x5F\x63\xB7\x1B\x2C\xA8\x6B\x63\xA9\xA5\xA4\x4E\xF8\x72\x35\xB7\xDB\x67\x1B\x48\x20\xBF\x7F\x42\xFE\x3F\x39\x8A\x7F\x59\x95\x13\x69\x74\xB3\x54\x25\x44\x90\x23\xA3\x19\x11\xE3\x6C\xC7\x53\xFB\x5A\xB1\x75\x4F\x90\x70\x2F\x7B\x0D\x57\x57\x86\x83\xE3\x26\x06\x22\x1E\xAC\xEC\x06\x6B\x83\x75\x5A\x4A\x9B\x34\x15\x05\xBD\x5C\xE3\xFC\x8E\x6B\x9C\xE5\x2B\xDF\x84\x90\x5D\x34\x8A\xA7\x2E\x30\x47\xFC\x7C\xE2\xFF\x83\x7B\x82\x0F\x48\xCF\x85\xC7\x56\x43\x64\x3F\x20\x07\x6C\x78\x74\x28\xF8\x80\xDC\x83\x0F\x97\x0B\xBC\xDE\x7B\x7C\xB8\x73\xEE\xE3\xE0\x42\xCE\x5C\xC5\x5D\x6C\x48\x48\x20\xA1\xEB\x8F\x35\x99\x9F\xAC\xE3\x09\x2E\x56\xEE\x67\x6B\xF6\x98\x2D\xC7\xAE\xDB\x0F\x09\xCD\x4D\xF2\xE3\xD4\x5A\x0E\x06\x97\xFC\xE4\xA4\x36\x0B\xF9\x71\xE3\xF9\x71\x03\xA1\xE3\xC7\x0D\x4F\x05\x5F\x30\x3F\x6E\x0C\x18\xB2\x19\x49\xEC\x6A\x83\x70\x87\x7F\x0A\x88\x27\x95\xA1\x4C\xBA\x75\x42\xE2\x6F\x2D\xF1\xD5\x08\x0A\xC8\xF0\x47\x46\x8B\x10\x4F\x38\x40\xB7\xBF\x0E\x13\x53\x0A\x41\xE6\x3D\x78\x33\x24\xFA\x06\xFF\x53\xE0\x7F\x46\x6B\x32\x28\xAF\x60\xA7\x94\x1A\x90\x5E\x76\x57\x96\xF1\x1E\xCB\x46\x11\xB6\xE3\xBD\xED\x96\xF2\xFE\xEA\xC5\xBC\x7A\xC9\xC2\xD5\x4B\xBA\xD5\xE3\x46\x21\x87\x98\xEC\x96\x62\x48\x16\xAC\x5E\x0C\x09\xAF\x5E\xDC\xAE\x5E\xEC\x2F\x2E\xFD\xEA\xC5\x14\xA3\x3E\x34\x5E\x12\x94\xE7\x1A\x30\x2E\x1F\x35\xA1\x1E\xB5\x77\x66\x0A\x6A\x8B\xE3\xE7\x48\x2B\x20\x9D\x40\x8A\x6B\x37\x66\xE5\x26\xA7\x02\xA4\xDC\x41\x12\xC9\x4B\xDA\x86\x23\x3F\xD7\x9D\x06\x2E\x7C\x30\xDD\x6D\xD3\x92\x4A\x44\xBB\xCC\xFD\xCA\xFD\xC1\x90\xF7\x92\x80\xE1\x9C\x90\x91\x24\x16\x68\x8C\x8D\x8E\x21\xA5\x34\xDF\x71\x2E\x0D\xFE\xE4\x78\xC0\x59\x65\xA0\xA8\x46\x6E\x0E\xD4\x78\xBC\x46\x09\xBD\x64\xF9\x56\x5A\x3E\x1E\x1B\x5B\xC4\x9C\x9C\x50\x6F\x3D\x31\x1F\xD9\x5E\x8A\x4E\x42\xE7\x15\x25\xF2\x97\xD4\x02\x09\x50\x94\xBC\x9B\x72\xF5\xBA\xFD\x32\x4F\x9A\x2C\x0C\x10\x45\xF2\x56\xC7\x0B\xB7\x3A\xEE\xB6\x5A\x32\xA2\xA0\x00\x88\x88\x22\x21\x9E\x3B\xAD\x24\xC4\x7C\x5A\x49\xBF\xD1\xB7\x7B\xFB\xB4\xB0\xA7\xF6\xC9\xF9\x90\x1D\x3B\x85\x83\xB3\x1E\xA2\x50\xFC\xED\x45\xA7\x62\xE7\x07\x98\x8F\xCF\xCC\x2A\x92\xDC\x40\x3E\xAB\x22\x11\xAE\x81\x5A\xBA\x58\x48\x48\xE2\x0F\x5A\xB5\x4E\xAB\x29\xB6\x5C\x88\xA5\x7B\xD8\x4B\x6E\xEF\x96\x87\x72\xC2\x6C\x37\x5F\x08\x61\xA4\x72\x9B\x9E\x06\xD9\xD3\x3A\x45\x87\x82\xB5\x5A\x72\x14\xB1\x68\x4D\x7E\xF9\x5C\x14\x31\x2C\xE0\x01\xA6\x64\xAD\xD0\x5A\x33\x17\x48\x0C\x22\xFB\x41\xE9\x19\x89\x9F\x51\x03\x17\x8E\x70\xC6\xFB\x81\x54\x4D\x3E\xCB\x47\xEE\x4C\x8B\xF1\xCC\xC5\x23\x6F\x6A\xC3\x73\x5B\x6C\x93\x88\x9C\xCC\xEF\x7B\x13\xEF\x61\xB0\x2B\xE4\x68\xD6\x28\x9A\x48\x2F\xA2\x03\x67\x40\xE8\x6A\x91\xBD\x0B\x31\x75\xCE\x02\x8E\xC2\x1B\x45\x40\x48\xC4\xCE\x1B\x9A\x0F\xD2\xDB\x7B\x1C\x92\x34\x64\x42\xCE\xD6\x1F\xBD\xE6\x38\x8E\x84\x2B\x05\x6C\xF2\xED\xA2\x00\xE4\x4F\xA1\x36\x4C\xCD\x6B\xC3\xD4\x53\xA7\x0D\x1B\xF0\x83\x9F\x5A\x79\x5A\x67\xF8\xB4\xCE\xF0\x96\xEB\x0C\x93\x9E\x92\xB0\xD5\x19\x96\xF4\xE8\xD2\x60\x8F\xA1\x24\xEB\x4E\xAF\x2E\xCC\xBC\xBA\x70\x3C\x54\x17\x96\xAC\x38\x5A\xEA\x2B\xAF\x6C\x50\x21\xF3\xBD\x44\x87\x50\xA7\x29\x4C\xBC\xA6\x70\xA9\x53\xCB\xE1\xE3\x32\x94\xD8\x44\x39\xD3\xC4\x58\x1E\x85\x72\xD0\x44\x52\x8D\x5B\x25\xE1\x0A\x2C\x0D\x94\x84\x4B\xB0\x8C\x8D\x2C\xCF\x34\x52\xCA\xA3\xB0\x3C\x68\x24\x67\xCD\x27\xE9\x03\x97\x61\x65\xA0\x1F\xE4\x46\xEB\xD5\x99\x46\x96\xE4\x51\x58\x1D\x34\x52\xB2\xDA\x93\x54\x81\xAB\xD4\x73\xA7\x1A\x3C\x40\x8D\xD6\x2B\x33\x8D\x2C\xCB\xA3\xB0\x32\x68\x64\x95\x75\x9E\xAC\x04\xBC\x0D\x56\x87\x5A\xC1\x55\x38\x80\xCD\x1C\x98\x69\x66\x45\x1E\x85\x03\x83\x66\x0E\xB2\xC2\x93\xF5\x7F\x07\xE0\xB6\xA1\x42\x90\xDB\x1D\x2A\x04\x5D\x7A\x23\x16\x69\x3B\x5D\x60\x6C\xA8\xCB\x4A\x50\x1B\x7E\x15\x5C\xCD\xD8\x6B\x01\x61\x0C\x09\x64\xAD\x2E\xD0\x85\xCA\x5D\x25\xBF\x99\x4E\x0D\x28\x3B\x35\xA0\xFC\xBC\xA9\x01\x97\x17\xAA\x01\x97\x6F\x4A\x0D\xB8\x7C\x53\x6A\xC0\xE5\x85\x6A\xC0\xE5\x56\x0D\xB8\x8C\x6C\xE1\xCA\x40\x0D\xB8\x7C\x03\x35\x60\x39\x54\x03\x2E\xB1\x3E\x89\x43\xD8\x95\xB0\x74\x82\xF9\x81\x9B\xD0\x00\xAE\xEC\xA5\x01\x5C\x9E\xD5\x00\x2E\x3F\x71\x0D\xE0\xF2\x13\xD7\x00\x2E\x3F\x09\x0D\xE0\xF2\x93\xD0\x00\x2E\x3F\x09\x0D\xE0\xF2\x93\xD1\x00\x2E\x3F\xAD\x01\x7C\x5A\x03\xF8\x79\xD1\x00\xBE\xD2\xA7\x2C\x66\xDF\x94\xDC\x05\xE2\x25\x56\x5B\xCF\xF0\xE6\xEA\x06\xBC\x39\x27\x34\x1B\x46\x5B\x63\xA7\xDD\x2E\x24\x0A\xA8\xF2\x2F\x2E\xF6\xE2\xB0\x81\x2A\xFF\x2B\x0D\xE8\x2F\x9F\xD2\x41\xD0\x30\xDC\x2D\xFB\x3E\x2E\x82\x73\xEE\x81\xCC\xC9\x1E\x64\x3B\xA3\x5E\xF7\xC7\x5B\x01\xD3\x8B\x97\xBD\xD8\x74\x63\x33\x2C\xDA\x5D\xD7\x9F\x25\x5B\xDD\x7E\x18\xBB\x3F\xFC\xDB\x3C\xF5\xDB\x67\xE6\x1A\x0D\xCC\x6D\xE6\x85\x44\x27\x56\x6C\xD5\x49\x2B\x2E\x26\xBC\x24\xEC\xC0\x61\x93\x9E\xE8\xD8\x71\xE6\xA4\xCB\x5A\xF0\x09\x24\x5F\x6B\x21\xA8\x45\x56\x4F\x9D\x19\xC6\x93\x11\x32\x63\xEF\xD3\x14\xB1\xFB\x16\x09\x3D\x95\x22\xBA\x43\xE7\x67\xEC\xDE\x2B\x8F\xB6\x5F\xE8\x02\x8F\x32\x4F\xC5\x24\x74\x3B\x89\x36\x74\x93\xA6\x49\x8C\x66\x27\xF1\x04\x42\x37\xA9\x41\xE8\x26\x0A\x52\x45\xFB\x51\x04\xF6\x77\xBF\x7F\xE0\x41\xF2\xD4\x88\xA2\xDD\xA4\x8A\xE1\xA4\x8A\xD9\x49\x2D\x8A\xB3\xB5\xC7\xA4\xE4\x70\x52\xDE\x68\x8A\x20\x11\x14\xE8\xF2\x4F\x3D\x95\x07\xED\xF0\xAB\x4F\xD7\xFF\xF8\x0B\x19\xD7\x9C\xA7\x6F\x0F\xDD\x5A\x3C\x8B\xC8\x5F\x90\x90\xCC\xE3\xDC\xD3\xC8\xF6\x34\xB2\x3D\x65\xC8\x46\xA0\x78\x13\xF8\xB6\x4A\x9E\xA6\xCC\xD5\x80\x2E\x3F\x43\x05\xBE\xFF\x80\xC8\x3D\x12\x5E\xD3\x8B\xAE\x57\xFF\x9D\xE4\xFB\xD5\x1F\x94\x7C\xC1\xFA\x36\x09\x8A\x2E\x68\x08\x3D\xDD\x8D\xE9\x35\xD9\x1C\x91\x3F\x21\xFD\x6D\xE9\x56\x9D\xB6\x28\x9B\xF6\x50\x36\x71\x78\xE9\x2F\xD9\x16\xA0\xEC\xE0\x13\x28\x76\x19\xC7\x11\x27\x84\xB2\x92\xF1\x37\xF4\x28\xEB\x7C\x7F\x3B\x94\x65\xFB\x53\xD0\xF3\x28\x2B\xA7\x9C\xBE\xB7\x45\xC9\xD4\x61\xE6\x1E\x71\xF7\xC5\x0D\xB3\x4C\x0C\x3E\x83\x53\x2B\xCE\xA2\x27\xDD\x55\x02\x0E\xD2\xA3\x67\x64\x61\xF6\xA6\x12\x7C\x1E\x9B\x0E\x08\x5C\xDB\x9C\x4D\x36\x07\xC1\x40\x20\x40\x2D\xB8\xA8\xEC\xA3\xE7\x8D\x67\x02\x31\x24\x67\x9D\xDB\xF3\xDE\x13\x22\x77\x6E\x57\x6A\xC1\xBC\x92\xF9\x79\x25\x9F\xF3\xBC\x24\xCF\x6C\x80\xB1\xE9\x10\x5D\x87\x40\x87\x50\x90\x40\x7C\x96\xCD\x4B\x35\x85\x86\x76\xA1\x03\xD9\xCE\xD4\x2F\x8F\xE8\x20\x4C\x71\xE2\xB9\x39\xEE\x12\xDF\x57\xA9\x63\xA0\x62\x14\x24\xA7\x55\xE6\x6E\xC2\x09\x7E\x39\xB4\x50\xC7\x74\x12\x10\xCD\x35\xE3\xAE\xCE\xFD\x3D\xBF\x9C\xA9\xF1\x85\x0D\x65\xE5\x96\x8F\xAD\xF9\x39\x4E\x04\x62\xC8\xCE\xFE\x77\x85\x2E\x3F\x25\xDD\x41\xC6\x8D\x76\x73\xE1\x7E\x92\xFD\x47\x76\xD6\xF2\x25\x02\x42\x87\xDE\x64\xB8\x73\xD7\x3B\x64\x55\x92\x6C\x12\x50\x36\x35\xDD\xEC\x1A\x48\xED\xF6\xD9\x2A\x67\x7B\x93\x82\x2A\x55\x23\xA7\xDA\x4E\xDD\x35\x90\x7D\x9B\x64\xB3\x8A\xB7\xC9\x06\x24\x5D\xD7\xB3\x51\x85\x00\x61\x77\x34\x84\x6D\x5A\x6D\xAA\xB4\xBD\x8E\x22\x10\xD9\x90\xB8\x3A\x0A\x72\x5F\xA7\x1A\xBB\xF6\x14\x16\x81\x31\x2D\x58\xAF\x0D\x61\xD8\x9A\x81\xF0\x62\x4C\xBA\xB7\xC6\x3B\x84\x53\x4C\xC2\x22\x80\x04\x04\x65\x98\x57\xF4\x17\x4A\x18\x9F\xE4\x1B\x13\x82\x82\xF1\x1E\x50\x40\x72\xF5\x78\x3F\xAD\xC3\x98\xFD\x70\x97\x16\xFA\xE1\x2E\x75\x5A\x87\xB1\x81\x31\x48\x18\x43\x79\xFF\xA4\x1E\x93\x12\xB0\x33\x3D\x60\xAD\xC3\x18\x96\xD8\x13\x77\xDC\x5E\xD9\x8F\x99\xD5\x1B\xB7\x57\xF6\xD8\xCC\xFC\x12\xF0\x64\x69\x05\x2C\xB9\x1C\x33\xCC\xB9\xFF\x70\x74\x31\x3C\x47\xDE\x4E\x6D\xB0\x7F\x14\x8C\xED\x4E\x72\xBF\x2B\x6B\x03\x2E\xAD\xDA\x2A\xE3\xF2\x07\xC8\xD8\xA1\xC4\xD5\x99\xDB\xB7\xDE\xC9\xBA\xCA\x07\x6B\xCE\xE7\x2A\x1F\xA4\xD0\x1C\x91\x77\x90\x2F\x4F\xD2\x40\x02\x0A\xC2\xF2\xAF\x2F\xCE\x45\x69\x43\x04\x48\x5C\x40\x79\x8E\x22\x63\x35\xC7\x57\x65\xE4\xC0\x6F\xFE\xF2\xF4\x8E\x45\x97\xA7\x09\x1B\xED\x97\xD3\x42\x52\x57\xB3\x65\x18\xC0\x22\x0A\xCD\x3C\x8F\x1C\xDC\x00\x7D\xA6\x08\x2E\x14\x4F\x6D\x4D\xDE\x51\x27\x3E\xF8\x1A\x23\xF8\x75\x31\xE5\x61\x24\x7B\x8D\x81\x8A\xB9\x0B\xDC\xF9\x51\x50\x6E\xB5\xF5\xAA\x64\x05\x80\x34\x50\x40\xDE\xC0\x68\x93\x2C\x11\xBC\x55\xC2\x98\xFB\xF7\x6A\x01\x17\x13\x60\x68\x95\xA0\xD8\x2A\xC1\x45\x05\xE8\x59\x25\x8C\x9F\x3C\x45\x9B\x49\x33\x25\x16\x6A\xD0\xC4\x8C\x06\x8D\x5A\x1B\x53\x8A\xA9\x31\xA5\x98\xF2\xA2\x78\x3F\xC5\x14\x69\xD0\x04\xED\x7D\xA7\x41\xE3\x67\xAF\x41\x13\x30\x26\x12\xC7\x15\x3F\x25\xA6\x44\x8F\x98\x01\x62\xE5\x1A\x01\x4C\x79\xDC\xEB\x55\x56\x6B\xE5\x2E\xBF\xCE\x8C\xA2\x40\x4A\x29\x03\x69\x28\x40\x23\x25\x69\x60\x05\x9C\x60\x33\x14\x22\x45\x02\x0C\x88\xF2\x37\x08\x92\x71\xB8\xE6\xFE\x49\xAD\x7C\x80\x79\x05\x79\x53\xE5\x38\x8C\xDC\x05\x87\x8E\x0E\x05\x6F\x93\x3D\xA8\xFB\x89\xB9\xAC\x78\x5C\xA4\x85\x3B\x47\xEC\xE6\xB7\x5C\x00\xA2\x8B\x63\x5F\x23\x7B\x4D\x7B\xFE\xF5\xC7\x9E\xD5\x59\x02\xBE\x23\x6E\x05\x46\xE4\xB4\x8F\xC8\x67\xF1\x9F\x1C\x42\x90\xCF\x96\xC9\xDD\x6A\x8D\x7F\xFD\xFF\xEC\xFD\x0B\xB0\x66\xD7\x55\x1F\x88\x9F\xFD\x38\xAF\xEF\x9C\xF3\x7D\xA7\x5B\x57\xD2\x95\xBE\xB6\xBD\xCE\x29\xFD\xAB\x6E\xA7\xDC\xE5\x4E\xFD\xC5\x6D\x95\xA3\x89\x7B\xF7\xA4\x1F\x57\x6D\xAB\x95\x4C\x31\x05\x55\x50\xE5\x9A\x78\x18\x7C\x6E\x23\xEB\xB6\xAE\x3B\x66\x68\xDF\x7B\x51\x37\xA2\x67\x12\x67\x04\xC8\xC4\x80\x8D\x05\x36\xD3\xB2\x71\x63\x63\x60\xCA\x21\x06\xDA\x06\x12\x01\x1E\x22\x6C\x19\x3C\x60\x40\x33\xC3\xC3\x06\x27\x38\xC1\x49\x4C\x00\xF7\xD4\x7A\xEC\xF3\xF8\xBE\xEF\xF6\xED\x46\x32\x83\x33\xB6\x4B\x7D\xBF\xF3\xDA\x8F\xB5\xD7\x5E\x7B\xED\xB5\xD7\xFA\xAD\xF2\x88\x39\x0C\x21\x18\xBA\x57\xF2\xAF\xF2\x88\x59\xC6\xA7\x48\xBA\x97\x6E\xB8\x6B\xC1\x39\x5C\x6E\x42\xD2\x30\xDA\x1B\x49\x87\xB9\xCD\x5B\xB9\xA0\x8A\x38\xD5\xA2\xCA\xCA\x8F\x10\x59\xA2\xDE\x0E\xB3\x5B\x4D\xBA\xC3\x4A\xC2\x4D\xA3\xC8\xBC\xFE\x57\x31\x39\x33\xC7\x6E\xC7\xB2\x6E\x1B\xBA\xFB\x9A\xF2\x03\xA2\xB3\x68\xF6\xF5\xC3\x15\x64\xBD\x36\xCE\x36\xAB\xFA\x3E\x97\x54\x1A\x12\x77\xED\x8B\x9F\xF8\x65\x3A\xF5\x34\xEE\xC3\xFE\x67\x88\x9B\x88\xA4\x4E\xDD\x64\xC3\x3D\x8B\xAD\xBE\x16\xF1\xF2\x96\xD7\xA3\x55\xFD\x0B\x72\x51\xD6\xD9\xAA\x7E\x46\x2E\x96\x6A\xBD\xAA\x3F\x26\x17\xCB\xAB\xFA\x59\xF9\x79\xA0\x4E\x56\xF5\x73\x11\x84\x9D\xE2\x6F\x21\x24\xC5\x1F\xB7\x2E\xA8\xF8\x13\xAC\x17\x0E\xC8\xAA\xFE\xB0\xED\xBD\xF8\xA4\xE6\x17\xFF\x89\xE6\x17\x2F\x6B\x7A\xF1\x29\xDC\x13\xBC\x1D\xEF\xE9\x55\x7D\x09\xDF\x11\x2F\xCC\x2F\x28\xF9\xB8\x3C\xA2\x3F\xA7\x20\x24\x3F\xCC\xCF\x28\x17\x40\xB6\x5E\x19\xFA\xF6\x12\x7E\xFB\x17\x8A\xB1\x0C\x5C\xD2\x9C\x2C\x02\xF7\xEF\x7E\x70\x00\x04\x6B\xB8\x83\xEE\xD9\x80\xBB\x8D\xC3\xFF\x5A\x08\xDB\xE3\xA9\x3A\x5F\xD5\xAF\x63\xD9\xD0\x8A\x25\x46\x1D\xCA\x5C\xFA\x48\x5D\xB8\x14\x8A\x29\xFE\xA1\x4C\x50\xE4\x19\xEE\xF1\x1F\xBF\x81\xF0\x1F\x19\x9B\x44\x0B\xF0\x88\x94\x43\x98\x91\x33\xCE\xE4\x0F\x0B\x4E\x64\x0A\xC5\xBA\x3F\xF1\x30\x67\x9B\xF9\x80\xB0\x45\x00\x35\xDB\x6E\x6B\x08\x50\x43\x9E\x38\x6F\x76\x17\x86\x00\x35\xB3\x9E\x38\xF4\xED\xB7\xB9\xFF\x71\x08\x50\x33\xEB\x89\x43\x6F\x7D\xAB\x7B\x13\x74\x85\xB8\x7F\xE4\xCE\x0F\x91\x69\x44\x82\xFB\xCE\xDF\x43\x94\xCA\x91\xF4\xBB\x78\x6B\xC5\x37\xEF\xAD\x95\x40\x8C\xFC\x57\xCC\xCA\x45\xBC\x5B\x15\x9D\x5C\x4C\x70\x1E\x3C\x13\x90\x2B\x1B\x98\xB3\x84\x45\x95\x38\x64\x7B\xFD\xC0\xB4\x4E\xA0\x60\x31\x19\x77\x62\x32\x81\x02\xC5\x64\x0C\x89\xC7\xFD\x23\x31\xC9\xD7\x5E\x4C\xC6\x0C\x58\xF2\x7C\xD0\x38\x0D\x86\x17\xF1\xBB\x21\x76\xDF\xBA\xAA\x2D\xEE\x08\x9D\xA2\x57\x50\x36\xDA\x29\x4E\x80\x1C\x62\xAE\xC4\x27\xA6\x19\xC2\xD9\x26\x3D\x38\xDB\x9B\x83\x1B\x32\x0B\xE1\x86\xCC\x4D\xC1\x0D\x99\x9B\x82\x1B\x32\x0B\xE1\x86\x78\xB7\xE7\xD2\xB3\x60\x3C\x9C\x6D\xE8\x76\xE2\x56\xD2\xB8\x4F\x45\xDD\xEF\x64\x55\xBF\x2D\xE6\xBF\x9F\xC7\x99\xF4\xE9\xC7\xDF\xF7\x53\x5B\xAB\xFA\x79\x94\x06\xEF\xFE\xB3\x4F\xFC\xF2\x47\xFE\xF8\xDA\x27\xBF\xE9\x88\xBE\x14\xFB\xEB\x27\x7F\xEE\xBD\x77\x1D\x31\x9F\x62\x17\xA6\x70\x45\xDF\x57\x1B\xB7\x85\x6C\xF2\x54\x07\x87\xC3\xC0\x3E\x21\xA7\xF9\xD6\x4D\x55\x60\xC1\x41\x53\x19\x17\x50\x5E\x76\x94\x6E\xB5\xA1\x00\x0C\xA7\xBF\x05\x67\x2B\xEA\x09\x94\x0F\xBC\x70\xF9\x03\xD5\x18\xF2\xAA\x80\x71\x11\x64\xE6\x28\x84\xEE\x69\xDB\xE0\xD8\x70\xCB\x41\x97\x8F\x5F\x64\x89\x49\x11\xAB\x6A\xD3\x85\x4D\x5D\xAC\xEA\x1D\x7C\xE1\x5A\x44\xEE\xA1\xD8\xB0\x67\xA2\x7A\x8C\x8C\x82\xBD\xBC\xAC\x1B\x7E\x08\x63\x28\xD6\x41\x97\xFF\x96\xD9\x21\x74\x01\xBE\xBA\x13\xAF\xAF\xB2\x08\xBB\xAC\x69\x71\xE3\x2F\x72\xA9\x9B\xA8\x07\xBA\xFC\x92\xD0\x8C\x4B\xE3\x7B\xBF\x23\xF7\xB8\x8D\xA1\xFB\x8C\x6A\x66\xDB\x89\xD2\xE9\x69\x6E\x3F\x97\xC6\xDF\xE3\x3B\xA1\x7B\x3A\x6E\xCA\x3F\xEB\x76\xAD\xA7\xF0\x9F\xA7\xE3\xBA\x38\xB7\xAA\x4F\xE9\xA3\x50\xB8\x8F\x80\x98\x2C\x68\x4A\x16\xBB\x4C\xC9\x22\xA3\x97\x17\x6C\xBE\xFA\x8F\x69\xD7\x15\x0F\x76\x5D\x79\x6B\xB0\x7A\x3F\x36\x91\x66\x4D\x4E\x16\xAB\xF7\xC7\xB3\x26\xAB\xF7\xC7\x64\xD6\x1C\x6C\xBC\xA4\x78\xBC\x4D\x53\x88\x37\x5E\x31\x14\xB3\x1B\xC9\x53\x6C\xAE\x38\x95\xB5\xAA\x98\xDB\x6E\xC4\x1D\x9D\x3C\x9B\xE3\x1E\xDA\x5D\xCC\xDB\x36\x48\x9D\x64\x23\xA3\x96\x59\xC8\x60\xD4\xAC\x8B\xDA\x11\x1E\x0A\x9E\xB6\x5E\xED\x08\x57\xF4\x87\xED\x9C\xDA\x41\xAF\x0C\x54\x4D\x1A\x82\x39\xC5\x83\x4B\xBB\xAC\x7B\xA5\xBD\x7D\x5E\x89\xA1\x57\x66\x4A\xBB\xBC\x48\x8D\x89\x69\x61\x6F\x99\x03\xA7\x15\x34\xDD\x5D\x62\x23\xA7\x19\xE1\x88\x6F\x5D\x62\x4E\x39\xF6\xF3\x5F\xFC\xC4\x2F\xFF\xDA\x7F\xA2\x19\x77\x3F\x93\x6D\xC5\x19\x54\x04\x64\x2D\xEC\xBE\xF0\xBC\x18\xAC\xEA\x6B\x96\x9E\x76\x55\xD0\x42\x3D\xC7\x55\xD7\x6C\x6D\xCE\xA1\x06\x70\x8A\x3A\x30\xE4\xAB\x5B\x16\xF5\x83\xC7\x0B\xF8\xAA\x73\x13\x7C\x4A\x7B\xBE\x62\x3F\xC1\xA7\xF4\x2C\x5F\x3D\xA5\x77\xE1\x2B\x3D\xCF\x57\x73\xFE\xFA\xC4\x57\xD4\x2D\xD1\x45\x65\x9B\x14\xB7\xB0\xD6\x56\x72\xFF\x31\xA8\x1D\x35\xEB\xDA\x3B\x2E\xED\x50\x7E\xCE\xD0\x3D\xFE\xCA\x55\x4D\x9C\x73\xE9\x87\x70\x21\xFF\x67\xEF\xBF\x84\x54\x2B\xBF\x9B\x04\x84\xFB\xDD\xC7\x2E\xED\x04\xEE\x7E\xA7\xDA\x37\xCA\xEF\xE1\x27\xA3\xEE\xCE\x93\x7C\x67\x47\x13\x60\x52\x07\xAB\xE6\x2E\xFF\x3E\x16\x16\x33\x02\x23\xAB\xD8\xDD\x88\xC9\x88\xF6\x46\xEC\x8B\xF3\x23\xA6\x45\x0E\xE8\x2F\xF7\x78\x25\x0B\xC6\x2B\xF9\xB2\x8D\x17\xC9\x01\x4F\x0A\xDC\xFB\x3E\x45\x94\xBA\x31\x2D\x12\x1E\x66\x48\x98\x16\x74\x00\xB3\x0B\x2D\xA2\x1B\xD3\x22\x9A\xA5\x45\xB4\x3B\x2D\x22\xDA\xEB\xDE\x98\x16\xD1\x90\x16\x52\x3C\xDE\xA6\x50\x1C\xA6\x45\x34\xA4\x85\x50\x82\x37\x02\xCC\x0B\xC2\x27\xDB\x27\x0A\x7D\x43\x3E\x59\x30\xB3\xFF\xBF\xC2\x27\x9F\xFF\xBD\x5B\xE6\x13\xF5\x5F\x08\x9F\xE0\xC6\x87\xC4\x9C\x17\x72\xC7\x0B\xC5\xB6\xFF\x08\x02\xF6\x82\x88\x3A\xDF\xF0\x50\x84\x61\xCF\x95\x7E\x3E\x21\xB7\x6F\x40\xEB\x48\xBF\x60\x61\x8C\x05\x04\x1C\xE2\x61\xE1\x71\xAF\xF0\x78\x51\xE1\xF1\xB0\xF0\x78\x7E\x9D\x14\x27\x7B\xEA\xE2\xCF\xC6\x1B\xB4\x8D\x8E\x32\x2F\xB9\x19\xBF\x51\xE3\x13\x02\x70\xE4\x97\xD9\x2B\x1C\x3B\xC7\x9E\xC4\x01\xEF\x0E\x5E\x78\x57\x21\xE6\x00\x20\xD5\x6D\x37\x5E\x58\xF7\xFE\xCA\x51\x4E\x83\xC7\x37\x3C\xFD\xBD\x66\x87\xC7\xBF\x38\x21\x86\x6C\x77\xCD\xEE\x72\xCC\xC8\x1E\x64\x3D\x46\x1B\x1C\x3A\x7A\xCB\xD8\x82\xB6\xAB\x1B\xB7\x5D\xCD\xB6\x7D\xC1\xA1\x85\x6F\x7B\xEF\xD4\x62\xD7\xB6\x2F\xB2\xF3\xC7\xF3\x76\xFE\x5E\x4F\x50\x60\x7C\x20\xF6\x96\xA3\xB7\xDC\xA6\xD2\xD6\x72\xB4\x30\x86\x74\xD6\x14\xAB\x25\xCA\x27\x66\xA3\x90\xDA\xA8\x38\x4E\xB4\x4E\xC9\x6E\x5F\x8F\x1E\xF0\xC1\x92\x79\x67\xB6\x5D\x1C\xA5\xA3\x25\xFE\x27\x81\xF8\xD1\xAA\x80\x11\xA4\x90\x4D\x7D\x68\x29\x7D\xB2\xE0\x74\xA1\x3D\xE1\x24\x74\x63\x48\xD9\x85\xB0\x70\x81\x0B\xE4\x9C\x81\x26\xBD\x96\x93\x06\x5B\xE7\x0D\x37\x8C\x6D\x47\xCD\x7A\xCD\x1E\x81\x5C\x03\x90\x90\xC2\x52\x73\x48\xDA\x40\xCD\xE3\x85\x91\xA2\xBD\x57\x44\x5B\x27\xC5\xAC\x0C\x6B\x73\x49\x45\xBE\x84\x34\x03\x7B\x25\xF3\x09\x44\xCE\x86\x71\x2C\xBE\x4E\x18\x94\x88\x22\x45\x73\x36\x64\x6B\x48\xBA\x93\x8C\xDC\x64\x7C\xD7\xB8\xA0\xD7\x1C\x3A\xF6\x48\xF8\xB4\x22\x87\x84\x1A\x91\x53\xB4\x2A\xDB\xE7\x47\xEB\xF5\xB8\x3D\xA1\x80\x1C\x02\xAA\x19\x12\x39\x90\x30\xFE\xC6\xE8\x54\x2F\xA6\x23\xD9\x65\xDE\xC9\x39\xDD\x9E\xD6\x8B\x9B\x0B\x78\x48\x70\x74\xB1\x8F\x63\x32\x56\x2C\x0A\x77\x48\x7C\xB8\x43\xD2\x9E\x50\x24\x7C\x88\x9F\xB4\x27\x14\x58\x0C\x75\xDB\xC8\xD9\x23\xF5\x93\x34\xC6\xA4\x1A\x41\x42\xE4\xE8\x9F\xBF\x2C\xF0\xFA\x54\x37\xEF\xF5\x39\x62\x9B\xF5\x78\xA1\xCD\x7A\xDC\xF5\x8E\x4A\x4B\x60\x04\x23\x48\x4E\x4D\xEB\x11\x8C\x17\x9C\xC0\x8C\x60\xCC\x27\x30\xA3\xB6\x7F\x23\x66\xAB\x51\xDB\xBF\x51\x06\xA3\xE3\x85\x91\xFE\x8D\xF8\xEC\x65\xC4\x2A\x31\x79\xFF\x8F\xB0\xC3\x38\xD6\x63\xBE\x39\xC2\xAB\x25\x0E\x6D\x94\x83\x98\xDA\x3C\x5C\x8F\x29\xA9\x3E\x6D\x52\x61\x0C\xE6\xEC\x3A\x9F\x4D\xCA\xEB\x78\x6B\x70\x68\x63\xF8\x5D\xD3\x1E\xF7\x8D\xE4\xF8\xA6\x03\x35\xCD\xDB\x13\x1C\xAC\x1E\x46\x7C\x82\xE3\xB6\x4F\x16\xCA\xED\xD8\x2A\xF3\x41\x5C\x79\x77\x20\xD3\x22\x23\xE9\xF6\x58\x46\xF3\x39\x45\xD2\x22\x9D\x77\x87\x14\xA3\x99\xD0\xC9\x5D\x0F\x29\x72\xBF\x33\xCE\xBB\x43\x8A\xD1\x8B\x1B\x3A\x69\x16\xB2\xB3\x99\x09\x9D\x1C\x51\xE8\xE4\x88\x42\x27\xCD\x82\xD0\x49\x73\x33\xA1\x93\x23\xEF\x9D\xBC\x38\x76\x71\x34\x63\x20\x1D\x7D\xE5\xC5\x2E\x7A\xEF\xEE\x51\x1B\xC2\x88\x7A\xE7\x20\xF9\x44\x21\xF9\xC6\x45\x70\x42\xE2\x14\xFE\x47\x50\x6A\xC9\xE0\xE8\x8A\xDD\xBF\x13\xE6\x8A\x04\xB9\x22\x61\xF7\xEF\xC0\xBB\x7F\x07\xC4\x15\x49\x8B\xA6\xA6\x76\xE1\x0A\x75\x2B\x5C\x41\x62\x20\x58\x28\x06\x82\x19\xAE\x30\xC4\x15\x86\xB8\x22\x98\x3B\xBA\xD2\x10\xF0\xD1\x95\x9E\x39\xBA\xD2\x83\xA3\x2B\x43\xAB\xB6\xF8\x80\xF7\x8F\xB0\xC4\xF9\x43\x4E\xE8\x21\xF3\x6A\xDE\x97\x07\x4D\xC0\xB4\xBE\x06\xEF\xD7\xFD\x28\xC6\xC8\xBD\x7F\x0E\x4D\xE0\xFD\x37\x42\x13\x18\xE8\xEA\x0B\x55\xA8\x5B\x1E\xA1\xC1\xE3\x05\x2A\x54\xD0\x6F\xBB\xA8\x50\xC1\x8D\xDA\xBE\x48\x85\xD2\x19\xEB\x4E\x94\xAF\xB9\xEB\x82\x0B\xAA\x34\x13\xCD\xC0\x1F\x27\xEB\xBD\x8E\x93\xFB\x5A\xF5\xA2\x83\xDC\xC8\x7D\xA0\x8D\xC4\xBD\x72\xBB\x89\xB7\xD5\x56\x9B\x2B\xEF\x09\xFD\x15\xE5\xFD\xD9\xF9\x7C\x32\x26\x95\xF7\x5E\x27\x68\x8D\x68\x17\x05\x30\x42\x12\x0C\x15\x3D\x54\xED\x34\x3B\x35\xA9\x36\x08\x33\xD8\xE5\x2C\x3A\x10\xDF\x96\x08\xB5\xC8\x68\x37\x27\x9B\xF6\x2D\xD4\x58\xA3\xCE\xC9\x66\x41\x2A\x48\xD2\xBF\xCB\x7E\x2A\xC8\xD0\x95\xB3\xDA\x77\xB9\x0B\xD7\x9B\x79\xFF\xC4\x68\xE8\xC2\xD5\x61\x3D\xDE\x72\xCC\xE9\xE0\xF1\x0D\x83\x8F\xCB\x61\xF0\xF1\x4D\x37\x3F\x5A\xD0\xFC\x5D\x1D\x5F\x77\x71\x1F\xED\xC6\x8C\xDA\xD9\x8F\x60\x58\xC0\x03\xAA\xCF\x03\x0B\x92\x97\xE0\x33\xE2\x05\x75\x13\x3C\xA0\x20\xFA\x8A\xE0\x01\xFD\x5F\x0C\x0F\x50\xB2\x53\xC5\xE9\x75\x38\x89\xCA\x03\x75\xC4\xC1\xD5\x75\x42\x51\xD7\xD1\xB4\x1E\x11\x22\x6B\x53\x47\xA4\xF4\x10\xE1\x5B\xD5\xB6\xE6\xB8\xE4\xAF\xD1\x01\x8C\xF8\xEF\x56\x9D\x1F\xA3\xAC\x17\x3B\x57\x6A\x4A\xC1\x49\x69\x30\x38\xFB\x45\x7E\xA5\x69\x85\x19\x52\xE7\xB4\x88\x1A\xE3\x34\x9B\x8F\xCB\x86\x8A\xA1\x32\x21\x29\xAF\x3F\xE6\x9D\x99\xD8\x22\x82\x2F\xB8\x80\x98\xAD\x7C\x8A\x97\x5D\x92\x9E\x6A\xF8\xB5\xA1\xEF\x51\xB4\x3D\xED\x0F\xEC\xE8\x61\x2A\x20\x2E\x4B\x90\x80\x29\xDF\xF3\x58\xDF\x59\x7A\x77\xB6\xBC\x39\x37\x99\x9B\xC3\x69\x11\x8B\x96\x01\x03\x11\xB9\xC9\x2C\xC2\x69\x31\x1E\xA7\xC5\xB4\x11\xE4\x86\x95\x50\xD3\x46\x90\xE3\x66\x83\xB6\x1C\xB4\xBF\xA0\x52\xC9\xFD\x22\x22\x4F\x2C\x67\x36\x38\x6E\x9C\xE8\xBC\xDD\x38\xB3\x46\x0B\xD8\x85\x8D\x6A\xE2\x02\x3A\xA9\x1C\x9B\xA3\x38\x32\x4D\x9D\x30\x6D\x4C\x53\xC7\x2B\x3A\xE0\xC3\x4A\xDB\x40\xEC\x6C\xD3\x5E\x27\x78\x9D\x74\xD7\x39\x5E\xE7\xFE\xDA\xB8\x12\x15\xFB\x09\x8C\x9D\x6D\xEA\xF1\x49\x32\x6D\x15\x28\x01\xC6\x4E\x6F\x56\x46\x6A\xF2\xF5\xB4\x5F\x59\xFC\xAA\x70\xDB\x4D\x5D\x14\x38\xA9\x3A\x1F\x93\xC4\xED\x70\x3A\x10\x67\xCB\x9F\x7D\x4C\x00\x76\xB4\x3E\xCA\x01\x88\x11\x8F\x77\xC6\x43\x89\xDC\x39\x33\xDE\x3A\x83\xAC\x37\xD8\x11\xBD\xC3\xE3\xED\x79\x6F\x27\xC1\x65\xBF\xE3\x35\x83\xDB\x55\x97\x37\xE5\xCE\xC5\x16\x55\x88\x02\xEE\x63\x7A\xA9\x92\xAF\x76\x76\x0E\x9F\xA6\xAA\xE3\xEE\x8B\x6F\xEF\xBE\x18\xDC\x7F\xEC\xA2\x87\x23\x2A\x38\x0A\xD0\xFA\x28\x40\x11\x20\x7C\x24\x48\xFA\x47\xB0\xC8\x08\xD7\x4A\x23\xC8\x56\x51\x0D\x62\xFF\x97\x68\x55\x97\x04\x68\xFB\xB2\x0D\x89\x68\x4B\x71\xBF\x30\x5A\xD1\xB6\xE9\x65\xA9\x78\x64\x83\xE4\x2E\x09\xE1\x47\x9D\xDA\x38\x57\x1B\x9C\xFE\x86\xF3\xC3\x57\x31\x8A\x51\xFC\xEF\xD4\x94\xEC\x13\x7A\x13\xB2\xDE\x06\x2E\x62\x55\x1D\x79\xA7\x8A\x78\x03\xA7\xFC\x06\x4E\x91\xAA\x1E\xD1\x89\x36\xED\xF3\x79\x2F\x74\xEB\x78\x47\x0B\x20\x18\xF6\xDC\xC2\x45\xBC\x67\x8F\x20\xA2\x3D\x7B\xB4\x60\x0B\x17\xF9\x2D\x5C\x34\x13\xA9\x19\x75\x91\x9A\x1E\x67\x7A\xC1\x8A\x16\xF0\x8A\x36\xDC\xC0\x45\xB3\x1B\xB8\xBF\xF1\x4E\x12\x94\xA6\xB3\x73\x92\x40\x09\xF8\x44\xAB\xA8\xBE\x5B\x6B\xB5\xA5\xB7\x51\xC3\x47\xC9\xAD\x48\x6C\xDB\x56\x6C\x87\x92\xE9\xDC\x8B\x6D\x7B\x85\xB3\x24\xB7\x6F\xDB\xAD\xDA\x5E\xA9\x23\x08\x9B\x3A\x84\x68\xED\xEA\x31\x78\x1C\xEC\x05\x32\x3A\x3E\x72\x11\xC2\xAB\x97\xDA\xA2\x38\x89\x68\x97\x6F\x19\xBF\x60\xF5\xDB\x38\xC5\x76\xFD\x10\x8B\xB5\xBE\x11\x57\x9A\xDA\x40\x38\x5B\xA4\xE9\x15\x19\xFA\xE4\xCC\x6D\x4A\x25\x6E\xAE\x2F\x05\x1B\x77\x0C\x2E\x83\xBD\x52\x2B\x88\x1A\xDC\x07\xAE\x5D\xBD\x80\x5B\xB5\xA6\x56\xA0\xD7\xAE\x5E\xC0\xB2\x55\xAF\x44\x49\xF7\xAC\x7D\x89\x8A\x53\xD5\xFD\xAA\x35\x7A\x5B\x6D\x29\xAF\xCE\x1F\xEE\x69\xF3\xA0\x9D\x39\x5D\x04\xEE\x13\xEF\x1A\xC4\xDD\x68\x30\x67\x8A\xC0\xBD\xF7\x2D\xFD\xBB\xB2\xC4\x6D\x3F\x0A\x9A\x5D\xB2\x74\x83\x8A\xC3\x09\x86\x43\x6A\x9C\x3A\xEF\x54\x73\xF5\x98\xD9\xAA\x63\x4A\xA3\xAB\xC1\x42\x7C\xA5\x4E\x68\x8E\xD5\x11\xB2\xFB\xE9\x29\x52\xD2\x6F\x10\xBC\x50\xB4\x9C\xD6\x58\xB0\xFB\x5C\x59\x3E\x71\xB1\x73\xDF\xA3\xAD\x11\x45\x80\x7E\xD7\xC5\x9E\xE0\xF3\xDF\x46\x9C\xFF\x68\x28\x72\xA4\x34\xFE\x1A\xA2\xEE\xEB\x45\x47\x0A\xA1\x3B\xDC\x26\xF4\x9D\xE0\xE5\xA5\xB7\xB6\xE7\xCE\x9F\xB0\x83\x73\xE7\xCB\xC3\x73\xE7\xCF\x7D\xFB\xA5\x9D\xC0\x1D\xBC\x99\x73\xE7\xF2\xAD\x8F\xF9\xDA\xA4\xF8\x52\x8A\x5F\x9A\x2B\x1E\x75\xA4\x17\xA3\x06\x80\xD0\x7D\xCE\xAE\xEA\x7B\xE6\x6B\x80\xB6\x86\x8F\xDE\x7A\x0D\x1F\x8C\x3B\x3F\xCF\x39\x84\x9F\x8E\x3F\xAC\x70\x08\x43\x08\x44\xCC\x67\xBF\xDD\xE3\x33\x88\xDC\xF5\xEB\xE6\xB4\xB3\x9B\x75\x8C\xEB\xF2\x39\xB2\xD4\x97\xE7\x69\x81\xB2\x6B\xD3\x3A\x76\x3B\x7A\x8D\x40\x72\x20\x75\xF6\x1C\xD0\x5B\xE7\x21\xC1\x27\x65\xFB\x40\xF3\x03\x2B\x0F\x2C\x3E\xA0\x3B\x5A\xEE\xA8\xD3\x35\x2E\xFC\x90\x9C\x6F\x70\x1D\x88\x85\xFB\x68\xE9\x74\x0A\x62\xAA\x1E\xA2\x93\x9B\x04\x23\xE7\x9F\xEE\xEC\x94\xBD\xE7\xD1\x49\x5C\x46\xAA\x38\xB7\x19\x3E\x5A\xA6\x85\xE7\xDB\x77\x76\x96\xD7\x0A\x43\xCB\x29\xE1\x8A\x24\xED\xDA\x6A\x73\x43\x01\x2B\xE7\xEB\xC4\x05\x90\x40\xDC\x60\x21\x6B\x53\x81\x59\x88\xD9\x74\x19\x54\x09\x99\x53\xBE\xAC\x2C\x47\xCB\x96\x9F\x2E\x23\x9A\x6A\x09\x27\x98\x4B\x5C\x50\xA5\xCE\x23\x5D\xC5\x7D\x25\x25\x18\xDC\xC0\x49\x96\xF1\xAB\x38\xAD\x46\xC0\xFE\x96\xB5\xE6\x8C\xBB\x55\x0E\x64\x65\x91\xAB\x14\xA2\xDE\x95\xEE\xF9\xE9\x92\xBE\x93\x43\xEC\x82\xF2\x83\xFE\x84\x23\x45\x32\x75\x97\x1A\x07\x8E\x2F\xF9\x24\x62\xD0\x2C\x4D\xD0\x8A\x8A\xE0\xEC\xEB\x80\x6B\xA8\xE9\x9C\xBA\xFC\x71\x7F\x1A\x83\xCF\xFD\xA3\x02\x82\x86\xB4\xC7\x18\xCC\x7A\x4D\xC9\x30\xC7\x8E\xFC\xDD\x5A\xC5\x1B\x52\x28\xA8\x09\x6D\x01\x45\x57\x40\xD5\x26\xD9\x9C\xFB\x8C\x00\x08\x9D\xE6\xCF\xAA\x98\xF7\x30\xC3\xC3\x47\x3E\x22\x81\x54\x6C\x8B\xE5\x4F\xD0\xC2\x9C\x56\x29\xE4\x55\x82\x7D\xCB\xE6\xA5\x51\x5A\xA5\x19\x8C\x16\x9D\x53\xA6\xC7\xD9\xF9\x7B\xCE\x72\x73\x7C\xF1\x51\x69\x17\x59\x08\x49\xF6\xD9\x17\x71\x0D\x78\x31\xD7\x81\xB9\x55\x40\xE7\x21\xAF\x02\x38\xCB\xBA\x85\x80\x58\xCF\x66\x03\xC1\x8F\xEA\xBF\x9E\x17\xFE\xF8\xDA\x57\x85\xF8\x4D\x48\x85\xFE\xEA\xF8\xDD\x17\x09\x19\x76\xF1\x52\xDA\x9D\xE4\x2F\x70\x0F\x20\x9F\x87\x6E\x09\xD5\xD9\x4B\x83\x5E\x10\x2B\x7B\xC0\xF5\x23\x5E\xAF\x69\x46\xAB\x20\xEB\x61\xD8\x9A\x0E\xC3\x21\xA6\x5F\xD8\xC3\xAD\x60\xC8\xBD\xDD\xE3\x49\xE7\x8D\x80\x93\x6C\xF0\xD1\x44\x8E\x62\xAD\x8F\x9C\x98\x7F\xB4\x30\x29\xA4\xF6\x49\x21\x39\x21\xE4\xDC\x19\xB0\x4F\x9F\xE1\x4F\x61\x39\x58\x05\xB4\x84\x9D\xAD\x83\x65\x17\xD6\x5A\xE3\xEC\x08\xF8\x3C\xDC\x42\xE0\xD4\x69\x69\x86\x71\xCB\xE4\x8D\x6E\xCB\xCF\x32\xA1\x5A\x0A\xD4\xCA\xED\x00\xDB\x45\x99\x2C\xCA\x5D\xA3\xEB\x2C\xFB\x8D\xA4\x8B\xCB\xFF\x80\xEE\xC5\x59\xF4\x4E\xCB\x43\x3E\x2D\x0F\xBB\xD3\xF2\x2A\x92\x73\x79\xF2\xDF\x97\xF3\x70\x01\xED\x8F\x76\x33\xE2\x2E\xB4\xD4\x7A\x15\x6C\x0F\x23\x6E\xFC\x02\x8C\xB8\x26\xEB\xF0\x30\xBE\x6C\xE7\x0B\x9D\x89\xAD\x3B\x5F\x60\x1B\xDB\xFB\xE7\x3C\x83\x6E\x78\xBE\x70\x03\xB4\xE2\x05\x31\xFA\x2D\x1E\x41\x1F\x74\x52\x67\xBD\x89\x49\xFC\x18\xEF\xC2\x8F\xB2\x0F\x5D\x14\xC1\xBA\xCB\x67\x8A\x3F\x63\x00\x04\xED\xBF\x56\x33\x5F\x7F\x25\x93\xB7\xEC\xA8\xFB\x95\xDC\x8D\xD6\xA0\xCD\xCE\x84\x9D\x14\xFD\xCB\xC7\x78\xA1\x0E\x7B\x47\x3D\xE1\xA2\xA3\x9E\x70\x78\xD4\x13\x2E\x3C\xEA\x99\xC1\x1F\xE8\xCE\x7D\x7E\x0C\x45\xF3\x9C\x62\xCF\xAE\x7D\x49\xDB\x1C\x72\x82\xEC\xE4\x3A\x5D\xB2\x0F\x75\xE7\x2A\x96\xD4\xE6\x1C\xAF\xEF\x3D\x1F\xE6\xBF\x9A\x7D\x72\xF0\xF8\x86\xE3\x33\x19\x0E\xCF\x4D\xC3\x6C\x2C\x30\x94\x9B\x9B\x82\x3A\xDD\xDF\x03\x6C\x60\xEF\x70\xBC\x15\xCC\xDF\xD2\x33\xB7\xBE\x5B\x75\x50\xB7\x87\xDB\x74\xC6\x60\x71\xB8\x96\x9B\x3A\x04\xED\x96\x9B\x83\x3A\x38\x62\xE4\x26\x34\xC8\x9C\xDE\xE4\x48\xB7\xCA\x86\x32\x35\xF5\xDF\x4A\xF0\x56\xE2\x6F\x85\xCE\xD0\xCB\xA0\xE9\x46\x22\x6D\xB5\xE4\xD9\x23\xCE\xEC\xB6\x5D\xB8\xFF\xBA\xDA\xA4\x6F\xA5\x4D\xFF\xD4\x68\xBB\xAD\x2E\x74\x86\x65\x39\xC5\xD9\x2B\x83\x7D\xEB\xCB\x51\x9B\xEE\x1C\x47\xE0\x7C\x39\x2D\x7B\x65\x8D\x04\x16\x07\x75\xC8\x26\xF9\xD1\x23\x85\x31\x04\x37\x46\x69\xD1\xDC\xE8\xE1\x1A\x6F\x1E\x2F\x50\xA3\xCC\xBC\x9D\xD7\xBA\x14\x1F\x6B\xB7\xDD\xD4\xDA\xA7\x3D\xCF\x38\xE1\xBC\x6D\x28\xA2\x38\x73\x7F\x3A\x44\x31\x21\x7C\xC3\xC0\x7D\xC7\xBB\x87\xD8\x26\xB3\x1D\xBA\xA5\xF4\x5D\x4E\xCD\x25\xBA\xC7\x79\xD3\xAB\x02\xCC\x87\x40\x7D\xA8\xFC\x9F\xF0\xE2\x5D\x75\xF4\xF7\x82\xE0\xDA\xF5\xEB\xD7\xFF\xC4\xBD\x4E\xA2\xEC\xF0\x0E\xFE\xEF\xF3\xEE\x1F\xB6\xBF\x83\xE0\x9B\xD8\xF6\xF8\xA3\x64\x40\xA1\xBE\xB9\x6D\xA2\x09\x98\x69\x76\x25\x56\xC9\xAD\x69\x1E\x3D\x5D\x03\xB4\xFB\xF6\x1D\xBB\x56\x04\xEE\xA7\xDF\x3B\x6F\x56\xEA\xA7\x31\xDB\xC5\xA2\x69\xFA\x67\x74\x06\x47\x37\x9E\x19\xDD\xC8\x8F\x6E\x42\xA3\x1B\x13\x82\x3B\x6F\x9B\x64\x74\x25\x80\x29\xC5\xD1\x4D\x7B\xA3\x1B\xE3\xE8\x52\x40\x16\x87\xF7\x6F\x37\xB8\xEB\xA0\xD1\x4D\xC8\x1F\xCA\x42\x22\xA3\x3B\x14\x08\x6E\xBB\x4A\xDB\xB0\x52\x1C\x09\xDE\x63\x6F\x13\x94\x89\x01\x5D\xFE\x13\xDA\xC2\x28\x86\x02\x18\x0A\x7A\x4B\xBA\x91\xD3\xE5\xCF\x90\x96\x18\xE3\xD4\x88\x05\xF1\x78\x8E\x20\x0B\x16\x76\x23\x0B\x7B\x3C\x87\x80\x41\x0D\x71\xC1\xDF\x51\x81\x53\x55\xDA\xC6\x3B\xB0\x0B\x62\xCC\x6A\x30\xAD\x31\x86\x3C\x86\xBD\xE3\x9F\x5F\x79\x0C\x25\xFB\xF6\xEA\x2B\x6D\xED\x32\x8A\x4D\x45\xED\x71\x45\x1F\x10\x0E\x52\xC7\xD9\xA7\x96\x78\xFE\x93\x43\xEE\x6E\x07\xFB\x7B\xDF\xDB\x67\x49\x77\xA1\xA9\x32\x09\x30\xA3\xB1\x20\xC4\x17\x50\x6B\xB3\x8B\x2C\x8E\x99\x1E\x2E\xB2\xB8\x75\x56\xB2\xCC\x52\x6C\x34\x36\x5E\x8B\x13\x59\xC4\x1F\x25\x4E\x6D\x36\x10\xFD\x1D\x45\x60\xD2\xF8\x1F\xE1\xE2\x33\x8D\x9A\x2A\xD1\x47\x99\x33\x18\x9F\x98\x5C\x3B\xBD\x88\xA8\x22\xC8\x19\x81\x3F\x26\x33\x11\x81\xF5\x57\x11\x24\xD8\xE6\x84\x81\x1D\x9D\xAA\x22\x02\x91\x44\xB2\xA7\x78\xB7\x08\x32\xC6\x79\xD0\x19\x8C\x5A\xC2\xCF\x9E\x80\x50\x72\xF5\x6C\x18\x8C\x8C\x3C\x92\xF9\x45\x3D\xDE\x6B\x51\xA7\x61\xBE\xC1\xA2\xDE\x5B\xC7\x21\xCD\x7E\x68\x9F\x51\xDB\xEA\x42\xD2\x5B\xCD\xE3\x6E\xBB\x5F\x04\xEE\x4F\x87\xA3\x65\xFD\x68\x7D\x6C\x26\x89\xFE\xA2\x57\x17\x08\x33\xBC\x2F\xA2\x46\x24\x49\x7D\xF4\x5D\xA0\x3F\xF4\xC3\x75\xF2\x7D\x72\xE3\x79\xF7\x0F\xC9\xCF\xF2\xBD\x34\x5F\x70\x7B\xBF\xF3\x26\x22\x00\x9F\x94\xE1\x5B\x9B\x0F\x5F\xFA\x5E\xF7\x4D\x92\x69\xA1\x1E\xB9\xEB\xD7\xAF\x5F\x8F\x4F\xF6\xBC\x45\x6F\x39\x85\xC7\xE0\xF1\x02\xE5\x81\x02\xEB\xE2\x01\xBC\x55\xEE\x2C\xC4\x33\xF0\x56\xF1\xEE\xF0\x56\xD9\xBC\xF2\x90\xF5\x94\x07\x0A\x38\x18\x6C\x9C\x5A\x0B\x8A\x76\x56\x26\x7E\xC1\xFA\x11\x6D\xF4\xB4\x33\x1B\xD5\x58\xC0\xA4\x75\x7B\xE6\xA9\x50\x3E\xE1\xC2\x76\x61\x83\x92\x76\x4D\xA0\xA8\x72\x73\x14\x72\x9C\xAE\xD9\x2B\x14\x4D\x8F\x1C\x32\x77\xA1\xF1\x17\xCE\x36\x90\xB9\xAD\xEE\x3A\xC7\x6B\xDD\x5D\x97\x0D\xCE\x0C\x97\x60\xC1\x25\x4C\x9C\x6D\xEA\x09\x1F\x7A\x8E\x19\x29\x74\x82\x93\xA9\xCA\xA0\x80\x09\xCD\x1E\xAA\xB0\xAD\x0D\xCB\xAA\x32\xAA\xA7\xCA\x61\x8C\x22\x73\x2C\x51\x82\x2F\x5E\xBE\x95\x7C\xE1\x41\x5A\xEE\x7A\xF9\x56\x20\x83\x11\x64\x90\x3E\x30\xAD\x33\xC8\x4F\x14\xA6\x3B\x44\x53\x90\x41\xFE\x6A\xBC\x03\x19\x44\x6D\xA6\x15\xFD\xD7\xC1\x50\x93\x21\x43\x4D\x66\x19\x6A\x37\x6D\xF4\x26\x19\xCA\x76\x2B\x08\x76\x6D\x98\x33\xA6\x0F\x59\xA3\x17\x08\xE7\x76\xBA\x3F\xDD\x17\xCE\xA4\x30\xE5\x50\xB0\x72\x93\x49\xA4\xBC\x3F\xAE\xC9\x50\xE7\xE0\xE3\x61\xED\xB6\x1A\x0F\xEB\x51\x34\xEE\x4D\x4D\x35\xC6\x47\x46\xFC\xE5\xC9\x46\x6B\xCB\x1F\x23\xDE\x2E\xBD\xEF\xB0\x86\x8C\xD3\x20\x65\x92\x1D\x06\xC7\xED\x54\xB7\x31\x78\x61\x08\xB9\xE2\x3A\x9B\x2E\x64\x97\xB4\xEF\x24\x09\x39\x25\xC9\xC9\x41\x13\x64\x86\x86\x74\x81\xEB\x02\xBF\x93\xB2\xFB\x82\x6E\x59\x47\x7B\x1F\x5A\x4F\x70\x5F\xDC\xF1\xC2\x0A\xFD\x34\x10\xE1\xA0\x64\xD7\x48\xB2\xA3\xE6\x9C\x62\xE7\xDD\x6C\x0B\xA2\xE2\x89\x8E\x39\xC1\x0E\x11\xC9\x5E\x3C\x4A\x94\x37\x49\x89\x32\x83\xAC\x1A\x41\x09\x38\xF9\xF3\x07\xA6\x75\xB9\x90\x12\xFC\x8E\x50\xA2\x6C\x29\x51\x32\x25\xCA\x96\x12\xBE\xB8\x96\x12\xB4\xB8\x36\xA0\xC5\x45\x74\xC2\x46\xF4\x0B\x4D\x35\x86\x09\xAE\xAF\x13\x66\x0B\x64\x09\xE4\x09\xD6\x31\xC8\x45\x20\xEF\x61\x32\xCF\xE6\x6F\xC9\x76\xD1\x91\xB3\x99\x83\xEE\x6C\xF6\xA0\x3B\x9F\x39\xE8\xCE\xE7\x0E\xBA\xF3\xD3\xD3\x2A\x9F\x3B\xE8\xE6\xBB\xC3\x83\xEE\x7C\xEE\xA0\x7B\xFE\x2D\xE0\xB7\x06\x07\xDD\xFE\xAD\xE1\x41\x37\xDD\xA5\xAE\xBB\xF4\x2C\xE4\x72\xD0\xED\x99\x21\xDF\x85\x19\x28\x5D\x4F\xBE\x17\x33\xE4\xCC\x0C\x73\x21\x04\xC1\x4C\x08\x01\x95\x96\xF1\xD4\xA5\xAC\x55\xE3\x39\x77\x84\x0C\xC6\xEC\x8E\xD0\x4A\xD2\xBB\x7D\xCA\xAA\xA8\xE7\x51\x1E\x42\xE4\x4D\x0B\xC5\x82\xF0\xAD\x6E\xF3\xFC\x4B\x89\x1A\x6D\xB3\x52\x2A\x0E\xEB\x74\x68\x88\xD2\xEA\xEA\xBB\xE7\xF6\x07\xD6\xA5\xB8\x1D\xFB\x47\x67\xC1\x36\xB5\x25\xC8\x02\x45\x3E\xF9\x64\xCA\xE2\x54\xD7\x60\x20\x6C\x68\xDC\xD9\x03\xC4\x1C\x95\x6D\xDA\xF1\x82\x0F\x60\xBC\x0B\x48\xEC\x73\x5F\x8B\x27\x48\x0C\xD1\x09\x54\xEE\xB6\x51\x63\x84\x68\x6D\x3A\xC9\x66\x72\x6C\x13\x36\x91\xAF\xA9\x4D\x0E\x23\x01\xC4\x29\xD8\x75\xDC\x4C\xA0\x40\xFD\xC1\x77\x0D\xF0\x15\x52\x67\x4F\x33\x9C\x4C\xBB\xEC\xA7\x6E\xBB\x19\xDB\x40\x69\x13\xE2\x56\x66\xF4\x70\x6D\xDD\xF9\xB3\x10\xA1\xC6\x69\xFD\x21\xC7\xEF\x05\x78\xFB\xED\xDB\xBD\xFB\xE4\x31\xF8\x99\x18\xEF\xBF\xED\x4D\xBD\xFB\xE4\x27\xF5\xC4\xAF\xD2\x07\x9F\xFF\xF1\xFE\x17\x3D\xB0\x9B\xD4\x7D\x5B\xBF\xD2\xA7\xDE\x12\xE1\xEB\xCF\xFF\xFB\x6F\x9D\xAD\xE0\xDA\xA5\xFF\x0A\x9F\x5C\xFB\xF3\x63\xB3\x55\x10\xDA\x7A\x6D\xDD\xCE\x27\xDE\xF1\x6D\x83\x4A\xF0\xD9\x47\x7F\xE3\x15\xF4\xEC\x67\x9F\x7E\xA8\x7B\x96\x71\xE0\x84\x1D\x3A\x75\x75\x44\xAD\x2D\xD1\x9D\xC8\x6E\x99\xEC\xBC\xB4\x58\x41\xFD\x72\x5B\x4D\xAD\x79\xE3\x14\xB3\x5D\xD9\x60\x81\x23\xB2\xB1\xA3\x42\x62\x7C\x69\x95\xC6\x2D\xDD\x5A\xC5\x14\xAF\x72\xEA\x70\x55\x18\xDE\xE1\x63\x13\x84\x10\xD9\xFC\x18\xE5\xA8\x70\x7A\xA0\x37\xA4\xD5\x38\x0C\x91\x52\x51\x06\x96\xD3\xF7\x9F\x47\xAE\xAB\xC6\x10\x55\x36\x27\x94\x6B\x1C\x9E\x88\x86\xA7\xBD\x6F\xF0\x3E\x0E\x4F\x44\xC3\xD3\xDE\xD7\x78\x9F\x86\x27\xE2\xE1\x69\x9F\x0C\x94\xC1\x62\x4C\x75\x5A\xAA\x94\x86\x27\xE2\xE1\x19\x56\x4C\xC3\x13\xF1\xF0\x0C\xAB\xE6\xE1\x89\x64\x78\x86\xD5\xF3\xF0\x44\x32\x3C\xBD\x06\xE0\xB3\x27\xFF\xF8\xD3\x86\x1E\x8A\x8F\x8C\x3C\xCD\x70\x04\x61\xCC\x28\xAA\x5A\x46\x0B\x62\x48\xCE\x36\x75\xDC\x8D\x5A\x4C\xA3\x26\xF3\x48\x62\x20\x54\x15\x77\xA9\x1E\x48\x26\x74\x03\xAE\x20\x39\x5B\x13\xDA\x12\x84\x14\x24\xC1\x47\x2B\x41\x2F\x59\x3D\x23\x00\x70\x46\x35\xCB\x49\x2A\x64\xBA\x6A\x8E\x36\x35\x2E\x78\x60\x92\x49\x03\xCC\xDA\x34\x7B\x5A\x77\xA6\x2B\x68\x73\xF0\x3B\x81\x13\xE2\x93\xE7\xD7\x90\xBA\xE5\x4F\x27\xEB\x00\x82\xA6\x36\x10\xAC\x15\x0A\x90\x8B\x94\x1C\x35\x06\x6E\xFB\xD1\xD3\xEC\xB4\xD8\x30\xD6\x26\xF8\x5D\x95\x21\xF6\x6C\xCF\x15\x35\x1D\xCB\xCD\x9D\x2B\xEA\x17\x70\x26\xD7\x15\x5F\x82\x5E\x7C\xAE\xA8\x5F\xD8\xB9\xE2\x2F\x59\x15\xB3\xD0\xB5\x10\xD2\x9E\x1B\xE5\xED\x87\x7F\x64\x46\x3B\xDC\x47\xE7\xBA\x94\x1D\x65\xE6\x49\x79\xBA\x45\xE2\xB7\x10\xB2\x2F\xA2\x47\xE9\x6F\xEA\xC8\xA9\xF3\x75\x4C\x25\x4B\x25\xF1\x7A\x6D\x21\x3C\x43\x02\x9B\x4C\x92\x9B\xCE\x9E\xAB\x13\x3E\x45\xAF\x53\xCE\xD4\x49\x8A\x42\x0A\x49\x03\x21\xD8\xD6\xCD\x87\xBE\xB0\x73\x5F\xF8\xF7\x35\x7F\x81\xAD\xA0\x2F\x38\x54\x16\x9F\x18\xBC\xD9\x94\xEF\x62\x1B\x89\xF8\xD9\xF6\x0B\xB1\xD2\x7E\xF2\x0A\x91\xCF\x51\x92\xE0\x95\x7C\x37\x21\x48\x45\x2A\xB0\x4E\xC4\x99\xC0\x00\x6E\xBC\xC9\xE7\x13\xF9\xB8\xA9\x47\xDC\xC1\xD4\x57\xC5\xE9\x25\xAC\x34\x4F\x3E\xAB\x32\x6A\x50\xDC\x10\x34\x11\xEF\x9C\xF0\x69\x53\xE7\x10\xE2\xE7\xA1\x7C\x1E\xE0\xBD\x04\x0C\xD0\x07\xCE\x6C\x62\xE7\x84\x1C\xD4\x8D\x54\x68\x91\xB5\xDD\x18\x41\x0E\x16\x32\x24\x5E\xD7\x8D\xC8\x6D\x6D\xD4\x86\x1A\x61\xF1\x41\xD3\xD6\x90\x10\xB5\xAE\xB0\xAD\xBE\x47\x97\x70\x86\x2E\x21\x21\xF9\xF4\xE8\x62\x7A\x05\x18\x2E\x20\xFB\x52\x38\xF0\x02\x8A\x7A\xF1\x21\x37\xE4\x2D\x76\x9A\x61\xF6\xFA\xCD\x1F\x5E\xB0\xD0\xEF\xEC\xEC\x2C\x9F\x96\x33\xE1\x6B\x8A\xF9\x0D\xD9\x04\x79\x83\xF8\xCD\xA9\x2A\xCE\x4D\xE6\x1B\x0B\x51\xEB\xBD\x5A\x27\x9C\x59\x20\xC1\x97\x91\xE0\x3C\x04\x77\xCF\xE3\x1B\x52\x19\xBA\x65\xDC\x94\x3C\x75\xFA\x8C\x9B\x78\xC6\xC5\xC5\x7B\xC0\xBC\x23\xA1\x16\x67\xF3\x4A\x38\xB5\x24\x8C\x98\x79\xF9\x1C\x5D\xE5\x36\x6B\xBF\xB3\x73\xDF\xF9\xAF\x34\x7F\x87\x83\x42\xDF\xB1\x23\xB4\x3C\xEC\x73\x71\x5B\x56\x2A\x23\x36\x9A\x19\xB1\x51\x43\x61\xAA\x52\x8A\xA6\x41\x4B\xBA\xCF\xB9\xB3\x5C\x8A\xD4\x5E\x8F\x5A\xAE\x1E\xF5\xB8\x7A\x84\x6C\x49\xBD\x2F\x5A\xAE\x4E\xFA\x5C\x3D\x1A\x70\x75\x32\xC3\xD5\xA3\xA6\x1E\x43\x88\x9F\x4F\x5A\xAE\x1E\xE1\x44\x99\xE5\x6A\x69\xA9\x6A\x7B\x56\x08\x95\xAC\xF4\x2C\xA4\xD8\x51\x1C\x46\x28\x60\xD2\xBE\x0F\x48\x83\xAD\x0D\x69\x4E\x88\xCF\x9A\xDA\xB3\xE7\x08\x42\xCF\x9E\x54\x6E\xD8\xA3\x98\x69\xCB\xA5\x68\x37\x30\x48\xB1\xA4\x5F\x2E\x0E\x5F\x57\x8E\x95\x72\x14\x39\x85\x39\x8A\x67\x8F\x3A\x8F\x9A\x38\xFB\x0F\x4A\x9B\x6D\xBD\xE5\x2E\xA3\x98\x65\x77\x28\xC3\x8B\x1C\xAE\x5F\x21\x04\x0F\x57\xDA\x3D\x49\x0F\x4D\x0F\xDB\x8E\x0D\xC1\xC8\xDC\xA0\xD7\x71\x33\xE8\xD4\x24\x03\x7B\xB5\x8A\x9C\xAA\x94\xF1\xA6\x21\xCA\x83\x67\xAE\x72\x62\x53\x73\xB5\x8E\x41\x5D\xDD\x82\xE8\xB1\x2B\x15\x43\xDE\xAD\x55\x04\xB7\xC4\xB7\x08\x25\xF2\xBC\x44\x1F\xE6\x3A\xCB\xB0\x28\x08\x71\x75\x0B\x6E\xB9\x68\x75\x7A\x71\xD1\xB4\xF4\xAA\xEC\xAD\x56\x47\xDB\x66\x0B\x17\xD5\xD7\x93\xE7\xAB\x7B\xEE\xCA\x8C\x25\x50\x39\xD5\x8C\x8D\x0A\x54\x90\xB9\xEF\x9E\xB5\x0A\xA3\x96\xEF\x3E\x33\xF0\x34\x72\xB8\xE1\x53\xE7\x6B\xE3\x6C\x79\xED\xB1\xD6\x15\x68\x92\xF5\x69\x1B\x12\x6D\xC9\x30\x1F\x23\x6D\x13\xA1\x2D\xDD\xAF\xC2\x1E\x6D\x21\xBA\x4A\x56\xE5\x98\x62\xDD\xC9\x6A\xA1\x90\xFB\x3B\x1A\x24\x4C\x83\xF0\x2A\x87\x40\x87\x57\xEB\x11\xD1\x20\x65\x1A\x24\x48\xDE\x18\x46\x30\xE2\x5B\x21\x7E\x70\xBE\x4A\x08\x7C\x02\xC9\x4B\x35\x10\x8D\x63\xA2\xF1\xAD\x14\xAD\x4E\x2F\x2E\x9A\xC8\x6B\x41\xAD\x6A\x14\xB2\x11\x6E\x7F\x2D\x6D\x9E\x91\xD2\x4E\x73\x2D\xB8\x0B\x51\xF8\xDF\xE9\x69\x1D\x12\x0A\xB8\x24\x46\x0B\x9C\x25\x2B\xE1\xD5\x6A\x84\x8B\x74\x53\x25\x10\xBA\x2F\x5D\xBF\x7E\x3D\xDE\xA8\x94\x53\x55\x88\xEF\x91\xC9\xFD\x42\x03\x21\x05\x3F\x40\x78\x95\x5B\x51\x67\x7F\x57\x82\x23\xAA\x04\x05\x12\xDF\xC3\xB6\xD1\xFA\xD7\xD4\xF1\xC9\x22\xC8\x20\xBB\x42\xE9\xF0\xC8\xE5\x83\x03\x27\x70\x06\x26\x6C\xD7\xCE\x9E\xD4\x37\x9C\x0D\xCE\xDC\x68\x36\x10\x8F\x0E\xA6\xC4\xEC\x88\xDD\xCC\x84\xB0\x0B\xB8\xD6\xFA\x11\xF3\xB3\x00\xA5\xF8\x70\xC4\x6E\x66\x42\x2C\x2C\x5A\x94\x5B\x25\x2E\x3D\x92\x61\x07\x3F\xA7\x57\xFF\xAE\x4E\xB2\xCF\x86\x36\xA5\x40\x4A\x9A\x30\x6A\xC1\x84\x71\xBF\x82\x53\xA1\xA5\x46\x89\x6D\x7B\xEB\x0F\x5F\xE2\xA8\x78\x5C\x2B\xCB\x35\x82\x77\xF6\xAF\xD3\x5E\x06\xAB\x03\xD5\x3A\xCF\x1A\x30\x2E\x39\x47\x79\xDA\xCA\xF3\xA0\xC4\x79\xD6\x90\xF3\x6C\x38\xAD\x23\x88\x9C\x3D\x07\xF4\xD6\x79\x08\xF1\x49\xD9\x3E\xD0\xFC\xC0\xCA\x03\x8B\x0F\xE8\x8E\x96\x3B\xEA\x74\x1D\xE2\x62\x14\x92\xF3\x2C\xD5\x85\x3A\xDE\x7A\x1D\x6E\xD6\x09\x7B\xC7\xA6\xE5\xDB\x2E\x4A\x92\xA2\xC4\x6D\x83\x2E\x9F\xA4\xB5\x78\x24\x4E\x2E\xB4\x3F\x93\x63\xD0\x53\x05\x2F\x6F\x95\x85\x80\x36\xE8\x06\x12\x18\x95\x3F\x29\x2B\x0D\x01\x0E\x66\x6B\xDE\x8A\x8C\x1A\x1E\x97\x5D\x04\xF3\x3E\x91\x33\x12\xA2\xF9\x1A\x8A\xCE\x57\xD4\x82\xEF\xA1\x98\x95\x02\x79\x8A\x9B\x45\x48\x95\xA7\xA7\x64\xA8\x9C\x98\xA3\x4E\x55\x16\x8A\x2A\x84\x09\xC3\x8E\xB1\x1F\xBF\xCC\xDC\x52\x66\x6E\x09\xF6\xEA\x16\xE4\x8F\x5D\xC1\xDD\x10\x52\xA2\x8A\xA0\x84\x92\x6F\x85\x44\x88\xCA\xF0\x09\x50\x00\x13\x88\x37\x65\x4F\x6A\xF1\x53\x7A\xAD\x2A\x8F\xA9\x6A\x9F\x0B\xAA\x90\xBA\xBA\xEF\xD8\xF5\xEB\xD7\xAF\x4F\x2E\xD6\xFB\xC0\x7C\x8D\x0E\xF0\x8D\xBF\x2B\x11\x44\x38\x18\x5F\xA3\x71\x26\xD4\xFB\xBB\xFB\x84\xB6\xB1\x4F\xCA\xDA\x07\x25\xEC\x97\xF6\xE0\xAE\x09\x17\x99\x14\x69\x45\x39\xA4\xEB\x09\x70\xC4\x12\xEE\x60\x29\x1C\x0B\x79\x65\x9E\x6A\x90\x0E\x87\x48\xE1\x34\xB3\xDD\xE8\xA4\x3E\xD4\x09\xA9\xEF\x47\x06\x27\x6D\x1D\xC8\xC8\x2C\xF0\x4E\x05\x9B\x7D\x21\x8C\x12\x62\x75\xB5\xA5\xB6\xF5\x8B\xC4\xEE\xFA\xE8\x4D\x33\xBA\x9D\xD6\x21\x84\x3D\x46\xB7\xC2\xE8\xF2\xA0\x63\x74\x2B\x8C\x6E\x5B\x46\xB7\xC2\xE8\x96\xB4\x2E\x61\x74\x64\xF3\x68\xB3\x8E\x9D\x6A\xC9\x95\xF0\xB2\x04\x06\xA2\xF5\x2A\x6C\x79\x3B\xED\xA8\x17\x0F\x79\x3B\x86\x64\xC8\xDB\x34\x5E\x99\x53\x10\x6E\x56\xA3\x59\xE6\xA5\xB4\x71\x2D\xF3\xE6\x9E\x79\x47\x4E\xE3\x7F\xA7\xA7\x55\xE1\x54\x35\x16\xE6\xCD\xAB\x10\xC6\x0B\x98\x77\x22\xCC\x3B\x21\xE6\xCD\x7A\xCC\x5B\xC2\x04\x26\x7C\xAB\x65\xDE\x92\x99\x77\x0C\x51\x8F\x79\xE5\xB5\x6A\x72\x63\xE6\xCD\x3A\xE6\x2D\xA1\xEC\x98\x37\x1B\x32\x2F\x97\xB5\x0F\x26\xB0\x5F\xDA\xE3\x99\x37\x26\xE6\x95\xEC\x9A\xC5\xC9\x76\xD6\x8F\x20\x6E\x67\xFD\xBC\xE2\xEE\x93\x92\x8D\x38\xA5\x68\x40\x20\x0D\xB3\x4C\x3E\xEA\x8F\x9A\x9C\xA8\xCB\x10\x8D\x86\x43\x34\x5A\x3C\x44\x0B\xCA\x44\xAD\x68\xC4\x0D\x73\xC1\x49\xF2\xCA\xC8\x7E\xC5\x1A\xB5\xAD\xB6\xFA\x8E\x33\xA6\x1F\x0E\xC1\x2B\xCC\x55\x5C\x50\x28\x98\xE7\x0A\x3B\xEA\xC9\xCC\xF8\xD2\xFF\x3A\x54\x8E\xE4\xF6\x2F\x5E\x99\xF1\xC3\xE6\x28\x55\xD0\xE5\xF7\xD1\xD6\xD8\xE2\x0E\x42\x21\x05\xD5\x5A\xA1\xC0\x5E\xF1\x8B\xE7\x49\xA2\x47\xC8\xFB\x73\x17\x94\xDF\xCF\x62\x99\x6E\xF0\xB7\xB4\x3D\x54\x9B\xEE\xC2\x46\xE3\x6F\x61\x71\x1A\x68\x40\xF4\x1A\x05\x27\x86\x33\x15\xF6\x3F\xEA\x6E\x51\x9E\x55\xAE\xA2\xB0\x58\x6D\x1E\xB6\x3E\x8C\x60\x3A\x3B\x89\x59\x6C\x27\x31\x2F\xC4\x4E\x62\x3A\x3B\x09\x17\x3F\x67\x27\x31\x3D\x3B\xC9\x0B\xAA\x01\xA4\x86\x39\xFF\x6B\xF3\xC2\xFC\xAF\x51\x29\xC3\x6A\x5A\xC7\xE9\x30\x7B\xCF\x58\xBD\xD4\x33\xD2\xCE\x4E\xD2\x1A\xB2\x40\x31\xAF\xD6\x96\x27\xAD\x75\x3B\xB8\x2D\xCE\xBC\x07\x06\xEE\x85\xAA\x90\x36\x57\x55\x04\xD6\xD9\x4D\x4A\x5A\x69\x37\xAB\x04\x6F\x6E\x54\x29\x58\xB7\xD5\x90\x6A\xB8\xDD\x54\x84\xE0\x7E\x16\xBF\xC9\x5D\x50\x15\xB4\x43\xBD\xB0\xB1\x5E\x8D\x49\x9C\xD5\x13\x67\x1B\xB0\x67\xAB\x12\x82\x6A\x9F\x39\x0A\x05\xA8\xB3\xD5\x7E\xD8\x57\xDD\x06\xFB\xAA\x25\x28\xAA\xDB\x69\x71\xB8\x1D\xF6\xD3\xDE\xAF\xBA\x03\xB7\xCB\x7C\x72\xBD\xB3\x63\x9B\xEA\x4E\xB8\x83\xD0\xA1\x33\x3E\xB8\x6E\xEF\x8E\xAB\x65\xB8\xAD\x32\x70\x47\x75\x97\x39\x0A\x77\x12\x50\x0F\x6F\x25\xEF\x86\xBB\x70\x17\x77\x27\xFE\x33\xC5\x7F\x0E\xC0\xA4\xA9\xEE\x42\xDD\x00\x45\xC8\xDD\x10\x36\x70\xA7\x7F\x1B\x2F\xA6\xFD\x8B\x03\xBD\x8B\xEA\x4E\x58\x46\x79\xB4\x8C\x82\xE5\x6E\x88\x9B\xEA\xCE\x2E\x72\xA2\x5A\x1E\xD6\xCB\xEF\x1B\xAC\xCB\xC0\x32\xCA\xFC\x65\x56\xC2\x8F\xD3\x34\x7A\x09\x39\x8A\xBC\x04\x17\x82\xA6\x7E\x09\x58\xB2\x61\x81\x3D\x0B\x2F\xC1\xCF\x5E\x82\xBF\x4C\x53\x2D\xB3\x55\x19\xCC\x3A\x4B\xE0\x97\x54\x77\xF1\x09\x29\xF6\x1A\xEE\x44\x0A\xD5\x77\xA1\xBE\x8B\x9D\x94\xBB\xCB\x74\x77\x2A\x8E\x73\x53\xB8\x9B\xA7\x89\xC6\x3E\x2F\x83\x6D\xAA\x65\xB8\xD3\x29\x6C\x3B\x8C\x28\x2E\x94\xBE\xA8\x96\xC1\xC2\x5D\xEB\xD5\x5D\x83\xC2\x49\x1F\x65\x0A\x7B\x63\xF7\xDD\x60\xE0\x4E\x58\x5E\x50\x83\x6D\xEA\xBB\xF1\x8D\x29\xDC\x8D\x04\x6D\xF0\xA5\x03\xF2\xD2\x01\x98\xE2\x9F\x3B\x21\x22\xAA\xB0\xF4\xBE\xCB\x6D\x35\xF5\x5D\x44\x95\x97\x80\x15\x09\xAD\x91\x27\x6E\x27\xB5\x9D\x10\x32\xA4\x7A\x41\x6E\x37\x48\xA7\x3B\x3C\x34\x3B\x3D\x5A\x86\x71\x75\x17\x8E\x3D\x09\xDC\xE5\xBA\x1D\x4B\xE4\x06\x98\xD0\xE0\x36\xF5\x72\x3B\x2E\x93\x06\x96\xF7\xBE\x53\xDD\xE1\x19\x04\x1F\x55\xCB\x70\x17\xF6\x0F\x1B\x0B\x77\xF2\xC8\xE3\xB8\x2B\x19\x77\xD3\xE3\x20\x1A\xF0\x3B\x3D\xBB\xF0\xB8\xF3\x26\x81\xD8\xB8\xBA\x93\xD8\x5B\x9D\x85\x82\x38\xBC\x7E\x69\xCB\xCF\xF5\xCB\xE8\x05\xEE\xD0\x52\x65\xE0\xA5\xD5\xDD\xE6\x28\x2C\xF7\x98\xF9\x4E\xB8\x1B\x59\x78\x79\xC8\xCC\x77\xFB\xB6\x32\xD9\x6F\x8E\x99\x17\x74\xE9\xFF\x1D\x66\x9E\xE3\xE3\x17\xCE\xC2\x1D\xF7\xFE\x35\x31\xEE\x50\x46\x51\x47\x5F\x86\x77\x5E\x3A\x60\x55\x16\x53\x4B\xC2\xAA\x77\xD6\x77\xF5\x58\xF5\xA5\xC8\x86\x77\x21\x1B\xF6\x59\x89\x99\xF7\xC6\x77\xAA\x97\xF6\x87\x7F\x20\xA4\xEE\x5A\x34\xAE\x7B\x31\x6B\x8F\x55\xBB\x8E\xF8\x39\xD7\x89\xDA\x05\x7D\xB8\xE3\xAF\xDE\x87\x3B\xF6\xEA\xC3\x2D\xF6\x60\x09\xF2\xA6\x5A\x82\xDB\xF0\xC1\x6D\x70\x3B\x2A\x5F\xB7\x93\x1F\x60\x86\x3B\x9E\xA6\xDA\x07\x05\xDE\x2B\xF8\x1E\x8F\x50\xE2\x0F\x8F\x9F\x89\x62\xDE\x69\xA8\xAD\x78\x5B\x6D\x65\x2D\x52\xD1\x8E\x6E\x5D\x97\x29\x16\xC7\xAB\x49\x62\x71\x43\xF5\xEA\x2D\xF3\xF6\x2A\x7D\xDE\xA9\xA6\x1F\xED\x48\xFA\x8C\x6A\x57\x58\xCD\x06\x18\x0D\x0A\xCC\xF4\x6A\x15\xF3\xB9\x72\x72\xB5\x4A\xC9\x1A\x50\x93\x4E\x49\xB6\x82\xCD\x2A\x07\xED\x15\x75\x7D\xD4\x7D\x80\x54\xFC\xB1\x7B\xD9\x06\xCD\x98\xC9\x8A\x0E\x20\x82\xB1\x33\xE7\xDD\x5F\x5C\xBF\x7E\x5D\x6D\x34\x75\xB9\xA2\x03\x9C\x6D\xE3\xAB\x10\x6F\xD5\xFB\xAE\x40\x02\xFB\x20\xDD\xB9\x72\x76\xBD\xDE\x0F\xFA\x2C\xAE\xF9\xB7\xE9\xA3\xB0\x9F\xCE\xE4\x61\x2C\xB7\x96\x5C\x50\xDD\x0E\xD6\x3D\xB1\xA3\xD8\x2D\xB9\x6C\xAA\x3B\x88\xF2\x28\xBC\x96\xD8\x2E\xBB\xB3\xA3\x60\x04\xB7\xAF\x3B\xBD\x09\xB7\xBB\x1D\xC0\x21\x1D\xAD\xD5\xCB\xD3\xFA\x2E\x6F\x19\xBF\xAD\xBA\x1B\xF6\x57\x21\xDC\x51\x4D\xCD\x51\x98\xD6\x77\xD0\xF7\x77\xB3\xE4\xBB\x1B\xE4\xBD\xEA\x00\xDC\x0D\xA6\x9E\x42\xFB\x5D\x04\x61\xBF\x13\x9C\x6E\x58\xBA\x1B\x4A\x77\x57\x74\x40\x41\x41\xE1\x7C\xCF\x42\xE9\xC6\xDD\x70\x80\x32\x8A\xD0\xDE\xB5\x57\x35\x1C\xE8\xAA\x2A\x01\x49\x04\x13\x5F\xDC\x9D\xD5\xED\x30\x45\x31\x27\x62\x3B\x80\x42\xD4\x53\xB2\x73\x19\xB7\x03\xD5\x5D\x9E\x34\x53\x26\xCD\x01\xC8\x48\x54\x1F\xC0\x6F\x21\x24\xCA\xA0\x28\x1B\xAD\xD5\x77\x4C\xEB\x03\xD2\xCB\x29\x45\x34\xDC\xD6\x88\x71\xF9\x40\xF9\xCE\xC5\x2D\xB8\x0B\x69\x59\xDD\x05\x77\xBB\x9D\x9D\x6D\x94\xEE\x54\x60\x43\xFB\x2A\x7C\x62\x60\x5A\x1D\x80\xDB\xAB\x29\xDC\xC1\x20\x22\xC8\xEB\x05\x8C\x71\xDF\x33\xA6\xB3\xC8\x2C\x9B\x0F\xA6\x63\x95\x10\xDB\xDD\x7A\x72\x86\x13\xBC\xD1\xEA\xD2\x76\xB1\x2E\x6D\x5F\x80\x1E\xFA\x8F\x27\x2A\xE2\x13\xC1\xC0\x7D\x82\x3C\x3B\x0F\x57\xCA\x5D\xFB\x7D\xDC\x93\x53\x52\x77\x54\xF3\xCF\xF3\x79\x52\x6D\x51\x37\x64\x5F\x0A\x08\x70\x07\x5D\xD1\x94\x70\xDB\x8F\xD2\xCE\x79\x5A\x2B\xC2\x44\xA4\x43\x5B\x9E\x3E\x81\xBB\x70\x86\x93\x57\xBB\x8F\xAB\x87\xEB\xD0\x7D\x5C\x9D\xAD\xA3\xF5\x8A\xAB\xD9\xF9\xC3\x17\x52\x0D\x7F\xBE\xEE\xF4\x1B\xAB\xC0\x90\x95\x74\x93\x6F\x51\x48\x5D\xD4\x54\xDA\x7D\x6A\xAB\x52\x8C\xE4\x08\xCA\xFD\xC5\x1F\xC8\xB3\xDA\x3C\x5C\x5B\x30\x6B\x85\xA5\xC3\x13\x73\x96\x42\x57\x41\x75\x5F\x2F\x7C\x43\xB9\xA4\xA1\xAA\xDD\xBB\x4C\xA5\xA4\xD0\x0F\x6C\x35\xBB\x16\xF7\xD3\x5B\x4D\x6D\x1F\xAE\x23\x30\x6E\xD4\xD4\xF2\x42\x04\xB6\x7D\xE1\xC3\xDD\x0B\x7A\xE1\x0B\xD7\xBA\x17\xEC\xC2\x17\x3E\xBA\x57\x09\xFF\x72\xAF\x12\x3E\xD6\xBD\x10\x2D\x7C\xE1\x57\xF7\xAA\xE2\xB9\xBD\x4A\xF8\xD4\x5E\x6D\xF8\xCD\xBD\xAA\xF8\x9D\xBD\x4A\xF8\xBD\xBD\xDA\xF0\x47\x7B\xBD\xF0\xB9\xBD\xDA\xF0\xEF\xF7\x2A\xE1\x3F\xEE\xD5\xC8\x2F\xEE\x55\xC5\x97\xF6\xAA\xE2\xB1\xED\x3D\xAA\xB8\xBC\xBD\x47\x09\x4F\x74\x2F\x24\x0B\x5F\x78\x72\xAF\x2A\xBE\x77\x7B\x8F\x5E\x7C\xFF\x5E\x25\xBC\x7D\xAF\x12\x9E\xDA\xAB\x84\xA7\xF7\xEA\xC5\x8F\xED\x45\x87\x0F\xEE\x55\xC5\x87\xF6\x2A\xE1\xA7\xF7\xEA\xC5\xCF\xEE\x55\xC5\xDF\xDB\xA3\x86\x13\x7B\x54\xF0\xEA\x3D\xBE\xFF\x07\x7B\x3C\xFF\x6F\xF7\x68\xDF\xD7\xED\x51\xFF\x37\xEC\xF1\xFD\x7F\xB7\x47\xFD\xAF\xDB\xA3\xFC\x66\x8F\xEF\x1F\xDE\xA3\xFE\x47\xF6\x28\x7F\x73\x8F\xEF\xCF\xDF\xF8\x7B\x92\xFA\x6E\xD4\xE0\x03\x0B\xEA\x2C\xD9\xB0\x3F\xAE\x9A\x8A\x42\xFD\x4E\x72\xF8\x10\x39\x1D\xD5\x1A\xB4\x3B\x7C\xA2\xD6\xD3\x8A\x70\x82\x1A\x5E\x08\xAB\x28\x0F\xB2\xCE\x68\xA6\xB3\x51\x00\x41\xF9\x7F\x90\x86\x4B\x3F\x7F\xB3\xFB\x39\x77\x77\x1C\xB8\xFF\xFB\xBB\x3A\x4D\xF6\x69\xA5\xCC\xB6\x47\x7B\xA5\xDD\x5B\x00\xEA\xA0\x0E\x8E\xE0\x1A\xEA\x92\x86\x56\x2F\xD9\x32\x51\x7C\xFC\x8A\x4E\xDA\x64\xA2\x6A\x45\xDB\x5A\xB9\x91\x38\xF7\x38\xB5\xC9\x48\xA9\xA8\x5A\x52\x38\xB7\x7B\xE4\x0C\x2D\x80\x65\xE3\x1E\xD9\xA8\x2D\xB7\x86\xDC\xEF\xF8\xF8\x0B\x2C\x27\xF2\x45\xFD\x89\x50\x9A\x0C\xAF\xE8\x8D\xD7\xE1\x82\x49\xDB\xCF\xCF\x19\x65\xB7\xBD\x5B\x16\x85\xC5\x33\x8A\xBF\xDA\xA0\x7D\x76\xE0\x3B\xA0\x8F\x82\xB9\xB9\x6E\x98\xDD\x3B\xE1\x35\xF8\x80\x22\xD8\x40\x49\x7B\xE8\xE0\xAE\xC1\xFE\x71\xB8\xBA\x3B\xDF\xB8\xFF\xFE\x4C\x81\x03\xB8\x84\xFF\x9D\x9E\x4A\x57\x35\x77\xB5\xFB\xB4\xDF\x69\x3D\xE8\x34\xBE\xD3\x54\x9C\xED\x93\xC8\x27\xCD\x23\x3A\x80\x99\xF2\xC9\x1D\x6A\xC9\x5B\x1B\xB8\x29\x21\x74\x64\xA7\xCE\x83\x99\xD6\xEC\x9E\x26\xE5\x4A\x97\xB8\xE1\x06\x59\xAB\x6D\x35\x56\xD1\x96\xEE\xDE\x50\x69\xF7\x06\x08\xD7\x41\x41\xB8\xEE\x54\xB3\x46\x69\x34\x42\x3E\xBE\xF8\x83\xEB\xD7\xAF\xC7\xA8\xCF\x91\xF9\x7F\xB3\x36\x40\x39\x2B\xE9\xC0\x84\xFA\x06\xC6\x65\x6B\x0C\x92\x8C\x35\x86\xA0\xFD\xB0\x2E\xAE\x3B\x64\xC6\xBB\x61\xF7\xA9\x6D\x59\x7F\xB4\x1B\xA5\xB7\xDB\x38\x45\x46\x8A\xC3\x2E\x53\x0A\xFE\x69\x6D\x41\x9D\xF1\xA1\xF0\xEB\xA0\xCB\x77\x5F\x6C\x41\x19\xC8\x2B\x98\x5C\xB5\xDB\x2E\x53\xC5\x93\x76\xEC\xDA\xBB\x3C\xD4\xD9\xAF\x68\xEF\xDC\xA6\x3C\x7E\x01\xC5\xE3\x19\xB7\xB5\xE1\xB6\x1B\x37\xE2\x06\x58\xAC\x9B\x8E\x3F\x39\x68\x13\xDB\x13\x4E\x6B\xBB\xCE\x51\x68\x16\x47\x31\x66\x9A\x44\xB9\x42\x2A\x1B\xF7\x06\x88\xD6\xE9\x49\x1D\x43\xB4\xBE\x56\x68\x9E\x1C\x09\xBF\x96\x52\xE8\x61\x8F\xE0\x11\x12\x3C\xA2\xB4\x30\xD1\xDA\xB4\x8E\x84\xE0\x11\x11\xDC\x64\x90\x42\x02\xE1\xB4\x4A\xC0\x08\xC9\x23\x6F\x44\x8D\x20\x01\xEB\x49\xDE\x1E\x5F\xCA\x94\x87\x00\x04\x87\xCC\x38\x75\xAE\x36\x74\xF6\x1E\x09\x6F\x3B\xB5\x31\xAD\xC9\x5F\x46\x83\x12\x94\x9A\xCE\x99\x5F\x09\xE4\x40\x2C\xD9\x04\x28\x3C\x6E\x93\x13\x31\x40\xBC\x60\xDC\xFE\xC0\x13\xB3\xA5\xE3\x90\x8A\x34\x82\x74\xB6\xC6\x28\x71\x48\x45\x3B\xAD\xCD\x3A\x83\xC2\x90\xCF\x35\x8D\x1E\xB5\xCE\xE2\x50\x9B\x06\x94\x87\x05\xBA\x9B\x60\x63\x08\x8D\x60\xD0\x3E\x2D\x8E\x0F\x0D\x81\x86\x33\x93\x6D\x72\x14\x36\xDD\x53\x99\x7B\x18\x39\x9E\x41\x67\x20\xEC\xC6\x42\x86\x2C\x71\x6F\xA0\xF1\x08\xFB\xE3\x11\xE2\x78\x84\x9C\xCB\x6B\x6D\x5A\x87\x32\x1E\x21\x8D\x47\x4A\xA9\x1F\xC0\x4E\xAB\x18\x52\x19\x8F\xD0\x33\x60\x08\x31\x98\x6E\x0A\x0C\xFB\xE0\x07\x2D\x1E\x8C\x0F\x41\xCE\xF1\x30\xA5\x32\x43\x42\xDA\x8D\x50\xB6\xDC\x59\x3A\x2F\x31\xF7\xAB\xF2\x87\xF0\xBA\xFC\x11\x72\x65\xFC\x21\x2D\x9B\xB2\xFE\xFF\x99\xEE\xDA\x93\x0A\xA5\x9D\xE9\x0D\x4C\xAD\x71\x59\x22\x1A\xD8\xDC\x66\x6E\x24\xAE\x0A\x75\xE8\x46\x27\x8B\x90\x89\x14\xB9\x11\x13\xF1\x0D\x38\xDF\xDD\x1B\x28\x5A\x8A\x38\x1D\xB4\x27\x98\xCA\x1C\x9B\x32\x6A\x92\x4F\x4D\x1D\x42\x08\x76\x6D\x5A\x5B\x21\x9B\x25\xB2\x85\x28\x11\x88\x5A\xE4\x19\xEC\xE1\x9E\x62\x72\xA9\x18\xA1\x5C\x8F\x3A\xB2\xC4\x4C\x8F\x90\xE9\x91\x31\x07\x37\xC8\x8B\x1D\xF0\x05\x83\x4A\x35\xEE\x4B\x94\xD9\x91\xDB\x6B\x21\xC4\xF9\x62\xE9\xFC\x19\x89\x47\xF1\xE2\x48\xC3\x1F\xD0\xB4\x8A\xCC\x73\x66\x2B\x59\x88\x0F\x3B\x16\x14\x01\xEC\xD1\x54\x40\xCB\xE6\x3E\x23\xD6\xA2\x81\x59\xC8\x8C\xC1\xA4\xCF\x86\x74\xFA\xE8\x1E\x26\x69\x05\x76\x7D\xAD\x08\x90\x94\xE4\x46\xD8\xE3\x37\x2B\xD4\x03\xC2\xBF\x9F\x27\x9C\x90\x5C\xE8\xD7\x71\x14\xB5\xD0\x48\xCE\x1C\x6E\x74\x47\x43\x2D\x2B\xED\x80\x94\x5D\xE3\xFB\x4C\x35\x16\x1F\xDD\xF2\x69\x62\xA7\x19\x1E\xE3\x9B\x8F\x8B\xC2\xD0\x49\xE6\x90\x24\x21\xD2\x2F\x9A\x8A\xD5\x0C\x2C\x21\xD7\x78\xAF\x4E\x50\x64\xB9\x41\x15\x66\x6D\x8A\x4A\x0D\xD8\xD3\x75\x3C\xE5\x08\xFA\x8E\xD4\xD1\x14\x68\xB5\x0D\xCB\x77\x90\x41\x2D\x40\x9E\xDA\x86\x18\xDF\x5F\x9B\x56\xC3\x09\xF0\x9E\x42\x8D\xB6\xCD\x51\x50\xEE\x4D\x84\xCA\xD9\x3A\x82\xF7\x38\x5F\x41\xB0\x5E\x87\x4E\xBF\xB1\x8E\xC6\x51\x14\x91\xA7\x7C\x46\xAF\xD6\x21\xB9\x85\xB1\xA0\x8F\x1F\xA4\xB3\xCE\x88\xA7\x5C\xBC\xAA\x69\x09\xC6\xB7\x62\x71\x12\xE3\x69\x23\xAF\x27\x14\x05\x0D\x11\x7F\x45\x60\x91\x31\x7F\x4C\x6F\x85\xFC\x56\xD4\x16\x2A\x13\x9B\xFD\x9D\xE9\xBB\x40\xEE\x40\x4C\x8E\x50\x13\x81\x11\xB5\xBE\x11\x14\x80\x1E\x42\xF2\x60\x61\xBB\x1A\x92\xAE\x5D\x55\x4A\x4A\x4D\x85\xA3\x6D\x9B\x6A\x24\x2B\x71\x5F\xEC\x5A\x8E\x84\x0C\xEA\x98\x5B\x9B\x9D\x2A\x14\x84\x0C\x1F\x16\x54\x39\x44\x95\x82\xB8\x4A\xFC\x13\x96\xF0\x1C\x47\x15\xC0\x68\xF0\x42\xEC\x3F\xD5\xB8\x6A\xF0\x0B\xDC\x7C\x2C\x68\x54\x29\xC8\xF8\xBD\x07\x29\x67\x39\x11\x10\x72\x7A\x83\x40\x64\xA5\x2B\x06\xA5\x28\x45\x63\x53\x57\x28\x1E\xB1\x0E\x61\x24\xC4\x32\xDD\x2A\x45\x1D\xEE\xC8\x68\x66\xC9\x48\x66\x21\xDB\xB0\x62\x17\xB8\xBC\x21\xA2\x80\x2E\xAF\x5E\x14\x50\x1C\x12\xE4\x7F\x4E\x59\x19\x88\x42\x09\xD3\x2B\x5B\x48\xA8\x68\x40\xA8\x88\x7B\x1B\x31\x2C\x8C\x1F\xAE\x2A\xA5\x4E\xC6\x3E\x16\x25\xEE\x8D\x56\xD2\x3D\xEC\xA8\x15\x31\x0A\x6F\x24\x3E\x73\x4C\xAD\xB4\xA5\x52\xC2\x54\x4A\x85\x4A\xD8\x89\x3A\x43\xB5\x56\xBB\xA4\x4A\xE9\xD4\x25\x23\x5E\xE3\xFA\x53\x6C\x33\x9D\x9F\x86\x4D\x1D\x63\xDF\x05\x06\x2D\xA4\xC6\x85\x1C\x81\x7F\xA1\xA9\x42\x2C\x59\x42\xD7\xA2\x53\x6C\x1B\x0D\x1B\x24\x01\xCA\x5D\x71\xBB\xCB\x70\xDF\x12\x62\x6D\x06\x52\x7C\x96\x52\x65\xD8\xFC\x2A\xCB\xD9\x3D\x2D\x72\xFA\x5B\xBC\x97\x0A\x51\xF3\x1D\x2F\x3B\xC5\x06\x39\x88\x9C\xFD\x16\x76\xAD\x84\x94\x1C\xC5\x51\x98\x30\xED\x39\x52\xD3\x76\xAE\x95\x69\x7F\x9A\x31\xC6\x42\xCE\x44\xD5\xD2\x79\xF6\x5A\x0C\x9C\xAA\x72\xCF\x2C\x7E\xE0\xB5\x9F\x26\x29\xBD\xC2\x96\xC1\xE1\x04\xB2\x44\x47\xFC\x98\xC7\x25\xE2\xD7\xFD\x4D\x86\xC4\x62\xA2\x77\xF5\xE1\xD3\xCC\x69\x8A\x5E\xEB\x03\x30\x08\xB7\xC6\x58\xBE\xC4\x28\x99\x2A\xA4\xC1\x08\x39\x0B\x51\x20\xD4\xE2\x72\xA3\x53\x84\x43\x68\x4F\x14\x11\x8F\x98\x74\x80\x46\x8C\x9B\x62\xA5\x03\xF4\x5D\x1E\x52\x9C\x51\x53\xD3\x37\x31\x51\x3A\x14\xC2\xA3\xB4\x41\x65\x3A\xC2\xB6\x06\x6D\x57\x12\xF6\xF1\xCA\x38\xF2\x5A\x35\x1C\x71\x87\xCD\xA8\x47\x10\x9D\x29\x02\x73\x14\x46\x3C\xF0\x29\x85\xA8\x51\x4E\x5D\x2B\xF2\x2D\x41\x6E\xA4\x20\x7F\x94\x12\x84\x6B\x7C\xA1\xA9\x23\xEE\x2B\xD5\xC3\x04\xC5\x5D\x44\x58\x8D\x38\x45\x07\xD5\x15\xC9\x1C\x45\xA2\x41\x7A\x62\x5A\xA5\x52\x7D\xCE\xDE\x36\x23\x0A\xFA\xF4\x03\xE6\xE7\x70\x2B\x18\xA4\xF7\x5D\x8B\x73\x1A\xB2\x11\xE8\xF2\xC7\x2E\xFA\xE5\x7E\x24\x6C\xE8\x6F\x92\xD3\x12\x4A\x9D\xE3\x45\x94\x87\x38\x40\x11\xC4\x85\x24\xAD\x09\x70\xB7\x32\x5A\x7F\xB0\x2D\xE7\x7D\xE4\x4F\x82\x65\x54\x01\x0A\x35\xFC\x45\x65\xC9\x83\x4A\xE5\x36\xE3\xD1\xD1\x38\x9A\xE6\x28\xA4\x4C\xA8\xC0\x13\x2A\xEE\x93\xEA\xC1\x96\x50\x29\x3D\xEA\x91\xEA\x54\x4B\xA8\x35\xCA\x38\x1A\xCD\x11\x8A\x87\xB2\xA2\xB9\x71\x82\x88\x49\x33\xE8\x04\xEE\xE5\xB3\xEC\x0F\x70\xE7\xCA\xE6\xED\xC8\x6F\x39\x83\x3A\xEE\x56\x13\x17\xF0\x5E\x80\x86\x8B\xE0\x42\x35\x65\x03\xA0\x78\x21\xCF\xCC\x29\x7F\xE6\x17\xA1\x07\x19\xE3\x99\x81\xEE\x68\xB0\xA8\x82\xD8\xE9\x16\x58\x44\x98\xC9\xBF\xA6\xE9\x35\x02\xF6\x50\x22\xC3\xFC\xB7\x54\xBC\xE6\x56\x51\xC9\x22\xC5\xE5\x13\x4D\xCA\x2E\x43\x88\x19\x6E\x04\x15\xAC\xB9\x60\x43\x2F\xFB\x78\xAF\x58\xDA\x79\x4A\x00\x19\xB8\xBA\xB6\x40\x11\x5C\x7D\x12\xC8\xAB\xBA\x99\x69\x19\x4F\x66\x67\x1A\x06\xBC\xB0\xFC\x89\x69\x99\x8D\xEB\x15\x01\xE2\x9B\x16\x73\x2B\xDA\xEA\x27\x99\x90\x10\x8C\x7F\xAF\xDF\x4C\xE3\xEB\x9E\xF4\xDA\x68\x7A\x6D\xD4\xBD\xD5\x01\xDB\x32\x11\x9D\x2A\xF0\xAF\xF3\xDC\xCE\x20\xCD\xFE\x24\x52\xC9\xB6\x53\x3E\x7C\xAC\xA7\x84\x38\xFD\xC6\x71\xC4\xE1\x81\x5E\x01\x91\xD8\xB8\x40\x24\x86\xA8\x68\x94\x68\x89\x06\x64\xD2\x53\x54\x78\xBF\x83\x6B\x95\xED\x51\x4C\x83\xE5\xEF\x34\xAF\x97\xB6\x55\x41\xA4\xCD\xB6\x2D\x56\xDE\xF2\xE8\x3F\xF4\x65\xE0\xFB\x11\x8A\x00\x9E\xF4\x36\x22\x54\x56\xC8\xFD\x97\x25\x58\x6A\x89\xFA\xAD\x23\xFC\xD2\x04\xF5\x6D\xBF\xCC\xF6\xCC\x43\xBE\xCD\x4A\x3A\xC9\x6D\xA6\xF8\x3C\xCD\x71\xA3\xB8\x22\xDA\x6A\x84\xCB\xA5\x7F\x22\xAB\x3F\xED\xD9\x79\x69\xED\x5E\x08\xFD\xA7\x5A\x5C\xDE\x71\x69\xD5\x7E\x69\x4D\xAA\x11\x28\x7E\x8F\x96\xD8\x56\x14\x51\x28\x39\xEB\x54\x9A\x57\x1E\xF6\x90\xF7\x5A\x9B\xE5\x27\x89\x90\x4C\x77\xCA\x32\x75\xB9\x23\xA6\x9E\x27\xE6\x1E\xAA\x08\xBF\x82\xA4\x89\x98\x50\xB2\xDB\xB4\x43\x0A\xE9\x1E\x85\x2C\x10\xF4\xBC\x95\x74\xC3\x78\xCD\x1A\x88\x28\x18\x7E\x83\x1B\x76\x0D\x81\xA8\x7B\x88\x2B\x8E\xFF\x5E\x67\xA8\x3B\xB3\x94\x62\xE6\xA8\x28\x94\x8B\xC8\x13\x79\x7F\x7E\xDB\x69\x20\x11\x19\xD6\x08\xE1\x37\x41\x11\x4D\xAB\x9A\xE8\x69\xD4\x8A\x58\xF0\xF1\xC9\xB1\x97\x8F\xDB\x64\xEA\x51\x13\x0D\x56\x29\x41\x9D\x21\x45\x5F\x12\x69\x45\x0F\x31\xA4\x87\x98\x0C\x35\x7C\x9F\x14\xBE\x4E\x5C\x72\xA2\x90\xDC\x3B\xE4\x47\xA9\x4E\x4A\x6C\x8B\x40\x2B\x45\x1C\xCA\x69\xD8\xE8\x49\x6F\x81\x3A\xC1\x6E\x94\xFF\x3F\xC1\xC0\x0A\xD8\x52\x27\xA0\x9F\x1D\x7A\xCE\x36\xE1\x5E\xEE\x68\x31\x1C\xF2\xCB\xF7\xE1\xF6\xD7\xDD\xCB\x51\x06\x44\x71\xDA\xE5\xBB\x5F\x7F\xEF\x27\xDF\xAB\xCE\xF0\x20\xBB\xE5\xB3\xE5\xDB\x2F\x4A\xC4\xFB\x61\xB2\xD7\xE6\x67\x79\xE3\xB7\xAA\x0F\xFB\xFD\xBB\x2D\xDF\xE1\xDF\xB9\x8F\x31\xE7\xC3\x55\x7D\x2F\x76\xEF\x41\xEF\x3A\x1A\x70\x9A\x7C\xF2\x20\xC5\xA2\x28\x4A\xD5\xE5\x67\x9B\x9A\x4E\x29\xC9\x61\xD2\xE2\x06\xC8\xD2\xD6\x81\x82\x21\x1B\x90\x6C\x20\x84\x63\x73\x82\x5A\x28\x1A\xDF\xA4\xDB\x02\xDF\xE7\xCB\xC3\xC2\x88\x0F\x92\x0E\xFF\x00\x49\x91\x75\x7B\xA5\x8B\x99\x09\xB7\xF5\x96\xD9\x26\xB7\x57\x7A\xEB\x3E\x0E\x09\x38\xA6\x98\xBF\x6D\x1D\xAE\x13\x87\x72\xB4\x8C\x6A\x88\xAC\x2B\xFA\x14\x59\xC9\x9A\x57\xA8\xE0\x2C\x2A\x9E\xAF\xDA\xB8\xFA\x38\x67\x4F\x04\xCA\x69\xF0\x2D\xCE\x6C\x36\x75\x7C\xD0\x04\x75\x7A\xF1\x58\xF0\x0F\x28\xC9\x6A\x02\xE9\xA5\x23\xC6\x6F\x25\x3F\xD0\x91\xC8\x4A\xDC\x9C\x28\xDA\xB4\x71\x71\x39\x03\x3B\xB1\xEB\x38\x45\x7D\xD8\xB6\x73\x96\x3A\xA7\xDB\x19\xC3\x92\x5B\x31\xB4\x41\x00\x6A\xDD\x0B\xC6\x60\x45\x3F\x84\x12\xDD\x99\xCD\x75\xDC\xD8\xC7\x1B\xCE\x20\x27\xFF\xE3\xF7\x0C\x71\x4A\x56\x28\x19\xB1\xD9\x6C\x56\x34\x47\x82\xD9\x56\xAF\xD4\xC8\x7E\x2C\x24\xC4\x27\x9B\xBC\x35\xD8\x50\x89\xCD\x17\xAA\x93\x41\x09\x7B\x62\x8E\xE2\x24\x9B\xB5\xB6\xF4\xDA\x13\x81\xAA\x43\x26\x10\x47\xB4\xC4\x1B\x6E\xBB\x19\xC7\xB8\x0A\xC4\x36\x1B\xF9\xD0\x45\xF2\xBD\x3C\x5F\x2B\x6E\x57\x50\xE7\xF8\xE2\x56\x33\x0E\x95\xD6\x3A\xA0\x4D\x57\xEE\xD4\x79\xF7\x45\x8A\xCB\x69\xA4\x7F\x5A\x00\x8E\xAC\x8B\x08\xCA\xE1\x99\x27\x7B\x5D\x8D\x57\xB4\xAD\x14\x8C\x2A\x9B\xA7\x19\xFE\xC1\x9D\xFA\xED\x45\x92\xC7\x99\xDB\x66\x7B\x06\x8C\xCE\xA0\x3C\x3E\x14\x10\x0E\x9D\x75\x40\x01\x76\xEE\xB7\x82\x53\x53\xFA\x71\x2D\x78\x00\x7F\xC4\x2E\xC2\xD1\xAF\x15\xD9\x09\x2D\xC4\x87\x02\x7B\xAA\x88\xE8\x49\xD8\x1C\x0A\x82\x07\x8A\x88\x7D\x61\xD6\x24\x9B\x81\x72\xEA\x7C\xE3\x07\x5E\x32\x1D\x7A\x59\xE6\x12\x3E\x61\x40\x46\x6F\x79\x23\xDE\x75\xC4\x62\x5A\xF2\x70\x12\x69\x16\x07\x4A\xD0\xD9\x16\xBD\x4E\xCB\x35\xF5\x05\x47\xE0\xE5\x0F\x16\xDA\x05\x10\xAE\xD3\xF6\x7D\xCF\x6F\x02\xDE\x6F\xAE\xB4\x20\x03\xDC\xB6\x1B\x7C\x2A\xFC\x84\x05\xB4\x80\x25\x24\x71\x57\x28\xE7\x8F\x2E\x7F\xE3\x12\x61\xA0\x3D\xBA\x51\x58\x10\x06\x48\x36\x78\x4F\xD3\xEB\x3D\x47\xD3\x27\x1B\xBC\xF4\xF5\x1E\xF0\x8A\xC5\x63\x63\xCF\x53\xCF\x2D\x05\x3E\x10\x97\x7B\x26\x6B\xDC\x9B\x70\x10\x88\xAE\xEE\x3F\x5E\xED\x5A\xD7\x4E\xEC\x75\x5A\x71\x76\x99\xD1\x76\x30\xA3\x6D\x3B\xA3\x6D\x37\xA3\x2D\x85\x84\xF2\x8C\xD6\x84\x0C\xC0\xE6\xAE\x3B\x18\x7D\x63\xE4\x71\xB6\x56\xF4\x01\x8A\x8F\x32\x79\x24\xA4\xD1\x94\xE8\xC2\x4F\x89\xE5\x9A\xE1\x89\xC9\xE4\x73\x28\xB8\xFD\x78\xA1\x29\x10\x16\x05\x5F\xF3\x1A\xB6\xFA\x1E\x20\x4C\xB8\x9C\x8F\x16\x0E\x50\x28\x3F\x97\x55\x2B\x64\x68\xE4\x3A\xD2\xC9\x94\xA8\x7D\xA0\xD6\xE5\xE4\xE3\x80\xC3\x55\xC6\xA9\x4A\xB9\xBC\xB2\xE6\xA8\x0C\x26\x6B\xF0\x15\xC1\x99\x2C\x93\x9F\x3E\xAE\xCE\x91\xDF\x20\xB3\x56\x98\x43\xB4\xEE\xA1\x28\x2B\x4B\xC7\x2E\xAC\xB3\x1C\x38\xC5\x51\x17\x95\x91\x16\x87\xD8\xE1\x13\x45\x28\x69\xAC\x69\x38\x09\x8E\x40\x11\x69\x18\xF7\xCF\x48\xFB\xEF\x43\xCE\x7D\x40\x90\x07\xD4\x46\xF6\xB9\x50\xAB\x2D\xBB\xDD\x1E\x4C\xD1\xEA\x7D\x10\x55\xCB\x23\x9C\x40\xE6\x0A\x61\xFC\xF0\x63\xBA\x09\xF1\x15\x02\x98\xED\x76\xA8\xC9\xA1\x80\x91\x09\x5D\x50\x51\x9A\xD7\x00\xBC\x1C\x48\x0E\x05\x8A\x10\xF8\xE9\x91\x06\xC2\x50\x6F\xF8\x1D\xA4\x5E\x73\x92\x81\x72\xA2\x55\x7D\x80\x57\xAA\x65\x08\x20\x7B\x65\x70\x07\x45\xFD\x66\x2E\x38\x79\xEE\x95\xC1\xED\xE4\xC9\x94\xF2\xC5\x52\x0B\x8D\xC8\xE8\x8C\xB2\xDA\x0A\xAD\xF3\x4A\x3B\x45\x80\x7C\x9C\x86\xB3\x71\x17\x1A\x7A\x8B\x80\x3B\x2B\x2D\xEB\x7B\x47\x48\xDE\x78\x25\x74\xBA\x45\xC8\x90\xC9\x8A\xBE\xE7\xAC\x7B\x55\x83\x6C\x57\x27\x04\xED\xA6\xCF\x50\x86\x4C\xB3\x59\x87\xBC\x00\x57\xC4\x18\x50\x8D\x58\xD7\x01\xDC\x0C\xD2\x46\xB1\x5B\x7B\x81\x61\x72\xEF\x69\x64\x1B\x84\xD3\x5E\x0E\x32\xF8\x44\x45\x3B\x0D\xDA\xE9\x07\xA6\x52\x89\xE5\x4A\xF4\x66\x1D\xF5\x2B\x59\xF1\x95\xAC\x2C\xA8\x64\x85\x2B\x79\x79\xAF\x92\x48\x2A\x71\x3B\xBE\x18\xEA\xEC\xE1\x4A\xB3\x39\xEF\x70\x8B\xCD\xC6\xC5\x64\xA4\x76\xB8\x6B\x81\xA4\xB9\x0D\xFC\x31\xC6\x92\x98\x41\x57\x44\x99\x95\x99\x9A\xD0\x4B\xE5\x8F\x5F\x24\x5C\x01\x3F\xE2\xB6\x4E\x04\xA0\x0B\x92\x46\x00\x2B\x35\x24\x67\x04\x43\x10\x67\xCA\x32\xC7\xD0\x25\xA0\xD7\x65\xC3\x29\xC9\x78\x29\x25\xE8\x01\x88\xCA\xB7\x3F\xD6\xC2\x52\x51\xDE\xC4\xF2\xF7\x2F\x09\x2C\x95\xC6\xCF\xFC\xE9\xDE\x5C\x83\x34\xB7\xA6\x1E\xA1\x22\x7E\x5C\x80\xFB\x51\x77\x66\xD7\xB4\x51\xA7\x72\x7C\xB2\x87\xAD\xF1\x6C\xD0\xC2\x08\x42\x7C\x2C\x38\x62\x4E\x41\xEC\x96\x84\x59\x62\x37\x6D\x5C\xB0\x1A\xE0\xAF\x15\xBC\x67\xF0\xD7\xE1\xF6\xD7\x7D\xED\xAF\x6B\x81\x44\x8C\xC7\x0C\x15\x1A\x03\xE1\x0B\x53\x89\xE0\x38\xE7\x42\x4C\x67\xBD\xB8\xC1\x71\xFA\x44\x4D\x71\xAF\x27\xCE\x81\x75\xC1\x49\x88\xDD\x33\x41\x03\x21\x50\x3E\x81\x53\x1B\xB5\x99\x3A\x05\x11\x98\x69\xF9\x41\xB1\x3E\x90\x88\x95\x68\x98\xA0\x7D\x9F\x92\x44\xB1\xBA\x46\xB2\x6A\x66\x4A\x39\x55\xB1\xAC\xBE\xAF\xC2\x35\x11\x55\xAB\x19\xFD\x90\x9E\x1E\xF6\x4F\x67\xB5\x47\x7A\xBA\xE2\x9F\xAE\xCC\x3F\x75\xCF\x06\xDE\xAD\x2F\xC8\x2E\x6A\x15\x6E\xFB\x53\x9D\x15\x1E\xF3\xFB\x6B\x0D\xAA\xE1\x43\x9D\xA3\xAF\xE6\x64\x5B\xEA\x95\xC1\x8A\xDB\x96\x90\x27\x54\xF9\xEE\x63\x7D\xF7\xD5\xA4\x4D\x26\x60\xA7\x14\x9E\x15\xD6\xA8\xFC\x56\x21\x18\xB0\x0F\x90\xB7\x73\x37\x0B\xAD\x33\x9B\xC2\xD7\xA1\x97\x40\xF7\x52\x4A\x8F\x90\x75\x28\x47\x9B\x95\x75\x7C\x4D\x66\x1B\x2B\xC0\x9C\x55\x2F\xA2\x97\x6A\xC3\x5A\x73\x1D\xA2\xFA\xBF\xCB\x07\x10\x75\xF3\x8C\x3B\xC3\xDA\x34\x2A\x82\x19\x1F\x13\xDF\x0F\xBA\xE3\xAD\xB7\x2B\xA5\xB7\x21\x70\xCF\x7C\x86\xFD\x1B\xB9\x6D\xF7\xF0\xDA\xA2\xCA\x4F\x22\x2F\x0B\xF9\xC8\x58\xF6\x29\xF6\xCD\xC3\xC7\xE6\xA8\xCC\x3C\x50\x52\xA7\x46\xE1\xED\xC5\xD3\x43\x95\x62\xF5\xF8\x21\x7F\x96\xAD\xDA\xB9\xBB\xA2\xEF\xF5\x4F\xEF\x9D\x7F\x5A\xFE\x6F\x17\xB9\x47\xD9\xBF\x51\x2A\xC2\xD6\x7D\xA6\xD7\x3A\x86\xEB\x97\x16\x70\x73\x13\xEC\x28\x87\xC6\x12\x00\x9C\x6D\xEA\xD0\x7D\x92\xF0\x7E\x58\x03\x21\xD6\xFB\x64\x70\x76\xDD\x1D\x3E\xDB\xAC\xE8\x97\x0B\xDE\x70\xF9\x4E\x4F\xAA\x65\xD4\x84\x09\xCC\x88\x36\xDD\x8C\x8B\x66\x85\x60\xCB\x1E\x1D\x98\x83\xA3\x9D\x7E\x23\x96\x85\x9B\x8F\x00\xF4\x94\xCE\xA4\xD7\x9D\x6A\xDC\x7B\x05\x42\x95\x56\x44\x3F\x0C\xF4\xA5\xBB\xD0\x10\xA6\xDA\x8A\x5E\xA6\xDD\x56\xA5\x70\x97\x97\xF1\x7E\xB6\xFC\x94\xEF\x6E\x19\x40\x50\xFE\xD4\xC5\x8E\xE4\xDF\x6F\x3D\x8B\xB2\x89\x83\x3C\x02\x68\x0F\x49\xAC\x29\xE0\x84\x34\x93\x39\xF1\x01\xB1\x15\x63\x3C\x90\xA5\x05\xB5\xAD\x88\xC4\x95\x9C\xB3\x6F\x12\x92\x93\x9B\x9C\xF6\xE9\x58\x2D\x0F\x75\xE8\xEC\x26\x2D\xF6\x11\x29\x9D\xA4\xB9\x10\xF8\x6A\xEC\xF1\x04\x36\x7C\x46\x9A\x05\x3A\x72\xC2\x3A\x32\x19\xE7\xE9\x6C\x3B\xA1\xB4\x05\x60\xCF\x01\x36\x81\x4E\xC9\x62\x52\xBD\x32\x50\x12\xB4\x6D\xFA\x2E\x25\xA6\x75\x29\xE1\x7E\x1C\x0A\x0E\x1F\xF7\xB8\xBB\xD2\x42\x0A\xAA\x32\x2F\xB8\x85\x76\xD8\x42\x73\x83\x16\x86\x92\xA9\xEC\xA0\xB6\x47\xB4\x6D\x33\x58\xF4\x08\xA6\x5F\x6C\x82\xE9\x1B\x34\xC7\x8A\xA3\x07\x37\x27\xFB\xE7\x7C\x40\x4D\x21\x80\x3F\x49\x2B\x89\xA5\xCD\xDC\x2C\x5B\x10\x53\x30\x4B\x50\x9A\x8C\xFF\xFD\x92\x40\xD0\xD2\x4E\xB5\xA5\x6C\x44\xEF\xB5\xB6\xDC\x7E\x8F\xFC\xB1\x85\xEF\x51\xBA\x6B\x8F\xC2\x01\x81\x53\xEC\x51\x74\x0E\x52\xDF\xA3\x14\x7B\x14\x66\x10\x4A\x9C\x29\xFB\x97\x70\xC7\xBA\x56\x12\xAF\xB7\x2E\x4D\x9B\x57\x8F\x7D\xFB\xCE\xCE\x4E\xB9\x75\x44\xFC\x4E\x84\x0C\xF8\xC8\xA9\x73\xBD\xA7\xBD\x22\x6A\xCE\x8E\x5D\x87\x44\x53\xDC\x56\xD8\xCD\x73\x10\x4A\x3B\xB2\xCF\x32\x52\x42\x3B\xAD\x38\x22\xB4\xA5\xA3\xBE\x01\x1D\x45\x5A\x97\x1F\xBF\x24\x21\x98\xC3\x16\x8A\x15\x69\x92\x0D\x0A\x0C\x6F\x50\x20\xED\x6A\x45\xB5\x30\x92\xE0\x05\x2C\x8F\x12\x39\x4D\xB0\xFB\x17\x4E\xEA\xA0\x77\xD7\xA9\x73\x84\x8D\x75\x50\x5B\xE1\x38\xE3\x9B\x51\x27\x57\xC8\xBC\x94\x48\xA6\x67\xD6\x85\x13\xC6\x56\x50\x2C\x7A\x93\x63\xD5\xE5\x2B\xE4\x65\x94\x10\x6E\x78\x79\xF1\x21\x82\x9B\x35\x3C\x00\x32\x84\x72\x5D\x2B\xD9\x2A\x10\x35\x89\x94\xCA\x93\xB2\x9B\x26\xB1\x4F\xB0\x97\x48\x94\xF2\x25\xE4\xD2\xFF\x86\x33\xCD\xBC\x98\xC4\xCD\xBE\x7E\xB6\xCC\x3D\xE9\xCB\x39\x1C\x5C\x50\xFE\xAA\x94\x1C\xCE\x95\x4C\x87\x87\x4E\xE9\x6D\x27\xDE\x57\xBD\x69\xC5\x29\x6B\x7A\x0C\x5A\xFE\x1A\x95\x63\x69\x6D\x6D\x13\x16\x60\x09\x5F\x3B\xDB\x36\x73\x83\xB6\x09\xB5\x5D\x50\x3E\x2B\xED\x32\x73\xED\xA2\x52\xDF\x2A\x9E\x16\x9E\x5D\x17\x89\x7E\xA7\x6F\x9A\x1A\xE2\x8C\x66\xCB\x7F\x7D\x49\x36\xEE\xF7\x02\xDB\xCB\xFC\xE3\x73\x2E\xE0\x87\x5E\x50\x68\x66\x4D\x8E\x6B\x45\x1E\xC2\xB5\x82\x9D\xE7\x88\x45\x38\xAF\x67\x8B\xE6\x71\x5C\xA6\xD4\x02\x41\x81\x12\xBD\xD2\x7C\x82\x40\x5C\x66\x80\x18\x2B\xEE\xB8\x0A\x14\x0A\x0A\xD4\xA0\x87\x92\x38\x1A\x52\xA7\x25\x4B\xE0\x3E\x1D\x30\x06\x7D\x70\x0B\x4A\x4A\x67\xC0\xFB\x3A\xFA\xEA\xB9\xEE\x2B\x50\xCE\x6C\xD0\x96\x42\xF5\xE0\x9F\x2F\x6C\x54\xD6\x25\x95\xF2\x26\xB6\x15\xFD\x10\xA8\xC6\xBD\x49\x54\x69\x7F\x7D\x61\xE6\x7A\x70\x61\xE5\x5A\xB9\x12\xB7\xBA\x56\x1C\x5F\x19\xF3\x99\x88\x6C\x9C\xDE\xA4\xD0\x86\xC1\xF7\x04\x3E\xAD\x7A\x39\x0F\xB8\xE5\xDF\x20\x7F\x5F\x87\x4A\x52\xC0\xEA\x7D\xF6\x47\x5A\xEB\x6D\xBD\xE5\x5A\x65\xB3\x65\xC6\xC8\x05\x0F\x16\xC1\xB1\x80\x65\x01\xAA\x61\xC0\x43\xCE\x89\x90\xFC\xFC\xE8\xD3\xB9\x8A\x21\x12\x7F\x99\x7B\xD9\x44\x27\x0E\x57\x57\x6A\x02\x6D\x6A\x97\x65\xF2\x01\x63\x66\x50\x72\x3C\x07\x41\xCB\x0A\xD1\x22\x56\x20\x4B\x04\x39\x1E\x92\xB8\xE7\x64\x73\x8A\xD2\x1A\xE1\xC2\x1C\x79\x76\x88\x9C\x3D\x5F\x07\x5D\x62\x5F\xEE\xEC\x41\x13\xCC\x8B\xBA\xA8\x13\x75\x2C\xE3\xD4\x8C\x8C\xD3\x64\xF3\x69\x5A\xEE\x93\xEB\x81\x8C\x8B\xE6\x64\x9C\x97\x69\xAD\x94\xBB\x74\xC4\x7C\x73\x06\x61\xF6\x9B\x9A\x92\xBA\x91\xD8\x68\x4F\x96\x48\x8E\x2B\x6C\x9F\xBD\xE2\xDE\xFA\x89\x7F\xF5\x3B\xE1\xD9\x5A\xB9\xE5\xF3\xA0\x1E\x95\x4B\xF7\xF4\xF7\xFD\xCC\x17\xDE\xEC\xAF\x80\x70\x09\x16\xBF\xF9\x68\xAD\x5C\xD1\xBB\xAE\x95\x9B\xE0\xA5\x4C\x36\xF3\xFA\x3A\xF4\xB9\xAC\x8C\xDB\x6E\x36\xF8\xAC\x5D\xE1\x7A\xCE\x70\xAE\x0A\xCC\x23\x15\xDB\x91\x18\x6E\x8E\xB8\x3C\x38\xEE\x37\xB9\x81\x77\xD4\x0C\x9D\x66\x90\xC0\xED\xA6\x05\xB5\x10\x2F\x1C\xF2\x11\x0E\xC1\x6C\x40\x74\xB2\x08\x89\x77\x0E\x9A\x04\xEC\xDF\x97\xB4\xD1\x01\x8D\x8E\xC9\xB2\xF9\xCF\xE8\xB0\x97\x1A\x12\x62\x43\xC8\x75\x22\x3A\x59\xD8\x7E\x21\xBA\x57\x08\x45\x93\xD1\xDC\xCF\xDE\x1B\x1B\xB5\x15\x6E\xEB\x37\x83\x46\x5A\x46\x2D\x85\x34\x52\x48\xEF\x42\xCB\xA8\x4F\xCB\xE1\x9B\x8F\xD6\x1A\x69\xD9\x5E\xD7\x1A\x69\xA9\x1F\xF5\xD0\xB0\x62\x13\x28\x3C\x4A\x3C\x24\xAF\xAF\x53\xB6\xC6\xC6\xD5\x88\xF1\xAC\xD6\xD8\x2E\xFF\x48\x35\xCA\x78\xF7\x3A\xF2\x24\x25\x7F\x4B\x7A\x9D\x02\x93\xE4\x96\xF1\x12\x34\x25\xB0\x0C\x86\x86\x59\x23\xAF\x86\xE4\x11\x3A\xEB\x84\xD1\xC9\xC2\xE2\x52\x6E\x12\x88\xFE\x41\x41\x08\x43\xE4\x5B\x44\xC5\xC7\x90\xE0\xB0\xD6\xFD\x6A\x08\xC5\x6A\xBB\xA9\xB2\x5E\x3D\x6A\x58\x0F\xA4\x90\x6D\x60\xC9\x7A\xAE\x64\x95\xB9\x65\xDE\x8C\xD6\x9A\x9D\x6F\xB4\x13\x38\xCD\x52\xA4\x0D\x07\x7E\x1D\x34\xC1\x11\x93\x30\x61\xFE\x96\x2E\xEB\x1C\x92\x9F\x78\x92\x52\x59\x39\xD5\xFC\x44\x5D\x7C\x23\xCE\x35\x6E\x8C\xDA\x84\xC4\x99\x35\x48\xB8\xB5\x2E\x38\x79\xEE\x9C\xE8\x8D\x05\xE4\x6F\x7D\x4B\x9D\xFF\xD7\x41\xB0\x73\xE6\x1B\x01\xFF\x06\xC1\x6B\x19\xE1\xFB\x3D\x84\xC1\xED\x82\x2A\xCB\x9C\x26\x02\x67\xE4\x06\x93\xAD\x4D\xEB\xD4\x29\x4F\xAE\x14\x8B\x24\xCF\x1D\x02\xF4\x4A\xCB\x1F\xA0\x9D\x34\x9B\x70\xAA\xC4\x63\x91\x70\x1A\x6E\x48\xC4\xEC\xB5\x56\x65\x7E\x4C\xF3\x9F\xA0\x0E\xEC\xDE\x8C\x94\x9B\xC1\xA9\x10\x20\xC3\x7E\xBD\x9E\x67\x55\xEA\x94\x03\x6A\xC1\xFF\xB0\xBE\x49\xA3\x3B\xE5\x93\x75\xDF\x0A\x1E\x41\x18\x41\xBA\x36\xAD\xB9\x76\x94\x61\xA3\xF2\xB7\xFD\xA6\x99\x99\x8A\x08\x53\x8F\x36\xA8\x8D\xF1\x06\x75\x85\x39\xEA\x0C\xB3\x17\x16\x2A\x8C\xD5\x3A\x36\x30\x73\xD5\xE4\x13\x12\x7B\xA3\x53\x22\x67\x73\xAB\x3A\x81\x11\x39\xE3\x7B\x8F\x85\x3A\x46\x89\x4C\x07\x16\xAC\x13\xF6\x6B\x8D\x61\xB4\xC1\x82\xB1\xCF\xC5\xB1\x67\xB3\xCE\xF0\x4E\x42\xD1\x27\x70\xE7\x43\xCF\x8C\x80\xCF\x94\x1F\xFB\x55\x4D\x60\xD9\xC9\x2B\x83\x56\x27\x7B\x3C\x31\xF1\xB6\xDE\x0A\x7B\x18\xB0\x3D\x14\x16\x5E\x6A\xFC\xA9\x8C\x8B\x37\x3C\x36\xDD\x1F\xBE\x6D\x90\x99\xC4\x67\xDC\x2A\x7F\xF1\xA2\xFF\xD2\xE0\xF6\xDE\x7A\x54\x95\xFE\x4A\x76\x8A\x7D\x6D\x7B\x4B\x98\xE8\xC7\x11\x05\x0E\xA3\x0C\x6C\x13\xB7\x88\xB4\xBB\xB7\x0E\xDB\xCD\x31\x3D\x8D\x6F\xBC\x40\xB5\x6F\xB7\x30\xCB\xF6\xBC\xC8\x52\xAE\xB6\x4E\x9A\x9A\x1D\x5D\x20\xFC\xDB\x01\x1D\xAF\xBB\xB0\xF9\xDB\x41\x40\x68\x5B\xB2\x07\x89\x37\xC8\x7D\xC5\x7D\xF4\x3D\x03\xF4\xEF\x7B\x21\x61\xE0\x3C\x3A\xC5\x75\xF6\xFC\xD5\x63\xFB\x1E\x87\xD1\xB1\xF4\xF1\x4B\x90\x1D\x53\x8F\x5F\x82\xF4\xFF\xAF\xF4\x31\x85\x6B\xCD\xB2\x2C\x7B\xCF\x06\x84\x4F\xEA\x96\x9B\xF2\xA3\x02\xCF\x13\x79\xEE\xED\x93\x27\xED\xC8\x53\x87\x84\x5A\x4A\x78\xFF\x22\xE6\x42\x6E\x7B\xD1\xD4\x63\x6C\xE1\x44\xCE\xC0\x02\x18\x1F\x0A\x6C\x35\xC6\xE2\xFA\x6D\xE5\xBB\xBE\xB0\xA2\xE9\x7F\xA3\x20\x14\x99\x4E\xBA\x3B\x1D\x63\x55\xC5\x6C\x09\x52\x61\xD9\xBE\xE0\x0B\x2B\x7B\x85\x2D\x24\x11\xBD\x50\x4D\x32\x82\xA2\x2C\xDC\x75\xB5\xD1\xD6\x45\x87\x69\x4E\x6D\xC0\xC4\xD9\xF3\xAC\xB9\xE0\x88\x92\xA0\x50\x5E\xBF\xE9\x8F\x32\x88\x65\x04\x52\x59\xFB\xE5\xBA\xE6\x84\x0E\x35\xE9\x1D\x29\xAD\xFD\x51\xB7\xF6\xFB\x8D\x49\x3D\x41\xFD\xD8\x36\x55\x08\x11\x0F\x75\xE4\x87\x9A\x3F\xA7\x93\xC7\x98\xE3\x08\x76\xEF\xD0\x84\x66\x00\xB3\x91\x81\x68\x6E\xCC\xC3\xFE\x98\xC3\x70\xAC\x79\x22\xDD\xEF\xB6\x1B\x0E\xA3\x7F\xEF\x8F\x0E\x8F\xD0\x64\x07\x55\x6B\x3E\xBB\x16\xB1\x1E\xF8\x8F\x56\xF5\xFD\x8C\xEE\x44\xAF\x22\x53\xD4\x5A\x5A\x03\x71\xF9\x0B\x17\x87\x81\xF7\x52\x20\xE1\x34\xD3\xE2\xDC\x69\x46\x41\xAB\xA7\x33\x80\x0E\x74\x00\x3A\x1F\xD2\x12\x03\x85\xE3\x6B\xFD\x01\x71\x6F\x16\x70\x28\x30\xB9\xE9\xC5\x1B\x12\x2C\x33\x3F\xF3\xC8\xD7\xC7\x32\xD7\x84\x0D\x45\x37\x94\xFF\xD2\x3B\xB6\x89\x0F\x96\x40\x22\x0D\x6C\x00\xAD\xF5\x68\x97\x0D\x50\x6D\x09\xD2\x95\x3B\x5D\x0F\x5A\x77\x5C\xE0\x7C\x28\x47\xDC\xE2\xBA\x66\x4C\x11\x33\xE5\x2F\xD8\x93\xCD\x94\xD5\xE6\x1D\x5C\xF0\x41\xF6\x7B\xBA\x75\x06\xBF\xB7\x36\xA2\x95\x5A\x7F\x42\xBD\x88\x97\xC8\x2D\x1F\xE7\x66\xC8\x79\xAE\x3A\xF1\x3A\xF7\xBE\x47\x2D\x0C\x69\x06\xC9\xBB\x87\x02\x7B\xD2\xCB\x45\x23\x85\xA8\x1B\x14\xC2\x95\x86\x99\x7F\x4D\x26\xB3\x61\x5C\x07\xB6\xBB\xEC\x56\x3F\x18\x9E\xBB\xFC\xCD\x49\xCF\xCB\x5D\x4F\xD5\x5E\x3D\x55\x3C\xE1\x39\x1D\xF4\x8D\x3B\xEB\x9F\xF3\x17\x10\x3C\x8A\xF3\x9F\x8E\x7F\xB3\x97\x4B\xC5\x2F\xA7\xAF\xDA\x9C\x80\xCF\xF0\xC1\x50\x7B\xFD\x4B\xB4\x78\xFF\xCA\x58\xE7\xDB\x7A\x6B\x2E\x23\x9F\xF8\xA2\xE9\x07\x04\xCE\xE4\x91\x0D\x0A\xE5\xD8\x79\xED\x06\x05\x71\x1C\xDD\x20\x78\x8A\xEB\xD7\x7F\x3E\x38\x51\x25\xF8\x2B\x78\x80\x60\x2A\x76\xCA\x53\xD5\x88\x9E\x98\x07\x08\xAB\xE2\xFA\xF5\xC9\x06\xFE\x73\xB2\xCA\xF1\xEA\x65\x1B\xEE\xFA\xCB\x4E\x56\x05\x68\xF7\xAA\x0D\xF7\xAA\x13\x0C\x60\xB1\xE8\xFF\xA0\xDC\x0E\x1D\x43\xA1\xFE\xE5\xAE\x73\xAA\x22\x85\x15\xA1\xF8\x2B\xD2\xCC\x5D\x9F\xF8\x7B\x13\xBC\x37\x3A\x4E\x37\xAF\x1B\x7F\xF7\xBA\xC1\xDB\x59\x91\x52\x14\xBF\x3A\x55\xC4\xA0\xDC\xD1\x0D\x54\x1E\xB5\x7B\x55\x35\x01\xE5\x5E\xB5\x71\x9C\xB7\x64\x96\xFA\x42\xB0\xA8\x6E\x47\x91\x25\xBB\xFC\x65\x3A\x71\xCF\x32\x88\xAB\x09\x8C\x8F\x17\x96\x8D\x10\x3B\xAF\xDD\xE0\x8C\x02\xD7\x5F\x46\x65\x5C\x7F\xD9\x06\x65\x6D\x9F\x40\xD1\xBE\xB2\xF3\xC8\x06\x84\xF4\x0E\x36\x92\x1A\x33\xD9\xA0\x95\x64\x02\x79\x41\x5E\x07\x9A\x31\x58\xA9\x2E\xF2\xD8\xB1\x2E\x6F\xCA\x0F\x7F\x07\x8A\xA8\x09\xF9\x54\x24\x72\x79\x52\x0E\xB9\x29\xB4\x6E\x72\xAE\x9E\xF8\xD1\x24\x48\xC6\x4A\xE5\x61\xE6\x9E\x9F\x85\x1B\x09\x5A\xFB\xF2\xC4\xA9\x07\x49\xE4\x29\xF6\x84\xC3\xF9\x54\xEB\xAB\xC7\xAE\xAB\x8B\xC7\xD2\xC7\xF1\x32\xA8\x0D\x5F\xAA\xC7\x2F\x1D\x31\x65\x6F\x21\x66\x9C\x57\x16\xCE\x62\x36\x58\xD1\xA5\xAC\x3D\x38\x9E\x38\xD5\xF0\x5F\xB2\xBB\x7E\x84\xD4\x45\x45\x8D\x2F\x9B\x55\xF2\x11\x58\x69\x80\x0B\x71\x3B\x9F\x25\x14\x09\x72\x1D\x41\x89\xFF\xF3\x9C\x87\x71\x45\xAF\xB8\xB2\x61\x1F\x24\x6C\xEB\x89\x42\x71\xEA\x16\x43\x64\xC1\x1E\x4E\x38\x5D\xA2\x96\xAD\xD7\x84\x8E\xB5\x39\xB5\x73\x72\x92\x3D\xA6\xC8\xEF\x09\xD2\xAB\x75\x89\x7D\x52\xEF\x3B\xB6\xEF\xF1\x4B\x50\xB6\x1D\x82\x41\x47\x12\x3A\xEF\xE4\x8E\x10\x61\x51\x01\x0F\x40\xBD\xE0\x0E\x30\x59\x52\x48\xFA\xA5\x31\x72\x26\x65\x32\x92\x64\x22\xDB\x94\x46\x91\xB7\xFA\xFD\xE1\xE1\xAC\x7F\xC8\x0B\x48\x80\x5A\x35\x14\x8A\xD5\x8D\x14\xF0\x9D\xDD\x06\x2B\x79\x31\x06\x2B\xB9\x95\xC1\xD2\xA7\x0A\x64\xC2\xAD\x6E\x6C\xD8\xCB\xAF\x9E\x70\xD3\xC1\x02\xFF\x22\xBC\xAE\x09\x07\x45\xE9\xAB\xBD\x21\x32\x57\xBB\x31\x1A\xB4\x9F\x61\x7E\x7A\x63\x44\xD6\x9B\xDD\xC7\xE8\xA6\xDB\xED\x93\x2B\x99\xD9\x31\xDA\xC6\xC1\x99\x78\x27\x0C\x3A\x30\x8D\xBC\x66\xFF\xBA\x7A\x82\xE4\x24\xDC\x4B\x2F\x5D\x05\xBE\x70\x02\x8A\x15\x0B\x7A\x48\x49\xD3\x82\x9E\x29\x38\x10\x5F\x3E\x51\x5C\xA8\x30\x50\xA2\xC6\xEB\x8A\x04\xF2\x64\x55\xBF\x8E\xFC\x60\x80\x59\xBF\x9B\xDF\x7E\xCE\xF6\x84\x85\xF3\xA7\x15\x5D\x2D\x13\x6F\xF3\xDA\xD7\x2A\x35\x13\xDE\x30\xD0\x98\xFB\x91\xF7\xB6\xAF\x49\xDF\xF6\xB5\x0F\x26\x7E\x53\xD8\x9A\x9C\xCA\x2B\x45\x00\xFB\x08\xA1\x56\x1F\x85\x7D\x57\x6A\x55\x04\x50\x76\xA5\x57\x13\x28\xC9\xF2\xA4\xF5\x51\x28\x7B\x96\xA7\x89\x74\x52\x89\xF6\x29\xD7\xF5\x84\x40\x6D\xEA\x09\xEA\x57\x8A\xF4\xAB\x49\xCF\xF2\x54\x7A\xCB\xD3\xBE\x9E\xE5\x89\x07\xC0\x7A\xF2\x2B\xA2\x75\x8A\x84\x0A\x67\x47\x00\xC9\x96\xF2\x18\xF0\x73\x47\x76\xDA\x21\x81\xDA\x61\x98\x74\xC3\x90\xCA\x30\x4C\x24\x7F\x38\x7E\x6D\x65\x18\x66\x64\x2C\x79\x3E\xBC\x5A\xE0\xB3\x7B\xA3\x91\xF5\x13\xC9\xFE\x67\xB6\x4A\xF2\xEA\x43\xA9\x1F\x79\x05\x75\x92\x1B\xA6\xD7\xA2\xD0\x0F\x59\xD4\x0E\x59\x38\x33\x64\xF4\xCF\xA9\xE9\xCC\xE0\xF5\xCD\xFA\xE4\xC3\x6B\x67\x06\x2F\xBE\x22\x41\x01\x9C\x3F\x9D\x07\x2F\xEE\xEA\xA9\x34\xC4\x34\x78\xE4\xE9\x36\x30\x1B\x86\x83\xC1\x93\xEB\x5A\x8B\x01\xA6\x1B\x3C\xDD\x1B\xBC\xD8\x0F\x5E\x34\x37\x78\x41\x37\x7B\x14\xC5\xC8\xA8\xF9\xD9\xA3\xC0\xCA\xEC\x51\x3C\x7B\xD4\xDE\xB3\xC7\xB6\xB3\xA7\x1D\xB6\x2C\xFB\x5F\xC6\x86\x68\xBF\x4D\x2B\x5F\xBF\x18\x31\xE9\x2A\x76\x76\xF2\x06\x7C\x21\xBF\xE9\xB6\x01\xA4\x4E\x7D\x54\xB9\xDF\x55\x24\x1F\x99\xDC\xFD\xF3\x19\xC2\x91\x53\x33\xE4\xB6\x48\x6E\x23\x89\x76\xC0\xF0\x49\x94\xED\xCA\xAD\x42\xB0\x44\x6E\xDA\x34\xF5\xC8\x2D\x7B\xBC\xF6\x24\x4A\xAE\x87\xE7\x7A\xC3\x43\x3D\x08\xC0\x7A\x72\x9B\x1E\xB9\x65\x67\xBC\xA3\xDC\xCF\xA9\x9E\x54\xD3\xAD\xDF\xC7\x70\x20\xF4\xE2\x81\xD0\xBB\x0F\x44\xD4\x1B\x88\xA8\x1B\x88\xCE\x1A\xD1\x0D\xC4\x0C\x9F\xEF\x4A\xE8\xE7\x95\xFB\x93\x1B\x12\x3A\x94\xA2\x0E\x9A\x6F\xDE\x9B\xC8\x6A\x17\x22\xAB\x19\x22\xCB\xF5\x1E\xC7\x7D\xBB\x11\x59\xF5\x88\xDC\xFD\x0A\x99\xDC\x37\x20\x7A\x48\x44\x0F\xE7\x89\x1E\xB6\x44\x0F\xE7\x0D\x1C\x7B\x12\x3D\x94\x73\xA3\x9B\x27\xFA\xE7\x95\xFB\xC2\x4D\x71\xF7\x4D\x11\xFD\xAF\x81\xB3\x17\x12\x7C\x8E\xD3\x87\x03\xC0\xE4\x57\xC3\x95\x43\xAF\xEA\xD7\x4D\x16\x2D\x1C\xBA\x5D\x38\x26\xF3\xEB\x46\xD8\x1B\x82\x70\x7E\x08\xC2\x8E\xEF\xFD\xC7\x37\x92\x2D\xD2\xEA\xAF\x2C\xD9\x02\x81\x97\x89\x37\x94\x2B\x5F\x76\x16\x87\xC0\x8B\x8C\xF9\xF1\xAF\xB5\xD7\xE3\x76\x69\x99\x5A\xDC\x32\x75\xCB\x2D\x53\x8B\x5B\x26\xF3\xAA\xD7\x32\xBD\x88\x27\x5F\xB0\x1C\xBE\x79\x7E\xCC\xB2\x3B\x67\xCC\x43\xA0\xC8\xCD\x6F\xD7\xFB\xD3\x80\xED\x4A\x2B\xEC\x2F\xA3\x18\x42\x82\xBC\x2E\xB3\xC7\x42\x6D\xB6\x95\xB7\x12\x3C\x1F\xF4\x12\xF7\x23\x8F\x7D\x43\xB7\x67\xC7\x9E\x6C\x37\x63\xAD\x02\xBD\xB7\x83\x88\x9C\x00\x75\x1E\x0C\xFD\x13\xDA\xDD\x3C\x18\xD8\xA5\x5C\xB3\x21\xE0\xCB\x56\x05\x58\xF7\xA9\x00\xF7\x10\x9F\x0E\x64\x17\x8E\x93\xED\x50\xF0\x75\x55\xE4\xBA\xCC\x02\x73\x02\xB7\x75\x40\x11\x7D\x25\x82\x68\xB7\x09\x1F\x83\xAE\x38\x68\x9C\xB7\x51\x21\x6E\x40\x22\xAE\x57\x76\xBB\x60\x21\x3E\xA2\xEF\x07\x0B\x66\x55\xDF\xC7\xD1\x17\xE1\x4C\xD5\xE2\x7C\x61\xC9\x1D\x74\xD0\x02\x76\xBD\x08\x21\xEC\xB5\x80\x4A\x5A\x19\xF8\x60\x54\x71\x46\x5B\x25\x97\x60\x77\xEF\x6B\x3A\xE4\x23\xAA\xFD\xE5\x64\x88\xE0\x87\x2B\x83\x87\xF6\xA0\xB9\xEF\x88\x59\xE6\x5F\x2B\x47\x0C\xEE\xC0\x9E\x09\x1A\x54\xE9\xDC\x72\xC3\xF6\x8B\x0F\x5D\x94\x7E\xE3\xDE\xCA\x1E\xD4\xAF\xAE\x62\xDC\x8C\x9D\xAA\x4C\x2F\xFB\x3D\x6A\x91\x9D\x27\x48\xAE\x5A\xB6\x03\xEB\x9E\xF7\xDE\xAC\xD9\xCF\x97\xA4\xE0\x19\x7F\xFC\xF2\xBC\xEA\x65\x1B\xEF\xB1\xEE\x70\xF8\xFB\xE9\x0F\xF7\xFE\x3F\x91\x18\xF9\xF7\xF6\x40\x25\xA9\xB1\x61\x14\x2F\x15\xA3\x6C\x3C\xD9\x7F\x5B\xB9\x2F\xD7\x4B\x33\x06\xCD\xFC\x76\x9A\x6E\xE5\xBF\x20\x43\xD1\x92\xD8\xD6\x56\x74\x59\xFE\xCC\x45\xF1\x3A\x79\xC4\xE9\x93\xC5\x6D\x10\xBB\x0F\x28\x42\x95\x2A\x7F\x96\x83\x58\x70\xAF\x9A\x34\x7C\xBF\x23\x6A\x0C\xF1\x41\xF3\x01\x75\xC4\x94\xEC\x9A\x1C\xB8\x84\x43\x3F\xF9\x61\x70\x50\x8B\x73\xF3\xB3\x58\x1A\x95\x02\xB1\x27\x32\xDF\xEE\x01\x57\x61\x61\xCF\xAA\x23\x3A\xC8\x6F\x23\x61\xD5\xF3\x46\xAC\x13\x4A\xF0\x2E\x6F\x46\xC8\x81\xFB\x21\x74\x39\x0E\x13\x0E\x0C\xB9\xFB\xC6\xEE\x30\x35\x32\x99\x69\xC0\x61\x88\x1D\xD0\x13\xDD\x6F\xBA\xA6\x87\xD0\x6B\xDD\xE1\x86\xDE\x94\x06\x26\xBB\x37\x10\x34\x57\x4D\xA0\x52\x61\x81\x0D\x9E\x6D\xF1\x5C\x7B\xF7\x41\xE8\xCA\xB6\xBD\x09\xB6\xCD\x56\x29\x47\x0D\x84\x94\x94\xD5\xE7\xDB\xC7\x86\x55\x23\x3E\x16\xAA\x33\x0A\x1B\x4F\xE5\x04\x8D\x93\x86\x42\xD6\xE2\xF0\xCE\x6F\xD0\x52\x7F\x58\xD0\x13\x23\x19\x44\xDE\xC9\x50\x1F\x85\xF4\x0A\x97\xC9\x45\x8C\xD8\x43\x43\x4E\x72\xC9\x91\x28\x83\x94\x96\xE9\x1C\x5F\xEE\x2D\xD3\x19\xE4\x03\x47\x0D\xB9\xAE\x33\x46\x73\xC8\xBA\xE3\x88\xAC\x5D\xA6\x47\xDE\x07\x2D\xF5\x3E\x68\x55\x4A\x46\xA7\x4E\x7C\x79\x6A\xD2\xC1\x68\x52\x04\xF9\xFE\x2C\x03\xFD\x0A\xA5\x71\xEB\x47\x7E\xF1\x43\x72\xC6\x4C\x68\x0A\xA9\x53\xFC\x06\x68\x77\x2D\xD8\x70\xD1\xF9\xF2\x9F\x13\x6F\x97\xFD\x02\x54\x5B\x00\xC5\xCB\xF4\x8B\xB8\xA6\xD8\x03\x1D\x5F\x98\x7D\xF6\x50\x8F\x33\x9E\x9D\xAB\x82\xB8\x41\x3F\xA7\xB8\x1B\xF1\x8A\xBE\x46\xC6\x9A\x49\xAF\xE2\xE7\x77\x6D\xF9\x6B\xA5\xE5\xCF\xF7\x8B\xFD\x69\x6A\xF9\xB8\x9B\x95\x54\x92\x53\x1B\xE5\xAF\xD2\x93\x82\x30\x64\xCB\xE3\x05\xAD\xFD\xCE\x52\x60\xA1\x24\x2B\xA8\x04\xA2\x81\xAC\x65\xF6\x44\x11\xB8\x3F\x7E\x7A\x70\x74\x13\xD1\xB7\x8C\x71\x91\xB3\x27\x67\xB1\xEA\x59\x95\x62\xFC\xC7\x99\x97\x27\x73\xDF\xB7\x84\xD0\x2B\x7A\x89\xCF\x6A\x86\x6D\x63\x5E\x2D\x88\x0E\x9F\x52\x55\x82\x7F\x9F\x53\x55\x8E\x7F\x9F\x55\x74\xF8\x08\xB1\x7B\x8A\x07\x22\x61\x4F\xDA\x9D\xC1\xD5\xE7\x03\x91\x2F\xCF\x0E\xE5\x4B\xBE\xAA\xDF\xAD\x20\xC6\xC6\x3E\xA5\x58\xDE\x3C\xAB\x8E\x98\x47\xF8\xE7\x53\xEA\x88\xD9\x51\x3C\xC6\x44\xD0\x1D\xD5\x70\x61\x32\x87\xE3\x15\xFD\x8C\xB4\xE7\x17\xA4\x3D\xD7\x54\x55\x08\x14\x02\x45\x3F\xE0\x22\x46\x2B\x15\x63\x83\xA8\xF2\x1A\xCB\x42\x17\x90\x89\xDD\x05\xE5\xBF\xE2\xF1\x66\x9C\xBD\xCF\x05\x9D\x38\x50\x07\xF5\xEB\x50\x5C\x71\xC6\x2B\xFC\x40\xB9\xA0\xFC\x30\x8D\xD6\x48\x7C\x93\xCB\x3A\x59\xD1\x73\xFE\x81\xA2\xC4\xE4\xA3\x0C\x12\xCA\x5B\xED\xEB\xD5\x47\x51\x7C\xE5\xE4\x04\xE1\x82\x93\x55\x81\x97\x09\x83\x01\xB7\xB8\x23\xAD\xF8\xCF\xD9\xED\xF9\x23\x74\xB2\xAC\x9D\xA5\xE3\x97\xDF\x0A\x4E\x15\x21\xF6\xF0\xA3\x1C\x48\xC3\xB7\xD9\xA0\x22\x17\x75\xE4\xFE\x2C\x78\x80\xE2\xA4\xE9\x63\x2C\xE4\x3F\x8B\x7E\x12\xBA\xDF\x42\x49\x02\x45\x95\xA1\x5E\x51\x54\xF8\xD2\xF3\xF4\xB6\x93\x1C\xFF\xAA\x3D\xAC\x51\x14\x0A\x91\xB1\xE9\x89\x37\xAC\x7C\xE3\x97\x7C\x78\x64\xD2\x70\xF0\x79\x42\xFD\x42\x71\x71\xD2\xA7\x08\x63\xEA\xCD\x11\x36\x4F\x5B\xD2\x79\xCF\xD4\xDD\x24\x6B\x9E\x22\x5F\x3D\xA1\x76\x9B\x67\x4F\x28\x99\x68\x4F\x50\xE4\xE1\xCF\x51\x2B\x13\x1E\x2D\xED\x14\x1D\x4B\x4E\x79\x89\xCB\x63\xBE\xAD\x9C\x1E\xDC\x66\x33\xA7\x4B\x9C\x1D\xDC\x0E\xF9\xB6\x75\xC9\xE0\x36\x5B\xE3\x5C\x29\x97\x86\x2F\x41\x2E\x75\xB6\x98\x0B\x54\x6F\x59\xC6\x0E\x3D\xAF\xBC\x12\xF1\xCF\x46\x2A\xF1\x0A\xC4\x53\xBA\x75\xE0\x00\xE3\x7E\xD2\x23\xFD\x19\xF7\x3E\x25\xC1\x46\xC6\xBD\xBB\xFB\xF9\x79\x25\xA1\x45\xC6\xED\x68\x9F\xCB\xD2\x7D\x51\x35\x28\x90\xFF\xC2\xD0\x13\x7C\xF7\xED\x8A\xFE\x9A\xCF\x28\x30\x3E\xE6\xE4\x6D\xCA\x7D\x51\x8B\x43\x92\x2D\x9F\xBB\xC4\xD0\x74\x4E\xAD\xEA\x5F\xC0\xB7\x82\x57\x06\xD7\x14\x18\xB0\xAB\xFA\xC3\xCA\x5D\x0B\xFC\x9B\xBE\xEA\xB7\xA9\xC6\xDD\xD7\x1C\xBB\x8C\x0B\x07\xD6\x63\x39\xCC\x89\x82\x10\x0F\xB7\xBF\x56\xDA\x5F\xD0\xFE\x5A\x6E\x7F\x95\xED\xAF\xC4\x77\x09\xEB\xFB\x98\x02\x5B\x7E\x42\x1C\x0B\x82\x43\x81\x96\x3C\xEC\xC6\x69\x6C\x1C\x9D\x33\x1E\x31\xCF\xCB\xCE\xF6\x98\xFA\xFA\x22\x70\x4F\xBD\x25\xEA\xB6\x4C\xFA\x8A\xFB\x33\x82\xAD\xDF\xD9\xD9\x49\x40\x1F\xBB\xFC\xE6\x0B\xC7\x92\xED\x2B\xA0\x8F\xFD\xA7\xC7\x76\xAE\x05\x5F\x3B\xAD\xC8\x1E\x6B\xB1\x9B\xAB\xFA\xB2\xE6\x70\xFE\x4F\x61\x77\xD5\x2B\x83\x67\x28\x2F\x72\xF9\x4E\x3A\xF5\x0A\x57\xF5\x93\x5C\x51\x78\x5C\x12\x8B\x19\xF7\x24\x8E\x4B\xF9\x6B\xB4\x23\x8A\xDA\xEB\x8F\xD3\x35\xA1\x7B\x3C\xA9\xAA\x10\xFF\x7E\x8A\x13\x17\x3E\xC3\x82\xCF\xB8\x27\x74\x4F\xF0\x19\x8E\x23\xA3\x3F\x4F\x68\xCE\x36\xBA\xB9\xAA\x3F\x84\xD4\xBF\x4C\xB8\x86\x6F\x53\xA8\x2B\xF3\xF7\x41\xF9\x09\x12\x51\x61\xF9\x97\xA4\x35\x9B\x43\xC1\x35\x45\x44\xF9\xA2\x2F\xFD\x72\x4F\xAC\xFA\xEE\x49\x77\xCC\x41\x73\x59\x1D\x31\x6F\xF2\xBE\x3D\xCF\xA9\x3A\xE1\x06\x4A\x74\x3F\x8E\x76\xCE\x3E\x65\x9C\xD6\x8B\x1E\x3F\xAB\x3C\xBE\xC0\x7C\xBC\x95\x85\xE4\xD5\x83\x78\x2B\x33\x1F\x6F\x65\x56\xF4\xC7\x54\x3D\xA2\xDD\x7F\xD2\x06\x5C\x25\xFD\xF8\x29\x22\xFD\xC7\x14\x5B\x04\x46\x1C\x38\x99\xB6\x11\x57\xE9\xEE\x5F\xB4\xA1\x8D\xD2\x1D\x6A\x78\xC6\x1C\xF4\x2C\x31\x49\xBA\xAA\x9F\x53\x44\x08\x89\x37\xA4\xE6\xF8\x86\x88\x33\x79\xC2\x59\xC2\xEE\x13\x1A\x32\x31\x7B\x23\xA4\xCF\x82\xE1\x4D\x06\x18\xF7\x99\xC1\x53\xAC\xE9\x9B\x99\xB8\x6F\x3A\x62\xEE\xE3\x5F\xDF\x7C\xC4\xAC\xF0\x48\xCB\x00\x62\xD9\x54\x84\x2C\x57\x7E\x9C\xC8\xA1\x88\x19\xA2\x4E\xDB\x22\xCD\x41\xF3\x84\x3E\x62\x2E\x2B\x6C\xEC\x87\x95\xA4\x1E\x57\xD3\x57\x06\x94\x73\x1C\x22\x50\xD3\x3A\x62\x58\x48\xF5\xCA\x80\x50\xD7\x57\xF4\x65\x85\x8C\x9A\x88\xE6\x68\x68\x9B\x48\xB5\x97\xCF\x49\x9D\xCB\xD2\x83\x4F\x0D\x7A\x20\xBB\xA8\xA4\x5F\xFD\xD7\xE1\xFE\xA9\xE5\x98\x64\xA6\x2F\xCB\x1C\x34\xD4\xAA\xCF\x33\x2D\xAF\xB9\xED\xFA\x0A\xF9\x2B\x60\x0F\x70\xB0\x72\xF9\x3C\x28\x3F\x49\xCC\x41\x14\xC7\xD6\xFD\xD4\x45\x91\x3A\x4F\xB5\xB8\xFF\xB6\x0B\x92\xFB\x43\xA3\xA2\x5E\x9E\xE5\x41\x72\x73\x76\x8E\xB1\xE2\x34\xFE\x40\xD1\xCF\x3A\x66\x51\xC1\xF9\xB6\x66\x3C\x0E\x42\xFC\x9F\xA2\x7F\x43\x1D\x66\x60\x5F\xA1\xA2\x3A\x3C\xCE\x70\xF5\xA4\x2B\x32\x74\x45\xE8\xD4\x89\xE9\x8A\x0E\xCA\x3F\xBF\x44\x76\xB7\x9C\x76\xF3\x04\x85\xA2\x1A\x9E\xE8\x64\x81\x78\x85\x8A\x9C\x16\x16\x67\x98\x9E\x9C\xC3\xE3\xA8\x60\x46\x10\x70\xB6\x32\x4E\xF1\x66\x05\x0C\x6E\xB7\x2D\x43\xE1\x5E\xBF\x6E\x36\x9C\x5E\x9B\x72\x1A\x50\x5F\x15\x27\x79\xA1\x6C\xF0\x04\x38\xC9\x25\xAD\xF9\x3D\x28\xEE\x5C\x5F\xA1\x74\xF9\xF8\x77\xC8\xDB\x56\xE6\x7F\xCC\xDC\x4F\x39\xCC\x56\x74\xCE\x77\x39\xA3\x99\xA5\x9D\x31\x41\x72\x50\x83\xD7\x06\x5D\x9B\xED\x58\x45\x2A\x5F\x4E\x47\x2F\x4A\x7A\xA9\x89\xC4\x44\xC7\x0C\x15\xC5\xA7\x07\xA0\x01\x40\xC1\x96\xB8\xA5\xBC\x2C\xAD\x22\x34\x29\x69\x82\x21\xDF\x78\xC7\xE5\x94\xFF\xB3\xBC\x91\xF9\x86\x53\x30\x7A\x46\x4E\xAF\x7E\x9D\xCB\xFE\xF2\x05\x0E\xF4\x64\x30\xD0\x73\x23\x4D\x79\xF6\xE7\x86\xBB\x91\x80\xC1\x85\x43\xFE\xF1\xDD\x86\xDC\x0F\xF4\x66\x43\x20\x39\x2D\x03\xC8\xD8\x07\x95\x91\xB4\x81\x7F\xF5\xB1\x77\xDB\xCD\xA9\xBF\x96\xE1\xFF\xF8\xDF\x90\xE1\xFF\xE0\x44\xB5\x69\x6C\x9E\x0F\xD6\xEB\xB0\xCB\x62\x23\x51\xA7\x91\x8F\x3A\x55\x2B\x3A\xA9\x63\xB6\xD3\x50\x70\xF9\x52\x9D\x36\xF5\x88\xA2\x4E\x33\x1F\x75\x3A\x82\x6C\x10\x75\x2A\x81\xF0\xEF\xF4\xF6\x8F\xE5\x5A\xB6\xE7\x04\xA1\x4D\x81\x35\x4B\xB8\x74\x10\x90\xD8\xB2\xF8\x2E\xD7\x23\x88\xD7\xEB\x78\x10\x75\x1A\x4F\x21\x85\xA8\x1F\x75\x3A\x1A\x44\x9D\x8A\xA7\x2E\x47\x9D\x62\x45\x55\x92\x41\x52\x45\x90\xB4\xF9\xC3\x18\x1F\x61\x97\x20\x4C\x06\xE4\xB6\xE4\x33\x0A\xCA\xE5\xD4\x1B\xE5\x21\x6D\xD6\xEB\x64\xD0\x9A\x64\x0A\xCA\x2D\xD1\x33\xF2\x38\xC9\xCB\xDF\xF1\x3D\x5C\xA2\xDA\xDB\x2E\x46\x19\x44\x90\x10\x91\x62\x4F\xA4\x04\xE2\x96\x48\x75\xE2\xB6\x71\xBF\x28\x50\x54\x89\xB4\x2F\xC1\x75\x73\x55\xBF\x1C\x12\x5C\x07\x81\xFF\x2C\x41\xE2\xCA\x7E\xA3\xB9\xD7\xCB\x84\x94\xB0\xAA\x97\x39\xF2\x27\x01\x1E\x1D\x4A\x28\x9A\x0C\x83\x82\x93\x5E\x50\x70\x9D\xB0\x17\x6F\x17\xC2\xC9\x9B\x23\xCB\xD9\x91\xA9\x68\xA0\xE1\x62\xE0\x84\x88\x5C\x21\x58\xA7\x3E\x40\xE7\x65\x38\x76\x11\xED\x0B\xA8\x61\xD2\xA6\x10\x92\x83\x3A\x41\xBD\x34\xA4\x7D\x3D\x6E\x2B\x43\x88\xF6\x5D\xF4\xB8\x4A\xB4\x97\x78\x36\x18\x58\x7E\xC2\x83\xE6\xA1\x23\xC2\xCD\xA8\x39\x9D\x91\x0A\x08\xB0\xD6\xDB\x28\x16\x54\x72\xB4\x5F\x09\x59\x2A\x42\xF7\xAF\x83\x06\xA2\x7D\x8F\x73\x75\x89\x3B\x30\x57\x5D\x22\xD5\x2D\xA1\x6A\x79\x28\x38\xC3\x5E\x8F\x89\x5B\x6A\xEA\xA4\xCF\x1D\x49\xC7\x1D\x3A\x83\x94\x75\x7C\x3F\x2E\xEE\xE5\x8D\x9C\x24\xC7\x28\x3A\xA2\xB1\xD6\x2A\xC0\xBD\x5C\x7E\xB6\x2E\xCA\x1F\xA4\xF9\x3D\xE6\x70\x00\xB7\xD9\xD4\x13\x97\x3F\x8C\xBC\x1B\x6F\x90\x39\x68\xCC\x28\xDF\x01\x44\xEB\xD5\x08\xAF\xCC\x51\x88\x58\x0F\x8F\x5C\x8E\x7B\xE6\x11\x8E\x6A\x06\xA3\x33\x95\x81\xAC\x1A\x71\xF4\x34\xF9\x07\x3D\x47\x19\xD6\xC7\x50\x08\x2A\xA3\xFF\xEC\xB9\xA0\xF1\x3F\x9F\xE9\x7E\xDE\xDF\xFE\x3A\xDC\xFE\xBA\xA7\xFD\xB5\xDC\x74\x95\xFA\x5F\xCF\x07\x28\xAD\x47\x12\xA0\x05\xE3\xD6\x43\x3E\x71\xD0\xA0\x40\x69\xFB\x7C\xDC\x83\x38\x8E\x18\x56\x93\x8C\x94\x31\xB2\x60\x84\x62\x5E\x62\x3C\x2D\xE7\x93\x95\x0D\x71\x1B\x3D\x52\xC5\x28\xB0\xF0\x7A\xBB\xE9\x04\x48\xF7\xBC\x8E\x4F\x16\x5A\xD8\xA0\x24\x36\x68\xCD\x8F\x2F\x87\x8C\xB0\x9D\xE2\x6E\x4C\x43\x88\xC5\x18\xE9\x19\x02\xDF\x87\x68\xDF\x77\x0A\x1F\xE0\x57\xD8\x56\xFA\xD0\x73\x00\x67\x20\x9F\xE5\x8F\x94\x5F\x54\x8D\x44\xE9\x2E\x73\x70\x0A\x16\x51\x45\x8C\x94\x99\x4A\xC3\x72\xF2\x7F\x87\xDD\xD8\x13\xC8\x28\xC9\x0F\x0F\x0C\x1E\x2E\x1D\x31\xE5\x80\x77\xA1\xA1\x77\x09\x34\x73\xB4\xEF\xF2\x4D\x4C\x16\x1A\x04\x8E\xD5\x41\x6E\x24\x61\xC4\xC0\xED\xEF\xE4\x9D\xAE\xEA\x24\x6E\xDD\x17\xAF\xD4\xF8\x64\x81\x84\x8D\x58\x70\xF4\x84\x6C\x7A\x13\x42\xF6\x78\xA1\xFD\x61\xCF\xCD\xCB\x1E\x5C\x1C\x22\xC1\x9E\x47\x99\x93\x9F\x6D\xD8\x84\x82\x4C\xC6\x52\x33\x90\xB9\x18\x79\xD3\x77\x3B\x7D\x13\x86\x80\x59\xE5\x08\xE8\x65\x8A\xF0\x69\xC9\x54\xA7\x8B\x88\xFD\x90\x74\x9E\x97\xB1\x85\x02\x7D\x45\x2F\x73\xE3\x6F\x41\xA6\x8F\xBA\xDE\x8D\x16\xC8\xF4\xB8\x2F\x6E\x6E\x24\xD3\xD3\x59\x81\xBE\xCA\x8D\xC1\xBE\x11\x3A\x68\xEF\xE4\xE3\x8A\x56\x09\x05\xDF\xED\x1D\xB7\x2B\x36\xB3\xEE\xBC\x07\xF7\x8F\x9A\xA2\x78\x6F\x29\x9C\xB8\x8E\x7A\x91\xF4\x12\x36\xEF\xF1\x69\x08\x16\x20\x21\xB5\xAB\x8D\x8D\xA5\x39\x62\x67\x62\x63\x17\x62\x8D\xA5\x1C\x1B\x9B\xB6\xB0\x00\x29\x01\x5A\x41\x72\x8E\x42\x0A\x18\x6D\xBC\x17\x1B\x2B\x95\x67\x1D\x06\x76\xD4\x80\x11\x7C\xE7\x99\x60\xD9\x9F\x36\xDA\x76\xC7\x94\x30\xA7\xDD\xCA\x19\x2C\xC3\x7F\x98\x15\x9D\xF0\x3E\x35\xF7\x78\x80\xF7\xCB\xB1\xBA\x50\xE4\x5E\x7C\xF8\x10\xC3\x9D\x30\xC6\x87\x60\x6A\x08\x8A\x87\x41\x09\xEC\xC1\x52\x24\x02\xF6\x1E\x06\xE8\xB8\x9F\x92\x06\x97\xBF\x7E\x49\xDC\xBB\xEF\x29\x7F\xBB\xFB\xF9\x69\x6F\x6E\x59\xD1\xF7\x1C\x0A\x34\xD9\x14\x97\x7D\x58\xDA\xB1\x00\x05\x85\x22\xAD\x0D\xB4\x9B\x34\xE5\x93\xDF\x21\xAA\xE3\xA1\xA0\xAC\x43\x9F\xC0\x84\x33\x06\x1E\x0A\x26\xEE\x3A\xE7\x61\x21\x48\x13\xB1\x28\x4B\xBE\x0F\x42\x6D\xA2\x7B\xCB\x1C\xF9\xA6\xCE\xE1\x6D\xFB\xF2\x20\x58\xD5\x07\xC8\xB6\x68\x71\xE6\xD2\x35\x64\x1C\x9C\x8F\x6F\xF7\x50\x4C\xEE\x91\xC3\xCE\xE0\xA0\x39\x55\x47\x64\x00\x3A\xB6\xB3\x73\x4D\xEC\x9F\x11\x92\xA8\xAC\x15\x0D\x04\x11\xF3\xD4\x94\xBC\xDD\xAF\x5E\x78\xF3\xB1\xCB\x6F\xBE\x50\x47\xC7\x02\x5C\xB0\xBE\x76\x4A\x31\x45\xD8\xC7\x7B\xC0\x40\x74\xC4\xBC\x16\x15\x53\xE8\x14\xD3\x31\x8E\x41\x00\xAA\x7C\xF6\x22\xE7\x9D\xC1\x21\x69\x2F\x9F\x57\xCA\x6E\xF3\xF7\x4B\x1E\x10\x45\x71\x0C\xA6\x5E\x77\xFA\x8D\xB5\xA1\x08\x66\xB1\xEA\xF6\x81\x49\xAC\xB3\x4D\xAD\x7D\xC6\x13\xA4\x90\xBC\xAF\x4F\x53\x84\xF4\xDF\xF2\xE8\x9A\xE4\x97\xCB\x07\xD7\xEE\x93\x82\xF2\x66\x57\x75\x29\xC9\xB1\x4F\xCC\x96\x4E\x08\x80\x9C\x35\x3B\xF3\x08\x2D\x0C\x60\xCA\x5E\xF7\x84\x57\xB6\xEE\x2E\x34\x4E\x9F\x77\xDB\x8F\xA2\xDC\x58\xD5\x49\xE6\x77\x59\x33\x50\x4F\x99\x0D\x82\x20\x3B\x70\x03\x85\x95\x1F\xDA\xFE\x43\x3B\xF3\x50\xF7\x1F\x76\x67\x73\xD9\xA7\xC3\x7E\x96\x0D\x89\xFE\x6E\xA1\xAD\x54\xBB\xE7\x62\x54\x18\x4E\x3E\xE3\x91\x3A\xC9\x18\xD8\x36\x93\x93\xB2\x70\xE8\xB8\xC4\x5B\xCC\x84\x8F\x87\x2E\xA1\x7D\x99\xA0\xAA\x11\x8E\x67\x2F\x7C\x9C\xAE\x2F\xCC\x5C\x0F\x2E\x6C\x6B\x5B\x2D\x3D\x30\xA8\x6D\x6A\x9F\x16\x58\x32\x2B\x38\xBD\xE9\x03\xD4\xBB\x12\x64\x83\xD7\x0F\x20\xC7\x61\x09\x8F\x17\x86\xE1\x54\x0D\x84\x14\xE7\xC6\x64\x78\x7D\xAD\x39\x00\x32\xA2\x3E\x6C\x78\x8E\xA6\x75\x30\x02\x35\x13\x52\x4C\x87\xD3\xEC\x3F\x63\xB0\x44\xCA\xD6\x49\x6E\xCF\x8C\xE0\x62\xDB\xC4\x2E\x8F\x78\x9F\x4C\x0B\x04\x4C\x63\xF8\xFC\xD0\xB4\xF0\x93\xD4\xFE\x3A\x66\xEC\x91\xA2\x45\x7C\x37\x55\x28\x40\xE2\x1C\x41\xC4\x10\x78\x04\x9C\x65\x0E\x9A\x84\x82\x5C\xFF\x3E\xF5\x2B\xE4\xCD\x03\xC3\xFB\x42\xDC\x90\x6A\xD2\x51\x91\x6E\x30\xAC\x82\x34\xB8\xC0\xFD\x6A\xD6\x8E\xAE\x0B\x16\x8E\x2E\xC5\x54\xF5\x73\x65\x10\x62\x50\xEC\xB3\xFF\xAF\x68\x0B\xF1\x46\x7D\xE3\x5E\x29\xEE\xD5\x71\x1F\x06\x75\xB3\x9D\x50\xB3\x9D\x50\x6D\x27\x18\x6A\xA5\x4F\xA2\xAE\x5B\x59\x96\x5D\xCE\x55\xD1\x8B\xC9\x54\x73\x99\xF1\x03\x3E\xFA\xF8\x64\xC0\x98\x93\xDA\xFD\x28\x6E\x71\x4A\x11\x1F\x2D\xC3\x7B\x41\xE1\xE5\x0B\x35\x2C\xE7\x38\x1E\x0F\xB9\xFB\x9A\x82\x27\x76\x88\x42\x04\xB5\xA6\x2D\xC2\xB6\x8F\x1A\x5A\x1E\x2D\xE8\x75\x36\x9F\x50\x12\x0A\x9F\xD7\x33\xA0\x25\x8F\x6D\x0C\x96\x00\x4B\x09\x2D\x18\xA2\x46\x96\x40\x0B\x21\x2A\x67\xEA\x8D\x4E\x41\x3C\xAD\x63\x3F\x37\x35\xC5\xAC\x7A\xD8\xB9\x3A\x81\x08\xF7\x6F\x58\x4B\x35\x62\xE4\xFE\x8D\xA6\xCE\x68\x16\x8E\x7C\x88\x3B\xEE\x92\xEB\x5C\x72\xCD\xC6\x1B\x14\x53\x9B\x91\x57\x08\xAA\x7C\xEB\x95\xC5\x2B\x5E\xBC\x85\xDA\xD8\xF1\x4A\x7B\x00\x5D\x4B\x20\x19\x60\xCF\x54\x05\x44\x95\x85\x82\xC6\x2A\x83\x51\x53\x59\xC8\xDD\x01\x96\xB2\xDD\xD7\x94\x5E\x40\x8E\xC1\xF8\x7A\xC6\x2F\xC0\xE5\x0D\xD9\x41\xDA\x6B\x5A\xD7\xCA\xEE\x7A\x09\xAF\x97\xBA\xEB\xE5\x86\xD6\xC3\xF6\xFA\x00\x5E\xB7\xFA\x1D\x2E\x1A\xD4\x5A\x68\xEA\x58\xB2\x85\xB3\x67\x78\xDE\x21\x31\x67\xAC\x1F\xD0\x59\x68\x7F\xCD\x06\x86\x53\xCE\xD8\x0F\x46\x56\x08\xDC\x16\xAF\xEA\x24\xB7\x99\x1F\x79\x82\x5D\xB6\xB4\x4E\x30\x13\x08\x8C\x3A\x23\x63\x9D\xE1\x60\x5A\x8E\x41\x57\xEE\xF3\x2F\xE3\xE1\xE1\x9A\x95\x4B\x9A\xF2\x77\xFD\x91\xD6\xC2\xA7\xCF\xFB\x96\x7C\x95\xF9\xBE\xCA\x7C\x1D\xF3\x91\x01\x8E\x10\x8E\xB9\x26\x1A\x10\x3D\xAD\xB5\x1F\x10\x72\x63\xA0\x85\xC3\x43\x1B\x60\x7D\x82\x5E\xB8\x49\x43\x4B\x8A\x23\xC4\xCD\x2A\xEE\x60\xB0\x3D\xF9\x80\x05\x2D\x31\x20\x99\xBD\xCA\xFF\xF3\x62\xAB\x84\x7A\xCD\x85\x40\xD4\x68\x97\x4E\x89\x72\x88\x94\x17\x9A\x5A\x97\xFF\x97\x38\x18\xC9\x4B\xB9\xEE\xA5\x13\xA4\x4E\xC5\x72\xFE\x9C\x1F\x61\x3D\x2B\xA8\x08\x56\x5A\x02\xF9\x69\xDB\x1C\xF1\x8D\x25\x6A\x13\x93\x8A\x20\xFB\x97\x18\x18\xBF\xE4\x70\x7F\x42\x36\x05\x39\x0A\x8F\xF1\xE7\xAC\x06\x55\x66\x33\x88\x87\xAA\xD5\x22\xB3\x7F\x6A\x55\xB6\xAD\x8F\x0E\x7A\x95\xB7\x81\x98\x3E\x91\x55\x6D\x5F\x43\xD9\x00\x0C\xD8\xE1\x44\x0A\x09\xA4\x0A\x34\x98\x75\x01\x03\x02\xD3\x26\x70\xE2\x89\x44\x07\xD0\x11\xE8\x86\x89\x8F\xC5\x84\x4D\x1B\x2F\x61\xD7\xEB\x88\xC7\x2D\x9A\xD6\x51\xA7\x61\x45\x54\x7A\x6B\xD7\x0E\x71\x63\x89\xB5\x54\x29\x67\x8E\xD8\x68\xEA\x11\x2D\xA3\xA9\x9F\x48\xB8\x0D\xAF\xB3\x6E\x22\x45\xE4\x61\xD4\x99\x90\x34\x5E\x11\x32\x9B\x69\xCF\x97\x48\xF9\x89\x18\x3D\x5E\x32\x6A\xE9\x33\x55\x4E\x30\xEF\x39\x03\x6E\x42\x8A\xEC\x9B\xF9\x89\xD4\x7D\x4D\x38\xE9\xA6\x9B\x48\xD1\x50\xF7\x24\x73\x11\x98\x6E\x22\x45\x6C\x97\x2C\xBB\xEB\x25\xBC\x5E\xEA\xAE\x97\xF9\x04\xAB\xBD\x3E\x80\xD7\x07\xBA\xD3\x30\xE0\xD6\x02\xCE\x77\xED\x27\x52\xC2\x13\x49\x68\x3B\xE2\x89\xC4\x59\x1B\x66\x10\xD4\x13\x46\x1F\x55\xAD\x8D\xA4\x9B\x4B\xDD\xA1\xD6\x3B\x3D\x3F\x0C\x91\xF3\x4F\x10\x9F\x9B\xD6\xF7\xD6\x4B\xDB\x3A\x3C\x43\x0A\x5C\xD8\x31\x86\xF6\x12\xD6\x78\x09\x1B\x72\xCA\xC2\x70\xBD\x36\x28\x8A\x7B\xDC\x91\xB4\x68\x8A\x84\xEB\x46\x12\x96\x0D\x15\xF4\x1D\x33\x46\x38\xAD\xC3\x8E\x31\xC2\xBE\x84\xC5\x9D\x6D\x95\x90\x6D\x08\x75\xE4\x8D\xA6\x36\xEC\xD8\x8D\x9F\x53\x38\x9C\xA9\x2C\x68\x3A\x3D\x20\xFE\x18\xF5\xF8\x83\x0E\xCC\x91\x33\x22\xFF\x16\xB9\x8F\x86\xDE\x2A\xC2\xEE\x39\xA8\x4A\x57\x96\xBC\x74\xEA\x0C\x22\xE4\x8E\xAC\x8A\x84\x3B\x0C\x72\x47\x04\x23\xCF\x1D\xDD\xD7\x14\xD4\x17\x76\xDC\x61\x67\xDC\x43\x2C\x72\x47\xD8\x71\x87\x45\xEE\x08\x3B\xEE\xB0\xC8\x1D\x61\xC7\x1D\xE4\x1F\x1A\x76\xDC\x61\xD9\x66\xDB\xD9\x70\x90\x3B\x42\x72\x64\x10\x6D\xB3\xCF\x1D\x3E\x39\x61\x8F\x3B\xFA\x9B\xA9\x76\xA2\x12\x78\xF3\x80\x41\x2C\x59\xB2\x7A\xA7\x9E\x5F\x15\x18\x5F\x15\x18\x5F\x15\x18\x5F\x15\x18\x37\x2D\x30\xDE\x61\x09\xAD\x97\x11\xA5\xF1\x8F\x39\x0C\x41\x0B\x4E\x1D\xB4\xF0\xD3\x81\xFB\x85\xA0\xBD\xF9\xB1\xEE\xE7\x73\xDD\xCF\x4F\x77\x3F\x83\x57\x06\x49\x97\x7A\x51\xB1\x1F\xE9\xB5\xA0\x71\xE4\x59\xC6\xAD\x31\x14\xCC\x5D\x5B\x47\xBA\xE1\x89\x29\xE1\x01\x53\x5A\x12\x54\xE8\xEE\xF3\x3F\xAE\x91\xD7\x5F\x43\xB3\x99\x78\xE5\x2A\xA5\xA0\xA8\x63\x67\xC9\x26\xF7\xB6\xED\x0B\xB5\x39\x62\x00\xF7\xF3\xC7\xB6\xBF\x96\x2D\x9F\x54\x3D\x27\x0B\x39\x62\x56\x18\x24\x63\x45\x2F\x81\x26\x72\xDC\xD3\x38\xBD\xE9\xCA\xE6\xEA\xB1\xA5\xAD\xAF\x1F\x7C\xE0\xEE\x11\x4E\x54\xDE\xA0\x10\x77\xCA\x2A\xED\x1E\x12\x31\x2C\xA5\xAC\x4F\x4A\xA6\xA6\x94\x8C\xE5\xBD\x47\xEA\x95\x01\xB0\x56\x78\x80\xF5\xC0\x65\xBE\x5A\x22\x10\xC1\xDE\x67\x39\xD3\x27\x9F\xA9\x19\x6E\xA2\x9E\xE4\xA6\xEB\x49\xDA\x7A\x4A\x08\xBB\x0E\x79\xD8\x76\xD5\xE1\xE1\x2B\xFC\xB4\x97\x65\xF2\x23\x61\x9B\x2E\x34\xAF\x95\x20\x44\x72\xF0\x2C\x25\x21\xF9\xCB\x1F\x9C\xC9\x25\xA1\xC5\xBA\xC8\xAC\xEF\x02\x3E\xAF\x31\xD3\x57\x17\x81\xFB\x81\x1F\x1B\xBC\x9C\x33\x42\x88\xB8\x4F\x0C\x2D\x37\x84\x12\x66\xF8\xA9\x9D\x7F\xDA\x43\xB7\x26\x6B\xED\xDF\xC4\x86\x09\xE4\x78\x1F\xD0\x92\xB8\x30\xF1\x9E\x67\x1E\x50\x9A\x5E\xBE\xBF\xE9\xB0\x30\x57\xF4\x7D\xB3\x9F\xCD\x7E\xE3\xC8\xB4\xE9\xD1\xD6\x7D\x3A\x9F\x0E\x06\xD3\xEC\x0E\x83\xA9\xD9\xCA\x78\x78\x06\x06\xF3\xF0\x0C\x0C\xE6\xE1\x3E\x0C\xE6\xE1\x85\x30\x98\xBC\xC7\x67\xD9\xC4\x96\xF6\x1E\x0C\xE6\xE1\x59\x18\x4C\x8F\x12\xE1\x61\x30\xEF\xCF\x6E\xB2\xAF\x54\x9A\x87\x1E\x3F\xBC\x08\x98\x7C\xC9\x3F\x5D\x9A\x7F\x9A\x7D\x6F\xA2\xF6\xF7\xFC\x83\x6C\xDF\x9A\xD6\x1D\x99\xD5\x61\xD3\xE9\x2B\xEC\x12\xBB\x51\xC5\xD0\xB9\xEF\x54\x49\x8B\xF0\x56\xA7\x9C\x65\xC2\x05\x84\x85\xC7\xFB\xC3\x9C\xA5\x6E\xC1\x0C\x34\x86\x14\x69\x31\x71\x1C\x6E\x91\xC9\xEB\xA5\x0B\xAA\x7D\xE6\x28\x8C\x21\x6D\x04\x7C\x15\x4A\x01\xBA\x2A\x8E\x17\x1A\x42\x50\xAF\x2E\x38\xE7\x05\xFE\x26\xB8\x17\xCA\x03\x02\x99\xD3\x9B\x90\x34\xB5\xE2\xF3\x4C\xB7\x7D\x82\x2C\xDD\xDB\x9C\xBD\x6E\x1F\xBE\xD6\x50\x73\xF4\x51\xC8\x57\xF4\x43\xF5\x7E\x01\xE9\xB9\x0D\x5B\xBC\xE4\xE2\x0D\x0A\xC8\xE3\xF3\xFC\xB1\x0D\x94\xD2\x2A\x83\x94\xC2\x4A\xF7\xC3\xD2\x30\xC1\x4E\x7D\xFB\x6C\x0E\x9E\xFD\x70\xFB\x2E\x39\x78\xA0\x70\xEA\x04\xEC\x23\xB8\x3A\x5C\x83\x75\xA5\xF2\x08\x5B\x71\x5B\x9B\x9B\x85\xB3\x31\xD0\x09\x39\x2F\x06\xDC\x58\xC0\xA6\x55\x4B\xDC\x5B\x6A\xE1\x85\x66\xAC\x03\xA5\xE9\x89\x85\xD1\x39\x0E\x39\xC9\x0F\x05\xEA\x38\x94\xE7\xAA\x32\x83\x89\x64\x90\xF4\xD6\xF3\x0C\x22\x9C\xC1\x3F\xD3\x4B\xFD\xE2\x14\xD1\x35\xE3\xCC\x1D\x19\xB2\x5D\x0A\x8C\x6D\xCD\x47\x9D\x6E\xFB\xC4\xB4\x22\x33\x8C\x81\x18\x46\xD3\xFA\xB6\x73\xFE\x04\x98\xCF\x3B\x38\x74\x51\x5A\x5C\xA7\x90\xBC\xA6\x50\x90\xA0\x42\x81\x63\x28\x83\x93\x12\x3E\x26\x83\x91\x84\xC8\xF5\xC2\x1D\xE4\x7C\xE2\x54\x55\x52\x12\x43\x6C\xAC\xA2\x77\xC9\x40\xAC\xC0\x4A\x82\x46\x05\x25\x95\xCE\x05\x65\xCC\x3D\xE4\xC2\x63\xB8\x72\x4F\x30\x0A\x3B\x7B\x9F\x17\x2B\x9A\x0C\x50\x29\x96\x2D\x2F\xB8\xAD\x96\xA2\x92\xC3\x50\x51\x66\x0A\x64\x43\x32\x09\x13\xAE\xBF\x8F\xB2\xD3\x9B\xF5\x18\x35\xA7\x25\x67\x69\x58\x4A\x1F\xB2\x19\x42\x89\x8D\xEF\xC3\x3A\x85\xDC\x90\x94\xC7\x6D\x45\x5B\xAE\x48\x93\xAB\x6C\x49\xB3\x3A\x83\x25\xAC\xAC\x84\x71\x53\x95\x92\x29\x54\xF9\x72\xB2\x45\xE5\x64\x52\x0E\xA7\x46\xE4\x9A\x95\x04\x26\xDC\xEC\x47\xB4\x26\xD6\x0A\xCA\x13\x85\xCE\x83\xBE\xED\x06\xBF\x80\xDB\x60\xE4\xCA\xCD\x73\xE5\x1F\x5C\x14\x28\x14\x2F\x72\x6D\xE7\x38\xA6\xB2\x3F\x33\xD6\x6C\xAB\xAD\xE1\xF9\x6A\x17\x05\x6A\x70\xB1\xA4\x60\xD0\x9C\x91\x31\x12\x81\x30\x22\x37\x04\x12\xBA\x87\xC9\xED\xCE\x2D\x37\xE5\x1F\x72\x88\x2D\xED\x98\x50\x0D\x96\x9C\x5A\xC1\x41\x03\x75\xCC\x36\xAE\x44\x86\x62\xA9\xC6\x4E\xD0\xB1\x7B\x9D\xA1\xCC\xB9\xB7\x79\x5F\x9D\x93\x1A\x72\x6C\x5B\x12\x79\x48\x16\x48\xC8\xDF\x7C\xC4\x00\x41\xAA\xE5\x8D\xA8\x07\xB2\xEC\x90\x57\x5A\xE0\xF2\xC6\x05\x90\x79\xDF\xE6\x54\x82\x69\xFD\xCA\xC5\x88\x33\xB8\x1C\x44\x8E\xC1\x0D\xD3\x76\x49\x48\x71\x49\x48\xDD\x81\x0A\x25\x59\x66\x28\xBB\xA4\x85\xA0\x71\x9B\x22\xB5\xFD\xF5\x9B\x66\xAE\x2F\xCC\x5C\xF3\x12\x81\x4B\x02\xE1\x97\x22\xBF\x67\xBC\x24\x10\x39\x32\x56\xB5\xAA\xA0\x57\x85\x7C\x62\x1B\xCA\x12\xDA\x06\x4D\x5A\x42\x16\xA6\x94\x82\xEC\x42\x23\x96\x43\xBC\xBF\xC2\x7A\x8C\xDF\x6B\x50\xB6\x38\x52\x80\x5B\x35\x93\x35\x6A\x0F\x3C\x65\x9B\xF2\x33\x17\x05\x28\x87\x0A\xB5\xDD\xD1\x6C\x90\x7D\xC1\xAA\x62\xBB\x55\x18\x8A\xC0\xFD\xDA\x77\xF5\x0E\xF4\x35\xEE\xA3\x3E\x11\xF0\x8E\x47\x73\x0A\x2E\x25\x89\xF2\x08\x34\xAC\x4B\x64\x6A\x49\x52\xB8\xA0\x62\xC3\xB3\xDB\xDA\xA8\x62\xF2\x19\x8F\x38\x20\x87\xF9\x9A\xBE\xC0\x25\x65\xE7\xF9\x77\x47\x67\x71\x17\xF8\x3D\xDF\x77\xF5\xF8\x59\x57\x9C\x6F\xEA\x85\x77\xE9\xCC\x8F\x72\x08\xC5\x60\x9D\xF6\x87\x85\x6C\x7C\x8D\xDA\x1C\x5D\x8B\x8B\xAC\x0C\x41\xC6\x7A\x61\xC6\xDB\xA6\xF6\x71\xFB\xCD\x39\x09\x20\x4C\x5F\x5F\x67\x7C\x6A\x38\x62\xD0\x56\xCE\x61\x3C\xAA\x72\xC2\x47\xE5\xFB\x8F\x54\xB9\x9C\x1C\xE6\x3D\x48\xDB\xA0\x05\xB2\x0D\xE8\xD3\xAA\x90\x83\x35\xCD\xA6\x86\x91\x4F\x81\x2E\x15\x18\x28\x36\x38\xD6\xDF\x70\xC9\x06\x4B\xA6\x84\x88\xF9\xC9\xC2\x48\x74\x61\xE0\xE9\xCE\x10\xA2\xFD\xA1\xE1\x44\x8B\xEE\x17\x87\xB7\x2C\x98\x13\x3E\x8D\xA0\x59\xD1\xC9\xC9\x82\xB3\x6A\xD5\x63\x3A\xE1\xC3\xCB\xB1\x0C\x20\xD9\x3E\x88\x72\x74\xDA\xB6\xA2\x03\x7C\x2A\x86\x03\xDE\x1A\x3A\xC5\x99\x1F\x61\x0C\x71\x1D\xE0\x22\x4B\x19\xF5\x39\xEF\x2A\x6E\xC9\x38\x04\x73\xF0\x0D\x44\x84\x8D\x46\x5B\xB4\x31\x25\xED\xD2\x9C\x2E\x5C\x0B\x83\x65\x19\x24\xD9\x07\x53\x3D\xDE\xD6\x6F\xEE\x41\xC2\x0F\xF9\xCE\x7A\xBE\x0B\x39\xD9\xAB\xAD\x25\xF9\x54\x84\x7C\x17\x73\xCD\xAC\x85\x44\x9E\xEF\x04\xE1\x74\x6B\x83\xD4\x8C\x08\x12\xC6\x78\x17\xBE\xA3\xAE\x86\x2D\x93\x84\x3D\xBE\x5B\x78\x97\x76\x9C\x94\x04\x2C\x45\xCD\xA7\xA9\x65\xE3\x19\xF3\x02\xDE\x65\x66\x59\xFC\x71\xD6\xBB\x6C\xDF\x11\x3E\xF3\x98\xBA\x59\x07\xD4\x9C\xBD\xBE\xCE\x3D\xE7\x65\xC4\x79\x05\x73\x5E\x81\x37\x98\xF3\xB2\x47\xAA\x42\xD0\x6E\x0B\xCF\x79\x6C\x08\x69\x7F\xE1\xA7\x0C\xBA\x27\x49\x24\x3B\xBE\x93\xE2\x43\x18\x6F\xF0\xD9\x6E\xC8\xE5\x86\x58\x2E\x21\xA0\x16\x0C\xB6\xEC\xD3\xDC\x5B\x72\xAD\x9A\xE7\xBA\x70\x86\xEB\x08\x14\x8B\x06\x5D\x73\x86\xD0\x04\x42\xE2\x3C\x92\xFF\xB6\x2E\xE9\x5C\x1E\x2F\x4B\x1C\x42\x23\x58\xA5\x2B\xFE\xBC\x3D\x24\xCE\x0B\xBA\x20\x43\xDB\x54\x51\xCB\x79\x25\xC4\x75\x8A\x9C\x97\x32\xE7\x51\x0A\xDC\x84\x33\x81\xEE\x1B\x7E\x03\xFB\x20\x61\xCE\x4B\xA1\x7C\xB0\x30\x92\x99\xDB\xD2\x1E\x21\xF7\x61\x57\x6C\x47\xB2\x22\xB8\xC5\xE4\xD5\x03\x89\xDE\x0F\x59\x1F\x24\xFA\xB6\x6F\x24\x94\x41\x95\xE1\xC4\xDD\x84\xCC\x99\x35\xC8\x78\x88\x18\x24\x9A\x85\xDF\x6D\xB0\xFF\xAD\x6F\xA9\xF7\x0B\x3A\xF3\xFE\x1E\x3A\xF3\xFE\x1E\x3A\xB3\x13\x73\x0D\x44\x10\x11\xDE\x72\xE4\x41\xA2\x23\x12\xFE\x1B\x45\x0B\xFE\x1B\x09\x3C\x73\xD8\xEA\xBF\x34\x09\x78\x08\x23\x36\x23\x9A\x35\x9F\xAC\xA5\x0F\x11\xBD\x57\x23\xD8\xCB\x15\x6B\xBA\x25\x88\xE8\x90\xD4\xBC\x69\xCD\xB5\x53\xEC\x62\xF9\x9D\x97\xFA\x10\xD1\x4C\x96\x3A\xDC\xA0\x36\x8E\x98\x85\x09\x9D\x2F\x3B\x43\x06\x4B\x66\x69\xE6\xE3\x76\x03\xC1\xBC\xCC\xD9\xB5\x43\x76\xB1\x68\x21\xA2\xE9\x20\x29\x6A\x21\xA2\x39\x81\x3B\xEA\x58\xE4\xE9\xC1\x6A\x9C\xD4\x1A\x71\xAD\xD1\x62\xE6\x56\x3C\x53\x7B\x59\x3E\xFB\xD6\x30\x4E\x7B\xEB\x54\x35\x99\x83\x88\x9E\x30\x44\x34\xBD\x70\xC9\xB6\x00\xA7\x1E\x99\x61\xB0\xB5\x66\xFF\x8B\x4C\x92\x6C\x31\x32\xB0\x75\x81\x18\x13\xE7\x36\xD7\xE4\x08\xCB\xA6\x46\x1A\x65\xE4\x49\x08\xA7\xA0\x4F\x17\x81\xFB\x93\xE1\x8B\xF9\xC4\xA7\xFE\x53\x7C\x68\xCC\xB5\xFF\xFA\x8F\xDC\x44\x71\xF8\xE2\xFB\x77\xAD\xB7\x7D\xED\x35\x0C\xB7\x6C\x49\x43\x6B\xBA\x84\xD4\x64\x98\xEC\x96\x59\xDC\xE6\xB8\xA7\xBA\x70\x67\x9F\x30\xF5\xCB\xD0\x9E\x60\x97\xF6\x04\x83\xF6\x9C\xE0\x54\x86\xBA\x7C\xFC\x52\x67\x06\xF1\xB9\x84\xE6\xC7\x87\xF3\xCE\x52\xA5\x37\x1C\x9A\x9C\x65\xE2\xA0\x05\xF8\x27\xFB\x79\x36\x12\x52\x9A\x2E\xF6\x71\x31\x6E\xFB\xD1\x2A\x74\x8A\x61\x12\xF9\x24\x75\x19\x8B\x7C\xF6\x6A\x7F\x45\x13\xD7\x25\xEF\x2F\x6D\x38\xA5\x32\xB9\xA6\xA9\xA6\x8E\xC6\x3A\x30\x38\xAD\x0E\x05\x50\x68\x50\x2E\x60\x6B\x96\x5A\xD1\x07\x6A\xCE\xF2\x7B\x00\xEF\x82\x29\x3F\x4B\xB1\x5E\x8C\x94\x62\xE6\x08\x2F\xD3\xB1\xAC\x63\x6F\xD7\x49\xA6\x60\x66\x89\x4E\xD8\x91\x31\xEF\xCF\xC9\x6D\x2D\x66\xB7\x59\xF3\x1A\xF6\x38\xB5\x94\xC6\xA3\xE9\x92\xA3\x9F\x29\x38\xD2\xC1\x6B\x5B\x27\x0A\xDA\xD1\x71\x2B\x7B\xCD\x1A\x7A\xE6\x04\x75\xBA\xA2\x1F\x92\x2C\x3C\x94\xF8\x34\x91\xCC\xB7\x91\x0E\x94\x09\x54\x94\x41\xE2\x92\x8D\x3A\x71\xE6\xBC\x53\x8F\x82\x41\x45\x22\x39\x5E\x58\x06\x02\x4C\x38\x49\xAC\x0C\x81\xA4\xAE\x67\xBC\x91\x58\x12\x48\x26\x92\x22\x4C\x8B\xAF\xED\xA9\x42\xA3\x2E\x45\x47\x3D\xF2\x2B\x39\xC1\x79\xBE\x49\x8F\xDE\xEE\xCA\x60\x12\x26\xBC\x29\x77\x7F\xD4\x1F\xAD\x88\x52\x77\xEE\xDD\x02\xAC\x9C\x10\x00\xAE\x3E\x3D\x93\xAB\x16\xC2\x8D\x82\x53\xC0\x12\x5A\xB6\x21\xC7\x5F\x2D\x00\x94\xCF\x1A\x15\x6F\xB7\x69\x0B\xC8\xAF\x73\x55\x1F\x10\x89\x28\xD8\x9E\x39\x69\x56\xD0\x80\x01\xC3\xA9\x78\xF1\xC7\xB5\xE0\x81\x29\x2A\x60\x6E\x47\x9F\xE0\x8C\x8F\xFC\xD5\x12\x2F\xAB\xE9\x03\x38\x3D\x0E\x05\x41\x61\x40\x23\x2B\x29\xAC\x05\x07\x89\x6A\x39\x40\x1B\x2E\x64\x25\xED\x02\x48\x3D\x2B\x79\x68\x2A\x72\xFE\xC7\x01\x4B\xDB\xED\x67\x9D\x8A\xF5\xC3\x06\x5A\x2B\xCA\x22\x6E\x20\x73\x11\x59\x33\xD4\xC6\xB4\x1E\x41\x46\x69\x80\x15\x69\x2E\x92\x06\x58\xD1\xE6\x87\x73\x68\x52\xD6\x2C\xB0\xE5\x1F\x89\x4F\x42\xB6\x1B\xEA\x31\x2E\x15\xAF\x50\x1A\x9B\x78\x9A\x14\xBA\xD4\x50\x42\x05\x2C\x27\xC5\x4D\x6E\x5E\xA5\x30\x6A\xAD\x28\x9C\x0B\xF9\x4D\x8D\x53\x6F\x6C\x28\x0F\xFC\x0D\xCA\xA6\x82\x69\x1D\x73\x5F\xBA\x6E\x36\x70\x25\x8F\x70\xFB\xD6\x02\x27\x5A\x0A\xD5\x1D\x1D\x0A\xC8\x62\x96\xF0\x5E\x91\x2C\x3A\x21\x81\x25\xE1\xFE\x51\x73\xC6\xF1\x3A\x6A\x5D\xCF\x49\x22\x90\x5A\x98\xFD\xDB\x50\xA5\x3D\xCB\x9A\x84\x9E\x3B\x2D\x7E\xD3\x8A\x3C\x52\xF1\xC7\xD8\x04\x81\x66\x3C\xEB\xD6\x5E\xE6\x7D\xAB\x77\xB4\xE4\x00\x5F\x92\x4D\x14\xE8\xC6\x7D\x3E\x68\x24\x31\xB3\x62\xB1\xC7\x27\x2B\x5D\x4E\x01\x72\x61\x62\x56\x22\x20\xF8\x8E\x95\x78\xAF\x86\x55\xB9\xC0\x7B\xFA\xF9\x65\x97\xA1\x0D\x12\xA7\x9B\xB1\xB6\x01\x6A\x1C\xCC\x2E\x11\xB3\x0B\x95\x74\x80\x82\x81\x91\x5D\x22\x14\x0F\x1D\xBB\xD0\xB1\x27\x05\xF3\x27\x3C\xAE\x99\xD7\xFE\x63\x14\x66\x24\x32\x2C\xC7\xDE\xB1\x87\xBE\xBB\xAE\x36\xEA\xD8\xE9\x73\xE4\x57\x3E\x3A\x51\xA7\xD3\xDA\xBA\xD1\x39\x9A\x58\x3B\xFA\x44\x9D\x4C\x09\x90\xBE\x3C\x4F\x7B\xF3\x84\xAC\x46\x21\x68\xF7\xBB\x6A\xC3\xFD\xDC\x76\x43\x50\x1D\x77\xAF\x15\x82\x74\xEB\x7E\x37\x38\x01\xDA\x3D\xC4\xF7\x47\x6B\xE7\x24\xC3\x4F\xC9\x69\x57\x73\x72\x5C\xCA\x20\x76\xDB\x8F\x6E\x38\x70\x25\x44\x6E\xC7\x6E\x1C\x87\xF0\xD1\x29\x8C\xCE\xD5\x23\xB6\xC2\xB9\x2E\x00\x41\x68\x66\x65\xFA\x59\x9A\x7E\x84\x99\xC2\xBA\x66\x1D\xBB\x6D\x1A\x18\x88\x1F\x98\xC2\x88\x3B\xED\xD9\x01\xDF\xA6\x4F\x5B\x76\x68\x17\x35\x1E\x9D\x84\x1C\x12\xC9\x65\x27\xE3\x58\xF2\x09\x9D\x09\x26\x38\x3B\xC0\xB8\x49\x53\xFE\xF1\xC5\xFE\x5A\x28\x83\xDA\x7D\x06\xA9\xDB\xD1\xE7\xF0\xE5\x43\xC1\xA4\x8E\x29\x7B\xAD\x3D\x07\x09\x84\x53\xD9\x06\xF9\x14\x11\xD6\x99\x13\x8C\x3E\x41\x60\x0C\xB8\x1F\x4B\xCA\xDF\xE7\x68\x61\xE6\xB6\x68\xC0\x6D\x11\xC5\x7D\xB1\x73\x6C\xC7\x6D\x11\xA3\x29\x32\xB7\xB1\xFA\x43\x07\xBC\xDE\x24\x94\xFD\xA4\x51\x76\x3E\xD2\x94\x91\x9A\xAC\x0F\x2C\xB3\x59\xF9\x1F\x2E\x89\x7F\x3B\x04\x84\x23\x45\x9E\x46\x6C\xE1\x47\xE5\x4E\x83\xEA\x52\xF4\x7F\x7A\xE1\x51\x5A\xEF\x80\xCD\x1F\xBB\xDD\xF2\x91\xC2\x8A\xBE\xFF\x96\xCE\x13\xEC\x2D\x9E\x27\x84\x72\x9E\x60\x5F\xA4\xF3\x84\x81\x5F\x34\x9F\x27\xD8\x5B\x3B\x4F\x40\x1E\x22\x82\x43\x2F\x44\x34\xCB\x09\x88\xA5\xFC\x13\x9A\xB2\x5F\x18\xA9\xFD\xDB\x62\xF7\x61\x0D\xDE\x34\x9E\x47\x42\xE4\x1F\x47\x91\x84\x7C\x24\x90\xB8\x54\x44\xBE\x77\x4A\x50\x2B\xFA\x9E\xE3\x34\x4F\x23\xB0\x55\x2C\xBC\x77\xB2\xD0\xED\x9D\x84\x6C\x2E\xB4\x24\xD3\x3D\xDA\xC2\x8D\xC8\x8C\x89\xB2\x17\xC5\x7B\x82\xA2\x88\xEC\xFF\x75\xE6\xB6\x4E\xF0\x2A\x7E\x8F\x53\x90\x89\xB0\xA1\xB3\x6D\x94\xFB\x94\x07\xC6\xA9\x07\x24\x0D\x77\x94\x61\x15\x2D\xFE\x17\x8E\x1C\x44\x64\x22\x77\x41\x95\x43\x82\xA5\xB3\x4D\x07\x89\x59\x13\x81\x08\x0C\x3D\xF3\x3C\xDE\x13\xA5\x1C\xD7\xEA\x31\x9E\xC8\x76\x57\x7E\x8E\xB6\x8B\x83\x27\x2E\x38\x14\x7C\xFE\x87\xBF\x93\xE3\x99\x73\xC6\xBB\xCE\x29\xA7\xFA\xFD\x6E\x74\xF6\x4C\x6F\xFE\xF9\xC0\x4A\xB2\x4A\x10\x44\x46\xC1\x01\x99\xCE\xAC\x51\x99\x2E\xA8\xC6\xB8\x71\x76\x01\x01\x31\x33\x6E\x4D\xE9\xB6\x99\x66\x63\xA7\x37\xEB\x1C\xC6\xEE\xFA\xF5\xEB\xD7\xCD\x06\x8C\x4F\x4E\x39\x06\xB1\x9E\x30\xC5\x72\x31\xFE\xB9\xC0\xA7\x0F\x4A\xB1\xE4\x7D\x5C\xF2\x7E\x24\x01\x6D\xE0\xB0\xD4\xC2\x6D\x53\x5C\x4E\xBD\x1F\x4B\xBD\x0D\xF6\xFB\x52\xF7\xB7\xA5\xE6\x14\xB1\x04\xB7\x75\xA5\x12\xEA\x14\x2A\x54\xE5\xBF\x91\x65\x7B\x24\x78\xD8\x63\x28\xC5\x98\x2C\x35\x27\x3C\xB3\xF6\x31\x5A\x12\xEC\x87\x42\x9E\xE3\x72\xCB\x0C\xB2\x7F\x40\xD3\x94\xB6\x8F\x9E\x3E\xFB\x65\x53\x0A\xFB\x20\x3F\x7E\x8E\x22\xB5\xDE\xE9\x0D\xFC\x23\x8F\x2B\x3E\xF1\x41\x70\xB6\xCA\x61\xBF\x58\x7B\x2C\x5B\x71\xB5\xEC\x61\x4F\x14\x86\x8F\x7D\x01\xD9\xC3\x3F\xED\x71\xAD\xE3\x84\x3A\xC2\xB7\x2D\xBF\x32\x07\x13\x85\x46\x6C\x63\x1A\xF1\x89\xCC\x48\xF8\x74\x64\x8E\x32\x7D\x46\xAC\x0F\x23\x9F\x2A\xCF\xA7\xB1\xE7\xD3\x11\xBD\xCA\x9B\xD1\x8C\xF9\x94\xD9\xD4\x10\x06\x1F\x7B\xB5\x80\x26\x7B\xE8\x08\x52\x4F\x3D\x3E\x43\x23\xC1\x25\xB9\x2C\xE8\x0C\x0F\xC5\x7D\x77\xA0\x46\x9B\xF3\x01\xF3\x32\xC9\xE4\x48\x40\xB8\x38\xE5\xC0\xD7\x1E\x9D\x5B\x3F\x79\x48\x98\x4C\xFD\x31\x15\x1B\x59\xDA\xB2\x9F\xF1\xEC\x97\x92\x83\x12\xA4\x9E\x51\xD2\x96\x51\x22\x66\x3F\xF1\x73\x41\xE5\xA3\x57\x20\x05\xB4\xA7\x60\x78\xF8\xEB\xF4\x38\x67\xC4\x36\x37\x18\x7C\xE3\x07\x3F\x61\x61\xCA\xA7\x06\x32\xD8\xD4\x40\x73\x9C\x8E\x00\x65\x9C\x65\x08\x53\x3E\x94\x22\xDF\x52\x20\x4F\x28\xCA\xBB\x87\x63\x9F\xEB\x0C\x6F\x0C\x47\xBE\x2F\xAF\x76\x1D\x77\x45\xD7\x37\x31\xF6\xFA\x66\xC6\xBE\x27\xA3\xA4\x9D\x19\xD9\x2E\xBD\x08\xFE\x77\xD4\xC5\x7F\x91\x89\x08\xB6\x95\xF6\xEB\x11\x6B\x84\x4E\x64\xB0\x97\xBF\xF1\xB0\x3F\x6D\x6F\x0C\xED\x3C\xA3\xB6\x57\xDC\x0B\x53\x85\xB4\x09\x41\xFD\xCD\xF7\x8C\x71\x5D\x28\xD7\x3D\x36\x97\xB3\xC5\xC7\x14\xED\xC8\x42\xD0\x6D\xF9\xC8\x0F\x86\xD5\xC4\x2E\x8E\x7C\x17\x13\xFA\x86\x18\x11\x45\x3A\xC3\xE8\x4B\xE5\x84\x23\xD4\x54\x9D\x30\xD6\xFD\xB4\x4E\x39\xC4\x14\x50\x68\x6A\xDA\x37\x71\xC0\x59\x82\x33\x02\x39\x7A\xB4\x40\x1C\xF3\x59\x8D\x4F\x8A\x48\xD1\x98\x9E\x7B\xFA\x4F\xE6\xC5\xB1\xB9\x45\x71\x1C\xA3\xD0\x4C\x17\x88\xE3\xD8\x8B\x63\x22\xE2\x84\xC5\xF1\xC4\xCF\x87\x49\x3B\x1F\xC6\x4C\xC0\x9E\x38\x8E\xB3\x4E\xD0\xCF\x8A\xE3\xC4\x8B\xE3\x64\x2F\x71\x9C\xCC\x88\xE3\x64\x4E\x1C\xF3\x64\x1F\xC3\xA4\x15\xC7\x52\x73\xBC\x9B\x38\xE6\xE0\x5E\x2F\x8E\x3B\x9A\x26\x73\xE2\x38\xD9\x45\x1C\x63\x9D\x63\xA4\xF3\xD8\x8B\x63\x33\x2B\x8E\x45\xCE\x24\x5E\x1C\x67\x03\x71\x9C\xEC\x22\x8E\x67\xE0\x85\x65\xCC\xF8\x96\xBC\xE6\x75\x89\x07\xDB\x07\x0B\xD4\x09\x3D\x50\x28\x42\x66\x68\x7C\x5B\xB7\x0C\x1D\xDE\x40\xAF\xC0\x57\x3D\x3F\xD3\xB9\x1D\x32\x31\x24\x55\x0C\x19\xD9\xF5\xFC\xD6\x89\xF2\xFD\x33\x6A\x46\xE2\xD1\x25\x9A\xBE\xF4\x4E\x50\x7A\x8F\x5A\xE9\x9D\x0C\x18\x5D\xEC\x2D\x34\x11\x3D\xC7\x27\x34\x5A\x71\x6F\x4C\x3A\xD1\x1D\xEF\x22\xBA\x39\x6B\x28\xB3\xAA\xF6\xAC\x8A\x85\xD6\x16\x12\xCF\x54\x49\x4F\x74\x13\xAB\x4A\xE8\x88\x77\xE1\xEF\x44\x37\xED\x46\x44\x74\x27\x22\xBA\xF5\x0D\x18\x45\x7B\x46\x89\x77\x13\xDD\x5A\x44\x77\xD2\x17\xDD\x89\x17\xDD\x21\x8A\xEE\xD4\x8B\xEE\x4C\x44\x77\x4A\x82\x6A\xB1\xA8\x1B\x08\xEE\x19\x96\x30\x34\xC8\xBB\x68\x99\x0B\xD9\x62\x28\xE7\x6E\xC4\x16\x3D\x39\xC7\xCD\x1E\x4A\xF2\x3F\xA5\x1E\xFF\xE1\x48\xDD\x36\x23\xC9\xBD\xE8\xB6\x6D\x7F\x84\x45\xAA\x90\x93\x94\x92\x82\x0D\xA6\x4A\x70\x15\x04\x8B\x8B\x4D\x1F\xAD\xD6\xE7\xDB\xB4\x10\xE2\xD8\x50\x14\x7E\x86\xAC\x94\x2F\x90\x99\x05\x4B\xC6\x0C\x8A\xCA\xB2\x2D\x84\xB4\x53\x19\xBA\xB9\xA7\x03\xD9\x49\x52\x55\xC3\xFF\xC3\xDE\xBB\x07\x5B\x76\x9D\xF5\x81\xEB\xB1\x9F\x67\xEF\x7D\xEE\xEE\xEE\x2B\xBB\xC3\xED\x4A\xD6\xD9\xA3\xC9\x5C\x55\xA9\xCB\xFA\x43\x75\x5B\x23\x67\xE2\x5E\x2A\xF7\xED\x87\x64\x49\x35\x43\xD5\x68\x6A\xA8\x1A\xFD\xE1\x3F\x5C\xFB\xCA\x13\xDF\xAB\x2B\x95\x92\x32\x7D\x1B\x68\x19\x07\xC2\xC4\x63\x0C\x81\xE0\x60\x41\x9C\x92\x1F\x22\x78\x82\x99\x31\xE1\x61\x85\x31\x55\x22\x81\x29\xA7\x06\x03\x13\x08\x60\x12\x88\x19\x48\x10\x13\x66\x30\x36\xA6\xA7\xBE\xDF\xF7\xAD\xB5\xF7\x3E\xE7\xDC\xEE\x36\x7E\x91\xA4\xDB\x65\xDD\x73\xCE\xDE\x7B\xED\xF5\xF8\xD6\xB7\xBE\xE7\xEF\xCB\x7A\x57\xDE\x82\x7D\x02\x08\x7D\x2E\xD9\x40\x03\xEF\xDC\x08\xE1\x3C\x30\xF0\x13\x41\x9E\xF0\x47\xF4\x41\x78\x67\xB3\x8E\x77\xB6\x74\x83\x72\xCD\xC0\xE5\xA4\x70\x64\x42\x2D\x9F\xE4\x96\x4F\x79\xAE\xC6\x07\x6F\xD4\x62\xCE\x4A\x78\xD2\x9D\xA2\x56\x37\xDD\xA9\xD0\xEA\xA9\xD8\x6A\xC3\x8E\xBE\xCD\xA1\xD5\xE4\x18\xDE\xD9\xBA\x0D\x77\x42\x78\xA3\xBC\x99\x53\x9D\x88\xA3\x39\xE5\x1A\x77\xCA\xCD\xE5\x7A\xC2\x79\x29\x5E\x2F\x4E\x4D\xE6\x35\x99\x6C\x89\x53\x5E\xEC\xCB\x27\x5D\x03\xDE\xD9\x4C\x79\x67\x8B\x20\x93\xC0\x3B\x4F\x35\xDA\xAB\x8D\x91\x14\x98\xAF\x48\x81\x39\xF1\x91\x22\xF2\x91\x5C\xA4\xC0\x22\xB2\x92\x3C\xA4\xA2\xE6\x23\x52\xC8\x1B\xD9\xCF\xD2\xC9\x7C\xC4\x4C\x92\xB5\xCC\xC4\x90\x9C\x12\x67\xB9\x0A\xB3\x8C\x20\xCF\xD2\xE5\x61\x96\xF3\x38\xCB\x5C\x5F\xD5\x09\xBC\xDC\xA2\x5C\x66\x26\xA5\xCB\x5D\x25\xCC\x24\x17\x66\xC2\x61\x1C\x43\xA7\xC6\x33\x57\xC9\xC9\x89\xB5\x6F\x62\x04\x6F\x64\x26\xC6\x55\xD4\xCA\xC0\x0D\xF3\x70\xC0\x58\x1C\x30\xA0\xCB\x90\xB3\x9C\xDF\x96\xE8\x67\x98\x61\x26\x41\x40\x32\x5C\x86\xAA\x80\xA2\x4D\x4C\x80\x74\xF9\x7C\xB0\x94\x1D\xA3\x9E\x26\xB8\x55\xD0\x78\x07\xD1\xCF\x10\xD3\x60\x93\x1D\x18\x86\xD9\xA8\x78\xEE\x0D\xC7\x49\x68\x09\xAA\x74\x1E\xC8\x0C\x71\x18\xB0\x9F\x2D\x18\x97\x09\xAC\x10\xF0\x4A\x15\xB1\x87\xF1\xA8\x2A\x66\x08\xBA\xE7\x1A\xDF\xCB\xC3\x0B\xCB\x07\x6D\x26\xE7\x61\xE5\x32\xAC\x9C\x83\x4F\x24\x8F\x98\x87\xA5\x57\x86\x95\xE3\xD6\xE9\xB0\xA2\x76\xB5\x21\x02\xDF\xD7\xE2\xDD\x81\xEB\xFE\xBF\x20\x8D\xDF\x8E\x26\x8C\xD0\x99\xCD\xCE\x8E\xD9\xAD\x18\x36\xE0\x25\x56\x02\x18\xCD\x33\x6C\x30\xB5\xE0\xBC\x36\xB8\xE5\x2D\xE0\x6D\xA4\x30\xB5\x63\x91\xB9\x33\xC7\x59\x0D\x18\x77\x36\x66\x0E\x90\xD0\x20\xC4\x3D\xB9\xB2\xCA\x6A\xF3\xDB\xE3\xB3\xCD\xC0\x67\xEB\x35\x32\xEA\x2C\xC8\xA8\x03\x9F\x2D\xD7\xCB\xA8\xB3\xD1\x5E\x1D\xF1\x59\xBB\x24\xA3\x42\x13\xB4\x41\x46\xB5\x37\x97\x51\x4B\x78\xCE\xC7\x32\xAA\x5D\xE1\xB3\xF5\x8A\x8C\x3A\xE5\xB3\x2C\xA3\x96\x23\x19\xD5\xB2\xB1\x4B\x64\xD4\x61\x42\xED\x8A\x8C\x6A\x59\x46\x2D\xC1\x67\xCB\x81\xCF\xD6\x2B\x32\xEA\xC9\x0B\x8D\x25\x19\x44\x22\xEC\x18\x1B\x82\xAB\xCA\x25\x72\xA0\x83\xA4\x13\x22\x88\x24\x6C\x40\x03\xBC\xD1\xF1\x96\x5B\x4B\xEC\x76\x1A\x12\xB5\xCC\x3F\xEC\xC0\x3F\x8E\x31\x1B\xAC\xE3\x1F\xA3\x8D\xB6\x2C\x74\x26\xC7\x0B\x9D\xC9\xF4\xB0\x48\xA6\x27\x85\x80\x16\xD8\x65\x17\x02\xA6\xB9\x88\xC7\xD9\x70\x52\xD8\x63\x4E\x0A\xAE\x58\xC2\x74\x92\x07\x3A\x49\x58\x43\x4A\x02\x9D\x24\xA3\x93\xC2\x8E\x34\xA4\x45\xBD\x7C\x52\xD4\x74\x78\xC9\x49\x91\xC8\x49\x91\x4F\x8E\xAF\xE9\x19\x9B\x87\x33\xD6\x1E\x77\x52\xE4\x72\x52\xC8\x51\x97\x84\x93\x22\x1D\x4E\x0A\x2B\xC5\xEE\x93\xDB\x90\x34\x4D\x88\xA5\xB1\x2C\x69\xA6\xF0\x0C\xE2\xC0\xB0\xBC\xE0\x56\x16\xDC\x86\xB8\x1B\xC5\xE0\x84\xB4\xE4\x17\xE5\x89\x41\xD2\x8C\x2B\x6F\xF1\xCC\xD2\xCA\xA7\x80\xBC\x34\xE1\x48\xC0\xC1\xC1\x47\x44\x3C\x2D\xBE\x06\x74\x18\x98\xEE\x1F\x61\x96\x7F\xB7\xFA\x8F\xDA\x6E\x1C\xFD\xF2\x25\xC2\x89\xA4\x24\x4F\x70\x8F\x56\xDB\xE6\x74\x37\x6B\x94\xFF\xC9\xA1\x76\xB1\xA3\x1F\x17\x33\xCE\xF9\x13\x03\xC7\x8C\xF5\x9C\x84\xAE\x9D\x71\xB6\x7D\x2F\xC8\xDF\x72\x21\x80\xC4\xD9\xF6\xB7\xAE\x07\x86\x27\x3D\x90\x48\xF7\xD2\xB1\x76\xF0\xB5\xB2\x56\x67\x77\xAC\xD5\x5F\x11\x6B\x75\xBA\x6C\xAD\x4E\xC6\xD6\xEA\xF4\xCF\xA5\xB5\x3A\x58\xAA\x9D\x09\xE7\xC7\x1D\x73\xF5\x7F\x9A\xE6\xEA\x3F\xC6\x10\xFF\x51\xFD\x9F\x88\xB9\x3A\x9E\x01\x05\x03\xC1\xD8\x85\x19\x9D\x01\x33\x3A\x03\xCA\xA5\x33\x80\x7E\x24\x26\x52\x39\xBB\x27\x50\xA2\xB2\xEE\x96\xAE\x9D\x71\x46\xCE\x00\xC3\x51\x90\xD6\x19\x39\x03\x18\x0B\x0F\x3D\xA0\x8F\x7F\x2E\x8C\xE4\xD9\x1D\x23\xF9\x57\xC4\x48\x9E\x2E\x1B\xC9\x93\xB1\x91\x3C\xFD\x0F\xD9\x48\x5E\xDD\xB1\x92\xDF\xB1\x92\x1F\x6B\x25\xFF\x3C\x46\xFC\xDD\x95\x6E\x27\xAA\x43\x7F\x53\x33\x39\x26\x28\x19\x22\xD7\x44\x85\x35\x21\xA7\x21\x09\xDC\xB9\x18\x71\xE7\x92\xB8\x73\xB1\xC4\x9D\x4B\x81\xB7\x05\xCE\xC6\x4C\xB8\x8F\x70\xE7\x92\xB8\xF3\x4C\xB8\x73\xC1\xA8\xA5\xD6\xCD\x84\x3B\x33\x47\x60\x8D\x12\xC5\x37\x8E\xFA\xCE\xDE\x82\x29\x8F\x2C\xF0\x4B\x4C\x79\x8D\xF5\x3D\x63\xA6\x9C\xDD\x8E\x49\xA8\x5A\xE1\xC8\xA1\xA4\xD5\x3C\x52\xF9\x3C\x50\x79\x0D\x88\x00\x57\x07\x2A\xAF\x47\x46\x72\x50\x79\xB6\xC2\x91\x61\x7A\xDF\xE0\x96\xDB\x90\x52\x33\x35\x0A\xB7\xD4\xEA\x09\xD7\x86\x56\xDB\xD1\xDE\x81\xEE\x7B\xE2\xD6\xA6\xF7\xC6\xD5\xD1\xB4\x3E\xE5\xC8\x1B\xB8\x9E\xB9\x36\x98\x8F\x47\xA6\xF7\x76\x32\xA1\x53\xB3\x40\x1B\xCC\x02\x1B\x2E\x03\x47\xCE\xA6\x1C\x19\x0E\x90\x66\xE0\xC8\x99\x6B\x97\xCD\xEF\x76\x45\xAA\xB5\x53\x8B\x8A\x5D\x31\xBF\xDB\x60\x7E\xB7\x23\x1A\xB0\x4B\xE6\x77\x7B\x4B\xF3\x7B\x1E\xCA\xBE\x62\xA6\xCB\x30\xD3\xD4\x68\x37\x73\x36\xCC\xB4\x8D\x33\xCD\xF9\x4F\x6E\x26\x5C\x6A\xB6\xCC\xA5\x48\xBA\x28\x85\x4B\x59\xE1\x52\xE5\xC4\xD2\x33\x35\xA8\x95\xC1\xA0\x26\xE6\x77\xC4\x12\xE6\x03\x97\xCA\x5D\x29\x5B\x60\x6C\x3D\xB3\x23\x3B\x55\xB0\xCC\xE0\x96\xDB\x11\x65\xF3\x91\xC9\x42\xE3\xFB\xCD\xCD\x16\x6A\x22\xCA\xAA\x5B\x98\xDF\x73\x56\x59\xA2\xF9\x3D\xBF\x85\xF9\xDD\x0C\x46\xC0\x5B\x9A\xFC\x82\x31\x48\xEA\xFD\xF4\x30\x04\x45\x0B\xD0\xD8\xFE\xA3\x26\xBA\x97\x5A\x32\xFC\x48\x3A\x40\x34\xB5\x14\xD1\xE4\x77\xDC\x4B\x0B\x7E\x69\xC1\x2F\x1D\x49\xC6\xFC\xD2\xE2\x66\x2F\x15\x81\x78\xF9\xA5\x76\x62\xDF\xF9\x02\x16\xFC\x47\x23\x93\xBE\xA9\x51\x7D\x6A\xF3\x21\xC5\x71\xC9\x7D\x99\x04\x36\x6D\x99\x4D\x9B\xDB\x67\xD3\x0C\x44\x38\x66\xD3\x66\x2D\x9B\x36\x2B\x6C\x5A\x66\x07\x6F\x4D\x97\x64\x8A\x55\x46\x6D\x03\xA3\xB6\xCB\x8C\xDA\x1E\xC3\xA8\xCD\x57\x80\x51\x37\xCC\xA8\x9B\xB0\xD1\x9B\x91\x90\x7B\x0C\xA3\xB6\xEB\x18\xB5\x0D\x8C\xDA\xDE\x8A\x51\xDB\x25\x46\xBD\x6A\xBB\x2F\x45\x34\x6E\x6E\x9B\x51\x0F\xB6\xFB\x76\x32\xA1\x76\x85\x51\xDB\x9B\x30\x6A\x76\x9D\x04\x46\x6D\xD6\x30\x6A\x11\x2C\xCD\xF1\x82\xA5\x99\xF2\x6D\x33\x65\xDA\x26\x58\xC2\xCD\x88\x1E\xCC\x92\x25\xDC\xDC\xD2\x12\x9E\x8B\x52\xC8\xB3\x5E\x86\x59\x37\xCC\xB4\x4D\x98\x75\x33\x62\xDA\xF6\x16\x4C\xDB\x44\xA6\x6D\x46\x4C\xDB\x8C\x3A\xB5\xCC\xB4\xCD\xC8\x12\x7E\x3C\xD3\x36\x3C\x00\x13\x98\x76\x32\xB5\x84\x03\xFB\xFD\x36\xA4\xC9\x3C\x66\x33\xB3\x34\x99\x04\x4B\x78\xBE\xDE\x12\x6E\x26\x9C\xF0\xA2\x3C\x31\x48\x93\xCB\x0C\x71\x6A\x7B\xE6\xE2\x4F\xF9\x60\x09\x27\x1E\x6E\x99\x87\xDB\x75\x3C\x3C\x89\xBE\xC9\xDB\x74\xDB\x18\x66\xA7\x50\xB8\x19\xAF\x85\xFA\x61\xC2\xB9\x63\x6E\x66\x3F\x33\xB8\xD5\x29\xEA\xB2\xBA\x4D\x1E\x6E\x97\x79\xB8\x9A\xF0\x70\x3B\xF0\xF0\x63\x5E\x2A\x3C\x7C\xF9\xA5\x62\xA3\xFF\x9E\x04\xF0\x44\x31\x95\x9F\x33\x2F\xD8\x49\x12\xBD\x47\x36\x6E\x98\x2E\x7D\x93\x90\xD3\x8E\xD9\x0C\xAE\xA6\x48\xF3\xFA\xAC\x3A\xC5\xEC\xCE\x92\x8E\x6A\x44\xF4\x31\x7E\xB6\x0B\x74\xEC\xA5\xB4\x0C\x4D\x3F\x21\x41\x3C\x26\x65\x94\x6F\x6A\x34\x4A\x6D\xBB\x94\x1E\x49\xE2\xE3\x66\x7D\x56\xC7\xAF\x33\x10\x5B\x7C\xFC\x61\xDA\x01\xD7\x88\x0A\x92\x85\xA9\x2D\x13\x6B\x86\xA4\x2D\xAF\x17\xD6\x1B\xEC\x3E\x9F\xB2\x33\x87\xB9\x38\x8A\x51\xB6\x5D\x7A\x80\x9D\xFD\x09\xD5\x0B\x3E\x75\x23\x09\x15\xCB\x09\x57\x4E\x70\x4F\x95\x2F\x7C\xED\xD2\x2D\x4E\x6D\x8B\x99\x4A\xFC\x14\x92\xDB\x7E\x2B\x70\x85\xE9\xAE\x95\xE6\xF5\xED\x34\x4F\x5D\x9B\xBC\x61\x7D\xFB\xC3\x31\xC2\x88\x1B\x4A\x96\x78\x93\x46\x2D\x87\xE7\xB6\x39\x8D\x6C\x84\xB3\xAA\xC5\x9C\xF1\x22\x6F\x1B\x52\x68\x2D\x6B\x91\x96\x1B\x3A\x51\x39\x5B\x7D\xDA\x68\x13\x72\x35\xEE\xDB\xEB\xCA\x58\x37\x61\xE6\x72\xC1\xF6\xE2\x02\x0A\x28\x38\x92\x21\xE1\x89\x8D\x61\x33\x4E\x07\x52\xAE\xA4\x3F\xF7\xBA\xD2\x29\x7F\xA6\xDF\x31\xDB\xAE\xE4\x52\x06\x28\xA3\xE3\xF8\xF2\x19\x57\xD2\x06\xB8\xDB\x95\x4E\x4B\x2D\xEB\x52\xEA\x46\x17\xA4\x52\x3D\xA8\x5A\x57\xBA\xE4\x41\x75\xD2\x95\x2E\x7D\x50\x9D\x72\xA5\x2B\xFD\x36\x83\x88\x38\xE5\xCA\xF6\x4F\x69\x16\xC6\x5D\x60\x33\xDA\xB6\xD9\x04\x7A\xBF\x51\x74\x74\x0E\x87\x00\x2A\x1A\xA6\x4F\x6F\xF5\x8C\xFB\xC3\xA6\x04\x57\x9E\x55\x27\xBD\x3E\xA4\xBF\x27\xBC\x39\x3C\xA0\x0F\xA7\x0E\x40\x54\xF6\xB0\xD3\xFE\x3D\xBF\x73\xFD\x1A\x70\xD4\xBC\x7E\xA6\xA7\xBE\xFA\x77\x85\x5F\x4E\x7C\x13\x52\x51\x16\x38\x00\xCF\xAA\x33\x93\xE9\x70\x39\xF5\xC4\x21\x6B\x63\x31\xAB\x5C\xE9\xEF\x0B\xE9\x30\xB3\xEA\x63\x46\xEB\xA3\x95\xD2\x14\xC3\x76\x8A\xE9\x73\xC6\xA9\x08\x04\xFD\xAA\xF1\x47\xBB\x0C\xFB\x0C\xAC\xC2\x33\xF8\x04\x54\x15\x7C\x3A\xCD\x40\xE6\xFE\x55\xC3\xB8\xC6\xFE\x7D\xD4\x68\xFB\xC3\xDF\x22\x80\xEE\x9F\x51\xFD\x02\xC5\x8C\xBE\xA4\x46\x5F\x58\xD7\xA8\x9E\x36\xFA\x87\xB7\xD9\xE8\x1F\x86\x46\x7F\x60\xDA\xE8\xEF\x51\xA3\x92\xEC\xA6\xC6\xA5\x21\x54\xF5\x5F\x6A\x7D\xC4\x9B\x24\x11\x90\xF8\x05\x75\x22\x20\xC0\xA1\x6E\x84\xBA\xC7\x3E\xF9\x90\x39\x6A\xFF\x15\xF3\x41\x40\xB3\xBD\xA5\xDA\xB9\xE5\x93\x16\x4F\x2E\x3D\x76\xD1\x98\x23\x7D\x75\xFA\x60\xA7\xCE\x2A\xB3\xD0\xC3\xD3\x06\x05\x2A\xE8\xE9\xCE\x3A\xFB\x90\x39\x72\x7A\x6B\x68\xC7\xEC\x98\xA7\xAA\xD3\x6A\x00\xD8\xFA\xD7\x74\x45\x60\xB6\xAA\xCF\x5A\xAB\x01\xA6\x33\xA4\x47\x15\x32\x5C\xD9\x4D\x05\xB6\x45\x57\x9E\xB3\x05\x2A\x8C\xA3\xB0\x67\x81\x5D\x83\x1F\x5D\xF9\x62\x08\x95\x50\x67\x95\xEE\x2A\xFA\x43\xD2\x20\x75\xB3\x6B\xB6\x88\x98\x59\xF6\x63\x33\xBC\x9B\x5D\xE4\x93\xBE\x76\x95\x9B\x5E\x9E\xB9\xF2\x21\xF7\xCE\x17\xFB\x91\xE8\x74\x51\xCC\x56\x9C\x7A\xBC\x8B\xE2\xC9\xB4\x3B\xD4\x45\xD2\x91\x76\x0F\x44\x42\x8A\x2A\xEB\xD5\x5E\xAA\x61\xA1\x60\x36\x49\x63\x8B\x9A\xA8\x2E\xBF\xB0\x98\x79\x0B\xDB\x7F\xCD\x5C\xB9\xF6\x7A\x91\x30\xDF\x48\xD9\x11\xED\xEA\x2D\x80\x7B\xEB\xA7\x69\xCB\x28\x97\xB4\xD7\xB8\xE8\x0C\x97\x6A\xA2\xFE\xD1\x98\x2E\x20\xD8\x26\xE9\x5D\xDE\xDE\x60\xAD\xDF\x9C\x77\xD9\x59\x05\xB9\xBB\x68\xFF\x88\x9E\x91\x3C\x31\xCF\x16\x74\x8E\x5A\x63\x61\xA5\x0E\x83\x2D\xB6\xB9\x86\x90\x40\xDA\x25\x68\x7B\x2B\x0E\x18\x98\x31\x52\x09\x57\x92\x6C\x93\x85\x40\xE2\x69\xD2\x16\x00\xB1\xC0\x90\x00\xDA\xA9\xDE\x65\x7B\x02\xE6\xCA\x40\x4E\x5D\xA8\x41\x84\xA6\xE9\x5A\x85\x7A\x3F\xB2\xB8\xA6\xFA\x13\xCD\x08\x01\x7A\x6D\x56\x76\xC0\x04\x30\x21\xF1\xDC\x6E\x01\xD5\x6C\x35\x1B\x9E\xA1\x6A\x59\x4E\xA6\xBB\xE9\xC6\x98\x0F\xAF\x57\xF3\xE1\xED\x90\x0F\x0F\x2C\x94\x51\x7E\xFE\x6A\x83\x9C\x1E\x2F\xED\x5E\x6A\x94\xFF\xA9\xE5\xCC\xF7\xE0\xD4\xA9\x46\x2F\x44\x0E\xA0\xE3\x2B\x03\x76\x19\xCB\xDC\x5C\x5B\xAB\xFA\x70\xA6\xE7\x01\x5F\x75\x5A\x86\xC4\xFC\x47\x5F\x86\x24\xBB\x00\x70\x1C\x94\x21\xC9\x88\xAC\x04\x98\x23\x96\x21\xC9\xA5\x0C\x49\xCE\x27\x09\xAA\x89\xE4\x4E\xBF\x0D\xE8\xEA\x1C\xDC\x19\xCB\x90\x64\x52\xFC\x4B\x73\x89\x8E\x92\x44\xD4\x87\xE1\xDB\x2B\x80\xB8\xC8\x01\x06\x6C\xBB\x8C\x20\x42\x9A\x41\x84\xF4\xDB\x82\x55\xCA\xBA\x1C\xB0\x61\x52\x73\xC4\x84\x0C\x6F\x48\xE9\x5D\xD5\xA3\x3E\x01\xF6\x17\x5B\xAD\x68\x5F\x5B\x57\xF0\xDE\x36\x9C\xE9\x5E\xB0\x62\x0F\x84\xB5\x45\xCD\x42\x6A\x01\x31\xA4\x00\xA4\xF2\xED\x60\x12\x31\x86\xF2\xAE\x48\x28\x11\x95\x08\x9A\x44\x13\x51\x89\x1A\x11\xC6\xB1\x41\x8E\x43\x25\xE2\x7A\x02\xC4\x64\x10\xAB\xD7\xB8\xA4\x4B\x77\x01\x9C\xAB\xFB\x05\x63\xB9\x2C\xE6\x8C\x60\xB3\x31\x7D\xC6\x6D\xB8\xF9\xAE\xE0\xCE\x35\x40\x25\xAA\x11\xC2\x27\x43\xAF\xF8\xFB\x02\x25\xEC\xB0\x06\x35\xCF\x55\xD5\x2F\x15\x2E\xC1\x0F\x80\xFF\xAF\x64\x9D\x18\x32\xDF\xDD\x4E\xF5\x95\x6A\x00\x13\xFD\x4D\xAD\xCD\x91\x53\x28\x0D\xE5\x5F\x51\x0C\x5B\x06\x57\x02\x6F\xAC\x87\xD4\x39\x9C\xA8\xEA\xF5\xBA\xE6\xA7\x8B\x08\xBA\x9F\xB0\x40\xDF\xB6\x7F\x93\xCB\x4E\xF9\x7B\x7B\x11\x0B\xE5\xA7\xB0\xCB\x05\x3A\x36\x60\xC9\x7A\x7D\xB8\x84\x22\xAB\x43\xC9\x69\xA4\xB6\x86\x1D\x8A\x5D\x7E\x1E\x9D\xB1\x3B\xBC\x80\x17\x26\xF5\x8C\xA8\x73\x4F\x72\x3E\x6E\x00\xB4\xFD\x8C\x92\x0A\xEB\x23\xC0\xD4\x2F\x18\x9D\x46\xC0\x94\x35\x58\xA4\x6A\x84\x45\x1A\x98\xA3\x0D\xCC\x31\x59\x0F\x98\x62\x25\xBF\x3B\x98\x41\x94\x4B\x80\x96\xFB\xE8\xB1\x65\x8F\x52\x17\x47\xA9\xC7\xA3\x64\x27\x34\x03\x6C\xA1\x90\x83\x98\x04\x02\xAE\x5A\x86\xDC\x74\x1B\xBE\x4B\x8D\xB2\x65\x9C\xD3\xD0\x1D\x46\x22\xAE\xA5\x3E\xBF\xD4\x32\x9E\x94\x32\x10\xCC\x59\x3D\xE1\xC2\xFA\x11\x56\x29\x90\xCB\x2B\xE0\x24\x6B\x27\xAA\x1D\xB8\xB2\x58\x9C\xF9\xD5\xCB\xB3\x34\x22\xB1\xBF\x9D\x44\xBC\xDA\x37\x4B\xFA\xF2\x60\x6B\x51\xFE\x93\x20\x03\x4E\xB0\xC6\x02\xB1\x80\xA7\xFC\x2F\x2B\x59\x89\xFC\x59\xFF\x27\x37\x6E\x6C\xEC\xF7\x82\x13\xE6\x6F\xD8\x7D\x06\x45\xC4\x7D\xBF\xCA\x0D\xD8\x7E\x7C\xEB\xF4\x46\xBB\x3B\x94\x9B\x4D\xBA\x49\xAC\x9E\x39\x1F\x90\x0B\xA9\xF1\x3D\x7F\xCD\x05\x3B\xE7\x42\x0B\x98\x00\x74\x7F\xC9\xA7\xF7\xAF\xA8\xBE\xFD\xF6\xEB\x01\x5D\xFB\x7C\x40\x39\x3D\xBF\x8C\x72\xCA\x03\xBE\xF7\x18\x00\x98\x7B\x23\x00\x0C\xE7\xA0\x7B\x35\x64\x92\xAF\x23\xB8\x07\x16\xBA\x1A\x72\xC1\x97\xE0\x54\xFD\x7D\x3C\x33\xE8\xB0\x6C\xFB\x35\x7D\x39\x7D\x4C\x5F\x4E\x0F\x7D\xD9\x8E\x7D\xB9\xFB\xF8\xBE\xDC\x1D\xFB\x72\xF7\x9A\xBE\x9C\xB9\x45\x5F\x9C\xAA\x3E\xFD\x15\xAB\xB3\x76\xCD\xAC\x2B\xB4\x76\x2D\xF9\xAA\x17\x5A\xFB\x77\x27\x74\x79\x14\x76\x34\xD1\x49\x07\x0D\x37\x71\x39\x63\xF5\x80\x8E\x60\xA9\x5A\x0B\xAC\x52\x92\x1A\x09\x82\x9D\xD1\xD3\x6F\x5C\x41\xF1\xC9\x83\xCF\x93\xA8\x9B\x8E\xB0\x85\x20\xA6\x31\xF6\x4F\xC5\x02\x66\xD6\x28\xFF\x63\x63\x38\xBB\x8C\xCF\x8A\x9A\x9E\x6D\x78\x71\x6A\x18\x70\xCC\x79\xB8\xA6\xFC\xFF\xF7\xE1\xC9\xDD\x6A\xDB\x3C\xC9\xF2\x55\xB6\x63\x9E\xAC\x5C\xB5\xC8\x82\x5C\x22\x68\x4B\x5D\x73\x0C\xEC\x4F\x1D\x60\x7F\xAA\x35\x58\x4B\x80\xFD\x01\x74\x49\xBD\x90\x3A\xCA\x74\x7F\xC5\xA8\x53\x2C\x09\xCF\x05\x7B\x70\x84\xBC\xB4\xC1\xE2\xEF\x9C\xCE\x89\xCA\x6D\xD0\xE0\x23\xF6\x92\x39\xEF\xE6\x6B\x61\xA6\x87\x17\x49\xBB\xB8\x55\xDE\xB7\x06\x69\x7A\xD4\x31\x86\xBF\x92\x44\xA3\xD0\xC1\x37\xF1\x6C\x2D\xC1\x4C\x2D\x8F\x28\xDE\x7E\xAB\xA1\x8F\xDB\x5D\x1A\x2E\x63\xBB\xF0\x60\xEB\x38\x58\x12\xAC\xDF\xFD\x81\xC9\x84\x7F\x59\xD6\xE0\x26\x1D\x99\x8F\x3A\x32\x3F\xAE\x23\xAB\x0D\xF2\x62\x49\xBB\xCB\x12\xFE\x74\x49\x86\x17\x76\x1B\x3D\x8A\x03\x33\xE6\x14\xD1\xAA\xEE\x3B\xFC\x20\x93\x10\xA4\xF5\xDA\x5F\x5D\xB3\xDA\xCB\xCB\xAC\x19\x60\x6B\x65\xA5\xBF\x6C\xD3\xB6\x6E\xE2\x36\x46\xB3\xD6\xD5\xCB\xF3\x06\x58\x94\xEF\x9C\x4E\x9C\x3C\xB3\xE0\x67\x86\xF6\xC3\x83\x01\xDA\xBC\x3E\x06\x49\xBD\xAE\x78\x70\x8B\xB9\xAB\xC3\x98\x35\x1F\xCB\xBA\x72\xF3\xAF\xC2\x98\x6F\x41\xB5\xEB\x07\x1D\x51\xC8\xC6\x03\xC6\x43\x74\xF0\x36\x63\x53\xB0\x19\xC3\x79\x6D\x9B\xC7\x21\x13\x09\x36\x54\x1D\xC0\xBC\x94\x35\x59\xA2\x67\xD5\xF1\xBC\x4F\xEA\x2F\x37\xAC\x57\x54\x01\xC5\xA9\x18\xC3\x6E\xD5\x05\x75\xA2\xD8\x6F\x32\xF0\x59\x7A\x4D\xCD\xDC\xBB\x86\x95\x9E\x9E\x44\xC0\xE3\x8C\x23\xA5\x6A\xC6\xE6\xBA\xC5\xBD\x11\x14\x8A\x84\x11\xFF\xED\xE3\x0A\xFC\x15\xC2\x24\x70\xEC\x3C\xDA\x24\xB1\x21\x28\x3F\x89\xC0\x73\x85\xB1\x3F\xC9\x67\x8D\x92\xF2\x64\x4F\xD2\x5B\x44\x2B\x28\x02\xD2\x21\xE6\x47\x6D\x9B\x47\xBA\x66\xAF\x33\x3E\x3F\xE4\x72\xC4\x66\x8B\x7E\xFC\x06\xFA\xCF\x13\x0C\x12\xEF\x8A\xF6\x3B\x42\xC9\xD8\x27\x70\xB1\x67\xBD\xF8\x91\x45\x53\xB9\xC6\x99\xA9\x40\x65\x82\x40\x25\x40\x38\x78\xA2\xE3\x61\xA2\xD9\x47\xD8\xD0\x4C\xAD\x1F\xF3\x64\x27\xDD\x35\xFE\x88\x8F\xD9\xC6\x9B\xC3\x35\x2E\x47\x18\x35\x64\xE2\xE8\x7C\x83\x36\x0B\x8C\xC6\x20\x17\xE3\x57\x40\xDF\x72\xD5\x46\x83\xEC\x6E\xC4\xCC\x8C\x71\xC1\xE6\x11\x17\x6C\x3E\xC2\x05\x5B\x47\x20\x80\x0A\xAB\xD7\xD1\xC9\xCD\xD0\xC2\x5C\xFD\x3A\x6D\x68\x14\x97\x9B\xEC\x66\xC8\x5F\x74\x1B\x4D\xAA\x71\x8D\x60\x7F\x3D\xDB\xC3\xAE\x50\xD5\xF9\xED\xD3\x2B\xEF\x75\xA6\xB8\xA4\xB6\x7F\x06\xEA\x5B\x45\x89\x23\xEA\x73\x89\x4B\xDB\x5F\x22\x52\xF0\x47\x4F\xEF\x37\xB6\x1A\xBD\x2A\x79\x56\xFA\x19\xA2\x42\x8D\x80\xEC\x63\x81\x1E\xFD\xA2\xCA\x63\x5E\x4B\x42\x79\xCC\x6B\x6F\xBF\x53\x1E\xF3\x4E\x85\xC2\x2F\x53\x79\xCC\x6B\x6E\xA5\x00\xE6\x77\x5D\x8F\xE5\x31\xD7\x5C\xFD\xEE\xEB\x77\xCA\x63\xDE\x21\xBE\xAF\x5E\x79\xCC\x81\x04\xC7\xE5\x31\xFF\xCE\xF5\xDB\x2F\x8F\xF9\x3D\xD7\xFF\xC3\x28\x8F\xF9\x6B\xA5\xAE\x8E\x38\xA2\x78\x90\x5B\x02\x9C\x68\x8A\x52\xCF\x2C\x68\x0A\x34\x1A\x6A\x87\xC4\x74\x32\x75\x56\xBD\x65\x11\xAD\xC4\x67\x55\xCD\x39\xCC\x95\xB8\x92\x60\x43\x82\xA3\x89\xEB\x0F\x00\xCF\x19\x79\xC5\x1D\xA2\x36\x2A\x18\x8B\x49\xA3\xB5\xAE\x82\xDF\x7F\x59\x6C\xEC\x1A\x24\xA6\x74\x90\xAA\xBB\x0D\xFA\xD8\x8A\xF4\x98\x2A\x63\xAD\xB6\x24\x42\x14\x17\x1B\x13\xA2\xA7\x48\x68\x76\xED\x52\x2D\x96\x36\xD4\x62\x31\x5A\x6B\xA5\xAB\xD5\x5B\x58\x08\x80\x44\xCB\xE1\x44\x16\xEB\xF4\x66\x74\x9D\xA9\x6E\x90\x65\x50\xED\xEA\x2D\x61\x28\x65\x18\x0A\xE7\x6E\xB0\x7A\x5E\xAD\xA8\xE7\xB3\x91\x7A\x3E\x23\xF5\x9C\x61\x62\xE1\x62\x13\x58\x50\x7B\xD8\xEF\x22\xCA\xA3\x0E\xA1\xD5\xE5\x90\xB9\xB7\x76\xDE\x6E\xF3\x55\x58\x35\xC1\x43\x5D\x94\x6E\x36\x7D\x27\xEF\x49\x3D\xF8\x3A\x16\x33\x73\xDE\x6D\x44\x18\x57\x4D\x32\xC8\xCF\x8F\x94\x01\xBA\x14\x20\xC1\x67\xB4\x4A\x33\xA0\xB5\x76\x99\xE8\x0F\x9D\x8A\xB5\x6F\x66\xB3\x99\x9A\xA1\x47\x6B\x6B\xDF\x94\x9C\xF5\x94\xC1\xF1\xF8\xCA\x7B\x46\x63\x68\x82\x0A\x96\x6C\x08\x7C\xFC\x2D\xD7\x83\xAE\x87\x72\xE5\x4C\x8F\x0D\xF3\x76\xB5\x6D\x36\x2F\x72\xBC\x7C\x7D\x21\x16\x54\xB8\x81\xA8\xC4\x19\x5B\x63\xE8\x59\x0E\xFE\x42\x5E\x3C\xEB\x29\x6C\x03\x85\xA3\xA3\x25\x41\xA8\x76\xED\xC5\xC6\x3A\x4E\x35\x7D\x13\x91\x48\xE1\xAA\x9B\x74\x86\x76\x29\x75\xE6\xB6\x7A\x8E\x9B\x57\x96\x99\x43\x19\xBF\x18\x9A\x2A\xD7\xD2\x54\xF9\xB5\xA0\x29\xC9\x09\xC2\xE1\xD9\xFA\xE4\x59\x67\x11\x0C\x3F\x6B\xFF\xD6\x75\x69\x73\xDD\xFB\x05\xB5\xE3\xD6\xAF\x2F\x7A\xCE\x29\xAC\x39\x5D\x64\x4C\xCB\x90\xBF\x2B\x97\x55\xFF\xBE\xB2\xE6\x48\x5F\x4D\x47\x5E\xF5\x7C\x90\x76\xC1\x9E\xE1\x05\xE6\x74\x8F\xB1\x83\x7D\x70\xB1\xE7\x13\x17\x7B\x67\x42\xDE\x6F\x11\x42\xA6\x66\x97\x1B\xF9\x21\x84\x4C\x55\xD1\x83\xDE\xD5\x0F\x37\xD1\xB3\xD2\x9C\x55\xAA\x9B\xF3\xD4\x1B\x71\xBE\x37\x82\x1E\x5C\xBB\x8A\x13\xDB\x12\xA6\x8A\x86\x5B\xCC\xB8\x22\xCD\xDC\xAB\x8B\x48\x83\x4F\x1E\x54\xA8\x5F\xE8\xF5\x61\x57\xA0\x04\x01\x74\x11\x83\x08\x2F\x3A\xDE\x2A\x57\x3B\x7B\x40\xF7\xFA\x23\x77\x7C\xCA\x4A\xB1\x63\xDE\xEC\x0A\x4F\xA7\xC2\x0B\x81\x16\xD5\x8E\xB9\x97\xFF\x9C\xE6\x43\x79\x93\x67\xBB\xE5\x7A\x84\x6F\x11\x3B\x68\x03\xDF\xD1\x62\x06\x3F\x14\xE3\xC8\xC2\xB1\x36\x5B\x24\xFE\x08\xBC\xC3\x30\x76\xBF\x39\xAB\x14\x3B\xF0\x0B\x47\xA4\xCD\xB2\x46\xFB\x3F\x21\xF4\x76\x26\x99\x23\x44\x69\x2F\xBD\x34\x5A\xEA\x22\x32\xA2\x4D\xA0\xFA\xCE\xE4\x24\x29\xB8\x42\xD4\xC8\xA6\xC0\x2A\x54\xC1\x7E\xA2\x7C\xDF\x7F\x63\x3F\x4F\x54\x96\x02\x1E\x1C\x51\x73\xEC\xE0\xC1\xEE\x2A\xB8\x0A\x12\xC8\xC2\x90\x92\x34\x63\x68\x6D\x57\x3D\x8C\x0A\x14\x74\x94\x85\x9A\x17\x17\x42\xF0\x9B\xDA\x95\x68\xB4\x6D\xA3\xF0\x5B\x25\xC1\xBB\x9C\xCD\x27\x0B\xE9\x2A\x67\xF6\xDA\x77\x5D\x97\x08\x10\x54\x49\xDC\x0A\x01\xA3\xB0\x01\xF0\x10\x40\xED\x3B\xE6\x49\xD2\xC3\xA2\x0D\xA0\x5A\xB6\x01\x54\x7B\xDD\x2C\xDA\x00\x66\x13\x1B\xC0\xEC\x22\xD2\x11\x56\x6C\x00\x33\xB6\x01\xB0\xDC\x37\xD5\xE4\x67\x6B\x6D\x00\xB3\xB1\x0D\x60\x16\x5A\x3F\xE6\xC9\x4E\xBA\x3B\x0B\x36\x00\xD4\x51\xAB\x02\x49\x55\x53\x1B\x80\x30\xB9\x37\xA3\x6C\x51\x60\x73\xF8\xD5\xC8\x98\x93\x85\xF5\x47\x8B\x2A\x48\xE9\x9B\x91\x52\x2A\x46\x1E\xB6\x80\x60\xE6\x30\x0F\xE6\x14\x81\x51\x20\xF5\xD4\x4A\xAD\x2E\x66\x54\xF6\x36\x18\x15\xC3\x33\x9B\xE8\x90\xE4\x40\x78\xA6\x2F\x57\x2C\x66\x2E\xC1\xF6\xE7\xB7\x17\x5E\xF1\x38\x37\xB7\xD0\x0B\xAF\x98\x5E\xBF\x6E\xC0\x79\x9E\x92\x2B\xFB\x4D\xC0\xE5\x47\xCC\xBC\xC2\x6C\x24\x9C\xC5\x55\x0C\x2E\xF3\x3A\xC0\xFD\x8B\x18\xC5\xCE\x16\x66\x24\xF0\x11\x24\x62\x04\xE0\xE1\x25\x2B\xC3\xD3\xA3\xE1\x69\x0C\x8F\x84\xF7\x0E\x51\x2C\x9C\xDC\x38\xB0\x41\xA2\xE9\xCE\x10\xE7\x28\x78\xE8\xE3\x6A\x38\xD6\xF3\x1E\xE4\x62\x22\x5C\x01\x87\x8B\xEC\xD5\xA1\xC0\xE1\x9B\x9D\xE9\xBB\xC4\x3F\xD7\x33\x04\x4E\x4A\x5F\xB5\x47\x41\x35\x89\x25\x61\x30\xE1\xE7\x7A\x97\xB9\x64\x4F\xD6\x97\x9F\xE2\x47\xB4\x7F\x7B\x8F\x6C\x8B\xE1\xEE\x78\x2B\x29\x15\x0B\x03\x92\x35\x4C\xDB\x01\xB4\x98\x8F\x4B\x6F\x0F\x3B\xD5\x77\x32\x53\x08\xB5\x51\x7D\x17\x9C\xFD\x32\x03\xCE\x70\x40\x4D\x3E\x04\xD4\x14\xD5\x65\xAD\x8F\x82\x13\xCD\x2E\xE5\x0A\xC0\x83\xC6\xE5\x36\x42\xB6\x9C\x1A\xE5\xD9\xA2\xE2\xA9\x0E\x49\xB6\xD5\x8F\x68\x6D\xC3\xD1\x71\x4D\xC7\xB3\x83\xB5\x22\xD5\xBE\x13\x9C\x4C\xB9\xFC\x21\x75\xCE\x16\xA1\x66\x54\x50\x30\x76\xBB\x12\xA1\x52\x07\x44\x23\x17\x99\xA6\x53\x57\xBA\xEC\xC2\x7E\x67\xB7\x3C\xCD\x84\xDD\x6A\xFF\x36\xB5\xC1\x1E\x9A\xC4\xC3\xCA\x97\xF2\xAD\xD9\x16\x3B\x50\x13\x3A\x51\x56\x03\xB1\x8A\xCA\xA9\xF6\xDB\xAE\x0B\xF6\xD1\x35\x3D\x0C\xFF\x23\x76\x54\x71\xE4\x56\x15\x2D\xC4\x81\x9E\x06\x07\x7A\xB6\xDE\x81\x9E\x8A\x4A\x02\x59\xCC\x2B\x97\x7D\x69\x15\x47\xD6\x35\xB7\x36\xA6\x29\xDE\x18\x6F\x5B\xAD\x38\xC2\x18\x4D\x28\x6E\xE3\xD2\x91\xBF\x43\x8F\xFD\x1D\x37\xAF\x38\xF2\x25\xF5\x67\xB9\xE2\x48\xE8\x8F\x9A\xF4\x27\x56\x1C\xF9\xCE\xEB\xE2\xD8\x37\xCE\xB6\xEF\xC1\x97\xCF\x7F\x0D\x23\xC0\xEC\x28\x02\xEC\xDD\x5F\xA9\x08\x30\x39\x34\x83\x91\xD3\xB2\x3B\x00\x01\x0C\xD5\x6F\x9B\x2F\xBA\x40\xCE\x71\x51\x0C\xC7\x4F\x61\xA8\x93\x7C\xF3\xF9\x4B\x86\xE1\x4A\xE9\x99\xF5\x33\x98\xCA\x64\xF0\x1C\xC6\x35\xD6\x2B\x6E\x93\xC9\x63\x01\x4C\x2C\xBE\x25\x3C\x2B\xC6\x99\xE4\xD8\x30\x8E\x7A\x29\x8C\x43\xEA\xD0\xAC\x96\xDE\x1D\x1A\x9E\xBC\x99\x5E\x72\xA7\xB0\xFD\x9D\xC2\xF6\x77\x0A\xDB\xDF\x29\x6C\x7F\xFB\x85\xED\xEF\x30\x8C\x3B\x0C\xE3\x0E\xC3\xB8\xC3\x30\x6E\x9B\x61\xFC\x6A\x69\x93\x23\x7D\x35\xBA\x96\x5F\x55\x53\x6B\x1B\xE7\x8D\xC0\xAC\x95\xF8\x60\xFF\x82\x25\xAB\xF4\x86\x3E\x5C\xD9\xEA\x66\x5E\x3D\x02\x08\xD8\x97\x15\x0C\x20\xB4\xA4\x36\x98\xA8\x48\xED\x42\xE2\xDB\x0B\x12\x5C\x2B\x16\xBB\x8A\x53\x62\xAA\x17\x01\xA3\x20\xF6\x3A\xFA\xD1\x55\x2F\x8A\xF0\xCA\x59\x21\xD1\x6A\xE7\x2C\xBF\x5E\x72\x65\x42\x5A\x87\x95\xF8\x9C\x60\xE0\xB3\xAE\x1A\xF2\x60\x38\x56\x26\xE9\x2C\xA9\x60\x1E\x48\x01\xB9\xAB\x77\x4C\xCB\x7F\x68\x28\xE6\xD2\xA2\x81\x39\x85\xA3\xD0\xC2\xA8\x1F\x65\xBC\x6B\x37\x67\x20\xB2\xDA\xD9\x9E\x0B\x8B\x6F\x7A\x73\x48\x44\x74\xC2\x95\xDE\xD2\xFF\xAF\x6C\xD1\x2F\xCF\xF5\x8B\x93\x32\x05\xF5\xC1\xE2\x14\x8F\xBB\x3E\x58\x6C\xCA\x8F\xC5\xC1\xE2\x2E\xFE\xB1\x38\x58\xBC\xC6\xCD\xA1\x1F\x87\x37\x44\x1C\x42\xE5\xEA\xB3\x4A\x4D\xD2\xD8\x13\x51\x24\xDE\xB0\xDF\x71\x38\x32\xAB\x9A\xF5\x10\x4C\x40\x93\x74\xDF\xE2\xB5\x01\xD8\x75\x4E\x6A\xFA\xB3\xEE\xC4\x53\x7D\x57\xD2\x5B\x5E\x2B\x18\x99\xEC\xC8\xBD\x8F\x93\x2F\x5F\x56\xFB\xDD\x69\xF7\x5A\xFF\x86\xFD\x03\xC9\x03\xBE\xC9\x0B\xC4\xEA\xD9\x28\xF7\x1A\x77\x97\x3B\xD9\x7E\x3F\x2F\x6D\xC3\x9E\x61\xC9\x41\x32\x8B\xC2\x6D\x4A\x42\xB0\x76\xC5\xB3\xDE\xC1\x9F\x04\x1A\xAC\x49\x69\xE0\xDF\x69\xE3\x0D\x17\xF9\x0A\xF8\x2C\xF2\x82\xE9\xA2\x13\xBB\x02\x56\x68\x13\xC5\x3A\x93\x46\xB9\xD3\x4D\x02\xE0\x3E\xC3\x25\xAD\xE8\xAD\xAE\x39\x70\xCD\x01\xDE\x6D\xDD\x29\xA9\x46\x6F\x9C\x0D\xCD\xEB\xF1\xBB\x19\xC5\xB3\x1B\x2E\xCA\xBB\x35\xBD\x5B\xCB\xC5\xF0\xEE\xD7\x2E\xB4\x43\x5C\x5C\xED\x5A\xAE\x62\x5F\xF2\x2A\xDC\xB7\x2B\x4A\xF2\x1B\xF6\x83\x21\x68\xCD\x6C\x51\x87\x84\xB8\x42\x5F\xB9\xA7\x03\x85\x7C\xF9\x3A\xCB\xD0\xC1\x0C\xC2\x0C\xF2\x8A\x14\xC7\x68\x71\xCF\xF5\xB2\x60\x61\x4A\xC3\x4E\xD9\x40\xE2\x34\x5B\x26\xC4\xCE\x14\x3E\xEE\x03\xC1\xA0\x5E\xB2\x35\x11\x89\x21\x29\xCC\xA2\x4E\x2D\x75\x01\x56\x15\xCD\x29\x26\xDD\x8C\x0F\xB6\xC1\x3B\xDD\x95\xE1\x60\x8E\x55\xC1\x67\x7B\xFC\xD3\xE0\xB2\x76\xE5\xDE\xCE\x50\x10\xEB\xB5\xAE\xF6\xA6\xEF\x6A\x5E\x87\x86\xE3\xF5\x6B\x24\x74\x8E\xDF\xA7\xA3\x45\xC9\x4A\xA0\x5B\x30\x2A\xBD\xAA\x06\xB3\xCA\xCB\xAF\xD5\xAF\x09\x9C\xED\x93\x6A\x29\x6A\x86\xB8\x48\x22\xB6\xAB\x27\xBC\xBA\xC8\x49\xFE\xB0\x08\x26\x5E\xB2\x97\x0A\xDE\x12\xCA\xDF\xC7\x6A\xB8\xFF\xAD\xCF\xDB\x2B\x8D\xE4\x29\x26\x47\x23\x7E\xD0\xA5\x2F\x1D\x71\xD9\xF2\xEC\xA5\xFF\x06\x8F\xDC\x2F\x51\xF7\x08\x3A\xEF\x9D\x1A\xA7\x63\x98\x21\x51\xC1\x0C\x89\x0A\xB9\x0B\x41\x3C\xE1\x78\x84\x27\x3C\xE1\xCB\xFE\x86\x0E\x96\x7F\xE2\x88\x5E\xED\x98\x4B\x4E\x3F\xA4\xCE\xC1\x8F\xAD\x76\xCC\xB6\x93\x3B\x17\xA5\xD3\x7E\xBB\xC7\x42\xB0\x01\xAF\xDC\x31\xF7\x2E\x5F\x6D\xC7\x57\xEF\xE7\x26\xDA\xC9\x4D\xED\x52\x13\x9B\xF1\x6A\xC2\x57\x87\x26\x12\x2E\xAB\xB7\xBE\x4A\xA9\x59\x29\x4B\xCA\x90\x07\xE8\x86\x8E\x15\x45\x5D\x11\x7C\x6D\xA1\xE6\xD9\xB6\x79\xE3\x42\x4A\xA4\x6E\x70\x3E\x0B\x49\x1C\x4E\xFB\x8D\xBE\xFD\xBE\xEB\xA1\x0A\x29\xDD\xB7\xCD\xCB\xF4\x07\x8A\x4E\x77\xBD\x6D\xEE\x5F\xCC\xE8\x15\x5C\xFC\xED\xBC\x2B\x82\x35\x3A\xD4\xF2\x7E\xAA\xAB\x11\xFE\xC8\x2B\x08\x13\x6B\xC3\x39\x47\x0A\xA9\x60\xC6\xDF\x37\x0A\xDC\x90\xFD\x46\xF7\x72\xAD\xC3\x06\x97\xBD\x5A\xCC\x5D\x0C\x3B\xAF\x98\x8A\x69\x16\x89\x69\x21\x84\x63\xC3\x1F\xF5\xDD\xC6\x10\x43\xDE\x72\x0C\x79\x2C\x09\x1B\x04\xB3\xC7\x9D\xED\x4E\xE0\xB6\x93\xD1\x35\x82\x70\x4B\x6D\xAD\xB1\x29\x1B\x87\x56\xAB\x97\x22\x91\xEB\x04\x08\xCA\x9D\xB8\xDC\x10\xCF\x39\xC1\x41\xBB\xAB\x11\xA8\xAE\xAB\x38\x02\xF5\x4C\x67\xD7\x47\xA0\x3A\x8E\xEC\x3D\xC3\xC7\x6F\xC5\xA6\x94\x10\x7F\xBA\x39\x18\xCB\x70\x57\xCB\x86\x94\x6A\x54\x2E\x98\x33\x36\xCB\x8A\x69\x43\x3B\x13\x31\x14\x80\x06\x21\x48\x87\xC6\x27\xCF\x62\x21\xBA\x53\x92\x7D\x82\x84\xE3\x72\xB1\xE9\x6C\x80\xBA\x2D\x5D\x76\x29\x36\x87\xB1\x13\x17\xE6\x71\x3C\x80\xCC\x21\x5A\xE2\xCE\x6C\xB9\x6A\xCD\x38\x1E\x60\x99\xE0\xFE\x00\x9A\x81\xB8\xCC\x4A\xC6\x71\xAF\x93\xD7\x82\x6A\x1E\x63\xE4\x83\x30\x88\x6A\x97\x91\x12\x40\x96\x01\x60\xB8\x72\x85\x3B\x89\x64\x48\xF8\x5A\x6E\xD8\xFD\xDE\x95\xB7\x49\x9E\x8C\xBE\xB6\xC9\x44\x48\x14\xC7\x2D\xAD\x0D\x73\x2C\x7D\x7B\xE8\xDA\x83\x8E\x0E\xE7\x03\x67\xDC\xDC\xEB\xFD\xAD\xC5\x5D\xEE\x24\xD1\x60\xE5\x4E\x9E\x55\x89\x1C\xAF\xC7\xB6\x21\xE9\x80\xD5\x59\xA5\xAF\x30\xCE\xC9\x80\xCB\x0A\x7F\x9F\xED\x8C\xBF\xA1\x19\xBE\x41\xA3\x94\xBF\x33\x74\xD9\x95\xBB\x72\x94\x34\x02\xDD\x9A\xF4\xC8\xC9\xEB\x4A\x7F\xDF\xBE\xBF\x8F\xF1\xD3\xEE\x12\xAF\x49\xE9\xEE\xDA\x45\x16\xA1\xA9\x11\x09\x7C\x93\xFE\x50\x5F\xCC\x08\x21\xEA\xE4\x10\x48\xC0\xC1\xC1\x9D\xF1\x9F\xA6\x7D\xF6\x69\xF5\xF0\x16\xE7\xFB\x8E\x63\x3A\x57\xEF\xB8\xC9\x04\x1A\x0C\xBC\xB3\xFE\x73\x74\xC0\x7C\x4E\x5D\xDE\xEA\xC0\xAC\xAD\x7F\xB2\x5F\xBC\xC6\x19\xFA\x63\xB0\xB7\x6F\x68\xFC\xE7\xCA\x16\xE3\xAE\xC4\x49\xAA\x30\x49\x25\x2C\xBF\x98\x1A\xA4\x15\xEA\xBE\x3B\x45\x4C\x80\xE6\xA9\xA2\x79\x72\xA7\x90\xAD\x59\x49\xE8\x4A\x15\x26\xCC\x30\x3D\x87\x09\x33\x3C\x61\x48\xDA\xBC\x6B\x57\xE8\xB9\x16\xA4\x46\xE3\x4A\xF7\x9A\x4B\x32\xE9\x27\x63\xAD\x5B\xB7\xB9\x28\x19\x93\xD3\xD2\xC7\xD1\xE6\x4E\x6E\xBE\xB9\xCB\x5B\x6E\xEE\x72\xB4\xB9\xCB\x63\x37\x77\x39\xDA\xDC\xE5\xCA\xE6\x9E\xAD\xDD\xDC\xE8\x74\x12\x10\x4D\x6B\x1E\xD3\xDC\xC7\xDC\xF4\x3A\x47\xF8\x0E\x83\x37\xCC\xC3\x68\x11\x00\x34\x66\x5F\xFA\xCF\x03\xFB\xCA\x2B\xF4\x71\x1D\x0B\xD3\x81\x0D\x33\xA4\x28\x9F\x09\x27\x2E\xCB\xB7\x5B\xF7\xDB\xAC\x2F\xCF\x1D\xFB\x6D\x42\xBF\xCD\x50\x9E\x7B\x73\x28\xCF\x3D\xF4\xDB\x0C\xFD\xB6\xA1\xDF\x45\x5C\x99\x78\x9C\x9E\xE4\xAC\x4B\x1C\x43\x88\xE4\xA7\x25\x48\x5C\xC5\x81\x79\x1B\xB4\x58\xD1\xF7\x13\x9C\xF1\x46\x69\x53\xD1\x93\x49\x67\x7D\x79\xA5\xD1\xDE\x39\x4B\x6C\x89\x91\x60\xE9\x02\xBE\x88\x47\xDE\x4E\x58\xF3\x57\x6B\xAC\xD9\xFA\xB1\x0E\xD5\xB7\x37\x38\x6A\xDA\xCD\x68\x57\x22\xCC\xE5\x7E\xB0\xFB\x51\xD9\x5E\x88\x70\xAE\x24\x49\xC0\xBC\xF4\x8D\xE7\xEC\x93\xCE\x8C\x93\x62\x9F\x40\x6C\x9F\xDD\x31\x4F\x2C\xA5\x77\xF2\x45\x5B\x71\x5A\xA4\x1B\x44\xD0\xF3\xD1\x50\x06\x5C\xFF\x90\xF6\xB1\xC6\x3B\x80\x64\x0F\x71\x0B\xD0\x9C\x0C\x74\xBD\x26\x59\x8A\xC8\x99\xA5\x8A\x33\x2B\xB9\xC6\x1A\xD8\x85\x7C\x75\x73\xF9\xAA\xC8\x45\xC7\x74\x60\x7B\xD4\x81\x07\xC2\x79\xBA\xB6\x03\xF7\xC7\x0E\xDC\xBF\xAE\x03\xF7\x86\x0E\xDC\xBB\xBE\x03\xE7\x59\x82\x06\xB6\xD3\x1B\x87\x27\x43\xF1\xDE\x7C\x08\x70\xFC\x64\x14\xDC\xCD\x60\x9F\xF8\xC1\x4A\x17\x47\x82\xDC\xCC\x70\x4B\x83\xB9\x2A\xE9\xD2\x3D\x67\x2E\xA3\xF0\x42\xE6\x0C\xE2\x83\x53\xA7\xF7\xBA\xC2\x21\xD1\x39\x15\x63\x66\xB1\xD7\xE5\xDC\x07\xD3\xBB\xD4\x1F\x3D\xDD\x2F\x4A\x97\x4A\x05\xA1\x7C\x9F\x43\x3D\x52\x60\x0C\x24\xE0\xF2\x0F\x2A\xA9\xAD\x90\x48\x12\xA0\xE4\x93\xE7\x3D\xF0\x54\xF3\x60\x1D\x1A\xDF\x3B\xFA\x62\xC6\x5F\xEC\xF8\x4B\x32\xFE\x92\x8E\xBF\x64\xE3\x2F\xF9\xF8\x4B\x31\xB1\x00\x65\xEC\x73\x2A\x78\x3A\x18\x8D\x37\xC3\xC8\xF6\x16\xA5\xCB\x18\x4E\xC3\x99\xBD\x6E\xE6\x60\xCB\x43\x00\x40\xE2\x32\x8C\x3A\x80\x58\x66\x7B\x34\xEC\x02\x50\xD4\x30\xDC\x66\x32\xF8\xD9\xF2\xE0\x91\xD8\xC0\x70\x5D\x1C\xA4\x3C\x0C\x7E\x7C\x2F\xEE\x1C\x8F\x7F\xE6\x4D\x3F\x9E\x82\x99\xB7\xFD\x78\x16\x68\x63\x8E\x27\x62\xC6\x79\x23\x71\x2E\x66\x9C\x5F\x12\xA7\x63\xE6\xF3\x7E\x34\x23\xE8\x5B\x41\x5B\x3B\x0D\x13\x93\xF0\xC4\x94\x6C\xB5\x2E\xF7\x48\x63\x99\x18\xAE\x13\xA0\x81\x03\x49\xC8\x3E\xBC\xE5\x41\x92\x6F\xDA\xA2\x13\x3E\xA7\xD3\x3C\x15\x1C\x3F\xA2\x87\x06\x29\xDA\xC8\x8E\x5C\xA2\x8A\x9C\xC3\x6B\x57\x88\x03\x3E\xE8\xFC\x31\x5E\x95\xAF\x26\x6D\xA0\x8E\xCD\x73\x7D\x97\x13\xBB\xE3\x0D\x22\x9E\xCE\xD9\x1E\x2D\x7D\xEE\x8F\x1E\x01\xE9\xCF\xF6\x48\x04\x44\xE0\x71\xE2\x66\x7B\x5D\xE2\xF5\x61\x37\x73\x33\x97\x5F\xDE\x42\x5C\x50\x1E\x8A\xBA\x5C\xDE\xEA\xB2\x21\xC0\x4E\x6C\xB3\x80\xD4\xA5\xC1\x93\x34\x5C\xF6\x5D\x0E\x4A\x33\x71\xFB\xE4\xB7\xDC\x3E\x39\x89\x53\x33\x70\xFD\xAF\xC1\x06\x8A\x36\x25\x2D\x1E\x90\xBD\x8E\xD1\xDB\x9D\x00\x7C\x1A\xD4\xEA\x08\x86\x12\xE4\x01\x14\xCC\x4F\xD8\x6E\x76\xF4\xB4\x13\xEF\x08\xB1\x18\x1A\xB7\x65\xE9\x86\xC6\x3D\x73\x49\x4F\x4A\x70\xA0\x51\xCB\x31\xFE\x08\x9D\xC7\xEF\x0B\x8D\x7B\x16\x9C\x38\x17\x26\x60\xE6\xB4\x3C\x40\xFB\xC7\xE9\x61\x13\xD1\xFE\x71\x7A\xD8\x44\xB4\x7F\x9C\x1E\x36\x11\xED\x1F\x44\xB8\xC7\xEF\x29\x7D\x4F\x87\xEF\x19\x7D\xCF\x86\xEF\x39\x7D\xCF\x87\xEF\x05\xED\x6C\xC9\xCB\x4F\x63\x29\x7B\x41\xF2\x2D\x26\xEE\x07\xB3\x0E\xE4\x20\x1F\x78\xF4\xBD\x31\x05\x9D\x73\x25\xDB\xF7\x72\xB4\x1B\x0C\x33\xF2\x25\x24\xA7\x57\x7F\x2C\x70\x18\x47\x5E\xB5\xDF\x1F\xA2\xB4\x68\x1A\xFC\x26\x42\x7B\xD8\x46\x2E\xE9\xDE\x08\xFD\x69\xBF\xF7\xF9\xC1\x66\x03\x8C\x52\x00\xA0\x1D\x41\x58\xB1\xC8\xCA\xAE\x19\x36\x41\xB7\x5F\x80\x11\xEE\xB4\x18\x4C\xD8\xDB\xD5\x7E\x96\x2B\xD3\xC3\x43\xC5\xF7\x76\x16\xA0\x74\x7F\xE1\x61\x52\x84\xE0\x8A\xE9\x6C\xFB\x3D\xCF\x23\xAA\x47\x1F\xD3\x8C\xD7\x07\xA1\x25\x81\x2A\x44\xAA\x3B\xBA\x7E\xD7\xC3\x1C\x12\xD6\xFE\x20\xB5\xF1\xA0\xD2\x12\xF4\xD5\xB2\xBD\x5E\xC3\x24\x50\xFD\x1F\x46\xA7\x47\x5E\x05\x70\xB7\x96\xF3\x58\x12\xF8\x5D\xF0\x89\x94\x9D\xCC\x7F\x23\xE3\xE2\x01\x4C\x26\xA5\x9F\xF2\xE9\x4F\xAF\xD3\xC6\x25\xAF\xD3\xE6\x69\x60\xB4\x82\x69\x01\xF2\x0C\x71\x48\x3E\xA1\xE5\x4D\xFD\xA6\x98\x50\x52\x9E\x99\x94\x8D\x3B\xA9\xB3\xED\xB5\xE7\x81\x67\xC7\xB5\xFF\x31\xA7\x19\x3F\x2F\x35\x28\xB7\xA9\xAF\xAC\x7D\x5B\xAF\xE0\x63\x4B\xE9\xD7\x4D\x46\x78\x94\xE7\xA3\xB3\x8D\x53\x59\xB6\x4D\xCB\x86\x11\x84\xD6\xB5\xDC\xCE\xA6\xD8\x93\x63\x91\x14\xC8\x29\xEC\x0F\xA2\x27\xB8\xA9\x4A\x66\x44\x73\xD4\x6C\xC2\xE0\x19\x89\xB7\xD8\xBD\xCE\xBC\x5E\x1B\x80\x3D\x02\x94\xA3\x75\x09\x2F\x4E\xE5\x6C\xF5\x41\x03\xD9\xC0\xAB\x6D\xF3\xD9\xF7\x3C\x7F\x4D\x75\x9A\x3B\xFE\x91\x7F\x7F\xFD\x9A\x5A\x18\xFE\xEB\xB4\xAF\x9F\xEA\x83\xC5\xC5\x99\x2E\xA1\x7D\xF6\x6C\x97\xD2\xCF\xAC\xB1\x65\x4E\x5D\x41\x12\x0F\x12\x1E\xD1\x8B\x74\x51\x20\x88\xD4\x2B\x40\xCE\x74\x99\xCB\xB7\xBA\x02\x2D\x41\xC3\x77\xD9\x1E\xFD\xA2\xBD\x62\xA4\x7C\x67\x61\xAD\xBB\xC0\x70\x37\x34\xB3\x8E\x5A\x6D\xD4\x24\xB1\x36\x71\x68\x42\x6C\xA3\xB4\xDD\xE7\x36\xD5\x16\xD2\x0F\x49\x3E\x6F\xEF\xE7\x86\xBF\x3A\xD5\x63\xC8\xD9\x1E\x2D\x32\xBB\x43\x04\x2F\xDD\x6B\x5C\xF2\x47\xE3\x3B\xD8\x91\xE7\xCD\x5F\xE3\x00\xB9\xD1\x4D\x40\x63\xAB\xBE\x99\x03\x76\x62\x28\x6E\x27\x3A\xB9\xD3\xED\xFB\x78\x7F\x92\x7E\x1A\x08\x9E\x94\xA5\x3F\xBD\xA1\xF6\x85\xDE\x6D\xB8\xD0\x99\xF6\xBB\x9E\x17\x20\xCF\xCD\x3E\xE0\x2C\x80\xD1\x6B\x0E\x92\xA4\x41\x0E\xBA\x2C\x50\x20\x19\x38\x75\xD3\x19\x67\xDA\xEF\x7E\x9E\xD9\xAB\xE1\xB7\x0A\x1C\x44\x6D\x18\x8B\xC9\x5F\x6B\xC5\x12\x4E\xDC\x3A\xF6\x86\xB4\x30\xE9\x08\xE3\xFE\xD1\x0F\x60\x1D\x8D\x5E\xBB\xD5\x3B\x0B\x1A\x82\x2B\xB2\xFD\x7B\xE0\x31\x95\xD7\x55\x47\x9D\x95\xFB\xE5\x18\xE0\xFB\xE5\x1E\xAF\xAB\xEF\x65\xDC\x0C\xF6\xCD\x9C\x0E\x30\x56\x20\xCE\x77\xB6\xFB\x9E\x31\xC3\x9D\xF1\xB5\xD3\x4B\x8F\x6E\xD0\xF1\xB8\xE9\xF4\x78\xDE\xF4\xF2\xBC\x41\x54\xDE\xA4\xE3\x88\xD5\x06\xAF\x7C\x29\x73\x82\x40\x55\x5F\x31\xC3\x93\x9F\xC0\x35\x43\xD7\x7E\xC6\x68\x73\x34\x24\x7A\x26\x91\x61\xA4\xFE\x35\x97\x1B\x46\xC0\xAD\x3B\xE5\x4B\xF6\xF5\x01\x8D\xD1\x24\x40\x97\x94\x05\x7F\x59\x3D\x4D\x84\xBA\x25\x35\x4B\x52\x8E\x1F\xD3\xA4\xC3\x81\x9F\xEC\x06\xEF\x5A\xEA\xFF\x7A\x2F\x40\xB2\xB8\x00\xB8\x16\x9F\x0B\x56\x64\x18\x13\x40\xC1\x65\xC2\x13\x31\x5C\x6F\x9B\x9A\x38\x5E\xE2\xF5\xEB\xB5\xA4\x27\xD5\xAC\x5E\x46\x06\x2F\x79\x68\xEC\x3C\xC1\x1D\x44\x1A\xBA\xFD\xBE\xE7\x11\x8F\xDC\x02\xB6\x14\xEA\xDD\x70\x92\xC8\xE1\x41\x9D\xC9\x3C\x62\xBB\x80\x9F\x0E\x8E\x0E\x21\x48\x0B\x74\x26\x9D\x4F\x60\xC1\x01\xD2\xA8\x25\x0E\x42\xFC\x98\xC6\x49\x3C\xE3\x41\x8E\x33\x05\xD7\x9E\x5D\x0C\x8A\x6B\x50\xD3\xC2\xF1\x83\xFC\xB5\x28\x17\xBC\x15\x9E\xF6\xEA\xE7\xB4\xCE\x8F\x30\x3D\xB8\x90\x7A\xBB\x4F\xF2\x00\x7E\x39\xEA\xBD\x7D\x0C\x3C\x13\xB0\x5F\xA9\x7F\xFB\x3E\x4A\x4E\xE6\x2E\x81\x1F\x3A\x0D\x68\xD4\xC1\x69\x4C\xA2\xAB\xB8\xF6\x2A\x78\x99\xBB\xDA\x59\x57\x3B\x7B\x69\xAB\xB3\xA4\xA2\xE3\xEF\x4C\xFE\x96\xF4\x17\x6C\xBC\xED\x17\x29\xF2\x12\x93\xBE\xCB\xF9\xD8\xCE\xB8\xE2\x17\x60\xC5\xF8\x65\x40\x96\x75\xD6\xE5\xF1\x31\xF8\xBE\x33\x12\xC6\x32\xF6\x74\x57\x89\xF2\xAA\x7A\x6F\xA2\x93\x11\x5E\x68\xBA\x04\x66\x19\xB0\x20\xFD\x67\xF2\x2B\x80\x54\xE2\xBF\xC6\xE9\x4B\x1C\x64\xAF\xFD\xEF\xE4\x97\x22\xF1\x19\x0F\x07\x90\x71\x36\x10\x1F\x84\xE8\x81\x68\xB9\x26\x06\x4D\xFD\x88\x64\x0D\x93\xEC\xF2\x53\xBC\x0B\xB3\x48\x6E\xB9\x9F\xB9\x0C\x34\x9C\xB9\x64\x4B\xE8\x2E\xA3\x45\x6F\x39\xF6\x02\x01\xEF\x99\xEC\x0E\x21\xC4\x8C\x09\x31\x19\x08\x31\x89\x84\x98\x08\x21\x16\x0C\x01\x07\x42\x4C\x5C\x31\x10\x62\x41\x84\x58\x4C\x08\x31\x68\xAD\x59\x24\xC4\x8C\x08\x91\x05\x65\x22\xC4\x82\x44\x47\x84\x4F\x66\x42\x88\xE2\xA1\x6F\xA9\xDB\x0F\x2A\xCD\x90\x66\x74\x04\x5D\xC1\xCC\x9A\x4B\x9C\x87\x92\xB8\xD4\xDF\x2D\xC0\x5A\xA9\x3F\x23\xB1\xFF\x74\x8E\xCB\x27\xFA\x53\xBB\xD4\xBF\x7C\xFD\xBF\xDA\x31\xDB\x2E\xF5\xBF\xFC\x3B\xD7\x91\x70\x0B\xFA\x2B\x7A\x97\xF9\xCF\xE4\x5E\xB7\x1F\xB8\xCE\xC8\xAA\x6B\x57\x04\x77\xB6\x1F\x04\x4F\x61\x3A\x8D\x5F\x01\xE8\xE8\xD2\x01\xEA\x34\xA9\xDE\x35\xD3\xF9\x28\x09\x27\x09\xD4\x81\x24\x12\xC4\x54\x5E\x10\xD0\x27\x46\xC6\xC3\x3E\xE0\xB4\xE2\xCD\xAE\xE8\xBB\xD2\x7F\x48\xFF\x8F\xDD\x2C\x98\x6F\x4A\x37\xF3\x1F\xD2\x4F\xED\xF9\xD3\x4F\xF5\x74\x07\xE3\xAF\xB3\x70\xC0\x81\xDF\x92\xBC\x88\x48\x92\x8C\xA3\xFD\x51\xEA\x27\x17\xE0\x58\x1A\x52\xE9\xB2\xBD\x2E\xF3\xE6\x19\x6A\x8B\x33\x2E\xB2\x2D\x57\xB8\xB4\xDF\xF3\xBA\xF7\xBF\x00\x7B\x40\xC9\xE1\xEB\x63\x60\xA3\x1A\x91\x0F\x35\x07\x93\xE7\x95\xCB\x17\xA9\xCB\x45\x75\x63\xD9\x95\x36\x49\x89\xD1\xD5\xE2\xB9\xE3\x9D\x8A\xC4\xE1\xF1\xFB\xF2\x2D\xA7\x84\x78\x00\xBC\x57\xB6\xAF\x84\xF4\x8E\x4D\xB4\x1F\x07\x91\xC2\x6A\x80\x69\xC8\xC2\x34\xE4\x2E\x8B\xD3\xC0\x98\x45\x9B\x8C\x59\x44\xBB\xFA\x88\x5D\x49\x03\xE4\xE2\x69\x92\x3E\x68\xFC\x1C\xDE\x5F\x48\x6A\x1C\x4A\x84\x73\xC3\xE9\xD0\x70\x3A\x6E\x18\x51\x50\x39\x62\x58\x8D\x54\xD4\x4A\xFC\x11\x9F\x60\x6A\xDB\x38\xE4\x70\xED\x18\x87\x74\x35\x75\x85\xA5\x86\x07\xD5\x19\x54\x4B\xCB\xE9\x55\x82\xB2\x7E\xE2\x1A\x63\xE5\x1A\x91\x88\x1E\x54\x1B\xD4\x8B\x70\x57\xE2\x37\x02\x28\x5D\x71\xE2\x6F\xE2\xD6\xCE\xE0\x14\x38\xAB\x36\x10\xFA\x25\x49\xC6\x3C\x52\xBF\x29\xF9\x47\x24\xC8\x66\x9D\x11\xB0\xDF\x02\x9E\x5D\x31\x97\x17\x0C\xCF\x26\x29\x2A\xE2\xD6\xE0\xDB\x79\xCB\x05\xDD\x14\xFB\xD7\x19\x54\xEB\x44\x44\x0E\x75\xC8\x78\xBD\xBB\xC5\x83\x0E\x26\x29\x0C\x10\x72\x67\x11\x02\x7B\x25\x2E\x89\x26\xC1\x9B\xC3\xCE\xB0\x5B\x30\x5E\xEF\xE4\xAC\xCF\x21\xBA\x30\x3D\x30\xF0\x5E\x80\x0E\x4E\xF9\x5A\x16\x60\x32\x53\xE6\x3F\x23\x48\x09\x26\xB5\xDA\xA2\x0A\x40\x0D\xDD\x91\x08\x91\x77\xCB\x5A\xAA\xDA\x36\xA7\x79\x79\xBF\x08\xC2\x32\xC3\xFA\x9B\x35\x84\x95\x4D\x08\xAB\x58\x25\x2C\xA4\xC1\x84\xC5\xA4\x8E\xF2\xB8\xC2\xF2\xD7\x27\xBE\x4D\x96\x1F\x2B\xE0\xB1\xBF\xF2\x40\xFE\x63\x24\x7A\x37\xDA\xC5\xDD\x78\xCB\xB2\x5B\x77\xCD\xAE\x4D\x79\xB0\xA3\x8D\x5B\xDC\xC6\xC6\x0D\x75\x9B\x97\xB7\x43\x3A\xDD\x67\xE9\x78\x9F\xA5\xC4\x70\xF8\xF8\x21\x22\x41\x2A\x39\x97\x52\xC0\x91\x0A\x71\x1D\xF2\x43\xCA\x97\x35\x42\x1A\x59\xA1\x19\xC1\xBE\xFE\x5B\xAD\x33\x52\x5E\x3F\xCB\xBC\x77\xCC\x0B\xF5\x84\x17\x1A\x00\x7A\x8A\xCB\xB8\x4B\x6E\xDA\x39\xBA\x43\x32\xE8\x47\xBC\x10\x80\xA0\x56\x02\xFD\x98\x17\x86\xE0\xBA\x38\xB1\x08\xAB\x33\x93\x59\x35\x5B\xA4\x41\x8D\xA7\x34\xBD\xC5\x94\x92\x58\xB4\xD0\xCE\x0A\xFA\x5E\xD2\xB7\x1F\x12\xF0\xBD\xAF\x20\x8C\xDC\xA7\xD4\x3A\x18\xB9\x0F\xE9\xAF\x3A\x8C\xDC\xBF\xD4\xA4\x93\x07\x2D\xE2\x35\x63\x2D\xC2\x48\x6C\x00\xF2\xD7\x5A\x9E\xFA\xCD\x01\x76\x1F\x79\x5E\x75\xF4\x19\x20\xCC\x4C\xFB\xB7\x3D\xC6\xD4\x89\x33\xBE\x62\x1C\xDB\xB6\xF7\x6F\xDB\x0F\xD1\xAF\x8B\xCC\x59\xC6\xBD\x25\xB9\x4C\x1F\xB0\xFF\x20\x93\x40\x49\x80\xB9\x1F\xC2\x84\x44\x7A\x55\x46\x47\x13\x31\x3A\x5A\xD4\x68\x9C\xCA\x9C\x16\x49\xC1\x20\xE8\x14\x91\x41\x18\x55\x54\x46\x82\x69\xE6\x67\x8C\x4E\x97\xAD\x2D\x41\x1A\xD3\x82\x2C\x1F\xA5\xB1\x38\x0D\x66\x55\x39\xA2\xCB\x69\xFB\x43\xD7\xD7\x60\xCE\x68\x41\x58\xAE\x65\x72\xA9\xC5\xD7\x5E\x64\x75\xBA\xEE\xEC\x85\x86\x5D\xA7\xA4\x24\xB2\xBE\x52\xC3\xBB\xEA\xEB\x8B\x8D\x71\xA9\xFF\xC6\x7E\x7E\x32\x31\x4B\xFF\x12\x33\xEE\xCF\x6B\xD7\x29\x6B\xA8\xE1\x43\x0F\x6B\x25\xFF\x8C\xFC\xD5\x4A\x0C\x34\x12\x49\xDB\x12\x49\xEC\x98\x1A\xCF\x68\xD2\x46\xEB\x95\x6B\xCE\x8A\xD1\x87\x4D\x0D\xB8\x12\xEC\x40\xBA\xFA\xF9\x04\x60\x83\xCB\x28\x59\x2D\x46\x2C\x86\x12\xA9\xE0\x98\xFA\xBB\x1E\x86\x67\x33\x01\x5E\x12\x0B\x77\x5E\x85\x08\x9F\xDA\x54\xF6\x3C\x1D\x12\x35\xAA\x5E\xB0\xF4\x76\x43\xEF\x83\xA0\x21\x57\x8A\x0C\xD9\x2F\x06\x00\x52\xA2\x99\xA1\xED\x5A\x23\x4D\x5B\x02\x2A\x78\xCE\x89\x6D\xA3\x7E\x43\xC2\xB1\x43\x97\x1B\x5D\xA7\x55\xB0\xE8\x3E\xDC\x24\x28\x7F\x9F\x0C\x58\xA9\x39\x1F\xE6\xC3\x77\xEE\xBD\x65\x53\x56\xCE\x75\x43\x13\x94\xAB\xA0\xC6\xC5\x76\x6C\xA9\xAB\xA4\xF9\x09\x64\x0B\x1D\xC9\x99\x64\x29\x92\x86\x31\xE3\xF3\x13\x6E\xC1\xA2\x67\x18\x60\x3A\x32\xB6\x10\x00\xDE\x53\xAB\x99\x45\xC5\x7D\x5A\xCF\xCF\x5F\xE7\xF3\x3D\x94\x05\xC9\x70\x1D\x47\xF2\xA2\x60\x61\x32\x0D\xC5\x2F\x0C\xAF\x05\x6C\x78\x22\xCD\x3B\x3B\xD5\x28\xED\x44\xA3\xB4\x83\x20\x6F\x39\x74\x7B\x2C\xC8\x47\x3F\x54\x1E\x8A\x91\x18\xC1\xBD\x43\x8C\xA1\x76\xE5\xC2\xB8\x32\xCE\x88\xA9\x39\x35\xBE\x44\xCD\x54\xCE\xE6\x51\x5E\xB7\xCF\x3F\x0F\x4D\x58\x44\x7D\xDA\xD9\x00\x49\xB1\x2E\x7B\x50\xE9\xA0\x81\x06\x17\xD7\xB0\x2B\x7F\xCC\x68\x7B\xE4\x54\xFB\x92\x30\x5A\xD9\x47\xA0\xCB\x48\x76\x4E\xF9\x6C\xD8\x70\x03\xB9\xF1\xB6\x1A\x6D\xDE\xDD\xA8\x4A\x69\x9F\x73\x34\x96\xBA\xC7\x24\x51\xFB\x77\xA1\x61\xE5\xF4\xEB\xB4\x19\x21\x86\xB0\xA1\x8F\x18\x15\x31\x13\x7D\x1B\x5A\xFB\x9F\x61\x8E\xE9\xA5\xA2\x2C\x69\x60\x4A\xC3\xAF\x82\x80\x4F\x78\x64\xB8\xC7\x83\xE9\x14\x97\x65\x0A\xCC\x82\x0D\x3E\x83\x19\xE4\x13\x1A\xB5\x00\x6E\x7F\xEA\xE4\x06\x88\xB7\x08\xCD\x1A\x55\x18\xAA\x1B\x09\xD8\x12\xE0\xE5\x36\x1A\xA2\x75\x34\x44\xB3\x45\x98\x47\xAC\x9D\x1A\x46\x4C\x6A\x42\x48\x6E\x0B\x16\xEF\x30\xE2\x8D\xB8\x20\xD9\xD2\x82\xC0\x54\x51\x5D\x1F\xE9\xDD\x6D\x2C\x22\xC1\x86\x50\xCE\xCC\x97\xF8\xE7\x2E\xF1\x29\x4A\x93\xC3\x37\x6F\x6C\x28\x88\xFF\xF0\x10\xB7\xE4\x64\x0B\xE9\x9E\xBD\xD2\x45\xCF\x92\x91\x62\x69\x6D\x0B\xD5\x12\x6A\x3E\xDC\x0D\x70\x9B\x31\xD4\x18\xFB\xDF\x76\x89\x93\xD4\xE5\xA0\xD5\x88\xA5\xFF\x7D\x23\x10\xE8\x56\x9E\xAE\xA5\xE2\xFE\x51\xEF\xC4\xD2\xB8\x29\xB5\x57\xD2\x21\x69\x4E\x0F\xA2\x4C\xB0\xE6\x72\x08\x3A\x8C\xBA\x6C\xC4\x4B\x9C\xDE\x63\x30\x9F\xF0\x56\xDC\xCD\xF9\x87\x43\x1E\x7A\x15\x98\xA0\x8C\x36\x16\x8C\x83\x78\x24\x09\xDE\xDD\xEA\xA8\x39\x40\x34\x45\x10\x35\x0C\xE3\x42\x19\x3A\x42\xE6\xD0\x78\x89\x7F\xA5\x73\xAB\x10\xF4\x30\x9E\x9E\x42\x6A\xA4\x26\x00\x95\x56\xEC\x07\x49\xDB\xBF\x05\xDB\x77\x3A\xF0\x22\x25\x6C\x8A\x36\x8E\x86\x43\x86\xD8\xF9\x0D\xBB\x3F\x1A\x41\x28\x5D\xC7\x12\x96\x89\xF2\x64\x75\xA3\x9C\x40\xBA\x59\x11\x33\x21\x1E\x9C\x76\x96\x16\xF6\x8C\xB3\x4C\x98\x16\xE9\xCA\x1C\x5C\x6B\x39\x95\xE4\x74\xDF\xFE\x23\xEA\x07\x83\xB5\xD7\x9D\xE2\xD7\x28\xFF\x14\xB1\x62\xE5\xDF\x42\xCC\x54\xF9\x37\x33\x70\x96\xF2\x6F\x13\xDC\x85\xD2\x29\x7F\xD8\x77\x08\x88\xD4\xBB\xC1\x44\xCD\x13\x49\xFC\xAD\xDC\xF3\xC9\x33\x97\x44\xD1\xF2\xFA\xD1\xA1\x0E\x44\xE9\x2A\x9F\x1C\xF6\x1D\x76\x6F\x85\xE1\x2E\x34\x57\x55\x93\x68\x0D\xA9\xB0\x00\x68\x3B\xA1\x9C\x11\x58\x94\x20\x88\xB7\x62\xD3\xB0\x52\x63\xA6\xAE\x13\x0E\x01\x77\xDA\x55\x3C\x26\x2E\x03\x54\x71\xAC\xB1\x94\x83\xEE\x14\x42\x41\xBA\xA6\xEF\xE6\x5C\x1E\x7B\xE3\x31\x94\xB4\x30\x87\x0B\x6D\x91\xC4\xDD\x23\xAE\x9E\xE3\x30\xE1\x74\xAB\x69\x5B\xD4\xC3\xE3\xBA\xE7\x18\xD0\xC5\xDC\x71\xC0\x67\xD7\xF4\xD2\xD6\x65\x36\x06\xB0\xAE\x49\x23\x3F\xEA\x03\x65\xA0\x72\x28\x6F\x33\xB7\x41\x9F\x6B\x67\xDA\x6F\x7D\x5E\xD0\xA6\xAC\x37\xCE\xB4\xEF\x78\x5E\xF2\xCA\x51\x01\x79\xD8\x98\xD2\x83\x92\x88\xB3\xEA\xA5\x74\xBA\xDC\x10\x5A\x5D\xBE\x87\x35\xDB\x45\x48\x84\x6A\xFB\xAE\x44\xFC\x2A\xAB\xA4\x15\x23\x89\x57\x70\xE9\x6F\x60\xFA\xAB\x3D\x6F\xF7\x19\xA5\x84\xA7\x03\x33\xB5\x68\xAC\xCC\x5C\x49\x53\x53\xBA\xA6\x9F\xCE\x0E\x22\x5E\x9B\xF0\xFA\x8A\x2B\x8C\xCF\x89\xA0\xE7\x0D\x87\xD1\xD8\xCB\x8D\x9E\x36\x89\xB7\x61\x65\x54\x68\x7D\x4E\xAD\xAB\xDE\xCD\x5D\x43\x7F\x03\xAA\xD8\xE8\x12\xED\x0B\xB9\x3A\x82\x31\x1B\xDF\x50\x0C\x37\x14\x6B\x6F\xA8\x87\x1B\x62\x96\x0D\x91\xD2\x82\x86\x9C\x70\xFF\xDF\xDE\x77\x55\x00\xE9\x77\xD5\x15\xA0\xFD\x17\x0C\x7C\xE4\x32\x44\xD0\xBB\x19\x42\x37\x2B\xD6\xF3\x88\x71\x40\x55\x9F\x0A\x76\xF3\xC4\x2A\x8D\x60\xF2\x6D\xA9\xD1\xE0\x94\x2B\xDA\x1F\x11\xC2\x1C\xFD\x6C\x5C\xD1\x7E\x14\x3F\x1B\x59\xB3\x4A\xF6\x22\xAA\x1F\x74\x7A\xCF\x9F\xD9\x65\xAC\x12\xFF\xB6\xC1\xEC\x4D\xFB\x4F\x90\xE0\x07\x58\xEF\x27\xFB\x4E\x40\x25\x74\xB0\x1F\xF8\xAF\xE7\x1C\x65\xDA\x74\xB4\x87\x6A\x6C\x99\x37\x63\xCB\xD8\xE5\xEE\xFD\x28\xBB\xC5\x12\x8E\x74\x96\x62\x0F\x9A\xF6\x72\x57\x21\xD2\x0A\x44\x6D\x77\xB7\x68\xA3\xEC\x6E\xB1\xA9\x25\x58\xBF\x78\xE0\xCC\xD2\x35\x57\x05\xA2\x1E\x2F\xA2\x4A\xAB\x18\x26\x8C\xF4\x43\x9E\xC8\x7A\xE8\xF9\x61\x0F\xE8\x9A\x37\x63\x2D\x44\x33\x43\xF9\x08\x55\x31\x33\xAB\xC3\xC9\x58\xC1\xD7\x2A\x7C\xAE\xAA\xBE\x6D\x10\xAB\xA7\xE9\x89\x7B\xFE\xCC\x5B\x3B\x76\xBC\xF9\x7F\xFA\xFE\x77\xFF\xAC\x7A\x4C\xBA\x98\x31\x97\xE7\x1B\x72\x00\xEA\xB8\xC2\x65\x97\xB7\xFC\xCF\xD2\x6D\x2E\xF7\xEF\x7F\xF7\xCF\xDE\x1D\x23\x18\x72\x36\xCC\x4A\x13\xDA\x65\xFE\xCC\x53\x41\xE0\xAD\x60\x30\x3B\xF3\x54\xDF\x95\xCC\xA6\xCA\x90\xD3\xED\xCA\x98\x34\x55\x46\x66\x95\xE3\x61\x46\x4F\x3A\x43\xFB\xC4\xBA\x84\x83\x60\x35\x42\x46\xD1\xD7\x37\x93\x6A\x6A\xE9\x73\x4E\x9F\x53\x58\x5D\x55\xB0\xF9\xC3\x04\x7F\xD8\x77\x05\xB7\x08\x11\x99\xAD\x46\xC4\x9B\xA5\xA6\x58\xE1\x72\x46\xEA\x60\xC3\x6D\x4E\x97\xD2\x98\x43\xF8\x5C\xCF\xE1\x1E\x26\xFE\xF2\xF6\x1E\xD2\x7A\x84\xEE\x63\xAF\x6F\x60\xB0\x9A\xBA\xA1\x7B\x06\xF1\x63\x8F\x7E\xC1\xE1\x30\x59\x64\xCB\x01\x9D\x91\xC6\xC0\x36\x2C\xBB\xDB\x18\x74\x8A\xA4\x75\x73\x9E\xED\x6C\x39\xCB\xF9\x49\xCF\x09\x67\x31\x70\x8B\x86\xB4\xA0\xD9\x79\x33\x0C\x8E\x88\xD2\xAB\xED\x18\x3B\x4F\xDE\x1A\x4E\x81\x52\x72\x5C\x26\xC1\x62\xD5\x3F\xCB\x74\x2D\x45\xED\xE0\x08\x66\xB5\x89\x4B\x8E\xDA\xF3\x2E\x95\x3A\x2B\x89\x93\xCC\xAC\x3C\xD0\x4D\xAC\x88\xD3\x15\x74\xDA\x5F\xED\xE7\x89\x61\xC4\xB8\x22\xD8\xAB\x3C\x76\x43\xC1\xD6\x42\xDC\x65\x76\xB7\x98\x38\x10\x40\x25\xD5\x82\xE0\x6C\x2F\xFC\x61\xEF\x4B\xB6\x45\xCD\x2E\x22\x43\x71\x93\x4D\x36\xDB\xA6\xE5\xEF\xC1\xD0\x98\x0E\x12\x16\x3F\xB9\xF9\x70\xA3\xBD\x76\xC5\xA1\x7F\xE7\x6F\xBC\xAC\x90\x5B\xE3\x52\x97\xB7\xDF\xF1\x3C\x9C\x95\x1C\x70\x97\x41\x9E\x83\xC6\xB4\x0B\xC1\x04\x6A\x10\x2C\xCF\x89\xCB\x98\xB7\x43\x5B\x5A\x14\x2E\x63\x8C\x7A\x8E\x9C\x8F\x02\xCD\x68\x88\x83\x49\x2E\x05\xCC\x0F\x36\x2A\x0F\x90\x18\xB7\x08\x3D\x55\xA8\x13\x00\x1B\x6A\x2D\xC1\xD7\x44\x05\x35\x5F\x4B\xA2\x7C\x54\x73\x79\xF8\xD4\x01\xC1\xBE\x26\xB9\x86\x71\x7B\x51\x5D\x7A\x2C\xD7\xD4\xFC\x7A\xB3\xA8\xC2\xC3\x23\xFB\x41\x25\x2E\xD0\x94\x74\xD9\x94\x71\xC2\x12\x14\xFD\x45\xA8\x4B\x80\x8E\x49\x5C\xB6\xB7\xA8\xE2\x80\xA1\xD4\x21\xAE\x8F\xD9\x67\xF1\x18\x23\x1A\x12\x07\x49\x89\x7C\x0A\xEA\x18\xBA\xCB\x11\xF8\xEC\x11\x8B\xA5\x9B\x2C\xC7\x3D\x75\x0D\x31\xBD\x39\x12\x80\xA1\x0E\xEB\xC7\x9A\x44\x7C\x9E\x85\x9B\xED\x75\x85\xB7\xCF\x74\x1B\x6E\xC3\xA5\x12\xFB\x84\xEE\x3D\x43\x1F\xDF\x70\x79\xAB\xDB\x18\xA2\x9F\x36\xC2\xF3\xA9\xDB\xF0\xC9\xE1\x10\x0B\x55\xB8\x39\x44\xA1\x9B\xF5\xAE\xC0\x33\xC4\x2B\xD8\x6F\x67\xCE\x87\xE2\x37\x85\x9B\xB9\x26\xDA\x73\x2C\x3F\x6C\x51\x75\xC4\x59\x36\xD2\x32\xDA\x5B\xD8\x64\x09\x9B\x80\x2E\x6D\xA1\xD0\x3D\x69\xF6\x4E\xEF\x22\x27\x13\xDE\xF7\xE2\x02\x4D\xE7\xA0\x60\xA2\x40\xD3\x20\x59\x26\xA3\x84\xCE\x10\x66\xE1\x37\x7B\xDE\x5C\x52\xD0\xB9\x18\x08\xCD\xA1\x8F\x39\x67\x13\xAA\xC1\x64\x5E\x86\x92\xB0\x39\x51\xD8\x37\xF6\xF3\x93\x4A\x2F\xFD\x23\x9A\x2C\xC3\xD5\x68\x98\x51\x83\x61\x06\xED\x63\x0B\x14\x71\x0B\x24\x40\x93\x7D\x9E\x6B\xA5\x66\x0B\xB9\xE7\x11\xB6\xA7\xCA\x26\xAF\x22\xF8\x79\x6C\x7D\xD9\x66\xA4\x0C\x6A\xFB\xB4\x5D\xB3\x6D\x6A\xD2\xDA\x1A\x80\x76\x22\xCE\x2A\xDF\x16\x1C\x55\x52\xD6\xC4\xBD\x9E\xFB\xA2\xEF\x72\xDC\x45\xFF\xA9\x05\x84\x09\xE2\x03\x02\x8A\x48\x6C\x16\xEC\xE9\xEB\x2D\xE3\x3A\xBB\xDC\xCF\xAE\xE0\x4F\x79\x45\xB2\x22\x25\x90\x62\xE6\x72\x97\x7B\xD5\xFE\xAF\x41\x35\xC0\x0F\x31\xE2\x61\xD8\x8A\x48\x3C\x2E\x98\xFE\x0B\xA2\xFF\x3A\xD2\x7F\xE5\x66\xAC\x5F\x3A\xD3\x7E\x3B\xE6\x83\x0B\x23\x60\x4F\xE4\xBC\x27\x9A\xC7\xF0\x7A\x50\x1D\xD2\xF9\x0A\x4E\x3D\x4A\x5C\xCE\xA8\xFE\x56\x3C\x29\x09\xC3\xE3\x85\x3D\xB1\x41\x7B\xA2\x85\xD1\x3F\xD2\xB4\xC4\x03\x36\x0E\x7B\xC6\x3E\xD3\x9D\x70\x27\x62\x3C\x20\xE4\xE3\xB8\x27\x4E\x60\x4F\x34\xD4\xE6\x89\xF0\x7C\xE6\x4E\x0C\x7B\xA2\x61\x84\x4E\xDA\x13\x37\xEB\x5D\x83\x67\x20\x93\xD3\x9E\xA0\x29\xDE\x90\xF4\x2B\x37\x73\x1B\xA3\x3D\x51\xF3\x9E\xC8\x79\x4F\x34\xAB\x7B\x02\x41\x44\xA7\x59\xF5\xC6\x54\x82\x72\x69\x87\x94\xC4\xD4\x0B\xDA\x21\x0D\xD7\x5F\x2D\x90\x48\x08\x6B\x7E\x10\x3F\x86\xCD\xF2\xAD\x99\x3E\x79\xE4\xC5\x20\x3D\x9C\x3E\x4B\x9B\x43\xEA\xF0\xCE\x62\xC9\xE7\xA4\xAB\x9D\x7E\xB4\x6B\xA2\xF0\x54\x47\xB0\x1E\x40\x06\xFA\xF4\xA2\x78\x04\x12\xE2\xBA\xF3\x06\xF1\x0B\x66\x68\x85\x44\x38\xE4\x91\x27\x44\x81\x73\xAF\x77\xB7\xC6\x4F\x73\x11\xAB\xE9\x23\x78\xD3\x7C\x88\xD3\x4F\x68\x0E\xAB\x50\x11\x53\x3C\xDA\x0B\xC0\xA7\xB5\x8B\x19\x4D\x51\xE9\xF5\x7E\x57\xB2\xB9\x4E\xA0\xC4\x66\x21\xBC\x8B\xCE\xD7\x39\x32\x5E\x72\xAE\xBD\xEE\x2A\xFE\x7B\x11\xA9\x18\x60\xA0\x73\x64\x62\xB9\xF9\x22\x63\x7B\x22\xB2\xE7\xE7\x6E\x76\x89\xE4\x09\xFA\xD5\x56\x2E\x5D\x94\x2E\xA1\x0D\x47\xDF\x0D\xBE\x73\x89\xC2\x02\x25\x07\xB1\x04\xD8\x3B\x2E\x73\x69\xFB\x6D\x20\xEC\x0D\xBE\x81\x8F\x3B\x36\x1A\x26\xAE\x58\xCC\x43\x89\xF3\xAC\xFD\x08\xF6\x8B\xBC\xB1\x96\xDE\xF0\xB6\xA9\x5D\xB1\x87\xDA\xD3\xFC\xE0\x89\xB0\x45\x12\xDE\x22\xA8\x5E\x92\x30\x11\x26\xAE\x25\x3A\x4A\xDC\x89\x1D\x60\xB0\x6F\xF0\x16\x49\x86\x2D\xC2\xD5\xC4\xBA\x93\x7B\xDD\x29\xDA\x22\x30\xF8\x27\xD3\x63\x83\x46\x7C\x72\xAF\x9B\xD3\x16\xB9\xCB\xDD\xE5\x92\xB8\x45\xE6\xA3\x2D\x72\x17\x26\x15\x45\x55\xEE\x1A\x8E\x8D\xBB\x86\x2D\x32\xAF\xDC\xDC\x6D\x62\x8B\xDC\xAC\x77\x73\x3C\xD3\x2F\x36\x78\x79\x13\x73\xDE\x9D\xE2\x95\x9B\xBB\x93\xEE\xD4\x68\x8B\x6C\xF0\x16\x49\x78\x8B\x60\x7F\x9D\xA4\x2D\x72\x32\x6C\x91\x86\xE6\x7F\x51\xB8\x6A\x91\xBB\x19\x36\x46\x5A\xB1\xBA\x97\x04\x49\x7D\x74\x6A\xFC\x92\xE1\x92\x3A\x96\x79\x1C\x07\x5A\x7D\x37\x47\x45\xE2\xCB\x15\x1A\x61\x08\xB6\x4B\x46\x85\xF1\x53\x89\xB5\x4B\x10\x21\x47\xF2\x9A\x5E\xE4\x12\x87\x67\x5D\xB2\x28\xC4\xA4\x85\x14\xEF\x67\xBB\xD2\xD7\x4F\x75\x33\x29\x2D\xA1\x5D\x7E\xA5\xAB\x91\x8D\x33\xEB\xB7\x89\x53\x3C\xDA\x90\xCE\xED\x15\x12\xA5\xBA\x99\xAB\xB7\x38\x84\xCF\x95\x38\xBE\x5D\xBD\xD5\x15\x12\x78\x67\x5D\xF6\x58\x63\x1C\x3B\x19\xD8\xF6\x60\xB9\x34\x4A\xCE\xA7\x58\xC5\x6A\x05\x49\xE5\xE6\xD2\x16\x4D\x05\x92\x7A\xD9\x9E\xCD\xD1\xBF\xFD\xDC\x6A\x3A\x3C\xFC\x91\x47\x71\xBE\xBF\x06\xFF\x31\xAA\x54\x3F\xED\xF5\x3E\x1F\x83\x05\xFF\x4E\x67\xA0\xD7\xFB\x30\x3E\xC1\x1B\xE7\x72\x52\xAC\x49\x22\x64\xCE\x8F\x17\xD2\x94\x70\x48\xD2\x23\xC0\xEE\xFF\xA0\x0E\x60\x0A\xF1\x1C\xD1\x3C\xC7\xC6\xD3\x4E\x02\xDB\x29\x05\x1D\x51\xCE\x13\xBA\xA5\x1C\x9F\x27\xF8\x61\x88\xA0\xF3\xD5\x25\xEC\xD2\x78\xBF\x87\x65\x9C\xEF\x47\xEC\x5F\x15\x03\xEE\x36\xAA\x69\xC3\xD3\x1B\x87\x65\xFE\xBA\xAA\xFA\xAE\xDC\x66\x47\xFA\xAA\x5E\x95\x1E\x3C\xE3\x6E\x68\x7F\xAD\x90\x7C\xFF\x94\x95\x36\x11\x40\xA1\xB9\x41\x32\xF8\x55\x18\x2A\x12\x12\x13\x74\x57\xF8\xD3\x87\xFE\xF4\x33\x5D\xB9\x63\xCE\xE0\x99\x02\x35\xA1\x3F\x29\xF7\xA8\x7B\xE8\x78\x3A\x67\x5B\x1C\x82\x57\xFB\x2E\x15\x50\x0B\x64\x52\x07\x5B\x75\xE6\x52\x9F\xD0\xFF\x2F\x6F\xB5\x3F\x19\x08\x12\x98\x19\x70\x0C\xFD\x38\x9B\xDF\xD9\x88\x75\x06\x07\xDC\x27\xC0\xC5\xBF\x70\xC3\xC2\xED\xF1\x85\x1B\x76\xC7\x9C\xA9\x11\x06\x96\x32\x27\xBC\x18\x2F\xE7\xF4\x97\xC4\xE5\x71\x8D\xCB\xE0\xC6\x3C\xEA\x71\x78\x2D\x4A\x76\x93\xA3\x32\x60\x36\xBA\x81\x2B\x87\xD2\x0D\x2E\xBB\xD0\x70\x99\xF8\x87\x9A\x73\x92\x07\x0F\xB9\xBB\x72\xB3\x17\x45\x7B\x29\xFD\xE7\xD5\x6E\x88\xD6\xEA\x69\x4E\x0A\x71\x26\x74\xA9\x2B\xF7\x68\xC0\xB5\x18\x66\xFD\x17\x94\x57\xED\x3F\x86\x83\x21\x43\xDC\x06\xF5\x9D\x81\x45\x6C\xFB\x63\x8C\x40\x48\x64\x3D\x3B\x07\x1B\x54\x22\xB9\x99\x09\xAA\xD3\xE1\x5E\x59\x93\x52\xD0\x4E\xC6\xBD\xAE\xF6\x98\xE1\x70\x0F\x60\x54\x0C\x6D\x72\x64\xE4\x22\xA5\xFD\xA0\x5D\xE1\x3F\xA9\x76\xB7\xE8\x48\xE1\x28\x26\x12\x6E\xF3\x3D\x71\x84\xCA\x1E\xCA\x58\x2A\x84\xDC\xF7\x6B\xB2\xAE\x1C\x6B\xC6\x3C\x0E\xDB\x98\xDE\xE4\xD9\x76\xC4\x31\xF0\xC8\xE6\xFA\xC3\x1F\xFC\x56\x14\xEE\x2B\xFD\x77\x5C\xBB\x7E\x4D\xED\x36\xA9\xB7\xC8\xB1\xC1\x57\x6F\xDB\xBF\xF7\xCD\xE8\x54\x4A\x7C\xE7\x93\xEF\xFB\xD6\x6B\xAA\x53\x5E\x3D\x82\xCA\x1C\x4B\x3F\x59\x97\x35\x5C\x2F\x76\xF2\xB3\xA9\xFC\x27\x3F\xFF\x0E\x7A\x49\x88\x8C\xCB\x2E\xD0\xBD\x00\x17\xB1\xC0\x60\x81\xB5\xF4\xA8\xEF\x14\x0B\x7F\xC3\xFD\xD1\x16\xD8\xE5\x23\xC4\xE1\xF6\x07\x38\x05\x5F\xE4\xD4\x8B\x38\xD9\xCA\x5D\xA0\x4C\x97\x2E\x0B\x5D\xA6\x83\x71\xDA\x78\x55\xC9\x3A\x11\xCF\xBE\xC7\xB6\xE7\x88\x53\x2C\x52\x46\x4E\x4A\x5D\xE5\x74\xFB\x8F\xAF\xB3\x1F\x20\x1D\x8B\x2A\x2E\xAD\x5E\xD1\x3A\x5D\x75\x18\x08\xD0\x0B\x2F\x98\xE2\x04\xDF\x0B\x8B\x84\xD8\xA1\x98\x65\x3D\x48\x63\xBC\x6B\x0C\xDC\x16\xB2\x65\x52\x76\x26\x65\xFE\xC6\x8D\xFF\x3D\x44\x2B\xBB\x94\x11\x1F\xAC\x4B\xF7\x60\x17\x48\xA5\x4C\x74\x16\xF6\x89\x95\xD0\x04\x54\xB6\xA7\x51\x1E\x5D\x41\x38\x60\xE5\x34\x48\xDF\x33\x3E\x40\x24\xFF\xC1\xE4\xED\x92\xEA\xE7\x53\x9D\x1D\xA3\x9A\x18\xAF\x63\x44\x66\x1D\x84\x94\xD4\x1B\xC6\x36\x85\xF7\xC8\x11\x57\xF0\x57\xFB\xD7\x69\x38\x4B\x95\xB3\xED\xFF\x76\x3D\x84\x05\x2C\x0C\x07\x6B\xA3\x86\x6F\xCD\xF1\xE8\x5A\xD2\xFB\x35\x9A\x4A\x86\xA6\x5A\x6E\x0A\x96\x13\x36\xA2\x0D\x4D\x15\xD3\xA6\x36\x2F\x85\x18\xE5\x70\x64\xEC\x07\x8F\x77\x1E\xEC\xE3\x12\x1C\x15\x15\x99\x74\x69\x18\x26\x86\xD2\x87\x61\x14\xD4\xFD\xC2\x9B\x7E\xCD\x30\x24\xAC\x3E\x0E\x43\x12\x1F\xD6\x36\x86\x81\x14\xDB\xA2\x62\xAF\x19\xC8\xD0\xD8\x66\x68\x6C\x74\x10\xA5\x7E\xE6\x67\xF1\x58\x49\xDB\xBF\xFB\x3C\xFB\x6D\xD2\xF6\xEF\xB0\xB3\x83\x91\x8D\x46\x6E\x2E\x71\x8D\x14\x90\x8E\x49\x4A\x2E\x7C\xA8\x8A\x67\x88\x53\xD3\x80\xBA\xF2\x31\x89\xC6\x28\xFD\x51\x1F\x7B\x84\x12\x54\xB9\x3C\x5D\xE1\x66\x9E\x00\xCE\x88\x96\x38\x55\x81\x81\xC9\xC7\x43\xA7\x99\xBE\xD4\x48\xEE\x64\x7C\xF1\xE8\xA5\xE0\x74\x8F\x1E\xF3\xD2\x4D\x7E\x69\x1B\x5F\xCA\x91\xB7\x65\x78\x6D\xB1\xFC\x5A\x57\x62\x4B\x3C\x2A\x00\xA0\xF8\x12\x1A\x1C\x6F\xCD\xEA\x23\x1A\x61\xD9\x03\xD6\x6B\xB7\x66\xCF\x49\x5A\x0C\xEF\x39\x3B\xC0\xBC\xCA\x31\xE5\x14\x1F\x50\xE2\x5C\x93\x52\xDD\x5A\x40\x8F\xFA\x9D\x50\xAA\x2D\x71\x16\x6C\xDB\x22\xF4\x01\x9B\x31\xDC\x26\x87\x16\xDF\xC6\x87\x16\xC7\xA4\xF3\x59\x64\xC2\x66\xF4\x47\xD5\xAB\x5A\x42\xA0\x61\x68\xC1\x39\x0A\x03\x4D\x32\x04\xE0\x6E\xC4\xDA\x72\x76\x38\xCF\x81\x90\xCB\xF1\xE9\x3F\xC9\xA2\x03\x04\xBA\x35\x07\xB3\x19\x0F\x97\x36\x82\xE4\xC5\x87\xE1\x9A\xC9\x70\xD3\xC9\x70\xD3\xF1\x70\x53\x0C\x97\xF8\xCB\x06\xB1\xCE\xDB\x1A\x71\xC2\x23\x4E\xE2\x88\x55\xF5\x9B\x33\x5D\xAC\x8B\x63\x1D\xEC\xCE\xE9\x85\x20\xC0\x64\x3C\x60\xF8\xD4\xCD\x43\x9A\xB9\x59\x3A\x4C\x60\xEC\x33\xBC\xE2\x0F\x15\xEB\x6E\x88\x5E\x41\x16\x65\x32\x89\xB4\x83\xE5\x64\xBD\x20\x53\x4F\x04\x99\x9C\x05\x99\xBA\x2B\xE2\x94\x89\x08\x53\x87\x70\xEA\x9C\x3B\x50\xF0\x9C\x98\x41\x74\xE9\xCA\x89\xF0\xD2\xCD\x44\xCB\x11\xD1\x25\x9F\xDC\xE0\xF2\x3D\xDC\x80\x11\xE4\xA4\xF8\x8E\x0B\xC2\x2F\xFF\xCF\x15\xFE\xF1\x7E\x7E\xD9\x04\x53\xCF\xED\xFE\x2B\xE9\x3F\x33\xF9\x92\x29\xA5\x72\x55\xA8\x4A\xA5\xC0\x8D\xBF\xA1\xAE\x34\x95\x2B\xDA\x77\xD0\xA1\x49\x9F\xFC\xE3\xBD\x9F\x5D\x6E\x48\x7A\x2E\x18\x30\x20\xAD\x9B\x8A\x54\xFF\x19\x4B\x18\x9F\xD5\x80\x57\x98\xE1\x56\x9C\x9E\xE8\xFE\x0C\xD6\xD6\x84\x83\x36\x72\x11\x3A\x81\x97\x96\x87\x47\x00\x2F\x2E\xC1\x89\x30\x44\x43\x2C\x62\xB8\x35\x7B\xE8\x8A\x1E\x0D\xC2\xC8\xE5\x12\x86\x9E\xB8\x59\x43\x57\xFB\xD0\x8C\xE9\x97\x1B\x81\x35\xF2\x4A\x93\x3B\x0D\xEB\x27\x46\x51\x62\x14\xE0\x5D\xB2\xE4\xCE\xB4\x3F\xCE\x07\xFE\xA5\xA6\x90\x45\xF7\x9F\x57\x9C\xE6\xBE\xEE\xA6\x60\x08\x08\xFA\x54\xC2\x67\xD0\xE3\x7D\x97\xD2\xA4\x71\x32\xD8\xC7\x89\xF4\xE6\x7F\x99\xE7\xDB\x1E\xFB\x4F\xAE\x33\x39\xA7\x1C\xF5\x73\x99\x27\xD4\xE5\xFE\x83\x1C\x93\xEA\xDF\x77\xD4\xFB\xEC\xF2\x16\x8F\xA8\x80\xEE\x5A\xD0\xC8\x4A\x10\xD0\x85\xA6\xC4\x8C\xA2\x53\xAB\x1D\x7E\xB4\x51\x75\x55\x51\x73\x5F\x50\x17\x9B\x9C\x86\x9F\x8F\xD7\xB5\xAC\xA4\x95\x30\x29\xC5\xF2\xA4\x84\x11\x7E\xFC\x88\x73\x47\x47\x63\x75\xB3\x30\xC6\xFC\xD8\x7F\x72\xBD\xE2\xA9\x2A\x3A\x22\x99\xCE\xF2\xE3\x05\x35\xFA\x45\x35\x11\xE7\x78\x76\x7B\xF3\xE5\x93\xC3\x05\xAB\x74\xF2\x1C\x5E\xC9\xCF\x16\xF4\xAC\x65\xCD\x92\x9F\x2D\xE2\xB3\x16\x51\x6A\x4E\xF5\x61\x9E\x80\xED\x30\x0B\xDF\x72\xFA\xD6\x84\x6F\xC0\x42\x28\xC3\x37\x54\xE3\xCA\xC3\x37\x94\x5B\xAA\xC3\x37\xA4\xF1\x57\xE1\x1B\x4A\xCD\x05\x7C\x7F\xB5\x50\x2E\xD6\x42\xC4\x22\x90\x5A\x03\x2A\x84\x74\x1A\xC4\xBA\x11\x9F\x9B\x44\xC6\xA6\xD5\x3F\xAF\xAC\x3E\xD2\x57\xDB\x55\x3E\xEB\x12\xC6\xF6\x4B\xA5\x44\x07\x4D\xE4\x43\x74\xA5\xFD\x7A\x12\x3F\x5E\xA4\x3D\xFD\x2B\xCC\x70\x6D\x60\xB8\xD9\xA0\x3D\x8D\x53\x8E\x72\xBF\xC9\x7C\x6E\x94\x64\x55\xC4\xAB\x65\xB4\x21\x17\x7E\x33\x40\xF4\xF0\x31\x96\x45\xCE\xC8\xE7\x05\x97\x25\xA8\x91\x32\x6E\xCE\xBB\x1A\x1B\xAE\x16\x03\x27\xFD\x19\x2B\x33\xDA\x55\xA4\x98\xC2\xB2\x0F\x24\x53\x59\x97\x86\xFB\x12\x25\xA9\xD9\x48\x92\xE2\xD8\x91\x5A\x2A\x79\x32\x93\x4F\x0E\x16\x73\x52\x4E\x5E\xFE\xDD\xEB\xD7\xD4\x62\x83\x3E\xFE\xC6\x1F\xD0\xC7\xD6\x6B\xB6\x99\x40\x26\x11\x1C\x9A\x09\xE7\xCD\x58\xE4\x44\x01\x1C\xFF\x2F\x55\xCC\xD4\xCA\xF8\xA7\xAE\xF2\x67\x2F\x1E\x0C\xF9\x81\x70\x00\xED\x43\x07\xA8\xF9\x74\x2D\xCF\x2A\x8D\x07\x47\xF5\x9A\x4A\x96\x0B\xE1\xE8\x29\x3D\xA9\x69\xD3\x53\xC9\x65\x72\x1E\x95\x21\x3A\x24\x5B\x3E\x8F\x70\xDA\x95\x74\x16\x69\x56\x0F\x2E\xB2\xE9\xA6\x61\xE2\xD1\x80\x24\x34\x74\x8B\xBA\xD4\xD4\x51\xFD\x81\xDC\x55\x4A\xB0\x48\xC5\x51\x73\x85\x84\xC5\x02\x51\x08\x1D\xB7\x97\xBB\x13\x18\xF4\xAF\x60\x19\x30\x84\x07\x2F\x72\x88\xC3\x25\xFC\x49\x2E\x71\x85\x86\xCD\xC5\x49\xA6\xE2\x53\xAE\x76\x65\xEF\xAF\xF6\x8B\x4D\x9F\x88\xF3\xB2\x74\x59\xDF\x11\xB7\xA6\x59\xC1\xF3\xC4\xB1\xE8\x0B\xCD\x07\x8A\x51\xB1\x84\x4C\xCF\xB9\xCD\x37\x41\x6D\x38\xD1\xD0\xAA\x93\xE0\xA3\xF9\xC5\xEE\xC0\xBF\x8A\x6F\xAC\x1E\xE3\xAD\xCE\xB6\x1F\x23\x1A\x9C\x9B\x7C\xA6\x2B\x87\x68\xBA\x84\xB4\x61\xEB\x5F\xFE\x3D\x5A\xD8\xCA\xB5\x8B\x4D\x51\x23\x61\xEB\x62\x2B\xFA\x5D\xB0\xA2\x8B\x1A\x79\x17\x71\x96\xBB\x58\x8D\x3C\x41\x6A\x64\x4E\x7F\x06\x35\x32\xAF\x5C\x45\x6A\x64\xE5\x36\x69\xA1\x36\x1B\x55\xA7\x55\x0C\x67\x9E\xF9\x6B\x92\xC7\x49\x54\x7D\x82\x2D\x31\xA0\xA3\x6A\x90\xDC\x36\x24\x2D\xF6\xFF\xBE\x0E\x5C\xB7\x0D\x89\xA9\x29\x45\x19\x56\x2E\xE3\xBE\xB0\x5F\x30\x73\xD5\x2E\x0D\xE3\x75\x4A\xB9\xEA\x75\x4A\xED\x02\x77\x91\xBA\x50\x8A\x53\x90\x0D\xD3\x0F\xA9\x73\x36\xD4\xC6\x85\x11\x29\x09\x86\x84\x9F\x60\x43\x42\x11\xC2\xEC\x92\xAE\x1C\x75\x0C\x12\xD6\xD9\x8B\x9C\x09\xC6\x44\x89\x12\x13\x51\xA0\x2B\xC5\xE8\x42\x9F\xE7\x4B\x6D\xCE\x50\x5A\xA1\x2B\x85\x1E\x1F\x85\x36\x9A\x08\x18\x17\xAC\x7F\xEC\x81\xB3\x4E\xBB\x7A\x6F\xC7\x14\xF5\x8C\x84\x82\x44\xB0\x08\x13\x57\x2E\xB8\xD2\xCC\xA6\xCB\x5C\xC9\x1B\x38\x39\x60\xB9\x1E\xCC\x13\x43\x84\x2C\xB9\xC8\xD0\x17\x89\x0F\x61\x24\x42\x7A\x90\x35\x24\x52\xD9\x89\xCF\x82\x37\xBB\x13\x22\x0C\x20\x48\x06\xC3\xC0\x2B\xCA\x50\x7A\xCD\xCD\x58\xB3\xAF\x6D\x35\x22\x8D\x13\x56\xC8\x62\xF3\x42\x63\xDC\x26\x2C\x12\x26\xAC\xF6\x09\x9A\xEA\x13\x0D\xD2\xA3\xB3\x4A\x42\xC4\x79\xAB\x44\x73\x0F\xEA\xE6\xD7\x62\x82\xD2\x0E\xB9\x5E\x27\x9D\xD4\x1B\x3D\xC5\xAF\xC4\xF3\xBC\x0A\x01\xE4\x36\x65\x8B\x51\x0A\x2E\x5E\x88\xEF\x16\x6F\x19\xB1\xF4\xAC\xFA\xB5\x1C\x2C\xBD\x5C\x63\x77\x30\xCC\xD2\xED\xC0\xD2\x83\x46\x93\x30\xAE\xE7\x43\xEE\x9D\x2F\x76\x29\xAF\xBA\x7D\x91\xD7\x7C\x3B\xAE\xF9\x1B\x62\xC8\x2A\xCA\xD4\xC4\x63\xC6\xA5\xC4\x68\x73\x92\xB5\x0C\x18\x6D\x28\x97\x94\x23\xCF\x80\x5D\xF7\x6C\x39\xCC\xCE\x2A\xEB\xFF\x8A\x84\x21\xD3\x59\xF1\x57\xBD\x11\x6B\x19\x67\x39\xE1\xB4\xC2\x66\xAD\x07\x6B\x99\xE4\xC8\xCE\x59\x04\xD6\xAC\xEF\xE0\x28\x84\xAB\x10\x35\x40\xAE\xE2\xAD\x09\x62\x2F\xC4\x06\xC9\x27\xDF\xA4\x2D\x0D\x64\x0A\xA0\x6F\x5C\x65\x37\x56\xE6\x8A\xF6\xE5\x71\xF2\x9D\x62\xDF\x55\x78\x1D\x3C\x3D\xA1\xEC\x8C\x33\xED\x87\xAF\x87\x83\x82\x4E\x5F\xCD\x85\x2B\xD3\x91\xF6\xA3\x5C\xBA\x27\xA4\x98\xB2\xDA\x91\x73\xB4\x39\xA4\x40\xCE\x07\xAB\x63\x05\x3D\xE4\x84\xD2\xFF\xA7\x9A\x58\x11\x8C\xA3\xAC\x84\xD5\xC1\xD2\xC3\x26\x50\x22\x77\xD1\xAA\x0C\xB0\x4D\xC2\xE4\xB3\x2E\x2C\x71\x90\xD5\x70\xFC\x57\xF1\x58\x4E\xB0\x03\x5C\xB6\x17\x1F\x21\x3E\x5B\xC0\xCA\x87\x40\x17\x75\xA9\xB1\xC7\x6A\x13\xAE\xF6\x1F\x3E\xEA\xE7\x7F\x29\x69\x56\xFF\xE9\xA6\x69\x0C\x3E\x59\x55\xB9\xDA\xFF\xB7\xFD\xFC\x64\x51\xD7\x75\x52\x8F\xFE\xE5\x8C\xA3\xB6\x7F\x01\x31\xEF\x24\x0F\x2C\x92\xBA\x60\x68\x33\xE6\x88\x89\x6F\x0F\xF0\x9B\x4B\xFC\xEF\xE3\x3A\xFC\x64\xC9\xF4\x9E\x3F\xE2\x27\x73\x18\x6F\x0E\xE4\x1E\x3B\xBD\xE7\xE3\x7C\x0F\xF6\xC7\xCB\x2A\xDC\x64\xB8\x03\x30\x44\x00\x6B\xD2\xA5\x17\x9A\x94\xF1\xC3\x66\x2E\xF1\xFA\x60\x91\x20\x66\x9C\x36\xFD\xDA\x04\x13\xC4\x0E\xB4\xB0\xB5\xC4\x2C\x75\x04\xD1\x97\xFE\xE8\xE9\x03\xBC\xDE\x72\x36\x0C\xB4\xEF\x7B\x2C\x69\x9B\x75\x02\xEC\x4C\x7A\x49\xC5\x09\x8C\x81\xAA\x6B\xA1\x6A\x58\x93\x99\xF2\x8A\xDE\x15\xD0\xA2\xC5\xA6\x68\xC6\x22\xDB\x75\x2D\x98\x00\x4E\x0B\xAE\x16\x64\x1F\xE3\x75\x44\x4B\x50\x8C\x0E\xD8\x25\xA4\x95\x91\xEC\x01\xC9\x39\x81\xA4\x4A\x5F\xBF\x8E\xBF\xD2\xC7\x5F\x67\xC3\x36\x24\x5C\xFA\xFE\x17\x1E\x6E\xA4\x9A\x71\x67\x9C\xBE\xBC\x40\xBA\xB8\x14\x0C\xF2\xFA\xE9\x05\x5F\xDC\xAF\xFE\xED\x19\x6B\x8F\xF4\xD5\x62\x0C\xD3\x1A\x23\x61\xFD\x03\xBD\x2F\x7A\x3E\x69\x10\x01\xFB\x40\x8C\x8E\xF5\x37\x34\x5B\x65\xB4\x77\xFB\x5D\xB2\xB5\x63\xEE\xE3\x5B\xB6\x1D\xDC\x3D\xFE\x81\xDE\x99\x2D\x9A\x88\xBB\x49\x96\x26\x76\x95\x9D\x33\x67\x88\x91\x9E\xB3\xAD\xCB\x88\x43\x2D\x94\xCB\x5E\x64\x75\x7B\x94\xCC\xD3\x25\x1C\x3D\x15\x4D\x08\x2E\xF1\xEA\xF5\x1A\x01\xD2\x82\x8F\x7C\x56\x29\x31\x99\xF8\x57\x54\x1F\xFE\xDB\xFE\x04\x6D\xB5\xF6\x1F\x30\x22\x99\x05\x34\xEC\x65\xD8\xED\x11\xA3\x8B\x20\x59\x07\xA0\xC1\x5B\x3E\x0C\x55\x1E\x41\x84\x9B\x97\x25\x6C\x18\xF1\xF5\x0F\x20\x10\xD2\x6C\x31\xD6\x15\xD6\x9D\x8D\xEF\x8A\xAB\x71\x87\xBA\x6B\x31\x38\x6F\xC8\x73\x08\x86\x67\xD4\xA2\xA6\x45\x55\x5E\x3F\x0B\x54\x89\x2E\xDF\xEA\x10\xDF\x6E\xD9\x15\x0D\x34\x9E\x7C\x8B\x36\x76\x3B\x88\x8B\xDA\x9B\x7D\x79\x97\x94\x68\xF4\x44\x62\x23\x7E\x63\x10\x97\x8D\xD1\x6C\x9B\x56\x18\x0F\x06\x71\xA9\xCB\x85\xF1\x58\x3E\x2D\xD8\x7C\x21\x9E\x97\x54\xE6\x06\x67\x6D\xCB\x5F\x36\x71\x0C\x6F\x72\x4E\x94\x4C\xDC\x70\xD1\xA9\x3D\x54\xBA\x23\x26\x93\x56\x88\x85\x76\x79\xFB\xC2\xF5\x38\xF1\x1C\xB6\xCF\xCF\x4D\xBB\xC9\xC1\x69\x1C\x3E\xFD\xC3\x02\x0C\xC3\xF8\x1A\x17\xD8\xBB\xC9\xE2\x79\x31\xF0\x59\x9F\x87\x5C\xB5\x2F\xF6\x7F\x2E\xBF\x57\x29\xFF\x0D\xFD\xFC\xBF\x4B\x95\x52\xDA\xCE\x0A\x36\x80\xAC\x9A\x47\x8E\xF9\x97\xD7\x2A\xB9\x99\x69\xA5\x32\x58\xDA\x95\x95\xA8\x6F\xB9\x0E\xF5\x9A\x75\x98\x7F\x49\xEB\xF0\x17\x97\xD6\xA1\x06\xEE\xB8\xDD\x36\xA7\x07\xB9\x40\x49\xFA\x28\x84\x00\x3E\xA4\xA1\x23\xF0\x93\xD6\xB7\x7D\xFB\x71\x1C\xA1\xA7\xEB\xBF\x18\x0A\xFA\x5C\x03\x02\xF6\x88\x83\x26\xC2\x3E\x15\xB3\x4F\x3C\x1A\xD2\xC2\xD8\xFF\xA6\xCE\xD9\x4B\xA1\xC9\x57\x54\x2F\x47\x6D\xB5\x34\x3C\x1E\xDC\x42\xD5\x9B\x42\x0E\x92\x53\x72\xCC\x5D\xA7\xE4\xAE\x7F\xC8\x02\xC0\xB1\xF7\x9D\x44\xF0\xEB\x69\x9A\x70\x7F\xBA\x8F\xFD\x3E\xE9\xE7\x4E\xF9\x76\x7F\xCB\x4D\x13\xD9\x8E\x6B\xE6\x44\x15\x42\x7F\x4E\x77\xF4\x1C\x73\x1E\xE5\xAF\xBD\xAC\x0E\x90\x5D\x3B\x6D\xFE\x94\xCB\x97\x9B\x55\x3B\xE6\x34\xE6\x63\x7C\xDF\x86\x5B\xCE\xEB\x3B\xE6\xFD\xAD\x0C\xF7\x07\x6F\xDA\xCB\x8D\x6A\x58\x3A\xA1\xB9\xB4\xFD\x29\x9E\xA1\xD7\xC8\xA5\x6D\x73\x29\xAC\x4E\x5B\xF9\x02\xE8\x0E\xBE\x64\x3C\x36\x7B\x8F\x6D\xBB\x0C\xCB\xE5\x72\x31\xC7\xB5\x18\xE7\x51\xDF\xE9\x1D\xB3\x29\x28\xE4\xF6\xAC\xFA\x0B\xCC\x7F\xE4\x07\xC4\x51\x43\xC7\x79\xC3\x45\x99\x99\xAB\x3D\xF3\x30\x88\x6D\x15\xA3\x30\xD7\x15\x23\xC2\x19\xAE\xEE\xF1\x40\xEF\x93\x03\x67\xB6\x58\x54\x4E\x3D\x1B\xFF\x5C\x05\x54\xF2\x02\x84\xE0\xB2\x17\xE9\xC5\x97\x50\xD3\x50\xEF\x91\xD6\xF0\x08\x0F\x24\x30\xEC\xBA\xFD\xFB\xA0\xD1\x39\x48\x62\x34\x04\x9C\x29\x5D\x75\xA1\x29\xD0\x08\x75\xEE\xF3\xEA\x62\x53\x38\x58\x1B\x48\x23\x72\xFA\xAC\xD2\x6C\xB0\x4B\x7C\xF9\x30\x8C\x7E\x61\x4F\xB0\x61\x51\x9F\x55\x46\x4C\xA0\x65\x05\xC3\x03\x90\xD8\x11\xF1\x6A\x01\xF5\xF4\x4F\x7E\xE9\x75\x57\x9A\xD2\x35\x4E\xED\x36\x25\x90\x82\x10\x4C\x35\x7B\xCA\xA1\xB8\x38\x7B\xE5\x00\xA2\x4E\x4C\xBE\xE6\x77\xD1\xB9\xCD\xB0\x42\x88\x27\xA9\xFD\xBD\x7C\x5A\x6B\x64\x42\xF8\xA3\xA7\x91\x77\xE0\xF2\x85\x22\x51\x88\x86\xAE\xF6\x48\x12\xAE\x30\x7A\xC6\x35\x9F\xD3\xAC\xE0\xB5\x85\xFF\x82\x5A\xD4\xFE\x68\xC1\x60\x6D\x34\xD0\x2F\xA8\xDD\x26\x8B\x27\x5A\xDD\xB7\x3F\x2D\x4A\x20\xAD\xFF\x23\x3C\x29\x60\x40\xF3\xB3\x4A\x2D\xEA\x3A\x1F\x91\x77\xE2\xAF\x99\x81\x3D\x68\x8E\x1B\x80\xCA\xFD\x9B\x90\xF5\x27\x04\xDC\xFA\x13\x4E\xFB\xDF\x54\xBB\x5B\x81\x11\x2C\x6D\x24\x33\x90\xE8\x55\x22\xD1\x2A\x32\x92\x11\x0F\x32\x03\x8B\xE6\x97\x7D\xFC\xA8\x9F\x9F\x50\xA9\x49\xC7\xFF\xEC\xF2\xEE\x39\x79\x9B\xEF\xDC\xA8\x9C\xF6\x9F\x53\x17\x1B\xBB\xBC\x4B\x6F\xEF\xF9\xB9\xBC\xB7\x69\x7F\xE0\xA6\xF7\x35\x95\x50\x3E\xF6\x8A\xD0\xFE\x8E\x69\x63\xE0\xA4\x9E\x9B\x52\x73\x38\x7E\x3B\x04\x67\xF0\x14\xFB\x9F\xC1\x47\xEB\x34\x35\xA0\x26\x0D\x60\xFB\x23\x06\x12\x98\x79\x3E\xA1\xFF\x0F\xE7\x4B\x7A\xCB\xF3\x25\x5D\x73\xBE\xE4\x5F\xD2\xF9\xF2\x9A\x75\xE7\xFC\x42\x87\xCC\xB6\x40\x38\x08\xBC\x21\xDA\x21\x21\xF4\x55\x35\x02\x92\xEA\xB4\xDF\x14\x5D\x41\x6C\x87\xC3\xA5\x5C\x6C\x87\x09\xB2\xD1\x5C\x1E\x02\x4F\x98\xAF\x6D\x9B\xD3\x2E\x71\xA9\x58\x7B\xAC\xCA\x90\x14\x34\x39\x77\x1A\x92\xB5\x18\x99\x2C\x90\x75\x19\x6C\x33\x10\x48\x66\xA2\x0C\x23\x1B\x71\xB0\xCD\x14\x63\xDB\x8C\x72\xC5\x22\x19\x32\x2E\x04\x5D\x61\x64\x9B\xC9\x9D\xA6\x69\xCC\x5F\xA7\x94\xD3\xB0\xCD\xE4\x15\x97\xF4\x45\xC0\x5B\xDF\x25\x22\x60\x0F\xDC\x98\xDE\x78\x1F\x7B\xEC\x2E\x34\x27\x46\x2C\x59\x4E\x69\xC8\x96\x13\xE6\x28\x29\x04\x30\xE2\x31\x70\x5C\x51\xB7\xB7\x75\x17\x91\xFD\xA2\x04\x02\xBC\xA4\xC8\x81\x40\x67\x91\x35\x6B\xA6\xAE\x11\xD0\x58\x39\x5E\x8F\x3C\xAE\x47\x22\xEB\xA1\x01\x35\x11\x54\x20\xCE\x3F\x81\x57\x74\xDB\x9C\x66\xFB\x46\x20\x0A\x59\x07\x98\x4A\xD4\x8E\xA9\x49\x73\x24\x12\xF8\x82\x98\x96\x69\x3B\x7D\x5E\x45\xDA\x39\x96\x10\x91\x94\x5A\xE1\x0B\x71\xBE\x39\xEE\xB9\xC4\x7C\xFE\x52\xC7\xBA\xC3\x8B\x08\x9D\x54\xCB\xE7\x12\x62\x2C\x71\x06\xD4\x01\x9D\x2B\x9C\x4E\x6C\x55\xDA\x24\x55\x20\x82\x43\x02\xA2\xEC\x36\x4E\xA7\x44\x4E\xA7\x52\xB2\x69\xD0\x82\x1C\x4D\x89\x0C\x1F\x59\x47\x34\xF2\xF6\x45\x9C\x4D\x96\xF4\x5E\x4C\xC4\x66\x35\x32\xC4\x86\x8F\x0C\xC9\x06\x68\xB8\x20\x27\x2F\x72\xA7\xA5\xC0\xF2\x44\x36\x36\xB7\xB3\xF2\x68\x58\x08\x7F\x1B\x74\x4B\x67\x55\xA7\xD8\xDD\xAF\x42\xC2\x36\x94\x94\xD7\x84\xB4\xD1\x90\x69\x1A\x50\x99\x44\xDD\x1E\x69\x2F\xC9\xA0\xC6\x70\xF4\x74\x00\xEC\xB1\xDB\xE6\x8D\xB7\xAB\xF9\xD8\x11\xB0\xB0\xAE\x5E\xD1\x3A\x3B\xF2\x0C\xEA\xC6\x29\x47\xD1\xF5\x2B\x68\x6A\x8F\xF7\x74\x32\xB3\xDF\x46\x00\x34\x70\x32\x27\xC4\x32\xF9\x64\x86\x9E\x0C\x28\x50\x3A\x5D\x73\xDC\xCF\x87\x69\x2A\xC1\x31\xA3\x20\x8D\x4C\x8E\x6C\x33\x54\x6C\x82\x95\x27\x77\x99\x9F\x3D\xD5\x2F\x32\x84\xEB\x84\xA0\x9A\x94\x8E\xC7\x1C\xC1\x27\xF0\xC9\xA2\xFD\xD9\xE5\x51\x89\x74\xA0\x4B\x3B\x53\x9D\x52\x4E\x09\x76\x53\x1C\x6C\x42\x43\x24\x8D\x81\x3E\x19\xFA\x8F\xAD\xDE\x59\x4F\xB2\x4C\xF5\x14\xE3\x2B\x66\x96\xFB\x0F\xE9\xCB\x82\x6E\xF7\x8B\x57\xB1\x2D\x18\xE7\x23\x64\x07\x99\x80\xE8\x31\xAD\x18\x26\x16\x38\x89\x61\x7B\x53\xC3\x00\x1C\xE9\xB4\xAC\x5C\xC6\x16\x32\x94\xF1\x4A\x58\x79\x34\x2E\x99\xA0\xF3\xC2\x28\x06\x13\x58\xC2\xC9\x42\x06\x61\x76\x12\xA1\x0B\xB0\x1C\x54\x0F\x23\xB5\x34\x94\x47\x81\x85\x6E\xA8\x1E\x06\xCC\xD6\x12\x6F\x59\xCC\x60\x0F\x7B\xFB\x7E\xDF\x55\x10\x78\x66\x21\xC7\x6D\xE6\xDF\xDE\x77\xF5\x50\x36\x2C\x27\x31\x12\xA9\x40\x5E\xB9\x7C\x6F\x91\xD0\x37\x44\x44\x9B\x98\x52\x85\x90\x1C\x00\x2A\x31\xF2\x6F\x97\xB9\xE4\xB1\x45\xE3\xB2\x45\xE2\x1A\x86\x45\x76\x33\x84\xD4\x86\xC2\x61\xC3\xD3\xF9\xA8\xF6\x0D\x7F\x1F\xAA\xE3\xF0\xF7\xA1\x10\x0E\x7F\x6F\x7B\x98\x69\xE2\xF7\xCD\x3E\x64\xDF\xF3\xF7\xD3\xF4\xFD\xF4\xF0\xFD\x0C\x7D\x3F\x33\x24\x80\x39\xEE\xAD\xEB\xBB\x9C\xD3\xBE\x2A\xCE\xB9\xAA\xD9\x94\x34\x4A\xC4\x42\x9E\x89\x20\xCF\x8C\xCA\xCA\x25\x88\x89\xDE\x59\xAA\x29\xC7\x86\x33\xAC\x3C\x30\x77\x50\x20\x2E\x94\x8D\x43\x5A\x9E\x62\x2C\x9A\xC7\x38\xD6\xDC\xE5\x1C\xE6\xFA\xEA\x5F\xE2\xE5\xE1\x37\x6B\x5F\xF4\xED\xCF\x46\x6B\xFB\xBA\xAB\xFF\x34\x40\x1E\xDD\x21\xBE\x3B\xC4\x37\x10\x9F\xD3\x90\xB7\x7C\xDD\xF3\x9B\xB0\x20\x66\x6B\x28\x96\xC4\x71\xCD\xE6\xB0\xCB\x03\xE8\x26\xBD\x4F\xEA\x7A\x1F\xF2\x79\x8B\xAF\x79\xCF\x58\x0E\x09\x75\x6B\x4C\x82\x1C\xB4\x88\xE4\xE8\xF6\x9F\x0D\xE8\x11\x01\x61\x08\x11\x3F\xC0\xF3\x42\xC1\x25\x4C\xE5\xDB\xFB\xCE\xB4\x3F\xC7\xD1\x76\xE1\xA6\xDA\x8C\x93\x1F\xE1\x58\xA8\xD8\xDB\x50\x9F\x0B\xF9\x96\x89\xC4\x33\x8A\xFF\xBD\x46\x3A\xA6\xDE\x36\x9B\xE8\x13\x4F\x15\xD8\x3D\x75\x34\x97\x9E\xF3\x21\x9A\x3B\xAE\xE4\xE6\x72\xFA\xB8\x8C\x74\xD4\x56\x4B\x38\xCD\x7A\x88\x55\xBB\x53\x1C\xF4\x4E\x71\xD0\x3B\xC5\x41\xEF\x14\x07\xBD\x53\x4D\xF8\x0E\xC3\xB8\xC3\x30\xEE\x30\x8C\x2F\x3F\xC3\xF0\x08\x2D\x17\xFB\x15\x23\xE3\xD1\x16\x7B\xC7\x4B\x43\x45\x21\x84\xDC\x14\xFB\xBC\xF3\x19\x21\x12\x86\xA1\xE4\xF0\x60\xC7\xA8\xEA\xBF\x5F\x69\x21\x3D\xAE\x85\x14\x52\xD5\x83\x2A\x71\x0A\xD5\x3F\xBC\xDE\x7F\xBD\x56\x4E\xF9\xB4\x77\x8C\xE4\xE4\xB0\x42\x87\x07\xDE\xA0\xED\x37\x7E\x91\xBD\xD3\xA1\x77\x06\x8D\xD8\x2F\x47\x23\x09\x1A\xF9\xAB\x4B\x8D\xA4\xEC\x7C\x5F\x69\xE4\xB9\x7D\x9F\x1E\xB0\x7A\xA2\x87\x5A\x1C\xD5\x7D\x2A\x3E\x4A\x63\xDF\x31\x6A\x63\xE9\xE1\xF8\x78\x86\xF7\x9D\x9D\x3C\x90\x1F\xFB\x40\x8E\xBB\xFF\xBA\x80\x60\x29\x00\x38\xD9\x82\xFF\xBC\x6A\xF8\xD7\xBB\x9D\x7A\xE8\xE8\x9C\x39\xC3\xBF\x5E\x82\x49\x8E\xD1\x28\x05\xD9\xC9\x02\x67\xBF\x0F\x90\x5F\x9F\x54\xF1\xC7\x5F\x1E\x3E\xFE\xC6\xF0\xF1\x33\xF1\x63\xF5\x37\x00\xEE\x46\x7D\x79\xAA\xBD\x7E\x3D\x76\xEC\x2D\xE1\xCB\xB6\x79\x62\xA1\xB9\x1B\x4F\x04\x27\xD0\x00\xA1\xB3\x6D\x1E\x0F\x57\x1F\x5F\x77\xF5\x52\xB8\x7A\x69\xF5\x6A\xF5\x8E\x52\x97\xEB\xCD\x24\x6E\x5A\x94\x55\xB1\x96\x61\x27\x95\x50\x47\x65\x50\x6B\x17\x2E\xA5\xAC\xB3\xC8\xA5\x14\xC2\xBC\x17\xC8\x6E\x3E\x50\x1B\x99\x20\xCE\x9B\x46\xCA\xD4\xA8\xE6\x9C\x5C\xE2\x04\x16\x4E\x1C\xBA\xD0\x68\x2B\x78\x2C\x89\x00\xFA\x1C\xF6\x1D\x80\x92\x0A\x3A\xF1\xA4\xC6\x85\xF5\x1B\x08\x50\x76\x85\xDF\xD8\x47\x99\x5C\x4E\x0D\xB4\x5E\x3F\xEB\x3F\x7B\xE3\xC6\x8D\x7C\xBF\x8F\x75\x46\x33\x94\x34\xE4\x7B\x9D\xF5\x6F\xDB\x3F\xE0\xC2\xBA\x5C\xD9\x2D\xF4\xCA\xFA\xE4\xD9\xD0\xB1\x9C\x0B\xCB\x49\x01\x20\x89\xC1\xCA\x60\x05\x7D\xE5\x3D\xA3\x82\x8C\x50\x5B\x43\x84\x58\x1E\xAD\x6D\x00\x4A\x4A\x38\xDA\x84\x36\x0F\x97\x02\xA2\x91\x0C\x7D\xBA\x88\x12\x95\xA1\xC2\xDC\x63\x82\x96\x60\x25\x8B\xE2\xAC\x52\xAB\xEF\x63\xA8\x27\x62\x54\x17\x46\xDD\x8E\x45\x3C\x53\x14\xE9\x95\x02\x32\x98\xCC\x50\x0D\x8E\xB8\x7B\xAD\x07\x9D\x8D\x17\x69\x30\x12\x4A\x11\xE0\xA5\x16\x25\xAF\x88\x43\x1C\x0B\x00\x66\xA4\x12\x45\x13\x41\x73\x18\xC2\xAA\xC0\x22\x25\x3C\xB8\xF1\x9C\x9A\xD1\x9C\x26\xB7\x39\xA7\x88\x7F\x34\xA3\xFE\x04\x72\xA9\x42\x94\x2E\x3F\xCF\xB3\x97\x32\xAD\x70\xD6\xD1\x9A\xF6\x78\x2D\x2C\x37\x2B\x20\x9F\x9C\xC9\xC0\xA8\x43\xBC\x1A\xE6\xA2\xB4\x07\xDC\x0A\x97\x9E\x55\x09\xD7\x9A\xE9\xCF\x2A\xED\x6F\xC0\xC7\x53\x72\xC2\x24\x32\x82\x64\x25\x53\xA9\x29\x98\x28\x63\x95\xC1\x0A\x3C\xBB\x48\xB9\x3C\x24\x13\x6A\xBC\x21\x31\x4A\xEA\x73\x4D\x08\x74\x61\x6A\x39\x49\xD1\x23\x3B\xEE\x91\x8E\x3D\xD2\xCE\x52\x8F\x40\xA3\xB1\x47\x3A\xC4\x7D\x12\x17\x04\xCC\x6E\x25\x55\x13\x98\x04\x78\x3F\x31\x7A\x36\x13\xC1\xDA\x3A\x79\x61\xE3\xC9\xD2\xD0\xE7\xD3\x5E\xB9\x74\x5D\xA5\xBC\xCD\x45\x5A\x0D\xC5\xF8\xC6\xB6\x8D\x8A\x33\xDF\x8C\xA0\x4D\x46\xD8\xD6\x35\x4A\xFA\x0D\xA3\x93\xA3\xA9\x75\x96\xEB\xA4\x74\x76\x10\xBC\x13\x4E\xC8\x4F\xFA\x05\x9D\x1D\x49\x6D\x2A\xBF\xB1\x48\xD8\x4A\x82\xD8\xE6\x2E\xF5\xB3\x5D\x92\x5B\x70\x7F\xDF\xA1\xC0\x0B\xCB\x67\x8E\x0B\x68\x78\x73\xE0\x12\xA9\x83\xB1\xBB\x05\x6F\x3C\x02\xC6\x81\xD1\xAB\x91\xA3\xFD\xF1\x71\x94\x97\x96\x08\x2F\x1D\x42\xC1\xCA\x87\x1B\x00\x58\x20\xCA\xCE\x70\x7E\xC4\xCA\x53\xA9\x3C\x95\xC6\xA7\xF8\xE5\x33\x7E\xC8\x25\xFE\xEB\x0E\x98\x15\xE8\x05\x2C\xDB\x08\x34\xE2\x21\xD8\xD5\xE6\x2C\x35\xE7\xEC\x10\x8D\x76\x20\x95\x16\x6F\xD6\xE9\xA1\xCB\x74\x3F\x27\x2E\xD0\xD8\x8D\x53\x4F\x6F\x49\xA5\x8F\x43\x7F\xF4\xB4\x77\xFB\x2E\x39\xA8\xBE\xD7\xEA\xEC\x88\xBB\x41\xF4\xF0\x7F\x7E\x78\x5A\xCA\x9A\x7F\xFD\xF5\xE9\xAF\xDA\x5F\x33\x97\x1A\xE5\x7F\xE3\x43\x4B\x37\xF3\xCF\xFF\x6E\xFA\x33\xE3\x02\xF8\xE7\x3F\x30\x2D\xDC\xCB\x81\x78\x4E\x0A\xF4\x72\x12\xF8\x47\x0B\x29\x56\xAC\xFC\xC7\xF8\x63\xC2\x85\x4B\x8D\x44\x39\xB3\x2C\x46\x9A\xC5\x8E\xF9\x68\xB1\x01\xA7\xCF\xB6\xF9\x08\xF0\xE5\x33\x6F\x9F\xE9\x10\x55\x61\x63\xB9\x71\xF6\x6A\xB2\x55\x33\xF1\xE6\x19\xE0\xA7\x59\xC1\xB3\x40\x50\xEB\x33\x0C\x6D\x71\x79\xAB\x4B\xA2\x78\x23\xB0\xC2\xE6\xB1\xC6\xD0\x2D\x01\xD0\x02\xBE\xB2\x9C\x8F\x20\xEE\x51\x1E\x1D\x54\xF4\xEB\x82\x8B\xE3\x63\x55\x05\x7C\xC4\xBA\xD4\x65\x11\xC2\x02\x62\xCD\xC7\x0A\x96\x6B\x3E\x2A\x52\xE8\x47\x8A\xE9\xDE\x19\xC4\xCB\xDF\x9B\xD9\xFC\xC8\x5C\x45\x15\x16\x1E\xA5\x8E\x73\x04\xF4\x6F\xD8\x1F\x91\x07\x8D\xFC\x0B\x1B\xB1\xBA\xF9\xE5\x89\x14\xFA\x73\x89\xFF\x26\x73\x09\x18\x71\x7F\x32\x5A\x1A\xE6\x7C\xCC\x7F\xF4\x7E\xDF\xE5\xF7\x58\xD5\x15\x0F\x91\x5A\xF1\x86\xFD\x1F\x7A\x47\x57\x7E\xCB\x43\xEA\xBF\x8E\x78\x99\x4B\x4F\xBB\xDC\xD1\xFD\xAE\xBC\x7E\xCE\x2A\x86\xDE\x29\xF8\x0B\x4D\x1E\xF1\xAB\x6A\xF2\x32\x13\xC0\xCB\xB2\x67\x69\xA1\x0E\xFB\x7B\xAC\x7A\xE8\xC8\x25\x78\xD5\xB7\x74\xE5\xE3\xE8\xEF\xF9\x7D\x57\xFE\x8D\x17\x0F\xD8\x46\x7C\x14\x41\xCF\xB2\xB9\x55\xDA\x90\x7C\x79\x8F\x2D\xF8\xDE\xF2\x6F\xBC\x88\xE0\x54\x00\x3B\xAB\x7B\x6C\x3B\xFC\x7C\x4D\x73\x03\xF4\xF3\xE9\xD1\xDD\xF4\x33\x47\x5D\x03\xFC\x98\x2B\xC8\xB0\x7C\x3F\xE4\x3A\xAB\x4B\xAC\x0C\xFD\x2E\x51\x3C\x0B\xE3\x18\x06\x5C\x5D\xD7\x50\xBE\xC4\xA5\x03\x6D\x13\x2D\xBE\xAC\xDE\xDA\xF1\x11\x12\x67\x90\x4E\x85\xEC\x70\x6F\x98\x44\x69\x6F\x16\xDA\x1D\x52\x5D\xC2\xB4\xE2\x05\x37\x50\x26\x65\x98\x4A\x46\x1B\xA9\x87\x77\x4E\xA6\x5D\x2E\x3B\xEB\x8F\x2E\xC5\xF9\x5E\xCC\x5C\x1A\x3A\x3A\x1B\x4D\xFE\x43\x47\x2E\x75\x29\x3A\x9B\xFB\xEC\xB0\xCB\xF6\x7E\xE8\x1D\x00\x9E\xE0\xB5\x90\x25\xC8\x78\xFE\x53\x9E\xFF\x50\x54\x22\x3F\x7E\xFE\xD3\xF5\xF3\x9F\xAE\x9F\xFF\xB4\x8A\x30\xBE\xB2\x0E\xA3\xF9\x47\x64\xED\x95\x00\x22\xC4\x8C\xE7\x77\x3F\xBC\x86\x84\xE2\x84\x0B\x0D\x61\xB2\xBB\x7C\x2F\x12\x52\xCE\xA3\x98\xF1\x28\x66\x21\x0D\xEB\xF8\x51\xCC\xD6\x8F\x62\xB6\x7E\x14\x33\x80\xFC\x01\x9E\xD7\x31\xAA\x0B\xD7\xA9\x78\x7F\xD1\xD5\x71\xCE\x50\x6E\xE0\x03\x45\x07\xF0\xAF\x3A\xC2\x31\xB8\x7C\x17\x8A\x3B\xFB\xF5\xE5\x4B\x41\x92\x5A\x16\xC0\xB8\x52\x06\x0C\x4B\x01\xC7\x1B\x93\x59\x5F\x28\x80\x10\xB9\x63\x5E\x28\x02\x30\xE4\x4B\x02\x8F\x95\x31\x92\x5A\xC6\xCA\x2C\x5B\x5F\x76\xCC\x07\xB8\x66\x88\xAB\x05\xC9\x67\x83\xF1\x25\x85\xB1\x59\x49\xA8\x6A\xE8\x7A\x43\x9C\xB1\x75\x2D\x90\x25\x99\x33\x36\x63\xCE\xD8\x80\x33\xA2\x7E\x44\x13\x1A\x48\xE8\x96\xC0\x19\x91\x67\xCE\x55\x97\x37\xB8\x33\x1B\xDC\x99\x0C\x77\xF5\x8B\xDC\x6D\x10\x67\x6C\x18\x67\x09\x68\xBB\xAE\x76\xF3\x11\x67\xCC\x77\xCC\x4B\xC4\x11\x1B\xEA\x39\xAB\xE9\xEF\x2F\x38\x97\x2C\x62\x6F\x56\xEC\x4A\x9E\x39\xBB\xDB\xA5\x28\x59\x8E\x42\x92\xE9\x16\x62\xAA\x04\x01\x81\x6B\x3C\xAA\x6D\xF3\xD1\x62\x97\xD1\xE3\xC7\xDE\x89\x8F\x14\x82\xB1\x2D\x0C\x18\xCB\x46\xAF\xAC\xFE\xD8\xE8\xFC\x48\x96\xD1\x30\x32\x35\x9D\x99\xFE\x03\x72\x10\x89\xD9\x87\xE1\x87\x59\xC0\x08\x26\xAF\xF0\xA5\xE8\x3B\xEB\x82\x4F\x84\x11\x33\x11\xBB\x91\x6C\x0C\x8B\x68\x87\x45\x24\x4D\xFE\x25\x29\xBA\xC3\xA6\x23\xCB\x30\x25\x72\xCC\x25\xB2\x8E\xCE\x6E\x54\xC1\x90\x33\x1C\x72\xF1\x8C\x62\xAD\x2D\x09\x76\x22\x39\xE4\x92\xB8\x94\xE9\x78\x29\x53\x86\xC2\xE2\x52\x20\xE1\x94\x84\x0C\x1B\x14\x3C\x60\x9F\xC8\x21\x67\x47\x87\x5C\xC2\x92\x6E\x38\xE4\xD2\xE1\x90\x23\xD9\x62\xE9\x90\xC3\x52\xA6\xB2\x94\x09\x2D\xE5\x8A\x10\x68\x47\x90\x64\xC6\xD8\x23\x7D\x15\xF5\xD1\xA1\xB1\xBE\xC0\x50\x07\xF6\x58\x26\xC0\x5C\x37\x19\x98\x00\x1D\xC1\xD9\x61\x97\x82\x09\x64\x8F\xA3\x14\x70\xB6\x86\x95\x25\x13\x26\x80\x1B\xB3\x15\x56\x16\x7F\x9E\xB2\xB2\xE1\x6E\x66\x65\x30\xB9\x08\x1B\xD8\x36\xAA\xFD\x17\x88\x21\x49\x58\x2B\xB8\xC2\x55\xA0\x7B\xCC\x1B\xAC\xA9\x76\xAF\x7D\x81\xAD\x7D\x29\xDB\x97\x6E\xE8\x47\xB8\x9A\x22\x28\x83\xD8\xE0\x87\x4E\xEA\x3A\x68\xE5\x2F\x17\x51\x2D\x87\xCE\x2D\x09\x68\x31\x41\x40\x13\xED\x70\xE8\xBE\x26\xA1\x20\x7C\x7C\x6F\xD1\x3F\x74\xED\x9B\xAF\x5D\x7B\x97\xC3\x77\xB9\x99\xAE\xBE\xBF\x70\x1A\xA9\xA8\xF8\x77\x74\xCE\x4A\x65\x95\x76\x0C\xB4\x61\xB9\xC6\xEB\x58\x20\x7F\x1C\xE2\x4D\x97\xF6\x1D\x20\xB2\xF3\x7D\x7F\xB5\x9F\x5B\x65\x35\x32\x77\xB3\xB3\x2A\xE9\x72\x97\x71\x35\xD6\xAE\x68\x7F\x19\xC6\x8B\x2C\xD4\x97\x0A\xCF\x98\xDD\x46\xF9\x9F\xFE\xE0\x44\x34\x2C\xCF\xAA\x50\x99\x31\xF7\x9F\x53\x9C\x29\xFC\x69\xC5\x01\xA2\x9F\xE6\xFF\x3C\xBC\xD5\x91\xB2\xFE\x39\x85\xFF\x5C\xDE\xEA\x0A\x9E\xDA\xDC\x3F\xD9\xBB\xC2\x3F\xD9\xF3\x0B\xA5\x3C\x52\xE6\x8B\x7D\x22\x68\xF4\x3B\x94\x9E\x4C\xA3\x0E\x95\xBA\xEC\x69\xD2\x49\x4D\x40\x16\xA0\xDE\x7B\xBB\x7F\xC1\x99\x03\xA9\x2F\xCA\x42\xD4\x47\x0B\x56\x99\x3E\x16\x60\x39\x1F\xFA\xB6\x6B\xD7\xAE\xBD\xA0\xCF\xB1\x71\x43\x04\x50\x36\xCF\x6A\xC8\x66\xF9\x5E\x57\xD2\xE6\x9C\x0D\x40\x9C\x24\x40\x9A\x00\xC4\x99\xEF\x75\x05\x6D\x4E\xB0\xB1\xB8\x39\x8B\xF1\xE6\x2C\xA2\x05\xD0\x15\x91\x51\xD3\x2D\x61\x73\x22\x2D\x7D\x86\xCD\x39\x1B\x75\x28\xC5\x3D\xFD\xA2\x90\x42\xB7\x9C\xCE\x83\xAA\x7D\x74\x16\xC6\xAD\x89\x5C\xF4\x8F\x15\x9C\x3B\xF7\xD1\x82\xAD\x34\x1F\x29\x9C\x44\xDE\x05\xED\xAC\xFD\xBF\xB8\x0E\xC8\xC2\xC4\xC5\x23\x4E\xE6\x7F\x68\x24\xC1\x63\x7D\x49\x9B\xBA\x6F\xFF\x82\xB3\x98\x3B\xC3\x39\x68\xE0\xB1\x15\xCF\xDD\xA2\xE6\xB9\x44\xFE\x56\xDA\xE5\x21\x75\x06\x91\x3E\x51\x47\x21\xA1\xFA\x7F\x80\x2A\x13\x6A\x36\xFB\x3F\x15\x88\xFE\x32\x24\xD7\x28\x97\x07\x74\x5E\x9E\xDF\x35\xBA\xD7\x90\x97\x93\x8E\xB4\xB6\xA4\x92\x84\x57\x28\x1A\xAC\x61\xE7\x52\xC9\xD0\x0A\x86\x80\xEE\xBB\x62\xDA\xE2\x63\xFC\x46\x57\x70\x5E\x79\xF8\x18\x8C\x1D\x5F\x7A\x1F\x20\xB6\x17\xF1\x7E\xCD\x50\x10\x92\x74\xA4\x5D\xE6\xFF\x54\x33\x2C\xB1\xE1\xEA\x20\x9A\x4B\x1A\x6A\x2E\x19\xB2\xF4\x66\x8B\x51\x65\x41\x5A\xF4\x37\xF4\x22\xF3\xD7\xCC\x22\xAD\xE2\x1C\x7E\xDD\x65\x2C\xD3\x45\x19\xF1\x3A\x65\x2F\x5B\xAB\xEC\xE5\xC0\x26\x5D\x52\xDF\x5C\x8E\x9C\xBA\xA8\xBA\xCD\x5C\xFD\x18\x50\x41\x91\x54\x3F\xE3\xC3\x9F\x89\x91\xC1\x41\xAB\xBD\x6E\x46\xBB\xC3\xF2\x5C\x4F\x76\x47\xE1\x6A\xBA\x5E\xD2\xEE\xA8\x5D\xED\x8A\xB8\x3B\xCA\xF1\xEE\x28\x07\x50\xAA\x72\xD8\x1D\xE5\xB0\x3B\x80\x53\x6B\xD9\x74\xC2\x1D\xB1\xDC\x11\xBA\x07\xBE\x90\x59\xC0\xA3\x25\xD9\x78\x52\xFC\xBA\xC6\xF1\xCD\x05\xBB\x11\x6B\x1F\x2B\xF7\x2F\xAA\x4A\x24\x6A\x84\x94\xD7\x4B\x3B\xA8\xC2\x0E\x0A\x7B\xE6\xCB\xBC\x03\x22\xA1\xCA\x3E\x78\x53\xF8\xE9\x36\x29\x37\x8D\x94\x3B\x21\x36\x3B\x21\x36\x2B\xC4\x96\xDC\x26\xB1\xC1\xA4\xF5\x67\xDA\x06\xE9\x64\x1B\xD8\x3F\x73\x3B\xEB\xB6\xF4\xAD\x89\xFF\x2B\x46\xF9\x40\xE7\x58\xA6\x70\x10\xF0\xC3\x22\x67\xFF\x59\x28\x5C\x1A\x48\xBE\x12\x14\x4E\x77\x2C\x2A\x24\xBB\x8A\xD7\xAB\xE0\x52\x0D\xEB\x36\xF0\xAD\xC9\xDF\x6B\x98\x4D\x06\x8B\x32\xEF\x86\x4A\x22\x7F\x21\x63\x88\x74\xEE\x94\x7F\x55\xF5\x08\x88\x6D\x7F\x45\x82\x8E\xF8\xFC\x94\xD4\x7C\xDE\x76\xFC\xEB\xFB\x97\x7E\x75\xDA\xBF\x5C\x04\xCB\xE1\xF7\x9C\xD2\x11\x7E\xE3\x9A\x5E\x72\x61\x9C\x55\x36\x54\x78\xB4\x6C\x85\x7C\xCB\x92\x5F\x43\xAF\xFA\x35\x3E\xC3\x85\x19\x06\xDF\xC6\x53\x6E\x7C\xF9\xB3\x6A\x7C\xED\x90\x9B\x7D\x92\x25\xAC\x27\x82\x78\xAB\xC3\x23\x5D\xB2\x63\xBE\x81\x6F\x7D\x02\x68\x01\xB6\xDF\x31\x4F\x8A\xC1\xF4\xFC\xA4\xE1\x97\x97\xDF\xFB\x46\x67\x06\xD9\xDC\x90\xEC\x95\x8C\x7C\x2C\x74\x7B\xDB\x77\xD9\xC8\xD3\xF2\x08\x37\xBB\x1D\x9B\xA5\x41\xDF\x17\x9B\xB4\x3B\xE6\x5E\x97\x8C\x2E\x6D\xF7\x23\x27\x8E\xDD\x31\xF7\xF3\xF3\x8E\x07\x73\x1A\xB1\x89\x2F\xAB\x1E\x77\xB2\x57\x07\xB3\xE3\xB4\xFF\x65\xD5\xB7\xBF\x76\x3D\x98\xD2\x3B\xEB\x61\x0A\x7E\xA4\x33\xA3\x58\x41\x92\xCD\x62\xF7\xF5\xB6\x39\xBF\x28\xA0\x51\x48\x07\xCA\x30\x57\x70\x33\x94\xB0\x08\xD8\xA1\x7C\x8C\xF6\x3F\x72\x4D\x42\x24\x4B\x57\xB2\x64\x83\x02\x83\x45\xDF\xFE\x86\x50\x8C\xF1\xCF\xF5\xAE\x04\xAC\xB6\xB7\x87\xFE\xB9\x5E\x00\xD9\x4A\xAE\x5D\xC9\x91\x1A\x35\xC3\xFA\xA9\x6D\x53\x70\xAD\x3C\x7B\xD3\x51\x7D\x9A\x49\xD6\xA0\xDE\x40\xC3\x05\xB0\x8C\x00\x10\x25\x4C\x8F\xE7\xA5\xC2\xFB\x10\x14\x69\xBD\x39\x5C\xA4\x2E\x75\x21\x3C\x03\x25\xD4\x59\xCE\xC7\xD6\xD1\x52\x68\x09\x9E\xC5\xE0\x5C\x63\x7B\xDF\x60\x63\x64\x05\x2C\x86\x78\x18\x8E\x8A\x8B\x01\x7D\x69\x1F\x32\x53\x89\xED\x8B\xFB\x06\xD3\xBE\xB0\xD1\x55\x4F\xAA\x68\x37\x54\xE7\x74\x66\xCF\xDB\x67\xC4\x5C\xCA\x55\x35\x78\xE8\xC5\x74\xD4\xBF\x19\x81\xEA\x3A\x8C\x9C\x0B\x85\xF8\xE2\x40\x32\xF3\x91\xC6\x1F\xAB\x7E\x72\x11\x10\xBD\x6D\x0A\x6A\xBD\xFD\x57\xE1\xE1\x62\x61\xD8\xE4\x9F\x06\x2D\x98\x7A\x87\x29\x13\x98\x71\x64\xAD\x6C\x47\x27\xE8\x7D\xF1\xD3\x03\xF1\xD3\xCB\xD1\x31\xEA\x66\x1C\x42\xB1\x63\xEE\x66\xAD\xB6\xF2\xF6\x19\x2C\x84\x1D\x21\x55\x03\xC0\xC6\xAB\xC5\xDC\x03\x5E\x66\xD1\x12\x67\xF3\xEA\x11\xC6\x7C\xA6\xFF\x5F\xD9\x12\x2F\x0D\xD7\xBD\xF7\xF9\x7E\x27\x68\x73\x06\x98\x9D\xE6\x59\xFF\x27\x37\x6E\xDC\xB0\x70\xBE\xF8\xE4\xD0\x91\xF6\x70\xE0\xD8\xF3\x0D\x2D\x93\x94\xBB\x6D\x1A\x8F\x1D\x30\xD8\x22\x60\xE2\x51\x4F\xEA\x34\x0D\xEC\x13\x28\x03\x03\xED\xF5\x65\x25\xB5\x2F\x94\xFF\x2B\xF8\x94\x61\x90\x82\x1B\xE2\xEF\xC7\xA7\x06\x53\x40\x9F\xE6\x4E\xF9\x7B\xF1\x69\x03\x13\x44\x9F\x5A\x81\x9A\x83\x8D\xA4\x71\x73\xB7\xE1\xDA\x9E\xFF\xB9\x19\x09\xE2\x9F\x79\x69\x6C\xC5\xC5\xF4\x07\x5D\xAD\xEE\x40\x22\x17\x1A\xED\xBF\x6F\xB8\x4B\x6A\x50\x01\xFB\x4D\x8A\xC9\x21\x6A\x86\x95\xF4\x22\x5E\x2E\x76\x4C\x11\x4A\xA8\xB1\xC3\xC8\xEE\x75\x86\x4E\xB5\x0C\xB5\x03\xFD\xD1\x23\xC4\x91\x42\xA5\x78\xF6\x41\xA7\x6C\xCF\x7E\x9C\x35\x1D\x62\xD7\xC1\x64\x19\x03\x2A\x5A\x06\xCF\x7D\x9C\xC9\xA4\xE0\xCA\x5D\xBF\xA0\x7A\x44\xF8\x03\xAD\x38\x1D\x58\xC0\xC8\x05\x9D\x30\x68\xF0\xD1\x23\x8D\x71\xC6\xEB\xB8\x45\x18\x04\xF4\x12\xC2\x48\x2D\x3B\xB8\xC7\x07\x85\x28\xB5\x87\xFC\xD6\x4B\xFC\xBA\x9F\x0B\xEC\x9B\x14\xEF\xB0\x47\x11\x79\x73\x77\x67\xF8\xD4\xE4\x9B\x89\xE3\x32\xCE\x01\xB3\x9A\xA3\x1E\x70\x1C\x9E\x7A\xF2\xF6\xFD\xBD\x45\xE6\x12\xA0\x06\x70\x96\xBB\x01\x61\x1C\xF6\xC8\xE1\x7C\xBD\x96\xD0\x87\x3E\xFE\x1A\x7E\x2A\x86\x9F\xBC\x8E\x37\xD6\xA3\x5F\x0D\x7E\x35\xBE\x05\x8A\x25\x02\x0D\x3A\xD5\x7B\x29\xFD\x89\x04\x54\xAE\xDD\x01\xC3\x91\x39\x04\xD3\x9B\xF4\x80\x9F\x47\x08\xB5\xE4\x45\x46\x22\x95\x13\xB5\xE8\x04\xDC\x5A\x8D\xCB\xD4\x08\x22\x40\x8C\xA2\x05\x76\xD7\xB2\x7B\x90\xF1\x5F\xF3\x25\x47\x60\xE5\x4A\x7E\x88\xDB\x3F\x0D\x0E\xC9\x9F\xCF\x74\xCA\xD9\x5D\xA9\xFF\x33\x2E\xAD\x63\x62\x69\x1D\x76\xBC\xC7\xA8\x04\x67\x2F\xA2\xEB\x74\xBB\x33\x1C\x55\x34\x11\x08\xB6\xBB\xF5\x6E\xCB\xED\x91\xBF\xF2\x01\x3E\x92\xEE\xEF\xD4\x3A\xA7\xE5\xFD\x8B\xE0\xF1\xBE\x3F\xA4\xF6\xAB\x91\xD3\xF2\xDE\x85\xD4\x4C\xBB\x77\xF9\x6A\x38\x01\xD6\x77\xE0\xFC\xA8\x03\x8F\x87\x33\x71\x6D\x07\x1E\x89\x1D\x78\x64\x5D\x07\xDE\x18\x3A\xF0\xC6\xF5\x1D\x78\xA2\x5B\xF3\xE3\x5B\x8E\xE9\xD5\x5B\x46\xBD\x7A\x8E\x7B\x75\xB8\xBE\x57\x87\xB1\x57\x87\xEB\x7A\xF5\x54\xE8\xD5\x53\xCB\xD7\xFC\x35\xBD\x52\x61\xE0\x0F\xFE\xA2\xDE\x18\xA1\xE8\x64\x01\x62\x47\xA2\xA9\x0A\x6A\xCA\xB3\x99\xBA\x7D\xF7\xF5\x61\xCB\xD7\x9D\x48\xE7\xBF\xF8\x0F\x96\xDC\x8D\x2D\x17\xBC\xF1\x72\x52\x95\x5B\x2E\xA7\xFB\x7E\x78\x3A\x88\x76\x51\xF0\x29\x54\x46\x94\x18\xAF\x1C\xDD\xFC\x70\xA8\xD5\x56\xC8\xC9\x99\x44\xEF\x4F\x37\x63\x9F\x19\xAC\x3C\x25\x1B\x57\x82\x69\x9E\xB1\x46\x16\xB9\xCB\x5C\x21\xFD\x46\xC4\x65\xEC\xF7\x90\x23\xB2\xBE\xE3\xC4\xC6\xBE\x36\xBD\x2E\x6E\xD6\xEB\x50\xA0\x50\xAA\x82\x95\x52\x17\xA9\x0E\x0E\x37\x48\x52\x69\x2C\x49\x53\xB2\x36\x90\x02\xDE\x89\xD1\x24\x4B\xEE\x53\xE1\xCA\xBD\xAE\xE0\xD0\xC4\x82\x55\x1A\xD2\x2D\x0D\xAB\x34\xFA\xF2\x96\xA8\xFA\x1C\xAD\x88\xFA\x05\xDD\x2C\xA4\x90\xEC\x70\xF9\x27\x6E\xBA\x70\xB3\x9E\x4B\xCA\x14\xC3\x4B\xB8\x62\x3E\xB3\x74\xB8\x43\x72\x74\x98\xD5\x20\xF8\x4E\x68\x54\xA8\xCE\x07\x78\xE0\xDC\xBF\x5D\x1E\x45\xC8\x0E\xE3\x7E\xD5\x6B\x97\x26\xA1\xA5\x29\xBD\x12\xC4\xC5\x2D\x57\x2F\x2F\x0D\x8A\x4A\x00\x79\xAB\x5E\xE4\x23\x64\x1F\x57\x72\xF5\xF2\xFA\x4D\x9C\xBE\x97\x0C\x95\x30\x18\xA7\xEB\x31\xF6\x40\x05\x79\xA9\xC6\x92\x90\x5C\x58\x4B\xE8\xCA\x60\x57\x45\x11\x96\xAE\x61\x70\xDE\x7C\x7F\x9E\x2B\xA5\xB5\x36\xA9\xA9\x5C\x46\xB3\xFD\x2C\x47\x4F\x35\x6B\x03\x80\xE8\x9E\x66\xDB\x20\x06\x0B\xA5\xED\xE6\x98\x8E\xC5\x06\x86\xBD\x41\x8D\xB6\xFE\xE8\x0A\x50\xD8\x96\x46\x6F\xC2\xE8\x8D\x8C\xBE\x5D\x1E\xBD\xE1\xD1\x9B\xF1\xE8\xC3\xC8\xDB\x87\xB9\x2C\xAB\x10\xA6\xA1\x39\x68\x07\xC2\x34\x91\x30\xF3\x81\x30\x5B\x4E\x4B\x05\xAA\xEF\xB6\x51\x7B\xBE\x7E\x2B\x54\xD1\xD5\x73\x66\xFA\x6A\x79\x0F\x1F\x3E\xFC\xFE\x65\xF6\x35\xED\xAA\xFF\xFD\x7F\x08\x40\x93\x49\x87\xDF\x84\x33\x69\x69\x22\x56\xC6\x18\x6F\xBF\xD5\x64\x8C\xDB\x5D\x1A\x3E\x57\x55\x0B\x24\x10\x06\x7F\xB1\x51\xFE\xDD\x53\x33\xC2\x97\x65\x55\x6E\xD2\x91\x62\xD4\x91\xE2\xB8\x8E\xAC\x36\xC8\x8B\x27\xED\x5E\x6A\x94\xFF\xA9\xE9\xEB\x47\x4B\x32\xBC\xB0\x3B\xD9\x73\x82\x93\xE1\xE7\x4B\x81\xD3\x8A\x93\xE0\x5A\x8E\x64\x2B\xFD\xD5\x35\xAB\xBD\xBC\xCC\xD4\xA9\x75\x2B\xFD\x65\x9B\xB6\x75\x13\x77\x72\x34\x6B\x0C\xE7\x3D\x9E\x37\xA2\x5E\xFF\x9D\xD3\x89\x93\x67\x16\xFC\xCC\xD0\x7E\x78\x10\x01\xE8\x78\xD5\x3A\x12\x8F\x83\x2E\x2B\x1E\xE3\x42\x4A\xF2\x85\xA1\x13\x7B\x1D\x06\x39\xAE\x89\xC4\xBD\x43\xD4\x9E\xCB\x43\xE9\xBB\x1C\xA9\x62\xE3\x2A\x48\x27\x69\x93\xB1\x3B\xA2\xF0\xBF\xF8\xA1\x4F\x7D\x48\x3F\xD6\x94\xB4\x8B\x1C\x5F\x29\x49\xA0\x3E\xE5\x4E\xC1\x1A\xF5\x29\xBA\xEE\x4A\xFF\xA1\x4F\x7D\xE8\x53\x30\x45\x0D\x4F\xA0\x7C\x49\x29\x5C\xFB\x94\x9B\xF9\xFA\x29\x6A\x11\xAF\x97\xDA\x7E\xA7\x5C\xD9\x2F\x4E\xB9\xC2\xD7\xFD\x82\x6B\xC6\xE7\xEE\x24\xA7\x82\x9F\x62\x1C\xBF\xCD\xA1\xD6\x17\x87\x40\x16\x28\xCA\xCA\x8F\x17\xC4\xB7\x0B\x3E\x93\x66\x28\x08\xD8\x95\x52\x4B\xB0\x10\xA6\x1E\x53\xF0\xB8\x4E\x21\x3F\x37\xA3\x39\x5B\xE4\x34\x22\xAE\xC2\x28\x6F\x43\x70\x5F\x7C\x29\xFB\xE7\x2D\xA7\x99\x97\x9C\xA8\x59\x92\x8C\x59\x0E\xB8\x90\x05\x8E\x1A\x12\x49\xF9\x58\x0A\xF2\x4D\xE9\xF2\x8B\xF4\x12\xDC\xEE\x0A\x88\xA4\x13\x6F\xC9\x57\x92\x24\x6F\xC1\x54\xD6\xD3\x64\x3B\x94\x2F\x0B\xF4\x08\xFD\x3F\x3E\x89\xA2\x35\x1D\x62\x55\x67\x43\xA4\x6A\x49\x7F\x8A\xAE\xE5\xF3\x3E\xD4\x0E\x45\x46\x04\xA3\x9E\x4A\x6C\xE7\x5E\x77\xD2\x9B\x67\xBA\x13\x40\x98\x0D\x67\x7B\x29\x22\x43\xEB\x8A\xBD\xAE\xF5\xFA\x19\x90\x55\x79\x39\x48\x04\xED\x58\x22\x68\x19\xD4\xAF\x06\x84\xA8\x34\x30\x43\xB7\x83\x7D\xA8\x62\x0F\xF3\x29\x57\xBA\x13\xC8\x65\x3C\xC1\x1D\x2A\x79\x70\x8B\xD6\x9D\xA0\xDE\x9F\x30\xE7\xDD\x49\x56\xDC\x88\xAC\x4F\x46\x73\xE7\xCC\x9D\x10\xDE\xB3\x03\xDA\x6C\xB1\xA4\x00\xB4\x0C\xEB\xBA\x41\x42\xC6\x86\xAB\xC4\x0A\xBD\x22\xAB\xE5\x5B\x38\x7E\xBB\xF2\x66\x02\x9B\xAC\x5B\xC1\xEB\x56\x46\x71\x6D\x16\xC5\xB5\x20\x04\x74\x2D\x8B\x6B\x33\x5E\xC3\x36\xAC\x61\x09\x44\xD6\x86\x97\x6B\x06\x0B\xC6\x6C\xAD\x24\x3F\xBC\x93\x5F\xD1\x56\xE8\x1F\x6C\x69\xAD\xF4\x60\x8D\x58\x3F\xEA\xAA\x9C\x8B\x6D\xEC\x70\x77\xD2\x95\x7C\x2E\x9E\x74\xB7\x1C\x66\x1C\xA4\xD4\x35\x5F\x19\xDA\xF2\xC0\x96\x0F\x9A\x63\x1B\xFC\xEA\xBF\x59\xF3\xB3\xC5\xDA\x23\x6E\x69\x92\xE3\x0B\xE9\x88\x6B\x77\x06\x88\x65\x20\x46\xB6\x4C\x66\xCC\xED\x4B\x3E\xE2\x8A\xD5\x23\x6E\xDA\x26\x56\x2C\x52\xDB\xAD\x96\xED\x8B\x98\xA9\x75\x73\x75\x72\x4C\x7D\xC5\x2A\xF5\x2D\x71\x90\xE1\x99\x45\x2B\xAA\x47\x68\x3F\x3C\x18\x4E\xB5\x35\x82\xDB\x78\x9C\x45\x38\xD5\x5A\x87\x26\xC2\xA9\x66\x2B\xD7\x8A\x4D\xCA\x54\x1C\x11\x95\xE2\x68\xBB\xD8\x24\x75\x56\xDD\x44\xD8\x9D\xA3\x59\xC6\x01\xA4\xA3\x10\xFA\x89\x6C\x28\xCE\xAE\x46\xC0\x73\x3C\x03\x59\xB5\x21\x7E\x64\x9E\xE9\x4E\x0E\x95\x00\x93\x50\xE6\x2C\x27\x0E\xB7\x47\x07\xE4\xB4\x12\xA0\x75\xB3\x31\xBF\x9A\x0D\xA5\x75\x67\xA1\x81\x14\xE1\x5F\xE3\xD2\xBA\x27\xC1\xA8\x4E\x3A\x91\x7A\x66\x52\xD1\xF4\xA4\xB0\x59\x46\x4B\x06\x4F\x72\x6D\x64\x54\x29\x2B\x68\x29\xA7\xAA\xA5\xE2\xCA\x19\x69\x3D\x81\x37\xC8\x6E\xAF\x89\x57\x0D\x6E\x96\x6C\xA8\xAD\x2E\xA0\x1E\x98\xFB\x4B\xD5\xB7\xCC\x53\x7B\xA4\xAF\x1A\x54\xB6\x42\xF6\xF6\x73\x7D\x00\xB9\x8C\x6A\x8C\x76\x6A\xAF\x4B\xBC\x7D\xA6\x4B\xE7\x59\x96\x29\x6D\x2C\x02\xAA\x9F\x83\xC7\x06\x27\xCE\x89\x6F\xA2\xC6\x01\x80\xA4\xEE\x61\x53\xA6\x4B\xE4\x60\x4E\x5C\x76\xCE\x70\xE5\xAD\x62\xE5\x09\xA2\x02\xB6\x26\x27\x93\x9F\x0B\xF1\x9F\x15\xD2\x64\x81\x26\xB9\x6E\x30\xCD\x43\xC6\x2D\xAF\x7B\x3B\xDF\xAA\xE4\x56\x15\xDF\x8E\x7E\x11\x3D\x72\xE5\x91\x58\x83\x78\x74\x43\xB1\x28\xF9\x01\x7A\xAE\x3C\x27\x28\x33\x93\x57\x24\x18\xD7\x22\x73\xC9\x9A\x16\x18\x59\x54\xF9\xB6\xC7\xA8\x9C\x69\x7F\xFB\x3A\xD7\x19\x93\x6B\x5D\x81\xAB\x5D\x8A\x4C\xA0\x2E\x89\x77\xF0\x2C\xE4\xCB\x6F\x33\xC3\xDB\x72\x79\x5B\x1E\xC6\x9E\x2E\xDF\x9A\xF2\xAD\xE9\xD2\xCC\x43\x67\x5E\xBA\x55\xDD\x63\xDA\x30\x4D\xC5\x39\x60\xAD\x62\xD4\xC5\xF2\x8C\x9A\xE3\x66\x94\x23\xDB\x13\xFF\x47\x7A\x28\x89\xF7\x6F\x62\xE5\xB4\xD4\x9B\xB7\x72\x38\x9F\xC4\x6B\xFE\xD8\x5F\x85\xD9\x14\x97\x92\xB7\xC2\x51\x45\xB3\x90\x0B\xBA\x44\x98\x87\xC5\x0C\x31\xD8\x93\x2E\xE8\x61\x0E\xAC\x0C\xCC\xA2\x0B\x88\x65\xCB\xC7\x37\xB3\x41\xBF\xE7\x28\xCD\x9C\x9F\xCA\x97\xA6\x83\x8B\x8D\xAC\x3C\x65\xE4\xA9\x38\x89\xD3\xF9\x46\x68\x8E\x5A\x79\xCA\xF6\x21\x22\x54\x88\x7E\x4A\x13\x6E\x86\x5D\xCD\x6C\x66\x4A\xF7\x70\xC8\x4E\x7E\x61\x64\x30\x8E\x2D\x9F\xB9\x94\xE6\x3E\x0E\x61\x3A\x70\x00\xE9\xE6\x6B\x97\x8A\x7B\x31\xED\x3B\x87\xA5\x09\xF5\x73\x45\x97\x35\x0D\x32\xFC\xF2\xF4\x49\xBC\x68\xA9\x9F\xB4\x20\xC7\xF4\xAB\xF2\x66\x31\x1B\x80\x51\xD5\xF2\x83\x48\x2C\x05\xA2\x99\xA0\x79\x25\xC2\x43\x18\xF9\xA8\x00\x4C\x9F\xD8\xA9\xC3\x2C\xD3\xDB\xED\x6E\x93\xD9\x63\x5A\x3C\xA6\x2B\x2C\x18\xE6\x88\x6B\xE0\x72\xB6\x76\xB7\xC9\x6B\xAE\xAD\xBC\xDC\x48\x75\x1C\xD3\x12\x6A\xE2\x94\x03\xEC\xDE\xE4\x31\xA4\xAC\x16\xC4\x27\x11\xA9\xCB\x21\xB9\x4B\x54\x58\x71\xE9\xA4\xC2\xC6\x21\xAE\xBC\x12\x04\x85\x04\xF1\x74\x51\x4C\x08\x6F\xA9\x0B\x09\xC8\x35\xDF\xDD\x5A\xE4\x81\xB8\x59\x0E\x2D\x58\x41\xC9\x97\x77\xB7\x72\x05\x37\x55\x2C\xF3\x8C\x61\x34\x33\x2E\x2F\xEA\x4C\xFB\x99\xEB\xA1\xB2\x4E\xC1\xEC\x59\x87\x1F\xD1\xD2\x42\x93\x16\x93\xD5\x69\x85\x1A\x85\x39\xEA\x24\x10\x9B\xD8\x73\xDA\x15\x7B\x8F\xC6\x76\xE0\x70\x45\x1B\x0B\x55\xDB\x0A\x9F\xD0\x96\x5C\x58\x68\x44\x47\x71\x09\x75\x8B\xB9\xCB\x69\x12\x13\x7B\x5E\x02\x58\xBB\x64\x65\x59\x12\x2C\xA2\xE5\xD4\x2B\x9C\x12\xD3\x69\xE6\x68\xD6\xCB\x8D\xF2\x09\x07\x65\x1E\xC3\x9B\x79\xDD\xE1\xE5\x40\xA4\x2D\x7F\x34\xA8\x24\x5B\x55\xFF\xA6\xD1\x63\x2C\xF7\x3C\x00\x41\x4B\x71\xF0\x50\xE7\xBD\xA0\x3F\x05\x57\x68\xD6\x1C\xB2\x0C\x0B\x24\x8C\xA2\x05\xEF\xF2\xCC\x8B\xF3\xBC\x9B\xB1\x62\x24\x37\xCE\xE4\xC6\xC2\x65\x0C\x54\x3F\x56\x32\xA8\x5D\x52\x43\x91\x72\x26\x55\x1E\x66\x70\x5A\x23\x75\x1E\x29\xD8\x8C\xD1\x8F\x23\x9E\x75\x51\x7F\x74\xA5\xAB\x56\xF5\xC9\x94\xC4\xC5\x99\x57\x8C\x81\x5C\xAC\x11\x06\x53\x94\xC0\xC5\x75\x98\x51\xA0\x49\x28\x57\x88\x30\x98\x4E\x04\x67\xDC\x07\xC1\xB9\x60\x49\x25\x4A\x83\x34\x3D\xAE\x8C\x76\x4A\x57\x3A\xBD\x0B\x1D\x61\x4D\xA7\x4C\xE8\x94\x39\xAE\x53\x86\x3B\x65\xB8\x53\x93\x0E\x99\xA5\x0E\x99\x51\x87\xC6\xE2\xE9\x60\x2B\xE5\x9C\xB8\x62\x6C\x2B\x35\xC6\x18\x6D\xD8\x0E\xB6\x46\x62\x2C\x68\x01\xAA\xB0\xD4\xA4\xE3\xD2\x52\x77\x35\xEB\xB8\x15\xAF\x61\x39\xC8\x8C\xBC\x56\x5C\x51\xDD\x48\x45\xF5\xA8\xE3\x4A\x1C\x96\xAB\xDD\x6C\xAF\xAB\x49\x66\xDC\x70\x1B\x23\x1D\xB7\x1E\xCB\x8C\xF5\xA0\xE3\xD6\xA1\x81\x1C\x76\xE2\x41\xC7\x2D\xDD\x1C\x32\xE3\x9C\x7B\x52\xB2\x19\x79\x51\xB9\x39\x9D\x2E\xF5\x50\x3D\xBD\x9C\x54\x4F\xCF\xB8\x7A\x7A\xC6\x85\x1F\x32\xD6\x71\xA7\x95\xA2\x8B\xB1\xCC\x48\x9D\xE0\xD8\x58\x99\x84\x8B\xE2\x0D\x24\x12\xB8\xC0\xA4\x51\x2F\xB4\x3D\x1F\x34\x23\xD0\x60\xBD\xBA\xDC\x36\xA8\x2C\x16\x10\x58\x6B\x96\xDB\xB2\x42\x62\x01\xF0\x13\x34\x97\x4C\x16\xDC\x46\x4D\xAA\x91\x25\xB7\xB2\xE4\xD9\x68\xC9\x39\x3C\x71\xE9\xED\xC7\x37\x7C\xAB\x1E\xAC\xEB\x43\x23\x86\xBE\xE1\xF5\xC5\xF1\x0A\xD1\xF0\xC4\xD0\xAC\x54\x6C\x0E\x8F\xB1\x3D\x05\x94\x53\xF1\x14\xAB\x5D\x49\x62\xA9\x38\xEC\x60\x9D\xFF\x61\x65\x47\x67\xB7\xBF\xA3\xB3\xB8\xA3\xB3\x9B\xEE\xE8\x4C\x42\xD4\xD6\x2C\xA6\x09\x8B\x69\x8E\x5B\x4C\xC3\x53\x69\xD6\x4C\x65\x34\xFF\xCF\xA2\xF9\x9F\x27\x75\xCE\xD6\xD3\x19\x4F\xEB\x7C\x32\xAD\xA4\x14\x71\xE8\xC1\x1A\x0B\xC7\xE4\x65\xD2\x36\x07\xD7\xF1\x3B\xD7\x9A\xFC\x47\x9D\x1B\x77\x8D\x0D\x1D\x31\xEE\x61\x0D\x39\x1D\xFB\xF4\xAD\xE6\x60\x42\x4C\x66\x4C\x4C\x66\x0D\x31\x2D\x1B\xFD\x9D\x71\x65\xFB\x9D\x02\x9E\x55\xC2\x60\xDB\xBE\x47\x1C\x72\x15\x9D\x24\x15\x44\x26\xE4\xB0\x24\x7D\xA7\xB1\xF4\x1C\x8E\x63\x42\x0C\x4D\x1E\x55\xBE\xC1\xDD\xFA\xF9\x13\xBA\x3C\x12\x97\x1E\xC7\x60\x80\xEC\xA0\x20\xEF\xA2\x4F\x9A\xD5\xE3\x2C\x7A\xD6\x32\xF6\xDC\xA4\xF1\x5C\x13\x67\x9C\xDC\x18\x5C\x70\x05\xEB\xA4\xC9\x34\xB8\x62\xEA\x89\x93\xF8\xC6\x9C\xCF\xB5\x15\x1F\x1C\x34\x05\x7B\x5E\x22\x2F\x11\xE1\xB8\xCE\x13\x57\x04\x4F\x5C\xB6\xC6\x49\x9A\xF0\x32\x24\xCB\xCB\x90\x47\x0F\x5C\x3E\x78\x3D\x92\xD1\x4A\x98\xE8\x14\xB5\xEC\x81\xCB\x87\x93\x2D\x0F\x27\xDB\x9A\x2E\xAD\xEC\x8E\x95\x2E\x4D\x29\x63\xD2\x21\x13\x3A\xD4\xCD\x8E\x27\x8E\xFC\xA2\xA4\x1A\xC0\xAE\x9F\xB1\x5D\x1F\xF6\x99\x2C\x18\xF4\x33\x12\x9E\x32\x39\x9A\x18\x6A\xA7\xAB\xE8\xE8\xB1\xCF\x74\xCD\x24\x97\x2B\x97\x5C\xAE\xC2\x55\x92\x63\x30\x77\xF3\x51\x2E\xD7\x24\xC7\x20\x1B\x62\x4C\x63\x03\x85\xCB\x96\x62\x4C\x1B\x76\x4C\x8A\x79\x1E\x97\xFB\xC5\xDC\x35\x24\xB9\x65\x5C\x67\x2B\x84\x99\xD6\xA3\x0A\xED\x73\x36\xF6\x67\x83\xB1\x7F\x12\x4D\xCD\x96\xF9\x8C\xDE\xFC\x1C\x89\xDD\xC5\x18\xB5\xE3\x26\x4B\x51\xDE\x72\x29\xCA\x09\xA3\x2A\x97\x97\x63\xB6\x64\x46\xCF\x06\x33\x7A\xBE\xDE\x8C\x3E\x1B\xDB\xD0\x69\x85\xD1\xD9\x2E\xF7\xC5\xFE\xF0\x34\xD0\x56\x5C\xEE\xF3\x7D\x1A\x40\x6D\xAB\x69\xD6\x0C\x67\x51\x65\xFC\x5C\xBE\x3F\x2F\xB4\xB2\xD6\xA6\x26\xD5\xD5\xB0\xF6\xF9\x68\xED\x47\xCE\x9C\x82\x37\xD1\xC8\x9D\x53\x4A\x7C\x71\x35\x98\xB2\x4C\x30\xBD\xE7\xC1\x13\x2F\xF1\xC5\xF9\xFA\xB5\x1F\x97\x75\x0E\x0D\xCC\xC6\xA1\x99\x58\xFB\x0A\xDD\xAE\x1E\x8A\xFE\x17\x5A\xFB\x1A\xB5\x70\xA0\x51\xC4\x10\xE3\x72\x14\x62\x8C\x42\x54\x05\x87\x23\xC7\xB5\x9F\x38\xF0\xB1\xF6\x79\x05\xFD\xA0\x2B\x64\x54\x7B\x5D\x4E\x1D\xD4\x28\x27\x25\x04\x51\xEC\xBB\xDC\xEB\xC3\x03\x86\x31\x60\xF9\x19\x86\xD0\x7C\xAD\x0F\x7C\x40\x72\x58\x63\x18\x2C\x8F\x33\x0C\x16\x7B\x5D\x85\xD9\x9A\x18\x06\x8B\x68\x18\x2C\xC4\x30\xD8\xB8\xE6\x8B\x31\x0C\x96\x2B\x86\x41\x48\x75\x5D\xCD\x3D\x09\x86\x41\x12\xF6\xC4\x30\x58\x05\xC3\x60\xE1\xAA\x91\x61\xB0\x5C\x35\x0C\x4E\x3D\x18\x32\x2D\x34\xB4\x67\xD1\x89\x2F\x1F\x4D\xA5\x5F\x2A\x4D\x65\x5F\x0B\x9A\x72\x59\x30\x15\x08\x45\x2C\xB1\xF1\x7C\x8B\xC3\x58\x93\x67\xBB\x15\x59\x6B\xCC\xCE\xA7\xA6\xF4\x2C\xF2\x90\xEC\xA6\x9E\xF5\x6C\x3D\x1B\x09\x0F\x0D\x8D\xBB\x92\xE8\x3F\x1B\xB2\xD5\x86\x56\x98\x9D\x64\x81\x9D\x7C\x19\xB9\x44\xF2\xA5\xAE\x68\xFA\xB5\x5C\xD1\x25\x3F\x81\xAA\x46\x18\x09\xFF\x3C\xB1\x19\xCC\xDC\xA3\xAC\xC9\x58\x89\xD3\x7F\x2C\x24\x4B\x26\xFE\x23\x45\xDF\xA5\xE1\xCB\x07\xE8\x68\x95\x2F\xF4\xE7\x85\x82\x93\x6F\xCF\x70\x64\x2C\x67\x43\xE6\xCE\x72\xC9\x6D\x11\x71\xB8\x1A\x33\x57\x1E\xA7\xB1\x97\x43\x62\xE3\x50\xA6\x78\x9F\x44\x92\x7B\xEC\xC7\xD8\x72\xF0\x10\x6C\x7E\xFC\xA2\xE9\xFB\x2A\x97\xD0\xE8\xDF\x4B\x2F\x9E\x8D\x92\x32\xAF\x9F\xB3\x1F\x2B\xF0\xF2\x82\xF5\x47\xF4\xEB\x34\x1B\x09\x0A\x14\x1D\x3C\x03\x40\xD7\x04\x30\x68\x3B\xA6\xC5\x27\xD7\xEF\x98\x1A\x9F\x5E\x28\x7A\x04\x0D\x81\x84\x03\x98\x4D\x09\x64\x22\x57\x9E\x55\x09\x63\x2B\x8C\xF2\x30\xA7\x17\x5D\x79\x5C\xBA\x26\xC3\xB6\x21\x81\xA9\xCB\x5D\x71\x56\xE9\xAE\x6A\xFF\xB5\x04\xE4\x1F\xFB\x14\x75\x63\x35\xC9\xB3\x5A\x93\xE4\x59\xBA\xCA\x7F\x4E\xE1\x3F\x44\x84\x57\x1A\x01\x8A\x2B\xFD\x93\x3D\x92\x3D\xF9\x65\x22\x32\x9E\xEE\x4A\x5E\x10\x22\xB0\x64\xDB\x9C\xD9\xEB\x4A\xCF\x09\x35\x37\x6E\xE8\x4B\x5B\x5E\x1F\xBA\xE2\x75\xDA\x1C\xBC\x5E\x9B\xCA\xEB\x45\x8E\x9B\x48\x2B\x4E\x50\x71\x91\x24\xDE\x33\xAE\x74\xE6\x0A\x97\x77\x1C\x25\x96\x7C\x77\x81\x12\x5D\x5C\x35\x5C\xCA\xB0\x33\x5A\x39\x63\xD9\xE2\xE7\x6F\x02\x91\x4C\xE0\x34\x12\x67\x7D\xF6\xEC\x80\x4C\xA0\x9D\x65\x20\x8D\x8C\x31\x20\xE2\xE3\xEC\x7E\x59\x6D\x22\xA4\x58\x77\x09\x91\x90\x72\x19\x40\x1E\x54\x80\x8F\x8A\x97\x53\x97\x0E\x97\xF5\xB1\x68\x1B\x89\xD3\xD4\x9F\x7C\x48\x92\xD6\x82\xB6\x81\x44\x66\xED\xCF\xEF\xAF\x4D\x91\x8E\x68\x0F\xC9\xFA\x14\xE9\x64\x7D\x8A\x74\xB2\x36\x45\x1A\xF0\x3D\x02\xF9\x90\x4C\x20\x1F\xA6\xD3\x61\x50\x3E\x59\xE0\x31\xEC\x00\xBB\x61\x05\x55\x46\x57\xCE\x8E\x74\x06\x67\x90\x00\x6E\x87\xB9\x36\x34\xFB\x80\xDC\x98\x4E\x77\x40\xDC\x90\xBF\x43\x84\xA3\xF1\x2A\xCE\xFF\x22\x07\x0E\xCD\x15\x4E\xC2\x97\xFC\x2A\x3B\x4C\xB2\xF1\x5A\xCA\x17\xE7\x71\xA9\x8E\xBD\x2D\x40\x1E\x9B\x09\x2C\x47\x8A\x37\x2C\xE1\x87\x18\xCE\xDC\x58\x02\xB4\xB0\x63\x40\x8B\x8C\x01\x2D\xB0\x50\x96\x17\xCA\x2E\x03\x5A\xAC\x2E\x94\x5D\xBF\x50\x76\xFD\x42\xD9\xB8\x50\x46\x16\xCA\x4E\xB0\x39\x9C\x0E\xC8\x1C\xE3\x2D\xB0\x3E\x37\x9F\x13\x9A\x26\xE3\xD1\x3C\x1E\x3B\x8C\xC7\x7E\x35\xC6\x23\xB8\x9C\x32\x18\xC6\xF7\xCA\x19\x33\x2C\x71\xB9\x4B\x2E\x6D\xB1\x70\xA8\xB6\x4D\x1D\x70\x45\x9D\xD9\xED\x12\x94\xBD\xB5\x5E\xF7\x2E\xD9\xEA\x74\x40\x96\xA8\x7E\x80\x0B\x08\xD2\xFD\xA7\x3B\xEB\xF5\xC3\x8D\xF2\xEF\xF9\xF0\x04\xC2\xF0\xF4\xC2\x56\x3E\x40\x00\x18\xD8\x49\x9C\xDD\x23\xAA\x30\x4E\x2F\x2C\xE7\xA2\x38\xEB\xF4\x1E\x10\x1F\xFC\x4F\x8F\x13\x39\x40\x38\x67\x80\xE8\xDE\xBE\x97\x7E\xEA\xEC\x05\x60\x54\xA9\x6D\xE3\xD0\xBA\xB3\xBD\x3F\xEA\xEF\x55\x5C\x26\x88\x21\xA5\x2C\x93\x26\x98\x58\xF5\x2E\x6B\xCC\x91\xBE\x8A\xDA\x21\xC9\xD8\x09\xE5\x8C\xD3\x93\x5F\xB2\x30\xE7\x52\x3A\x34\xBB\xD0\x70\x95\xD3\x45\x8E\xE2\x8B\x2C\x4A\x19\x97\xB3\xBF\x28\x5D\x6E\x32\x3A\xA6\xF2\x58\x59\x14\x49\xF9\x63\xC7\x54\x1E\xD2\x32\xD4\x9A\x06\xD9\x31\x35\x7D\x12\x2F\x5A\xEA\xE9\x05\x5A\x8D\xF5\xFD\x82\xF3\x82\x73\x65\xCC\xF4\x11\x7E\x21\x2A\x9C\x0E\x0E\xA4\x9C\x1D\x7E\x4B\xED\x23\x13\x4B\xF7\x1B\xD5\x71\x6F\x61\x57\xF9\xCA\x23\xA6\x0F\xDE\xE7\xD5\x19\x00\x04\x2B\x92\xF5\xAB\x1F\xB7\x26\x3D\x32\x57\x07\x3F\x44\x28\xF6\xB8\xD6\x6D\x9E\xB2\x1F\x39\x5B\x72\x9B\xE7\x31\xB5\x26\xA7\xE9\x80\x9B\xB7\x98\xB8\x79\x8B\x73\xA6\x58\x75\x9C\x6B\x76\xF3\x16\x13\x37\x6F\x31\x5E\xA2\x42\xCA\xAC\x72\x9B\x25\x5F\x6E\xC5\x4D\x2E\x77\x15\xE1\xF7\xE8\x3E\xCF\x96\x9D\xB9\xEC\x61\x83\x7C\x12\xA4\xF0\x0C\x2F\x62\x1F\x1B\x7C\xE2\x7A\xB7\x91\xB2\x5E\xC4\x25\xBA\xD4\x65\x2B\x6E\x44\xF6\xEF\xD8\x82\x73\x9C\xE1\x20\xCA\xFA\x0E\x70\xAF\x89\x6F\x7B\x6E\x19\xE9\xBA\x6F\x83\x6F\x91\x71\x24\x33\xFF\x5C\x2F\x25\xD4\x59\x62\x58\x72\xDA\x51\x23\xFE\x74\xCF\x20\x35\x96\xCB\x8A\x4A\x88\x59\xEE\xD2\x45\xC6\xDE\xBE\xC4\xE9\x8B\xB1\x9A\x50\x40\x90\x2F\x46\x71\x14\x2B\x4B\xE8\xF5\xB4\x50\x22\x22\x2A\xBC\x7D\x66\x9E\xA5\xE9\x38\x9A\x22\x5D\x5E\x96\xE4\x38\xC7\x72\x9D\x48\x38\x45\xBA\x3E\x9C\xA2\x98\x7A\xB8\x17\x5A\xC2\x29\xB4\xB4\xC9\x1E\xFD\x42\x96\xBA\x08\x71\x02\xEB\x5E\xBF\xD6\xF9\x9F\x8C\x3C\xCA\x9A\x37\x6E\x11\xE3\x30\x47\x77\xAC\x06\x54\x14\xCB\x71\x0B\x76\xF0\x7C\x2E\x35\x61\x6F\x12\x51\xC1\x4C\x41\xB1\x6B\x17\x5B\xA1\x60\xF2\x49\x97\x42\x2A\xF4\xF2\xFB\xF4\xF0\xBE\x61\x2F\xAE\x8F\x93\xD0\x83\x27\x75\xC9\xAF\x5F\xAC\xFA\xC4\x8F\x0B\xA9\x58\xF1\xD3\xEB\x63\x67\x55\x57\x31\x48\x64\x69\x51\x25\x68\x64\xD2\xBF\x10\x94\xD9\xC0\xAB\x73\x7C\x07\x56\x62\x3A\x6E\xDE\x81\xB0\xAC\xB3\xC9\x66\xCF\x8E\xD9\xEC\xD3\x58\x99\xE3\x3D\xD1\xD3\xF9\xE3\xDA\x5C\x5D\x11\xB7\xBA\xC7\x68\xB0\xE1\xE9\x30\x4E\x57\x37\x7C\x12\x37\x7C\x2A\x1B\x3E\xED\xBB\x82\x36\x7C\x31\xDE\xF0\x69\xD8\xF0\x38\x36\x52\xB8\xE9\x79\xC3\x17\x6B\x37\x7C\x8A\x0D\x0F\xB4\x14\xD9\xF0\xF0\x92\x77\x33\x5F\xEC\x36\x26\x6E\xFE\x04\x28\xD1\x58\x1A\x6C\x7E\xDA\xD1\x0C\xFF\x57\xF4\x4E\xEF\x02\x25\x72\x28\x21\x9C\x54\x7F\xBF\xD4\x49\xE0\x03\xF7\x45\x3E\x60\xCE\x7B\xB5\x6D\x7E\xE2\xFD\xDF\x7A\x4D\xF9\x23\xCE\x39\xB6\x7E\xBB\x47\xCD\x4A\x80\x98\xEF\x98\xC2\xE3\x32\xBA\xEC\x3F\x4A\x0F\xB5\x3F\x1C\x72\xDE\xB4\xC4\x6D\x25\xED\x07\x03\x0E\xAE\xE3\xB2\x24\xF7\xD8\x88\x2E\x8C\x04\xC2\xF0\xDD\x19\x7C\x6A\x19\x2B\x69\x9B\xFF\xDC\xE7\xB8\x9C\x2B\xBD\x67\xC7\xDC\x8B\x54\xD3\x90\x16\x7A\x56\xDD\x1F\x4A\x46\x71\x52\x6A\x00\x30\x7E\x40\xAA\x5B\xEB\x8A\xC4\xE1\xD7\x6B\x2C\xE1\xCB\x5C\x35\x91\x1F\x3C\x1F\x1F\x7C\x45\x8D\x9F\xFC\x84\x1A\x1E\x4D\xE2\xA3\x9C\x67\x4F\x1A\xEE\x93\x0E\xB7\x7E\x83\x43\xD9\xD2\x27\xF8\xC7\xC7\xF9\x47\x2E\x30\x39\xAD\x74\x7C\x5A\x32\xB5\x91\xB8\x88\x22\x57\x9D\xF6\x7A\xBF\x4B\xB7\xD8\x60\x00\x41\x4E\x7B\xFD\xAC\xA3\x5F\x6A\x01\x81\xF4\x77\x7B\x46\x17\xF0\x66\x77\xAB\xD3\x4E\xFB\xF7\x6C\x1C\x00\x5C\xE0\xAC\x52\x24\xBB\x99\x03\x40\x0E\xB8\xF0\xFD\x5A\xCB\x3F\x2C\xC2\x0F\x05\x7F\xEF\xE2\x0D\x8E\x7F\xF8\xCF\xF0\x83\x53\xFE\x6E\xA0\xDC\xE8\xA7\x0F\xE8\xA2\xE6\x8B\xFF\x79\xBC\x5B\xDA\xFF\xCB\xE1\x07\x69\xFE\xBF\xE0\xA7\x51\x59\xEC\x9F\xA0\x7A\xAB\xDE\x31\xDB\x26\xA2\x5E\xC6\xDA\xCB\x9A\xEF\x79\xD7\xF3\x0C\xFE\xAC\xB1\x72\x5C\x4F\x4F\x7B\x1A\xCF\xC6\xE5\xAD\x1D\x73\x5F\x30\x04\x9D\xE9\x52\xE4\x9F\xB7\x02\x37\x8D\xA9\x2B\xB8\xEA\x29\x30\xA8\x3A\xED\xDF\x26\xE0\x6E\x90\xD1\x2B\x86\x05\x6D\x7B\xFF\xB6\xFD\x80\x14\xBF\x10\x24\xCD\x1A\x4E\x06\x7D\xC0\x13\x9C\xEC\x98\x96\xCB\x12\x6A\xAF\x0F\x1F\x54\x05\x4F\xB0\x3E\x58\x24\x74\x16\xB0\xFA\xE9\x74\x34\xE5\x24\x4E\x33\x0E\x36\xAC\xF4\xA3\xA2\x64\x4C\x61\xC1\x4C\x69\xE2\x06\x34\xA1\x26\x40\x04\x7E\x0F\x16\x2E\x8D\x84\xD6\x9F\x53\x7C\xB9\xFD\xCE\x50\xCE\xDD\x22\xB3\xB9\xE6\x84\xB6\xED\xF6\xF3\x88\x69\x31\x02\xA4\xFE\x38\xEA\x16\xD1\xDE\x80\xFE\xF7\x8E\x87\xEC\x51\xFB\x8B\x21\x7D\xFE\xEB\xA9\x4F\xC3\x34\x2F\xB4\xBF\x2F\xE6\x2C\xEB\x87\xFE\x97\x6B\xD7\xAE\xBD\x90\xF0\x11\x81\x24\x7B\x43\x03\xE9\x9C\xF2\x1B\xD8\x47\xBC\x0C\xC6\xAB\x7B\xD4\xFB\xBF\xEF\xFA\x35\x75\x4E\x71\xEF\xE9\x87\x97\x86\x1F\x5A\xFC\xF0\xD1\xE1\x87\xD3\xF8\xE1\x27\x86\x1F\x5C\xEF\xD5\xEB\xD4\x27\xE8\x87\xD7\x2B\x15\xA1\xFC\x1E\x6F\xFF\xE7\xE7\x81\x70\xF0\x04\x7F\xFF\xFA\xF6\xEF\xD2\xD0\x1E\x54\xDF\x20\x7E\xF6\xBB\x6F\xBF\xD8\xDB\x7D\x2B\xEE\xC0\xB7\x69\x7D\x34\xD0\x17\xEC\x73\x1E\x43\x02\x6B\x60\x18\x65\x9A\xFD\xE9\x3D\x09\xDF\xF3\x09\xBE\x07\x1E\xC8\xE3\xEE\x79\x65\x7A\x4F\xF5\x4D\x5A\x90\xD3\x05\x47\xFD\x10\xDF\xCC\x53\xB4\x95\x51\x58\x8D\xBE\xDD\x8F\x33\x30\x80\xA4\xBB\x1E\x26\xAF\x36\xB0\x37\xFA\x96\xF0\x97\xD7\xF4\x5E\x33\x8B\xD9\xEC\xC1\x57\x40\x51\x6C\x66\x04\xDB\x91\x8F\x88\x42\xFA\x1D\xC6\x49\xAF\x0E\xB4\x5E\x05\x33\xE7\x2C\x67\xA9\xA1\xF5\x14\xB3\xE6\xD0\xEB\x33\x82\x6D\x40\x4C\x5A\x03\x0B\x02\x3D\xF5\xBF\x07\x60\x84\xB6\xF7\x3F\x3A\x61\xD2\xDB\xE6\x49\x64\xC0\x06\xD8\x62\xA7\xAA\xB3\x82\x21\x8F\x54\x6F\x52\x8A\x05\xBC\xFD\xF1\x48\x93\xF6\xA8\xFD\x25\xAE\xB4\xF4\x64\xF5\x0B\x06\x10\xF8\xB8\xA1\x13\x2C\xF6\x58\x3E\x93\xA8\x7B\xDD\x8F\x5F\x1F\x7E\xFC\x14\xD7\x50\xFD\x96\xE1\xD2\x93\xC7\x5F\xBA\x57\x2E\xC5\xD3\x80\xDD\x9C\x7A\x20\x25\x3D\x22\x25\x3D\x79\xF6\xCD\xA3\x67\x3F\x21\xCF\x72\xC4\x7B\xD2\xFE\x3E\x83\x3D\x4C\xEE\x7F\xCB\xE8\xFE\x57\x96\xEE\x7F\x75\xE5\xFE\xD5\xD3\x64\x72\xF1\xFE\xB5\x1C\xE4\x26\xBB\xA0\xBA\xA4\x06\x41\x95\x8E\x51\xE9\xA3\x1A\x75\x98\x47\xDE\x86\x91\x73\xD1\x88\xE5\xC1\x57\x0F\x2F\x37\xF4\xEA\xB8\xA1\x57\x43\x43\x7C\xAE\xC4\xB6\xCE\xAC\x6D\xEB\x5F\x98\x41\x48\x78\x97\x8E\xE9\xCF\x9E\xF5\x61\xB5\x6D\xEE\xE3\x71\x3C\xE1\x52\xE0\x60\x7A\xE5\xDF\xA9\x19\x8D\x80\xCB\xB0\x43\xDE\x21\xE6\x4A\xD2\xCB\xC1\x56\x97\xFB\x93\x97\x1A\xE5\x8F\x60\x63\xEF\x4A\x97\xB3\xD1\x7E\xDF\xE5\x17\xB7\xDA\xEF\x67\xF7\x3D\xA2\xB6\xB8\x8D\x32\x60\x24\x6A\x2F\x01\xFD\x86\x04\xDB\xF6\xFF\x91\xE8\x3F\x62\x31\x2C\x7C\x15\xB0\x65\x6F\x9B\x84\xEB\x5C\xEC\x89\xB1\x87\x21\x06\x20\xE7\x21\xEE\x0F\x31\x61\xA9\x57\x57\xB6\x00\xB1\x84\xF0\xC8\xA3\xBE\x4B\x21\x72\xFA\xA4\x5F\x68\xB6\x0A\x59\x06\x75\xCF\x43\x99\x10\xC5\x4F\x5B\x96\x70\x72\xF9\x4F\x72\xE2\x5A\x30\x20\xD4\xAA\x62\xD3\x6C\x79\x01\x80\x7C\x30\x7B\x97\x38\x6B\xCA\xF6\x7D\xE2\xE6\xF2\xEF\x0A\x09\xE6\x2E\xAF\x3E\xC3\xA6\xD9\x7C\xB4\xD9\xF3\x38\xBD\xC5\x30\xBD\xC6\x59\xC9\xC6\x49\xBA\xD2\xD9\xCB\x5C\x4B\xF6\x1E\xA3\xBA\xFF\x9F\xBD\x77\x81\x93\xA2\xBA\x12\x87\xCF\xA9\xAA\xAE\x7E\x54\xF7\xD0\x32\x18\x51\x06\xBB\xBA\x17\x75\x50\x06\x06\x54\x54\x14\xA5\x40\x5E\xF2\x8C\xE0\xFB\x31\x14\x33\xCD\x4C\xF7\xF4\x74\x0F\xDD\x3D\x28\xC6\x30\x93\x28\x6A\x8C\x6B\x4C\xB2\x49\x36\x89\x89\xE4\xEF\x46\x44\x62\x56\xF3\x36\xD1\x85\x3C\x75\x93\x35\x81\x28\x6A\xDE\x98\x64\xB3\xEB\x26\x21\x68\x7C\xF0\x50\xEA\xFB\xDD\x73\xEE\xAD\xAE\x2E\x06\x93\xFD\xFF\xF7\xFB\xBE\xDF\xEF\xFF\x4B\xC1\x99\x5B\x75\xFA\xBE\x9F\xE7\xDE\x73\xEE\x39\x89\xAD\x36\x9B\xCB\x1E\x6F\x47\xED\xC4\x5C\xFB\x8E\xAD\xB6\xD6\x6F\xEB\x76\xBC\xBF\x98\x4B\x8A\x79\x21\x3A\x97\x49\xAB\x76\xED\xCA\x6C\xAA\xC1\xCD\x5D\x4D\x96\x9C\xF9\x9E\x98\xA6\xCE\xFB\x5B\x3A\x00\x73\x63\x58\x8B\x8D\xBE\x30\xA5\x39\x5A\x36\x6D\xC7\x52\x68\x39\x48\x67\xF1\xDA\xC2\x09\xB6\x61\xC7\x26\xF0\xAD\x0F\xA5\x26\x84\x06\x42\x4E\x17\xB4\x8C\x2E\x48\x1C\x7D\x02\x2B\x38\xD7\x52\x68\x27\x6D\x63\x31\x55\xEB\x53\xD2\x2B\x69\xFB\xC1\xA2\xAD\x4F\x90\x2A\x49\xA1\x03\x96\xCE\x67\xFB\x9B\x8B\xC8\x9C\xB1\x51\xCC\x1E\x67\x5B\x24\x95\xAA\x77\x00\xD8\x31\x91\x4F\x8D\x68\x48\x5B\x0B\xE8\xD3\x9A\x20\xDC\x85\x29\xC3\x8E\x89\x66\x8C\xD1\xC1\x51\x4E\xB7\x8F\x5B\xEC\xAB\x2B\x25\x0B\xA4\xA2\x0F\x11\x89\x2C\xF2\xD2\x4F\x35\x22\x7E\x10\x33\xA5\xA3\x39\x68\x13\xA9\x95\x4D\x5B\x0E\x88\xD9\xF5\xDF\x68\x2D\x00\xE7\x78\x31\x30\xB2\x9A\xDD\xD2\xAE\x2D\xCA\xD2\x5D\x8D\x49\xD9\x98\xC8\xEB\xB5\xA4\xDB\x0C\x66\xC1\x18\x5B\x77\x60\xA1\x9D\x12\x85\x31\xED\x94\x6D\x2E\x9A\x90\xD3\x1D\x73\xD1\xFA\x6C\xD2\x19\x19\x49\xDB\xB1\x81\xEC\x71\x0E\x28\xF3\x09\xFE\x9E\x3A\xDD\xA2\x83\x86\x71\x96\x7C\x21\xB6\x3E\x0D\x90\x31\x72\x7D\x4C\xFF\x81\xA8\xAB\x16\xDA\x6F\xA4\x92\x86\x45\x3A\x61\x30\x1B\xB3\x5B\xD8\x7C\x1F\x3A\x82\x8E\x12\x83\x66\x4C\xD1\x81\xF4\x66\x32\x17\x6B\xD8\xD1\x0E\x18\x93\xD2\x93\x51\x4B\xF6\x1B\xDF\xB7\x92\xCD\x09\xF8\x26\x03\xFE\x63\xE6\xA7\xE2\x94\x84\x6E\x49\xAD\xCD\x98\x8D\x25\xE3\x56\x20\x13\x71\x99\x09\x3A\x02\x8C\x51\x2C\x69\x95\x01\x7A\x1D\x25\x0F\xF3\x53\x31\x0A\xA3\x59\x8E\x96\x25\x46\x45\x8A\x74\x07\xC7\x9D\xF7\xD8\x8B\xD7\xB3\x39\x69\xDD\xC1\x4B\x52\x68\xC7\x9D\xF8\x62\x96\xE7\xB2\xB8\xC7\x1D\xB7\x84\x72\x91\x92\x1E\x28\x55\xFA\x6E\x14\x9D\xB4\x5C\x87\x53\xD6\x64\xCA\xA6\x45\xFE\x58\x4E\xD9\x9C\x9F\x32\x54\xF2\x14\x1F\xA5\xBB\x50\xEC\xE3\xED\xE8\x64\x3D\xCD\x3B\x77\x91\x49\x8A\x31\x1B\xA3\x6B\xAE\xBC\x23\x61\x79\x90\x9A\x9D\x9A\x5F\xE5\xA9\x6D\x35\x19\xD1\x26\xD5\x45\xB6\x9E\xFE\x38\xA7\x2A\xE9\xB8\xD5\xA4\x55\x85\x8A\x20\x66\x03\x2A\x05\x49\x80\xA3\x9D\x92\x01\xBE\x78\x4B\x23\x40\xD3\x0F\x1F\xA4\x98\xB8\xA4\x2C\xDA\x45\x03\xC7\x54\xBA\x2E\x49\x26\x8E\x38\xFD\x52\xB5\x01\xF5\x60\xDA\xD7\x91\x58\xCA\x72\xF2\x45\x07\xF7\x0E\x99\xE7\xEA\x77\xF4\xBA\x54\x04\x63\x91\x1D\x47\x39\xD3\xC4\xAC\x71\x44\x3E\x90\x9E\x0A\xCD\xD6\xD3\x2F\xF3\xB2\x0C\x96\x45\xBB\xAC\x7F\xBC\xFF\xF6\xE0\xE7\xBD\xF4\x79\x3E\x93\x41\x68\x0A\xC2\xE7\x7C\xA4\xD3\xB8\xF3\x51\xE3\x3D\x0F\xB2\xF1\x1E\x31\x56\x5A\x04\x09\x83\x92\xD6\x61\x0A\xC7\xFA\x12\x4A\x13\x39\x6A\x8C\x83\x73\x73\xB1\xE5\x38\xD4\xE9\x01\x5D\x3E\xA4\xB8\x56\xDA\x12\xC6\xF9\x29\xED\x2F\xAE\xEB\x36\xDF\xD0\xF1\x43\xA0\x8D\x3C\xB3\x8A\x1F\x94\xA2\x1D\x49\x4A\xC8\x19\x37\xE0\x5B\xAC\x21\xC9\xA2\xD2\xBF\x91\xBE\xB7\x69\x45\xB7\xC1\x7A\xDC\x40\x7D\x58\xAC\xF7\x36\x2C\xC8\x69\xB3\x20\xC9\x64\xB7\x2E\x55\xCD\xE8\x1D\x90\x64\x15\xFC\x4C\xD6\xC6\xC8\xC2\x69\xCE\x10\x73\x25\x30\x7F\x5E\xDA\x76\x30\x84\x4F\x36\x04\x95\x35\xA4\x1C\x8D\x54\x5B\xB5\x30\x70\xBF\x80\x79\xE1\x74\x82\x8F\x4C\xD9\xA1\xD4\xB9\x26\xAF\xFD\x92\x59\x2F\x83\x89\x7A\x4E\x90\xCE\x06\x6C\x7D\x21\xF3\x35\xA4\xB2\xFE\x99\x5A\x4C\x99\xDD\x34\x54\x20\x41\x75\x26\x79\x27\x90\x54\xF7\xD8\x39\x39\x96\x4D\xE2\x7C\x52\xD4\x16\x5F\x66\xA0\xBB\x5B\x1A\xF3\x19\x48\x24\x54\xE3\x54\x74\xC9\x49\x10\xDD\x90\x75\x45\x05\xFA\x6A\x73\x49\xB5\x60\x49\xE9\xE8\x20\x8B\x72\x13\x76\x74\xD2\x4D\x85\xC4\x63\x16\x52\x5A\x24\x40\xD6\x1E\x46\x1A\x9C\x51\x05\x3A\xBA\x90\x06\x5B\x19\x31\xB8\x3A\xFF\x77\x0A\xA9\x14\x62\xD1\x5E\x4F\x9A\xDE\xD1\x65\x42\x62\x87\xB6\x20\xA7\x8B\x77\x83\xC9\x7A\x2D\x8B\x44\x7E\x58\x96\x75\x67\x04\x23\xC7\x32\x41\x34\x0D\x8D\x9C\xE6\x1C\xF1\xC4\xD2\xCD\xE7\x2B\x2F\x34\xCE\x57\x46\x21\xDD\x9D\x17\xE4\xE6\x26\x5D\x74\xBE\xD6\x20\xDD\x29\x22\x31\xAF\x42\xBB\xF6\xAC\xF0\x91\xD3\x99\x36\x21\x31\x83\x08\x71\x01\x3D\x4F\x5F\x2F\xFE\x28\x89\x3C\xB6\xD0\x46\xDE\xF4\x39\x4A\xAE\x28\x9D\xD3\x6C\x90\x62\xB7\x90\x13\x2B\x75\x84\x0D\x90\x68\x36\xAC\x10\x93\x24\x47\x1B\x91\xD6\x6A\x74\xDB\xA0\xD3\x27\x36\xA5\x26\xEF\x29\x44\x1A\x26\x6B\x02\x56\xFA\x35\xD2\xC0\x65\x6B\x73\x25\xCB\x01\x98\xA5\x2F\x2B\x76\x5C\xB1\x21\xD6\xA9\xF3\x46\x83\x8E\x00\xA4\x0E\x38\x9D\x15\xA2\x13\xDD\x15\x91\xEA\x35\x22\xE9\x43\x6C\x28\x9E\x3A\x07\x89\x56\xC5\x2C\x8A\x4A\x29\x77\x23\x35\x66\xAA\xEF\xFD\xDF\x59\x5C\x10\x2D\x2A\xE6\x60\x8A\xD3\xF3\x74\x79\x54\xC5\x3C\xDA\xF3\xD1\xB0\x9A\x77\x73\xE7\x89\x4D\x33\x38\xA8\x6C\xE3\xEE\xA2\xDE\x36\xDA\xEE\xF7\xD9\xD0\xEE\xF7\xE3\x26\xD9\x4F\x51\x9D\x16\x03\x5D\x89\x3B\x2E\xF7\x3B\x1A\x6A\xA8\xAC\xDF\x33\x2B\x50\xDD\x3A\x37\xB8\xAE\xD9\x74\x47\x8A\x6E\xD4\x18\xA2\x9E\x0D\x4A\x24\xAB\x26\x83\xF9\x29\x43\xCC\x0F\x96\x8D\x82\xD4\xCE\x22\xD5\x33\xC5\xC6\x38\x79\x78\x00\x64\xC8\x44\xD5\xB7\xC1\x13\x81\x31\x57\x5E\x3E\xA2\xFA\xD6\x79\x26\x34\x9A\xEA\x5B\x93\xF5\x8D\x74\x93\x06\xD9\x24\x25\x4F\x26\xA2\xBE\x95\xA4\x37\xCA\xFA\xA6\xD9\xC3\xAF\x6F\x43\x2D\x1A\xCE\x70\x51\x5A\xB5\xF1\x3C\x9D\xEF\x10\x6A\xE7\xA3\x11\xAA\x08\xBA\x06\x47\x56\xAE\xCD\xAC\x6E\x47\xB2\x48\x8A\xCC\x68\xE3\x98\xB6\x61\x71\xCE\x98\x20\x4A\xCD\x96\x8C\x65\x9E\x44\xEB\x9B\x74\xCB\x88\xFC\x2C\x91\xBA\x83\x69\x8A\x46\x79\x58\x94\xA3\x3B\xB0\xBA\x52\xC6\x4F\xB7\x90\x78\xB5\xA0\x2F\x4E\x80\x7D\x49\x25\xFD\x62\xA9\xCA\xA9\x62\x5B\x4A\x01\x0A\x2D\x50\xDC\x39\xE9\x08\x8B\x4A\x3A\x2C\xF5\x13\xE9\xE9\x23\xDC\xE1\x78\x1D\x1C\x63\x85\x9B\x3A\xE2\x37\x75\xE4\x7F\xB4\xA9\x23\xFF\x63\x4D\x1D\x39\xBA\xA9\x23\xCD\x4D\x1D\x09\x36\x35\x4D\x1B\x4D\x4D\x4D\xE3\x95\xAE\x84\x8A\x06\xCF\x11\xB5\x43\x2D\xBE\x9E\x94\x83\x79\x44\x2D\x59\xFB\xE3\x68\x0E\xDB\xC8\xA7\xD0\x9A\x7F\xC1\x44\x9F\xAF\xB6\x66\x06\xDB\x77\x33\x24\x5D\x40\xEC\x37\x5E\x93\x52\x48\xCA\x1E\x25\x9F\x18\xC5\xEA\xE2\x2F\x47\x0E\xD0\xDC\xC2\x6D\x15\xCB\x49\xEB\x70\x24\xE7\x64\x2A\x93\x69\x1A\x89\xD2\x3B\x74\xD6\x99\xA5\x25\x89\xE4\x44\x95\x2C\xF5\x4C\x4D\xCC\x65\x1D\x90\x94\x16\x5B\xA5\x4E\x48\xEA\x1D\x24\xF7\x4E\xB7\xD8\x63\x4C\x80\xB0\xB9\x92\x99\xF2\x76\x35\x9B\xFE\x89\x09\xCA\x7A\xA6\xD2\x86\x69\x90\x14\x9E\x62\xA5\x46\xE5\xFA\xA9\x09\xDA\x2C\xC9\xEB\x20\x9D\x25\x2C\x98\x90\x95\xDA\x86\x3D\x79\x9A\xC1\xBB\x18\x31\x05\x8B\x6A\x10\xF9\x61\x8B\x45\x86\x22\x96\xC8\x84\x84\x14\xC0\x58\xCF\xBB\xBE\xA3\x57\x70\xAE\x55\xCE\x81\xDC\xDA\x73\x0E\xB0\x79\x05\x67\x2D\x96\x6A\x05\xC7\x99\xEA\x00\x14\x78\xA0\x33\x03\x19\xE4\x9C\x2D\x32\x4B\xD1\x67\x8D\xA0\xF2\x2B\xC3\x57\xE0\x27\xAA\x6F\x3E\x6F\x48\x0C\xBF\xC7\x23\x21\x89\x9E\x02\xA2\x07\x55\x61\xD4\x75\x50\xC9\x60\x11\x54\x90\xA9\xA8\xAE\x31\x96\x1F\x83\xE8\xDE\x92\x42\xE2\x93\x5F\x8D\x0B\xCC\xD1\x31\xDB\xD9\xC8\x71\x47\x96\xAA\xE8\x88\x9C\xB3\x58\x02\x80\xCA\x4A\xE9\xC9\xFB\xF0\x46\xA0\xF5\x0C\xD5\x70\x06\x59\xD9\xA0\x19\x4D\x92\x36\xB2\x9D\x64\xED\x1A\xD2\xEA\x1C\x99\x92\x63\x3B\xE9\xC8\xC4\xED\xFF\x41\x8A\x4C\x6F\x1F\xAB\x1D\xB9\xFC\x5C\x2C\x49\xD3\x72\x13\x1A\xCD\x4D\xC8\xDA\x27\x54\x13\x1A\xDC\x76\x92\x9E\xD5\xD8\x78\x08\x4D\xB9\xAA\xF5\x40\xB6\x9E\x9C\x06\x55\x13\xF9\xF5\xAB\x7E\x33\x9A\xDB\xD4\x6F\x3E\xD9\x78\x84\xA6\x39\x65\x21\x5F\xA5\xE0\x66\x93\x99\xE5\x5D\x88\x8A\x54\x93\xD6\xF5\xB4\x51\x1B\x8F\x6B\xF4\xFF\xB0\x2A\xFF\x62\xE3\xC9\x65\x86\x4E\x9B\xFF\x27\x1A\xCF\xE0\x2E\xEA\x93\xDA\xC6\x02\x25\xD7\x6B\xB0\xC5\x3E\x4A\xC3\xB2\xB6\x6B\x01\x62\xC0\xA4\xF9\x15\xD2\x07\x1B\x47\xB4\xCC\x0B\x86\x99\x5A\xD2\x01\x56\x59\xAC\x96\xCA\xF5\xE4\x99\xCD\x0A\xD3\xB4\xC5\x1C\xF2\x76\xD2\xDF\x87\x0B\x26\xE4\x22\x59\x93\xCD\xCF\xA8\x8B\xAB\xA8\x8C\x3D\x23\x13\xCB\x4C\x77\xA8\x23\x64\xC2\xF2\x10\x42\xD2\x61\x9A\xD3\x88\xEA\xD1\xE4\x2C\xCD\xE1\x4C\x0A\x10\xCA\x20\x6D\x1A\x92\xB4\xCC\x28\x7B\xD0\x4A\x8F\x37\x15\x6A\x71\x4A\xCA\x84\x69\xA4\xA3\xBD\xB1\x6F\x84\xF3\xD1\xB4\x9A\x0B\x2C\xE6\x23\xB4\xF5\x69\x68\x0A\xBA\x9F\x8D\x02\x8A\x09\x59\xD0\xFF\x87\x94\x3C\x95\xD8\xC1\x92\x81\x89\x9F\x7F\x24\x6C\xE8\x85\x48\xD1\x94\xE4\x06\xD8\x7C\xDA\x39\x53\x03\x69\x78\x52\xEC\x3C\xC4\xCE\x5F\x2B\x90\x34\x16\xF9\xA2\xFD\x8C\x98\xF6\x9D\x61\x52\x9D\xCD\x7A\x63\xC5\x5E\x42\x14\xDC\xD6\xFC\x33\x53\x39\xF1\x19\xCC\x8B\x52\x96\x50\x48\x26\x93\xF5\xBB\xEA\x2C\xEE\x36\xCE\xD6\x8A\xA2\x33\x28\xEB\xC2\x52\xC1\xAA\x3C\x3E\x6B\xA8\x36\x37\xD4\xC6\xBA\x5D\x4B\x4B\x2E\x15\x2D\x99\x69\xB5\xA3\x96\xC2\x75\x62\x94\x5B\x2F\xF3\x7E\x9F\x2D\xF9\xE9\x3C\x37\xE2\x34\xD4\x78\x19\xC3\x31\x72\x5F\xE8\xD0\x5D\x70\xDD\x39\x82\xEB\x1D\xB2\xFC\xA8\xFC\xA0\xBF\xEA\xEB\x7C\xC8\xA0\x3B\x51\xD2\x29\xEB\xBC\x67\x84\xA6\x37\xDD\x49\xCA\x63\x06\x9D\x97\x7E\x64\xFD\xD7\x31\x3A\xEB\x8D\x51\xE3\x3B\xDA\xE2\x09\x62\x5F\x4F\x8C\x35\xDA\x2E\xCA\xBE\xA3\x8F\xB1\x9A\x73\x73\x74\x6A\x74\xA4\xA1\xDB\xC4\x0D\x96\x9B\xCD\xB7\x4F\xDB\x6A\x01\x27\x26\x56\xC1\xF4\x7B\x36\x8B\xF6\x6D\x01\x27\xDA\xFC\x19\x0F\x7C\x1E\x34\x31\xA6\x4C\x6F\x6B\x2D\x1A\x82\x46\xBA\x78\xA0\x91\x15\x12\xB2\x03\xC7\x14\x39\x34\xF8\xD8\xC5\xE0\x63\x17\xC3\xD6\x45\x1E\xE9\x4B\xCC\xB4\xEA\xBC\xC5\xF0\xCF\x5B\x68\x38\x8F\x1E\x46\x7B\x9B\x30\x4E\xD0\x70\x9B\x4A\x9A\x49\x7A\x79\xFD\x43\x67\xA5\xF2\xEA\x56\x5D\x44\xD0\xB0\xCC\x31\xCD\x19\x59\x24\x43\x49\x7A\xFA\x51\x3A\x2D\xD4\xF8\x70\x8F\xF6\x32\xA6\x9A\x89\x45\x44\xC6\xA2\x00\x4D\x67\x72\xF6\x4C\xCE\x9E\xC9\xD5\x6E\x72\xB5\x9B\x7E\x35\x9B\x7E\x35\xD3\x9E\x99\x36\x3C\xE5\x9C\xA0\x9E\xB9\xAF\x93\x2E\x50\xE3\x7C\x34\xD9\x46\x58\xCC\x8E\x66\x63\x64\x1D\x30\x17\xCF\x26\x6C\xCC\x5A\xFA\x9C\x06\x15\x19\xAA\x10\x4A\xD1\xE0\x14\x0D\x3F\x45\x43\xA5\xE8\xBC\xE9\x8D\x91\x89\x88\x29\x98\x76\x5D\xA6\x6D\xB0\x9C\x7F\xD2\xB6\xF8\x77\x56\xB5\x1C\xE3\xC9\x3B\xC6\xF2\xE6\xF4\x53\x31\x6B\xD9\x09\x91\x91\x04\x1B\xE5\xA5\x1B\xE1\xC4\xDC\x8B\x8B\xB8\x4A\x54\x88\x22\xD9\xC6\x17\x9D\x2E\xBD\x99\x19\xA7\x62\x9E\x6A\xD4\x11\xB7\xE4\xDB\x57\x55\x8B\x5F\x55\xB2\x41\x55\x6D\x2F\x4E\x81\xF3\xA1\x7B\x02\x16\x62\x45\x0F\x0F\x34\x88\xB6\x48\x9A\x40\x96\xA5\x54\x3E\x44\xD4\x52\x75\xB6\x44\x50\x8B\xB2\x4A\xD2\x2C\x29\x92\xD7\xD7\x67\xD5\xF9\xB2\xAF\x45\x9A\x74\x48\xB3\x91\x25\xF1\x13\xAB\x49\x28\xE6\x92\x64\xDE\x34\x97\x90\xF2\x34\x49\x31\x4E\x13\x4E\xAC\xA8\x94\x46\x38\x69\x3A\xB5\xE0\x9B\x99\x31\x16\x54\xB3\x98\x4E\x20\xA5\xAA\xB4\xBD\x01\x5B\x6A\xA7\x88\xA9\x59\x4D\xB1\x62\xFC\x2A\x1F\x2E\xE6\x2C\x0A\x1C\x99\x9F\x6A\xA8\xE8\xB6\x6C\xB3\xC1\x3F\x1E\x0B\x4E\x84\xD9\xAB\xAA\xBE\x05\xCA\x0C\xA1\x16\x8A\x0D\xB3\x3F\x2B\x34\x06\xBC\x8D\x3C\x3F\x58\xAC\xE1\x1A\x42\x13\x43\x63\x2A\xD2\x68\x77\xD2\x1C\x8F\xC1\xA1\x0D\x27\x41\x23\x8E\x43\x1B\x2A\x34\x9D\x80\xA5\xA9\x13\x72\x24\xFE\x4E\xC4\x36\xAC\xD7\xD5\x84\xEA\xC0\x25\x81\xF1\x73\x8C\x03\x59\x8D\xF9\xCA\xA3\x9C\xC4\x4A\x2A\x8D\xC6\x26\x71\xB5\x1A\x33\x6D\xC0\xFF\xE8\xB1\xEA\xE7\x37\xC7\x65\x33\xFB\x3B\x38\x7D\x72\x40\x9D\x03\xFA\x93\xA9\xC1\x53\x77\x8B\x3F\x6B\xCA\x28\x68\x57\x2C\x72\x91\x1E\x61\x01\x11\xD1\x8C\x6A\xCF\x45\x5B\x62\x6B\x7E\xB0\xFE\x34\x3F\x0A\x8D\x13\xD2\x38\x21\x8D\xEB\x55\x73\xC6\xB1\x86\x70\x95\x43\x79\xFC\xA2\x59\x37\x9B\xC8\x37\x1E\x36\x09\xD7\x3F\x93\x3B\x89\x0E\xD4\x72\xDA\x64\x60\xB3\x11\xB1\x62\xCE\x98\xCC\x62\xAA\x4A\x80\x11\x27\x03\x09\x3B\xB0\xCC\x1E\x0B\x81\x46\xED\xC8\x39\xD2\x3E\x7A\x8E\x3C\x8A\xFE\xA4\x0B\x14\xDA\xA6\x70\x0C\x3B\x46\x02\x12\xE4\xCF\xFA\x63\xB4\x61\xF4\x2C\xDD\x9F\xD3\x1A\xE7\x80\x8A\x68\xC3\x14\xDB\xEC\x5B\xA0\xC8\x23\x9D\x96\x8A\xBB\xA5\xA0\x88\x58\xAB\x59\x64\xC8\x70\xDE\x55\x74\x4C\x92\xBB\x66\xFD\x94\xDA\x5C\x66\x50\xF0\xBF\x80\x59\x33\x83\xB8\x62\xEF\x2A\x2A\x35\x12\x9A\x48\x41\x90\x37\x86\xB2\x35\x4D\xC2\xB8\xB1\x22\x99\x1A\xD0\x59\x70\x95\xCD\x2E\xC6\x78\x9E\x5B\x2F\x96\x4F\x5B\x97\x57\xC2\x22\xCD\xCB\xAA\xC4\xC7\x55\x1E\xA3\x6A\xBF\x6A\xB0\x3E\x14\xBA\x98\x6D\x28\xA5\xA9\x62\x94\xD2\x11\x2C\xDF\xF2\xD0\xA4\x16\x8A\xC6\x05\x45\xC3\x57\x22\x21\x65\xEC\x2D\x75\x37\xD1\x08\x28\xA0\x50\x17\x13\x0D\xC9\x88\xF2\x0D\xD9\x1A\x2B\x52\x31\x3A\x4C\x91\x4B\x67\xCA\xB2\x53\xFE\xC5\x44\xCA\x43\x8A\xCF\x5A\xB2\x31\xBE\x98\x68\x34\x2E\x26\xA6\xEC\x44\xE0\x62\xA2\xC6\x1A\xCB\xE4\x1E\x4A\xB3\x53\x61\xED\x13\xC1\x2D\xA3\x1D\x2C\x6A\xBC\x51\xD4\xE6\x82\xA6\xE4\x8D\xB4\xC4\xA8\x05\x8D\x89\x82\x5A\xB6\x15\x28\x68\xC2\x2F\xA8\xD5\x30\x66\x68\x35\x0A\x6A\x35\x0A\x4A\xC7\x2B\x74\xE5\x32\x97\xE0\x1C\x18\xF4\x73\x31\x1B\x67\x25\x4E\x7C\x9B\x5C\x5A\x31\x4C\xD9\xB1\x40\x41\xE3\x5C\x50\x8B\x0B\x4A\x2B\x57\x4A\x14\x34\xD5\x10\xBE\xA1\x13\x1B\xB2\xC1\x14\x95\x9D\x67\xB1\x7F\x86\x21\x37\x3B\xFD\xA2\x68\x86\xE4\xDD\x1A\x2D\x5A\x04\xD0\x72\xDE\xB7\x3D\x74\x4D\xB0\xE1\x97\x8C\x93\x3B\x5A\xDD\x86\xA2\x73\x33\xED\x30\x48\xF8\xDC\xD9\x54\xE4\xA5\x90\x34\xA1\x66\x59\x49\x96\xBE\x4C\x5A\xB2\xD0\xFB\xD3\x1F\xE0\x09\xA2\x03\x80\x18\x28\x02\x49\xC6\x63\xF8\xA6\x9D\xC4\x6B\x56\x60\x5B\x01\x0D\x0A\x30\x07\xA3\x53\x86\x39\x7F\x8C\x19\x96\x8A\x83\x79\xF6\x7A\x52\xF3\x17\x0C\x1A\x8F\xB4\xD7\x14\xBB\x80\xF9\xEA\xE4\xCC\x68\x88\x29\x69\x0D\x21\x4A\xDD\xFA\x90\x8E\xDA\xB0\xCF\xE6\xCA\xD1\x29\x09\x1D\xD6\xA0\xBA\x08\xD2\xC4\x00\x16\x2B\x67\xCB\xF1\x89\x04\x6A\xBA\x6E\x18\x46\xC4\x4C\xF0\x03\xB1\x68\x5C\x52\xA9\x30\x0D\xB5\x9A\x33\xB2\x13\xD6\xCF\x1F\x23\x67\x87\x76\x8D\x75\x2C\x2E\x4C\xC5\x82\x7E\x50\xF8\x90\xBB\x0A\xF9\x3B\x0D\xEF\x66\x3F\xA9\xA8\x4F\xB2\x63\x80\x74\x67\xDE\x38\x45\x30\x0D\x4D\xAA\xA7\x05\x63\x9A\xB2\xC0\x39\x08\x61\x52\xBA\xB4\x6B\xD9\x14\xE3\xC2\x94\xD1\xC8\xE5\x82\xA3\xB3\x7D\x54\x80\x05\x32\xE3\x2A\x00\x97\x82\xC5\x22\x89\x89\xD7\xAE\x19\x7C\xE4\xC5\xC6\x6F\x1C\xBD\xCE\x65\x14\x6F\x82\x2E\xE1\xBD\xA4\xF8\xA3\xF9\x25\x71\xD8\xA8\xB5\x75\x9F\x86\xCA\xCA\xB4\xD8\xE3\xD2\x65\x20\x5D\x74\x47\x1B\x17\xA7\x1A\x67\x12\x8E\x56\xCF\xF2\xDD\x18\x31\xFD\x4A\x23\x37\x24\xCF\xAD\xD3\x1E\x59\x53\xEB\x55\xCE\x90\x96\x48\x4D\x3B\x78\xBD\x9A\x72\x47\x17\x51\x05\x41\x27\x48\x60\xB6\x55\xDA\xB8\x60\xAD\x05\x2F\x58\x6B\x4D\xB4\x35\x1B\x6C\x6A\x98\xE2\x6E\xE4\x86\x6F\xD9\x06\xF3\x84\x36\x53\xE2\x7C\x16\x1D\x51\xB6\xB8\x0D\x3B\x12\x30\x53\x4A\xDB\x7B\x5A\xA4\x0D\x2E\x01\x84\xED\x60\x04\x2C\x94\x1A\x4D\xBC\xAA\xE6\x35\x0A\x59\xA2\x51\x74\x61\x3E\x22\x44\x1B\x9D\x96\x25\x13\x1C\xAD\xEE\xEC\x7F\xF9\xD6\x11\x60\x9B\x44\xE9\xCF\xD0\x46\x80\x24\x34\xB5\xD1\x25\x34\x1B\x66\x9E\xFC\x7C\x19\xF2\x1A\x7C\xD0\x52\x0B\x4B\x68\x02\x1B\xF9\x88\x48\xF1\xCC\x88\x20\xBE\x02\x37\xC8\xA5\x78\x26\x6F\x93\xD3\x45\x5F\xFA\x8E\x96\xC4\xB4\xCA\x6A\xC4\x8E\x1C\x9D\x55\x43\x66\x35\x22\xB2\xAA\x18\xB8\x11\xCE\x6A\x24\x24\x4C\xAA\xDB\x1A\x5B\xFC\xD5\xA4\x30\x29\x30\x1F\x90\x55\xC3\x8B\xBA\xA7\xAC\x6A\x0E\x56\xB3\xBA\x24\x3B\x9B\x5B\x42\xB7\x23\x9C\x55\xCA\x27\x2D\x19\xE4\x9D\xA4\x4A\x73\x11\x16\x91\xE1\x79\x32\x42\x42\x33\x13\xD2\x9F\x55\x5A\x3C\x02\x82\xA7\x5A\x80\xFB\x4D\x05\xFE\xEC\x3F\x8B\x59\xF6\x01\x16\xAA\xA6\xA8\x51\x8E\x96\x98\x8C\x80\x67\x43\x59\x3B\x92\x62\x26\xE9\xEA\x34\x49\xB3\x6B\x3E\x87\xA8\xD1\x11\x46\xB0\x21\x6A\xD8\x19\x34\xD0\x3A\x49\x96\x01\x9D\x36\xDF\x14\xEB\x38\xF5\x26\x9C\xA4\x8D\xCE\x97\x5E\xB9\x55\xEE\xB4\xE7\xEE\xBC\x75\xF6\x39\x7A\x3B\xCF\xBB\x40\xAC\xA7\x07\xE5\xFD\x06\x65\x85\x24\x56\x4C\x6F\x53\x57\x1E\x3A\x1B\xAC\xAA\x4F\x18\x74\x6C\x22\x69\x23\x07\xB2\x6C\x49\x06\xD2\xF7\xF3\xFA\x2B\xE5\x82\x44\xD7\x30\xC4\xD7\x7A\xC9\xE7\xE4\x33\x6E\x67\xD3\x7A\xA2\x68\xC0\x46\x36\x1F\xD3\xCE\xBC\x70\xAE\xC7\x09\x36\xB0\x34\x3D\x6D\xEB\x8F\xFA\xC9\xC1\xA2\x32\x6E\x33\x0B\x7C\x43\x48\x26\xD9\xAE\xC9\x49\xDD\x35\x24\x16\x36\x4A\xAC\x68\x2B\x0B\x52\xB3\x00\xC8\x70\xBF\x18\xA0\x7E\xB6\x89\x7D\x63\xF2\xA9\x8A\x1F\x30\x17\x99\x20\x6D\xF0\x88\x01\x4C\x7B\x25\x31\xC8\x1F\xDF\x4C\x83\x1C\x98\x67\xA2\x8E\xB7\x88\x1E\xE6\x70\xD9\x08\xF3\xC9\x4D\xBA\x54\x47\xD7\x95\x22\x13\xFA\x83\x69\x71\x12\x39\x12\x28\x27\x39\x73\x96\xDE\x16\x3D\x5B\xEC\xD4\x12\x36\x3A\x89\x25\x13\x72\x1A\xDB\x7D\xDA\x24\x91\x30\x61\x81\x3C\x73\xD3\x9C\xC4\x42\x25\xA1\x6C\x3A\x98\x95\xC3\x0C\x8B\xFE\x00\x96\x55\x00\x13\xC8\x00\x14\x2C\xCC\x82\x1D\x21\xBA\x48\xA4\x57\xCC\x45\x68\x66\xB5\xC5\x7E\xD5\x1F\x09\x8A\xDC\x0D\x18\xC1\xD3\x9C\x04\xDF\x28\x30\x58\x4A\x1B\xF9\xDE\xBC\x48\x89\x07\xDB\x98\x86\xC4\x36\x58\xD6\x5D\x06\xC6\xB9\x6F\x84\xC4\x42\x54\xDF\x04\x49\x7B\x21\x8B\xEC\x33\x83\x44\xCE\x0B\xE4\x47\x9D\xC7\x45\x68\x03\x24\x68\x1A\x56\x2D\x49\xCA\x68\xA4\xAD\x13\x3B\x22\xE8\x83\x2C\x5F\xE7\x65\x99\x45\x07\xF9\xA6\x87\x20\xF4\x68\x0A\xB1\x44\xF5\x26\x9D\xB1\x62\xE5\x61\x05\x85\x64\xBF\xC1\x28\xB2\x02\x32\x07\xC6\x70\xE2\x74\xF9\x3F\xE9\xBC\x9B\xA5\xD4\x70\x89\xAC\xDF\x84\x1D\x61\x3D\x2D\x32\x31\x3B\x5E\xE4\x7E\xC7\x87\x55\x27\xB1\x82\x01\x41\xBE\x52\x1E\xD9\x4C\x89\x62\xF9\x49\x2A\x3D\xD6\xA0\xD2\x83\x41\xD5\x9E\x38\x19\x0C\xBF\x58\x5E\xFD\x8A\x34\x02\xFB\xA9\xDA\x31\x3B\xD1\x4F\xEB\xF9\x31\x0E\x20\xE2\xBC\xD9\x8A\xF3\x66\x2B\xCE\xBB\x4B\x9D\xE9\xCD\x38\x05\x14\xBB\xAF\xB8\xBF\x39\x8B\xAB\x0D\x98\x65\x5B\x22\x56\x7B\xBD\x54\x60\x22\x2A\xCD\xA1\x53\x5C\x4B\x6C\xF8\x13\xEA\x3E\x8A\xC5\x24\x40\x36\x61\x5B\xED\x5A\x5A\x50\xB5\x71\x07\x16\xDA\x09\xDB\xB0\x31\xFD\xA1\xCD\xF2\x2E\x9A\x20\x56\xF9\xAA\xBD\xE8\x0B\xB6\x61\xFD\x0C\xD5\x4C\xF1\x57\xF7\x06\xDD\xEF\x06\x8A\x3E\x33\x02\x5D\xC1\x60\x91\x4F\x45\x41\x1A\x0E\x2E\x60\xF2\x91\x48\x43\x41\xB8\x06\xDA\x91\x6B\xCC\x5E\xAF\x68\x5F\xFA\x41\x67\x9D\xA0\xE9\x62\xD6\x20\xE6\x1E\xFA\xF2\x46\x59\x43\x1A\x0A\xB3\x58\x50\x90\x6F\x1B\xA8\xC2\x89\x55\x06\x02\x85\xD3\xAD\xD3\x10\x87\xF9\x2C\x3D\xC6\x12\xE6\x72\xC3\x89\xBE\x7C\x38\x5D\x13\xB1\xBE\x98\xC0\xE8\xB0\xA4\x3B\x34\x5B\xCC\x23\x34\x04\x0F\x01\x9B\x16\xFB\x26\x30\x46\x73\x7E\x0A\xF4\x67\xD1\x84\x9C\x61\xA3\xF3\x4D\xA0\x3F\x97\x4C\xC8\x45\xA4\x8A\x5B\x31\x11\x0C\x8B\x6C\xF5\x3B\x58\xAC\x3B\xC3\x35\x3B\xE2\xEC\x18\x2E\xD6\xD9\x58\x9C\xE6\x1C\x02\xFA\x23\xC3\xBF\x08\xF4\xA7\x29\xBC\x71\x54\xF8\xFB\x38\xBC\xC1\xE2\xA1\x69\xA5\x2C\x88\x26\x81\xAC\x61\x47\xA5\x6D\x3F\xB1\x48\x4B\x62\x70\x51\x2E\x46\x23\xCF\xD7\x6E\x45\xD6\xBD\x68\x3B\x60\x2E\xE0\xF3\xC4\xB4\xCD\x17\xD9\xE9\xE8\x7A\x79\x8A\xE5\x7E\x16\xF1\xE5\x51\x5F\x95\xC1\xE8\xD1\xEB\xA3\x47\xDF\x1C\x39\x6B\x07\x25\xB5\x5D\x74\x93\x76\x5C\x2E\xEE\x0C\xD7\xB8\x4F\xF8\xBA\x27\xD8\xD6\x1C\x87\x89\x91\xD6\x46\xDB\x58\x90\x42\xD6\xFD\xA9\x8C\xE8\xE7\x48\x9F\xAA\x2E\x19\xCC\xE4\xAB\xC1\x97\x8F\xB0\xAE\x47\x32\x5E\x46\x8A\x1A\x68\x1D\x13\xA4\x2D\x99\xD8\xB7\x41\x6C\x45\x93\xF4\x93\x1D\x99\x40\x76\xC3\x1A\x22\x2E\xC3\x45\x56\xB7\x63\x48\x66\xBD\x11\x9C\x52\x8F\x78\xDF\xA6\xD6\x6F\x54\xB6\xBC\xB8\xDF\xA8\x94\x08\x29\x2B\x18\xA7\xF4\xBD\x5C\x22\x2B\xC5\xD6\x96\x8E\x56\x2F\x24\xF9\x9C\x5E\xE4\xD7\x0B\xD7\x8A\xF6\xFF\x76\xAD\x8C\x5A\x1D\x7F\x5D\x2D\xF8\x87\xD8\x69\xD6\x63\xC2\xD5\x10\x9D\x4F\x99\xF8\x6F\x75\x3D\xAE\x03\x8D\x39\x29\xE9\x45\x29\x2D\x58\x07\xFF\xFF\xD4\x00\x71\x6D\xA8\xF0\xE3\x8E\x2E\x3C\xEF\x9B\x44\x49\xED\x28\x9B\xA5\x14\xD3\x42\x3F\xE7\x92\x65\x1C\xA4\xA0\x50\x08\x4B\xC2\xDA\xBC\x3C\x49\xC6\x9D\x66\x47\xDB\xB5\xF4\x72\x96\x64\x15\x39\x42\x3B\x9A\x35\x93\x24\x57\x2B\xEB\x6C\x79\x4A\x27\xE5\x53\x59\x83\xCD\x6C\x90\xE0\x0B\x2B\xB8\xC9\xC6\xED\x48\x36\xAA\xCC\x91\x0B\x24\xA9\xB7\xCE\x9A\x76\x9C\x95\x4A\x99\xD9\x28\x6D\xB5\x29\xF6\xC0\x99\x66\xC4\x36\x49\x12\x65\xAE\xBC\x7A\xA9\x3D\x3C\xD7\xBE\xCD\xC6\x87\x6F\x3D\x47\x4B\x8B\xF5\xD2\xE7\x66\x25\x89\x96\xF7\xB9\x9E\x49\xD6\xC8\x17\x91\xFB\x48\x29\x38\xC5\x87\x67\x5C\x65\xA2\x9C\xE3\xC4\xE4\x7A\xA7\x64\xCD\xB1\x5D\x50\x3E\x05\xD4\x17\x48\x12\x4C\xCB\xF9\xED\x9B\xFE\xE8\x66\xDF\xA0\x95\xA1\x24\x71\x9A\x84\x17\x64\xCB\x1E\x43\x12\x47\x0B\x49\xE2\x68\xBE\x24\x8E\xD8\x44\x10\x41\x69\x5D\xE7\x93\xBB\x74\x16\x49\xEB\xCD\x7C\xE6\x95\xF2\xF9\x68\x63\xA7\xBC\x88\x05\x87\xA0\x21\x38\x04\x2C\x38\xA4\x2D\x60\x62\x2D\x6D\x23\x19\xD6\xD1\x68\xDB\x6B\x6B\x0B\xAD\x4F\x46\xE8\x8A\xB7\x4E\x47\x9F\xB4\x32\xC8\xD9\x81\xDE\x0F\xFA\x6F\x2F\x42\x60\x1A\xF7\x3C\x2F\x23\xE6\x7B\xEC\xDF\xB0\x5E\x2E\xA1\xB4\x12\x89\x89\xFF\x27\x7E\x90\x9D\x34\xC9\x80\x7F\x4D\x52\xAE\x21\x14\xF8\xA7\x2A\x30\x31\x4E\x55\xB9\x58\x32\x86\x29\x77\x14\xAD\x4A\x83\x50\xCB\x9A\xB6\x4E\xC7\x8F\x51\x41\x10\xF9\xB3\x12\x5E\x32\x81\x54\x64\xC7\x6C\xF4\x67\x25\xD1\xA5\x6D\x53\x94\x17\xC5\xF4\x96\x76\x3C\x2A\x0D\xDA\xE6\x64\x2D\x9D\x8B\xCF\xB5\xEF\x90\xF7\x8B\x49\x51\x69\x94\x27\x41\x39\x0E\x4D\x65\x50\x8D\xAC\xA9\x59\x52\xDD\x5D\x31\x6B\xD9\x51\x36\x0E\xD3\x3C\x0E\xA3\x3C\x0E\x79\x10\x25\xB6\x66\xA3\x76\x7C\x6B\xD6\x1F\x91\xE6\x31\x47\xA4\x6E\x9B\x72\x44\xFA\xD6\x3A\xED\xA8\x58\x0E\x2D\x39\x55\xE4\x74\xB6\x7A\x66\xD1\x3A\xE6\xCB\xA6\x51\x6E\xB5\x6C\x54\xD4\x07\x51\xDB\x59\x53\x8E\xC3\xA8\x6C\x58\x39\x0E\x63\x72\x1C\xEA\x8D\x71\x68\x2D\x4F\x69\x3C\x0E\x4D\x1E\x87\x31\x31\x0E\x4D\x1E\x87\xA6\x1D\xE3\x71\x28\x39\xEB\x81\xA1\x16\x63\x76\x59\x4C\x69\xB9\x8A\xCF\xF5\x3C\xCF\x1B\x73\x8B\x1D\xA1\xC1\x46\x06\xF6\x82\x83\x8D\x55\x29\x88\x40\x49\x56\xD0\x16\x93\x83\x2D\x26\x45\xA9\x82\x03\x8D\xCD\xF4\xD1\xF1\x89\x61\x5B\xFD\x64\x20\x9A\xAD\x6A\x41\x4E\xA7\xAE\xFF\xBB\x84\x66\x0D\xE3\xA6\xA6\xF3\x30\x5F\x3C\x5B\x90\xAF\x1A\xEF\x22\x7C\x6B\xD1\x06\x5B\x8B\x46\x9F\x5B\x8B\xC5\x2C\x2A\xE3\x8A\x1A\xF7\x34\x52\x95\x8D\x62\x87\xC2\xEA\x13\x74\x52\xEE\xA1\x2F\x4B\x69\xB6\x4E\xB7\xF2\xF6\x7B\x3B\xE8\xC2\x26\x89\xD0\x99\x7C\x9D\x21\xE2\x40\x16\x1D\x3A\xF2\x6F\x8E\x8A\x6F\xED\x48\x65\x6B\xBC\x0B\x17\xC5\xCB\x06\x0E\xBC\x51\x6E\xCE\x73\x52\x74\x25\x85\x62\xEF\x99\xD4\x9A\x87\x3C\xDD\x41\xD1\x94\x7E\xF5\x98\x3F\x57\xC4\x42\x73\x85\xD4\x73\xA7\xF1\x9C\x6F\xB2\x0A\x18\x93\xE6\x02\x4D\xF4\xC8\xB4\x94\x40\x24\x8D\x98\x72\x13\xC1\x33\x64\x54\xCC\x90\x6C\x76\x92\x7B\x90\x69\xEB\xFD\xB9\xB8\xA3\xCB\xB3\x67\x6C\xD6\x7F\x87\x76\x44\xFC\x6E\x3A\xDA\x90\x58\x4F\x6D\xF4\xB5\xA5\x99\x47\xEB\xD3\xC4\x26\x7D\x9A\x66\x50\x9F\x26\x49\x54\x90\x86\xE3\x5C\x22\x9C\x13\xBD\x4E\x07\x54\x71\xA9\x65\x4A\x6C\x1A\xFC\x63\x11\xB4\xB5\x62\x36\xC2\x36\xD8\x4D\x5F\xB2\xCB\x97\xEF\x17\x1D\x27\xD6\xAE\x8D\x93\x35\x1D\x53\x35\x1D\xF3\x67\xDF\x98\x18\xCF\xB1\xBF\x5C\xA3\xA3\xCD\xBE\x51\x6A\x43\x59\xCB\xF4\x2D\x27\x92\x60\xB5\x46\x1F\xF6\xFB\x4B\xB0\x66\x35\x59\xB3\x64\x37\x38\xD6\x54\xB3\xDA\x7F\xA7\x66\xB1\x51\xB3\x7E\x04\x62\xAF\xDB\x5C\xB3\xAC\x8B\x2D\x16\xCE\x8C\x7F\x6A\x40\xB5\xAA\x35\x4C\xE1\x1F\xB3\x3A\x59\xF4\x25\xD2\x9F\x43\x47\x1B\x72\xD0\x46\x95\x29\x6C\xCA\x54\xC0\xB2\xB9\x9F\x93\xDC\x68\x03\x87\x25\xAB\x7D\x0E\x2B\x38\x49\xD2\xF0\x4B\xE3\xA6\x71\x9B\x42\xAE\xCC\x52\x68\x93\xEA\x8D\xAF\x69\x88\xCA\xE3\x2B\xE0\xB8\x64\x02\xD9\xBE\x46\xC9\x84\xD2\xA5\xB2\xBD\xB8\xBA\x4D\x6E\x93\x6C\x01\x4B\xC4\xC4\x49\xC3\x2A\xAB\x8B\x8B\x08\xA4\x9C\x23\xC9\xF8\x2A\xD1\x2A\xA8\xE6\x48\x31\x08\xE9\x1C\x95\xE7\xC8\x88\x98\x23\x91\x85\x5B\xD0\x9F\x23\x35\x96\x38\xD5\x48\xD5\x7F\x43\x42\xB6\x31\x47\x66\x63\x76\x7C\x32\x19\xB7\x8E\xB1\x18\x88\x3F\x59\x26\xC5\x14\xA9\xA9\x29\xD2\x52\x32\xA7\xA4\x0F\x96\xA5\x0F\xB5\xBF\x34\x45\x5A\x8A\xBB\x92\x10\x24\x26\xC9\x4A\x50\x2E\x7C\x13\x38\xE0\x8C\x8C\x7C\x87\xAF\x85\xF6\xF3\x71\x2E\xEF\xD3\x6A\xB4\xCC\xAE\xE7\x03\x3D\xDE\x7A\xF9\x28\x23\xDC\xFA\xD6\xFF\xD2\xD0\x14\x7B\x44\xF4\x67\x57\x31\x8D\x3A\xE9\xA2\x14\x15\xA2\x13\x92\x24\x1F\x94\x18\x2C\xFC\xC2\x5A\x1E\xD2\xC4\xB6\x8B\xA9\x19\x56\xA9\xEF\x61\xF5\x17\x24\xDA\x4A\xBD\xA4\x68\x47\x26\x6B\xE9\x73\x1A\xD2\xB7\x46\xCE\x60\xE1\x7A\x39\x54\x23\x62\xA8\x12\xD1\x4B\x9B\x0F\xA2\xDB\x98\x05\x69\x34\xC8\xA9\xC6\x50\x6D\x12\x98\x25\x7A\x01\x69\x4A\x34\x17\xA8\x33\x29\x43\x8A\xCC\x53\xDD\x8E\x63\xF1\x59\x64\xF5\xDB\x68\x6B\x4B\x53\xE0\x7C\xF8\xE1\xB0\xEA\xA9\x18\x9F\x2D\x11\x4B\x41\xD4\xA2\xAC\xB0\xF9\x54\x58\xAB\x15\x6C\x50\xE7\x91\x8A\xB1\x73\xB2\x7F\xB1\xD8\x5A\xA8\xA8\xA3\x49\xD4\x72\x93\x2C\x5B\xB3\x0C\xB0\x35\xEB\x13\x1A\xEA\xEA\xC0\x74\x17\xF8\x27\xA6\x0E\x5D\xC5\x67\x35\x12\x7C\x18\x71\x81\x7F\x6C\x7A\x56\xF0\xD8\x74\x8A\xA0\xA4\x6E\x9D\x3D\x53\x5B\x64\xA3\xF3\x2C\x9F\x9F\xB6\x13\x7D\xD5\xAE\x0E\x4E\xB7\x6C\x26\xF6\xBC\x2F\x89\x78\x31\x9B\x28\x26\x59\x26\xE6\x17\x6B\x74\x2C\xAB\x52\xF2\x8F\x65\x59\x6C\x41\xA5\x64\xA8\x94\x6C\x1B\x9D\x57\x39\x25\x75\x0B\x4C\x26\x22\x88\x4B\x8E\x57\xF6\x21\x4E\xB0\x8D\x6E\xEE\xD0\xF8\xD3\xE5\x0D\x6E\x1B\xD3\x9F\xD9\x2C\x27\x72\xA7\xBD\x28\x3F\x6C\x74\x76\x81\x3A\xCA\xD5\xAC\x0F\x26\x30\x1A\xE0\x7C\x1B\x8A\xAB\x40\x57\xC8\x69\x4B\x31\x3F\xE5\xDB\xCD\x8E\xE5\x4C\xBE\x78\x1B\xA5\x36\xCD\xC5\x8A\xB9\xB8\xF3\x10\x96\x73\x0D\x0B\x0E\x76\xC2\x79\x08\x4B\xFD\xCE\xF8\x52\x91\xB6\xE5\xAC\x58\xD3\x17\x4B\x1B\x9F\x13\xE1\xB3\x51\x26\x3C\x4C\xEE\x1A\x64\xEF\x3D\x3A\x53\x1B\xAF\x3A\x71\xDC\x36\x79\x6E\x16\x71\x39\xC3\x7C\xE9\xD2\x8E\xD9\x91\x62\xBF\x83\x45\xE7\x59\x92\xFA\x8B\x33\x73\xD2\x27\xE0\x88\xE3\x74\x33\x29\xA1\x24\x35\x68\x44\x58\x45\x78\x87\xE5\x5F\xAD\x16\x73\x4B\x9C\x4A\x97\x94\x97\x5B\x63\x5C\xCA\xFE\x5C\xB4\x29\xBD\xE8\x04\x7F\x20\x46\x49\x7D\x65\xFA\xCB\x9B\x65\x19\x88\x6A\x1A\xEF\x17\x82\x54\xBC\x44\xA9\x1A\x4C\x5F\xDD\xB7\x6D\xFA\xD5\x90\x23\x83\x91\xE3\xA4\x4C\xB6\xA0\x33\x49\x33\x39\x4F\x3F\x5C\x29\x34\xC1\xCC\xD4\xC6\xB3\xB4\x1F\x69\xF6\x69\xD7\xC6\x15\xA9\xA6\x29\xE2\x48\x23\xE2\x48\x30\x62\x45\x4D\x2A\x1B\x3F\x51\x9A\x66\xFC\x81\x60\xE7\xF8\xCE\x80\xCD\xCA\x64\x79\x2F\x80\xB3\xA0\xCD\x26\x8D\xC4\xC4\x82\xA5\x39\x71\x9C\xBC\x11\x2D\x65\x67\xF9\xB2\x6C\xB4\xE1\xCB\x70\xC6\x14\xA5\xD7\xD8\x71\xEF\x23\xAF\x39\x8D\x16\x8D\x0E\x18\x43\x24\x7B\xC3\x34\xE6\x38\x22\x79\xA5\xF5\xDB\xC8\x34\x34\x73\x5A\x8B\xA6\x01\x99\x6A\x8E\x3A\x69\x65\xCD\x31\x16\x90\x68\xE2\x5B\xDF\xBE\x77\x79\x2E\x22\x35\x84\xCB\xF3\x0F\xBA\x38\xE9\xC4\x8A\x34\x65\x11\x2F\x61\x82\xAF\x2E\x52\xE7\x9B\xE2\x82\x38\xB0\x02\x87\x03\x52\xC3\xBC\x2D\x2A\xC1\xD1\xEA\x39\x8D\x17\x99\xC6\xE1\x41\x8C\x6B\x2A\xCA\x12\x96\xD4\x1F\xA8\x8C\xE9\x86\x32\x34\xA9\x25\x97\xAF\x1E\x48\xAF\x14\x9E\xEF\x01\x71\x4F\x23\x3A\x9F\xB4\x74\xAB\xBD\x07\x0D\x96\x51\x3B\x55\xBB\x36\x9E\x5B\xF7\xBF\xD1\xAF\xB4\x46\xF3\x6B\xA3\xF4\x2B\xB3\xA9\x5F\xC5\x8E\xEE\x57\xAC\xD3\x4A\xB6\xA5\xC8\xA8\x2C\x87\x2C\x6E\xF2\xB8\x3B\x65\xEB\x53\x03\xB0\x50\xAD\xDA\xE4\x49\x53\xA3\x4A\xC2\xB6\x31\x88\x73\xC1\x11\x1B\xE3\xF5\xEF\xE8\x41\x1B\xE1\xC2\x06\xC6\x6D\xEC\xAF\x18\xB7\xF3\x99\xE5\x77\xD4\x68\x88\x34\x0F\xB3\x48\x70\x98\x45\x68\x46\xA2\x9E\xC0\xEB\x9B\xE9\x0B\x52\x73\x37\x20\x71\xAF\x88\x54\xFF\x2E\x82\x92\xCA\x6F\xBE\x2A\xD6\x90\x0E\xD0\xAC\x3F\x22\x9A\xC3\x36\x38\xFF\xF0\x67\x9E\x7F\x03\x93\x21\x36\x4D\x86\x1A\x8B\x41\xE9\x3C\x19\x1A\x6F\x9B\x3D\xE1\x43\x52\x63\x81\xC9\xB0\xA1\xD4\x35\x96\xD5\x78\x32\x34\x24\x2F\x7B\x7C\x63\x45\xD7\x04\x21\x1B\xAC\x57\x6D\x02\xE9\xBE\x09\x54\x6A\xE4\x2F\x54\xAA\x2E\x36\x59\x68\xEB\x7C\xDC\xE4\x18\xC5\xF4\xA3\xBC\x1A\x80\xF5\x75\x0D\x8D\x61\x07\xB3\x18\x92\x7A\x20\x1E\xCC\x70\xB1\xE5\x78\xD3\x34\x4D\x00\x5D\xD7\x0D\xCD\x94\x0F\x9A\x86\x94\x55\xD2\xE6\xA7\x22\xD2\x88\x15\xB7\x96\xE6\x68\x8B\x27\xD0\x7D\xA6\x0E\x20\xF2\x79\x8C\x25\x85\xC6\x38\x0D\x9D\xCE\xC0\x73\xF2\x86\x14\xFA\xAA\x1B\x0C\x25\x5A\x66\x50\x38\xD2\x06\xA0\xAD\xE0\x1B\xC2\x4D\xD7\x81\x49\x56\xDE\x81\x85\x92\xF0\x5A\xB8\x5E\x0A\x22\x90\xB4\x7D\x18\x3F\x9A\xA8\x49\x07\xA0\x03\x0B\x8F\xF5\x6B\x8E\xF2\x8D\x16\x67\x9C\xEB\x80\x14\x0E\x83\x13\x5F\x91\x02\xE7\x02\x1B\x36\x38\xB8\x9E\x68\x52\x2B\x01\x36\x32\x4F\x27\xF8\xFA\x9B\x24\xB1\xC0\x48\x3C\x23\xC7\xF2\xED\x87\x6F\xF5\xEF\x69\x31\xB5\x12\x59\x28\x15\x3C\x1B\xD9\x98\x6D\x32\x21\xDD\x90\x10\x53\x34\x27\x99\x59\x66\x1D\x19\x92\xF7\xF4\x00\xD1\x00\x71\x8A\x5B\x10\xD9\x74\x4E\x9F\xD2\xE8\x3A\x3A\x19\xC7\x22\x29\x4B\xB1\xFE\x90\x44\x7F\x7C\x31\x9D\xC6\xF0\x47\x94\xAE\x49\xB2\x40\x87\x12\xB5\x89\x92\x64\x3F\xED\xCC\x12\x0B\x53\x86\xF4\x9A\xB5\x98\x45\x15\x95\x11\x3A\xC8\x27\x1D\xB6\x65\x9B\xB9\xE4\x42\x92\xB5\xF2\x65\xFC\x69\xC2\x30\x64\x00\x07\x69\xAB\x27\xD6\x24\xE9\x23\xAA\x22\x42\x52\x72\x36\x0D\xCD\xC5\x29\x48\x46\x2C\x8B\xD8\x5F\x59\x93\x6D\x05\xA5\x78\x19\x6B\x69\x78\xA7\x82\xB4\xD8\xA9\x05\x29\x48\xEA\x16\x2B\x34\x30\xA4\xEE\x46\x19\x99\xB9\x22\xA5\xB4\x7A\xD0\x45\x3E\xBF\x82\xF8\x0E\x94\x99\xDE\x4A\x82\xA2\x7E\x9C\xB2\x45\xE4\xED\x81\x26\x33\x14\xBA\x62\x80\x99\x24\x26\xCB\x82\x22\x66\xE3\xF7\x37\x45\xAD\x24\xE9\xF8\xC1\x20\xAD\x72\x2C\xE4\x1A\xA5\xFE\x04\x8E\xB6\xA0\x48\xA9\x34\x7D\xC6\x9A\x3F\x93\x81\xCF\x2C\xAF\x85\xD9\xA8\x9D\x64\x9B\x28\x26\x6F\x72\xE3\x7C\xC7\x85\xA6\x10\xDA\x69\x35\x25\x41\xA1\xA8\x28\xA4\x3F\x3A\x2E\x82\xA8\x4D\x10\x75\x2B\x92\xFB\x49\x1F\x20\x2D\x55\x96\xEC\x81\x76\xAC\x9F\xC4\x5F\xC2\x62\xC6\xBE\xBE\x70\x41\x16\x58\x62\x06\x8A\x31\xA3\x6F\x98\x4C\x87\x48\x1D\x2F\x7A\x93\x8E\x17\x3A\x1D\xB2\xEC\xE8\xF9\x52\xF8\x28\x67\xCE\x4F\x19\xA4\x70\x9D\x16\x77\xC5\xEC\xA4\x8A\x8A\x2B\x4D\x9B\xEA\xEA\x4F\x8C\x4A\xB1\x90\x2C\x08\x34\x3A\x0F\x9F\x83\x24\xA5\xD8\x35\x6B\x63\x17\xFD\x22\x6E\x29\x2E\xA9\x1C\x0B\xDC\x33\xFD\xBE\x63\x93\x2A\x8A\xC6\xD0\x7A\xBB\x7A\x50\x63\x31\xAA\x96\x36\x5F\x94\x3A\xA9\x4A\x63\x91\xB8\x38\x55\x00\x23\xA2\x62\xEB\x49\xAA\xE4\x93\xA2\x03\xE9\x2C\xB3\x10\x63\x2B\x32\x52\x90\x3A\xCA\xDD\x22\x25\x15\x0D\x8A\x2E\x91\xB4\xA3\xC5\x1C\xDA\x86\x70\x94\x01\x71\x64\x1D\xE2\xBE\x28\xB5\x6C\xF9\x54\x93\x28\xB5\x49\xF6\x28\x7D\x51\x6A\xD1\x31\x7C\x51\xEA\x86\xE9\xF1\xE0\xD8\x10\x3D\x55\x4C\x2F\x62\x47\x26\xEB\x38\x30\xB0\xCD\xC6\xC0\xA6\x9D\x51\x78\x3C\xC7\xE5\x4C\x63\xDA\x71\x1E\xD1\x56\x68\x44\x6B\xFF\xCD\x11\xAD\xAB\x11\x1D\xFF\x2B\x46\x34\xB5\xA0\xB9\x38\x38\x37\xBE\x7D\x03\x1E\xBB\x09\x53\x8D\x26\x4C\x1D\xD5\x84\xC8\x4D\x98\xFA\x4B\x4D\xD8\x12\x68\xC2\x14\xB5\x1D\x37\x61\x52\x36\x99\x68\x3D\x12\x89\x6F\x6E\xC2\x96\x70\x13\x42\x53\x13\xA6\xFE\xDA\x26\xB4\x02\x2B\xC3\x86\x62\x8B\x06\xA8\x89\x52\x20\x6D\xD9\xD3\xB6\x35\x86\xBE\xDA\xB5\xF4\x4C\xF5\x65\xDD\xA6\xF9\x4C\x09\x11\x6A\x63\xD1\xD1\x97\x28\x29\x09\xED\xE8\x55\x4D\x67\x46\xFA\x38\x52\xD8\x62\x1B\x75\xE7\x8E\xBD\x3B\x61\xFD\xFC\x86\x5E\x2E\x15\xC1\xB1\x6E\xE8\xFC\x7D\x43\xD2\xD9\x08\x0B\x1C\xA6\x48\x6D\x8C\x61\x1F\x9D\xAE\x8D\x1C\x90\xE4\x40\x47\x5B\x6A\xA5\xB0\x01\x65\x81\x8E\x53\xE5\x59\x90\x24\xC6\xD3\x0B\x8F\x4E\x11\xD7\xCF\xE7\x33\x5F\xC3\xB2\x0D\xEB\xB9\x28\xDD\xF0\x15\x5D\x4D\x63\xC1\x3F\x9D\x75\xE9\x92\xF4\x8B\x1E\xBE\x63\x24\x30\x9A\x03\x0E\xA4\xDF\x47\x2B\x69\xA4\x89\xE6\xD1\xB9\x26\x5A\x0C\x40\x4D\x17\x04\xB8\xBA\xB3\xEC\x0C\xF3\x69\x3E\x21\x50\x21\xB4\xA0\x0F\xE4\x99\x4C\x20\xE8\x96\x81\x7F\xA9\x8B\x22\x4D\x2C\x48\x81\xF3\xEF\x0F\x04\x0F\x3C\xDA\xB5\xB4\xA4\x44\x67\x6A\xE9\xA3\x85\x4E\x45\xA8\x77\x17\x5B\xC6\x1A\x91\x58\x4C\x64\x27\x66\x18\x31\x7A\x8C\x98\x48\x65\x78\xA6\x96\x4E\x46\xD4\x1B\x25\x2B\x4D\x7A\x48\xE5\x0A\xC4\xD1\x8F\x8C\x96\x36\xDB\x75\x23\x1B\x48\xC3\x0B\x48\xBD\x0E\xD3\xC2\x69\x55\x42\x19\x11\x69\x2E\x68\xD7\xA0\x28\x2F\x34\x8F\x5A\x0E\x79\xEE\xCE\xBB\x4E\x99\x2D\xFD\x98\x49\x5B\xAC\xDA\x81\x12\xD6\x8E\x15\xA7\x9F\x3F\xA9\xA0\x39\x4D\x79\x93\x95\x2D\x95\x80\x70\x21\xFF\xBA\x12\x62\xA0\x84\x7C\x59\xF8\xA8\xCE\x91\x33\xA9\x2B\x8A\x68\x7E\xBF\xB5\xD9\xB4\x4F\x3A\x1B\x75\x0C\x92\x15\x61\x12\x48\xF3\xEF\x0A\xFE\xFE\xC1\xA6\x6A\xE0\xBC\x2E\x4F\xC1\x31\x3D\x11\x6B\x82\x25\x81\xC8\xFC\x96\x5A\xE1\x4C\x22\x84\xE8\xAC\x8C\x8A\x18\x65\xA2\xC6\x7C\xBB\x22\xFA\x16\xFC\xA2\xAC\x91\x24\x22\x08\x1E\x51\x44\xEB\x6D\x93\x4F\xF3\x09\x1D\x48\x1E\xAD\xEF\x77\xA1\x64\x9C\xD8\x46\x91\x8D\x31\xFC\xBE\xD9\x8E\x0C\xFF\x70\xAC\x88\x23\x32\xEA\x7E\x5B\x13\xF3\xF6\x9D\x3C\xB4\x9A\x6E\x48\x39\x9A\x03\x3C\xA1\x9F\x64\x83\xE8\xF2\x40\x67\xDD\x74\xE3\x5F\xAE\x2B\x1A\xED\x79\xE5\xE9\xD5\xE8\xA4\xB5\x3A\xEE\x53\x3B\x0B\x9D\x55\xBA\xA8\x9F\xEF\x8B\xFA\x77\xF6\x42\xC3\x89\x6E\x39\x9A\xB4\x9F\x41\x44\x04\x88\x44\x22\x86\x89\xF2\xD1\xC5\x7C\xCB\xEB\xCB\xFC\x54\xF4\x6D\x68\xE4\xF8\x42\xBA\xC6\x40\xE7\x2E\xFE\x52\x9A\x60\x92\xDD\x4E\xF0\x4A\x1B\x0F\x2D\xA4\x2A\x80\x5C\x48\xCD\xC6\x42\x6A\xAA\x88\x9A\x16\xD2\xB8\x25\xA8\x30\x96\x57\x6B\xA7\x63\x6D\x93\xD7\x53\x33\x98\xAE\x9D\xB2\x93\x82\x34\x8E\x5A\x52\xA9\x4E\x90\x18\xB2\x31\xFD\xB9\xCD\x2C\xE9\x2B\x7E\xE1\xDB\x9C\xD2\xEE\x80\x4F\x85\xD1\x15\x44\xFF\xEC\x9D\xF4\x5E\xD0\xAA\xFB\x30\xB5\x1E\xDA\x86\x2A\xC7\xA8\x71\xE4\x62\x7E\x2C\xD2\xE4\xC2\x28\xAB\x06\x79\x08\x45\xD3\xD8\x0B\xD9\xA6\x1D\xF3\x6F\x9D\x1A\xD3\x50\xC5\xE7\x53\x02\x66\x80\x0C\x10\x44\xD5\x4C\xBE\x8A\x4C\xD4\xE7\x51\x99\x32\x8F\x9D\x05\xF3\x18\x59\x50\x09\x39\x56\x30\x21\x83\x0F\x54\xE8\x16\xA2\xE9\x9F\x2E\x8B\x3E\xFF\x60\xB3\x2D\x1A\xE4\x7B\x89\x6A\x5D\x36\x68\x95\x8E\x32\x59\x7A\x8C\x94\x8E\x22\x6E\xA2\x41\xE2\x46\xDE\x67\xC6\x20\x7D\x1A\xB5\xA9\xBC\x51\x9F\x3E\x95\x08\x62\xD1\x44\x25\x7D\x8A\x41\xE2\x06\x7D\xE2\x06\x1B\xF4\x29\xFA\xF4\x29\x16\x73\x24\xB5\x57\xCC\x59\x92\x98\x49\x88\x7A\xB3\x1A\xC4\x0D\x0A\xE2\x06\x8F\xA6\x4F\x83\xC4\x0D\x32\x7D\x6A\x30\x83\x41\x12\x37\x52\x18\xD6\x9F\xD5\x14\x71\x23\x46\x90\xA8\x8E\xA8\x35\x20\x75\x90\x8A\x2A\x1D\x27\xAA\x74\xDB\xD6\x51\xEE\x5B\x13\x23\x91\x76\xF1\x0B\xF8\xEA\x95\xDF\x50\x48\x34\x00\xD7\x5D\xA0\xFD\x92\x4D\x68\x5B\xAE\xF5\xD6\x8B\x06\x1A\xC3\x8D\x4B\x1C\xBE\x4E\x1C\x1B\x25\x89\xE5\x13\x05\xE9\x5B\x36\x37\x05\xB5\xF5\xF4\x7B\x25\x06\x1B\xD2\xFF\x74\x5C\x63\x70\xFB\xC8\xEB\x6E\xD9\x08\x2F\x0C\x76\xC4\xF6\x6F\x31\x4B\xAD\x2B\x11\x1B\x69\x66\xE7\xBD\x1D\x5D\x3C\xA2\x9B\xA5\xCD\x29\x48\xDD\x17\xB6\x9E\xBE\x8D\x67\x4D\x5B\x53\x46\x2E\xF9\x22\xA0\xAD\x55\x1B\xAC\x2E\x47\xB3\x75\x55\xCC\xC6\x2D\x41\x79\x91\x17\x9A\x8F\x19\x1C\x5C\xDE\x30\x0F\xD7\x54\x0A\x3F\xFB\x86\xC8\xE4\x31\xB3\xEF\x37\xA0\x68\x62\x3F\x83\xD1\x26\x19\x2E\x91\x77\xB1\x81\x8C\x4A\x49\x4A\xBF\x70\xB7\x92\x67\x53\x0A\x1B\xF3\xF9\x2D\x3A\xC3\x35\x5B\x2B\x92\xD4\xB7\xF4\x99\x35\xFC\xA2\x45\x54\xD1\xB2\x11\x27\x26\xFB\x3F\xDA\x11\x87\x6E\xB6\x46\x94\x36\x31\x32\xEC\x88\x7C\x19\x2A\xD2\x48\xA7\x21\x81\x6D\xD9\x51\x12\xAC\x20\xE5\x4F\x91\x40\xD5\x61\x23\x7E\x4B\x54\xAE\x7F\x79\xE8\xED\x32\x00\x8D\x0C\xB0\x9C\x27\x67\x00\x58\xF5\x4A\x84\x79\x52\x92\xB3\xA2\x78\x52\xC4\x74\x1A\xED\x87\x17\x11\x0D\x29\x31\x3A\x4E\x1D\x22\x22\x5F\x75\xD1\xE8\x96\x96\x4E\xE3\x56\x67\x06\x67\xF0\x30\x8F\xAE\xE7\x91\x76\x0D\x19\x48\xF9\xD7\x96\xD0\x21\xC0\x9E\x86\x19\x23\x5D\x2C\x72\x74\x5D\xDD\x79\x08\xF9\x38\xD1\x50\x64\x01\x69\x99\x69\x8E\x9E\x54\x20\x69\xAC\x7C\xBE\xE9\xF2\x0D\x32\x9D\xA0\x91\x14\x9A\x73\x73\xD1\xD1\x36\x38\xC3\x35\xD1\x85\x67\x6A\x31\x29\x64\xA3\xF4\xB0\x34\xF4\xF1\x19\x00\x54\x01\xBA\xF8\x83\xD6\x1D\x49\x4C\x8D\xAE\x60\x4D\xF3\xF5\x9A\xA6\x73\x9A\xF3\x10\x09\x91\x88\xDC\x3C\xB7\x89\x15\x16\x28\xF5\x27\x24\x4C\xA5\xA9\xE2\xAB\x5A\xA3\x7A\x48\x4A\x1D\x7E\x9C\x93\x08\x5F\x45\xD3\xEC\x48\x3F\x71\xCF\x8B\xCE\xA6\x01\x31\x59\x99\x6C\x0C\xCE\xB0\xB5\x7E\x96\x44\x23\x99\x82\xF4\x7D\x7C\x21\x1C\xA4\x9E\x10\x92\x51\x64\x56\x2A\x15\xDA\x6C\x5C\x11\xA4\x63\x7C\x1C\x72\xD0\x8E\x4E\x68\x5C\x24\xD2\xD8\xC0\xAA\xE9\x8B\xE9\x98\x62\xE7\x2B\x52\x21\x39\xF3\xA8\x73\xF3\xFA\x62\xCE\xA2\x6B\x8C\x09\xA5\x26\x23\xE1\xDC\x5C\xCC\x25\x45\x45\x62\xD1\x89\xAE\x27\x71\x22\x8B\xAD\xA1\x81\x1D\xED\xCF\x1A\xE2\x8B\x84\xAC\xB4\x66\x4D\x0F\x72\xD7\x48\x3A\x30\x4C\xDB\x58\x21\x88\x81\xAC\x61\xA7\xF8\x12\xB6\x9D\x60\x61\xF2\x36\xEE\x3B\x8D\xD0\x22\x98\xBA\x2D\xC1\xDF\x31\x56\xE3\xEF\x7F\x27\xC5\x77\xB2\xF1\x9D\xE6\xBB\x28\xFE\xF7\xB8\xA2\xDA\x04\xF0\x37\x99\x01\x18\xDF\xF8\x6E\x13\xDF\x6D\x0D\xBD\x14\x36\xE7\xD6\x16\x34\xAC\x94\xDB\x00\x3B\xCE\xDA\xF2\x65\xDD\x5A\x3C\x7E\xE8\xA4\x27\x28\x9F\xC0\xA7\xE0\x86\xA5\xD4\x22\x32\x82\xEE\x6A\xC6\x78\x33\x23\x45\xFF\x45\x93\xF1\x9D\x46\x3E\x20\x27\x2E\x22\xF0\x19\xF9\x0A\xEA\x11\x26\xD9\x02\xB6\xD1\xD9\x9F\x51\xDA\xD4\xE8\x9A\x95\x13\x2B\xA6\xBF\x22\x86\xA4\xD8\x38\x8D\xFA\xEB\x57\x15\x33\xE6\x6F\x9D\xEF\x6F\x9D\xAF\xD1\xF9\x48\xC7\x23\x31\x8D\x39\x25\x6A\x10\x6D\x42\xCE\xBF\x19\xC8\x9A\x09\xB4\x7A\x2E\xAA\x08\x03\x5E\xAD\xE4\xA2\x2F\x9A\x36\x4D\x9F\x51\x12\xBB\x43\x16\xB9\x09\x76\x41\x45\x41\x3B\x46\x31\xFD\xB5\xCD\x92\x07\xDF\x98\x8E\x17\xB2\xE4\x61\x52\x27\x39\x29\xAE\xCA\x9B\x8B\x39\x2D\xFD\xD8\x66\x56\x85\x27\x3D\x25\xB5\xA0\x68\x8A\x28\x54\x54\x1A\xB9\x4A\x9E\x23\x85\x2A\x48\x4D\x03\x92\xB0\x05\x33\x8A\xF8\x16\x7D\xBB\x36\x8E\xF2\x24\xC9\x6C\xB1\xCE\x89\x8C\x46\x65\xCE\x59\x72\x23\xCA\x72\xAC\x68\x47\xC5\x6B\x78\x59\x48\x5B\xCD\x97\x17\x03\x0A\x0E\xAD\xBB\x0D\xB4\x86\xB5\x39\x4D\xA5\x92\x63\x88\xEF\x7A\xF2\x40\x32\x96\xB1\xF5\x12\xDB\x68\x1E\x48\x91\xA2\xD4\x93\xA6\xF7\xF3\x9A\x63\x92\x4C\x55\x70\x20\x91\x18\xA3\x69\x6B\xC5\x9C\xBC\x56\xA9\xDB\x11\x35\x90\xE8\xB2\xA7\xC9\xED\x66\x4E\x90\x76\xAB\x89\x1A\x32\x29\x76\xD9\x6A\x51\x3B\x22\xB6\x56\x22\x95\x6C\x9C\xB8\x28\x62\x20\x25\x58\x42\x4E\x0D\xA4\xB8\xA8\x78\xAB\x31\x90\xCC\x14\xD8\x09\xDA\xBE\x3B\x60\x9B\xFD\x59\x4D\x7C\xD1\x1D\x20\x5D\xD9\x74\xE3\x6B\xDA\x72\xFF\xA6\xF1\xE5\x2B\x6D\x45\x36\x69\x47\xB2\x9A\x9D\x64\xF1\x55\x3B\x2E\xBA\xAF\xA5\x06\x52\x23\xB4\x08\x46\x51\xF8\xDF\xD2\x8A\x85\xFF\x9D\x2C\x92\xC2\x1B\xFF\x3B\x5D\x24\xC1\x20\xFF\xBB\xE9\x44\xCA\x36\x45\x5F\xD6\x1B\x03\xC9\x14\x03\x49\x6F\x0C\x24\x5D\x0C\x24\x91\x5B\x5B\x8C\x77\x4D\x0D\xA4\x18\x0F\x24\x59\xB7\x89\x00\x15\x14\x6C\x73\xC9\xCB\xB4\x58\x2B\x1D\x84\xC7\x52\x83\xEA\xBA\x4F\xF5\x07\xD5\x01\x78\x26\x5D\xE0\xAB\xA0\x6A\x9E\x6D\xC9\x58\x1D\x5D\x06\xF0\x3B\x86\xA6\x66\x58\x5D\xCD\xB0\x11\x69\xC2\xAB\x3F\x47\x57\x42\x03\xBD\x23\xE6\xF3\x55\xE9\x42\x2B\xCD\xB0\x52\x3A\x40\x84\xE3\x8E\x11\x99\x90\x8B\x34\x3A\x46\x24\x38\xC3\x46\xC5\x0C\x4B\x6A\x25\x1C\xBD\x48\x9D\x42\xA7\xFE\x48\xC9\x12\xD3\x46\x67\x81\x27\xD5\x3F\x12\x81\xFE\x41\x77\x24\x45\xCF\x30\x95\x2F\xBE\xD7\x20\xEB\x47\x52\xDB\x81\xB3\xA0\x9C\x65\x9B\xA2\x77\x58\x59\x53\xF6\x0E\x5D\xF4\x0E\xD3\x4E\xA8\xDE\xD1\x08\x2D\x82\x51\x14\xFE\x77\xAC\x48\x82\x12\xFE\x77\x52\x7C\x27\x1B\xDF\xE9\x22\xDD\xA0\xF5\xBF\xC7\x15\xD5\x05\x0B\xFE\x1E\x2F\xBE\xC7\x37\xBE\xDB\xC4\xB7\xDF\x3B\x22\xA2\x77\x44\xC8\xDE\xAB\xD8\x29\x87\x7A\x87\x29\x47\x5E\xA0\x77\x04\x29\x44\x7F\xA0\xEA\xCE\xCD\xC5\xE6\x0E\x62\x10\xBB\xDA\x68\x74\x90\xBF\x4D\x18\x7F\x9B\x30\xFE\x36\x61\xFC\x6D\xC2\xF8\xAB\x27\x8C\x1A\xEA\xC3\x6C\xCA\x95\xFB\x87\xD2\x24\xBE\x8C\x9D\x7E\x69\x16\x37\x07\x7C\x94\x0D\x74\x12\xC1\xEA\x30\x05\xA5\xAC\xF3\x51\x38\xDF\x18\xD2\xD8\x60\xAA\x4E\xF2\x3D\x62\x67\x1C\xC9\x6A\x96\x33\x2C\xE8\xAC\x7E\xD1\x53\x16\x4C\x60\xAD\x15\x47\x10\x23\xC3\x36\x4E\x21\x7B\x17\x22\x22\x9D\x4D\x32\xDA\x1A\xD9\xA6\x95\x87\xD5\xCE\x08\xD6\x72\x11\x67\x27\x5F\xDE\x22\x2B\xBC\xEB\x73\x86\xF3\x2B\xBA\xA3\x6A\x38\x3B\x71\x71\x0A\x1D\x8D\xEE\x41\x9B\x75\x67\xE7\x98\xF5\x76\xA4\x9A\xD3\x9D\x11\x81\x27\xE5\xB3\x1D\xA0\x51\x1C\x66\x73\x1C\xCE\x9F\xC8\x02\xAC\x4E\xF7\xA0\xCD\xBA\x18\xD3\x75\x67\x64\xAF\xBE\xBE\x6A\x9B\x14\x41\xBA\x11\x81\x4E\x11\x20\x47\x20\x83\xBF\x41\xC1\x8D\x60\x70\x11\x92\x32\xE1\x79\xB0\xDE\x46\x11\x8B\xE7\xE9\x4B\x52\x68\x39\x6F\x79\x3A\x17\x4F\x0E\x0C\xB0\x0D\x6B\x37\x69\x31\x63\x8D\x13\x8E\x07\x6C\x48\x01\x67\x01\x38\xA8\x94\x8D\x8E\x91\x48\xE7\xA2\xF5\xCE\x08\x56\xA5\x1D\x02\xC7\xDC\xE0\xEC\x14\x5F\xE0\x68\xC2\xA7\x88\x9D\x54\x5A\x7B\xDE\xB7\x61\xC9\x04\x56\x62\x1D\x0E\xA9\xD1\x57\x72\x83\xB3\x97\x42\xAA\x78\xFC\x88\x1D\x5D\x2A\x04\x56\x18\x52\xD2\xE4\x8C\xDD\xE0\xEC\x1F\x35\x80\x1F\x5F\x23\x06\xC3\x5A\x8D\x38\xEC\x48\x8D\xC1\xEA\xDA\x2A\x88\x06\x06\xD1\xA4\x74\x7C\x46\x28\xAA\xC1\xBD\xA2\x89\x6C\x74\xF4\x06\x6A\xBF\x42\x2D\x64\x2E\xA5\x98\xDA\x2C\xEB\x81\xFB\x75\x0D\x9C\x91\x11\xC3\x7A\xF6\x3E\x7D\x13\xDC\x0C\x8B\xCB\xEB\x0A\xE5\x42\x7D\x23\x38\xE5\x8D\xE0\x56\xAB\xEE\x46\xA8\x0C\xBA\x5D\x03\x6E\xA9\x54\xE9\x9E\x65\x17\x4A\xA5\x7C\xAF\x5B\xB2\xD7\x0E\x15\x4A\xF5\x42\xD9\xEE\x76\xBB\xFB\xF2\x76\xA1\xDC\x93\xBF\x11\x4E\xE9\x9C\x71\x23\x9C\x72\x23\xF9\x2F\x0F\x0D\xAC\xCD\x57\xBB\xEA\x95\xAE\xB5\xEB\x66\xD9\x85\xF2\x06\xB7\x54\xE8\xB1\x19\x6B\x8F\xE6\xA5\xB2\x21\x5F\x5D\x57\xAA\xDC\x00\x8B\xF2\x6B\xAB\xF9\x1B\xE0\xDA\x1B\x60\x69\xA1\x36\x04\x17\x17\x36\xE4\x6B\x5D\x4E\xFF\x50\x75\x08\x96\x0F\xD5\xFA\x86\x60\x75\xBE\x34\xD4\x3B\x04\x4B\x0B\x03\x6B\x87\x60\xA5\x3B\xD4\x35\xAF\x50\xEE\x5A\xE4\x0E\xC1\xD2\x21\x18\x18\x2A\x75\xD5\x0A\x37\xE5\xBB\xEA\xED\x93\x1B\x71\xCE\xB2\xBB\xFB\xF2\xDD\xFD\x76\xBD\x2F\x6F\x77\x57\xCA\xF5\xFC\x8D\x75\x70\x7B\x7A\xFE\x4A\x9F\xB5\xA1\xB5\x7F\xA5\xCF\xA1\x72\xFE\xC6\xC1\x7C\x77\x3D\xDF\x63\xE7\xCB\x3D\x76\x65\x9D\x5D\x28\x0F\x0E\xD5\x61\xB5\x5B\xEE\x1D\xAA\x43\xA9\xB0\x76\x60\xB0\x27\x2F\x6A\xB1\x5C\xCF\x57\xCB\x6E\xC9\xCE\x57\xAB\x95\xAA\x5D\x28\xDB\x5D\x03\x83\x3D\x5D\x6B\xDD\x5A\xBE\xAB\xDC\x53\xD8\x30\x50\xE9\x99\x65\x0F\x96\xF2\x6E\x2D\x6F\x57\xF3\x83\x95\x6A\x1D\x96\xF4\x15\xEA\x6E\xB9\x6B\x95\x68\x88\xAE\x55\xDD\xD5\xC2\x60\x9D\x6A\x71\x6D\xA1\x5E\xEB\x72\xCB\x3D\x5D\xE5\x4A\x1D\xE6\x6D\x1C\xAC\x16\x2A\x75\x70\x7B\x7B\xAB\xF9\x5E\xB7\x9E\xAF\x51\x5A\xA0\xEA\xBF\x9A\xAF\xE5\xAB\x1B\xF2\x3D\xB6\x5B\xED\x1D\x1A\xC8\x97\xEB\x30\x30\xD8\x23\xBC\x14\x02\x91\xD5\xFA\x0A\xEB\xEA\x50\xCB\xD7\xE1\x62\xE1\x3D\x2F\xF2\x5F\xE8\xBA\xBC\x90\xAF\x43\x65\x6D\x31\xDF\x5D\x87\xA5\x8D\x18\xBB\xFB\xDC\xAA\xDB\x5D\xCF\x57\xED\xEE\x92\x5B\xAB\xC1\xFC\xDE\x8D\x83\xF5\x82\x5B\xEE\x5A\x54\xC8\x57\x2B\xBD\xA5\x8D\x83\x7D\x35\x70\xCA\x6E\xBD\x52\x0A\x63\x97\xE5\xAB\x95\x42\xBD\xD0\xDD\x84\xBC\xB6\x06\x57\xD7\x60\x65\x0D\xE6\xD5\xE0\x94\xDA\x2C\xFE\x6F\x9F\x52\x6B\xEA\x31\xD5\x8D\x5D\x85\x72\xBD\xD1\x19\xAB\xF9\xC1\x6A\xD0\x83\x5B\xEB\x5A\x57\xAA\xB8\xA3\xF9\xD8\xE0\x96\x86\xF2\x5D\xB5\x3E\xD1\x9D\x6F\xE8\xEA\xAE\x0C\x6E\x94\x81\x46\xF1\x5B\xAC\x55\xCA\x5D\x37\x54\x0B\xF5\x7C\xB5\x2B\x3F\x50\xA8\x1F\xDB\x67\xA8\xB7\x87\x7F\xA6\x6A\xBD\xB1\x52\x85\x72\xC5\xAF\x79\x7B\x5D\xA5\x2A\x7C\xE4\xEB\x85\x7A\xA1\x52\xB6\x2B\x83\xF9\xAA\x5B\xAF\x54\x61\xAD\xDB\x33\x2A\x5E\xD5\xF8\x60\xBE\x5A\x6A\x60\xCB\x15\xEE\x43\x4D\x7D\x8F\x10\x7E\xC2\x95\x2A\xF4\x15\x02\x0D\x55\xEB\xAB\x0C\x95\x7A\xEC\xB5\x79\xBB\xB7\x9A\x77\x05\xA6\xDE\xE7\x96\xED\x52\xA5\xE1\x27\x18\x5B\x29\x5F\xEE\xAD\xF7\x89\xCE\xFC\x97\x7D\x04\xD3\x01\x55\x0D\xB5\x7A\xB5\x50\xEE\xB5\xF3\xB5\x6E\x77\x30\x1F\xF8\xBD\x27\x5F\x2A\x0C\x88\xCA\xB5\x0B\x35\xBB\x5C\xA9\xDB\xAE\x5D\x2B\x94\x7B\x4B\x41\x3F\x7E\x1A\xDD\xA5\x8A\x18\x0B\x6E\xB9\x37\x3F\x4A\x12\x43\x65\x8E\x3D\xD0\x1F\x61\x49\xDF\x40\x3E\xD8\xE6\xDD\x95\x81\x41\xB7\x9A\x97\x6D\x45\x3F\xB8\xD5\x42\xBD\xAF\x6B\x60\xA8\x54\x2F\x0C\x96\x36\x86\xA7\xA9\xA0\x97\x42\x79\xA8\xF6\x36\xBF\x0F\x96\x8E\xF1\xF3\xDB\x4D\x83\x81\xF0\x3D\x85\x0D\x85\x9E\xFC\xDB\x78\xA8\x56\x86\xCA\x3D\x47\xFD\xBE\xCC\xED\x77\x6B\x6E\x15\x96\x6D\x74\xCB\x03\x6E\xA3\x8B\x94\xDD\x01\x51\x15\xEE\x60\x7D\xA8\x2A\x5A\xB9\x32\x34\x08\x57\x0F\xC2\xB2\x6A\x05\x16\xB9\xE5\xA1\x72\xA5\x02\xCB\xFB\x2B\x70\x85\x5B\xEE\xEE\xAB\xC0\xDC\xCA\x60\x65\xA0\xB2\xAE\x02\xAB\x2A\x1B\x2B\x03\x6B\x2B\xB0\xAC\xE0\x56\x60\xA9\x2B\x10\xB0\xB2\x02\xCB\x2B\xB0\xB4\x02\xF3\x2A\x70\x4A\x25\x50\xA6\x5A\x17\x35\xC6\x2C\x9B\xEB\xB5\x50\xAB\x94\x61\x5E\x65\x60\xA0\x52\xF6\x7B\xDE\x2C\x5B\xFC\x15\xB3\xE3\x86\x7C\xB5\x56\x68\xFA\xC5\x5D\x7B\xD4\x0F\x47\xC5\xEB\xFF\xBA\xD4\xAD\x17\xCA\xB0\x2C\xDF\x93\x5F\xE7\x16\x7A\xAA\x85\x32\x0C\x95\xF2\x65\xFB\xC2\xD9\xF6\x86\x52\xBE\x1C\xEC\x8A\xF5\x4A\x7F\xBE\x0C\xCB\xDC\x6A\x77\x5F\xBE\x0C\x17\x0F\x0D\x96\x2A\x1B\xDD\x32\x38\x1B\xF2\xB5\xBA\x5B\x86\x55\xEE\x80\xA8\x4F\xB7\x0C\xAB\x0B\x6B\xF3\xC2\x9D\x5F\x5A\xEB\xD6\xDC\x32\x2C\x72\xEB\x55\xB7\x0C\x65\xE1\xA9\xCF\xDD\x50\x70\xCB\xB0\xA2\xD4\xD3\xB5\x52\xE4\x40\xBE\x2F\x1A\x2A\xF7\xBA\x55\xF1\x35\x8F\x1D\xA7\x3A\x90\x2F\xF3\xF7\x50\xB7\x2B\xFC\x75\x39\xA5\xB5\x2E\xA3\xFA\x2A\x55\xB7\x36\x20\x5E\x97\x55\xCA\xBD\x34\xFB\xC1\xE2\x72\x8D\x26\xEE\x42\xA5\xEC\x96\xBA\x56\xBA\xD5\x7A\x9F\x40\x2F\xCC\x57\xAA\xBD\xE2\x65\xE9\xC6\x1E\x95\xD8\xAA\x4A\x6F\x0F\xA3\xBA\x85\xB3\xB2\xAF\x92\x2F\x17\xBA\x1B\xBF\x0E\xD5\xFB\xBA\x9C\xAA\xBB\x56\x61\x96\x57\xAA\x01\xCC\xDA\x4A\xA5\x94\x77\xCB\xB0\xDC\x5D\xEB\xD6\x5D\xF1\xB6\xCC\x2D\x17\xBA\xFB\xE8\x75\x91\xF8\x2C\x43\xD9\xBE\xD0\x9E\x6E\x9F\x7A\xAA\x5D\x16\xBD\x77\x40\xD4\x26\x23\x3B\x05\xB2\xF1\x3D\x9B\x11\xB5\xBA\x2B\xE6\xEA\x6E\xB7\x56\xBF\x60\xA8\x50\xAE\x4F\x9F\xD9\x55\xBF\xB0\xBD\x3C\xD9\x9E\x2D\xFC\xCD\x75\x07\x86\x06\x60\xDE\x50\x39\x5F\x58\x57\xA9\x0E\x80\xD3\x57\x19\x80\xEE\x41\xFB\x02\xBB\x27\xDF\xCD\x13\xC0\x19\x03\xF9\x01\x70\x7A\x4A\xEE\x00\x2C\x73\x4B\xEE\x46\x57\xBC\xAD\xE8\xED\x73\x07\x60\x55\xA1\xA7\x47\xB8\x62\x9D\x59\x2D\x5E\xE6\x11\x76\x00\x96\x0E\xC0\x80\xCA\x11\x65\x65\x20\xD0\x85\x06\x86\x4A\xA2\x2C\xBD\x43\x25\xA8\xDB\xD9\xD9\x76\x57\x57\x79\xA8\x54\x82\xB5\x81\x77\x37\xF0\xBE\xDA\x1D\x28\x94\x60\x50\x8C\x88\x0D\x85\x6A\x7D\xC8\x2D\xC1\x3C\xB7\xEC\xF6\x50\xB3\xAD\xAD\x54\x0B\xBD\x85\xB2\x5B\x82\xAB\x4B\xB0\xBC\x04\x4B\x4B\xB0\x2C\x9F\xAF\xE7\x0B\x5D\xCB\xDC\x8D\xF9\x7E\x58\x58\xCD\xE7\xFB\x61\xAE\x5B\x77\xFB\x61\x55\x3F\xAC\xAC\xB9\x25\xB1\x2A\xAC\x74\xFB\x4A\xEE\x86\xC2\x51\x0D\xCB\xD8\x65\xF9\x72\x4F\xBE\x6B\x49\xA1\xDF\xED\x1F\x2A\xC0\x15\xAE\xE8\xD8\x5D\xF3\x0A\xF5\x02\x2C\x1C\x2A\xBA\x55\xB7\x5E\x80\xD5\x6E\x7F\xB5\x00\xAB\x36\x96\x2A\xF5\x42\xD7\x72\xB7\xB7\x5A\x80\xE5\x6E\xB9\xA7\x50\x76\x45\x57\x83\x8B\xF3\x1B\x5C\xF9\xBA\xAE\x5C\x80\x65\x43\xA5\xBA\x2B\x5C\xB7\xCF\x2D\x8A\x97\xB9\x55\xB7\x6F\xA0\x00\x73\xF3\xE5\x5E\xB7\x54\x80\xB9\x7D\x6E\xA1\xBF\x36\xD4\x5F\x80\x25\x7D\x95\x62\x7F\x01\x56\x94\xBA\xE6\xF5\x15\xC4\xB7\x5B\xA8\xF7\x09\xB4\x5B\xAD\xD4\xFA\xC4\xEB\xC2\xA1\xEA\xC0\x50\x7F\x5F\x01\x96\x55\x7A\x44\x8C\x35\xB7\xEA\x0E\x74\x2D\xAC\x94\x7B\xC4\x6F\xE5\xA2\x5B\x72\xE5\xD7\x55\xF9\x9B\x0A\x3D\x22\xEC\x50\x8F\x7B\x83\xDB\x53\x80\xD5\x7D\x6E\x01\x2E\x77\x0B\x70\x55\x01\x56\x8A\xB8\x37\xBA\x7D\x5D\x4B\x0B\xB0\xBA\xB0\x4E\x64\xBC\x0F\x2A\x83\xDD\x95\x9E\x7C\x3B\xF5\x8D\xFE\xC5\xE5\x5A\xDD\x29\xD5\xED\x9B\x6F\xB6\x47\xC3\x2F\x73\xEB\xDD\x7D\x30\xD7\xAD\xD5\xDC\xAE\xCB\xDD\x3E\xA8\x55\xBB\xA7\x29\x42\x69\x5A\x7D\xE3\x60\x9E\x26\xC0\xA9\x7D\xB0\xDA\xED\x75\x4B\x95\x5E\x58\xE9\xF6\xB9\x37\xF4\x75\x2D\x1A\xA8\x94\x7B\x61\xF9\xC6\x82\xDB\x9F\x2F\xF7\x76\xAD\x1C\x72\xBB\xFB\x86\xF2\x12\xBD\xAA\xD0\x5B\xBE\xA2\x5A\xA8\x17\xCA\xBD\xC0\xBD\x0F\x2A\x83\x5D\x94\xEC\xA5\xF9\xDE\xFC\x8D\x83\x4B\xC5\xBA\xE3\x96\x56\xF1\x6F\xAB\x2A\x55\xB7\x6B\x55\x65\x60\x30\x5F\xEE\x85\x4B\xF3\x45\x97\xFC\xCB\xE5\xB3\x9C\xEF\xFD\x4B\xB4\xC5\x3A\xE8\x9C\x3E\xE3\xCC\xB3\xCE\x9E\x79\xCE\xB9\xE7\xB9\x6B\xBB\x7B\xF2\xEB\x60\xE5\x3A\x98\xB7\x2E\x40\x77\xF9\xEB\xB9\xA0\x24\x1B\xA4\xCF\xBC\xA1\x6A\xAD\xB0\x21\x0F\xF5\xEA\x50\xDE\x5F\xCD\x68\xBD\x82\xE5\xF9\x1B\xBA\xC4\x80\x58\x3A\x94\x6F\xCA\x8C\x5B\xCF\xC3\x3A\xB7\x54\xCB\xC3\x5C\xB7\x54\x28\xE7\xC5\xCB\x50\x2F\xBF\x5C\xE2\x6E\x70\xE9\x65\xD5\x50\xB9\x87\xDF\xAE\x76\xCB\xEE\x5A\xF7\x26\xB7\xDA\xB5\x6A\xFD\x90\x5B\xCD\x43\x37\xAD\xFE\x62\xBD\x1D\x2A\x57\xF3\x82\xD8\x18\x2A\xE7\x83\xC3\x4A\xAC\x75\x76\xA5\x09\xE7\xF6\xF4\x10\x66\xA5\x5B\x1A\xD8\x58\xCD\x97\xF3\xA2\xE7\x89\xEC\x1E\x8B\x0E\xE4\x95\x3A\x90\xD6\xA0\x5B\x55\xEB\x37\x14\xCA\xB5\x7A\xD7\x35\xD5\x4A\xA5\x7E\xDD\xD4\x63\x75\x95\xB7\xF5\x33\x77\x63\x3D\x7F\x29\x45\xB5\xA2\xE6\xF6\xE6\x61\x5E\x5F\xBE\x5A\xE9\xCF\xE7\x65\x5B\x37\xD6\xC9\xA1\x72\x41\x04\xF5\x73\x29\x69\x90\x5A\x7E\xFD\x50\xBE\xDC\x9D\x87\x95\xA2\x29\x68\xDA\x59\x9A\x0F\x51\xEF\x4D\xC4\x35\xCC\x1D\xEA\x2B\xF4\xC0\xBA\xC1\x6A\xA1\x5C\x5F\x37\x8B\x4A\x54\x18\x18\x2C\xE5\x05\x25\x97\xEF\x81\x75\x44\x25\x8E\x82\x1F\x1C\xAA\x77\x1F\x8D\x5E\x5C\xEE\xCB\x8B\x00\x3D\x32\xC3\xB5\x59\x76\x6F\xBE\x6E\x8B\x69\xD5\x5E\xE7\x16\x4A\xC1\x1F\xEA\xD5\xA1\x72\xB7\x5B\xCF\xAB\x1F\x06\xDD\xBA\xD8\x3B\xD8\xF5\x4A\xC5\x2E\xB9\xD5\xDE\xBC\xDD\x41\x4B\x70\xA1\xE4\x7B\x09\xEE\xD7\x02\x28\xBF\x29\xE1\xDA\x1E\x58\xD9\x03\xCB\x7B\xE0\x94\x1E\x28\xDF\x50\xA9\xF6\xD4\xC4\xDC\x5A\xCD\xD7\x86\x4A\xF5\x8E\x0B\x29\x24\x2C\x5B\x79\x71\xD7\xB2\xC5\xCB\x9D\xA5\x4B\x57\xCC\xB3\x2F\x08\xFF\x3A\xAF\x32\x58\x2F\x74\xC3\x65\x62\x6A\x12\x2F\x0B\x4B\xAE\x58\xE3\xC4\xEB\xFC\x7A\x5F\xA1\x32\x58\xE8\x86\x4B\x45\xED\xAB\x55\x74\xA0\xD0\x0D\xF3\x36\x56\x0B\xA5\x92\xC4\x2D\xAE\xBB\xEA\x75\xF5\x50\xB5\x5F\xC4\x51\xA9\xF7\x15\xBA\x81\xD6\xB0\x6E\x98\x5F\xDA\x38\xE0\x16\xBA\x61\xF1\xC0\x60\xBE\x5A\x70\x4B\x62\x6D\x23\xC4\x32\xB7\xDC\x23\xDC\x55\x1B\xAB\x05\xB7\x1B\x56\x75\xC3\xCA\x6E\x58\xD6\x0D\xF3\xBA\x9B\xE7\x0E\xFA\x5B\x18\x70\x4B\x53\xBB\xE1\x94\xB5\xB0\xA2\x36\xE0\x96\x37\xBA\xB0\xA2\x5A\xD8\xE8\x8A\x75\xA3\xB0\xAE\xD0\x75\x69\xA5\xAF\x50\xEE\xDD\xE8\x8A\xC9\x65\xAD\x5B\xBE\xC1\x15\xA3\xCE\x85\xD5\x85\x6A\xDF\x50\xDD\x85\x55\xEE\x50\xD5\xAD\xF5\xD5\xAB\x2E\x5C\x5C\xE9\xAD\xBA\xB0\x44\x2C\x00\x6E\xD9\x85\x45\x85\xAA\xDB\x2B\x5E\x56\xF7\xB9\xC2\x99\xD7\xE7\xF6\x0F\xB8\xB0\xAA\x50\xEE\x73\x4B\x2E\x2C\xAC\xBA\xE5\x7A\x9F\x0B\x4B\xF3\x83\xDD\x7D\xAE\xA0\x28\xAA\x6E\x8F\x08\x5E\x2E\x0B\x77\x65\x9F\xDB\x5B\xEB\x5A\xE9\x02\x57\xAA\x58\xA4\x5C\xE8\x14\xB5\x4C\xFB\x4C\xB8\xE6\x7A\xB8\x66\xD6\xF5\xD4\xD9\x66\x5D\x07\xD7\xCC\x6A\xBC\x5D\x7F\x63\x4F\xA1\xB7\xC0\xEF\x81\xD7\xEB\x1B\xAF\x01\xE4\xE0\x50\xB9\x5B\x46\xE0\xBF\x5D\x5F\xAA\xDC\x90\xAF\xD2\x6B\xE3\xED\xFA\xA1\xC1\x41\xF9\xDA\x78\xBB\xDE\x2D\x95\x87\x06\xE8\xB5\xF1\x76\x7D\x77\xB9\x5E\x2D\xD1\x6B\xE3\xED\xFA\xB5\x25\xB7\xDC\x4F\xAF\x8D\xB7\xEB\xDD\x5A\x77\xA1\xC0\xC1\xFD\xB7\xEB\x7B\xAB\xEE\x60\x1F\xBD\x36\xDE\xAE\xAF\x0D\xBA\xDD\x79\x7A\x6D\xBC\x5D\x2F\xBA\x25\xBD\xF9\x2F\xD7\xBB\xA5\xC1\x3E\x57\x66\x48\xBE\x5D\x7F\xED\xD4\xEB\x60\xA0\x50\x13\x1B\x07\xFB\x3A\xA8\x8B\x59\x49\xBC\x5E\x0B\xD7\xC0\xD5\xD0\x3D\xD8\xE1\x93\x1F\xF6\x05\xB6\xE8\xD2\xAB\x56\x2D\xBE\x7A\x7E\xD7\x32\xE7\x4A\xB8\xF6\x0A\xB8\x76\x15\xAC\x84\xDA\x72\x77\x39\x94\x45\xE5\x9F\x79\x3A\x75\x7A\xE7\xCA\xD5\x97\x3A\xCB\x57\x2D\x58\x71\xE9\xB2\xAE\x19\xF2\xA7\xD1\x7E\x58\x06\x25\x57\xFD\xB6\xC4\xB9\xD4\x59\xBD\xEA\xB2\xB9\x4E\xD7\x5C\x67\xD5\xFC\x79\xCE\xAA\xF9\x62\x12\xBB\x41\xB4\xEC\xF2\xCB\x96\x2E\x95\x3E\xCF\x3C\xBD\x7D\x94\x88\xA6\xCD\x98\xDC\xEC\xF9\xDA\x8B\x61\x1E\x2C\x2D\x94\xF3\x6E\xB5\x6B\xAE\x7A\x71\xC4\x5E\x48\x44\x72\x1E\x13\x65\x67\x9D\xDB\xD8\xFD\xD4\xD7\x75\x9C\xEB\xCF\x71\x97\xAD\x5E\xD0\x71\x2E\xFB\x99\x3E\xD3\x27\x94\xCF\x82\xAA\xE8\x5B\xFC\xBA\x72\xBA\x88\x68\xA0\xD2\x53\x1E\x1A\x20\x62\x8F\xDF\x2E\x98\x6D\xAF\x3C\x13\xD6\xBA\x35\xF2\x36\x03\x36\x4D\x87\x5A\xA1\xB7\x2C\xA6\xDD\x8E\xE9\x22\x8B\xEA\x6B\x3A\x6C\xEA\x0C\x13\x66\x44\x23\xF6\x15\xD6\xD5\x05\x1A\xA8\x17\xD6\xE8\xF5\x06\xCA\x01\x7B\x1D\x92\xEF\x50\xF3\x5F\xAA\xDD\xEA\xB5\x27\xDF\xDD\x71\xA1\xFA\x08\xC6\xAE\x3E\x61\xC0\xBD\x71\xB0\x9A\xEF\x56\x3F\x15\xCA\x85\xBA\xFA\x16\x35\x7C\xE1\x6C\xBB\xB4\x56\xFC\x50\x5A\x4B\x28\x26\x5E\xA1\xC0\x4E\xB7\x5B\xAD\x6E\xBC\xA6\xF3\x3A\x51\x00\x0A\xCE\x88\xE9\x61\xC4\x0C\x46\xC0\x34\xE8\x10\x44\xD7\x19\x30\xF5\x74\xC8\x0E\x0C\xF6\x74\x15\x6A\xB5\xC1\x7C\x77\xC1\x2D\xB5\xF3\xB0\x9D\xAC\xD0\xDD\x15\xB1\x58\xF5\xB8\x75\x37\xFC\x4B\xAD\xCF\xAD\xE6\x7B\x9A\x7F\x6A\xAF\x0C\x76\xD1\x52\xD6\x2E\x69\x91\x4B\xF3\x83\x79\xB7\x3E\x19\x0A\xB5\x41\x31\x22\x67\xB4\x57\x2B\x37\xD4\x26\x43\x77\xA5\x54\x13\xDE\x02\x47\x5A\x33\xA6\xD8\xFC\xDB\x51\x71\xCC\xEB\x73\xAB\xF3\xC4\xC2\x1B\x88\x46\x44\x10\xF8\x2C\x4F\x06\x49\xAE\xB7\x0F\x8C\x12\x83\xA4\x88\x26\x43\x7B\xA9\x32\x59\x74\x85\xF6\xBE\xC2\x64\xC8\xF6\xB9\xB5\x2E\x3A\xDD\x6B\x17\x5F\xDD\x95\x72\xDD\x2D\x94\x6B\xE2\xA3\x5D\x2D\xCB\x32\x1A\xB1\x30\xCF\x1F\x18\xAC\x6F\xBC\xA2\xD0\x53\xEF\x1B\xFD\x77\x45\xEF\x1D\x3B\x7D\xA6\xC8\x46\x0F\x3D\x8F\x37\xB4\xA3\x15\x3F\xF0\xCB\x51\xC1\x7C\x6A\x61\x32\x70\xAB\x14\xF8\x88\x33\xDF\xBE\xB6\x81\xA9\xE7\x7B\xF3\xD5\x76\x77\x72\xB8\xA9\xDD\xC9\x70\x03\x8D\x08\x51\x23\xD3\x2F\xBB\xE0\x82\xE9\x33\x27\xC3\xD0\x51\x98\xEA\x51\x98\x5A\xB5\xFB\x28\x9C\x5F\xB5\x33\xCE\x3E\x7B\x32\x88\x0A\x0E\x7C\x75\x07\x3F\xCA\x93\x45\xA7\x6D\x9F\x31\x19\xDA\x07\xAB\xF9\x75\x85\x1B\xB9\x0B\x34\xB0\x85\x1E\x2E\x61\x99\x08\xA5\x8E\xE9\xA3\x7A\x9C\x2E\x93\x14\xEF\x9D\x32\x41\xF5\xDE\xDD\x78\xAD\x77\x5C\x58\xCD\xAF\xE3\xF8\xC4\x77\x65\xA8\xDE\xC5\xF5\xD8\xD5\x40\xD6\xEA\x6E\xB5\xAE\xEA\x55\x20\xAA\x43\xE5\xF5\x1D\x17\x8A\xD4\x82\x58\xF1\xCD\x31\x07\x37\x8C\xDC\x7F\x2F\x6C\x1F\xEC\x18\xEC\x9C\x2C\x52\xAC\xD4\xDD\x12\x07\x0A\x7A\x2B\x94\xEB\x17\xB6\xAF\x2B\xB9\xF5\xA9\x32\x5A\xAE\x89\x0B\x46\xF3\x46\xA4\x64\xAD\xEB\x68\x9F\x8D\x4C\x41\x7B\xB9\x56\xEF\xF7\xC3\x77\xF7\xFB\xBE\x21\x5B\x18\xEC\xB8\xB0\xE4\xD6\xEA\xED\x93\xA1\xD0\xC3\x33\xC0\xCD\x37\xDB\x83\xD5\x4A\x6F\x57\xC7\x85\xA2\x4E\xDB\x0B\x3D\x1D\xD3\x27\xFB\x9E\xA8\x04\x34\x6B\xBA\x5C\xC5\x22\xA0\x7D\x41\xF0\x53\x7A\x09\xA2\xD4\xCA\x34\x39\x48\x5C\x4E\x86\x5A\xBE\xDE\x0E\x93\x98\x8A\x17\x13\x4F\x4E\xCD\xE4\x34\xD2\x72\x74\xF0\xD9\x33\x8B\xCF\xF1\x66\xD9\xEA\xF3\x06\xB7\x5A\x26\xEA\x36\xB0\xDB\x80\xA6\xE7\xA8\x5D\x08\x1C\xEB\x77\x67\xEE\xBC\x8B\xE7\x2F\x08\xFF\xCE\xD8\x85\x8B\x16\x5F\xB2\x64\xE9\xB2\xE5\x2B\x56\xBE\xF3\xD2\x55\xAB\x2F\xBB\xFC\x8A\x2B\xAF\xBA\x9A\xE3\xEB\xED\x2B\x14\xFB\x4B\x03\xE5\xCA\xE0\xFA\x6A\xAD\x3E\xB4\xE1\x86\x1B\x37\xDE\xD4\x88\xF3\x8C\x69\xFF\xB3\xF1\x75\x74\x89\x38\x7E\xD2\x82\x3E\x64\x4C\x84\x97\x53\x8D\xEF\x8B\x01\x9B\xD2\x8B\x43\x0A\x10\x6C\xD8\x0C\x79\xD8\x02\x23\xEF\xD8\x7B\x3C\xD8\x09\x1B\xA1\xDD\x9E\x6C\x23\x4C\xB3\xBB\xEC\x4E\x80\x4E\xE8\x54\xA1\x1C\x00\xB8\x5A\xBE\xDB\xD2\x75\x01\xE0\x26\x00\xD8\xEB\x79\x1E\x48\xF7\xCB\x00\x20\xE0\x3F\x34\xF6\x23\xDC\x9D\x00\xF0\x5C\x28\xEC\x0B\x00\xF0\xCB\x10\x6E\x2F\x00\xBC\x1E\x8A\xEF\x00\x00\x1C\x09\xE1\xC4\x8B\x80\x8D\x32\x1C\xB9\x08\x30\x8D\x32\x7B\x9C\x0F\x22\xF3\xAA\x00\xE7\xC9\xF0\xD3\x91\x61\xBC\xFC\x16\xEE\x0C\x04\x38\x27\x14\xF6\x3C\x04\x58\x14\xC2\x5D\x82\x00\x37\x84\x70\x37\x22\x10\x88\xE7\x76\x19\xE7\x46\x04\xD8\x14\xF2\x37\x8C\x40\xF0\xEC\x11\xF6\x23\xDC\x11\x04\x10\xF0\x6D\x59\x0E\xE1\xBE\x07\x19\xC4\xF3\x63\x89\x7F\x2F\x02\x6C\x0E\xC5\x77\x1B\x02\x81\x78\x7E\x28\xFD\xDD\x8E\x00\x77\x84\xFC\xBD\x0F\x01\xEE\x94\xFE\x9E\x96\xFE\xDE\x8F\x00\x77\x85\xFC\xDD\x8D\x40\x20\x9E\x15\xD2\xDF\x07\x90\x41\x3C\xDF\x97\xB8\x7B\x90\x41\x3C\x3F\x90\xB8\x0F\x22\xC0\x87\x42\xF1\x7D\x18\x81\x20\x98\xEE\x3F\x20\x83\x78\x7E\x24\x71\x1F\x41\x06\x37\xD0\xA7\x3E\x8A\x0C\xE2\x79\x46\xE2\x3F\x86\x0C\xE2\xD9\x2D\x71\xFF\x88\x00\x1F\x0F\xA5\xFB\x09\x04\x82\xFB\xA5\x1F\xE1\xDE\x8B\x0C\xC1\xF8\x3E\x85\x0C\xE2\xD9\x23\x71\x9F\x46\x86\xF7\xAA\x7A\x07\x80\xFB\x90\x01\x02\x7D\x78\x0B\x02\x7C\x36\x94\xEE\x03\x08\x04\xE2\xF9\xA9\xF4\xB7\x15\x01\x1E\x0C\xF9\xDB\x86\x40\x10\xF4\xF7\x39\x04\x78\x38\xE4\xEF\xF3\x08\x04\x41\x7F\xFF\x8C\x00\x8F\x84\xFC\x3D\x8A\x00\x5F\x90\xFE\x7E\x22\xFD\x7D\x11\x01\xBE\x12\xF2\xF7\x55\x04\x02\xF1\xFC\x4C\xFA\xFB\x1A\x02\x3C\x16\xF2\xF7\x38\x02\x3C\x11\xC2\xED\x40\x20\x38\x57\x86\x13\xEE\x77\x90\x41\x3C\x1A\x0F\x3F\xF8\x2E\x32\x90\xC2\x33\xE9\xF7\x7B\xC8\x70\x44\x8E\x0D\xF1\xDB\x93\x08\xF0\x64\x28\xEC\x53\xC8\x10\x0C\xFB\xAF\xC8\x10\x0C\xFB\x7D\x04\xF8\x7E\x28\xEC\x0F\x90\x21\x18\xF6\xDF\x90\x21\x18\xF6\x69\x04\xF8\x79\xA8\x6C\xBF\x40\x20\x78\x54\xFA\x13\xEE\x2F\x11\xE0\x4F\x21\x7F\x2F\x23\x10\x04\xD3\x7D\x05\x19\x82\xE9\xFE\x19\x19\x82\xE9\xBE\x8A\x00\xAF\x85\xE2\x7B\x1D\x81\x40\x3C\xF7\x49\xBF\x6F\x20\x83\x78\x9E\x92\xB8\x03\x08\x90\xD1\x9A\xC3\xDA\x1A\x10\x88\x67\x93\xF4\x97\xD3\x00\xCE\x0C\xF9\x9B\xA5\x01\x81\x78\xCE\x38\x9D\xDD\xF3\x35\x80\x0B\x42\xFE\x66\x6B\x40\x20\x9E\xEB\x64\x7C\x17\x6A\x0C\xE2\x69\x97\x61\x2F\xD2\x00\xE6\x68\x00\x17\xA9\xEF\xD3\x01\x1C\x0D\x60\x6E\x28\xBE\x79\x1A\x10\x88\x67\xB6\x8C\xEF\x62\x8D\x41\x3C\xF3\x65\x5D\xCD\xD7\x18\xC4\xB3\x50\xE2\x16\x68\x00\x2B\x42\xF1\xAD\xD4\x80\x20\x23\xD3\x15\xEE\x3B\x35\x86\x36\x89\x13\xEE\xA5\x1A\xC3\xC9\x12\x27\xDC\x55\x1A\xC3\x54\x99\x0F\xE1\xAE\xD6\x00\x04\xCC\x90\x38\xE1\x5E\xAE\x01\x5C\x21\xEA\x50\xE2\x84\x7B\x95\x06\x20\x60\xA6\xC4\x09\xF7\x1A\x0D\x40\xC0\xD9\x12\x27\xDC\x6B\x35\x00\x01\x2B\x3E\xCB\xE9\x0A\x77\x8D\xC6\x10\x8C\xCF\xD5\x00\x04\x2C\x91\xFE\x84\xDB\xAD\x31\x4C\x57\xEB\x93\xE7\x79\x79\x0D\x40\x40\xBB\xF4\x27\xDC\x75\x1A\xC3\xC5\x12\x27\xDC\x3E\x8D\x61\x9A\x0C\x2B\xDC\x82\x06\x20\xA0\x43\xE2\x84\x5B\xD4\x00\x8A\xA1\xB0\xFD\x1A\xC3\x1B\x93\x19\x27\xDC\x92\xC6\xE0\x48\x7F\xC2\xAD\x68\x0C\xC1\xF8\xD6\x6B\x00\x02\xDE\x92\x61\x85\x5B\xD5\x18\xCE\x90\xFE\x84\x3B\xA4\x01\x08\x38\x5D\xE2\x84\xFB\x6E\x0D\x40\xC0\x7F\xC8\xB0\xC2\x1D\xD1\x18\x4E\x95\xFE\x84\xFB\x5E\x0D\x40\xC0\x3C\x99\x17\xE1\xDE\xA2\x31\x04\xFD\xDD\xAE\x01\x08\x38\x5D\xFA\x13\xEE\x1D\x1A\x43\xD0\xDF\xFB\x34\x00\x01\xDF\x90\x38\xE1\xDE\xA9\x01\xBC\x5F\x03\x38\x4D\xE2\x84\x7B\x97\x06\x20\xE0\x31\x89\x13\xEE\x87\x34\x00\x01\xA7\x48\x9C\x70\x3F\xA5\x01\x08\x38\x5E\xA6\x2B\xDC\x4F\x6B\x0C\x63\x25\x4E\xB8\xF3\x75\x86\xD5\xB2\x7F\x0B\x77\x50\x07\xA8\xE9\xCD\x7D\x7C\x83\x0E\x70\x43\x08\xF7\x2E\x1D\xE0\xDD\x7A\xF3\x7A\x34\xAC\x33\x88\xA7\x2E\xF1\xB7\xE9\x0C\x54\x56\x89\xBB\x43\x07\xB8\x53\xE2\x4E\x91\xB8\xBB\x74\x06\xF1\xCC\x91\xB8\xBB\x75\x80\x0F\x48\xDC\x45\x12\xF7\x41\x1D\xE0\x33\x12\xA7\x68\xB3\xFB\x75\x80\xED\x21\xDC\xE7\x74\x86\x9F\xCA\x7A\x11\xEE\xC3\x3A\xC0\x3F\xEB\x00\x3F\x93\x38\xE1\x3E\xAA\x03\x7C\x53\x6F\xA6\xE1\xBE\xA5\x03\x08\x78\x51\xE2\x84\xFB\x6D\x1D\xE0\x07\x21\x7F\xFF\xA6\x03\x08\xD8\x29\x71\xC2\x7D\x5A\x07\xF8\xA1\x88\x53\xE2\x84\xFB\x23\x1D\xE0\x47\x32\x7F\x31\x99\xBF\x5D\x3A\xC3\xB7\xA4\x3F\xE1\xEE\xD6\x01\x04\x3C\x29\x71\xC2\xDD\xA3\x03\x08\xD8\x2D\x71\xC2\x7D\x4E\x07\x10\xF0\x7D\x89\x13\xEE\xF3\x3A\x80\x80\x03\x6A\x6E\xF6\x3C\xEF\x05\x1D\xE0\x4F\xA1\x76\xDB\xAF\x03\xC1\x43\xD2\x9F\x70\x5F\xD6\x01\x04\x3C\x22\x71\xC2\x7D\x45\x07\x10\x10\x95\xF9\x15\xEE\x9F\x75\x86\xBB\xA4\x3F\xE1\xBE\xAA\x03\xBC\x2A\xCB\xF6\x1D\x89\x7F\x4D\x67\xD8\x22\xBF\x85\xFB\x86\xC8\x5B\x28\x2F\x07\x75\x20\x10\xCF\x41\xE9\xF7\x90\x0E\x70\x38\xE4\xEF\x2D\x1D\xC0\x93\xFE\xD4\x9A\x02\x06\xC0\x18\x83\x71\x2B\x65\x1E\xD3\x06\xC0\x34\xA3\xB9\x1F\x74\x1A\x00\x2B\x8C\xE6\x76\x5B\x69\x00\x74\x19\xCD\xE5\x5D\x63\x00\xBC\xC7\x68\x4E\xF7\x4E\x03\x60\x47\x08\xB7\xD3\x00\x02\xF1\x8C\x91\x69\x7C\xD3\x00\xF8\x61\xC8\xDF\x8F\x0C\x20\x78\x59\xA6\x21\xDC\x5D\x22\x7F\x91\x66\x7F\xD3\x23\x00\x97\x47\x64\x5E\x65\x7C\x6E\x04\xE0\xB6\x08\xC0\x2E\x19\x56\xB8\x5B\xD2\x00\xDF\x4D\xF3\xEF\x6B\xDA\xD8\x7D\x32\xCD\x10\xC4\x3D\x9D\x66\x08\xE2\x76\xA5\x01\x0E\xA5\x01\x76\x5A\xE0\x3F\x6F\xA5\x01\xBC\x10\x6E\x4B\x2B\xC0\x9F\x5A\x65\x98\x8F\xB1\xBB\xBF\x15\xE0\xB5\xD6\xE6\xBE\x7B\xA0\x15\xE0\xAD\xD6\xE6\xBE\x36\xD2\x06\x20\xE0\x43\xFF\xC1\x38\xE1\xBE\xA7\x0D\x40\xC0\x87\x25\x4E\xB8\xEF\x6D\x03\x10\x70\xAF\xC4\x09\xF7\x96\x36\x80\x5B\xDB\x00\x3E\x2D\x71\xC2\xDD\xDC\x06\x20\xE0\x53\x12\x27\xDC\xDB\xDA\x00\x04\xFC\x93\xC4\x09\xF7\xF6\x36\x00\x01\x3F\x93\x38\xE1\xDE\xD1\x06\x20\xE0\x5B\xEF\xE3\xFC\x0A\xF7\x9E\x36\x80\xAF\xCB\xFA\x98\xF3\x2A\xFB\x7D\xA2\x0D\x60\x47\x08\xB7\x71\x22\x83\x71\x27\xE3\x85\xFB\xEE\x89\x0C\xBF\x6B\x61\x9C\x70\xEF\x9E\xC8\x70\xAE\xF4\x47\xEE\xC9\x00\x1F\x39\xB9\xB9\x7D\x3F\x79\x32\x10\x7C\x57\xD6\x95\x70\x3F\x7D\x32\xC0\xA7\x4F\xE6\x70\xCE\xAF\xE5\x18\x39\x19\xC0\x0B\x85\x85\x0C\x40\x34\xD3\xA8\x77\xE1\xC6\x32\x00\x63\x32\xB2\x1D\x64\x9C\xE9\x0C\xC0\xF1\x21\x7F\xE3\x33\x00\x13\x43\xFE\xEC\x0C\xC0\x69\x21\x7F\xED\x19\x80\x69\x21\x7F\x9D\x19\x80\x73\x42\xFE\xCE\xCD\x00\x5C\x14\xF2\x37\x27\x03\x30\x3F\xE4\x6F\x51\x06\x60\x59\xC8\xDF\x3B\x33\x0C\x41\x7F\xAB\x32\x0C\x41\xDC\x65\x19\x86\x20\xEE\x8A\x0C\x43\x10\x77\x55\x86\x21\x98\xC6\x35\x19\x86\x20\xEE\xBA\x0C\x43\x10\xD7\x95\x61\x08\xE2\xD6\x64\x00\x7A\x43\x69\xF4\x65\x00\x2A\x21\x7F\x83\x19\x80\xF5\x19\x80\x4B\xA4\x3F\xE1\x56\x33\x00\x43\x19\x80\xCB\x25\x4E\xB8\x1B\x32\x00\x37\x64\x00\x7A\x24\x4E\xB8\x37\x66\x00\x36\x66\x00\x46\x24\x4E\xB8\x37\x65\x00\xDE\x95\x01\x18\x94\x38\xE1\xDE\x9C\x01\x78\x77\x06\x60\x93\xC4\x09\x77\x24\x03\x70\x7B\x28\x7F\x77\x64\x00\x3E\x10\xCA\xDF\x3D\x19\x80\x8F\x85\xFC\xFD\x63\x06\xE0\xBE\x90\xBF\x2D\x19\x80\xAD\x21\x7F\x0F\x66\x00\xFE\x39\xE4\xEF\x91\x0C\xC0\xA3\x21\x7F\x5F\xCC\x30\xC4\x25\x4E\xB8\x5F\xCB\x00\x3C\x16\x0A\xFB\xF5\x0C\xC0\x37\x24\xEE\x2B\x12\xF7\x78\x86\x41\x3C\x6F\x48\xDC\xBF\x64\x18\x7E\xF6\x1B\x39\x7E\x7F\xE3\x79\xDF\xCE\x00\x7C\x3B\x94\xC6\x53\x19\x80\x1F\xC8\xB0\x6A\xED\xFA\xB7\x0C\x43\x30\xBE\x5D\x19\x80\xDD\xA1\x3C\xBF\x90\x01\xF8\x49\x28\x7F\x3F\x15\x69\x4A\xDC\xBD\x6A\x6D\xC8\x00\xBC\x18\x0A\xFB\xEF\x19\x86\xE0\x3A\xF8\x9F\x19\x80\x97\x42\xF1\xFD\x57\x06\xE0\xF7\x12\x77\x8F\xC4\xFD\x21\xC3\x00\x81\xF5\xED\xCF\x19\x86\x60\xD9\x0E\x64\x00\x0E\x4A\x7F\x23\x6A\x1D\xCC\x00\x1C\x96\xB8\xF7\x4A\xDC\x9B\x19\x86\x60\x79\x4F\xCD\x32\x88\xE7\x7E\x39\x97\x9C\x9E\x65\x10\x8F\xF3\x2B\x49\xF3\x66\x19\xA8\x6D\x24\x6E\x46\x96\x41\x3C\x72\x1A\x84\xE5\x59\x86\xDF\xCA\xF8\x85\xBB\x26\x0B\x50\x91\xFE\xE4\xD2\x02\x83\x59\x80\xE1\x2C\xC0\x7E\xE9\x4F\xB8\xB7\x64\x01\x6E\x25\x7F\x8D\x39\xEC\x2B\x93\x00\x7E\x34\x89\xC3\x9C\x24\xC3\xEE\x9A\x04\xF0\xD2\x24\x80\xDF\xC9\xB0\xE4\x4E\x01\x98\x3A\x85\x7F\x57\xEB\x60\xE7\x14\x80\xEB\xA7\x34\xAF\x83\x6B\xA6\x00\xB8\x53\x9A\xD3\x58\x3B\x05\x08\xA8\x3E\x9F\x63\xBF\xDD\x53\x18\x28\xDD\x97\x19\xD7\x33\x85\x41\x3C\x27\x4A\x7F\xF9\x29\x0C\x7B\xF6\xF0\xB7\x70\xD7\x4D\x01\x10\xF0\x82\xC4\x09\xB7\x77\x0A\x40\x29\x94\xEE\xC0\x14\x20\xA0\x7A\x92\x7E\xCB\x53\x18\xC4\xA3\xCB\x34\x2A\x53\x18\xC4\xF3\xA2\xF4\x37\x38\x85\x41\x3C\xBF\x96\xB8\xEA\x14\x80\x5A\x28\x8D\xA1\x29\x00\x1B\x42\xB8\x4D\x53\x00\x86\x65\xD8\x6F\xCA\xB0\x23\x53\x00\x7E\x13\xF2\xF7\xFB\x29\x00\xFB\x42\xB8\x57\xA6\x00\xFC\x39\x84\x83\x0E\x80\x53\x3A\x00\xB6\xC8\x31\x28\xDC\xD3\x3A\x00\x4E\x0B\xE1\x3A\x3A\x00\x3A\x42\xB8\x39\x0F\x00\x0C\x3C\xD0\x1C\xDF\xC8\x03\x00\x9F\x0C\xE1\x72\x5B\x01\xA6\x6D\x6D\xC6\xCD\xD8\x0A\x50\x09\xE1\x36\x6E\x05\xB8\x39\x84\x7B\xF7\x56\x20\xA0\x31\x30\xC4\x69\x6F\xDA\x0A\x70\x7B\xC8\xDF\xFB\xB7\x02\xDC\x15\xC2\xFD\xFD\x56\x20\xA0\x76\xBC\x5A\xCE\x93\x5B\x01\x3E\x1C\xF2\xF7\x0F\x5B\x81\xC0\xEF\x7B\x00\xF0\xD1\xAD\x00\xDB\x42\xFE\x1E\xDA\x0A\x04\xE2\x79\x5C\xC6\xB7\x7D\x2B\x83\x78\x1E\x95\xB8\xCF\x6D\x65\x10\xCF\x97\x25\xEE\xE1\xAD\x0C\xE2\xD9\x21\x71\x9F\xDF\xCA\x10\x8C\xEF\x91\xAD\x0C\xE2\xD9\x27\x71\x8F\x6E\x65\x10\xCF\x73\x12\xF7\x85\xAD\x0C\xE2\xF9\xBD\xC4\x7D\x71\x2B\x03\xB5\x8F\xA4\x99\xBF\xB4\x15\x60\x47\xA8\x1C\xDF\xDA\x0A\xF0\xED\x10\xEE\x3B\x5B\x81\x40\x8D\x35\xE1\x7E\x77\x2B\x83\x78\x9E\x90\x69\x7C\x6F\x2B\x83\x78\x9E\x92\xED\xB1\x6A\x3B\xC3\x9A\x37\xE5\x9A\xFA\xA6\xE7\x0D\x6E\x07\xD8\xB1\x1D\xA0\xB3\x4F\xD2\x15\x7D\x9E\x97\xF5\x00\x66\x31\x05\xEF\xD3\xE8\x8E\x07\x70\xB5\xD7\x4C\xA3\x83\x81\x70\x9A\xC1\x07\x4E\xED\xD2\x5F\xBB\x81\xB0\xC2\x40\x78\x41\xFA\x13\xEE\x23\x06\xC2\x33\x21\x7F\x2F\x18\x08\x87\x43\xFE\x46\x92\x08\x5F\x48\xB2\x3F\xB5\xBF\xDC\x99\x44\x78\x25\x89\x4D\xFB\xB7\x2D\xE3\x11\x76\x8C\xC7\xA6\xFC\xED\x1C\x8F\xF0\xAB\xF1\xD8\x94\xBF\x39\x65\x84\xAE\x72\xB3\xBF\x35\x65\x84\xE1\x72\xB3\x3F\x78\x09\x21\xFB\x12\xFB\xCB\x49\x7F\xB9\x97\x10\xE6\xBD\x84\xF0\x4B\xE9\x4F\xB8\x86\x5C\x6F\x92\x12\xE4\x16\x83\x1E\xB1\xF7\x3D\x5D\xD0\x38\x00\xF0\x2E\x38\xFA\x09\xFF\xFE\x6E\x00\x90\xD3\x03\x5C\x37\x8A\x7F\x7E\xF6\xCB\x2D\xF5\x24\x59\x1D\x57\xB1\x3B\xE7\x03\xEC\xDE\xF1\x6D\x76\x8B\x07\xC8\x1D\xB9\x75\xAA\x43\xFD\x62\x6C\x37\xB9\xB7\x3E\xFF\x31\x72\xF3\x4F\x3F\xED\xDC\x23\x63\xC4\x00\x94\x76\x23\x8C\x86\x3F\xF7\xC7\x08\x1F\x3C\x86\xFF\xE0\xA3\x89\xF9\x13\x00\x22\x72\xCD\x15\xDB\x94\x94\xEC\xA9\x62\x3B\x72\x02\x00\x4C\x04\x22\x91\xE9\x0C\x61\xB2\x58\xE3\x00\x60\x1A\x00\x9C\x0D\x00\xE7\x03\xC0\x6C\x00\x98\x27\xCF\xEF\x16\x4B\xDE\xC1\x2A\x51\x52\x79\x96\x9F\x07\x80\x5E\x00\xE8\x07\x80\x01\x00\x58\x0F\x00\xC3\x00\x70\x0B\x00\x88\xAD\xC3\xFB\x01\xE0\x23\x00\x20\xB6\x42\x9F\x92\xE7\xF5\xA2\xDB\x3F\x2C\xC6\x99\xE4\x25\xED\x10\x73\xB0\x18\x27\x62\x3F\x26\xCF\xF0\xF1\xFF\xC3\xFC\x6E\x1C\x25\xCF\x1F\x38\x46\xBE\xB7\x85\xF2\xFE\x8D\x51\xF2\xBF\x3B\xD0\x06\x22\xDF\xE3\x00\xE0\x78\x00\x78\x87\x2C\xC3\xF8\xC0\xEF\x27\xCA\x35\x7D\x82\xA4\x1F\x26\xF2\x16\xE8\xA8\x47\xF5\xFB\xBF\x93\xAE\x24\x07\xFC\xB3\x1F\x75\x2E\x74\x9A\x74\xD5\x38\x9E\x1C\x8A\xE7\x74\x59\x67\xA2\x6F\x77\x00\xC0\x54\x59\x7F\xC1\xDF\xC5\xBC\x3D\x5D\xD0\x36\xF2\xF7\x93\x09\x77\x1E\xC4\x21\x01\x49\x60\xCD\xF5\xE2\xDB\x81\xAB\xA1\x0B\xBA\xC0\x85\x9B\xE0\x3F\x53\x48\x6D\x96\x3E\x1B\x9B\x38\x91\xB7\x1F\x87\x74\x46\x1E\xC6\xDF\xA2\xF1\xF7\xB8\xB3\x11\xF4\x00\x7E\x7F\x9A\xFD\x87\xF1\x77\x01\xFB\xB7\xCF\xC6\xA6\x71\xFD\x47\xE9\x3F\x8C\x37\x65\x9E\x45\x1E\x45\xFE\x94\x2B\x5A\x3A\x0E\x71\x2A\x83\xE8\x49\xC3\x30\x4C\xFE\xB2\xB0\x89\x7E\xB7\xE5\xAE\x21\x0B\xD3\x60\x16\xCC\x81\x6B\x60\x0D\xBC\x0B\x36\x11\x3F\x95\x8C\x88\x49\x2E\x69\xB8\xFC\xFC\xBD\x00\x5C\x60\x56\xF3\x3B\xD2\x9C\xDF\xEF\x84\xCA\x61\xC9\xFC\x86\xF1\x5B\xA5\xFF\x5D\x67\x23\x68\x01\xFC\xBD\xD2\x7F\x18\x7F\x99\xF4\xFF\x42\xA8\x5E\x2F\x91\xFE\xC3\xF8\x39\xD2\xFF\xCF\x43\xF1\x9C\x2D\xFD\x87\xF1\x67\x48\xFF\xBF\x0D\xE1\x6D\xE9\x3F\x8C\xFF\xD6\x18\xF6\xFF\x87\x50\xBA\x5F\x1D\xC3\xFE\xC3\xF8\xA2\x8C\x7F\x7F\x08\xDF\x25\xE3\x0F\xE3\xFF\x20\xE3\x7F\x35\x84\x7F\x51\xC6\x1F\xC6\x7F\x54\xFA\x3F\x10\xC2\xBF\x5F\xFA\x0F\xE3\x9F\x97\xFE\x61\x66\x73\x3F\xFA\x37\xE9\x3F\x8C\x1F\x96\xF9\x4F\xCF\x6C\xAE\x87\xBA\xCC\x7F\x18\x8F\xD2\xFF\xF8\x99\xCD\xE9\xBE\x2E\xE3\x0F\xE3\x3F\x2C\xFD\xDB\xA1\x74\xDF\xA7\xFA\x7B\x08\xFF\x39\x99\xFF\xCE\x99\xCD\xFD\x6A\x8B\x8C\x3F\x8C\x6F\x83\xE6\xE7\x8B\x3A\xC2\xE3\x3A\xC2\x4F\x92\x08\x67\x68\x08\x67\x26\x11\xBE\x9A\x46\x78\xE7\x09\x08\xD7\x9C\x80\xF0\xCD\x34\x82\xAB\x23\xBC\xD3\x42\xD8\xAC\x23\x7C\x4A\x47\x78\xE2\x38\x84\x9D\x91\x46\xAE\xC5\xDC\x71\x1E\x00\xCC\x0A\xCC\x1D\xE7\x07\xD2\xB8\x40\xCE\xCF\x17\xCA\xF5\x76\x8E\x94\x3B\x08\xFE\x3E\x57\xCE\xDF\x17\x4B\x3E\xD2\x82\xD0\xEF\x0B\x65\xF8\x45\x72\x7E\x57\x73\xA5\x92\x5F\xB0\xE5\x3C\x5F\x94\xB4\x43\xBF\x9C\xEB\x77\xD8\x00\x25\xB1\x3F\x90\xF8\x1A\x00\x08\x10\xE4\x4C\x3D\x20\xE7\x10\x94\x6F\x50\x72\x0D\x76\x40\x9E\x41\xBC\xFF\x4A\xC2\x8E\x89\x2C\xD3\xF0\x5B\x19\xF6\xDF\xC5\x5E\x57\xE4\xD9\x06\xF8\x5D\x40\xD6\x21\x28\xE3\xE0\x05\x65\x1B\xA4\x4C\x83\xA8\x41\x25\xA3\x20\xDE\x95\x6C\x82\xF0\xAE\x64\x12\x04\x5E\xC9\x22\xDC\xEE\x79\x9E\x92\x41\x10\x7E\x82\xB2\x07\x41\x99\x03\x25\x6B\xF0\xE3\x80\x8C\x81\x88\x47\xC9\x16\xFC\x30\x20\x53\x20\xE2\x51\xB2\x04\x4F\x07\x64\x08\x04\x5E\xC9\x0E\xAC\x08\xC8\x0C\x7C\x3F\x20\x2B\xF0\x83\x80\x8C\x80\xF0\xAF\x64\x03\x9E\x0E\xC8\x04\xFC\x28\x24\x0B\xA0\x64\x00\x9E\x09\xF0\xFE\x77\x07\x78\xFE\x22\x9F\x41\x5E\xBF\xE2\xF1\x3F\x13\xE0\xED\xEF\x09\xF1\xF4\x15\x2F\xFF\xB9\x00\x0F\x5F\xC4\xA3\x78\xF7\x3F\x0D\xF0\xEC\x45\x3E\x15\xAF\xFE\xA7\x01\x1E\xBD\xF0\xAF\x78\xF3\x3F\x0D\xF0\xE4\x85\x7F\xC5\x8B\xFF\x49\x80\x07\x2F\xF0\x8A\xF7\xFE\xB3\x00\xCF\x5D\xC4\xA3\x78\xED\xE2\x3D\xC8\x63\x57\xBC\xF5\x30\x4F\x3D\xC8\x4B\x57\x3C\x74\x81\x57\xBC\x73\x0C\xF1\xCC\x15\xAF\x3C\xCC\x23\x0F\xF2\xC6\x15\x4F\x5C\xE0\x83\xBC\x70\xC5\x03\x17\x61\x15\xEF\x5B\xF8\x51\x3C\x6F\x0C\xF1\xBA\x15\x8F\x5B\xE0\x15\x6F\xFB\x3E\xCF\xF3\x14\x4F\xFB\x29\xD1\xCF\x25\x2F\x9B\xE6\x2D\xC9\xC3\xDE\xE4\x79\x9E\xE2\x5D\x0B\xBC\xE2\x59\x9F\x71\x7A\x83\x57\x2D\xD2\x55\x3C\xEA\xEB\x3C\xCF\x53\xBC\xE9\xF6\xD3\x9B\x79\xD2\x8A\x17\x2D\xFC\x2B\x1E\xF4\x6C\xCF\xF3\x14\xEF\x79\x7E\x80\xE7\xBC\x30\xC0\x6B\x16\xE9\x06\x79\xCC\x41\xDE\x72\x90\xA7\x1C\xE4\x25\x07\x79\xC8\x41\xDE\x71\x90\x67\x1C\xE4\x15\x07\x79\xC4\x41\xDE\x70\x90\x27\x1C\xE4\x05\x07\x79\xC0\x41\xDE\x6F\x90\xE7\x1B\xE4\xF5\x06\x79\xBC\x41\xDE\x6E\x90\xA7\x1B\xE4\xE5\x06\x79\xB8\x41\xDE\x6D\x90\x67\x1B\xE4\xD5\x06\x79\xB4\x41\xDE\x6C\x90\x27\x1B\xE4\xC5\x06\x79\xB0\x41\xDE\x6B\x90\xE7\x1A\xE4\xB5\x06\x79\xAC\x41\xDE\x6A\x90\xA7\x1A\xE4\xA5\x2A\x1E\xAA\x68\x3B\xC5\x3B\x15\xEF\x41\x9E\xA9\xE2\x95\xD6\x03\x3C\xD2\x53\x03\xBC\xD1\x53\x02\x3C\xD1\x39\x01\x5E\xE8\x45\x01\x1E\xA8\x2D\x79\x9F\x02\xC4\xBE\xE2\x9F\x24\x1F\xD4\x0E\xF1\x3F\x83\x7C\xCF\x47\x75\x80\x47\x25\x1F\xF3\x0B\x3A\x80\x80\x93\x69\x0D\x05\xF8\x92\xC4\x7F\x59\x07\xF8\xB2\x4C\xF7\x2B\x3A\xC0\x57\x25\xFE\x6B\x3A\x80\x00\x41\xF7\x3F\xA6\x33\x44\xDA\x00\xBE\xAE\x33\xCC\x14\x7B\x0A\x9D\x41\xF8\x7F\x5C\x07\x10\x70\xF0\x4D\xCF\x7B\x42\x07\xD8\x21\xF1\x3B\x75\x00\x01\x62\x4F\xF1\x4D\xC9\x7F\x15\x74\xBB\xE2\xBB\x8A\xBA\xFA\xB6\x0E\xF0\x5D\xE9\xFF\x7B\x3A\x80\x80\x31\x62\x8F\xA2\x03\x3C\x25\xF1\xFF\xAA\x03\x08\xB8\x6E\x22\xC0\xF7\x03\xFC\xD9\x20\x5F\x36\xC8\x8F\x55\x7C\xD8\x58\x88\xFF\xAA\xF8\xAE\x62\x7F\x12\xE4\xB7\x06\xF9\xAC\x41\xFE\xAA\xE2\xAB\x12\x0D\x18\xE0\xA7\x06\xF9\xA8\x41\xFE\x69\x90\x6F\xAA\xF8\xA5\xFF\xE4\x79\x5E\x90\x4F\xAA\xF8\xA3\x62\x7E\x50\x7C\xD1\x83\x9E\xE7\x29\x7E\xA8\x48\x4B\xF1\x41\x89\x07\x2A\xF9\x9F\x2B\x03\x7C\x4F\x5B\xF2\x3B\xA7\x4B\x7E\xE7\x0C\x03\x40\xC0\xF2\xF1\x00\x67\x1A\x0C\x02\x7F\x96\x01\x20\x60\xD9\x78\x80\xB3\x0D\x80\xD9\x12\x7F\xA1\x01\x20\xE0\xE2\xF1\x00\x17\x19\x00\x73\x24\xDE\x31\x00\xE6\x0A\x18\x0F\x30\xCF\x00\x58\x2C\xF1\x97\x18\x00\x02\x2E\x18\x0F\xB0\x24\xC0\x67\x0D\xF2\x57\xD7\x18\x00\x6B\x0D\xCE\x7F\xB7\xC1\x30\x69\x3C\x40\x8F\xE4\xB7\x0A\xBC\xE2\xB3\x8A\x77\xC5\x5F\x1D\x13\xE0\xAB\x8A\x3A\x09\xF2\x53\x15\x1F\x55\xF8\x57\xFC\xD3\xCE\x10\xDF\x54\xF1\x4B\xD7\xB4\x35\xF8\xA4\xE2\x5D\xF1\x47\xC5\x7B\x90\x2F\x1A\xE4\x87\x2A\x3E\xE8\xAE\x8F\x35\xF8\x9F\xB1\x10\xDF\x33\xC8\xEF\x0C\xF2\x39\x83\xFC\xCD\x5B\xDA\x00\x6E\x91\x7C\xCD\x5B\x25\x8F\x53\xE4\x39\xC8\xDB\x0C\xF2\x34\x83\xBC\xCC\x20\x0F\x53\xF1\x2E\xE7\xBC\xEA\x79\x8A\x67\x29\xDE\x83\xBC\xCA\x20\x8F\xB2\x89\x37\x79\x32\xC0\x9A\x93\x39\x5D\xF7\x64\x06\x41\xE3\xAE\x95\xBC\x4A\x81\x0F\xF2\x28\x15\x6F\xD2\xF9\xB5\xE7\x29\x9E\x24\x51\xC9\x01\x5E\xA4\xE2\x41\x8A\x7A\x08\xF2\x1E\x15\xCF\x51\xE0\x83\xBC\x46\xC5\x63\x14\xF8\x20\x6F\x51\xF1\x14\x05\x3E\xC8\x4B\x54\x3C\x44\x81\x0F\xF2\x0E\x83\x3C\xC3\x20\xAF\x30\xC8\x23\x54\xBC\x41\x11\x56\xF1\x04\xC5\xBB\xE2\x05\x8A\x77\xC5\x03\x14\xEF\x41\xDE\x9F\xE2\xF9\x09\x7C\x90\xD7\x17\xE4\xF1\x05\x79\x7B\x41\x9E\x5E\x90\x97\x17\xE4\xE1\x05\x79\x77\x8A\x67\x27\xE2\x0F\xF2\xEA\x14\x8F\x4E\xE0\x83\xBC\x39\xC5\x93\x13\xF8\x20\x2F\x2E\xC8\x83\x53\xBC\x37\xE1\x47\xF1\xDC\xBE\x22\xE6\x5E\xC9\x6B\x7B\xC3\xF3\x3C\xC5\x63\xBB\x3D\xC4\x5B\x53\x3C\x35\x31\x77\x29\x5E\x9A\xF0\x1F\xE4\xA1\x29\xDE\x99\x88\x5F\xF1\xCC\xEE\xF5\x3C\x2F\xC8\x2B\x0B\xF2\xC8\x14\x6F\x4C\xF8\x57\x3C\xB1\x7B\xC4\x1E\x5A\xF2\xC2\xC4\x9C\x16\xE4\x81\x29\xDE\xD7\x88\x98\xEB\x24\xCF\xEB\xBD\x9E\xE7\x29\x5E\x97\xC8\x8F\xE2\x71\xDD\xF2\x6B\xCF\x53\xBC\xAD\xEC\xAF\x3C\x4F\xF1\xB4\x3E\xF1\x2B\xCF\x53\xBC\xAC\xB6\x10\x0F\x4B\xF1\xAE\xD2\x21\x9E\x95\xE2\x55\x89\xB9\x45\xF1\xA8\x4E\x0A\xF3\xA6\x24\x4F\xAA\x33\xC4\x8B\x52\x3C\x28\x31\x2E\x14\xEF\x29\xFE\x9C\xE7\x29\x9E\xD3\x49\x2F\x7B\x9E\xE2\x35\x9D\xF8\x9C\xE7\x05\x79\x4C\x41\xDE\x92\xE2\x29\x89\x3C\x28\x5E\xD2\x6F\xF7\x78\x9E\xE2\x21\xE9\xCF\x79\x9E\xE2\x1D\xBD\xB8\xC7\xF3\x14\xCF\xE8\xD7\x7B\x3C\x4F\xF1\x8A\x44\x1E\x14\x8F\x48\xC4\xA3\x78\x43\xDF\xDC\xE3\x79\x8A\x27\x24\xFC\x28\x5E\x10\xD1\xC9\x92\x07\x44\xE3\x3A\xC0\xFB\x09\xF2\x7C\x82\xBC\x9E\x39\x0F\x00\x5C\xF2\x00\xFB\x5F\xF2\x00\xC3\xEC\x0D\x9E\xB7\x54\xF2\x7E\x04\x5E\xF1\x7C\xC4\xBB\xE2\xF5\xD0\xDE\x50\xF2\x78\xC4\xBB\xE2\xED\x88\x3C\x28\x9E\xCE\x9B\x43\x9E\xA7\x78\x39\xC2\x8F\xE2\xE1\x08\x3F\x8A\x77\xF3\xC2\xD5\x9E\xA7\x78\x36\xC2\x4F\x90\x57\xA3\x78\x34\x02\xAF\x78\x33\x8F\x5F\xED\x79\x8A\x27\xF3\xE8\xD5\x9E\xA7\x78\x31\x5F\xBE\xDA\xF3\x14\x0F\x66\xC7\xD5\x9E\xA7\x78\x2F\xC2\xBF\xE2\xB9\xEC\xBB\xDA\xF3\x14\xAF\xE5\xB9\xAB\x3D\x4F\xF1\x58\x7E\x7F\xB5\xE7\x29\xDE\xCA\x16\xBD\xC1\x53\x11\xE9\x2A\x5E\x0A\x9D\x99\x49\x1E\x8A\xE8\x27\x8A\x77\xF2\xC4\xD5\x9E\xA7\x78\x26\x4F\x0D\x79\xDE\x93\x5B\x01\xBE\x2F\xCB\xF8\xDA\x56\x80\xD7\xE5\x7B\x90\x87\x12\xE4\x9D\x28\x9E\x89\x1D\xE6\x95\x48\x1E\x49\x7B\x88\x37\xA2\x78\x22\xED\x21\x5E\x88\xE2\x81\xCC\x09\xF1\x3E\x14\xCF\xC3\x0E\xF1\x3A\x14\x8F\xC3\x0E\xF3\x36\x24\x4F\x23\x17\xE2\x65\xAC\x53\xC7\x29\x2F\x21\x2C\x79\x09\x61\xE5\x4B\x08\x57\xBD\x84\x70\xFD\x4B\x08\x5D\x2F\x05\x4E\x88\x4E\x40\x38\xE9\x04\x84\x89\x27\x20\x9C\x71\x02\x42\xE7\x09\x08\x17\x9D\x80\x00\x17\x23\x2C\x58\x80\x00\xA6\x61\x9A\xA6\x65\xA6\xCC\x93\xCC\x36\xB3\xCD\x3C\xD9\x3C\xD9\xB4\xCD\x8B\x4C\xC7\xBC\xC4\xBC\xDC\xAC\x98\xEB\xCD\x9F\x9B\xBF\x34\x3D\x73\x65\x74\x38\xBA\x25\xF6\xA5\xD8\x57\x62\x4F\xC6\x9E\x89\xBD\x18\xFB\x4D\xCC\x8B\xAD\x3C\xFC\xCD\xC3\xCF\x1C\x9E\xFD\xD6\xCA\xB7\x3E\xF0\xD6\x87\xDE\x7A\xF2\xAD\xFD\x6F\xBD\xF5\xD6\xE0\x91\xFA\x91\x0D\x47\xDE\x3C\xA2\x92\x5F\xD3\x82\xB0\xA9\x05\x01\xF6\x21\xE8\xFB\x10\x22\xFB\x10\x32\xFB\x10\xB2\xFB\x10\x72\xFB\x10\x26\x49\x38\x4D\xC2\xE4\x7D\x08\x33\xF6\x21\x9C\xB5\x0F\xE1\x9C\x7D\x08\xE7\x49\x38\x5F\xC2\x5C\x09\x0B\x25\x2C\x96\xB0\x44\xC2\xB2\x7D\x08\x2B\xF6\x21\xBC\x73\x1F\xC2\xA5\xFB\x10\x56\x4B\xB8\x42\xC2\x55\x12\xAE\x91\x70\x9D\x84\x2E\x09\xEE\x3E\x84\xB5\xFB\x10\x7A\x24\xF4\xEE\x43\x28\xEE\x43\x28\xED\x43\xA8\xEE\x43\xA8\xEF\x43\xB8\x61\x1F\xC2\xC6\x7D\x08\x37\xEF\x43\xD8\x24\x61\x64\x1F\xC2\xFB\xF6\x21\xBC\x7F\x1F\xC2\x27\xF7\x21\x7C\x66\x1F\xC2\xFD\xFB\x10\x3E\xBB\x0F\x61\xDB\x3E\x84\xED\xFB\x10\xBE\xB1\x0F\x61\xFF\x3E\x84\x97\xF7\x35\xDA\x67\x7A\xE4\xF2\xC8\x55\x91\x3B\x23\x7F\x1F\xF9\x40\xA4\xF5\xF0\x09\x87\xC1\x42\x38\xDB\x42\x38\xCF\x42\xB8\xC8\x42\x80\x09\x4B\x26\xAC\x9C\x70\xF3\x84\x2D\x0F\xBC\xF1\x00\xF4\x21\x9C\x5B\x6C\xE6\xDD\xEC\x2A\x22\xFC\xB1\x88\xB0\xBF\x88\xF0\x5A\x11\x61\xE7\x84\x3F\x4F\x78\x73\x82\x37\x41\xFD\x3E\x12\xBF\x25\xBE\x39\x7E\x57\xFC\x03\xF1\x7B\xE2\x1F\x8E\x3F\x18\x7F\x28\xFE\x48\xFC\x0B\xF1\x2F\xC4\xBF\x12\x7F\x2C\xFE\x78\xFC\x3B\xF1\x27\xE3\x4F\xC5\x7F\x10\xFF\x61\xFC\xF9\xF8\xF3\xF1\x9F\xC7\x7F\x11\xFF\x55\xFC\x37\xF1\xDF\xC5\x8F\xC4\x1B\x07\x80\x08\xB1\x36\x84\x44\x1B\xC2\xCC\x36\x84\x73\xDB\x10\xE6\xB7\x21\xAC\x6C\x43\x28\xB5\x21\xFC\x97\xF6\x7B\x2D\x32\x7D\xDA\xF4\x2D\xD3\x77\x4C\x27\xFF\x69\x84\x65\x69\x84\x4B\xD3\x08\x95\x34\xD2\xF9\xA7\x00\x68\xF7\xDA\xE1\xA4\x09\x27\x9D\x7C\x52\xE6\xA4\x39\x27\xAC\x3A\xA1\x51\x82\x0C\x0C\xC3\x7D\xF0\x30\x3C\x0C\x60\x46\xA8\x0F\xFE\xC2\xFC\x85\x39\x26\x3A\x26\xFA\xEB\xD8\xAF\x63\x2D\xE3\x5B\xC6\x5B\xF6\x18\xFB\x74\x7B\xAA\xBD\xC6\xEE\xB1\xD7\xD9\x15\x1B\x5E\xF0\x0E\x78\x47\xBC\x23\x07\xBD\xC3\xDE\x13\x69\x04\x01\x4F\x4B\xE8\x3C\x0B\xE1\xDC\xB3\x10\xB6\x3C\x8E\x70\xFF\xE3\x08\xB5\xDD\x08\x37\xED\x46\x40\x68\x21\xB0\xA1\x05\x86\xA1\x85\x24\x0A\xDF\xF2\xC6\x90\x94\xC5\x5B\x82\x18\x1D\x37\xFC\x8E\x47\xC6\xBF\x36\x7E\x8B\x86\xB0\x4B\x6B\xD4\x71\x67\x04\xA1\x3B\x82\x50\x91\xA0\xF2\xAB\x9E\xFF\x4E\xBE\x55\x9E\x83\xED\xF7\xBF\x9F\x7F\xF9\x1C\x87\x70\xD6\x71\x08\x33\x8F\x43\x58\x78\x1C\x02\x3C\x34\xF3\xA1\x39\x0F\x2D\x7B\x68\xE5\x43\x57\x3D\x74\xED\x43\x5D\x0F\x6D\x69\x7D\xAD\xF5\x40\xEB\x5B\xAD\x83\xDB\x77\x6C\x7F\x64\x0C\xC2\x0F\xC6\xA0\x98\xD0\x0E\x34\x22\x08\xD5\x83\xCF\xCB\xD8\x06\xDB\xE1\x31\xF8\x06\xEC\x80\xE7\xE1\x79\x78\x03\xDE\x80\xC7\xB4\x5F\x69\xFF\xAE\xBD\xA4\xFD\x41\xF3\xB4\xBA\x5E\xD7\x37\xE9\x9B\xF4\xCD\xFA\x66\xFD\x76\xFD\x76\x3D\x62\x46\xCC\xA4\x99\x34\x27\x98\x13\xCC\x8C\x99\x31\xE7\x98\x73\xA8\x3E\x44\x5D\xF4\xC4\xF3\xF1\x8B\x5A\x2E\x6A\xD9\x33\xE6\x85\x31\x87\xD3\x87\xD3\xBF\x7F\xC7\x1F\xDF\x71\xF6\x09\x33\x4F\xD0\xC6\xEB\xE3\x23\xE3\x23\xE3\x9F\x69\x7B\xA6\xED\xC5\xB6\x17\xDB\x5E\x6A\xFB\x43\xDB\xBE\xB6\x3F\xB7\xBD\xD6\xF6\x46\xDB\xA1\xB6\x43\x6D\x60\x5B\x76\x0B\xD5\xDD\xA0\x5D\xB7\x37\xD9\x23\xF6\xDD\xF6\x16\x7B\x87\x0D\xD9\x53\xB2\xA7\x65\x27\x67\xA7\x64\xA7\x67\xCF\xCC\x2E\xCB\xAE\xC8\x76\x65\xDF\x97\x7D\x7F\xF6\x9E\xEC\xA9\x93\xE6\x4C\xBA\x64\xD2\x9A\x49\xDE\x69\x30\xB9\x76\xC6\x86\x33\x3E\x72\xC6\xC7\xCE\xF0\xCE\x80\xA9\x97\x4E\xDD\x3F\xED\xF0\x34\xE8\x34\x3A\xCD\x4E\xB3\x33\xD6\x69\x77\x76\x76\x9E\xD3\x79\x41\xE7\x45\x9D\x9F\xEC\xBC\xB7\x73\x4B\xE7\x96\xCE\xC3\x9D\x6F\x76\xDE\x33\xFD\xBE\xE9\x3B\xA7\xFF\x66\xBA\x3D\xA3\x6B\xC6\xF0\x8C\x1F\xCD\xF0\x66\x78\x33\xAE\x3C\xD3\x3B\x73\xE7\x32\x6F\x19\x6C\xCD\x6E\xBD\x63\xEB\x9D\x5B\x3B\x1F\x3C\xEF\xC1\xA9\xDB\xA6\x6E\xFB\xD1\xB6\x1F\x6D\xBB\x66\xFB\x35\xDB\x8B\xDB\xFB\xB7\x5F\xF8\xD6\x45\x6F\xA5\x8F\x9C\x78\xA4\xF3\xC8\xA5\x47\x56\x1F\x59\x77\xA4\xEF\x48\xFF\x11\xD1\xD6\xE8\xD9\xDE\xF9\xDE\x1C\xEF\x1A\x2F\xEF\x0D\x7A\x83\xDE\xA7\xBD\xFB\xBC\xBD\xDE\xEF\xBC\xFF\xF4\xF6\x79\x07\xBD\xB7\xB8\x1F\x20\x82\x86\x08\x51\x44\x38\x13\x11\xCE\x41\x84\x8B\x10\xE1\x1E\x44\xB8\x17\x11\x76\x21\xC2\x9B\x88\xF0\xA2\x86\x70\x58\x43\xF8\x75\x05\xE1\x37\x95\x46\xFF\x80\x5D\x08\xAF\xED\x42\x80\xDD\x08\xA7\xEE\x46\x98\xBC\x1B\x61\xDD\x6E\x84\xA2\xEC\x37\xB7\xEC\x46\xB8\x75\x37\xC2\x5D\xBB\x11\xB6\xED\x46\xF8\xFC\x6E\x84\xFF\xDC\x8D\xB0\xF7\xC7\x08\x7F\xFE\x31\x02\x3C\x83\x70\xF9\x33\x08\x6B\x9E\x41\xB8\xF1\x19\x04\x78\x16\x61\xF5\xB3\x08\x97\x3F\x8B\x70\xEF\xB3\x08\x9F\x7E\x16\xE1\xBE\x67\x11\xFE\x97\x84\xCF\x3E\x8B\xF0\xC0\xB3\x08\xDB\x9E\x45\xF8\xDC\xB3\x08\x9F\x7F\x16\xE1\xB1\x67\x11\xBE\x21\xE1\x89\x67\x11\xBE\xFD\x2C\xC2\x77\x9F\x45\x88\xEC\x41\x88\xEE\x41\x48\xEC\x41\x48\xED\x41\x18\xB7\x07\xE1\x1D\x7B\x10\xDA\xF6\x20\x9C\xBC\x07\xE1\xBC\x3D\x08\xE7\xEF\x41\xB8\x70\x0F\xC2\x9C\x3D\x08\x17\xEF\x41\x58\x20\xE1\x92\x3D\x08\x2B\xF7\x20\x5C\xBA\x07\xE1\xB3\xCF\x21\x3C\xF8\x1C\xC2\x0F\x9E\x47\xF8\xE1\xF3\x08\xDE\xF3\x08\xEB\xFF\x80\xF0\xA5\x3F\x20\xE0\x1F\x11\x66\xFF\x11\x01\xF6\x23\x9C\xB1\x1F\xA1\x73\x3F\xC2\x87\xF7\x23\x6C\xD9\x8F\xF0\xF9\xFD\x08\x8F\xEE\x47\xD8\xB1\x1F\xE1\x9B\xFB\x11\x7E\xB4\x1F\x61\xF7\x7E\x84\xD7\xF6\xF3\x41\xE6\xC3\x2F\x23\xFC\xEE\x65\x04\xEF\x65\x04\x7C\x05\x41\x7B\x05\x21\xFD\x0A\xC2\xF9\xAF\x20\xCC\x79\x05\x61\xD1\x2B\x08\x2B\x5F\x41\x78\xE7\x2B\x08\x6B\x5E\x41\xC8\xBF\xC2\x07\x9D\xCF\xBF\x8E\xB0\xF7\x75\x84\x3F\xBC\x8E\xB0\xFF\x75\x84\x37\x5F\xE7\xC3\xCD\xDA\x1B\x08\x23\x6F\x20\xBC\xF0\x06\xC2\xDE\x37\x10\x7E\xFF\x06\xD2\xC5\x1D\xEB\x00\x42\xFA\x00\xC2\xC2\x03\x08\x2B\x0F\x20\x5C\x75\x00\x61\xCD\x01\x84\xDB\x0F\x20\xDC\x73\x00\xE1\xE1\x03\x08\x8F\x1C\x40\x78\xF4\x00\x02\x1C\x44\xB8\xF1\x20\xC2\x4D\x07\x11\x7E\x70\x10\xE1\xE9\x83\x08\xAB\x0E\x21\xAC\x39\x84\x30\x70\x08\x61\xF0\x10\x42\xFD\x10\xC2\x8D\x87\x10\x6E\x3A\x84\x30\x72\x08\xE1\xB6\x43\x08\xF7\x1C\x42\x78\xF0\x10\xC2\x23\x87\x10\xBE\x72\x08\x61\xE7\x21\x84\x6F\x1D\x42\xD8\x75\x08\xE1\xB9\x43\x08\x70\x18\xE1\x43\x87\x11\xFE\xE1\x30\xC2\xF7\x0F\x23\xEC\x3F\x8C\x70\xF0\xF0\xD1\xF3\xCA\xAF\xF5\x3F\xE9\x23\x53\xFE\x3C\xE5\xE0\x14\x6F\x0A\xBC\xE0\xFD\x0A\x6C\x84\x8F\xFF\x1D\x02\x4C\x42\x28\x4F\x42\x18\x9C\x84\x50\x9F\x84\x30\x32\x09\x61\xDE\x29\x08\x10\x43\x88\xC4\x10\x62\x12\x12\x31\x84\xB3\x63\x08\xE7\xC4\x10\xCE\x8D\x21\x5C\x20\xE1\x22\x09\x60\xDC\x6A\xDC\x6E\x4C\x8B\x8C\xB4\xDD\xD1\x76\xC6\xC4\x33\x26\xDE\x38\xF1\xC6\x89\x7B\x3B\xBC\x8E\x39\x0F\xDC\xF7\xC0\xD4\x23\xD3\x8E\x80\xA4\xAD\xFC\x27\xBE\x32\x7E\x59\xBC\x3B\xBE\x2E\x3E\x1C\xDF\xFB\xA0\xF7\x20\x9C\x88\x60\x9E\x88\x10\x97\x90\x3C\x11\xA1\xF5\x44\x84\xE3\x4F\x44\x78\xC7\x89\x08\xE3\x4F\x44\x38\xFB\x44\x84\x73\x4E\x44\x38\xF7\x44\x84\xF3\x4F\x44\x58\x70\x22\xC2\xCA\x13\x11\xAE\x3A\x11\x01\xC6\x23\x9C\x3F\x3E\x10\xFF\xE3\x08\xC5\xC7\x11\x06\x1F\x47\xB8\xF9\x71\x84\x91\xC7\x11\xEE\x78\x1C\xE1\x9E\xC7\x11\x3E\xFE\x38\xC2\xBD\x8F\x23\xDC\xF7\x78\xC0\x7F\x27\xC2\xD4\xB3\x1A\xF3\x34\x44\x10\x4E\x8B\x20\xEC\x1D\x83\xC4\xA3\x83\xB1\x8B\xC6\x5E\x32\x76\xD9\xD8\x95\x63\x2F\x1F\x7B\xE5\xD8\x2B\xC7\x5E\x3D\xF6\xBA\xB1\x6B\xC6\xDE\x31\xF6\xCE\xB1\x7F\x3F\xF6\x9E\xB1\x8F\x8C\xFD\xC2\xD8\x2F\x8F\xFD\xDA\xD8\x7F\x19\xBB\x73\xEC\xCE\xB1\xDF\x1A\xFB\xDD\xB1\x4F\x8D\x7D\x6E\xEC\x0B\x63\xD3\xAD\x63\x5B\x8F\x6F\x1D\xDF\x7A\x75\xEB\x75\xAD\x37\xB7\x8E\xB4\x7E\xBC\x75\xA4\xE3\xA3\x1D\x5B\x3A\x1E\xE8\x78\xB0\xE3\xF3\x1D\x8F\x74\x7C\xA5\xE3\x6B\x1D\xFF\xD2\xB1\xB3\xE3\x7B\x1D\x4F\x75\xFC\xB0\x63\x57\xC7\x73\x1D\x2F\x74\xFC\xB2\x03\xB7\x9B\xDB\xE3\xDB\x5B\xB6\x1F\xB7\xFD\x1D\xDB\xED\xED\xA7\x6E\x6F\xDF\x3E\x75\xFB\x96\xF4\x77\xD3\x4F\xA6\x9F\x4C\x3F\x9D\x7E\x3A\xBD\x2B\x7D\x28\xFD\x66\xDA\x4B\xDF\xD3\xF6\xF5\xB6\x27\xDA\x76\xB4\x41\xC7\x29\x1D\xA7\x75\x9C\xD6\x21\x1E\x98\x32\x75\x4A\xE7\x94\xEB\xA7\xC0\x5E\x04\x73\x2F\x42\x6C\x2F\xC2\xF8\xBD\x08\x13\xF6\x22\x64\xF7\x22\xFC\xDD\x5E\x84\x49\x7B\x11\x4E\xDD\x8B\x70\xFA\x5E\x84\x4E\x1D\xE1\x12\x1D\x01\x5A\x11\xF4\x56\x84\x48\x2B\x42\xB2\x15\x61\x4C\x2B\x42\xBA\x15\xA1\xB5\x15\xA1\xBD\x15\xE1\xF4\x56\x84\xCE\x56\x84\x19\xAD\x08\x67\xB6\x22\x9C\xDD\x8A\x70\x5E\x2B\xC2\x05\xAD\x08\x17\xB7\x22\x2C\x6C\x45\x58\xD4\x8A\xB0\xA4\x15\x61\x59\x2B\xC2\x4A\x09\x57\x48\xB8\xAE\x15\xA1\xBB\x15\x61\x5D\x2B\x42\xA9\x15\x61\xB0\x15\xA1\xDE\xDA\xA8\xFF\x41\xBD\xA6\x0F\xE9\x37\xE8\x37\xE9\xEF\xD6\x87\xF5\x61\xFD\x56\xFD\x56\xFD\x36\xFD\x36\xFD\x0E\xFD\x4E\xFD\x2E\xFD\x2E\xFD\x6E\xFD\x33\xFA\xFD\xFA\x8B\xFA\x7E\xDD\xD3\x4F\x9D\x78\xFA\xC4\xEB\x26\xBA\x13\xD7\x4D\x2C\x4E\xDC\x31\x71\xC7\x44\xC8\x1C\x9F\x19\x9F\x99\x98\xB1\x33\xF3\x33\x8B\x32\xCB\x32\x2B\x33\x57\x64\xAE\xCA\x5C\x95\xB9\x26\x73\x4D\xE6\xBA\xCC\x75\x99\xAE\xCC\xBB\x33\x23\x99\x2F\x65\xBE\x92\xF9\x4E\xE6\x7B\x99\x67\x32\xCF\x65\x7E\x96\xF9\x45\xE6\x4F\x99\x57\x32\xAF\x66\x5E\xCF\x1C\xC9\x9C\x9A\x3D\x35\x9B\xDF\x9E\xDF\x1E\x5C\xBF\xE7\x20\xC2\xDD\x88\xB0\x45\x02\xFC\x18\x61\xFE\x8F\x1B\xF9\x7D\x4F\xE2\x96\xC4\xE6\xC4\xDF\x27\x3E\x90\xF8\x60\xE2\xC3\x89\x07\x13\x0F\x25\x1E\x49\x7C\x21\xF1\xC5\xC4\x97\x13\x8F\x25\x1E\x4F\x7C\x37\xF1\x64\xE2\x5F\x13\x3F\x48\x3C\x9D\xD8\x95\xD8\x95\xD8\x9B\xF8\x4D\xE2\x77\x89\x97\x13\x07\x13\x5E\xC2\xA7\x9F\x27\x22\xE4\x27\x22\xF4\x4E\x44\xE8\x9B\x88\x50\x9C\x88\x70\xF7\x44\x84\x7B\x26\x22\x7C\x70\x22\xC2\x87\x27\x22\xFC\xE3\x44\x84\x2D\x13\x11\xB6\x4D\x44\xC0\x84\x9E\x88\x24\x12\x89\x31\x89\x74\xA2\x35\xD1\x9E\x38\x3D\xD1\x99\x98\x91\x38\x33\x71\x76\x62\x66\xE2\xDC\xC4\x79\x89\x0B\x12\x17\x24\x2E\x4C\xCC\x4D\x2C\x4C\x2C\x4A\x2C\x49\x2C\x4B\xBC\x33\xF1\xCE\xC4\x55\x89\x6B\x13\xD7\x27\xAE\x4F\xAC\x4B\x6C\x48\x8C\x4C\xFD\xF8\xD4\x4F\x4E\xFD\xF3\x54\x98\xB6\x67\x5A\xA4\x33\xD2\x19\xED\x8C\x76\x66\x3B\x27\x77\x9E\xDB\x79\x7E\x27\x9C\xB5\x63\x19\x2C\x7F\xF3\x3E\x38\x38\x70\x68\xF0\xD0\x4F\x0E\x89\xFC\xED\xAF\x20\xBC\x5C\x41\x92\x22\xFA\xC5\x03\x1A\xC0\x56\x0D\xCE\xFA\xAA\x06\x73\xBE\xAA\xC1\xC4\xAF\x69\x60\x7F\x4D\x83\xCF\xFC\x50\x83\x47\x7E\xA8\xC1\xDE\xDF\x6B\x00\x07\x34\x98\x78\x88\x25\x8E\x2E\x69\xD5\x01\x8E\xF3\x8E\x9B\xDA\x39\xAD\x73\xFA\xF4\xBB\xA7\xC3\x8C\x93\x67\xAC\x99\xB1\x69\xC6\x9A\x6D\x37\x6F\x83\xCF\xDD\xFF\xFC\x23\xCF\x7F\xEF\xF9\x1F\x3C\x7F\xF8\xF9\x2D\xDE\xBF\x78\xDF\xF2\x9E\xF4\xBE\xEF\xFD\xC8\xFB\xB1\xF7\xBC\xF7\x53\xEF\xE7\x0D\x1A\x2A\x85\x70\x5A\x0A\xA1\x33\x85\x70\x5E\x0A\xC1\x3E\xE1\xAC\x13\x82\xED\xB3\x37\x86\xF0\x4A\x0C\xE1\xD5\x18\xC2\x6B\x31\x84\xC3\x31\x04\x2F\xD6\x68\x9F\x0F\x46\x9E\x8C\xEC\x8A\xFC\x57\xE4\x4F\x91\x57\x23\x13\x0F\xCF\x3C\x7C\xEE\xE1\x0B\x0E\x5F\x78\xF8\xC2\xC3\x73\x0E\x3B\x87\xE7\x1D\xBE\xF8\xF0\x82\xC3\x2B\x0E\x3B\x9D\x1F\xED\xFC\x54\xE7\x7D\x24\xAE\x88\x8F\x20\x9C\xFC\x28\xC2\xCA\x47\x11\x2E\x7D\x14\x1B\x97\x3F\x55\x7F\x88\x21\x5C\x16\x43\xB8\x22\x86\xD0\x25\xD2\xD1\x2B\xFA\x66\xE3\x36\x63\x89\x79\x99\x39\x68\x0E\x9A\xEF\x8C\xAF\x8E\x3F\x72\xD2\xCE\x93\x76\xB5\xFD\xB8\xED\xD9\xB6\xBD\x6D\xBF\x6E\xFB\xCF\xB6\x3F\xB6\xFD\xB1\xED\xD5\xB6\x57\xDB\x0E\xB4\x1D\x6C\xDB\x39\xF1\xE0\xC4\xC3\x13\xBD\x89\x49\x3B\x65\xEF\xB2\xF7\xDB\xA7\x77\x76\x74\x7E\xBC\xF3\x13\x9D\x70\x64\xCC\x11\xFB\x48\xC7\x91\xB7\x10\x41\xC0\x5E\x8D\xE1\xFC\x56\x86\xDE\xDD\x08\x85\xDD\x08\xEF\xDA\x8D\xF0\xDE\xDD\x08\x9B\x77\x23\xBC\x7F\x37\xC2\x43\xBB\x11\x1E\xDE\x8D\x00\xD8\x02\x7F\xC2\x16\x58\x63\x21\x54\x2D\x84\x1B\x2D\x84\x61\x0B\x61\x8E\x85\x70\x99\x85\x70\xA5\x85\xD0\x65\x21\x8C\x6C\x7B\x7A\xDB\xAE\x6D\x3F\xD9\xF6\xCB\x6D\xBF\xDA\x26\xCA\x33\x92\x66\x9E\xBE\xA2\x4F\x47\x92\x77\x25\xEF\x4E\xDE\x93\xFC\x50\xF2\xC1\xE4\x43\xC9\x2F\x26\xBF\x9C\x7C\x2C\xF9\x78\xF2\x3B\xC9\xEF\x25\x9F\x4A\x7E\x3F\xF9\x74\x72\x4F\xF2\xB9\xE4\x2F\x93\xBF\x4C\xEE\x4D\xFE\x26\xF9\xBB\xE4\x9F\x92\x2F\x27\x5F\x49\x06\xDB\xE3\x33\x9D\x87\x3A\xDF\xEA\xF4\x3A\xF7\x4F\xF7\xA6\xEF\x9A\x71\x64\x06\x9C\x79\xC5\x99\xEB\xBC\x8A\xB7\xDE\xFB\x14\xB7\xE9\x23\xCC\x48\xED\x79\x14\xA1\x57\xD4\xEF\xB6\x8E\x6D\xD3\xB6\x4D\xDB\xE6\x47\x90\x40\xD0\x13\x08\x91\x04\x82\x99\x40\x48\x26\x10\x5A\x13\x08\xC7\x27\x10\x4E\x48\x20\x9C\x98\x40\x38\x3B\x81\x70\x6E\x02\x61\x56\x02\xE1\xA2\x04\xC2\xA2\x04\xC2\xCA\x04\xC2\x95\x09\x84\xDF\x56\x18\xE0\xFD\x08\x7B\xEE\x42\x18\x39\xE1\x17\x27\xEC\x3D\xE1\xA5\x13\xF6\x9F\x70\xF0\x84\xBD\x27\x7A\x27\xC2\x58\x84\xE3\xC6\x22\xB4\x8E\x45\xB8\x70\x2C\xC2\x23\x63\x11\xFE\x6B\x2C\xC2\xFE\xB1\x08\x07\xC7\xA2\x2F\x1F\xF4\x10\x3C\x04\x5F\x86\x2F\xC3\xD7\xE1\xEB\xB0\x13\x9E\x83\x17\xE0\x75\x38\x00\xDF\xD4\xBE\xA7\xED\xD6\xF6\x6A\xBF\xD5\xFE\xA0\xFD\x41\xDB\xA7\xED\xD3\x06\xF5\xBA\xBE\x21\x30\x5F\x1D\x3D\x57\xBD\xA6\xBF\xA1\xBF\xC7\xB8\xD3\x98\x16\xE1\xBD\xE0\x55\x91\x35\x91\x3B\x64\x9F\x7C\x25\x62\x9B\x97\x98\x65\xDA\x87\x3F\x63\xEE\x31\xF7\x98\xFF\x6E\xFE\xCE\xDC\x67\xFE\xC9\x3C\x64\xBE\x69\x7A\xA6\x67\xA6\xA3\xE9\xE8\xD8\xE8\xB4\xE8\xB2\xE8\x67\xA3\x8F\x46\x1F\x8D\x7E\x3F\xFA\x5F\xD1\x57\xA3\xAF\x45\x0F\x45\x0F\x45\x21\x76\x7C\xEC\xA4\xD8\x49\xB1\x49\xB1\x49\xB1\xF6\x58\x7B\x6C\x4E\xEC\xCA\xD8\x9A\x58\x31\xA6\xF6\xF1\x46\xFC\xBC\xF8\xEC\xF8\xEC\xF8\xCA\xF8\xCA\xF8\x95\x71\x37\xBE\x3E\x3E\x72\xCC\xBD\xE2\x13\xF1\x27\xE2\x3F\x8C\xFF\x50\xEE\x11\x5F\x8C\xEF\x8F\xBF\x1C\x7F\x33\xFE\x66\xFC\xED\xE6\x1A\x35\xAF\x54\x13\xF5\x63\xCE\x81\x4F\x24\x9E\x90\x73\xDF\x8B\x89\x83\x89\x83\x89\x88\x95\xB4\xC6\x58\x69\xAB\xD5\x6A\xB7\x4E\xB7\x3A\xAD\x19\xD6\x99\xD6\xD9\xD6\x79\xD6\x6C\x6B\xB6\x75\xAD\x75\x9D\xD5\x65\xB9\xD6\x7A\x6B\xBD\x75\x8B\x75\x8B\xB5\xD9\xBA\xD3\xBA\xDB\xBA\xC7\xFA\x90\xF5\x11\xEB\xE3\xD6\x27\xAC\x7B\xAD\x7B\xAD\x4F\x5B\xF7\x59\xF7\x5B\xFF\x64\x3D\x68\x3D\x64\x7D\xDE\x7A\xCC\xDA\x65\xED\xB2\x22\xC9\x64\xB2\x25\x99\x4E\x8E\x4D\xB6\x27\x4F\x4F\x9E\x97\x9C\x9D\x9C\x9D\xBC\x32\x79\x75\x72\x4D\xD2\x4D\x8E\x24\x47\x92\x9B\x47\xE9\xD3\x4F\x24\x9F\x90\xFD\xF8\x45\xEA\xC3\x46\x2A\x99\x6A\x49\xA5\x53\x63\x53\xB3\x52\xB3\x53\xB3\x53\xCB\x53\xCB\x53\xAB\x53\x97\xA7\xBA\x52\x6E\xEA\xA6\xD4\x70\x6A\x73\xEA\xA3\xA9\x4F\xA4\x1E\x4D\x7D\x31\xF5\x8D\xD4\x13\xA9\x27\x52\x3B\x53\xDF\x4B\x61\x4B\x67\xCB\x8C\x96\x33\x5B\xE6\xB4\x2C\x68\x79\x4F\xCB\x7B\x5B\x6E\x6D\xB9\xB5\xE5\xB6\x96\x3B\x5B\xEE\x6A\xB9\xFF\xFF\x61\xEF\x4B\xE0\x9B\xA8\xB6\xFF\xCF\x99\x4C\xDA\xA6\x69\x9A\x14\x4A\x97\xA4\xBD\x49\x2C\x10\x02\x43\x08\x8B\x2C\x65\x2B\xBB\xA2\x2C\xAE\x80\x0B\x69\x68\xD3\x36\xD0\x26\x21\x49\x59\x5C\xA0\x6C\x2A\x08\x82\xFB\x86\x52\x15\x67\xDA\x54\x05\x14\x10\x65\x2B\x2E\xB8\x22\xA8\x2C\x0A\x28\x05\x15\x51\x40\x70\x57\x96\xE4\xFF\x99\x99\x9B\x36\x8D\x45\x7D\xEF\xFD\x7C\xEF\xFD\x7F\x3F\x87\xCF\xA1\xB9\xE7\xDE\xB9\xFB\xFD\x9E\x73\xB7\x33\xA9\x4F\xA7\x3E\x9D\x2A\xA4\xAE\x4E\x7D\x3E\xF5\x85\xD4\xCD\xA9\x9B\x53\xEB\x53\x5F\x4B\xDD\x9E\xBA\x3D\xF5\x60\xEA\xA1\x54\xD0\x82\xB6\x40\x3B\x42\x7B\xA9\xB6\x5C\x7B\x87\x76\xB1\x16\x74\x1D\x75\x03\x75\x03\x75\x63\x75\xD7\xE8\xAE\xD3\xDD\xA8\x73\xEA\x9C\x3A\x97\xAE\x44\xE7\xD1\xF9\x74\x95\xBA\x39\xBA\xBB\x74\x77\xE9\x7E\x2B\xF7\xFF\x35\x3D\xA4\xAA\xF5\xD2\xD6\xD1\xB9\x23\xA6\x97\x67\x78\x33\x66\x67\xCC\xC9\x78\x24\xA3\x3A\xE3\xEB\x8C\x6F\x33\x7E\xC9\x80\xCC\x94\xCC\xD4\xCC\xB4\x4C\x53\x66\xD7\xCC\x82\xCC\x2B\x32\x0B\x33\xCB\x33\x3D\x99\xBE\xCC\xAA\xCC\x17\x32\xF7\x65\xEE\xCB\x3C\x98\x79\x30\xD3\x94\x35\x23\xAB\x2A\x6B\x7E\xD6\xED\x59\x35\x59\xA1\xAC\x50\x96\x38\xBF\x86\x6C\x92\x3D\x36\xBB\x22\xDB\x97\x1D\xCC\xAE\xCA\xAE\xCB\x5E\x9D\xFD\x56\x36\xE8\x33\xF4\x26\xFD\xD5\x7A\x41\x2F\xE8\x95\x86\xEE\x86\x61\x86\xCB\x0C\xF3\x0C\xD5\x86\xE7\x0C\xAB\x0C\x2F\x1B\xBE\x30\x40\x4E\x5E\xCE\xA8\x9C\x31\x39\xD7\xE5\xDC\x9A\x23\xEA\x8D\x51\x7D\xA6\x69\xBE\xF8\x93\x3C\x5F\x14\x65\x3C\xF9\xB3\x32\x7E\x93\x71\x8B\x71\x8B\xF1\x15\x49\xD6\xBF\x6B\xDC\x45\xE5\x7D\x83\xF1\x04\x95\xF7\xE7\x8C\x53\x4D\x53\x4D\xB3\x4D\xB3\x4D\xCB\x4C\xCB\x4D\x8C\x99\x31\x27\x9A\x13\xCD\xC9\xE6\xD6\xE6\x36\xE6\x36\xE6\x6C\x73\xAE\xB9\xAD\xB9\xAD\x59\xD4\x09\x3A\x98\x3B\x98\x3B\x9A\x3B\x9B\xBB\x98\xFB\x98\xFB\x99\x07\x9A\x87\x99\x2F\x35\x8F\x36\x8F\x36\xCF\x33\xCF\x37\x47\xF5\xAC\x42\xEE\x73\xEE\x38\xF7\x0D\xF7\x1D\xF7\x3D\x17\xAB\x87\xD9\x3B\x97\x76\xF6\x76\xF6\x76\xFE\x33\xFA\x5E\x17\x5B\x17\x9B\xD2\x9E\x60\xEF\x6A\xBF\xD8\xDE\xD7\xDE\xCF\x1E\x95\x4B\x22\xA6\x9E\xB3\x47\xEC\xCA\xAE\x5D\xBA\x8A\x72\xB4\xBA\xEB\xD6\xAE\x22\xBE\x36\xCA\xEB\xEA\xC5\x2B\x77\xAD\x3C\xBF\x12\x9E\x4E\xE1\x75\xBC\x91\xEF\xC8\x77\xE2\x0B\x78\x0F\x3F\x9B\x7F\x8C\xAF\xE6\xBF\xE0\x33\x05\xA3\x70\x91\x70\x87\x70\xA7\xB0\x55\x78\x45\x78\x5B\xF8\x51\xC0\x1A\x45\x8D\xB2\x26\xB1\x26\xB9\x26\xA5\xE6\xA2\x9A\x82\x9A\x40\xCD\xDC\x9A\x17\x6A\xBE\xAB\xF9\xB9\xE6\x6C\xCD\xD9\x9A\xF3\x35\xE1\x9A\xE4\xDA\x76\xB5\xF6\xDA\xE1\xB5\xA2\xAC\x9E\x5F\xFB\xBC\x34\x1F\x6E\xA8\xFD\xBC\xF6\x68\xED\xA9\xDA\x33\xB5\xE1\x5A\x08\x75\x08\x15\x84\x06\x87\x86\x86\x2E\x0B\x15\x86\xA6\x85\x6E\x0A\xDD\x14\x9A\x15\x5A\x15\x5A\x13\x5A\x13\x5A\x17\x5A\x1F\x7A\x29\xB4\x39\x54\x1F\xAA\x0F\xBD\x12\x7A\x25\x74\x20\xF4\x49\xA8\x21\xF4\x75\xE8\xBB\xD0\x0F\xA1\x96\x74\x5A\x7B\xDD\x75\x75\x37\xD4\xB9\xEB\x7C\x75\x47\xEA\x62\xF5\x82\x46\x3D\xE4\x6C\xC2\xD9\xD6\x67\x33\xCF\xE6\x9E\xCD\x3D\x6B\x3C\xDB\xE1\x6C\xC7\x16\xE5\xF8\x9A\xB8\xB5\xCA\xB3\x8D\x6B\x95\xE6\x48\x7E\x64\x50\xE4\xBA\x48\x49\x8B\x7A\x06\x82\x1A\x50\xB2\x92\xD3\x1E\x10\x3A\x00\x42\x3E\x20\xF4\x03\x84\xFE\x80\x30\x10\x10\x46\x01\xC2\x58\x40\xB8\x11\x10\xAA\x00\xE1\x8C\xF8\x97\x41\x58\xCE\x20\x34\xAE\x27\x29\x10\x8C\x0A\x84\xCE\x0A\x84\x02\x05\xC2\x60\x05\xC2\xA5\x0A\x84\xB1\x0A\x84\x4A\x05\x42\x15\x3D\x2B\x57\xAD\x40\x78\x55\x81\xF0\xA6\x02\x61\xA7\xA8\x6F\xB3\x08\x8F\xB1\x08\xD1\x75\xE6\xE8\x1A\x73\x74\xDE\x11\x5D\x9F\x82\x04\x84\x9E\x89\x08\x05\x89\x08\xD7\x24\x22\x14\x26\x22\x94\x26\xFE\x63\xF3\x33\x51\x6F\x29\x4C\x42\x98\x96\x84\x50\x95\x84\xF0\x78\x12\xFE\x46\x7F\x02\x15\x42\x1B\x15\x82\x49\x85\xD0\x47\x85\x50\xA5\x42\x78\x51\x85\xB0\x45\x85\xB0\x55\x85\x92\x7C\x16\x49\x77\x01\xB9\x5C\x98\x8C\x70\x4B\x32\x42\x55\x32\xC2\xF2\x64\x84\xFA\x64\x84\x37\x92\x11\xDE\x12\x65\xB3\xF8\x2E\x5D\xFF\x8C\xEA\x25\x51\x7D\xA5\x4A\x8D\x70\xB7\x1A\xA5\x83\x78\x97\xA4\x88\x7A\x88\xBC\xBE\x1E\x5D\x5B\x17\xF5\xC0\x3C\x0D\x42\x55\x2A\x42\x6D\x2A\xC2\xEA\x54\x84\x35\xA9\x08\xA0\x45\xC8\xD1\x22\x58\x28\xD9\xB5\x08\xC3\xB4\x08\xAB\xB5\x08\xAF\x69\x9B\xE6\x6D\x0A\x1D\x42\x2F\x1D\xC2\x3C\x1D\xC2\x2A\x1D\xC2\x2E\x1D\xC2\x31\x1D\x82\x22\x0D\xA1\x7D\x1A\xC2\x50\x4A\x23\x28\x8D\x4D\x43\xF0\xA7\x21\x4C\xA3\x34\x2F\x0D\xE1\xF9\x34\x84\x6D\x69\x08\xAF\xA5\x21\xEC\xA7\x74\x90\x52\xAC\x5E\xD1\xA9\x15\x42\x55\x2B\x84\xDB\x5A\x21\xDC\x41\x69\x51\x2B\x84\x25\xAD\x10\x96\xB6\x42\x78\xAC\x15\xC2\x8A\x56\x08\x35\x54\xFF\xF8\xB4\xD5\x3F\x3E\xDF\xEA\x4F\x29\x3A\xBF\x12\xE7\x55\x4E\x71\x1E\x95\x8E\xD0\x23\x1D\x61\x44\x3A\xC2\xC8\x74\x04\x47\x3A\x82\x33\x1D\xA1\x2A\x1D\x61\x55\x3A\xC2\x6B\xE9\x08\xAF\xA7\x23\xBC\x41\xA9\xAA\x0D\xC2\x73\x6D\x10\x3E\x6A\x83\x70\xA0\x0D\x02\x64\x20\x74\xC9\x40\x18\x4A\xA9\x2A\x03\x21\x94\x81\xB0\x81\x52\x74\xBF\x41\x9C\x67\x77\xCA\x42\xA8\xA6\x7B\x1C\x91\xAC\x3F\x9E\xB7\x77\xC9\x46\x18\x48\x69\x10\xA5\xEA\x6C\x04\x21\x1B\x21\x94\x8D\xB0\x2B\x1B\xE1\x30\xA5\xCF\x28\x81\x5E\x26\xB5\x1E\xA1\x9B\x1E\x21\x9F\xD2\x58\x4A\x37\xE8\x11\x16\xEA\x11\x1E\xA3\x54\xAF\x47\xF8\x45\x7C\x27\x66\x5D\xDB\x96\x83\x50\x40\xC9\x9F\x83\xB0\x34\x07\x01\x72\x11\x12\x72\x11\x92\x72\x11\x54\xB9\x08\xEA\x5C\x04\x7B\x2E\xC2\x70\x4A\x2D\xCD\xCB\x16\xD2\xB9\x98\x48\x0D\x04\xE1\x3B\x82\xB0\xDA\x28\x53\x74\x1D\xA5\x71\xDD\x84\xAE\x23\x34\xEE\xC3\xD0\xB5\xFD\x82\xC9\x08\x13\x27\x63\xE3\xBA\x3E\x4C\x41\xE8\x32\x05\xA1\x60\x0A\xC2\x90\x29\x08\x45\x53\x10\xA6\x4F\x41\xB8\x75\x0A\xC2\x52\x91\x4F\xF7\x8A\xC0\x8B\x30\xD2\x8B\x30\x96\xD2\xBD\x5E\x84\x15\x5E\x84\x06\x2F\xC2\x61\xBA\x66\x28\x12\xF8\x10\x7E\xA6\x87\x39\x45\xFD\x17\x96\x20\x24\x2D\x91\x75\xEE\xD8\x79\x4D\x54\xFF\xF6\xAD\x41\x38\xFB\x3C\xFE\xE1\x3A\xC9\x7F\xF3\xBA\x61\xFD\x5E\x84\x57\xF6\x22\xEC\xDF\x8B\x70\x70\x2F\xC2\x99\xBD\x08\xE7\xF6\x22\xA4\xEF\x43\xC8\xD8\x87\xD0\x63\x1F\x42\xCF\x7D\x08\xA3\xF7\x21\x8C\xDD\x87\xE0\xD9\x87\xE0\xDB\x87\x70\xC7\x3E\x84\x45\xFB\x10\x6A\xF6\x21\x84\xF6\x21\xBC\xB2\x0F\xE1\xB5\x7D\xF2\x3A\x24\x1C\x46\xE0\x0E\x23\xF4\x3A\x8C\xD0\xFF\x30\xC2\x68\x4A\xF5\x47\x10\x8E\x1F\x41\x80\x63\x08\xAF\x1D\x43\x69\x7F\x6E\x08\xDD\xA3\x13\xE9\xEF\x7D\x30\x79\x1F\xEC\x9F\x5D\x2F\x68\x59\x8F\x6E\xD2\x9F\xDF\x4C\xDD\x91\xBA\x2B\xF5\x63\x59\x8F\x96\xCE\x7B\x37\xCD\xD7\x9A\xCF\xD5\x36\x48\xF3\x34\xC8\x6D\x97\xCB\xE5\xDE\x90\x3B\x29\xD7\x95\x3B\x25\x77\x7A\xEE\xCC\xDC\x2D\xB9\x40\x22\x24\x56\xF7\xEB\x68\xEE\x64\xEE\x66\xEE\x26\xE9\x74\x85\xE6\x3B\xCC\x85\xDC\x6C\xEE\x22\xE1\xF6\x18\x7D\x29\x22\xC8\x3A\x4A\x71\x5D\x49\x9D\xBB\x4E\xD4\x47\xA2\x3A\x05\xE4\xF4\xCA\xE9\x9B\x73\xA9\xA4\xBF\x4A\xF3\xD7\x6C\x92\x6D\xCA\xEE\x94\x6D\xCF\xEE\x9B\x5D\x90\x5D\x90\x3D\x34\x7B\x4C\x76\xEC\xFC\xF8\x82\x72\xBC\xF1\xF9\xE7\xF4\x91\x5D\x2B\x23\x2B\x1B\xB1\x28\xE6\x11\xEB\x46\x9C\xC7\x1E\x92\xEA\x24\x02\x88\x88\x0A\x54\xA0\x12\x95\x98\x88\x89\xA8\x42\x15\xAA\x51\x8D\x1A\xD4\xA0\x16\xB5\x98\x86\x69\xD8\x1A\x5B\x63\x1B\x6C\x83\x99\x98\x89\xD9\x98\x8D\x06\x34\x60\x2E\xE6\xA2\x11\x8D\x68\x46\x33\xE6\x61\x1E\xB6\xC3\x76\x68\x41\x0B\x5A\xD1\x8A\x9D\xB0\x13\x76\xC6\xCE\xD8\x05\xBB\x60\x57\xEC\x8A\xDD\xB1\x3B\x5E\x8C\x17\x63\x2F\xEC\x8D\xF9\x98\x8F\xFD\xB0\x1F\x0E\xC0\x01\x58\x80\x05\x38\x18\x07\xE3\x50\x1C\x8A\xC3\x71\x38\x5E\x82\x97\xE2\x65\x78\x19\x8E\xC2\x51\x38\x06\xC7\xE0\x15\x78\x05\x5E\x85\x57\xE1\x35\x78\x0D\x8E\xC3\x71\x38\x01\x27\xE0\xF5\x78\x3D\xDE\x88\x37\xA2\x03\x1D\xE8\x44\x27\x16\x61\x11\xBA\xD0\x85\xA5\x58\x8A\x6E\x74\xE3\x14\x9C\x82\x15\x58\x81\x5E\xF4\xE2\x54\x9C\x8A\x01\x0C\x60\x25\x56\xE2\x74\x9C\x8E\x37\xE1\x4D\x78\x0B\xDE\x82\xB3\xB0\x0A\xE7\xE1\x3C\x5C\x80\x0B\xF0\x0E\xBC\x03\x17\xE3\x12\xBC\x07\xEF\xC1\xFB\xF1\x7E\x7C\x18\x1F\xC5\xC7\xF1\x71\x7C\x02\x9F\xC0\xA7\xF0\x29\x7C\x1A\x9F\xC6\x1A\xAC\xC1\x10\xD6\xE1\xB3\xF8\x2C\xAE\xC6\xD5\xB8\x16\xD7\xE2\x7A\x5C\x8F\x2F\xE1\xCB\xB8\x19\xB7\xE2\x76\xDC\x8E\x6F\xE1\x5B\xF8\x2E\xBE\x8B\xEF\xE1\x7B\xB8\x0B\x77\xE1\x07\xF8\x01\xEE\xC6\xDD\xB8\x17\xF7\xE2\x47\xF8\x11\xEE\xC7\xFD\x78\x10\x3F\xC1\x43\x78\x08\x0F\xE3\x61\xFC\x0C\x3F\xC3\x2F\xF0\x0B\xFC\x12\xBF\xC4\xAF\xF0\x2B\x3C\x8E\xC7\xF1\x24\x9E\xC4\x53\x78\x1A\xBF\xC7\xEF\xF1\x47\xFC\x11\x7F\xC5\x5F\xF1\x2C\x9E\xC5\xF3\x78\x1E\x23\x18\x41\x64\x90\x51\x30\x0A\x46\xC9\x28\x99\x44\x26\x91\x51\x31\x2A\x46\xCD\xA8\x19\x0D\xA3\x61\xB4\x8C\x96\x49\x63\xD2\x98\xD6\x4C\x6B\xA6\x0D\xD3\x86\xC9\x64\x32\x99\x6C\x26\x9B\x31\x30\x06\x26\x97\xC9\x65\x8C\x8C\x51\x9C\xD2\x30\x79\x4C\x1E\xD3\x8E\x69\xC7\x58\x18\x0B\x63\x65\xAC\x4C\x27\xA6\x13\xD3\x99\xE9\xCC\x74\x61\xBA\x30\x5D\x99\xAE\x4C\x77\xA6\x0F\xD3\x8F\xE9\xC7\x0C\x64\x0A\x98\xC1\xCC\x60\x66\x04\x33\x82\xB9\x94\xB9\x94\xB9\x8C\xB9\x8C\x19\xC5\x8C\x62\xC6\x30\xF7\x32\xF7\x33\xAB\x98\xA9\x8A\xA9\x8A\x80\x22\xA0\x98\xAE\x98\xAE\xB8\x59\x71\xAB\x62\x99\x62\x99\xE2\x19\xC5\x7B\x8A\x5D\x8A\xF7\x15\x7B\x14\xFB\x14\x1F\x2B\x3E\x56\x1C\x50\x1C\x50\x7C\xA2\xF8\x44\x71\x48\x71\x48\x71\x58\x71\x58\xF1\x99\xE2\x33\xC5\x17\x8A\x2F\x14\x5F\x2A\xBE\x54\x7C\xA5\xF8\x4A\x71\x5C\x71\x5C\x71\x52\x71\x52\x71\x4A\xF1\xBD\xE2\x47\xC5\x8F\x8A\x5F\x14\xBF\x28\xCE\x2A\xCE\x29\xEC\xAC\x83\x75\xB2\x4E\xB6\x88\x2D\x62\x5D\xAC\x8B\x2D\x65\x4B\x59\x37\xEB\x66\xA7\xB0\x53\xD8\x0A\xB6\x82\xF5\xB2\x5E\x76\x2A\x3B\x95\x0D\xB0\x01\xB6\x92\xAD\x64\xA7\xB3\xD3\xD9\x99\xEC\x4C\xF6\x66\xF6\x66\xF6\x56\xF6\x56\x76\x36\x3B\x9B\x9D\xC3\xCE\x61\xEF\x64\xEF\x64\x97\xB0\x4B\xD8\xA5\xEC\x52\xF6\x6E\xF6\x6E\xF6\x5E\xF6\x5E\xF6\x7E\xF6\x7E\xF6\x41\xF6\x41\xF6\x61\xF6\x61\xF6\x51\xF6\x51\xF6\x31\xF6\x31\x76\x05\xBB\x82\x7D\x82\x7D\x82\x7D\x8A\x7D\x8A\x7D\x9A\x7D\x9A\x15\x58\x81\xAD\x65\x6B\xD9\x3A\xB6\x8E\x7D\x96\x7D\x96\x5D\xC5\xAE\x62\xD7\xB0\x6B\xD8\x17\xD8\x17\xD8\x75\xEC\x3A\xF6\x45\xF6\x45\xF6\x25\xF6\x25\x76\x23\xBB\x91\xDD\xCC\x6E\x66\xB7\xB2\x5B\xD9\x57\xD8\x57\xD8\xD7\xD8\xD7\xD8\xED\xEC\x76\xF6\x4D\xF6\x4D\xF6\x6D\xF6\x6D\xF6\x5D\xF6\x5D\xF6\x3D\x76\x27\xFB\x3E\xFB\x3E\xFB\x21\xFB\x21\xBB\x87\xDD\xC3\xEE\x63\xF7\xB1\x1F\xB3\x1F\xB3\x07\xD8\x03\xEC\x27\xEC\x27\xEC\x21\xF6\x10\x7B\x98\x3D\xCC\x7E\xC6\x7E\xC6\x7E\xC1\x7E\xC1\x7E\xC9\x7E\xC9\x7E\xC5\x7E\xC5\x1E\x67\x8F\xB3\x27\xD9\x93\xEC\x29\xF6\x14\xFB\x2D\xFB\x2D\xFB\x3D\xFB\x3D\xFB\x23\xFB\x23\xFB\x33\xFB\x33\xFB\x2B\xFB\x2B\x7B\x96\x3D\xCB\x9E\x67\xCF\xB3\x11\x36\xC2\x8A\x83\x57\xA1\x54\x28\xC5\x27\x51\x99\xA8\x54\x29\x55\x4A\xB5\x52\xAD\xD4\x28\x35\x4A\xAD\x52\xAB\x4C\x53\xA6\x29\x5B\x2B\x5B\x2B\xDB\x28\xDB\x28\x33\x95\x99\xCA\x6C\x65\xB6\xD2\xA0\x34\x28\x73\x95\xB9\x4A\xA3\xD2\xA8\x34\x2B\xCD\xCA\x3C\x65\x9E\xB2\x9D\xB2\x9D\xD2\xA2\xB4\x28\xAD\x4A\xAB\xB2\x93\xB2\x93\xB2\xB3\xB2\xB3\xB2\x8B\xB2\x8B\xBC\x3E\xA5\x3B\xA3\x3B\xAF\x8B\xE8\x7E\x69\x7D\xBE\xB5\x38\x27\x87\xDC\x4E\x14\x3F\x1F\xC9\x45\x82\x44\x41\x14\x44\x49\x94\x24\x91\x24\x12\x15\x51\x11\x35\x51\x13\x0D\xD1\x10\x2D\xD1\x92\x34\x92\x46\x5A\x93\xD6\xA4\x0D\x69\x43\x32\x49\x26\xC9\x26\xD9\xC4\x40\x0C\x24\x97\xE4\x12\x23\x31\x12\x33\x31\x93\x3C\x92\x47\xDA\x91\x76\xC4\x42\x2C\xC4\x4A\xAC\xA4\x13\xE9\x44\x3A\x93\xCE\xA4\x0B\xE9\x42\xBA\x92\xAE\xA4\x3B\xE9\x4E\x2E\x26\x17\x93\x5E\xA4\x17\xE9\x43\xFA\x90\xBE\xA4\x2F\xE9\x4F\xFA\x93\x81\x64\x20\x19\x44\x06\x91\x21\x64\x08\x19\x46\x86\x91\x11\x64\x04\xB9\x94\x5C\x4A\x2E\x23\x97\x91\x51\x64\x14\x19\x43\xC6\x90\x2B\xC8\x15\xE4\x2A\x72\x15\xB9\x86\x5C\x43\xC6\x91\x71\x64\x02\x99\x40\xAE\x27\xD7\x93\x1B\xC9\x8D\xC4\x41\x1C\xC4\x49\x9C\xA4\x88\x14\x11\x17\x71\x91\x52\x52\x4A\xDC\xC4\x4D\xA6\x90\x29\xA4\x82\x54\x10\x2F\xF1\x92\xA9\x64\x2A\x09\x90\x00\xA9\x24\x95\x64\x3A\x99\x4E\x66\x92\x99\xE4\x66\x72\x33\xB9\x95\xDC\x4A\x66\x93\xD9\x64\x0E\x99\x43\xE6\x91\x79\x64\x01\x59\x40\x6E\x27\xB7\x93\x85\x64\x21\xB9\x93\xDC\x49\x96\x90\x25\x64\x29\x59\x4A\xEE\x26\x77\x93\x7B\xC9\xBD\xE4\x7E\xF2\x18\x59\x41\x56\x90\x27\xC8\x13\xE4\x29\xF2\x14\x79\x9A\x3C\x4D\x04\x22\x90\x5A\x52\x4B\xEA\x48\x1D\x79\x96\x3C\x4B\x56\x91\x55\x64\x0D\x59\x43\x5E\x20\x2F\x90\x75\x64\x1D\x79\x91\xBC\x48\x5E\x22\x2F\x91\x8D\x64\x23\xD9\x4C\x36\x93\xAD\x64\x2B\xD9\x46\xB6\x91\x57\xC9\xAB\xE4\x75\xF2\x3A\x79\x83\xBC\x41\xDE\x22\x6F\x91\x77\xC8\x3B\x64\x07\xD9\x41\x76\x92\x9D\xE4\x7D\xF2\x3E\xF9\x90\x7C\x48\xF6\x90\x3D\x64\x1F\xD9\x47\x3E\x26\x1F\x93\x03\xE4\x00\xF9\x84\x7C\x42\x0E\x91\x43\xE4\x30\x39\x4C\x3E\x23\x9F\x91\x2F\xC8\x17\xE4\x4B\xF2\x25\xF9\x8A\x7C\x45\x8E\x93\xE3\xE4\x24\x39\x49\x4E\x91\x53\xE4\x5B\xF2\x2D\xF9\x9E\x7C\x4F\x7E\x24\x3F\x92\x9F\xC9\xCF\xE4\x57\xF2\x2B\x39\x4B\xCE\x92\xF3\xE4\x3C\x89\x90\x44\xA3\xCE\xD8\xC6\x68\x32\x5A\x8C\x76\x63\x2F\x63\x81\x71\x98\xB4\xE6\x52\x68\x2C\x35\xFA\xA4\xB5\x96\xDB\x8D\xCB\x8C\x0F\x1A\xAB\x8D\x82\x71\xB5\xB4\xEE\xF2\x62\xCC\xBA\xCB\x1B\x74\xDD\x65\x9F\xB1\xC1\xF8\x25\x5D\x77\xF9\xD9\x98\x6C\x4E\x36\xA7\x9A\xB5\xE6\xD6\xE6\xD6\xE6\x2E\xE6\x2E\xE6\x1E\xE6\x1E\xE6\x3E\xD2\xDA\x4A\x7F\xF3\x70\xBA\xB6\x32\xDF\x3C\xDF\x6C\xE7\x26\x72\x4E\xCE\xC9\xB9\xB8\x12\xAE\x8C\x2B\xE3\x26\x73\x93\xB9\x72\xAE\x9C\x9B\xCA\x4D\xE5\x02\x5C\x90\x9B\xC6\xDD\xCC\xCD\xE1\xE6\x70\xF3\xB8\x79\xDC\x02\x6E\x01\x77\x3B\x77\x3B\xB7\x90\x5B\xC8\xDD\xC9\xDD\xC9\x2D\xE1\x96\x70\x4B\xB9\xA5\xDC\xDD\xDC\xDD\xDC\xBD\xDC\xBD\xDC\xFD\xDC\xFD\xDC\x83\xDC\x83\xDC\xC3\xDC\xC3\xDC\xA3\xDC\xA3\xDC\x63\xDC\x63\xDC\x0A\x6E\x05\xF7\x04\xF7\x04\xF7\x14\xF7\x14\xF7\x34\xF7\x34\x27\x70\x02\x57\xCB\xD5\x72\x75\x5C\x1D\xF7\x2C\xF7\x2C\xB7\x8A\x5B\xC5\xAD\xE1\xD6\x70\x2F\x70\x2F\x70\xEB\xB8\x75\xDC\x8B\xDC\x8B\xDC\x4B\xDC\x4B\xDC\x46\x6E\x23\xB7\x99\xDB\xCC\x6D\xE5\xB6\x72\xDB\xB8\x6D\xDC\xAB\xDC\xAB\xDC\xEB\xDC\xEB\xDC\x1B\xDC\x1B\xDC\x5B\xDC\x5B\xDC\x3B\xDC\x3B\xDC\x0E\x6E\x07\xB7\x93\xDB\xC9\xBD\xCF\xBD\xCF\x7D\xC8\x7D\xC8\xED\xE1\xF6\x70\xFB\xB8\x7D\xDC\xC7\xDC\xC7\xDC\x01\xEE\x00\xF7\x09\xF7\x09\x77\x88\x3B\xC4\x1D\xE6\x0E\x73\x9F\x71\x9F\x73\x27\xB8\x13\xDC\x37\xDC\x37\xDC\xF7\x71\xEB\x47\x83\xF8\x41\xFC\x10\x7E\x08\x3F\x8C\x1F\xC6\x8F\xE0\x47\xF0\x97\xF2\x97\xF2\x97\xF1\x97\xF1\xA3\xF8\x51\xFC\x18\x7E\x0C\x7F\x05\x7F\x05\x7F\x15\x7F\x15\x7F\x0D\x7F\x0D\x3F\x8E\x1F\xC7\x4F\xE0\x27\xF0\xD7\xF3\xD7\xF3\x37\xF2\x37\xF2\x0E\xDE\xC1\x3B\x79\x27\x5F\xC4\x17\xF1\x2E\xDE\xC5\x97\xF2\xA5\xBC\x9B\x77\xF3\x53\xF8\x29\x7C\x05\x5F\xC1\xCF\xE1\xE7\xF0\xF3\xF8\x79\xFC\x02\x7E\x01\x7F\x3B\x7F\x3B\xBF\x90\x5F\xC8\xDF\xC9\xDF\xC9\x2F\xE1\x97\xF0\x4B\xF9\xA5\xFC\xDD\xFC\xDD\xFC\xBD\xFC\xBD\xFC\xFD\xFC\xFD\xFC\x83\xFC\x83\xFC\xC3\xFC\xC3\xFC\xA3\xFC\xA3\x7C\x9E\x90\x27\xB4\x13\xDA\x09\x16\xC1\x22\x58\x05\xAB\xD0\x49\xE8\x24\x74\x16\x3A\x0B\x5D\x84\xAE\x42\x77\xA1\xBB\x70\xB1\x70\xB1\xD0\x4B\xE8\x25\xF4\x11\xFA\x08\x7D\x85\xBE\x42\x7F\xA1\xBF\x30\x50\x18\x28\x0C\x12\x06\x09\x43\x84\x21\xC2\x30\x61\x98\x30\x42\x18\x21\x5C\x2A\x5C\x2A\x5C\x26\x5C\x26\x8C\x12\x46\x09\x63\x84\x31\xC2\x15\xC2\x15\xC2\x55\xC2\x55\xC2\x35\xC2\x35\xC2\x38\x61\x9C\x30\x41\x98\x20\x5C\x2F\x5C\x2F\xDC\x28\xDC\x28\x38\x04\x87\xE0\x14\x9C\x42\x91\x50\x24\xB8\x04\x97\x50\x2A\x94\x0A\x6E\xC1\x2D\x4C\x11\xA6\x08\x15\x42\x85\xE0\x15\xBC\xC2\x54\x61\x86\x70\x93\x70\x93\x70\x8B\x70\x8B\x30\x5B\x98\x2D\xCC\x11\xE6\x08\xF3\x84\x79\xC2\x02\x61\x81\x70\xBB\x70\xBB\xB0\x58\x58\x2C\xDC\x25\xDC\x25\xDC\x2D\xDC\x2D\xDC\x2B\xDC\x2F\x3C\x28\x3C\x28\x3C\x2C\x3C\x2C\x3C\x2A\x3C\x2A\x3C\x26\x3C\x26\xAC\x10\x56\x08\x4F\x08\x4F\x08\x4F\x09\x4F\x09\x4F\x0B\x4F\x0B\xE2\x53\x2B\xD4\x0A\xAB\x84\x55\xC2\x3A\x61\x9D\xF0\xA2\xF0\xA2\xF0\x92\xF0\x92\xB0\x51\xD8\x28\x6C\x16\x36\x0B\x5B\x85\xAD\xC2\xAB\xC2\xAB\xC2\x9B\xC2\x9B\xC2\xDB\xC2\xDB\xC2\x4F\xC2\x4F\xC2\x19\xE1\x8C\xA4\x2B\x16\xD6\x95\xD5\xF9\xEA\xB6\xD6\x45\xD7\xAE\x24\x5D\x31\xF6\x6C\x21\x5D\xE7\x69\x3C\x3B\x48\xE7\xD1\xD1\xB3\x81\xFA\xDD\x08\xDD\x77\x23\x8C\x8E\x99\x77\x95\xEE\x46\x98\xBB\x1B\xE1\xD1\xDD\x08\xEB\x7F\x67\x7E\xB5\x73\x37\xC2\xD7\xBB\x11\x14\x74\x3E\xD5\x8B\xCE\x97\xA6\xEC\x41\xB8\x6D\x0F\xC2\x8A\x3D\x08\x2F\xEF\x41\xF8\x70\x0F\xC2\x37\xE2\x5C\x6C\x2F\xC2\x45\x7B\x11\xFA\xEE\x45\xB8\x76\x2F\x82\x77\x2F\xC2\xA2\xBD\xF2\xFC\x2A\x76\x5E\x75\x38\x6E\x5E\x65\x88\x9B\x57\x5D\x13\x37\xAF\x5A\x1A\x37\xAF\x7A\x8B\xCE\xAD\x44\x8A\x9E\x75\x5C\xCD\xFC\x76\x4F\x25\xA8\x08\x2A\x6E\x52\xDC\xA4\x98\xA0\x9C\xA0\x2C\x48\x28\x90\xF6\x44\xA2\x7B\x1E\xB1\xFB\x1D\x53\x55\x53\x55\xC3\x53\x87\x4B\xF3\x82\x73\xBA\x73\xBA\x7D\x99\xFB\x32\x87\x64\x0D\xC9\x12\xF4\x82\x7E\x46\xCE\xAD\x39\x5C\xEE\x64\xE9\xFC\xC1\xA3\xB9\x5B\x73\x63\xF5\xFD\x5B\xB8\x5B\x39\x6F\x67\x2F\x5D\x6F\x55\xD2\xF5\xD6\xBE\xF6\xC7\xEC\x8F\xDB\xCF\xD9\xC3\xF6\x36\xD5\x6D\xAA\x7F\x59\x79\x7E\x65\x0A\x9F\xC2\xCF\xE6\x67\xF3\xCB\xF9\xC7\xA4\x35\x54\x9F\xE0\x13\xEE\x10\xEE\x10\x7E\x11\x7E\x15\x76\xD6\xEE\xAC\x3D\x5A\x7B\xB4\xD6\x17\xF2\x85\x3E\x09\x7D\x12\xFA\x3E\xF4\x43\xE8\x86\x3A\x47\x9D\xBB\xCE\x5D\x17\x3D\x0B\x24\xED\x17\xD2\x79\xFD\x85\xE6\xEB\xD1\xF9\x65\x74\xFE\x18\x9D\xC7\x6C\xC4\x8D\x58\x8F\xAF\xE2\x7D\xCC\x7D\x4C\xD3\x5E\x91\x7C\x6E\x33\x76\xBF\xE8\x8F\xF6\x8A\x20\xA9\xCD\x1F\xEE\x0B\xF9\xFF\xDE\x17\xFA\x8B\xF6\x85\x86\xFD\xA9\x7D\xA1\x7F\x76\x4F\x08\xD2\xFE\xF5\x3D\xA0\x7F\x74\xEF\x47\xDE\xF7\x19\x9C\x35\xF4\x1F\xDC\xFB\xF9\xFD\x7D\x9F\xE9\xBF\xD9\xEB\xB9\xD8\xDC\xDB\x6C\xEF\x5C\xFA\xA7\xF6\x4D\x12\xEC\x09\xF6\x7E\x74\xBF\x64\x05\xDD\x2F\x89\xFC\xEE\x7E\x49\x7A\x75\x86\xB4\x67\xF2\xF3\x4A\x78\x5A\xDD\xB8\x67\xE2\xE1\x3D\xD2\x7E\xC9\x52\x61\xA9\xF0\xB3\xF0\xB3\x70\xF6\x9F\xD8\x27\x69\xA8\xFD\xBC\xF6\xCB\x16\xF7\x48\xBC\xA1\xA9\x7F\xB8\x4F\x72\x90\xEE\x93\x7C\xD7\xE2\x3E\x49\xFD\x05\xF6\x47\xFE\xD9\x7D\x91\xE8\xBE\xFC\xEF\x9E\xBF\xF8\x0F\xEF\x8F\x8C\xA5\xFB\x23\xFF\xD7\xF7\x43\xFE\xDE\xFF\xF8\xEF\xDC\xFF\xF8\x7B\xDF\xE3\x5F\xDB\xF7\x88\xDF\xEF\x88\xDF\xE7\xF8\xCB\xF6\x33\xE8\xFA\xFE\x05\xD7\xF5\xFF\x5E\xC7\xFF\x97\xD6\xF1\xA3\xCF\xEB\xF8\x3A\xBE\x89\x6F\xE2\x3B\xF8\x0E\x7E\x87\xDF\xE1\x1D\xC6\xA5\xC6\x87\x8C\x2B\x8C\x35\xC6\x55\xC6\x4D\xC6\x4D\xC6\x77\x8D\xEF\x1A\xCF\x19\xA3\x96\x38\xE5\xF5\x7C\x79\x0D\xFF\x53\x51\xF4\x20\x83\x0C\xB2\xC8\x62\x02\x26\x60\x12\x26\x61\x32\x26\x63\x0A\xA6\x60\x2A\xA6\xA2\x0E\x75\xD8\x0A\x5B\x61\x3A\xA6\x63\x06\x66\x60\x16\x66\xA1\x1E\xF5\x98\x83\x39\x48\x90\xA0\x09\x4D\x78\x11\x5E\x84\x6D\xB1\x2D\xB6\xC7\xF6\xD8\x01\x3B\x60\x47\xEC\x88\x1C\x72\x68\x43\x1B\xDA\xD1\x8E\xDD\xB0\x1B\xF6\xC0\x1E\xD8\x13\x7B\x62\x1F\xEC\x83\x7D\xB1\x2F\xF6\xC7\xFE\x38\x10\x07\xE2\x20\x1C\x84\x43\x70\x08\x0E\xC3\x61\x38\x02\x47\xE0\x48\x1C\x89\x97\xE3\xE5\x38\x1A\x47\xE3\x58\x1C\x8B\x57\xE2\x95\x78\x35\x5E\x8D\xD7\xE2\xB5\x38\x1E\xC7\xE3\x75\x78\x1D\xDE\x80\x37\xE0\x44\x9C\x88\x85\x58\x88\x93\x70\x12\x16\x63\x31\x96\x60\x09\x96\x61\x19\x4E\xC6\xC9\x58\x8E\xE5\xE8\x41\x0F\xFA\xD0\x87\x7E\xF4\x63\x10\x83\x38\x0D\xA7\xE1\x0C\x9C\x89\x37\xE3\xCD\x78\x2B\xDE\x8A\x73\x70\x2E\xCE\xC7\xF9\x78\x1B\xDE\x8E\x0B\xF1\x4E\xBC\x0B\xEF\xC6\x7B\xF1\x3E\x7C\x00\x1F\xC2\xE5\xF8\x18\xAE\xC0\x6A\x7C\x12\x9F\xC4\x95\xB8\x12\x79\x14\xB0\x16\x6B\xF1\x19\x7C\x06\x9F\xC3\x55\xB8\x06\x5F\xC0\x75\xB8\x0E\x5F\xC4\x0D\xB8\x09\x37\xE1\x6B\xF8\x1A\xBE\x81\x6F\xE0\xDB\xF8\x36\xEE\xC0\x1D\xB8\x13\x77\xE2\xFB\xF8\x3E\x7E\x88\x1F\xE2\x1E\xDC\x83\xFB\x70\x1F\x7E\x8C\x1F\xE3\x01\x3C\x80\x9F\xE2\xA7\xD8\x80\x0D\x78\x04\x8F\xE0\xE7\xF8\x39\x1E\xC5\xA3\x78\x0C\x8F\xE1\xD7\xF8\x35\x9E\xC0\x13\xF8\x0D\x7E\x83\xDF\xE2\xB7\xF8\x03\xFE\x80\x3F\xE1\x2F\x78\x06\xCF\xE0\x39\x3C\x87\x61\x0C\x23\x30\xC0\x88\x0F\xCB\xB0\x4C\x02\x93\xC0\x24\x31\x49\x4C\x32\x93\xCC\xA4\x30\x29\x4C\x2A\x93\xCA\xE8\x18\x1D\xD3\x8A\x69\xC5\xA4\x33\xE9\x4C\x06\x93\xC1\x64\x31\x59\x8C\x9E\xD1\x33\x39\x4C\x0E\x43\x18\xC2\x98\x18\x13\x73\x11\x73\x11\xD3\x96\x69\xCB\xB4\x67\xDA\x33\x1D\x98\x0E\x4C\x47\xA6\x23\xC3\x31\x1C\x63\x63\x6C\x8C\x9D\xB1\x33\xDD\x98\x6E\x4C\x3E\xD3\x97\xE9\xCF\x0C\x60\x06\x31\x83\x98\x21\xCC\x70\xE6\x12\xE6\x12\x66\x24\x33\x92\xB9\x9C\xB9\x9C\x19\xCD\x8C\x66\x7C\x0A\x9F\xC2\xAF\xF0\x2B\xA6\x29\xA6\xB5\x70\x1E\x6F\xA9\xE2\x6E\xE9\x4C\x5E\x9D\x62\xA7\x62\xA7\xE2\x03\xC5\x6E\xC5\x47\x8A\x8F\x14\xFB\x15\xFB\x15\x07\x15\x07\x15\x9F\x2A\x3E\x55\x34\x28\x1A\x14\x47\x14\x47\x14\x9F\x2B\x3E\x57\x1C\x55\x1C\x55\x1C\x53\x1C\x53\x7C\xAD\xF8\x5A\x71\x42\x71\x42\xF1\x8D\xE2\x1B\xC5\x0F\x8A\x1F\x14\x3F\x2B\x7E\x56\xFC\xAA\x38\xA3\x38\xAF\xE8\xC2\x16\xB2\x85\xEC\x24\x76\x12\x5B\xCC\x16\xB3\x25\x6C\x09\x5B\xC6\x96\xB1\x93\xD9\xC9\x6C\x39\x5B\xCE\x7A\x58\x0F\xEB\x63\x7D\xAC\x9F\xF5\xB3\x41\x36\xC8\x4E\x63\xA7\xB1\x33\xD8\x19\xEC\x4D\xEC\x4D\xEC\x2D\xEC\x2D\xEC\x2C\x76\x16\x5B\xC5\x56\xB1\x8B\xD8\x45\xEC\x62\x76\x31\x7B\x17\x7B\x17\xBB\x8C\x5D\xC6\xDE\xC3\xDE\xC3\xDE\xC7\xDE\xC7\x3E\xC0\x3E\xC0\x3E\xC4\x3E\xC4\x3E\xC2\x3E\xC2\x2E\x67\x97\xB3\x8F\xB3\x8F\xB3\xD5\x6C\x35\xFB\x24\xFB\x24\xBB\x92\x5D\xC9\xF2\x2C\xCF\xD6\xB0\x35\x6C\x88\x0D\xB1\xCF\xB0\xCF\xB0\xCF\xB1\xCF\xB1\xAB\xD9\xD5\xEC\xF3\xEC\xF3\xEC\x5A\x76\x2D\xBB\x9E\x5D\xCF\x6E\x60\x37\xB0\x2F\xB3\x2F\xB3\x9B\xD8\x4D\xEC\x16\x76\x0B\x5B\xCF\x6E\x63\x5F\x65\x5F\x65\x5F\x67\x5F\x67\xDF\x60\xDF\x60\xDF\x62\xDF\x62\xDF\x61\xDF\x61\x77\xB0\x3B\xD8\x5D\xEC\x2E\xF6\x03\xF6\x03\x76\x37\xBB\x9B\xDD\xCB\xEE\x65\x3F\x62\x3F\x62\xF7\xB3\xFB\xD9\x83\xEC\x41\xF6\x53\xF6\x53\xB6\x81\x6D\x60\x8F\xB0\x47\xD8\xCF\xD9\xCF\xD9\xA3\xEC\x51\xF6\x18\x7B\x8C\xFD\x9A\xFD\x9A\x3D\xC1\x9E\x60\xBF\x61\xBF\x61\x4F\xB3\xA7\xD9\xEF\xD8\xEF\xD8\x1F\xD8\x1F\xD8\x9F\xD8\x9F\xD8\x5F\xD8\x5F\xD8\x33\xEC\x19\xF6\x1C\x7B\x8E\x0D\xB3\x61\x16\x94\xA0\x64\x94\x8C\x92\x55\xB2\xCA\x04\x65\x82\x32\x49\x99\xA4\x4C\x56\x26\x2B\x53\x94\x29\xCA\x54\x65\xAA\x52\xA7\xD4\x29\x5B\x29\x5B\x29\xD3\x95\xE9\xCA\x0C\x65\x86\x32\x4B\x99\xA5\xD4\x2B\xF5\xCA\x1C\x65\x8E\x92\x28\x89\xD2\xA4\x34\x29\x2F\x52\x5E\xA4\x6C\xAB\x6C\xAB\x6C\xAF\x6C\xAF\xEC\xA0\xEC\xA0\xEC\xA8\xEC\xA8\xE4\x94\x9C\xD2\xA6\xB4\x49\xE7\x2B\x63\xCF\xC9\x55\xB7\xFE\xB1\x75\xE3\xD9\x78\x02\x62\x37\x23\x2C\x61\x49\x02\x49\x20\x49\x24\x89\x24\x93\x64\x92\x42\x52\x48\x2A\x49\x25\x3A\xA2\x23\xAD\x48\x2B\x92\x4E\xD2\x49\x06\xC9\x20\x59\x24\x8B\xE8\x89\x9E\xE4\x90\x1C\x22\x3E\x26\x62\x22\x17\x91\x8B\x48\x5B\xD2\x96\xB4\x27\xED\x49\x07\xD2\x81\x74\x24\x1D\x09\x47\x38\x62\x23\x36\x62\x27\x76\xD2\x8D\x74\x23\x3D\x48\x0F\xD2\x93\xF4\x24\xBD\x49\x6F\x92\x4F\xF2\x49\x3F\xD2\x8F\x0C\x20\x03\x48\x01\x29\x20\x83\xC9\x60\x32\x94\x0C\x25\xC3\xC9\x70\x72\x09\xB9\x84\x8C\x24\x23\xC9\xE5\xE4\x72\x32\x9A\x8C\x26\x63\xC9\x58\x72\x25\xB9\x92\x5C\x4D\xAE\x26\xD7\x92\x6B\xC9\x78\x32\x9E\x5C\x47\xAE\x23\x37\x90\x1B\xC8\x44\x32\x91\x14\x92\x42\x32\x89\x4C\x22\xC5\xA4\x98\x94\x90\x12\x52\x46\xCA\xC8\x64\x32\x99\x94\x93\x72\xE2\x21\x1E\xE2\x23\x3E\xE2\x27\x7E\x12\x24\x41\x32\x8D\x4C\x23\x33\xC8\x0C\x72\x13\xB9\x89\xDC\x42\x6E\x21\xB3\xC8\x2C\x52\x45\xAA\xC8\x5C\x32\x97\xCC\x27\xF3\xC9\x6D\xE4\x36\x72\x07\xB9\x83\x2C\x22\x8B\xC8\x62\xB2\x98\xDC\x45\xEE\x22\xCB\xC8\x32\x72\x0F\xB9\x87\xDC\x47\xEE\x23\x8F\x93\xC7\x49\x35\xA9\x26\x4F\x92\x27\xC9\x4A\xB2\x92\xF0\x84\x27\x35\xA4\x86\x84\x48\x88\x3C\x43\x9E\x21\xCF\x91\xE7\xC8\x6A\xB2\x9A\x3C\x4F\x9E\x27\x6B\xC9\x5A\xB2\x9E\xAC\x27\x1B\xC8\x06\xF2\x32\x79\x99\x6C\x22\x9B\xC8\x16\xB2\x85\xD4\x93\x7A\xF2\x0A\x79\x85\xBC\x46\x5E\x23\xDB\xC9\x76\xF2\x26\x79\x93\xBC\x4D\xDE\x26\xEF\x92\x77\xC9\x7B\xE4\x3D\xB2\x8B\xEC\x22\x1F\x90\x0F\xC8\x6E\xB2\x9B\xEC\x25\x7B\xC9\x47\xE4\x23\xB2\x9F\xEC\x27\x07\xC9\x41\xF2\x29\xF9\x94\x34\x90\x06\x72\x84\x1C\x21\x9F\x93\xCF\xC9\x51\x72\x94\x1C\x23\xC7\xC8\xD7\xE4\x6B\x72\x82\x9C\x20\xDF\x90\x6F\xC8\x69\x72\x9A\x7C\x47\xBE\x23\x3F\x90\x1F\xC8\x4F\xE4\x27\xF2\x0B\xF9\x85\x9C\x21\x67\xC8\x39\x72\x8E\x84\x49\x98\x24\x19\xB5\xD2\x39\xC1\x0E\xC6\x2E\xC6\xDE\xC6\x81\xD2\x59\xC1\xE6\xE7\x04\x1D\xC6\x32\xA3\xD7\xB8\xC1\xB8\xD1\xF8\xA6\xF1\x1D\xE3\x47\xC6\x03\xC6\x63\xC6\x13\xC6\x5F\x8C\x67\x8D\xD1\xB3\x80\x6A\xB3\xC6\xAC\x33\xB7\xBA\xE0\x79\x40\xBB\xB9\xBB\x79\x80\x74\x1E\x70\x98\x79\x9E\x79\x9E\x74\x16\xB0\x90\x2B\xE4\x26\x71\xC5\x5C\x29\x57\xCA\xB9\x39\x37\x37\x85\x9B\xC2\x55\x70\x3E\xCE\xCF\xF9\xB9\x4A\xAE\x92\x9B\xC5\x55\x71\x73\xB9\xB9\xDC\x7C\x6E\x3E\x77\x1B\x77\x1B\x77\x07\x77\x07\xB7\x88\x5B\xC4\x2D\xE6\x16\x73\x77\x71\x77\x71\xCB\xB8\x65\xDC\x3D\xDC\x3D\xDC\x7D\xDC\x7D\xDC\x03\xDC\x03\xDC\x43\xDC\x43\xDC\x23\xDC\x23\xDC\x72\x6E\x39\xF7\x38\xF7\x38\x57\xCD\x55\x73\x4F\x72\x4F\x72\x2B\xB9\x95\x1C\xCF\xF1\x5C\x0D\x57\xC3\x85\xB8\x10\xF7\x0C\xF7\x0C\xF7\x1C\xF7\x1C\xB7\x9A\x5B\xCD\x3D\xCF\x3D\xCF\xAD\xE5\xD6\x72\xEB\xB9\xF5\xDC\x06\x6E\x03\xF7\x32\xF7\x32\xB7\x89\xDB\xC4\x6D\xE1\xB6\x70\xF5\x5C\x3D\xF7\x0A\xF7\x0A\xF7\x1A\xF7\x1A\xB7\x9D\xDB\xCE\xBD\xC9\xBD\xC9\xBD\xCD\xBD\xCD\xBD\xCB\xBD\xCB\xBD\xC7\xBD\xC7\xED\xE2\x76\x71\x1F\x70\x1F\x70\xBB\xB9\xDD\xDC\x5E\x6E\x2F\xF7\x11\xF7\x11\xB7\x9F\xDB\xCF\x1D\xE4\x0E\x72\x9F\x72\x9F\x72\x0D\x5C\x03\x77\x84\x3B\xC2\x1D\xE7\x8E\x73\x27\xB9\x93\xDC\x77\xDC\x77\x5C\x01\x5F\xC0\x0F\xE6\x07\xF3\x43\xF9\xA1\xFC\x70\x7E\x38\x7F\x09\x7F\x09\x3F\x92\x1F\xC9\x5F\xCE\x5F\xCE\x8F\xE6\x47\xF3\x63\xF9\xB1\xFC\x95\xFC\x95\xFC\xD5\xFC\xD5\xFC\xB5\xFC\xB5\xFC\x78\x7E\x3C\x7F\x1D\x7F\x1D\x7F\x03\x7F\x03\x3F\x91\x9F\xC8\x17\xF2\x85\xFC\x24\x7E\x12\x5F\xCC\x17\xF3\x25\x7C\x09\x5F\xC6\x97\xF1\x93\xF9\xC9\x7C\x39\x5F\xCE\x57\xF1\x55\xFC\x5C\x7E\x2E\x3F\x9F\x9F\xCF\xDF\xC6\xDF\xC6\xDF\xC1\xDF\xC1\x2F\xE2\x17\xF1\x8B\xF9\xC5\xFC\x5D\xFC\x5D\xFC\x32\x7E\x19\x7F\x0F\x7F\x0F\x7F\x1F\x7F\x1F\xFF\x00\xFF\x00\xFF\x10\xFF\x10\xFF\x08\xFF\x08\x7F\x91\x70\x91\xD0\x56\x68\x2B\xB4\x17\xDA\x0B\x1D\x84\x0E\x42\x47\xA1\xA3\xC0\x09\x9C\x60\x13\x6C\x42\x37\xA1\x9B\xD0\x43\xE8\x21\xF4\x14\x7A\x0A\xBD\x85\xDE\x42\xBE\x90\x2F\xF4\x13\xFA\x09\x03\x84\x01\x42\x81\x50\x20\x0C\x16\x06\x0B\x43\x85\xA1\xC2\x70\x61\xB8\x70\x89\x70\x89\x30\x52\x18\x29\x5C\x2E\x5C\x2E\x8C\x16\x46\x0B\x63\x85\xB1\xC2\x95\xC2\x95\xC2\xD5\xC2\xD5\xC2\xB5\xC2\xB5\xC2\x78\x61\xBC\x70\x9D\x70\x9D\x70\x83\x70\x83\x30\x51\x98\x28\x14\x0A\x85\xC2\x24\x61\x92\x50\x2C\x14\x0B\x25\x42\x89\x50\x26\x94\x09\x93\x85\xC9\x42\xB9\x50\x2E\x78\x04\x8F\x30\x53\x98\x29\xDC\x2C\xDC\x2C\xDC\x2A\xCC\x12\xAA\x84\x2A\x61\xAE\x30\x57\x98\x2F\xCC\x17\x6E\x13\x6E\x13\xEE\x14\xEE\x14\x96\x08\x4B\x84\x65\xC2\x32\xE1\x1E\xE1\x1E\xE1\x01\xE1\x01\xE1\x21\xE1\x21\xE1\x11\xE1\x11\x61\xB9\xB0\x5C\x78\x5C\x78\x5C\xA8\x16\xAA\x85\x27\x85\x27\x85\x95\xC2\x4A\x81\x17\x78\xA1\x46\xA8\x11\x42\xC2\x73\xC2\x6A\x61\xAD\xB0\x5E\x58\x2F\x6C\x10\x36\x08\x2F\x0B\x2F\x0B\x9B\x84\x4D\xC2\x16\x61\x8B\xF0\x8A\xF0\x8A\xF0\x9A\xF0\x86\xF0\x96\xF0\x96\xF0\xA3\xF0\xA3\x60\x8E\xE4\x37\xBB\x9F\x1A\xBD\xF7\x1E\x3D\x7F\x18\x3D\x77\x17\xBD\xC7\xDE\x68\x9B\x77\x37\x42\xF6\x6E\x84\x1E\xBB\x11\x46\xED\x46\x28\xDB\x8D\x30\x87\x9E\x05\xFA\xB3\xE7\x81\xD6\xED\x46\xD8\xB5\x1B\xE1\xAB\xDD\x08\xEC\x9E\xDF\x3F\x07\xD4\xFB\x4F\x9E\x03\x2A\xDF\x83\xB0\x60\x0F\x42\xF5\x1E\x84\x97\xF6\x20\xEC\xDE\x83\x70\x72\x0F\x42\xD2\x5E\x04\xF3\x5E\x84\x7E\x7B\x11\xAE\xD9\x8B\xE0\xDB\x8B\xB0\x30\xE6\xBC\xD0\x11\x7A\x4E\x28\x87\x9E\x0F\xBA\x96\x9E\x0B\x5A\x46\xCF\x03\xBD\x4D\x29\x7A\x3F\x3F\xBA\x9E\x11\x5D\x07\x18\x48\x09\x14\x5E\xC5\x3C\x76\x21\x7B\xB7\x72\xB3\x72\xAB\x72\xAB\x72\x9B\xF2\x15\xE5\x6B\xCA\xD7\x95\x6F\x28\xDF\x50\xEA\x12\xF4\x09\x97\x25\x38\xA4\x7B\x21\x7B\x13\x0E\x26\x1C\x4A\xF8\x3C\xE1\xCB\x84\x63\x09\x5F\x27\x9C\x4C\x48\x4B\x4C\x4B\xB4\x27\x8E\x4C\xE4\x13\x57\x27\x1E\x4F\xFC\x3E\xF1\x7C\xE2\xF9\xC4\x8C\xA4\xEC\x24\x43\x52\x5E\x52\xBB\x24\x4B\x92\x35\xA9\x73\xD2\x84\xA4\xEB\xE9\x5D\x7C\x85\x2A\x5F\xD5\x4F\x35\x40\x35\x46\x75\x85\x6A\x9C\x6A\x92\xAA\x48\x35\x47\x35\x4F\xB5\x49\xB5\x49\xB5\xA5\xF1\x8E\xF7\x0E\xE9\x8E\xF7\x11\xD5\x67\xAA\xB0\x2A\xAC\xC2\x64\x45\x8B\x77\x81\x7C\xC9\x53\x93\x2B\x93\x2B\x93\xE7\x24\xCF\x4B\xDE\x94\xBC\x29\x79\x4B\xCC\x1D\xA5\x23\xC9\x9F\x25\x9F\x49\x8E\x24\xA3\x5A\xA1\xEE\xA7\xEE\xA7\x1E\xA0\x1E\xAA\x1E\xA1\xBE\x44\x7D\x99\x7A\x94\xFA\x1A\xF5\x38\xF5\x24\x75\x91\x7A\xAE\x7A\xAE\x7A\x8B\xFA\x15\xF5\x76\xF5\x9B\xEA\xB7\xD5\x3B\xD4\xFB\xD4\xFB\xD4\x90\xC2\xA6\x0C\x48\x19\x9A\x32\x3C\xE5\x92\x94\x91\x29\xA3\x52\xAE\x49\xB9\x36\x65\x52\x4A\x51\xCA\x9C\x94\x79\x29\x9B\x52\x36\xA5\x6C\x89\xBB\x3F\x72\x24\xE5\xB3\x14\xD0\x28\x34\x7D\x35\xFD\x34\x03\x34\x43\x35\xC3\x35\x97\x68\x46\x6A\x46\x69\xC6\x69\xC6\x69\x26\x69\x8A\x34\x73\x34\xF3\x34\x6F\x6B\xDE\xD6\xEC\xD4\xEC\xD6\xEC\xD5\xEC\xD5\x7C\xA4\x39\xA4\xF9\x4E\xF3\xBD\xA6\x6B\x6A\xD7\xD4\x1E\xA9\xF9\xA9\x23\x52\x47\xA7\xAE\x49\x5D\x93\xBA\x36\x75\x93\x74\x1E\x28\x4B\x9B\xAD\xBD\x58\x7B\xB1\xB6\x97\xB6\x97\xB6\x8F\xB6\x8F\x76\x80\x76\xA0\x76\xAA\x76\xBE\xF6\x36\xED\xED\xDA\x25\xDA\x07\xB5\x0F\x6B\x37\x69\xB7\x6B\xB7\x6B\x3B\xE9\x06\xE8\xAE\xD5\x4D\xD0\x4D\xD4\x15\xEA\x26\xE9\x8A\x75\xA5\xBA\x0A\xDD\x54\x5D\x50\x37\x57\xB7\x44\xB7\x54\xB7\x54\xF7\x88\xEE\x31\xDD\x8D\xAD\x1D\xAD\x5B\x65\xA6\x67\x76\xCB\xEC\x91\x79\x65\xE6\x55\x99\xFE\xCC\x40\xE6\xDA\xCC\x0F\x33\x3F\xC9\xFC\x24\x53\x9D\xA5\xC9\x5A\x90\x75\x5B\x56\x6D\x56\x6D\x56\xF4\x8C\x4F\xA6\xDE\xA0\xBF\x46\x3F\x51\x5F\xA8\xBF\x45\x3F\x5B\x3F\x5B\xBF\x5A\x5F\xAF\x07\x03\x6B\xE8\x61\x18\x6A\x98\x62\x08\x18\xAA\x0C\x73\x0D\x4F\x18\x9E\x35\x1C\x35\x7C\x6F\x68\x9B\xD3\x2B\x47\xBE\x17\xD4\x74\x27\xE8\xE7\x98\x3B\x41\xBB\x4C\xA7\x4D\xA7\xB8\x6F\xB9\xD9\x9D\x67\x77\x6E\xE8\x1C\xE9\xDC\xD1\xDE\x45\xBA\x1B\xE4\xE5\xFD\x7C\x90\xBF\x95\x7F\x9C\x5F\xC1\x9F\xE6\xBF\xE5\x99\x1A\xA6\x26\xA1\x26\xA1\x46\x5D\xA3\xAE\xC9\xAB\xB1\x88\x22\xA0\xA6\xAA\x66\x4E\xCD\xDA\x9A\xD7\x6B\x1A\x6A\xBE\xAD\x89\xD4\x44\x6A\xDA\xD7\x76\xAE\x1D\x51\x7B\x55\x6D\x55\xED\xBC\xDA\x17\x6A\xEB\x6B\xBF\xA8\xFD\xA2\xD6\x1A\xEA\x19\x1A\x12\x1A\x12\xBA\x3C\x34\x2A\x74\x73\xE8\xD6\xD0\xEA\xD0\xEA\xD0\xF3\xA1\xB5\xA1\x17\x43\x1B\x42\x5B\x42\x5B\x43\xDB\x42\xDB\x42\xC7\x43\xA7\x42\x3F\x86\x7E\x0A\x7D\x56\xF7\x75\xDD\x89\xBA\x93\x75\xE4\x2C\x39\x2B\xDF\x4D\xEA\x22\xD9\x7C\x88\xBF\x9F\x34\x4D\x81\x70\x93\x02\x01\x5B\xB8\xB7\xA3\x8D\xBB\xA7\x23\xD2\x17\xC9\x08\x47\x93\x11\xDA\xD2\x7B\x5D\x75\xA9\x08\xCF\xA4\x22\x0C\xD7\x22\x8C\xD5\xA2\x64\x5F\x80\xD1\x21\xF4\xD6\x21\x0C\xA7\xB6\x05\xE6\xEA\x10\x56\xEB\x10\x5E\xD6\xA1\x74\x1F\x9E\x49\x43\xB0\xD0\x7B\xF1\xC3\xD2\x10\x86\xA7\x21\x04\x28\x55\xA5\x21\xCC\x4D\x43\x78\x21\x0D\xA1\x3E\x0D\xE1\xAD\x34\x84\x77\xD3\x10\xDE\x4B\x43\xD8\x99\x86\xC0\xB5\x42\xE8\x45\xEF\xFE\x88\x74\x88\xDE\xFF\x89\xDE\x7B\xEC\x4B\xEF\x32\x0E\xB8\xC0\x7D\xC6\xE8\x3D\xC6\x49\x17\xB8\xC7\x78\x71\x3A\xC2\xF0\x74\x84\x89\x94\x56\xA7\x23\xBC\x9A\x8E\xB0\xAA\x0D\xC2\xBA\x36\x08\x1B\xDA\x20\xD4\xB7\x41\x38\xD8\x06\xE1\x93\x36\x08\xF6\x0C\x84\x82\x0C\x84\xBA\x0C\x84\x17\x33\x9A\xEC\x78\x70\x59\x08\xF9\x59\x08\xF6\x16\xEE\xBB\x0E\xC8\x46\x28\xA0\x34\x38\x1B\x61\x48\x36\xC2\xFB\xD9\x08\xFB\xB2\x11\xF6\x67\x23\x34\x64\x23\x7C\x4E\x09\xF5\x08\xC9\x7A\x84\xEE\x7A\x84\x3E\x7A\x84\xBE\x7A\x84\x01\x7A\x84\x11\x94\xAE\xD0\x23\x5C\xAF\x47\x58\xA4\x47\x78\x58\x8F\xD0\x25\xC6\x0E\xC4\xC0\x1C\x84\x7B\x72\x10\x84\x1C\x84\xDA\x1C\x84\xF5\x39\x08\x5D\x73\x11\x7A\xE6\x22\xE4\x53\xEA\x97\x8B\xD0\x3F\x17\x61\x60\x2E\xC2\xB0\x5C\x84\x11\x94\x16\xB5\x70\x8F\xF1\xC1\x5C\x84\xEF\x09\xC2\x4F\x44\xB6\x67\xF1\xC3\x64\x04\xFB\x14\x84\x9E\x53\x10\xC6\x78\x65\xBA\xC2\x8B\x70\xBB\x17\x61\xA9\x17\xE1\x1E\x6F\xD3\x7D\xAE\xE8\x3D\xC4\xC7\x36\x21\x3C\xBE\x09\xC1\x45\xEF\xC3\x55\xBC\x8F\xE0\xFF\x9D\x7B\x71\x83\x3F\x40\x18\xFA\x01\x02\xEC\x47\xE8\xB9\x1F\xA1\xEF\x7E\x84\xF2\xFD\x08\x95\x94\xE6\x53\x7A\x74\x3F\xC2\x8A\xFD\x08\x4F\xEC\x47\x58\xB5\x1F\xE1\x4F\xDF\x9B\x3D\x8C\xD0\xF3\x30\xC2\x89\x23\x08\xA7\x8E\x20\xEC\x3A\x86\xB0\xF7\x18\xC2\xD0\xAF\x10\x46\x7E\xD5\x74\x2F\x6F\x2C\x5D\x47\x6D\x20\x08\xBF\x90\x98\x33\x7A\x8D\xFB\x4B\x51\x1C\x1C\xA3\xB9\x5A\x53\xA4\x29\xD1\xCC\xD6\x14\x24\x5D\x9F\x34\x31\x69\x62\x52\xAC\xB8\xAE\xA7\xE3\xE7\x78\x32\xC2\x4F\xC9\x08\x3E\xBA\xAE\x17\xDF\x4E\x8D\x4F\x0B\xEB\x7D\x17\x6A\x3F\xB1\xDD\xC6\xE6\x22\x4C\xC8\x6D\x7A\x5F\xA1\x52\xA8\xFA\xAA\xFA\xAA\x06\xA8\x0A\x54\x97\xAA\x2E\x57\x8D\x56\x8D\x51\xCD\x55\xCD\x53\x6D\x51\xD5\x53\xB9\xF3\xAE\x24\x77\x14\xC9\x8A\xE4\x01\xC9\x05\xC9\xF3\x92\xE7\x25\x6F\x49\xAE\x4F\x7E\x4B\x92\x29\xEF\x26\x33\x6A\x85\x7A\x80\x7A\x80\xBA\x40\x5D\x40\x65\xC9\xE5\xEA\x71\xEA\x71\xEA\x2D\xEA\xAD\xEA\x6D\x8D\x72\xE4\x5D\x49\x8E\x60\x8A\x22\x65\x50\xCA\xD0\x94\xB9\x29\xF3\x52\xB6\xA4\x6C\x49\xA9\x4F\x79\x2D\xE5\x0D\x49\x6E\xBC\x23\xC9\x0D\x46\xA3\xD0\x0C\xD0\x14\xD0\x7A\xBA\x5C\x92\x17\x73\x35\xF3\x34\x3B\x35\xEF\x37\xCA\x08\x11\xFB\x67\x6B\x67\x6B\x3B\xE9\x38\x5D\x57\x5D\x57\x5D\x6F\x5D\x6F\x5D\x5F\x5D\x3F\xDD\xB5\xBA\x71\x8D\x98\x3F\x4F\x37\x5F\x77\xBB\x6E\x31\xC5\xFC\xE5\xBA\xF5\x99\xEB\x33\xB7\x64\xBE\x9E\xF9\x46\xE6\x9B\x99\x79\xD9\xED\xB3\xAD\x12\xB6\x77\xCD\xEE\x9E\xDD\x3B\x3B\x5B\xAF\xD7\x5F\xA3\xBF\x46\x3F\x4E\x3F\x4E\xEF\xD4\x3B\xF5\x45\xFA\x62\x7D\x85\xDE\xAF\x67\x0D\xAC\xE1\x62\xC3\xC5\x86\xBE\x86\xBE\x86\xFE\x86\x41\x86\x21\x86\xA1\x86\xB9\x12\xD6\x3F\x61\xE0\x0D\x82\x21\x64\x08\x19\xBE\x34\x7C\x69\xF8\xDA\x70\xC2\xF0\x8D\xE1\x1B\xC3\x77\x12\xFE\x77\xCA\xE9\x91\x73\xB1\x64\x0B\xE3\xE7\x9C\x9F\x73\x6C\xF6\x2E\xF6\xBC\x9A\xB6\x35\x96\x1A\x0B\xC5\xEF\x57\x6B\xAE\xAC\xBD\xAA\x76\x5E\xED\xBC\xDA\xB5\xB5\xEB\x6A\x5F\xAE\xDD\x58\xBB\xA5\xB6\xBE\xB6\x4B\xC8\x1E\xEA\x1E\xEA\x11\x1A\x25\xE1\xB6\x88\xDC\xB7\x86\x8E\x87\x8E\x87\xBE\x91\xB0\xFA\xC7\xD0\x67\x75\x9F\xD7\x1D\xAD\xFB\xB2\xEE\x2B\x09\xB3\x4F\x34\xBB\xBF\x2C\xE2\x69\x14\x53\x19\x8A\xA5\x51\x3C\x7D\x5E\x27\xDB\x76\xDE\xA0\x43\xE0\xD2\x64\x8A\xE2\xE9\xDC\x18\x1C\x5D\x97\x86\xB0\x95\xE2\xE9\x7B\x94\x44\x1C\xB5\xB5\x42\xE8\xD6\x0A\xA1\x7B\x2B\x84\x8B\x29\x35\xB4\x42\x38\xD2\x0A\x81\xA1\x78\x2A\xE2\xE8\xC0\xD6\x08\x83\xFE\x24\x9E\x8A\xF8\xD9\x2B\x1D\xA1\x20\x1D\x61\x50\x3A\xC2\x30\x4A\x22\x8E\x3E\x9F\x8E\xF0\x12\xA5\x8D\xE9\x08\x5B\xD2\x11\xB6\x51\x12\xF1\x75\x0D\xC5\xD7\x8D\x6D\x10\xB6\x50\x12\xF1\xB5\x5B\x06\x42\xDF\x0C\x84\x7E\x19\x08\x03\x28\x3D\x43\xE9\xB9\x0C\x84\x55\x19\x08\xEB\x29\x99\x32\x11\xCC\x99\x08\xED\x29\x89\x38\x6C\xCB\x42\xE8\x4D\x29\x1E\x8F\xFB\x53\x8A\xC5\xE3\xC1\x14\x8F\x3F\xCC\x46\x38\x98\x8D\x70\x28\x06\x8F\xFB\x50\x1C\x1E\xA7\x47\x18\xAF\x47\x78\x90\x92\x88\xBB\x22\x0D\xA0\x54\x4B\x69\x0D\xA5\xB5\x94\xA2\xB8\x2A\xE2\xE9\x7D\xB9\x08\x0F\x50\xFA\x91\xE2\x6A\x14\x3F\xA3\x78\xE9\xA2\xF6\x42\xA2\x38\x79\x07\xBB\x90\xDD\xA2\xDF\xA2\xFF\xC4\xD4\x60\x3A\x62\xFA\xDC\xE4\xE3\xFD\xBC\xA8\xE3\x3F\xE2\x41\x68\x08\xFD\x14\xAA\xAF\x3B\x59\x77\xBA\xEE\xD7\xA6\x3E\x44\xD7\xC3\xDF\xA0\xB8\x56\x4D\xF7\xBF\x36\xA9\x10\x76\xAA\x10\x3E\x50\x21\x44\xE8\xBE\xD8\x0A\x55\xD3\xFA\xFD\x98\x18\x1C\x5F\xE1\x8D\xC1\x23\x49\x5F\xBE\xFD\x2F\xD6\x97\x19\x55\xBE\xA4\x31\xF7\x53\x0D\x52\x5D\xA2\x1A\xA5\x1A\x15\xA3\x37\xCF\x91\xF4\xE6\x6D\xAA\xD7\x54\x3B\x54\x3B\x62\xF4\x65\x46\xD2\x97\x07\x5D\x50\x5F\x9E\x2B\xE9\xCB\xDB\x24\x7D\xF9\xCD\xE4\x1D\xCD\xF4\x65\x94\xF4\xE5\x81\xEA\x81\xEA\x41\xEA\xA1\xEA\x51\x92\xBE\x7C\x6D\xA3\xBE\x5C\xAF\xAE\x57\xEF\x50\xEF\x50\x43\x8A\xA8\x29\x8B\xBA\x72\x41\x0B\xBA\xF2\x1C\x49\x57\xDE\x9A\xB2\x35\x65\x7B\xCA\xF6\x94\x77\x53\x76\x50\x3D\x19\x25\x3D\x79\x90\x66\xA8\x66\x94\x66\x14\xD5\x8F\xE7\x48\xFA\xF1\x07\x54\x3F\xFE\xB3\x7A\xF1\x54\xED\x2C\x6D\x55\x0B\x7A\x71\x67\x9D\x5D\xD7\x4D\xD7\x4B\xD7\x47\x97\xAF\xEB\xAF\x1B\xA0\x1B\x4F\x75\x64\x59\x37\x9E\xAB\x5B\xA0\xBB\x4D\xB7\x44\xB7\x44\xF7\xD8\x05\xF5\xE3\x75\x99\x2F\x66\x6E\xCE\xDC\x9E\xB9\x3D\xF3\xAD\x16\x75\xE5\x8B\xB2\x2D\xD9\x1D\xB2\xBB\x65\x77\xCB\xEE\x23\xE9\xCC\x59\x7A\x83\xDE\xA0\xBF\x56\x7F\xAD\x7E\xBC\xA4\x3B\x17\xEA\x27\xE9\x27\xE9\x5D\xFA\x72\x7D\xA0\x51\x8F\xDE\xAC\xDF\x2A\xE9\xD2\x0A\x43\x0F\x43\x0F\x43\x4F\x43\xBE\xA1\x9F\xA1\x9F\x61\xB0\x61\x30\xD5\xAB\xE7\x18\x9E\x34\x3C\x6D\xA8\x31\xD4\x1A\xEA\x24\xFD\xFA\xA8\xE1\x98\xE1\x2B\xC3\x49\xC3\x49\xC3\x29\xC3\xB7\x06\x2E\xA7\x7B\x4E\xCF\x46\x7D\xFB\xC2\xF7\xF0\x77\x99\x0E\x9A\x0E\x9B\x0E\x9B\xBE\xF8\x8D\xEE\xDD\x99\xEA\xDE\xDE\x0B\xEA\xDE\xED\x6A\xDA\x4B\xBA\xF7\x6B\xBF\xD1\xBB\xAF\xA8\xAD\xAA\x9D\x5B\xFB\x42\xED\x0B\xB5\xEB\x6B\x5F\xAA\xDD\x54\xBB\x99\xEA\xDF\xB6\x50\xD7\x50\xB7\xD0\xC5\x8D\x7A\xF8\xE5\xA1\x5B\x42\xB7\xB4\xA8\x87\x9F\x08\x9D\x0C\xFD\x14\xFA\x29\xF4\x45\xDD\x17\x75\xC7\xEA\x8E\xD5\x9D\x6C\x41\x17\xFF\x77\xE8\xE3\xD2\xF7\x0A\xE2\xF4\xF1\x39\x3A\x84\x17\x74\x08\xEB\x75\x08\x2F\xB5\xA0\x97\x77\x4A\x43\xE8\x4C\xF5\xF3\x58\xBD\x7C\x4E\x1A\xC2\xFA\x34\x84\x2D\x31\x7A\xF9\x4E\x4A\x5D\x5A\x21\x74\x6D\x85\xD0\x83\x52\xCF\x16\xF4\x74\x91\x3E\x8B\xD1\xD7\x31\x46\x5F\x2F\xA0\x14\xAF\x8F\xF7\x4E\x47\x18\x98\x8E\x30\x38\x1D\x61\x28\xD5\xCD\x63\xF5\xF3\x17\xD2\x11\x36\xA4\x23\xBC\x4C\x69\x6B\x3A\x42\x7D\x3A\xC2\x2B\x54\x6F\x7F\x9E\xEA\xED\x9B\xDA\x20\x6C\x6E\x83\xB0\x35\x4E\x7F\xEF\x9E\x81\x90\x9F\x81\xD0\x9F\xD2\xC0\x18\x7D\x5E\xA4\x67\x29\xAD\xCE\x40\x58\x47\x75\xFC\xA8\x9E\x6F\xCC\x44\xB8\x28\x13\xA1\x5D\x26\x82\x85\xEA\xFD\x5D\xB2\x10\x7A\x65\x21\xF4\xA1\xFA\xBF\xA8\xEF\xF7\xA3\x3A\xFF\x00\xAA\xEB\x8B\xB4\x3B\x46\xDF\x3F\x40\x75\xFE\x86\x38\x7D\xBF\xF7\x05\xF4\xFD\x6B\xF5\x08\x13\x62\xF4\xFE\x07\xF4\x08\x0F\x51\xFD\xDF\x1E\xA3\xFF\xF7\xA7\x73\x80\xD8\x79\x40\x28\x07\x61\x75\x0E\xC2\xF3\x39\x08\x2F\xE4\x20\xAC\xFB\x07\xE7\x05\xD1\xF9\xC0\xFD\x94\x1E\x8C\x99\x17\xFC\xF0\x3B\xF3\x82\xF8\xF9\x40\x74\x1E\xF0\x47\x76\x31\xFE\x5B\xF4\xFF\xC6\x87\x9E\x2B\x18\x9B\x81\x30\x21\x43\x3A\x47\x90\xC5\x66\xB1\x59\x09\x59\xA9\x59\xBA\xAC\xEC\x2C\xF9\x4E\x7D\x28\xAB\x30\x03\xA1\x3C\xA3\x69\xDF\xBC\x70\x32\x82\x7B\x32\x82\x67\x32\x82\x37\xC6\x46\xE0\x9F\x39\x67\x51\x2B\x8E\x17\xDD\x0A\x5D\x43\x6D\xB8\xB6\x30\x34\x3B\x14\xAB\x1E\xDA\xA1\x0F\x3C\x0F\x2F\xC0\x4B\xF0\x12\x6C\x82\x2D\x50\x98\xE0\x4E\x38\x9D\xF0\x6B\x42\x7D\xE2\x5B\x89\x25\x2A\xAF\xEA\xA8\xEA\x94\xEA\x07\xD5\xAF\xAA\x92\x64\x6F\xF2\xD1\xE4\x53\xC9\x25\x6A\xAF\xDA\xAF\x9E\xAE\x3E\xAA\xFE\x4E\x5D\x92\xE2\x4D\x99\x91\x32\x4B\xB2\x03\x32\x5E\x33\x51\x53\xA2\x99\xA1\x39\xAA\x39\xA5\x19\x9B\x3A\x41\xBA\xA3\x65\xD2\x76\xD7\x16\xE8\x2E\xD5\x2D\xD3\x3D\xAC\x73\xB7\xBE\xA5\xF5\x37\x19\xA7\x33\xA2\xB6\x38\xC4\xB2\x0E\xCF\x1E\x93\xBD\x2B\x7B\x7F\x76\x95\x7E\xA1\x7E\x99\xFE\x61\xFD\x58\xC3\x04\xC3\x6A\xC3\x4B\x86\x82\x9C\x4B\x73\xC6\xE6\x4C\xC8\xF1\x49\x76\xE1\x66\x9A\xAA\x4C\x0B\x4D\x63\xCD\x73\xCD\x0B\xCC\x0B\xCD\x85\x6D\x1F\x6D\xFB\x75\xDB\x48\xDB\x69\x96\x7B\x2D\xE7\xB9\xF3\x5C\x93\xFD\x9A\x7C\xFB\x3D\x5D\xEF\xEF\x6A\xEA\x66\xED\x76\x49\xB7\x31\xDD\xAE\xE8\xE6\xE8\x56\xD5\x6D\x61\xB7\x35\xDD\xB6\x76\x33\xF1\x56\xFE\x28\x7F\x8A\xB7\xD7\x5C\x5C\xB3\xAB\xE6\xE3\x1A\xA8\x55\x49\x36\x50\x4E\xD7\xFE\x5A\x3B\x36\x34\x21\x24\xEA\x3B\xBA\x48\xB6\xB4\x2E\x1A\xB5\xD3\x56\x80\x08\x33\x10\x61\x11\x22\xDC\x19\x63\x9F\xCD\xA4\x40\xC8\x53\x20\x0C\xA2\x34\x92\xD2\xFB\x0A\x84\x3D\x0A\x84\x6A\x16\xA1\x96\x45\x18\x4F\xED\xC3\xCC\x4C\x42\x98\x9D\x84\x20\x24\x21\xAC\x8A\xB1\x4F\x93\xA1\x42\x30\x50\x3D\x6A\xB3\x0A\xA1\x3E\x4E\x9F\x2A\xA0\x76\x4D\x6E\x4D\x46\x98\x95\x8C\xF0\x58\x32\xC2\x0A\x3A\xEF\x3B\x95\xDC\x64\xCF\x25\x6A\xE7\xA5\x56\x8D\xB0\x4A\x8D\x70\x26\x05\x21\x92\xD2\x64\x2F\x27\x6A\x9F\x33\x57\x8B\xD0\x5E\x8B\x70\x85\x16\xE1\x6A\x2D\xC2\xEB\xD4\x56\x60\xD4\x8E\xE3\x69\x1D\xC2\xAF\x3A\xD9\xCE\xE0\xC0\x34\x84\x5D\x69\x08\x1F\xA7\x21\x1C\x4E\x43\xF8\x21\xAD\xC9\x3E\xCA\xD8\x74\x84\x09\xE9\x08\xBB\xD2\x11\x3E\x4E\x6F\xEA\xC7\xF5\x19\x08\x6F\x65\xC8\xF6\x47\xFB\x66\x22\x34\x64\x21\x7C\x97\xD5\x64\x97\x2B\x6A\x4F\x32\x3A\xBF\x8C\xDA\x33\xAA\x37\x22\xEC\x36\x36\xD9\x1F\x8B\xF6\xF3\xB1\x53\x10\x26\x4C\x41\xB8\x7E\x0A\x82\x73\x0A\x42\x95\x07\xE1\x01\x4F\x93\x5D\xBB\xA8\x3D\xBB\xA8\xBD\xB8\x82\xC3\x08\x97\x1E\x46\x38\x7D\x04\xE1\xD7\x23\xB2\xBE\xBA\xF3\x58\x93\xDD\xD4\xA9\x27\x10\xEA\x4E\x20\x3C\x7B\x02\x61\xD5\x09\x84\x35\x31\xF6\xE5\x3A\x9F\x44\xE8\x12\xB5\x33\xF7\x2D\x42\xCA\xB7\x4D\xF6\xD3\xA2\xE7\xC2\xA2\xED\x17\x7B\x6E\x27\x8A\xBF\xA2\x5E\x1F\x3B\x9E\x5A\x1A\x43\xB1\xE3\xE7\xA8\xFA\x94\x34\x76\xC4\x71\x53\xA2\xF1\xC6\x8D\x19\x6B\xE3\x98\x11\xC7\x4A\xD3\x38\xF9\xF8\x82\xE3\x44\xEC\xDF\x7F\xD4\xAF\xA3\x4F\xB4\x7F\x46\xFB\x47\xC9\x1F\xB4\xFF\x3F\xD2\xEE\x7D\x68\xBB\x7F\x15\xD7\xEE\x13\x5A\x68\xF7\xF8\x76\xBE\x50\x3B\x46\xDB\xAF\xB1\x3D\xE8\xB9\x52\x11\x37\x3E\xCD\x3E\x94\x0D\xE9\x08\xD7\xA7\x23\xDC\x48\xCF\x79\xD5\x27\x9E\x49\x3C\x9F\x18\x89\x7E\x62\x16\x44\xCC\x29\x94\xB0\xE3\x0E\x73\x2C\x56\x88\x78\x00\xD4\xBE\x57\x10\x2F\x3C\x9E\xA3\xFD\x32\xFA\xC4\x62\xA5\x88\x8B\x22\x16\x9E\x56\x7F\xA7\x16\x71\x50\xC4\x40\x9F\x66\x86\xA6\xA3\xB6\xBB\x56\xC4\x3B\x11\xE7\xF6\x67\xEF\xCF\x6E\x8E\x63\x0E\xF3\xC2\x16\x70\xEC\x42\xD8\x25\x62\x56\x6C\xFB\x45\xF1\xA9\xF2\x0F\xF0\xE9\x3F\x8D\x3F\x7F\x06\x77\x5C\xBA\x26\x7C\xC9\xA7\xB8\xF1\x35\xC5\x8D\xEB\x28\x5E\x44\xF1\xE1\x8F\x70\x20\x3A\xDE\xFF\xD1\x71\x7E\x98\x9E\xC7\x6F\x3C\x37\x46\xCF\x85\xD9\xE9\xB9\xFC\x68\x7F\x1C\x7D\x18\x61\xCC\x61\x84\xAA\x8C\xE5\x19\x63\x73\x66\x37\x7E\x36\x2B\xDE\x4E\x45\xB4\xFC\x4D\xF3\x5D\xB9\x3D\x3A\x2B\x10\xBA\x28\xE4\x73\xA0\x62\x3D\x8E\xA5\x7A\x7A\xEC\x79\x57\xA9\xBF\xD1\x73\x9C\x85\xC9\x08\xB3\x93\x63\xCE\x7F\xD2\x07\xD5\x0A\xF5\x85\xCE\xCA\xF7\x8B\xDB\xAF\x91\xCF\xCD\x17\xA9\x4B\xD4\xD3\x1B\xBF\x4C\x1F\x6F\x67\xA4\x8A\x9E\xAF\x8D\xE2\x82\x19\xF2\xA0\x1D\x74\x04\x4E\xFA\xEE\x5D\x5F\x18\x28\xD9\x8B\xBD\x51\xFA\xBE\xDD\xCD\x70\x33\xDC\x0A\xB7\xC2\x13\xF0\x04\x08\x20\x40\x1D\xD4\xC1\x7A\x78\x11\x36\x4A\x56\x64\xB7\xC2\x2C\xC5\x2C\xC9\x56\xEC\x75\x4A\x87\x72\xA1\x72\x91\x72\x8B\x72\x8B\xB2\x5E\x59\xAF\x7C\x55\xF9\xAA\x72\xBB\x72\xBB\xF2\x7B\xE5\x0F\x4A\x55\x42\x72\x42\x4A\x82\x46\xB2\x23\x4B\x12\x8C\x09\x93\x13\x2A\x12\x76\x27\xEC\x4E\x80\x44\x4D\xE2\xCF\x89\xBF\x26\xDA\x93\x06\x48\x6B\x9E\xC5\x2A\x97\xCA\xA7\xF2\xA9\xCE\xAB\xCE\xAB\xA6\x25\x4F\x4B\x3E\x9D\x7C\x3A\x79\x7A\xCA\xF4\x94\xF9\x29\xF3\x53\x7E\xD0\xFC\xA0\x19\x93\x3A\x26\xF5\xBA\xD4\xEB\x53\x59\x6D\x2B\x6D\xBA\x36\x5D\x9B\xAF\xED\xAF\x5D\xA0\x5D\xA0\xDD\xA5\xDD\xAD\xFD\x58\xBB\x5F\x3B\x52\x37\x46\x77\x56\x77\x56\x57\xD8\xBA\xAC\x35\xA4\x43\xBA\x27\xC3\x93\xF1\x68\xC6\xF2\x8C\xA8\x6D\xDA\xDD\x99\x7B\x33\x3F\xCA\xDC\x9F\x09\x59\xC9\x59\x43\xB3\x87\x65\x13\xBD\x51\x5F\xAD\xE7\xF5\x35\xFA\x67\xF5\xD7\x19\x0A\x25\x1B\xCB\x7D\x73\x06\xE6\xCC\xCA\x99\x9D\x53\x9F\xF3\x46\xCE\x87\x39\x1F\xE6\xE8\x4C\x16\x93\xDD\x34\xC4\x34\xCC\x74\x85\xE9\x2A\xD3\x44\xD3\xAD\xA6\x59\xA6\x25\xA6\xBB\x4C\x49\x79\xEA\x3C\x6B\x5E\xC7\xBC\x32\x4B\xA5\xE5\x75\xCB\x76\xCB\x51\xCB\x29\xCB\x3C\xEB\x43\xD6\x8F\xAC\x07\xAC\xE7\xAC\xE7\xAD\xBF\x72\xE7\xB8\x30\x17\xE1\x7C\x9D\x7D\x9D\xC1\x66\xB3\xD9\x6D\x63\x6C\x57\xDA\xAE\xB4\xA1\x5D\x61\x4F\xB2\xA7\xD9\xD3\xED\x46\xBB\xF8\xF4\xB7\xF7\xA7\xB6\x6A\xCF\xDA\xC3\x2B\x23\x2B\x35\xBC\x96\x0F\xF0\x01\x7E\x16\x3F\x8B\xFF\x8E\xFF\x99\x0F\xD6\x4C\xAF\x79\xAF\x66\x67\xCD\x2F\x35\x67\x6A\xCE\xD5\x9C\xAB\xB1\xD5\x76\xA9\x75\xD4\x3A\x6A\xB7\xD5\xEE\x90\xEC\xB9\xDD\x10\x72\x84\x3E\x0D\x1D\x0A\x9D\x0E\x7D\x1B\x3A\x5E\x77\x3C\xCE\x6E\xAD\x33\x5C\x14\x2E\x0A\x97\x85\xCB\xC2\x93\xC3\x53\xC2\x18\x51\x44\x94\x91\xE4\x48\x4A\x44\x1B\xD1\x47\x0C\x11\xA3\x64\xCB\xB6\x7F\x64\x60\x64\x60\xE4\xFA\xC8\xF5\x91\x1B\x23\x37\x46\x1C\x11\x57\x24\x6A\xBF\x76\x85\x42\xA6\x5D\x94\xA2\xF6\x92\xC7\x51\x5B\x78\x46\x95\x4C\xD1\x7D\xD1\xA8\x3D\xB6\xD9\x94\x4E\xD3\xF5\xEA\xA8\xDD\xEB\x87\xD5\x08\xCB\xD5\x08\xCF\xA6\xCA\x74\x8D\x16\x61\x82\x16\x61\x04\xB5\x2F\xBD\x51\x87\xB0\x49\x87\xB0\x85\xDA\xA7\x2B\x48\x43\x18\x92\x86\x10\x4C\x43\xA8\x4C\x43\x78\x3D\x0D\xE1\xCD\x34\x84\x1D\x94\x0E\x50\xFA\x24\x0D\xE1\x50\x1A\x42\xEF\x56\x08\xFD\xA9\xEE\x2B\xD2\x65\xE9\x08\x63\xD2\x11\xAE\x8B\x91\x17\x22\x6D\xA7\xB4\xAD\x0D\xC2\xBE\x36\x08\x83\x32\x10\x86\x64\x20\x44\xF5\xEF\x7E\x99\x08\x03\x32\x65\x9B\xA2\x22\x0D\xA5\xF6\x46\x8F\x50\x1A\xA8\x47\x18\xAE\x47\x78\x44\x8F\xB0\x5C\x8F\xF0\xB8\x1E\xE1\x49\x3D\xC2\x20\x6A\x47\xDB\x97\x83\x30\x35\x07\xE1\x67\xBA\xEE\x1F\x31\xCA\x14\xB5\xB3\x1A\xD5\xE7\x7F\xA4\xD4\x6B\x0A\x42\xDF\x29\x08\x43\x29\x3D\x48\xD7\xE6\x8E\x78\x65\x5A\x41\xED\x95\xDE\xBE\x1F\xE1\xCE\xFD\x2D\xD8\x83\x9F\x82\x30\x6C\xCA\x6F\xF5\xA7\xF8\x73\xB6\x85\x54\x2E\x34\x9E\x1B\xA6\x8F\x38\xCE\x07\x9A\x0A\x4C\x57\x9B\xAE\x36\x75\x0F\xF7\x08\x8F\x0A\x8F\x09\x8B\x7D\x22\xEA\xDF\x19\x3A\xC3\x22\x3A\xB6\xC5\x71\x95\x90\x95\x90\xA5\x33\xB5\x31\x65\xDA\x32\x6D\x7A\x9B\xDE\x96\x6F\xEB\x6B\x2B\xB0\x15\xD8\x72\xEC\x39\x52\x9F\x16\xFB\x73\xD7\x70\xB7\xF0\xF8\xF0\x78\xA9\xFF\x69\x22\x9A\x48\xB4\xCD\x41\xFA\x2E\xA9\x15\x6E\x84\x1B\x25\x5C\xE9\xAB\xED\xAB\xED\xAF\xED\xAF\x5D\x9E\xB1\x3C\x63\xB8\x69\xB8\x69\x96\x69\x96\xE9\x2E\xD3\x5D\x26\x55\x9E\x2A\x4F\x9D\xA7\xCE\xEB\x98\xD7\x31\xCF\x6D\x71\x5B\xA6\x58\xA6\x58\x2A\x2C\x15\x16\xAF\xC5\x6B\x99\x6A\x99\x6A\x09\x58\x02\x96\x4A\x4B\xA5\x65\xBB\x65\xBB\xE5\x4B\xCB\x97\x96\xAF\x2C\x5F\x59\x8E\x5B\x8E\x5B\x4E\x5A\x4E\x5A\x4E\x59\x4E\x59\xE6\x5B\xE7\x5B\x6F\xB3\xDE\x66\xBD\xC3\x7A\x87\x75\x91\x75\x91\x75\xB1\x75\xB1\xF5\x2E\xEB\x5D\xD6\x65\xD6\x65\xD6\x7B\xAC\xF7\x58\xEF\xB3\xDE\x67\x7D\xC0\xFA\x80\xF5\x21\xEB\x43\xD6\x8F\xAD\x1F\x5B\x0F\x58\x0F\x58\xCF\x5B\xCF\x5B\xF3\x6C\x79\xB6\x76\xB6\x76\x36\x8B\xCD\x62\xB3\xDA\xAC\x36\x95\x5D\x65\x57\xDB\xD5\x76\x8D\x5D\x63\xD7\xDA\xB5\xF6\x34\x7B\x9A\xBD\x8D\xBD\x8D\x3D\xD3\x9E\x69\xCF\xB6\x67\xDB\x0D\x76\x83\x9D\xD8\x8D\xF6\x01\xE7\x07\x9C\xCF\x0A\x67\x85\x7B\x86\x7B\x86\x7B\x87\x7B\x87\xF3\xC3\xF9\xE1\x7E\xE1\x7E\xE1\x01\xE1\x01\xE1\x82\x70\x41\x78\x70\x78\x70\x78\x68\x78\x68\xF8\x92\xF0\x25\xE1\xEB\xC2\xD7\x85\x6F\x08\xDF\x10\x9E\x18\x9E\x18\x56\x45\x54\x91\xFE\x91\xFE\xD2\xD8\x2B\x8C\x14\x46\x8A\x22\x45\x11\x11\x5B\xB3\x4D\xD9\xA6\x5C\x53\xAE\x29\xDF\x94\x6F\x52\xD8\x14\x36\xA5\x4D\x69\x4B\xB6\x25\xDB\x34\x36\x8D\x2D\xD7\x96\x6B\x33\xDB\xCC\xB6\x82\x9A\xE9\xB2\x9E\x40\xE5\x7A\x74\x3C\x8A\x18\x9D\x65\xCA\x32\x19\x4C\x39\x26\xA3\xC9\x68\xEA\x63\xEA\x63\x62\x6C\x8C\x8D\xB5\xB1\x36\x95\x4D\x65\x4B\xB1\xA5\xD8\x72\x6C\x39\x36\x93\xCD\x64\x03\xE9\x7B\xA7\x22\xFE\x5B\xA0\xA3\x24\x03\x38\xB0\xC5\xC8\x81\x1B\xE0\x86\x46\xEC\x17\x71\xFF\xB7\x98\xBF\xF0\xDF\x86\xF7\x17\xC2\x7A\x11\xE7\x7F\x8B\xF1\xCA\xAC\xC4\x7F\x00\xE7\x33\x4C\x99\x26\x93\x84\xF5\xBD\x4D\x7D\x4D\x03\x4C\x83\x4C\x43\x4C\x23\x24\xCC\xBF\xCA\x74\x8D\x69\xA2\xA9\x39\x9E\xA3\x2D\xC1\x96\x64\x53\xDB\xD4\xB6\x54\x5B\x86\x2D\xCB\x96\x6D\x33\xD8\x0C\x36\x62\x33\xDA\x3A\x4A\x58\xDF\xC7\xD6\xCF\x36\xD0\x36\xC8\x36\xC8\x36\x24\x06\xF7\x45\xAC\xFF\x9F\xC2\x79\x5D\x38\x23\x9C\x2D\xE1\xBC\x3D\x3C\x2C\x3C\x3C\x7C\x69\xF8\xF2\xF0\x58\x09\xF3\xC7\x85\x1D\x61\x67\x1C\xE6\x27\x46\x92\x25\xDC\x4F\x89\xA4\xC6\x60\x7F\xBF\x48\xBF\x88\x33\xE2\x8C\x14\x8B\x98\x1F\xFB\xFC\xC5\xF8\xFF\x37\xEE\xFF\x6B\xB8\xDF\x01\x3A\xC0\xF5\x70\xBD\xA4\x9B\xE5\x6B\xF3\xB5\xFD\xB4\xFD\xB4\x8F\x66\x3C\x9A\xA1\x37\xE9\x4D\xC4\x44\x4C\xC3\x4C\xC3\x4C\xB7\x9A\x6E\x35\x2D\x31\x2D\x31\x25\xE5\x25\xE5\x25\xE7\x25\xE7\x59\xF3\xAC\x79\x65\x96\x32\xCB\x64\xCB\x64\x4B\xB9\xA5\xDC\xE2\xB1\x78\x2C\x3E\x8B\xCF\xE2\xB7\xF8\x2D\x41\x4B\xD0\xF2\xBA\xE5\x75\xCB\x51\xCB\x51\xCB\x31\xCB\x31\xCB\xD7\x96\xAF\x2D\x27\x2C\x27\x2C\xDF\x58\xBE\xB1\xCC\xB3\xCE\xB3\x2E\xB0\x2E\xB0\xDE\x6E\xBD\xDD\xBA\xD0\xBA\xD0\x7A\xA7\xF5\x4E\xEB\x12\xEB\x12\xEB\x52\xEB\x52\xEB\xDD\xD6\xBB\xAD\xF7\x5A\xEF\xB5\xDE\x6F\xBD\xDF\xFA\xA0\xF5\x41\xEB\x47\xD6\x8F\xAC\xFB\xAD\xFB\xAD\xE7\xAC\xE7\xAC\x17\xD9\x2E\xB2\xB5\xB5\xB5\xB5\xB5\xB7\xB5\xB7\x75\xB0\x75\xB0\x0D\xB6\x0D\xB6\x25\xD9\x93\xEC\xC9\xF6\x64\x7B\x8A\x3D\xC5\x9E\x6A\x4F\xB5\xEB\xEC\x3A\x7B\xBA\x3D\xDD\x9E\x61\xCF\xB0\x67\xD9\xB3\xEC\x7A\xBB\xDE\x9E\x6B\xCF\xB5\x0F\x3C\x3F\xF0\x7C\x66\x38\x33\x7C\x71\xF8\xE2\x70\xAF\x70\xAF\x70\x9F\x70\x9F\x70\xDF\x70\xDF\x70\xFF\x70\xFF\xF0\xC0\xF0\xC0\xF0\xA0\xF0\xA0\xF0\x90\xF0\x90\xF0\x88\xF0\x88\xF0\x84\xF0\x84\xF0\xF5\xE1\xEB\xC3\x37\x86\x6F\x0C\x27\x45\x92\x22\x7D\x23\x7D\x25\x5D\xC6\x11\x71\x44\x26\x45\x26\x49\xFD\x3A\x7A\xAF\x2A\xDA\xFF\xA2\xF3\x16\x7B\xED\x55\xD2\x18\xAB\xCE\xF8\x3A\xE3\x9B\x8C\x5F\x32\xA2\xDD\xBF\x2D\xB4\x85\x4E\xD0\x09\xFA\xC1\x00\x98\x08\x13\xA1\x10\x0A\xE1\x16\xB8\x05\x66\xC1\x2C\x78\x12\x78\xA8\x81\x5A\x78\x06\x9E\x81\xE7\x60\x0D\xAC\x85\xB5\xB0\x01\x36\x34\x7E\x4F\xE1\x15\xE6\x75\xE6\x03\xE9\x9B\x0A\xC7\x99\x93\xCC\x49\xE6\x14\x13\x61\x2A\x15\x95\x8A\xF9\x8A\x05\x8A\x9F\x14\x3F\x29\xE6\xB2\x73\xD9\x25\xCA\xA5\xCA\x84\x84\xA4\x04\x75\x82\x3A\x21\x35\x41\x9B\xF0\x69\xC2\xA7\x09\x5F\x25\x7C\x95\x70\x3E\x21\x9C\xF0\x53\xE2\x4F\x89\xE1\xC4\x48\xE2\x77\xAA\xEF\x55\x67\x54\x67\x55\xDF\x26\x7F\x9B\xEC\x53\xFB\xD4\xDF\xAB\xCF\xA8\x67\xA7\xCC\x4E\x19\xA3\x19\xA3\x99\xA9\x99\xA9\x19\x98\x3A\x30\x15\xB5\x0A\x6D\x6B\x6D\x6B\x6D\x1B\x6D\xA6\x56\xAF\x35\x6A\x7B\x68\x7B\x68\x7B\x6A\x7B\x6A\x7B\x6B\x7B\x6B\xB7\x68\x5F\xD7\xBE\xA1\x7D\x57\xFB\x9E\x76\xA7\x76\x8F\xF6\x23\xED\xE3\xBA\x15\xBA\x65\xAD\x1F\x6E\x5D\x91\x51\x91\x71\x20\xF3\x40\x66\x41\x76\x41\xF6\xA7\xD9\x91\x6C\xA7\x61\xB2\x21\x68\xB8\xC5\xB0\xD9\xB8\xD9\xB8\xD5\xB8\xCD\xB8\xC3\xB8\xD3\xF8\x89\xF1\x90\xF1\xA4\xF1\x94\xF1\xBC\x31\x6C\x1C\x6A\x1A\x6A\xBA\xD2\x74\xA5\xE9\x26\xD3\x2D\xA6\x45\xA6\xC5\xF4\x5B\x0D\x68\x56\x98\x13\xCC\x49\x66\x95\x39\xDD\x9C\x6E\xCE\x30\x67\x99\x89\x39\xCF\xDC\x4E\xFA\x86\x83\xC5\x6C\x35\x5B\x45\xF1\x64\xCE\x37\xF7\x35\x17\x98\x87\x9A\x47\x4A\xDF\x73\x18\x63\x5E\x24\x7D\xCF\x21\x31\x2F\x25\xAF\x43\x5E\xA7\x3C\xF9\xBB\x0E\xCB\xDB\x7E\xD5\x16\xDA\x95\x5A\xEE\xB3\xBC\x66\x79\xC3\xF2\x85\xE5\xB4\x65\xAE\xF5\x61\xEB\x3E\xEB\x41\xEB\x59\x6B\x38\xE6\x7B\x0F\x5F\x70\x5F\x73\x63\x6D\x57\xD8\x62\x6D\x59\x9F\xEE\x72\xB6\x0B\x6B\x67\xED\xAD\xEC\xAD\xED\x26\xBB\xC9\xDE\xD3\xDE\xCB\x3E\x80\x7E\xFF\x61\x59\xD7\xBB\xBB\x3E\x40\xBF\xFD\x00\xDD\x48\xB7\x8E\xDD\x46\x74\x1B\xDB\x6D\x6C\xB7\xC2\x6E\xB3\xBB\x2D\xEA\xB6\xBA\x5B\x7D\x37\xF9\x1B\x10\xCB\x56\x6E\x5F\x09\x42\x86\x60\x12\xCC\xC2\x42\x61\x91\xD0\xA1\xA6\x53\x4D\xCF\x9A\x3E\x35\xD3\x43\x33\x43\xD1\x6F\x41\x58\xCF\x5A\xCF\x3E\x7F\x76\xDB\xD9\x73\xE7\xCF\x9F\x9F\x14\x9E\x14\x2E\x0E\x97\x84\xDD\x61\x77\x98\x8D\xB0\x11\x75\x44\x1D\xC9\x89\x90\xC8\x80\xC8\x80\x48\x41\xA4\x20\x72\x43\xE4\x86\xC8\xC4\xC8\xC4\xC6\x6F\x42\x9C\x8B\x9C\x8F\x44\xBF\xFF\x30\x13\x11\x16\x22\xC2\x62\x6A\x5F\x3C\xFA\x3D\x88\xA8\x9D\xF1\xE8\x77\x21\xA6\x27\x21\xCC\x48\x42\x78\x33\x59\xA6\x81\xF4\xFB\x3E\x7B\x8C\x08\xDF\x1A\x11\xFA\x4D\x41\x18\x48\x75\x3E\x91\x96\x53\x9B\xF6\x2D\x7D\x37\xA2\x98\x7E\x37\xA2\xFC\x77\xBE\x1B\x01\x1F\x20\x0C\xA2\x36\xCE\x87\xC5\x7C\x3F\x62\xDB\x5E\x99\x0E\x50\x3A\x4B\xA9\xCD\x3E\x99\x2E\xA6\x34\x86\x92\x97\xD2\x42\x4A\xB5\x94\x5E\xA5\x04\x1F\x21\x44\x3E\x46\xE8\xB5\x1F\x21\x7F\x3F\x42\xC5\x7E\x84\xE0\x7E\x84\x69\xFB\x11\xE6\xED\x47\x58\xB0\x1F\xE1\xB6\xFD\xF2\x9C\x5B\xA4\xC8\x11\x99\x9E\x39\x21\xD3\x6A\x4A\xB6\x93\x32\x35\x7E\x77\xE7\x1F\xFC\x4E\x84\x26\xE6\x3B\x11\xFF\x97\xBE\x0F\x01\x49\x9D\x93\x44\xDD\xAB\xAA\xE6\xF5\x9A\xF7\x6A\x3E\xAE\x89\xC5\xBB\x27\xE1\x69\x58\xAA\x5C\xAA\x14\x31\x29\x8A\x3F\x67\x29\xFE\xFC\xAA\xFE\x55\x2D\xE2\x8D\x88\x1B\xE2\xF8\xEF\x5D\xD3\xBB\xE6\xDC\xF9\x73\xE7\xA3\x63\xA0\x21\x72\x38\xF2\x45\xE4\x68\xA3\x0E\xF1\x89\x11\xA1\xC1\xD8\xD4\x7E\xD1\x76\xAB\xA2\xF2\x7A\x2C\x8B\x30\x3B\xE6\x7B\x10\x55\x31\xE7\xE4\x3E\x69\x23\xF7\x93\x3B\x5B\xDA\x9F\xA1\xCF\x1C\xCD\xBC\x16\xEF\xE0\xC7\x9F\x1F\x3D\xAA\x39\xA5\xF9\x4E\xF3\x83\x26\xBA\xDE\x14\x7D\xA2\x98\x5E\x03\x35\xB0\x0A\x56\x35\x62\xF8\xEF\x61\xF7\x85\xF0\x51\xC4\x9A\x58\xFC\x88\x62\x86\x88\x17\x51\x4C\xF8\x2C\xF2\x59\xE4\xEC\xF7\x08\x91\xEF\xE5\x3C\x44\x65\x4B\x54\xA6\x88\xB2\x64\x0D\xAC\x69\x94\x21\xA2\x9C\x10\xE5\x43\x3C\xF6\x66\x99\xB3\x24\x3C\xBD\xCC\x7C\x99\x79\x99\xF9\x3E\xF3\x23\xE6\x47\xCD\xD5\xE6\x6A\xF3\x53\xE6\xA7\xCC\xBC\x99\x37\x3F\x67\x7E\xCE\xFC\x9E\x79\xA7\xF9\x03\xF3\x07\xE6\xDD\xE6\xDD\xE6\x1F\xCC\x91\x8B\x4C\x79\xE6\xBC\x5B\xF2\x6E\xC9\x7B\x34\xEF\x85\xBC\x83\x79\x87\xF3\x5E\x6C\xF7\x62\xBB\x6D\xED\xB6\xB5\xFB\xA5\x5D\xA4\x9D\xB7\xBD\xB7\x7D\x7D\x23\xEE\x46\x2C\x60\x8D\xC5\xDE\x48\x47\x7B\xA7\xA1\x9D\x46\x74\xBA\xBC\x93\x88\x7F\x51\xDC\x8B\xE2\x5D\x14\xE7\x8E\x44\x8E\x44\xBE\x8A\x9C\x68\xA6\x3F\xFE\x4F\xE3\x46\xFC\xF7\xB5\x78\xE0\xA1\x16\x6A\xE1\x39\x78\x0E\x56\xC3\x6A\x90\x65\xEA\x5D\xCA\x96\x64\xE9\x19\xD5\x19\x95\x2C\x3F\x7F\x51\x9F\x89\x93\xA1\xFF\xAC\xFC\x8C\x97\x9D\xBF\x95\x87\x99\x17\x94\x87\x23\xCD\x23\xCD\x97\xC7\xC8\xC3\xFB\xCD\x0F\x9B\x97\x9B\x57\x98\x9F\x30\x3F\x69\x5E\x69\x7E\xDA\x2C\x98\x9F\x35\xAF\x32\xEF\x30\xEF\x32\xBF\x6F\xFE\xD0\xFC\xA1\x79\x8F\xF9\x7B\x33\xE4\x89\x32\xD3\x98\x77\x91\x24\x37\x6F\xCE\xBB\x35\xEF\x91\xBC\xB5\x79\x07\xF2\x8E\x34\x93\xA1\xEB\xDB\x6D\x68\x57\xDF\xEE\x95\x76\x3F\xB7\x83\xF6\x9E\xF6\xBE\xF6\xA2\x4C\xDD\x6A\x91\xBE\xD1\xD5\xA9\x4B\xA7\x61\x9D\x86\x77\x1A\xD5\xE9\x9F\x93\xA7\xFF\x88\x2C\x95\xE5\x67\xAF\x9A\x3E\x54\x86\x9E\x3F\x7F\xFE\xFC\xE7\x91\xCF\x23\xC7\x22\xC7\x22\x27\xFF\x02\x79\x78\xD0\x88\x70\xF8\xDF\x2C\x17\xFF\x51\x39\x16\x95\x5F\x51\xB9\xF5\xEF\x90\x57\x67\x44\xAC\xF9\xE1\x7F\x87\xDC\xB2\xD3\xF9\x62\xF4\x89\xDA\x31\x88\xEE\x5F\x8D\xA5\xF3\xB0\x2A\xC3\x56\x83\x38\x9F\x87\x1A\xAE\xD9\xFE\x09\x24\x6A\x12\xB5\x89\x23\x13\x47\x25\x8E\x49\x2C\x4C\x9A\x9C\x24\xDB\x3A\x49\x6F\xB4\x71\xE2\xCF\x0C\x64\x46\x6D\x98\x88\xC1\x4D\x8D\x67\xFB\xE5\xBD\xB7\x6A\xFD\xB3\xFA\x2A\xC9\x56\xC7\x21\x69\x1F\xBD\x8A\xDA\x05\x88\xEE\x83\xCD\xFD\x93\xF6\x69\x62\xEF\x50\xEC\x52\xEF\x92\xCE\xBF\x1E\x55\x9F\x51\xD7\xD3\xFE\x1B\x9D\x2F\x36\x50\x9B\x41\xB1\xF7\xE0\x23\x8B\x62\xEE\xC1\xFF\xC6\xC6\x4D\xF3\x7B\x18\xB2\xBD\x9B\xA2\x94\x92\x14\x6F\xCA\xF4\x94\xD9\x29\x55\x89\x6B\x12\x31\x35\x3F\xB5\x20\xF5\x7A\xF9\x50\x02\xB5\x39\x33\x95\x9E\x07\xDB\x22\xE1\x9C\xBC\xAE\x22\xDB\xAE\xF7\x75\x9E\xDD\x79\x76\xE7\x46\xF9\x4C\xED\x26\x44\xF7\x01\xA3\x76\x3B\xA2\xEB\x03\xF0\x74\x27\x3E\xB6\xBA\xEB\xE9\x3E\x5E\x54\xFE\x57\xD3\xFD\x9D\x08\xA5\xA8\x3D\x8B\x3A\xBA\x26\xD8\x68\xD7\xA2\x7A\xF1\x4A\x11\x43\x4C\x60\x82\x6A\xA8\x86\xAA\x8C\xAA\x0C\x30\x25\x9B\x3A\x98\xAC\xA6\x2E\xA6\x2E\x26\x87\xC9\x61\x02\x3B\xD8\x81\x9E\x6B\xE9\x20\xF9\x58\x4D\xF1\xE1\x9B\x85\x05\x80\x49\x49\x08\xB1\x16\x52\xAB\x46\x22\x28\x00\x60\x64\x1C\xBF\x9A\xF2\xC7\x33\xCD\xF9\x1B\x46\xCA\xEE\x05\xA9\xF2\xDF\xFA\x91\x28\x7D\x7B\xDD\x74\x19\x4A\xDF\x3C\xEF\x92\x28\xF3\x0B\x2E\x47\x60\x63\xDE\xFB\x38\xA1\x79\x3C\x63\x2F\x47\x60\x00\xA4\xB9\xA7\xC8\x2F\xA4\xEE\xE8\x93\x4F\xF3\x53\x76\xB9\xFC\xB7\x9C\xFE\xBD\x29\x39\x2E\xFF\xF4\x3D\x93\x4A\xE6\x2F\x8B\x8B\x67\x03\xE5\x57\x5F\x8E\xB1\x5F\x15\x94\xF6\xFB\x62\xE3\x69\xA0\xF9\xFD\x49\x49\xF9\xA3\xE4\xF2\x47\x9F\x35\x71\xE1\x75\xD4\xBF\x53\x8A\xCC\xEF\x30\x2A\xAE\x9E\x68\xB9\xB8\x51\xCD\xF3\xA3\xD7\xC8\xFC\x1E\x71\xE1\x17\xA5\xD1\x7A\x1B\x85\xD2\xF7\x85\xAB\x46\xA1\xF4\x4D\xE3\xD3\xB4\x1E\xDE\x8C\x8B\xA7\x43\x62\xF3\xF7\x77\xD1\xF8\x7A\xC7\xF1\x1B\xE8\x7B\x9B\x69\x7B\x9D\x8E\x8B\x67\x34\xCD\x27\x8C\x46\xD0\x88\xE9\x8F\x46\xE9\xDB\xCB\xF9\xDA\xB8\x7A\x1E\x2D\xBF\xB7\x90\xE6\x67\xD9\xE8\xE6\xED\xBB\x81\xD6\x43\xF5\xE8\xE6\xF5\x76\x59\x5C\x7E\x9E\x19\x2D\xBB\x73\x68\x7F\x58\x4B\xDD\xF5\x34\xFE\x02\xCA\xDF\x35\x1A\xE1\x7A\xD1\x3D\x16\xE1\x4A\x00\xE8\x48\xF3\xBF\xEB\xCA\xE6\xF1\x2F\xA6\xFD\xF3\xE0\x95\xCD\xD3\x29\x88\xEB\xCF\x0D\x57\xCA\xF9\xDD\x86\xCD\xF9\x70\x15\x42\x02\x00\x38\x68\xFC\xF6\xAB\x10\x62\x2F\x1C\xE8\xE2\xFA\xFF\xD8\xAB\x64\xF7\xDD\xB4\x3F\x14\x5E\xD5\xBC\x1E\xEE\x83\xE6\xE1\x7D\x34\x3E\x43\x5C\x7D\xAE\xA6\xF1\xEC\x8A\x1B\x17\xF5\x57\x21\x28\x01\x60\x48\x5C\xBA\xA7\xAF\x92\xEB\xE7\xFB\xB8\xF0\x70\xB5\xEC\x5E\x9C\xDA\x9C\x9F\x44\xF9\xC3\x28\x5F\x77\xB5\xFC\xED\xDD\xE8\x33\x9D\xB6\xCB\xB2\xAB\x11\x92\x63\xF8\xF9\x34\xFC\x86\xAB\xE5\xF4\xEA\xAF\x96\xF3\x33\x2B\x2E\xFE\x63\x34\xFE\x4B\xE3\xCA\x75\xFA\x6A\x94\xBE\xB7\xAF\xA7\xF5\xE3\xBB\x06\xC1\x2C\x7A\x5C\x2B\xB7\x9B\x9B\xF2\x4D\xD7\x36\x1F\x8F\x49\x71\xE3\xBA\xF0\x5A\xB9\x5D\x4E\x44\xC7\xF5\xB5\xF2\xB8\x88\x3E\x47\x69\xFE\x77\x5D\x8B\x20\x6D\xEE\x8E\x93\xEB\xF9\x79\xDA\xEE\x05\xE3\x9A\xC7\xFF\x75\x5C\xFE\xAB\xC6\xC9\xE5\xFB\x8C\x8E\xF7\x65\xE3\x9A\xFB\x9F\x8D\xAB\xE7\xEA\x71\x72\xFE\x17\xD0\xF6\xAD\x1F\x27\x8F\xCF\xE8\xD3\x91\xD6\xC3\xE7\x34\xDE\xD3\x34\xFC\x7D\x71\xE9\xEA\xC6\xCB\xFE\x65\x14\x07\x4C\xE3\x11\xD2\xC5\x7A\x1A\x2F\xE7\xBF\x4F\x1C\xCE\xAC\xA6\xE1\x0B\xE3\xC6\x51\x3D\xE5\x6F\xA3\xE3\x77\xD7\xF8\xE6\xE3\x62\x7F\x3C\xBE\x45\xE3\xA1\xF9\x3C\x3D\x5E\x1E\xEF\xD1\xC7\x4C\xF9\xF6\x09\xF2\xF8\x1F\x3B\x41\x0E\xDF\x93\xB6\x4B\xE1\x84\xE6\xB8\x71\x38\x2E\x7E\xDF\x04\x39\xFF\xCB\xE3\xC6\xD7\x6A\xFA\xDE\x0E\x96\xD6\xDB\x84\xE6\xE3\xE5\xED\xB8\x78\x76\xD1\xF0\xB9\x71\xFD\xA1\x81\xF2\x17\x50\x9C\x3C\x3D\x01\xE1\x16\x04\x68\x70\x20\x7C\x07\x00\x29\xB4\xBD\x7C\xA5\xB4\x3F\xD0\xE7\x21\xCA\xAF\x2E\x6D\xDE\xFF\xAF\xA0\xE5\x35\x95\x35\xAF\xB7\x67\x69\xFB\xDA\xCB\xE4\x7E\x1F\x7D\xEE\x4F\x8B\xC3\x01\xFA\xDE\xE2\x38\xBE\xAF\x4C\xAE\xBF\xA5\x34\x9E\x9A\x32\xF9\xEF\x33\xF4\x6F\x32\x2D\x6F\x7D\x19\x42\x11\x03\x30\xD6\x2F\x7F\xFB\xFA\x6E\xDA\x6F\x0B\x03\x08\xBD\xC4\xFE\x1B\x94\xFB\x7F\x76\xB4\x5C\x41\x84\x8B\xC4\x1F\x33\x11\xDE\x10\xF1\x9F\xE2\x43\xC1\xEC\xE6\xE3\x77\x2B\x4D\xD7\x37\x1B\x61\x02\x03\xD0\x70\x07\x42\x3B\x00\xB8\x37\xAE\xFF\x24\x2D\x92\xDD\x55\x71\x7C\xDD\x22\xB9\x9E\x6D\xB4\x5C\xA6\x45\x08\x1B\x45\xFE\x12\x84\x52\xB1\x3C\x71\xED\x75\xC9\x32\xD9\xBD\x5A\x19\x57\x3F\x94\x5F\x16\x2D\xD7\x32\xB9\x5E\xA2\xCF\xD3\x74\x5C\xDC\xB2\xAC\x79\xBF\x3A\x10\x97\x9F\x65\xD4\xFF\xCD\xF8\xF1\xB8\x4C\xAE\xFF\x33\x71\xF9\xA9\xA7\xE9\xBC\x1C\x1D\x77\x77\x23\x54\x88\xFC\x7B\x10\x06\x03\xC0\x0F\x54\x4E\xED\xBA\xAF\x79\xFB\x3E\x1E\x17\xFF\xE7\xF7\xC9\x6E\x0D\x8D\xFF\xC4\x7D\xCD\xFB\xC9\x98\xB8\x74\xE1\x7E\x39\x9F\x6E\x75\x5C\x7D\xDE\x2F\xBF\xE7\x89\xC3\xF3\x0E\xF7\xCB\xEE\xC4\xB8\x74\xED\x34\xFC\xD7\xB4\x1E\xC6\xDE\x8F\xF0\x81\x58\xDE\x87\x10\x2A\x01\xE0\xC7\xB8\x74\xC7\x2E\x97\xD3\xBD\x96\x86\x2F\x5C\x2E\xF7\x9B\x19\xCB\x65\xF7\xA1\xB8\x76\xA9\x5A\x2E\xC7\x5F\x1D\x17\x4F\xF5\x72\xB9\x3E\x36\xD0\xF0\x6F\x2E\x6F\x5E\x5E\x8E\xF6\x87\x86\xE5\x08\x43\xC4\xF1\xF7\x98\x5C\x9F\xEF\xC7\xB5\x17\x54\xD3\x71\x1A\xAF\x2F\x55\xCB\xF1\x7D\x43\x71\xCF\x5E\x2D\xF7\xCB\x5D\xD5\x08\x59\x00\xB0\x84\xD6\xDB\xB2\x27\x9A\xE3\x83\x22\x4E\xBE\x54\x3F\x21\xC7\x7F\x9C\xE6\x73\xF5\x13\xCD\xFB\x4F\x62\x74\x7C\x3D\x21\xF7\x83\x86\x27\xE4\x72\x65\xD0\x7A\xD6\x3D\x89\x90\x2B\xA6\xF3\x24\x42\x47\x31\xBE\x38\x79\xDD\xF0\x94\x1C\xDF\xF3\x71\xF8\x73\xFA\x29\x39\x5F\x7E\xDA\x9F\x75\x2B\x9B\xFB\xBF\x4F\xD3\x4D\x8F\xE3\xEB\xE3\xEA\xC7\xB4\x52\xAE\x87\xB2\x38\xB9\x50\xB0\x52\x4E\xF7\xC5\xB8\xF0\x63\x69\x7C\x57\xC7\x85\x1F\x4F\xF9\xA9\x71\xE1\x0B\x69\x3C\xB7\xC7\xF1\x7D\x34\xFC\xD3\x71\xFC\x19\x94\x1F\x88\x97\x8F\x94\xFF\x39\xE5\x2F\x5B\xD9\x5C\x9E\x3E\x9F\x12\x87\xDB\x34\xDD\x83\x71\xF1\x34\x50\xFE\xF7\x3A\x8A\xDB\x2B\x11\xE6\x8B\x1E\x02\xC2\xC5\x62\x7E\xE2\xEA\x79\x75\x8D\xDC\x5E\xE6\xB8\xF8\x3F\xAA\x91\xDD\x2F\xC4\xB7\x17\xE5\xBF\x18\xD5\x6F\x6B\xE4\xFE\x1F\x7D\x8E\xD3\xFE\xA6\xAB\x45\x48\x13\xEB\xB1\x56\x76\x7F\x4B\xE3\x2F\xAC\x45\xB8\x24\x26\xFC\x65\xB4\x1F\x56\x85\x9A\xE3\x6A\x19\xED\x87\x35\xA1\xE6\xE9\x3F\x12\x57\x9F\xAB\x43\x72\x79\xBB\xD3\x72\xD5\x87\x9A\xCB\xA3\xD6\xB4\x1F\x9E\x0E\x21\xCC\x11\x19\xCF\x22\xF4\x10\xF5\x6A\x8A\x0F\xD5\xCF\x21\x5C\x16\x13\xDE\x1A\x2F\x1F\x57\xC9\xFD\x47\x49\xF3\x79\x6C\x55\x9C\x3E\x1F\xAD\x87\x38\xFE\xB7\xB4\xFE\x61\x35\xC2\x7D\x62\xBF\x7A\x1E\xE1\x0A\x51\x8E\xD3\xFC\x7C\xB4\xB6\x79\x78\x1D\x2D\x6F\x43\x1C\x7F\x6D\x34\xFE\xB5\x08\xAD\xC4\x7E\xBB\x8E\xCE\x13\xE3\xC6\xE9\xF8\x75\xB2\x3B\x21\xBE\x7F\x52\xFE\x8C\x38\xFD\xDC\xB7\x4E\x4E\xE7\xAD\xF8\xF9\x1D\x8D\x7F\x10\x8D\xBF\x7A\x1D\x42\x8A\x58\x0F\x34\x9E\xF6\xB4\x7E\x1A\xD6\xC9\xE3\xFA\xEA\xF5\xB2\xFB\xAE\xA8\xDC\x59\x2F\xF7\xB3\x82\x17\xE5\x76\xD0\x45\xF5\x81\x17\x11\x7C\xA2\xFF\x4B\x08\xC3\x01\xE0\xCE\x78\xB9\xB3\x51\x76\xFF\x12\x97\x9F\xEA\x8D\x72\x3E\x23\x71\xB8\xBA\x9A\x86\x7F\x9B\xB6\xCB\x86\x8D\xCD\xEB\x6D\x46\x14\x97\x36\x36\x7F\xEF\xD9\xE8\xBC\x66\x63\x73\xDC\x7B\x8C\xA6\xDB\x10\x17\xCF\x19\x1A\xFE\xD8\xC6\xE6\xF8\xFC\x33\xED\xCF\x3F\xC4\x85\xAF\x8A\xD6\xF3\xA6\xE6\xF2\x2E\x8B\xF6\xB7\xF4\x4D\xCD\xC3\xFB\xE3\xFA\x9B\x89\xFA\x7F\x42\xE3\xB1\x6F\x92\xF5\x70\x1F\xE5\x8F\x8F\xD3\xF7\xAA\x36\xC9\xE5\xE0\xE9\x38\xAD\xDE\x24\xB7\x57\xF4\xE9\x4E\xDB\x71\xD7\xA6\xE6\xEF\x59\x68\x79\x77\xC7\xE5\xE7\x38\x6D\xAF\x86\x4D\xCD\xF5\x87\x81\x34\xFC\xB9\x4D\xCD\xEB\x21\x29\xAE\x1F\xEA\x36\xCB\xF1\x75\x8D\xC3\x25\x13\xE5\x73\x34\x1E\xFB\xE6\xE6\xFE\xA7\xE2\xDA\xB7\x80\x86\xBF\x31\x5E\xFE\x52\x7E\x46\xFC\xFC\x65\xB3\x9C\x2F\x3B\xE5\xCF\xD8\xDC\xBC\x5C\xEF\xD1\xF1\x58\xB5\xB9\xF9\x7C\x73\x77\x9C\xFE\x50\x4D\xF3\xC5\xD2\x74\x6B\xE2\xF2\x79\x90\xE6\xF3\x99\x38\xFE\x6D\xB4\x7D\x57\x6F\x6E\x5E\x6F\xF5\xE2\x7F\x83\x1A\x56\x24\xAA\xEB\x15\x52\xBC\x70\xE1\x47\xF4\x4F\xFA\x03\x7F\xDD\x1F\xF8\x17\xFC\x8E\xBF\xF8\x54\xFD\x8E\x5F\xBE\x1A\xA1\xF7\x07\x52\x39\x66\x21\xC0\x0C\x04\x28\x17\xFB\xB0\x51\xD4\xCB\x00\xE0\x64\x24\x32\xBD\xA7\x9C\xCE\x4F\x00\xBA\xBE\x28\x59\x6B\x94\xCB\x94\x44\xF3\x66\xA2\x79\xA8\x8A\x66\xE8\x2F\x0C\x20\x7A\x8A\x32\xA3\x58\x1C\x9F\x0A\x00\x9D\x05\xA0\xFA\x36\x84\x82\xC1\x5A\xA8\x7A\xE0\x21\x80\xC3\x3F\x2A\xE1\xED\x47\xFA\x8A\x11\x68\xE9\xCB\xA2\x4C\x6A\x45\x23\x46\xA9\x5C\x72\xBB\x2D\x7E\x9F\xCA\xDD\xF7\xA3\xED\x1A\x91\xBE\xD5\x18\x89\xE8\x40\x2C\xB3\x38\x47\x80\x07\xDF\x63\x58\x8F\xB3\xC2\x85\x5F\xEE\x60\xB6\x28\x21\xC5\xEB\x73\x3A\x26\x55\xBA\xCB\x83\x6E\x8F\x1D\x63\x5D\x5D\x99\x58\x57\x37\x45\xAC\xAB\x3B\x1B\xEB\xEA\xA1\x4C\x16\x5D\xCE\x49\x5E\x7F\xD0\x91\xA0\x91\x7E\x97\x96\x3A\x8A\xBC\x95\x9E\x60\xA2\x3A\xEA\x0C\x54\x56\x24\x69\xA3\x0E\x9F\xDF\x5B\x5C\x59\x14\x54\x35\xFA\x56\x38\x67\x24\x37\x39\xDC\x1E\x75\x4A\xE3\x7B\x5E\x7F\x30\xA5\xD1\xCB\x59\x5E\xAE\x69\x72\x78\x66\xA6\xA6\xD2\x4C\x38\x2A\x5C\x15\x93\x5C\x7E\xAD\xB6\xB9\xBB\xBB\x4E\xCE\x8F\xDF\x1D\x2C\x73\x38\x27\x05\xD2\xB4\x4D\x4E\xBF\xB7\xD2\x53\xDC\x2A\xB5\x89\x51\xE4\x72\x97\xB7\x8E\x09\x50\x52\xEE\xF5\xFA\xD3\x63\x02\xF8\xCA\x2B\x03\x6D\x62\x02\x54\xB8\x3D\x95\x81\x8C\x56\x31\x8C\xCA\xF2\xA0\xDB\x57\x3E\x33\x53\xD7\xC4\x2B\x76\x4F\x73\x17\xBB\xB2\x62\xF2\xE1\x77\x55\x64\xD3\x00\x7E\xE7\x4C\x47\x91\xD7\x53\xE4\x0C\xEA\xB5\x4D\x9C\x40\xB9\xBB\xC8\x65\x48\x6B\x62\xF8\x5D\xD3\x5C\xFE\x80\x2B\x47\x2A\xFA\x24\x77\x30\xE0\xF0\xFA\x73\x53\x1A\x1D\x4E\x4F\x31\xD1\x36\xBA\x3C\xAE\x52\x67\xD0\x65\x6C\xF2\x9E\xE1\xF5\x9B\x5A\x35\xBA\x02\x65\xEE\x92\x60\xB9\xAB\x24\x68\x6E\xDD\x9C\xE7\x77\x97\x96\x05\x2F\x92\x52\x2D\x72\x17\xFB\xC5\x7C\x05\x9D\x6E\x4F\x20\x2F\xD9\xE7\xF4\x07\x5C\x12\xB3\x6D\x92\xFC\xDB\xED\x6B\xD7\xBA\x31\xA0\xDB\x13\x14\x73\x57\x14\x0C\xB4\x97\x7A\x42\x51\x85\xCF\xE1\x9A\x6A\x51\x47\x7F\x7B\x5C\x53\x3B\x34\x7A\x94\x06\xAD\xEA\xA6\xDF\xAE\x8E\x8D\x1E\xE5\xC1\x4E\xEA\xA6\xDF\x2E\x4E\xAA\x21\xD7\x34\x67\xB9\xA3\x28\x38\xC3\xE1\x71\x4D\xEF\x9C\xD1\x8C\x13\x70\x05\x1D\x6E\x8F\xAF\x32\x68\x4B\x8A\xF2\xBB\xB0\xE2\xFF\xF6\x36\xBF\x09\x57\xEC\x0C\x3A\xBB\x1A\x7E\xC3\x76\x79\x82\xFE\x99\x3E\xAF\xDB\x13\xEC\x96\xD9\xCC\xB3\xD4\x15\x74\xF8\x5D\x81\xCA\xF2\x60\x77\x83\xC3\x51\xE2\xF5\x17\xB9\x1C\xEE\x0A\x9F\xD8\xC1\x63\x3A\x7E\xA0\x87\xD4\xA8\x41\xAF\xC3\x53\x29\x76\xB6\x8B\xE5\x0A\x75\x06\x5C\x3D\x7B\x38\xDC\x01\xC7\x34\x67\xB9\xBB\xB8\x67\x1A\x65\x94\xBA\x3C\x8E\x62\x57\x91\xB7\xD8\xD5\x2B\x2D\x26\x9C\xCC\xEA\x1D\xCB\x72\x79\x44\x56\x9F\xD8\x17\x65\x56\x7E\x9B\x98\x50\x95\xFE\x72\xFA\x72\xDF\x78\xB6\x1C\xBA\x9F\xD4\xE6\x93\x03\x5E\x8F\xA3\xD2\x53\xE1\xF4\x07\xCA\x9C\xE5\xFD\x75\x8D\x3C\xCA\x19\x20\xA5\xEC\xAF\xF4\x04\xDD\x15\x2E\x87\xCB\xEF\xF7\xFA\x07\x66\x46\x07\x51\xA9\xDF\xE9\x13\xFB\xAB\xB3\xA8\xCC\x39\xA9\xDC\x55\x90\xD5\xF8\x76\xB9\x6B\x86\xC8\x2F\xA6\x45\x1F\xD4\x82\x4F\x20\xE8\x77\x7B\x4A\x07\xA7\xFD\xC6\x67\x48\x9B\x46\x96\xDC\x9B\xE4\x90\x43\xD3\xE3\xD8\x41\xEF\x14\x97\x67\x58\xAB\xF8\xC0\xAE\xE0\xF0\xF8\x08\xBC\x93\x26\xBB\x8A\x82\x23\x52\x9B\xB3\x2F\x91\xC6\xC4\x34\x67\x79\xA5\x4B\x66\x5C\x9A\xD3\x18\x60\xBA\xDF\x1D\x74\xF9\x1D\xAE\x0A\x77\xD0\x31\xC9\xEB\x2D\x77\x39\x3D\x23\x5B\xF6\x15\x7B\x77\xA9\xCB\x7F\x99\xA1\x45\x5F\xB9\xF8\x97\xB7\xEC\x29\x97\x6B\xD4\x45\x2D\x7A\xCA\x23\xDB\x55\xEE\xAA\x70\x79\x82\xA3\xF5\x2D\x86\x91\xF2\x3E\xC6\xD8\xA2\x5F\x91\xB7\xBC\xDC\x55\x14\x74\x7B\x3D\x63\x4D\x2D\xA7\x2E\xF6\x70\x39\xFA\x2B\x2E\x1C\xA2\x5C\x74\x3A\xCB\xAF\xCC\x6B\x31\x84\x5C\xB1\xD1\x68\xAE\x6A\x13\x1F\x48\xFA\x73\xB5\xA6\x91\x5D\x5C\x59\xE1\xBB\x26\xB5\xA9\xDA\x45\xF7\xB5\x52\x33\x54\x38\xCB\xCB\xBD\x45\x0E\xB7\xC7\x1D\x1C\x27\x75\xC2\x32\x97\xD3\xE7\xF0\x05\xFD\xE2\x60\x1B\xDF\x9C\x13\x70\x05\x27\x24\x37\xBD\x74\x9D\x34\xC0\x4B\xFC\x2E\xD7\xF5\x12\x42\xF8\x5D\x12\xFB\x86\x36\x31\x63\xD1\x51\xE4\x2C\x2A\x73\x89\x71\xDD\xD8\x02\x3B\xE0\x0A\x4E\x94\x92\xA8\x70\x55\x78\xDD\x37\xB9\xA4\x6C\x38\x9A\x71\x7C\x95\x81\xB2\x42\x6D\x33\x8E\xD7\xE7\x6C\xD5\xFC\xA5\x80\xCB\x1F\x9C\xD4\x2C\x50\xA9\x2B\x58\x24\x21\x6D\x85\xAF\x58\x8A\xB5\x58\x2B\xFE\x2A\x76\x95\x38\x2B\xCB\x83\x22\x98\xB8\xD4\x22\xA3\xC2\x39\x43\x74\x94\xA8\xA3\x61\x8B\x5D\xE5\xA5\x52\x06\xE4\x1E\x24\xA2\xC8\xA4\x92\x32\x89\x33\xA9\xA4\x09\x53\xDC\x59\xF1\x1C\x87\xC7\x2B\xD5\xC5\xE4\xB4\x26\x9F\x49\x25\x12\x8E\x4F\x89\xB2\x24\x50\x97\xF8\xE5\xEC\x54\xE7\xA4\x40\x45\xD2\x54\x67\x71\xB1\xC3\xEB\x71\x79\x58\xF1\x97\x37\x69\x6A\xA0\x72\x92\xE8\xF6\xB1\x53\x2B\x2A\xCB\xA7\xB2\x53\x9D\x9E\x62\x7F\x92\xF8\xBF\xC3\xE3\x0D\x06\x14\x53\xBD\xFE\x20\x3B\x75\x86\xD7\x5F\xC9\x4E\xF5\xB8\x4A\xA7\xA5\x35\x65\x35\xE0\xF0\x3B\x3D\xA5\xAE\xE9\x69\x0E\x87\xCF\x19\x2C\x93\x12\x92\xFB\xC9\x8C\x14\x07\x1D\x90\xA2\xC7\x4C\x9D\x43\x02\x52\x3A\x20\x83\x65\x81\x9B\x34\x0E\xD9\xE5\x77\x55\x78\xA7\xB9\x6E\x8E\x3A\x4B\xDC\xE5\x41\x97\xFF\x96\x36\xD1\x46\xA3\xBD\x4E\x66\xDF\xDA\x2A\x8E\x5D\xEA\x0A\xCE\x8A\x0F\x2A\x47\x38\x3B\x3D\x8E\x5D\xE9\x71\x7B\x3D\x55\x98\xE8\x70\x54\xB8\xFC\xA5\xAE\x39\xD8\x3A\x1A\x20\x26\x1B\x73\xE3\xB8\x72\xB2\xF3\x64\x6D\x48\x92\x1E\xEE\x92\x92\xF9\x98\x1E\x75\x36\x8A\x3B\xB7\xD7\xB3\x00\xDB\x50\x76\xA0\x19\xFF\x36\xD4\x44\x83\x4B\x79\xB8\x1D\x53\x1B\xC3\x49\x8C\x3B\x50\xEA\xE0\x81\xA0\xBF\xDC\xE5\x59\x88\x6A\xEA\xF0\x14\x55\xF8\x16\xC9\x2E\x77\xA0\xD8\x5D\xEA\x0E\xDE\x19\x75\x05\x7C\xCE\x22\xD7\x62\x54\xC9\xAE\x32\xD7\x8C\x25\x28\x8D\x0B\x77\xD0\xEB\xBC\x4B\x8E\xCD\x19\xF4\xBA\x7B\xF6\x58\xDA\xE8\x28\xE9\xD9\x63\x19\xB6\xA2\x51\xBB\x3D\xA5\x01\xAA\x6B\xDC\x4D\x0B\xD3\xC4\x94\x04\xFD\x3D\xCD\xD9\x2E\x4F\x71\x60\xBA\x3B\x58\x76\x2F\x66\xC4\xB2\x4B\xBC\xFE\x0A\xA7\x54\x0B\xF7\x61\xEB\x58\x0F\xB7\xA7\xD8\x35\xC3\x5B\x72\x7F\x73\xAE\xDF\xE5\x2B\x77\x16\xB9\x1E\xA0\x15\xD5\x9C\xEB\xF0\x3C\x18\x1F\x5A\x52\x74\x1E\xC2\xB4\x58\x6E\xC0\x57\xEE\x0E\x3E\xDC\x3C\x1B\x81\xA0\xD3\x1F\x94\xF2\xF7\x48\xF3\xA8\x03\x95\x93\xE4\x9F\x8F\xA2\x2E\x96\x1F\xF4\xBB\x2B\x96\x37\x0F\x2A\xB2\x1C\xA2\x2A\xF4\x58\xF3\xB8\x25\xBE\xA4\x0E\x3D\x8E\x99\xBF\xF1\xF0\xF9\x5D\x25\xEE\x19\x2B\x5A\xF0\x09\x54\x96\x94\xB8\x67\x54\xB7\x10\x99\xD4\x78\x4F\x60\x72\x93\xE3\xC9\xE6\x45\x2C\xF7\x4E\x77\xF9\x9F\x6A\xCE\xAB\xF4\xF9\x5C\xFE\x95\x72\x05\x05\x67\xFA\x5C\x01\x51\xA9\x90\xC7\xE0\xD3\x71\x5C\xF9\x15\x5E\x6E\xC0\x46\x2E\x15\x6C\x82\xDC\x07\x1A\xD9\x92\xE4\xA9\x91\x6B\xA7\x29\x06\x57\xB0\x36\x2E\x52\x79\x10\x85\xE4\x4C\xC5\x64\xA0\xBC\xBC\x4E\xEE\xCE\x32\x4F\x9C\x4E\x3C\x23\x57\x6C\xA5\xC7\x2D\x6A\x1E\x8E\x40\xA5\xDF\xEF\x15\xB5\xD0\x67\xE5\x5A\x8A\xF2\x65\x7D\x45\x74\x06\x9F\xC3\x9C\x16\x7C\x1A\x5F\x5C\xD5\xF2\x8B\xC1\x92\xDE\xAB\x9B\xFB\xC8\xBA\x8E\xE4\xB3\xA6\x79\x26\xCA\x9D\x81\xA0\xC4\x7F\x5E\xAE\x96\x28\x5F\x2C\xAC\xD8\x02\x2F\x34\x67\x07\xBD\x72\x23\xAC\xFD\x0D\x5B\x6A\x87\x75\x18\x23\xD6\xC4\x82\xAF\x8F\x65\x94\x39\x03\x65\x2F\xCA\x15\x25\x33\x8A\xBC\x15\x3E\xA7\xDF\xB5\x41\x0E\xD4\x84\x5C\x2F\xD1\xA1\xEE\x92\x5C\x2F\xCB\x2D\x13\x15\x01\xFE\x99\xE2\xB0\xDA\x28\x63\x87\x1C\x4F\xA9\x2B\xB8\x09\xB3\x7E\x13\x2F\x7D\x65\x73\x4B\x5E\x72\x5F\xD8\x22\x57\x53\x73\x2F\xA9\xE5\xB7\xB6\xF4\x92\x9C\xC3\x7A\xB9\x06\xE3\xE2\x73\x05\xB7\xA1\x36\x26\x9F\x62\x61\x5F\x89\x2D\xBD\xA8\x0A\xBC\x2A\xF7\x27\x99\x51\xEE\xF2\x94\x06\xCB\x5E\x93\xDF\xA2\x85\x9F\xE2\x9A\x19\x78\x3D\xF6\x2D\x51\x88\x6D\xC7\x18\x1D\x4D\xC2\xE8\x37\xE4\x5A\xA4\x2F\xC9\x22\xF7\x4D\x4C\x73\x38\x62\xEB\xD1\xEF\x9D\xFE\x96\x5C\x91\xB4\x8B\xBF\x2D\x47\x4C\x33\xE8\x77\x95\xBC\xD3\x8C\xE1\xF6\x04\xDF\x6D\x04\x5C\xB7\xA7\x74\x07\xE6\x36\x25\x2B\x6A\xC1\xE5\xDE\xE9\x8E\x22\xAF\x6F\x26\x4D\xE1\x3D\xD4\x5F\xC0\x3F\xE0\x0A\xEE\x6C\x6A\x42\x67\x71\xF1\x2E\x4C\x95\xF3\x26\x35\xA9\xDF\x3B\xFD\x7D\x79\xF8\xFF\xF6\xDD\x0F\xE4\x1E\x4F\x7B\x90\xDF\xE9\x09\xB8\x83\xEE\x69\x2E\x47\x51\xB9\x37\x50\xE9\x77\x7D\x88\x44\x8E\xE8\x42\x01\x76\x63\xCC\xA4\xD1\xE9\xF3\xB9\x3C\xC5\x7B\x64\x11\x20\x0E\xC9\xBD\xCD\x9A\x28\xE0\xBE\xC9\xB5\x4F\x6E\xE7\xA6\x3A\x71\x48\x6A\x93\x33\xE8\x2A\xFE\x48\x1E\xE9\x8D\xB5\x23\xAB\xB9\x1F\xC7\xA2\xA3\x23\xE8\xF2\x57\xB8\x3D\x62\xE8\xFD\xB1\x72\xA1\x29\x92\x03\xB2\x2C\x92\xF2\x73\x10\x5B\x35\xE5\x6D\xBA\x34\x99\x76\xFA\x3E\x91\xAB\x5C\xAE\xD2\x4F\x31\x91\xD6\xD2\x21\x8A\xCB\xAE\x60\x63\xC8\x86\xD8\x2E\x19\x2D\x82\x2C\x6A\x0F\x63\x6A\x53\xCC\x01\xAF\x3F\x78\x44\x4E\x4A\x0E\x2B\x6A\x35\xA2\x6A\xF1\x59\x6C\xC7\x95\xC5\xBA\xC4\xFF\x5C\xCE\x42\xB9\xD7\x3B\xA5\xD2\xF7\x85\x9C\x70\x85\xD3\xE7\x13\x0B\x22\x16\xFB\xA8\x1C\x59\x94\x25\x87\xFB\x12\x4B\x3D\xE2\x6C\x2B\xDF\xEF\xEA\xD6\x21\x10\x2C\xCE\xCF\x77\x38\xBA\xE6\xE7\x4F\x72\x06\xDC\x45\xB4\x16\xFA\x15\x95\x39\xFD\x9C\xA9\xC9\x53\x74\x8B\x6D\xE6\x0E\x06\x24\xBF\x01\xB1\x9E\xB4\xCA\xBC\x7E\xD9\xCB\x34\xC0\x54\xE4\xF5\x04\x82\xED\xAD\xC7\xB0\xD8\x15\x28\x72\xFA\x5C\x7F\x6D\x32\x5F\x61\x03\x53\x5A\xEE\x9D\x24\x77\xAA\x72\x67\xD0\xD5\x41\x0C\x21\x7B\x77\xE4\x4C\x95\x9E\x80\xBB\xD4\xE3\x2A\x36\x95\x7B\x3D\xA5\xB1\x31\x4E\x73\x15\x89\xD1\xFD\x25\x99\x6B\xD9\xF3\xAF\xA9\x87\xA6\xBA\x88\x0D\xF7\x3F\x98\x42\x47\xEB\xD7\xD8\xA1\xDC\x35\xC3\xE5\xCF\xCF\x97\xFE\xFC\x4E\x05\x5B\x8F\xA3\x96\x06\x9D\x25\x87\xB5\x9E\xC0\xD6\x94\xE3\x71\xCD\x08\x76\x90\x26\xBE\x1D\xAD\x27\x1B\xB9\x25\xAE\x60\x51\x99\x08\xB3\x15\x1D\xAC\xDF\x60\x7A\x33\xAE\xA4\x8C\x77\xB0\x9E\xC2\xDC\x66\xEC\xA0\x18\x91\xDB\x13\xA4\x79\xB0\x9E\x46\x8D\xDC\xA9\x67\x89\x7F\x3A\x58\xBF\xC5\xD6\xB2\x5B\x86\xD8\x0E\xA2\xA3\xA3\xF5\x3B\xD4\x4B\x1D\x45\xD2\xE2\xE5\x02\x75\xE4\x4C\x92\x5F\x47\xEB\xF7\x68\x90\xF8\x7E\x47\x85\xD3\xED\xE9\x10\x08\x3A\x83\xAE\x8E\x9C\x49\x0E\x65\xFD\x01\x73\xA8\xAF\x9C\xA3\x38\xEF\x1F\xE5\x71\x2C\xC5\x5E\xE1\x0C\x16\x95\xFD\x84\x47\x12\x9B\x6A\xD4\x21\x49\x15\x49\x92\xC4\xF5\x03\xEA\x23\xE6\xE1\xB7\xDC\x26\x71\xDC\x4F\x9E\xE9\x4D\x71\xCD\xFC\x8B\x9A\x58\xEC\xB1\xD3\xBC\xEE\xE2\x8E\x03\x3A\x0E\x30\xFD\x26\x23\x41\xE7\xA4\xF2\xFF\x7C\xFE\x62\xD3\xAF\xF4\x78\xFD\xC5\x2E\xBF\x4B\x9C\x7B\xFA\xA4\xDC\xB8\xFC\x2D\x67\xE2\x3F\x9D\x57\x31\xF5\xA6\x24\x9B\x79\xB9\xA6\x56\x3A\xCB\x1D\x41\x6F\x33\xEF\xA0\xBF\xD2\xF5\xBB\x85\x95\xDE\xFA\xEF\x2C\x6B\x8B\x05\xFA\x9D\xAA\xF8\x4D\x59\x5B\x82\xCA\xFF\x44\xA1\xC4\x7F\xF9\xF9\x25\x6E\x4F\x71\x4C\x7E\x3B\x34\xFE\x8C\x8A\x9E\x9F\x31\x51\xE2\x75\xB0\xFE\x82\xDB\x99\xA6\x18\x7D\x4E\xF7\x05\x7A\xE3\xFF\x68\x2E\xE3\x13\xFA\xAB\xF0\xBF\x3D\x67\x2A\x71\x96\x07\x5C\x31\x35\xF0\x97\x25\x65\xFD\x15\x7F\x52\xC5\xD5\xE4\xFF\xAF\x40\xCA\x99\x44\x05\xFE\x6F\x3C\xFD\x1B\x4F\xFF\xC6\x53\x11\xAD\x1C\x0E\x57\x85\xBC\x32\x56\xE9\x71\x4F\xAD\x94\x92\x74\x38\xFD\xA5\x81\x96\x73\xF0\x6F\x42\x51\xD3\x6F\x81\xFD\x3F\x90\x8B\xF6\xED\xAD\x67\xF0\xA6\xFF\x80\x08\x99\x25\xA6\xD4\xC1\x7A\x16\x97\x32\xFF\x87\x00\x21\x3F\xDF\xEB\x93\x65\x48\x07\xEB\x6F\xE5\xBA\xFC\xF7\x1C\x5E\xF5\xBB\x51\xFD\x6E\x1C\x9C\xE9\x02\xB1\x9E\xC7\x15\x09\x7F\x0B\x84\xFF\x74\xFB\xFF\x2D\x10\xFE\x3B\x04\x82\xDF\x25\xA6\xDD\xA1\xF9\x12\x42\x58\x5E\xB7\xF2\xBB\x4A\x5D\x33\x1A\x8F\x1A\x44\xE4\x25\x40\x99\x29\xCD\xB1\xAB\x98\xF4\x22\x6F\x85\xCF\x5D\xDE\x6C\xC1\xC7\x3A\x87\xD1\xF9\x5D\x95\x01\x57\x07\xBF\xAB\x5B\x7E\xFE\x95\xC3\xBA\x75\xB4\xCE\x65\x5E\x49\xF9\x9F\x54\x22\xFF\xA2\x25\xA3\xC6\x0C\xFF\x8B\x93\xF1\x7F\x43\xF6\xFE\x10\x2A\xFE\xA2\x3C\xFC\xD7\x14\x5B\x1A\x6A\xFF\xCE\x11\xFF\x97\xA5\xF5\x27\xF1\xF0\x7F\x7B\x83\xFE\xF5\x15\xFD\xEF\xEA\x3C\xFF\x24\xE8\xFF\xE5\xB5\x1D\xB3\xA8\xF2\x17\x95\xFD\xAF\xDD\x68\x98\xC7\x7C\x98\x1A\xA7\xA0\xFF\x7F\x2A\x54\xFE\x85\x85\x89\xBF\x65\xCB\xDF\xB2\xE5\x6F\xD9\xF2\xB7\x6C\xF9\xAF\x94\x2D\x17\x5E\x60\xFA\xCB\x7B\x58\x9C\x38\xF8\xCB\x8A\xF9\x97\x8A\xB8\x7F\x7F\x81\xDA\xB7\xB7\xCE\x67\x2E\x6A\x9A\xD8\x89\xDA\x81\xC3\x59\x5E\x1E\x3D\x8F\x11\xA8\x9C\x24\xCD\xF5\x16\x30\x5F\xAA\xFE\x96\x56\xFF\x95\xE0\xF6\xB7\xB4\xFA\x5F\xD6\xA0\x7F\x4B\xAB\x7F\x97\xB4\x6A\x71\xF5\xEB\x36\x26\xD1\x1D\x70\x96\xFB\xCA\x9C\xB7\x8B\xBF\xA4\xF3\x99\x77\x88\xBF\xA4\x73\x9D\x0B\x19\x65\xB9\xB7\xB4\xAB\x7D\x11\xA3\x0A\x78\x7C\x7E\xB7\x27\x58\xE2\xB8\x93\x49\x76\x4C\x8B\xBA\x16\x33\x6A\x87\xB7\x32\xE8\x98\x54\x59\x52\xE2\xF2\x2F\x61\x54\x92\xCB\x53\x59\x5E\x7E\x17\x93\xE2\xF0\x04\xBD\x4E\x7A\xC8\x79\x29\xA3\x74\x94\x04\xBD\xCE\x65\x8C\xD2\xE1\x0A\x7A\x9D\x77\x33\x49\x52\x50\xBF\x6B\xDA\x3D\x4C\x62\x89\x1C\xDB\xBD\x4C\x42\x89\x74\x0D\xE2\x3E\x46\x59\xE2\xAB\x0C\x16\xDD\xCF\x28\xA5\xBB\x70\x0F\x30\xF2\xF9\x38\xF1\xF7\x83\x4C\x82\x7C\x97\xE1\x21\x86\x2D\xF1\xBB\x5C\x0F\x33\x09\x45\x92\xFB\x11\x26\x91\x5E\x67\x78\x94\x49\x08\x04\xFD\x41\x6F\xF9\x72\x26\xA1\xC2\x55\x51\x54\xE6\x7F\x4C\xFE\x51\xE1\x7B\x5C\xFE\xE1\x9B\xB9\x82\x49\xAC\x70\x55\x54\x78\xA7\xB9\xAA\x25\x56\xC0\x15\x7C\x42\x7A\xAD\xDC\xE5\x79\x92\x49\x91\x2E\x16\x4C\x72\x06\x5C\xCE\xE2\xE2\xA7\x98\xD4\x58\x67\xD0\xBB\x92\xD1\x48\x8C\x40\x99\xD7\x1F\x74\x16\x17\x3F\x4D\xDD\x62\x00\xB7\xA7\xC8\xCF\xC7\xBC\x1F\xA8\x9C\x24\x30\xBA\x58\x67\x89\xDF\x5B\x51\x13\x1B\x43\x45\x65\x79\x6D\xCC\x1B\x15\x95\xE5\xA1\x58\xEF\x62\xF7\xB4\x3A\x46\xDB\xE8\x5D\xEC\x9E\x56\xE1\x2D\x7E\x26\x86\x23\x5F\x43\x7B\x36\x9E\xE3\x7F\x8E\x72\xA2\x89\x38\x8A\x56\x31\xAD\x62\xAE\x54\x14\x79\x3D\x41\xD7\x8C\xE0\x6A\x26\x95\x5E\xAB\xA0\x8C\x35\x8C\xA6\xC4\x13\x74\x14\x79\x3D\xD3\xBC\xE5\x95\x41\xD7\xF3\x4C\x9A\xE8\x76\x56\x06\xBD\x8D\xBC\x17\x18\xB6\xC8\x1F\xEC\xBE\x96\x49\x12\xBD\x8A\xDD\x25\xDD\xD6\x31\x89\x81\x60\xB1\xA3\xC4\x13\x5C\xCF\xA8\xC5\x5F\x6E\xCF\x34\xD1\xF5\x22\xA3\x29\xF1\x56\xFA\x1D\x81\xA0\xCB\x27\xBA\x37\x30\x69\x92\x4F\x2C\xEF\x25\x46\xBA\xEA\x31\x35\xE0\x8A\xDE\xF4\x79\x99\x49\x16\x39\x41\xAF\x23\x50\xE4\xDE\x48\x1B\x40\x74\x49\xBE\x9B\x98\xD4\xE9\x5E\x7F\x0C\x63\x33\xA3\x12\x03\x48\xAD\xBF\x45\x7E\x55\xEE\x14\x5B\x19\xE9\xCE\x08\xED\x18\xF5\x4C\x8A\x5C\x21\x72\xC8\x6D\x4C\x92\x94\xAC\xC7\x35\xFD\x15\x26\x4D\xF2\x99\xEE\x96\x4E\x7C\x79\x1D\xC5\x33\x3D\xAF\xCA\xB9\xA2\xEF\x8A\x9C\xD7\x64\x8E\x94\xB6\x74\xA5\x20\xF0\xBA\x9C\x58\x45\x40\xE4\x6D\x97\x6B\xD2\x1D\x70\x7B\x4A\xDC\x1E\x77\xD0\xF5\x86\x9C\x2D\x77\xC0\xE3\xF4\xBC\x19\xF5\x94\x2E\x15\xBA\xA7\xB9\xDE\x92\x5F\x75\x07\xA6\x7A\x9C\x9E\xB7\x19\x8D\xEC\x08\xF8\x5C\x45\x6E\x67\xF9\x3B\x51\xCF\x9B\x5C\x7E\xEF\xBB\x72\x3E\xC5\x71\xBB\x43\x0E\x57\xE9\xF6\x04\x1D\xA2\xD7\x7B\x4C\x22\xBD\x02\xB3\x53\x2E\xE9\x54\xBF\x2B\xE0\xBE\xC9\xB5\x4B\x0E\x17\x70\x05\xE5\x7C\xBE\x4F\x4B\xEE\x0A\x3A\xC4\xE4\x3E\x60\x74\x51\x67\x34\x3B\x1F\x36\xB1\x7C\x5E\xF9\xB8\xEC\x6E\x26\x3D\xCA\x2A\x9E\xE9\x71\x56\xB8\x8B\xA4\xEB\x7F\x7B\x1A\xE3\x76\x94\x94\x3B\x4B\x03\x7B\x65\xB7\x98\x9D\x22\xAF\xAB\xA4\x64\x9F\xEC\x9E\x2A\xF5\x28\x57\x49\xC9\x47\x72\xAD\x89\xE0\x55\x2E\x85\x0A\x7C\x1C\x2D\x2D\xBD\x11\xB6\x9F\x49\x76\x34\x32\x0E\xC8\x15\x15\x70\x05\x69\x5D\x1C\x6C\xCC\xBB\x74\xA7\xEE\x93\x68\x67\x91\x7A\x8B\x58\xD8\x4F\x69\x82\x25\x6E\x8F\xB3\xDC\x7D\x93\xEB\x10\x1D\x4D\x25\xEE\x19\x0E\x8F\xD3\xD3\x40\xBB\x4F\xD1\xFF\x63\xEF\x3B\xA0\xE3\x28\xD2\x84\x91\xAA\xFA\x27\xFC\x1C\x0B\x3E\x6E\xD9\x5B\x96\x5B\x73\x0B\x43\xD7\x00\x5E\x4B\xC2\x0B\x68\x85\x6E\x65\xC9\x02\x3D\x6C\x59\x48\xB2\x7D\xEF\x71\xDC\x6C\x6B\xA6\x24\x35\x6A\x75\xB7\x3A\x58\x92\xDF\x9D\x0E\x30\xC9\x44\x63\x72\x32\x39\xE7\x9C\x61\xD9\xDD\x63\x03\x9B\x97\xCD\x91\xBC\xE4\x9C\xD3\xBD\xAE\xAA\x4E\xD5\xD5\x3D\x3D\xB6\x47\x7E\xC7\xF1\xFC\x9E\x3C\x53\xF5\xD5\x97\xFA\xAB\xAF\xBE\xFE\xEA\xAB\x9A\x51\x5C\x1E\x2B\xE1\x29\xF3\xF1\xC6\xBF\xA7\x18\xC8\x34\x29\xA9\x3A\x09\xE3\x9F\x68\xDC\xC6\xA3\xE1\x58\xAA\xA9\xE1\x27\x29\x4D\x62\x91\x6A\x4B\xF3\x53\x0C\xC5\xC4\x08\x76\x88\xF2\x9F\xF6\xFB\x29\x8E\x67\xD8\xD7\x11\x0A\xFE\x2C\x7D\xEE\x13\x65\xC3\x9C\xFE\x1B\x7B\x2E\x74\x96\x3E\xD7\xB8\x13\x41\xA4\x98\xA6\x36\x4D\xCF\xCD\x96\xF0\x54\x19\xDB\xF6\xF3\xCC\x18\xCB\xE3\xE6\x0B\x8D\x5B\x51\x86\xC7\xCD\x17\x1B\x77\xF4\x3F\x96\x6C\x65\x1C\x97\x94\xCA\xE1\x78\xCA\x7C\x89\x49\x48\x4E\x57\xEA\x95\x97\xA3\x34\xAC\x57\x18\x26\x45\xAF\xBC\xEA\x63\x52\xCC\xD7\xA8\xA5\x4C\x18\xD6\xEB\x6C\xF0\x84\x52\xA9\xD8\xEE\xD0\x1B\xD1\xC1\xFA\x9B\x6C\xF0\x94\x61\xBD\xE5\xE3\x19\xB2\xDF\xA6\xE6\x38\x41\x8E\xE9\xBE\xC3\xA4\x33\x35\xD7\x7E\xD7\x87\xA9\x54\xDE\x63\x1F\x6D\x77\xE8\x7D\xF6\xB1\xA2\xAE\xFC\xA0\x71\xEB\x92\xFF\xF9\xC3\xC6\x1D\x02\x17\x55\xD2\xA9\x23\xFB\xC8\x67\x86\x7E\xFD\xD8\x07\x1F\x77\xB5\x4F\x98\xD2\x3D\xD7\xD5\x5C\xD2\x70\xF3\x11\x80\x76\x8E\x8D\xBB\xDA\x91\xE0\xFF\xD3\x67\xAC\x7B\xAE\xED\x28\xF0\x77\x41\x8F\xD7\xB4\x1A\x6C\xE5\x63\x39\x1A\x50\x8D\x4F\x50\x65\x3B\x06\x3D\x86\x68\x29\xDA\x31\x0C\xC8\xC2\xE3\xC7\x82\xCF\x05\x64\x4B\x78\x4A\x29\x3B\xC7\x81\x1D\x12\xA3\x8E\x07\x73\x12\x6D\x53\x6B\x00\x55\x8D\x63\xB9\x7A\xF9\x04\xF6\x85\x9C\x77\x3E\x11\x30\x2B\xC0\xAA\x76\x12\xA0\x46\xAD\xAE\xC2\xAA\xEE\xC9\x7F\x32\xA0\x56\x8C\xA7\xC8\x59\x58\xB7\xE9\x6B\xA7\xB0\x16\x76\x3A\xD6\x6D\xFA\xDA\xA9\x20\x5C\x15\xCA\xE3\xE6\x5A\xB0\x1D\x15\x70\xD2\xB0\xC6\x3C\x4C\xA7\x79\x0D\x8A\xA5\x38\xB6\x3B\xE4\xBD\xD3\x94\xD7\x81\x39\xF1\x06\x4F\x17\xA7\x83\xED\x7C\x5B\x19\xC3\x96\x8E\xB5\x33\xC0\x8E\xC1\x19\x3B\x0F\xA0\x64\x2A\x96\x32\x6E\x9F\x09\x76\x08\x9B\x27\x5B\xC8\xFB\xCE\x59\x60\x5B\x5B\x9D\x0A\xFC\xF4\xD9\x60\x7B\xCF\x77\x47\x9B\xCE\x01\xDB\x91\x52\x68\xD3\xB0\x71\xC9\x34\x26\x9B\xCF\x05\xDB\xDB\x13\xAE\x62\x61\xD2\x4C\x9A\xCE\x03\x73\xEC\x49\xF2\x8E\xA2\x0D\x5B\xC6\x24\x6D\x3C\x1F\xEC\x1E\x0D\x8B\x74\x3C\xE5\x94\x4C\x4B\x1D\xC7\x5C\x84\x72\x01\xD8\xD9\xDF\xF9\x9E\xAB\xE3\x49\xAE\xF7\x42\xF0\xF9\xA0\xB7\x82\x35\xEC\x60\x99\xE4\x1D\xD1\x7A\xB0\x4B\x74\xD8\xA1\x87\x71\x03\x2F\x02\x5F\xE0\x06\x1E\x7A\x18\x1B\x7A\x31\x98\x53\x2A\x95\xA7\x94\x92\xE9\x5A\xB8\xB4\x52\xB5\x1C\x57\xD1\x2E\x01\x0F\x37\xD4\x25\x54\x8B\x23\xAB\x6F\x52\xF9\x52\xF0\x60\xBD\x84\xF0\x95\x79\x40\x7D\x25\xB8\x0C\x58\x75\x12\x80\xAE\x9B\x32\x57\xC0\xEF\x75\xA3\xCB\xEB\x46\x94\x9E\x3D\x11\x12\xBD\x02\x4C\xD7\x97\x68\x39\xBD\xAC\xFE\x4A\x70\xF8\xAC\xD1\x46\x57\x81\xE1\x3A\x11\x33\x5D\x7B\xB4\x34\xA4\x94\xC7\x08\x3D\x74\x35\x38\xBA\x5E\xD6\x5F\x2A\x29\xB6\xA7\xC0\x12\x9E\x72\xB0\xA5\x2B\x5A\x96\x76\xAF\xA9\xDF\x93\x25\x4C\x64\xD1\xBE\xB6\x7E\x4F\x36\x41\x1B\x5D\x07\xD6\xD7\x4B\xE1\xEC\x44\x1D\x3F\x73\x04\x13\x29\x45\x11\xB4\xF9\x7A\xA0\x78\x3E\x3F\x96\x0E\xB1\x0D\xCB\x89\xBD\xB1\x6B\xD8\xB6\xDB\x54\xDD\xD9\x6B\xAE\xAA\x3B\xED\x05\xF2\x5F\xB1\x5D\xF6\xFE\xD2\xCF\xF1\x74\x0A\x07\x8E\x6E\x00\xFF\x11\x90\x56\x75\x27\x41\x6B\x41\xAD\xC4\x92\x7F\xB3\xC8\xDF\x08\x56\x0E\x19\x86\x16\x83\xA1\x07\x55\x54\x43\x27\x0C\x94\x54\xDD\x53\xA7\xB7\x06\x6E\x4A\xB9\x6F\x02\xDD\xD1\x6E\xEF\x7D\xAC\x64\xE8\x65\xEE\x99\xCD\x5D\x69\x68\x8A\xA3\x6A\xB8\xC0\x76\x0C\xE9\x7F\x73\xE5\x22\x62\xCB\x31\xBA\x19\xEC\x4E\xB2\x19\x0B\x55\x67\xC0\x51\x1C\xDC\xDA\xDA\xE7\xDA\xA3\x32\x21\x15\x35\xB7\x5B\x40\x91\x83\x1B\xB4\xA6\x07\xB0\x62\x95\x05\xC0\xB7\x82\x31\x0E\x98\x41\x92\xD6\x01\x62\xA5\x7D\x2A\x2E\xE3\x20\x5B\x9D\xDA\xE1\xA9\xD7\xFF\xCB\x03\x51\x4D\xA1\xDB\xC0\x49\x0D\xA4\xAB\xCF\x32\x46\x7C\x52\x3E\xE9\x0D\x20\x19\xC1\xD5\xA1\x97\x47\x0D\x2B\xD6\xB4\x44\x71\xCA\xA3\x07\xAB\x7A\x25\x95\x9F\xDB\xC1\x2E\xA4\xA7\x93\x96\x34\x59\xAD\xAD\x1D\xDE\x1C\xEB\xD1\x6D\x72\xD2\x09\xDD\x01\xBE\xC8\xF5\xCF\xF8\x1F\x65\x74\x27\x40\x34\xBB\x84\x47\xF0\x94\xD9\xDA\xBA\x42\xD1\xC6\xB0\xD5\x46\xDA\xBA\x2D\x65\xA4\xBD\xB5\x75\x86\xB6\xC9\xE8\xAE\x2C\x44\x77\x03\x99\xEB\xEC\x54\x1C\x39\xC0\xC3\xB8\xF7\x3E\xA2\x7B\xC0\x6E\x1C\xE8\x80\xA3\x58\x51\x58\x4F\xFF\xE8\x5E\x66\x2A\x21\xD8\x21\x2E\xB6\x9D\x04\xDC\x7D\x60\x47\x0E\xAE\xD7\x30\x65\x74\x7F\x82\xA1\x85\xD3\x0E\xEE\x27\x87\xB0\x7C\xCB\x66\x18\x1E\x00\xFF\xC8\x81\x12\xAD\x13\xED\x3D\x08\xF6\xE0\xFA\x16\x8D\x9B\xCE\xF4\x0A\xB5\xE2\x30\xFB\x22\xDF\x97\x9A\xE8\x21\x50\x48\x68\xC0\x74\x5C\x0B\x47\x39\xF6\x50\x7E\x0B\xEC\xCC\x73\x86\x47\x54\x9D\xB2\x86\x1E\x06\xBD\x5C\xEF\x32\x9D\x94\xD5\x55\xFA\x5D\x1D\x7B\x32\x0C\x90\xF3\xF4\xE1\xD4\xA3\xFE\x97\xFB\x4A\x4D\xD8\x23\xF7\x6D\x70\xB7\x94\x48\x5A\x6E\x44\xB9\x41\x7C\xCA\x53\xE7\xEC\x79\x89\x8D\x2B\x3D\x4B\xC7\x5A\x75\x9F\x44\x34\x74\x63\x49\x91\xF4\x72\x12\x50\x9C\xEE\x16\xC2\xE5\xDC\x13\xA8\x07\xF3\xD9\x8C\xE5\x11\x72\x03\xF3\xDF\x69\xCC\x45\x0A\x78\x04\xE4\xE4\x64\x9B\xFF\x8A\xF0\x1D\x70\xE7\x96\xA9\x3B\x8B\xB3\x61\xCA\x1B\x51\xF0\xF2\x99\x45\x7F\xEA\x2D\x3A\xBD\x6C\x20\x5B\x09\xA6\xB7\x7C\x4F\xAA\x36\x2E\x11\x3B\xB7\xDC\xB2\x53\x72\x04\x5B\xE9\x8E\x6B\x6A\x22\x1E\x18\x68\x7B\x12\xB6\x7D\x6E\xC6\x74\xAA\x07\x0B\x05\x01\x74\x7B\xA1\x80\xBE\x0B\xF6\xE4\xA3\x92\x4A\x85\xAE\x5C\xFD\xB8\xEC\x5A\xB6\xBA\x32\x5C\x87\xD1\x7F\x27\xD6\xCE\x6E\x55\xAF\x24\x17\x6C\xF4\x48\x62\x39\x0E\x80\x16\x79\x36\x12\x42\x7E\x4F\xC4\x80\xB7\x84\x0A\x43\x80\xEF\x83\xAF\x66\x40\x2F\x1B\xEC\xDE\x8F\x1F\xF1\x03\xD0\x94\x31\x62\xB1\xE2\xA8\x7A\x13\x3F\xE6\x87\xE0\x9F\x92\x63\x4A\xFB\xCD\x2F\x35\xCD\x1F\x1E\x1E\x1E\x96\xD1\xA3\x09\x80\x4E\xC3\x9C\x0E\x03\x08\xF4\x23\xD0\xCC\xC7\x4E\xA3\x86\xE5\x2C\x57\x6D\x95\x45\x46\x34\x92\x2B\x46\x03\xAE\x1F\x83\x7D\xB9\x41\x7D\x16\xCE\x1C\x42\xF9\x2D\xA2\x9F\x80\x5D\xB9\x91\x8B\xE9\xC5\x68\x72\x28\xD4\x4F\xC1\x12\x1E\xBB\x61\x67\x73\x24\xFE\xC8\xE2\xD9\x9F\x81\x05\x09\x1D\xD0\x62\xFD\x38\x36\x1A\xE1\x04\x06\x89\x7E\xCE\x6C\xA8\xC7\xA6\x61\xB4\x17\x54\xC6\x39\x60\x04\x7E\xC1\xE2\x4F\x1F\x70\x91\x5E\x11\x81\xFD\x12\x1C\x5C\x2D\x34\xF6\x9A\x7A\x74\x96\x25\xC8\x54\x25\x7A\x0C\x7C\x81\x93\xAA\xCB\x70\x48\xDC\x8B\x7E\x05\xBE\x92\x34\x7E\xD5\x1E\x8D\x61\x44\xBF\x66\xE1\xB0\xCF\x0E\x83\x1E\x34\xBC\xF7\x04\x39\x54\xC3\x6F\xC0\x5E\x62\xB8\x7E\x7A\x89\x14\x07\xFE\x5B\xE0\x44\xFD\x5F\x05\x7B\x6E\x6C\x48\xB1\x31\x15\xD4\x93\x90\xBC\xD8\x44\xE4\x16\x7B\xD1\x54\x70\x76\x5A\x2C\x8A\x5B\x46\xBF\xAB\xE1\xBD\xE3\xF7\xE0\xA0\x6A\xB0\x79\xED\xF9\x0F\xE0\xDF\xAB\xA2\xDA\x28\xE3\xFD\x23\xF3\x23\x19\x04\xB8\x19\xFD\x27\x70\x03\xFC\x2C\xBA\xF8\x2C\xBA\x10\x45\x17\xC2\x32\x8F\x3F\x03\x3B\xC4\x49\xE6\xD4\x26\x9E\xAA\x25\x72\x7F\xD0\x90\x52\x1E\x2B\x95\x15\x53\x29\xAB\xCE\xB4\x8C\xFE\x02\x4E\x6D\x88\x65\xBB\x4C\x4D\xF5\x6B\x45\xD2\x11\x16\x6B\x64\xA0\xC8\xE5\x7A\xD3\x01\x0B\x05\xF4\x57\xB0\x76\x76\x59\x1A\xB6\x0C\xDD\xC9\xE6\xE9\xF1\x59\x52\x53\x21\xBF\x9A\x9E\x00\x67\xCE\x2E\x4B\xD5\xD4\xE4\xBF\xDC\x3D\x09\x5A\x49\x6B\x57\x77\x07\xF9\x23\x07\x69\xAF\x62\x5A\x0A\x2C\x5C\xB7\x9E\x02\xDB\x87\x83\x67\xBC\xD1\xE8\x69\x96\x39\x21\x4D\x2B\x0C\x6B\x6C\x62\xD0\xE8\x24\x99\x93\x48\x6A\x2E\xEC\xF4\x89\xC4\x5A\xA2\xD9\x5D\xF4\x0C\x98\x17\x42\x44\x31\x05\x69\x53\x6E\xC0\xB3\xE0\x9D\x86\x5A\xDF\x4F\x43\x0A\x03\xEC\x6A\xA0\x6A\x69\x94\xE4\x08\xAE\xE5\x20\xC5\x1E\x4D\x34\x92\xF0\x38\xE3\xA1\x46\x10\x86\x6F\xEB\x89\x1E\x99\x6F\x09\xEE\xEC\x02\x8F\xF1\xF7\x98\x6C\x3A\x4D\x54\x79\x0B\x9F\x1D\x85\xA4\xBF\xEC\x55\xA5\xEF\x6B\x29\x55\x7D\xA9\x23\xD0\x73\x60\x3E\xE9\x1A\x20\xB7\x48\x0D\x60\x67\xB0\xCD\xD3\x4A\x7B\x6B\x6B\x0F\x49\xFC\x07\xE1\x67\x98\xF0\x7B\x1E\xEC\x1F\x62\xEB\xA8\x54\x06\x8D\x43\x5C\xEC\x8A\xEC\x3F\x69\xC1\x2F\x80\xE5\x21\x58\xBF\xAB\x13\xC8\xA5\xBA\xF7\x92\x95\x6B\xFE\x24\x30\xFA\x91\xD7\x8B\x2C\x30\xF2\x11\x13\x29\x13\x88\x7D\xFD\x79\xAC\xBC\xC4\x42\x44\x3A\x02\xDB\xD8\x21\x53\x30\x02\xDD\xBF\x62\xB1\x51\x1E\xC3\x56\x11\xBD\xCC\x82\x5E\x8A\x84\x64\xE4\xBB\xBB\xBB\x65\xBE\xAD\x8F\x14\x47\x14\xD1\x2B\xA0\x97\xEC\xA3\x84\xFD\x3D\xBA\xA6\xEA\xB8\x42\xC1\x16\x1B\x86\xD9\x46\x2E\xAE\x61\xF7\xD7\x04\xD7\xD8\xA4\x21\x7C\x55\x44\x7F\x30\x15\xFC\x35\xB0\xA4\x46\xFA\x24\xC0\x48\xC5\xF7\xBA\x80\xFC\x60\xBA\xF8\x6F\xE4\x27\xEF\xD1\xAD\x2A\xFD\x9B\x22\xF2\xE9\xD2\xBF\x05\x16\xD7\x46\x3E\x5B\xF8\xB7\x93\xD4\x07\x33\x9E\xFD\x3B\x39\x84\x8F\x48\x5D\x55\xF8\x77\x45\xE4\xD3\x85\x7F\x2F\x87\xF0\x31\xF2\xD9\xC2\xBF\x2F\xA0\x9E\xF1\xE4\x3F\xC8\x4D\x3D\xD7\x83\xFF\x50\x44\x3D\x5D\xF6\x8F\xC0\xC1\x35\x51\xCF\x16\xFD\x63\xE6\x19\xA9\x9B\xD3\x15\x6D\x7A\x15\x8E\xEE\xFB\x09\x86\x7C\x02\x8E\x6C\xE0\xFB\x36\x72\x93\x30\xFC\x5B\x8C\x6D\x4A\xFA\xDE\x91\xF7\xDB\x45\x74\x04\xDC\x23\x12\xDC\x1C\x88\x9D\x58\xE8\x13\x09\x77\xD0\x91\x70\x32\xB1\xA3\x1D\x6C\xBD\x96\x4C\xCB\x98\x9A\x6E\xE3\x73\x6F\x39\x51\xB7\xB6\xEE\x56\x9A\x5F\x88\xEE\x3E\x16\x0B\x85\xF6\xB9\xED\x6C\x9F\xF6\xA8\xFA\x52\x6E\x4A\xA7\xBC\xBA\xBE\x94\x9B\xD3\x29\x1F\x0D\xBF\x1C\x19\xD6\x45\xEA\xC8\x02\x5C\x5D\xDD\x1D\x45\x74\x0C\xBC\x24\xB9\xF3\x1B\x80\xCC\xDA\xA6\x2F\x33\xB6\x14\xF3\x3A\x16\xCA\x29\x01\x43\xD9\xC2\x8A\x83\xE9\x85\xE4\x64\x6F\xF3\x38\x78\x46\x32\x5C\x9D\xDD\x90\x4A\xF8\x86\x7B\x3C\xA4\xFB\xB1\xBD\x1E\x78\x6F\xEC\xC5\x00\xAD\x81\xD1\x78\xA8\xC3\xB2\x94\xE9\xB6\x10\x78\x70\xD4\xC2\x4A\xA5\x18\x96\x80\x79\x62\x9E\x00\xB7\x0F\x21\x66\x3C\x7C\xE8\x44\xA8\x64\x65\xBD\x22\xB8\x32\xA4\x89\x40\x89\x52\x5C\x27\xC1\xFF\x0A\xC1\x48\x18\x46\x61\x27\xE4\x1C\x02\xF8\xA1\x14\xF9\x93\x6A\x45\xB1\x9A\x98\x04\x12\x74\x32\xFC\xA6\x30\x53\xB0\x61\xE2\x09\xD3\x02\xA7\xC0\x63\x1A\x42\xE8\x01\x07\x9B\x39\xA5\xAB\x41\x07\x79\xC4\x47\xA7\xC2\xE1\x08\x1F\x75\x2C\xFF\x58\x0B\x9B\x23\x33\xB5\x8F\xDC\x7A\xDF\x51\x2E\x63\x8D\xB8\x91\x94\x0A\xA5\xD3\xE0\x71\x49\xCF\xD1\x3B\xEB\x9E\xC3\xE3\x7F\x1D\xDC\x3F\xA1\x7B\x55\x77\xDA\x3D\xAE\x04\x6F\x13\xE4\x8F\xFF\x32\x72\x3A\x4C\x3E\x5D\x3A\x36\xE1\x5B\xCE\x80\x5F\x4F\x27\xB3\x68\x4A\xB5\x1D\x55\x1F\x09\xC8\xF1\x84\xCE\x84\x4E\x76\xCA\x22\x6A\x28\xF9\x6C\x58\x98\x5A\x8A\xF6\x17\x0A\xE8\x2C\xE8\xD6\x91\x6C\x24\x2F\xC2\xD1\x3D\xBB\x1E\xE2\x0A\x52\x44\x1C\xD9\x73\xE0\xAA\x3A\x92\x4D\x11\xD7\x7F\xC4\xE7\xC2\x13\x93\x53\x62\xA9\x8E\xFB\x14\xDB\xDE\x0C\xD3\xE2\x3C\xF8\xF9\x08\x7C\x8F\xED\x73\x82\xCE\x87\xD1\xC2\x47\x76\xF5\x37\x57\x6C\x23\xD2\x4C\x0C\x84\x18\x01\xBB\x52\x3A\xAA\xF2\x49\x4B\x31\x49\x5A\x24\x0E\xCE\xBC\x48\x7B\xA2\x08\x52\x00\x55\x40\x17\xC0\x75\xB1\x65\xDC\xB1\x30\x73\xF4\xC1\x76\xA4\x7F\x86\x30\xD8\x9E\xC4\xB6\x9D\xF1\x40\x03\x38\xC2\x77\x05\xDB\x8E\x65\x4C\xCB\x3C\x8D\x48\xD6\x26\x42\x88\xE5\x6C\xD0\x85\x90\xEE\x8F\xFA\x3B\x21\x7D\x24\x1C\x09\x6B\xFE\x68\x7B\x7C\x8F\x6B\x3D\x3C\x28\x75\xCC\x12\x65\x7A\x08\x77\x92\x1F\x6A\xA1\x4F\x4F\x0E\x17\x87\x18\x7C\xB7\xA6\x8C\xD8\xE8\x22\xB8\x13\xE9\xEA\x9C\x2E\x6B\xB8\xDB\xD0\xC8\xD6\x2C\xF1\x4C\x17\x43\x94\xC9\x58\x64\x87\x13\x5D\x02\xF7\xC9\x84\x1D\x50\xC7\x4D\x0D\x2F\x8D\x89\xB1\xD4\x44\x97\xC2\x7F\xCE\x1C\xD6\x65\x38\x32\xBA\x0C\x0E\x54\x51\x90\x89\x15\x87\xC7\x5D\x65\xF1\x42\x97\x33\xF7\xEE\x8D\x76\x54\x47\x35\x74\xBA\x05\x95\xBA\x47\x16\xC9\xD1\x5C\x01\x3B\xD3\xC6\xA6\x6C\x8A\x85\x31\x4A\x90\x12\x45\x57\x06\x3A\xE3\xB1\xA4\x6E\x55\x7B\xC3\xAE\x82\x7D\x55\xD5\x41\xD1\xC9\x39\x22\x23\xA2\x8B\xAB\xE1\xBF\xC4\x50\xB2\xCD\x38\xBA\x0C\x65\xEC\xDD\x86\x7B\xDB\xD7\xC0\xAF\xA4\x63\x08\x36\x28\xAF\x85\xA3\xF9\xF6\x50\xF9\x8D\xA7\xD4\xBC\x7A\xB8\x09\xC4\x05\x95\xD7\xC1\x03\x52\xB5\xD4\x65\x2C\xC6\xC3\x4E\x9F\x62\x61\x3D\xCD\x81\xA2\xEB\xA1\x9C\x31\x7E\x39\xB6\x1C\xB5\xAC\x68\x0B\x15\x4B\x46\x37\xB0\x40\x47\x0C\xDA\x69\x68\x9A\x62\xDA\x98\xB3\xFC\x1B\xD9\xCB\xAD\x78\x50\xBF\x3A\x32\xCA\x18\x44\x37\x71\xAA\x8D\x43\xB2\x1D\x70\x74\x33\x6C\x89\x41\xF5\x93\x1F\xB1\x58\x8C\x95\x8A\xAA\x8F\x30\x3F\x90\xB4\xA5\x5B\x60\x7C\xEB\xBE\x5B\xF1\xBC\x76\x87\x46\x9E\x36\x31\xA1\xE8\x98\x58\x9C\x29\x70\x25\xB7\xC2\x35\x0D\xDC\x3B\x29\x5B\x06\xC8\x88\x6E\x4B\x19\xC7\x19\x4F\x95\xF4\x73\x19\x64\x12\x45\xDB\x9A\x31\x49\x7E\x8E\xA3\x2D\xC6\x0D\x2D\xCE\xF6\x73\xC5\x5C\x2B\xBA\x0D\xAE\xA3\xCB\x66\x42\xA6\x9E\x71\x53\x6B\x6D\xED\x37\x5C\xBD\xD2\x5C\x8B\x7C\x7B\xA5\xC8\x35\x60\x6A\x6A\x39\x4B\x30\x0A\x40\x7E\xE6\xE1\xF6\x3C\x5C\xB5\xCC\x36\x57\x77\xE4\xE1\xAA\x69\xB6\xB9\xBA\x13\xD2\x6D\xAF\x8E\x4A\xC5\x5B\x97\x70\x85\x16\x24\xD1\xD5\x6A\x54\xB1\x3A\x35\xC5\xB6\x17\xBA\xAA\x56\x21\xBF\x32\x11\x73\xB1\xE8\x2E\x78\xA0\x10\x90\x56\x1F\x79\x88\x08\x9F\xBC\x8B\x14\x58\xF5\xDD\x50\xA7\x01\x8F\xD7\xB4\x8C\xFE\xAC\xD3\x81\x96\xE1\x9A\x09\xDF\x51\xCC\x50\x47\x1A\xD7\x91\x01\xDE\x9C\x76\xED\x22\xBA\x87\xB9\xAE\x1E\x7B\xB9\xA2\xA9\x15\x52\x53\x95\x1D\xCE\xC5\xC7\xDF\x0B\xFF\xD5\xD7\xDB\xB2\x08\xA3\x69\xC4\x29\x4C\xF0\x3E\x96\xA9\x8B\xFB\xD8\xC2\x17\xE1\x64\xD0\xF0\x03\x86\xA2\x30\x66\x14\x30\x78\x3F\x34\xD2\x57\x30\xEF\x73\x67\xA7\xC7\xAC\x52\x76\xB0\x25\x52\xB2\x98\x56\x96\x4A\x1E\x80\x0B\xC3\x67\xB8\x88\xFE\x64\x4E\x36\xE2\xD8\x78\x6A\x52\x0F\xC2\xC3\xAA\xB0\xED\x6B\xB8\x8A\x65\x14\xC5\x6A\x79\x88\xD9\x7B\x2A\xFE\x3E\x6C\x69\xD4\x6A\x13\xF8\xD1\xB7\x58\xBA\x87\x19\x0D\xAB\xA8\xEF\x55\xC6\x53\x4F\x5A\xA0\x87\x99\x9D\xC4\xE8\xE5\xB2\x34\x81\x69\xC7\x45\xF9\x36\x2C\x86\xFA\xEE\xA1\xE7\xDD\xD3\x14\x8E\xBE\xC3\x80\x93\x71\xD0\x0C\xDF\x24\xA3\xEF\x66\x05\x1A\xA4\x62\x88\x94\x49\xE6\x0A\x47\x1E\x81\x6D\xE9\x50\xD5\xA3\xC0\xEF\xC1\xEE\x8C\xE1\x35\x04\x82\xDF\x87\x23\x55\x0A\x55\x36\x20\x1E\x12\x66\xA1\x7E\x00\xFF\x33\x6F\x85\x83\x87\x29\x5F\x6D\x03\x81\xCC\xAE\x47\x21\x20\x85\x02\xFA\x21\x9C\xA9\x3B\x7D\x61\x55\x85\xCF\xC0\xA3\xF5\x53\x40\x66\xA5\x89\x4F\xFF\x47\x70\x75\xEE\x1A\x93\x0D\x66\x20\x5D\x03\xFE\xD4\xFF\x31\x7B\xF5\x60\x2F\xF1\xBA\xED\x78\x7F\x55\xA7\x43\x73\xE4\xF8\xA6\x74\x6C\xD3\xFB\x27\x2C\x57\xC5\x0F\x13\x1D\x18\x4A\x8E\xFE\x29\xFC\xAA\x70\xB4\x7F\xFA\x27\x39\xE2\x67\x29\xF4\x52\x4E\x16\x71\xA3\x7F\x1E\xDB\xA7\x08\x47\x87\x67\x96\x7E\x01\x0B\x42\x88\x5E\xC3\x8C\xA9\x01\xFD\x12\xEE\x2C\x04\xEC\x56\x54\x4D\x46\x8F\xC1\x1D\x22\xBD\xA4\x6E\x14\xFD\x0A\xCE\x89\xB4\xCD\xD0\xC6\x5F\xC3\x7F\x88\x34\x2E\x35\x1D\x75\x5C\x5D\x85\x65\xF4\x9B\x98\x9C\x44\x1A\xB1\x9F\x17\x66\x74\x7F\xCB\x9C\x9D\xF7\x14\x96\x28\x66\x10\xE8\x2C\x51\xAC\xB1\xB0\xCC\xFB\x77\xF0\x4B\x62\x28\x6C\x8D\x60\x19\xFD\x1E\xEE\x2A\xEC\xEE\xC7\x65\x43\x33\x2C\xA2\xAE\x3F\xB0\x35\x86\x87\x21\x1F\xE2\xC7\xAC\x7C\xDF\xFE\xC7\x98\xE2\x3A\x8D\x71\xD3\x75\x30\x1B\x2F\xA3\x3F\xC1\x1D\x23\xBD\xDD\x9A\xE2\x38\xDE\x1B\xCF\x9F\xE1\xA5\x8D\x91\x76\x4F\x8E\x01\xB7\x5C\xC6\xB6\x6D\x58\xB6\x38\xB1\x2A\x4A\x94\x27\xE6\x4F\xE2\xE7\xE8\x58\x03\xD1\x91\x68\x9A\x51\x6F\x5A\xAD\x5C\x31\x1F\x16\xEF\x5F\xEA\x66\x94\x80\xCB\x2A\xE8\x8A\xE8\x2F\x2C\x8A\x4A\x26\x8E\x35\x6C\xA7\xA8\xC9\xB3\xDA\x0A\x9E\x5A\xAE\x68\x2E\x9F\x2A\xAC\x0E\x89\xFE\x0A\x1F\x6A\x14\x9F\x1C\x26\xBB\xD5\x72\xB1\x80\xEA\x41\x36\x07\xA8\x5F\x6F\x95\x09\x93\x07\x0F\x5D\xDB\xEB\x26\x0A\x7A\x1C\x5E\xC9\x9B\x76\x97\x41\x7E\x27\x93\xCE\xB1\x74\x33\xFE\xBF\x66\xDE\x4F\xC0\x47\x1A\x62\x4E\x51\x75\x16\xAB\xF4\xA4\x6E\x0D\xCA\xE0\xBC\x76\xC6\x6A\x1A\x02\x6D\x6A\x49\x9E\x84\x5A\xD2\x07\x1E\xA4\xEA\x8E\x2D\x6F\x42\x7E\x03\x47\xFF\x14\x3C\xB9\x61\x36\xE9\x91\x3A\x85\xD8\x2F\xB5\x04\x5D\xD4\xEC\x9F\x66\xD9\xC5\xC4\x46\x60\xA9\xDB\x0B\x54\x3A\xF4\xCA\x42\x2F\x62\xCA\xD8\x15\x7C\x06\xAE\x05\x99\xD7\x09\xEC\xF3\x69\xF2\x40\xFF\x3B\x3C\xD9\xB3\xF0\xD1\xE4\x62\xC0\xDD\xB1\xD0\xF2\x69\x7A\x2C\xF5\x55\xE7\xDF\xE0\x63\x8D\x35\x5C\x59\xF1\x99\x5E\x73\xEA\xF5\x39\xD8\x41\x5F\xC0\x17\x35\xD3\x90\xBD\x5A\x82\xC3\x83\xF3\x22\x73\x43\xB7\x7D\x24\xCF\xC3\xD5\x7C\x5E\x3B\x67\xAD\xD5\xC6\x50\x65\x95\x67\x61\xDD\xD5\x0B\xF0\x1B\x21\x70\xFF\xA2\xE6\x0D\xC0\xF9\x22\xDC\x25\x82\x22\x72\xCE\x8D\xB9\xEA\x97\xA0\xB5\x11\x82\x0A\x10\xC6\x8B\xD8\xFA\x17\x35\x33\x0F\x1F\x95\xEB\x65\x56\x76\x44\x70\xCC\x78\x82\xA1\x57\xE0\xEA\xFF\x97\xD8\xA9\x8D\x36\xD4\xFF\x2E\xD9\xE4\x19\xB1\x71\xC5\x8C\xFF\xB4\xFD\x6C\xDC\x53\x3C\xFB\x82\x92\x2B\x6D\xEA\x42\xAB\x86\x0B\x83\x67\x47\xEC\xB9\x39\xB6\xED\x37\xC7\xC3\xF0\x4B\x03\x5E\x85\x4F\xC0\xBC\xD3\x80\x0B\x43\xEB\xF8\xC3\x75\xC9\x79\xC0\xD1\xDE\x2C\x7C\xF9\x37\x31\x6D\x90\x8D\xD5\x93\xC9\x8D\x30\xB2\xFA\xEA\xCE\xB7\xB2\xD7\xA0\x13\xFA\xDF\x2E\x83\xA6\xC7\x72\xAC\x2D\x7E\xF9\x50\x2C\x56\x2F\xC6\x20\x2C\x76\xFD\x41\x31\xBA\xBD\xC4\x96\x86\xD7\xE1\xCA\x10\xB2\x0A\xD1\xEC\x1B\xCF\x92\x2C\x89\xB7\x1A\x18\xE1\x37\x60\x39\x1C\xD1\xA7\x58\x8E\xAA\x68\x84\x7E\x6F\x75\xA9\x45\x6A\x10\x0B\xF9\x26\xA4\xB7\xAC\x11\xC0\x51\xC3\x18\xB3\x3D\xE5\xF6\x1A\xCE\xA8\xF7\xF4\x62\xAD\xDD\x1D\x24\x45\x4B\x4E\xED\x90\xF3\x3B\x2C\xB8\x4A\xED\xF7\x97\xF3\xB7\x58\x11\x01\x03\xA4\x35\xDD\x1C\xEC\x41\x86\x31\x26\xA3\xB7\xE1\x61\x35\x70\x43\x4F\xA6\x28\xAA\xE6\x5A\x38\xC9\x4B\xB4\x37\xF8\x9D\x6E\xB8\xBB\x80\x93\x28\x24\xE5\xE3\x5D\x96\xE9\x0C\xCB\x14\xF8\xE2\xA6\xA0\x72\x47\xB0\xFD\xF8\x1E\xAB\x3D\xF3\xBB\x66\xD8\x70\xF4\x3E\x3C\x2B\x19\xA1\x91\xE9\x35\xA4\x68\x8A\x17\xBB\x28\xC3\x0E\xB6\x58\x40\xDD\x26\x9C\x84\xB4\x10\xC5\x3F\x49\x27\x9E\xA8\x31\x98\xBD\x92\xD4\x12\x30\xE8\x03\x96\xF2\xF5\x59\xEE\xD1\xCB\x16\x1E\x96\xD1\x87\x70\xC5\x06\x46\x5A\x1C\xA2\x44\x98\xF8\x11\x47\xB1\x0B\x53\xC0\x8F\xE1\xF1\x29\x3A\xB2\x48\xAD\xC8\x6C\x6A\xE5\x13\xEE\x41\x76\x31\xEF\x88\x8E\x90\x76\x8D\x75\xB0\xAB\x42\x06\x8D\xB0\x94\x0D\x1D\x29\xF5\xC7\x60\x06\x1C\xC5\xEA\xD3\x5C\x7B\xA9\x15\xB9\xF7\x4C\x68\x4C\xE9\x1B\xF4\xE8\x28\x69\xBF\x78\x97\xE6\xDA\xA2\x8B\x0D\x04\x43\x57\x73\x43\xC3\x4B\xDA\xAA\x0E\x3D\x5A\xDA\x3F\xD6\x95\x10\x20\x63\xEC\x31\xD2\x37\x63\x5D\xB4\xE4\x6F\x69\x50\xBD\x81\xB3\x14\x51\xB5\x7C\x83\x54\x75\x1D\x2B\x7D\x43\x40\xA2\xA6\x9A\xA0\xE3\xA4\x78\x85\x51\xC0\x5D\xAF\x41\x8B\x4D\x6A\xC2\x76\x3C\xCF\x50\xF4\xE2\xB8\x6A\x1A\xA3\x7E\x79\x8D\xD4\xC5\x79\x20\x13\x2B\x39\x75\x1E\xC9\x7A\x9D\x20\xB5\xC7\x80\x58\x31\x64\x68\xA6\xD5\x44\x39\x51\xEA\x8C\x75\xF5\xE2\x49\xAE\x7E\x20\xF8\x9A\x61\x04\x27\x49\x5F\x8E\xDB\xDE\xA4\xC2\x95\x8A\x9E\x2C\xCD\x8B\x41\xD0\xDB\x7F\x52\xC5\x2D\xA2\x53\x24\x5A\xF6\x39\x68\x98\x55\x61\x4F\x95\x0E\x15\xA7\x14\x03\xF0\xF4\xD2\x59\x0A\xE1\xEF\x55\x0B\xEE\x04\x46\x6B\x63\xC2\xD1\xDA\x82\xD6\xD6\x41\x3C\xE5\xF8\xAF\x97\xA7\x49\x5F\x8C\x2B\xD1\x1D\x67\x26\x61\xCB\x68\x9D\xC4\x0E\xB1\x75\x1A\xFA\x4A\x6C\x39\x9E\x23\xB1\x07\x8D\x85\xD3\x0E\xB6\xC3\xBA\x7D\xFF\x39\xD5\x25\xCC\x2A\xA2\xD3\xA5\x35\x0D\x9C\xBD\x4D\xB8\xAA\x85\x2B\x34\x0F\x5A\x9F\x8B\xA3\xB9\xB3\x4F\xFE\xE4\x42\x67\x48\x47\x65\x31\xD3\x6D\x58\xF4\x80\x46\x5D\x99\x42\x67\x32\x03\x4B\x96\x66\xF1\x2D\x32\x3A\x2B\x15\xD6\x2F\xE3\x0A\xB3\xCD\x67\x4B\x5F\x4F\x87\x4D\x9B\x5B\x7E\x31\x14\x3A\x47\xDA\x2D\x8D\x2B\x43\x77\x14\x55\x27\xB5\x62\xE8\x5C\x69\x8F\x14\x30\x5A\x71\xD9\x31\x64\xD0\xFB\xBA\xD0\x79\xCC\x7A\x93\x90\xBD\x78\xC4\x73\xCF\xE8\x7C\x69\xF7\x14\x88\x03\xB1\x13\x32\x8C\x2E\x90\xE8\xE9\xD1\x88\x75\x07\x15\x32\x89\x36\x19\x5D\x28\xB5\xA6\x81\x57\x2F\x6A\x59\x2F\x2D\x48\x1B\x9C\x59\x95\x7C\x91\xB4\x53\xDC\x11\xA9\xE3\xA6\xA6\x0E\x4F\xCB\xE8\x62\xE9\x90\x58\x4F\xF4\xA6\x1F\xDF\x09\x54\xBF\x21\x2A\xF8\x46\x16\xA6\x4B\xA4\xBD\x72\xE0\x0C\x0A\x7D\x2E\x95\xBE\xC4\xAD\x63\x64\x4F\x84\x16\xA5\xCB\xE8\x32\xFF\x99\x1A\x8A\x86\xED\x32\xF6\x05\x0E\x2F\x20\xF2\xFD\xDD\xE5\xBE\x89\x71\x90\xD9\x77\x8B\xF9\xA3\xAF\x90\xFE\x4D\x38\x3A\xF3\x26\xA5\x1C\xDF\xFC\x7A\x72\xA9\x4D\x2C\x86\xA2\xFB\x4D\x59\xDC\x5D\x25\x1D\x20\x1C\xDE\x65\x08\x47\x27\x9C\xCB\xD5\x4C\x8B\xFE\x93\x4F\xD7\xE2\x35\x4C\x8B\x3C\x64\x3E\x2D\x5E\xCB\x62\x00\x7E\x74\xD6\xD5\x56\x31\xEB\x29\xA2\xEB\xD8\x83\x48\xA0\xD8\x34\x0F\xE2\x7A\x69\x1F\xB1\x7C\xEC\x3B\xE7\x8B\x7C\xC1\x6E\x90\x06\x33\x87\x09\x23\x95\xEA\xF5\xAB\x37\x4A\x05\xE1\x73\x9D\x89\x37\xC8\xE8\x26\xA9\x37\xC7\xA4\xAA\x41\xCD\x37\x33\xCA\xBC\x40\x33\xF1\x06\x19\xDD\x22\xAD\xCA\x77\x2E\x20\xC6\x4B\xFE\x8B\x77\xC2\x58\x43\x70\x52\xE0\xD6\xDA\x7C\xC9\x6D\xD2\xE1\x79\xB4\xB4\x69\x2C\xE9\x76\xE6\x8D\xB3\x89\x25\x67\xD8\x1D\xD2\x74\x95\x8A\xC2\x4D\xA2\x49\x61\x8D\xE1\x9D\xD2\x39\xB5\x5D\xA3\xE4\xE3\xAC\xE1\x26\xA5\x60\x48\x8E\x6B\xB0\x02\xD8\x42\x01\xDD\x25\x9D\xBB\x39\x78\x4B\xBF\xE8\x29\xCA\xDC\xDD\xB3\xAA\xB8\xEA\x17\x63\x45\x79\xBB\x47\x5A\xBF\x39\x78\xCB\xA1\x38\x3F\x03\x75\xAF\x44\x8F\xD1\x45\xB2\x77\xF4\x6E\x26\x99\xBB\xF6\x3B\xF2\x4B\x08\xF7\x49\xDB\x78\x9D\x8E\x61\xB9\x3A\xBE\x5F\xDA\xC6\xFB\xCF\x31\xBC\xA6\x07\xA4\xAD\x86\x5D\x4D\xF3\x1A\xB6\x9C\xD3\xB0\xC5\xE7\x4A\x25\xDB\xF1\xCC\xDC\x34\x54\x2F\x48\xD9\x7A\x87\xC6\x2D\xB6\x9C\x67\x19\x15\xC5\x51\x1A\xA4\x79\xDE\x7F\x5B\x54\xB6\x36\x2D\xA3\xE2\x96\xB1\x65\x37\x6C\x6B\x5A\x46\x19\xDB\x36\xAE\xEC\x3D\x34\xDD\xB0\xED\xB2\x21\x57\x77\xDC\xB9\x65\x4D\xD1\x47\x0E\x68\x6A\x99\x37\x7F\x5E\xD3\xDE\x7B\xEE\xD9\x3C\xBF\xB9\xA9\x69\x7E\x73\x4B\x53\xF3\x82\x96\xA6\x7D\xF6\xDC\xB7\xA5\xA2\xE0\xA1\x96\xCA\x82\xF9\xFB\x0E\xEF\xDD\x34\x83\xA7\xCC\xA6\x99\x08\xC8\x7E\x0B\xF6\x9B\xD7\xD4\xF2\x3F\x01\x00\x00\xFF\xFF\x89\x10\x7A\xF3\x3E\x70\x06\x00") +var gzippedCallGraphCSV = []byte("\x1F\x8B\x08\x00\x00\x00\x00\x00\x00\xFF\xEC\xBD\x5B\x6F\xDC\x38\xB2\x38\xFE\xEE\xCF\xF1\x43\x1E\x82\xE0\x8F\x8C\x67\x77\x91\x31\x72\x02\x38\x17\xEF\x19\x9C\xD9\x24\x27\xC9\xEE\xBC\x04\x10\x68\x89\xDD\xAD\x13\xB5\x24\x53\x94\xD3\x9E\x87\x7C\xF6\x3F\x44\xEA\xC2\x4B\x91\xA2\xEE\x6A\x47\x0F\x33\x71\xB3\x8A\x75\x63\x55\xB1\x44\x52\x54\x92\x22\x0F\xED\xF7\x9E\x9F\xE4\x31\x7D\x56\xFC\xBA\x47\x51\x8E\x3D\xFA\x90\xE2\x0B\x19\xE8\x1F\x10\xA1\x09\xC9\x63\x15\x50\xFC\x8A\xF3\xE3\x2D\x26\x5E\x18\xD3\x1A\x98\xE5\x47\x13\xC1\x02\x74\x4C\x03\xEF\x2E\xC6\xDF\xB5\xC6\x23\x3A\x79\x3E\x3D\xE9\xC8\x19\xA6\x5E\xF8\xEB\xA5\xC6\x00\xDD\x26\x84\x6A\xE8\x01\x8E\x34\xCC\x52\x4A\x9A\x78\xB7\x3B\x09\x78\x87\x82\x40\xC3\xBE\xDD\x15\x98\xBC\x4F\x0D\x4C\x49\x12\xE4\xBE\xD1\x54\x15\x58\xD3\x4E\x04\xA8\x1A\x4A\x9D\x54\x2D\x45\x86\xB2\xA6\x62\x37\x51\x5B\xB1\x07\xA8\x71\x85\x70\x77\xCC\xE1\x5E\xA0\xE6\x47\x74\x32\x69\x2D\x83\xFC\xE4\x98\x22\x22\x40\xC3\xD8\xD8\x51\x02\xA9\x1D\xB3\x84\x18\x0D\x5D\xC3\x10\x21\xE8\xC1\xFB\x1E\xD2\x83\xE7\xA3\xD4\x04\x47\x69\x8A\xE3\xC0\x44\x39\x3B\xA0\x28\x4A\xBE\x7B\x7E\x92\x3E\x98\x28\x64\xA2\xE9\x51\x14\x99\x04\xAB\x40\xB7\x49\x12\x61\x14\x37\xED\xF1\x83\xB1\x4B\x09\x32\x75\x29\x5C\x62\x8F\xE9\xC5\x6D\x1E\x46\x34\x8C\xBD\x23\x2E\x06\x46\xA0\x16\x52\x4C\xCC\x50\x6B\xD7\xCA\xE6\x00\x42\x25\x8E\x0C\xFA\x55\xD5\xC2\x0C\xD6\x19\xFF\xDA\xCA\xF9\x57\xDD\x12\xA4\x18\x5B\x74\x9B\x19\x1C\xBA\x06\xCB\x31\x27\x35\x4B\x11\x27\x77\x40\xB7\x19\xD0\x5C\xC7\x93\xC4\x5C\x88\x3F\xA9\x5D\x8F\x17\x06\x26\x49\x1E\x07\x36\xA9\x39\x02\x20\x77\x03\xD0\x25\x17\x3A\xB1\x3F\x0B\xAA\x75\xDE\x55\x30\x64\x35\x1A\x79\x54\x45\x1A\x88\x41\x15\x1F\x87\x91\x4D\x13\x06\x07\x14\xA9\xDB\x75\x3D\x9A\x2E\xC5\x5F\x10\x40\x96\xBE\x16\xC1\x20\xE2\x2E\x4A\x12\x62\x93\x91\x23\x00\x42\x36\x00\x5D\x4A\xA1\x13\xFB\x13\x04\xC9\x82\x36\x82\x18\x24\x4D\xA3\xDC\xEA\xCC\x35\x5C\xA7\xCD\x40\x80\x0A\x75\xBB\xAE\x41\xD3\xA5\x9E\xE4\xE4\x76\x80\x03\xE4\x23\x35\xC0\xA0\xD5\x31\x8C\xED\x6A\x71\x04\x40\xF8\x06\xA0\x4B\x2F\x74\xCA\xF2\x5B\x10\x20\xCB\xDF\x88\xA1\x2A\xD0\x40\x5A\x35\xB0\x91\x2D\xB2\x71\x10\xEE\x24\xCD\xF2\x88\x86\x69\xF4\x60\xD5\x5E\xC4\x01\xE8\x57\x60\xC8\x3E\x22\x0C\x30\x91\xD4\xB5\x9E\xD3\x75\x98\x81\x23\x68\x2B\x11\x68\x30\x57\x10\xDE\x87\x01\xB6\xE9\x2C\x60\xE8\xFC\x4B\x20\xA0\xAF\x00\x09\xF0\x0E\xE5\x11\x55\x34\x16\xBB\x06\xE1\xBD\xA9\x2B\xC0\x0D\xD2\x55\x00\x99\x12\x39\x36\xD5\x8F\x35\xB8\xE0\x18\x66\x61\x4C\xF1\x1E\xEC\x0A\x64\x63\xAC\x16\xC1\x52\x33\x90\xF8\xAB\x0E\x04\x1F\x81\x66\x9D\xBA\x41\xA3\xA2\x9A\xF1\x93\xD8\x47\x70\x71\xA5\xC2\xA1\x02\x0B\xC6\x11\x8B\x2C\x5E\x33\x45\xA1\x8F\xCD\x4C\x1A\x70\x65\x58\xF2\x20\x4C\x65\x32\x8A\x51\x0C\x15\x45\x93\x82\xE0\x7B\x4C\x32\x8B\x1C\x22\x82\x91\x8D\x8E\x24\x30\xBA\x0D\x69\xE6\x99\x66\x9F\x0A\xA8\xBB\x88\xD8\x4D\x1C\x42\xB1\x47\x16\xEE\x63\xA9\xB1\xAE\x58\xEA\x86\x2C\xBF\xF5\x92\xF2\xB1\xAC\xC1\x52\x44\x2B\xB2\xBF\x8E\x15\xE3\xBD\x26\x89\xE6\x32\x35\x76\x42\x34\x1E\x5E\x9C\xD0\xA6\x11\x99\xEA\x9D\x1A\x6A\x30\x42\xD5\x51\xB3\x42\xD5\x47\x36\x43\xD1\x2A\xDB\x81\xB5\x68\x86\x60\xAD\x09\x51\x3B\xAA\x96\x60\xAD\x92\x29\x90\xA9\x1C\x12\xA8\x88\x06\xAE\x5A\x64\x73\xC4\x78\x8F\xA8\x21\x43\x8A\x08\x06\xA3\x08\x50\x59\xFD\x12\x20\x5B\x40\x60\xC6\xA5\xBE\xDD\x31\x80\x09\x83\x35\xC1\xF2\xE8\x86\xAC\x39\xAA\xA6\xAB\x00\x92\xF5\x34\x51\x54\x03\x9E\xAC\xB1\x72\x32\x07\xCB\xC9\x14\x2D\x27\x30\x5C\x4E\x5A\xBC\x9C\xC0\x80\x61\xAD\xA7\x04\x60\x64\x96\x5F\xB7\xC5\x49\x8B\xA8\xEC\x10\xEE\x68\x84\x77\x86\x87\x71\x00\xC7\x30\x74\x32\x92\xA6\xBD\x0C\x56\x92\x9C\x45\x12\x21\xDF\x2A\x58\xD2\xCC\x04\xC0\xC4\xE9\x09\xEA\xCA\x7E\xC6\x26\xB0\x45\xF8\x66\x76\x86\xAD\x03\xB9\x2D\x64\x42\x75\xC8\x18\x0E\x09\xF7\x87\xD6\xB1\xE0\x48\x06\x07\x54\xC8\x18\x4D\xDD\x10\xD1\x75\xD5\xA5\x00\xC7\xA1\x21\x21\x3B\xB5\x00\x94\x7D\x5B\x04\x68\x2E\xAE\x70\xB6\xF9\x59\xC3\x98\x0F\x23\xF1\xC2\x38\x8D\x90\x6F\xA5\x66\x1C\x97\x4A\x52\x35\x5A\x44\xA0\x1E\x34\xAA\xA4\xC2\x70\xFA\x61\x40\x8A\xE2\x83\xA2\x30\xCE\xA0\x51\x90\x11\x52\x44\x32\xCC\xDA\x8C\xC0\x30\x35\x10\xAE\xD6\x21\x1A\x1A\x4D\x0F\xA1\x8D\xF9\x2D\x4D\xC2\x7F\xFC\xED\xA2\x02\x3F\x3B\xE2\xA3\x7F\x10\x58\x16\x9E\x44\x32\xEC\x53\xB3\xC8\x02\x0A\x24\xB4\x42\x41\x5C\x23\xF1\x8F\xA9\x87\xEF\x0C\x4B\x69\x02\x50\xED\x13\x5B\x3B\xC5\x86\x5E\x7B\x6A\xE9\x54\x02\xF5\x3E\xD8\xDA\x09\x83\xBD\x22\x1B\xA7\x08\xE6\x14\x59\x39\x45\x00\x27\x7C\x8F\xA2\x22\x99\x79\x31\xFE\xCE\xF3\x2B\x8A\xA2\xC4\xAF\x61\x4A\x5E\xAA\x9B\x0E\x18\xA5\x5E\x4A\x49\xF1\x48\x28\x43\x38\x6F\x36\x88\x0D\xA0\xF8\x1F\x84\x16\xE4\xC7\x54\x6E\xFF\xBF\x2C\x89\x79\xB3\xE7\xED\x12\xE2\x63\x2F\x3C\xA6\x09\xA1\x1E\x93\x9C\xAF\x97\x95\x4E\xC0\x7F\x3C\x77\x45\xFC\xC5\x15\xF1\xD2\x15\xF1\x57\x57\xC4\xBF\x31\x1D\xEB\x50\x86\xA2\x40\x06\x2A\x7B\x0A\x32\x10\xD1\x64\xF7\x0F\x88\x64\xD9\x8B\xE0\x32\x0F\xA1\x0C\xFF\xE3\x6F\x5E\x98\x15\xBC\xC2\x00\x4C\xDA\x0A\x4A\xF9\x7B\x8F\x63\x2F\xC0\x7E\x12\xC0\x58\x3B\x82\x61\x80\xE8\x5D\x1A\xA9\x22\x2D\x14\xDE\x02\x00\xB8\xCF\xE9\x00\x95\x51\xD9\x6C\x56\xA4\x44\xB0\xAA\x21\x10\xC9\x28\x09\x63\xB6\x6E\x9D\xF8\x88\xE2\x40\x44\xC3\x71\x0B\xAF\x12\x41\xE0\xC5\x5B\x0C\x44\x34\x5E\x5A\xC7\x67\x42\xEC\x95\xC0\x9C\x44\xED\x4A\x0B\x48\x56\xC5\x15\x62\x36\xE5\x0B\xD4\x56\x03\x08\x48\x56\x23\x28\xC4\x40\xBE\x2C\xEC\xF3\xF8\x88\x48\x76\x40\xE0\x06\x03\x80\xC1\x9A\x9A\x4C\xC3\x7E\x6A\xF0\x3A\xC5\x48\xE0\x8C\x92\x08\xC7\x70\x37\x50\x40\x92\xC7\x34\x3C\x62\x0F\x13\x52\x96\xC7\x21\x4D\x90\x01\x24\x50\xD7\x81\xC2\x28\xCB\xC0\x2C\x4E\x49\x18\xD3\x9D\x67\xE8\xC8\xD3\x70\xB5\x67\xB0\x27\x28\x3D\x78\x04\x23\xFF\x80\x6E\x23\x6D\x9C\x6C\x68\x19\xB6\x93\x61\xCF\xF8\xED\x18\xD5\x2A\x40\xBB\x44\xFB\x16\x86\xEA\x2E\x8F\x09\xA7\x5A\xC5\x65\x43\x16\xE1\x53\x81\x12\x88\x19\x30\xCC\x82\x70\x1F\x52\x00\x89\x0F\x6A\x89\x74\xC0\x27\x1D\xA5\x1A\xB9\xD8\x17\x1D\x46\x82\x86\x59\x96\x56\xB5\xA0\x0E\x85\x18\x1A\x09\x19\xE4\x04\x08\x09\xE5\x5F\xE3\xF0\xA2\x3E\x82\x43\xC1\x08\xE6\x88\xD3\x50\xF3\x38\x2C\x02\xB5\xCC\x13\xC5\x4F\xEA\x80\x9D\xE5\x84\x24\xC5\x73\xAF\x05\xB7\x29\x22\xDC\x38\xD3\xDD\x0B\x07\x6C\x9E\x57\x5C\xB1\x4B\xDA\x46\x71\x69\xF2\x0D\xC7\xE5\x54\x1A\x45\x66\xA8\x58\x41\x81\x08\xF0\x90\x88\xE4\xAB\x99\xDA\x3C\x2A\x0D\xB6\xA6\x94\x19\x95\x87\xAE\x9D\x52\xE5\x59\x4E\x0C\x59\x63\x0B\x3F\x71\x39\x10\xC4\x4A\x6E\xFF\x0F\xFB\xDA\xE0\xC3\x0A\x62\x37\xBC\x36\x92\x00\x9D\x0C\x53\x07\x90\x9C\x64\x64\xB0\xD5\x7C\x12\x92\xC5\x78\x5C\xF2\x76\x7A\x2A\x9E\x1B\x49\xD8\x2E\x1A\x82\x17\xC6\x19\xD6\xE2\xB1\x55\x28\xB3\x34\x42\xD9\x6F\xA0\x02\x62\x80\x4A\x7D\x27\x21\xC5\xC4\xC3\xC7\x90\x56\xA1\x06\x46\x95\x88\x57\xAE\x56\xC8\x33\xB3\x15\x4B\x2D\x01\x4C\x78\x36\xAE\xC2\xD4\x63\xA2\xD1\xDA\xD1\x91\xBE\x92\x3F\x45\x2C\xCB\x5C\x00\xA0\xC9\x65\x86\x86\xC5\x23\x1A\x47\xF8\x88\xA5\x63\x51\x7B\x6C\xE0\xAE\x77\xD0\x50\x18\x05\xB8\x37\x03\x19\xEB\x3D\x18\xD3\xA6\x64\x83\x65\x72\xA5\xCE\xFD\xD4\x9C\xEB\xD8\x4D\x9D\xB7\x1D\xBB\xF9\x49\x14\x61\x9F\x86\x49\x0F\x41\x31\xF5\xA2\xE2\x27\x8A\xE0\xBE\x0D\xED\x56\x3B\x2A\xA8\xC2\x61\x1B\x23\xEB\xBE\x2E\x20\x88\x2D\xF0\x8A\x70\xBC\xA7\x07\xB7\x1E\x56\xAF\x57\x70\x3B\x1B\xBC\x4C\x98\x7A\x44\x98\x5D\x15\xE8\xD2\xC5\x20\x2D\xDD\xD9\x3F\xCE\x3D\xCB\xE2\x8F\x62\x72\x0C\x63\xB9\xCE\x70\xD2\xB2\x7E\x06\x77\xE9\xD2\x09\xD9\x36\x6A\x46\x71\xA0\x34\xC4\xFE\xB1\x91\x6C\x10\x1C\x87\xA1\xE9\x20\x6B\x54\x3C\x4C\x5A\xC6\xA3\x59\xD4\xB2\x20\x71\x19\x95\x9C\x4E\x70\xD3\x2A\x3E\x20\x96\xCD\x47\x7C\xF4\xCB\xF3\x79\x22\x66\xB3\x3E\x52\x3E\x38\xF9\xC8\x3F\x30\x1B\x29\xE4\x65\x78\xA6\xC1\x8F\xF8\x98\x84\x7F\x61\x2F\x8C\x43\x6D\x6C\x34\x98\x50\x62\x54\xB0\x34\xCF\x0E\xA6\x7E\x35\x0C\xE8\xC7\xAB\x10\x68\xC9\x0B\xC0\xD0\x2B\x97\x0A\xA9\x52\xB8\xA6\xC1\xA7\x5F\x10\xA7\xA4\x52\xF9\x11\xDB\xEB\x28\x34\x13\x8E\x5C\xF8\x49\x4C\xF1\x09\x80\x1F\xD1\xC9\x08\xAB\x77\x8C\xE4\x56\xF1\x4C\x6B\x0D\x51\x8C\xCF\x77\x4A\xA4\x1D\x13\x71\x8F\x06\x4A\x39\x12\x5C\x62\x2F\x43\x9A\xE1\x90\xDB\x1B\x7F\xD2\x29\x15\x22\x0B\x73\x9E\x26\x4A\x23\xB9\x04\xAA\x9D\x11\xA6\x58\x19\x41\x82\xCA\xB5\x88\xB8\xDD\xC1\x3B\xEE\x85\x8E\x1A\xB4\xDE\x5D\xDA\xD9\x57\x4F\xB5\x8E\x34\xF1\x32\x3F\x84\xFB\x0A\x55\xA1\x89\x6E\xBD\xBE\x2A\xC0\xBD\x38\x61\xD1\x68\x17\xBB\xC6\x6A\x91\x52\xA2\x66\x90\x56\xA2\x65\x90\x1A\xE2\x27\x4B\x5F\xEE\x5E\x29\x7B\x9E\x3A\x48\x3C\x6E\x79\x32\x20\xC9\xE3\x21\x75\xAF\x96\x54\x64\x90\x92\xA3\xB4\x6E\xCD\xDE\x9F\x4E\xB1\xDE\xFD\x03\x7A\xFD\x85\xC3\xF8\x16\x95\xAB\x82\x0A\x82\xB0\xC4\xAA\x13\xC5\x27\xB6\x8C\x9F\xFF\xF2\x0F\x13\x46\xB9\xD0\x0F\x63\x34\xE9\x58\xDC\x13\xD4\x54\x51\x40\xA2\xD9\x55\x90\xA0\xA5\x00\xD2\xEC\xA6\x74\x13\x06\x42\x13\x43\xB5\x8D\x88\x20\xD9\x46\x15\x45\xB7\x8D\x8A\xA1\xDB\x46\xC0\x60\xB6\xB9\x93\xCE\x2A\xDF\x49\x07\x91\xEF\xE4\x93\xC6\x77\xE2\x71\xE4\x6A\x07\x55\xEC\x2B\xB6\x14\x8F\xEC\x75\x8B\x48\x43\x40\xAA\xE8\xC8\x34\xE4\xFE\x4A\xDF\xA6\x5F\xB9\xA5\x2C\xF4\x95\x5A\xB2\xFC\xB6\x69\x11\x68\x88\x48\x8C\xCE\x31\x17\x8E\x0A\x37\xBF\x8E\x79\x09\x13\xFA\x56\x40\x2E\xB7\x74\x26\xA7\xDA\xBA\xBE\x43\xC2\x59\xE7\xFA\x47\xA9\x5B\xF5\x0B\xC5\x41\xD3\xBF\xD2\xAD\x21\x27\xEC\x5D\x57\x87\x78\x0C\x9C\x18\x48\xE4\x56\x37\x08\x5E\xD5\xB4\x73\x67\x92\xF0\x44\x1F\x92\x98\x35\x62\x35\xB8\x95\x16\x4D\x8B\xE0\x5F\x35\xA0\x74\x2B\x11\xED\x94\x10\xA5\xA5\x32\x81\xA8\x9C\xA8\xB7\x74\xDE\xA5\x56\x59\x38\xE0\x7C\x27\x1E\x9E\xAE\x7F\x14\x7C\xA4\xA5\x79\x81\x90\x48\xFE\x04\xD3\x3F\x89\x0C\x4E\x12\x87\xFA\x17\xD3\xE5\x24\x33\x39\xC1\x5C\x62\xBC\x87\xB8\x14\xCD\x0D\xE1\xFA\x17\x3B\xD3\xDB\xFC\x66\x32\x54\x14\x4A\x3E\x02\x41\xE5\x80\x03\x9F\x49\x32\x8F\xA0\x78\x6F\x38\xE2\x25\xA3\xE8\x07\x4A\x74\x78\x3D\x47\xE8\xD4\xD5\x82\x43\x82\xD5\x0B\x9E\x3A\x0C\x9A\x0A\x6D\x64\xC4\x75\x4C\x19\x45\x3A\x53\xA2\x80\xC4\xD3\x1D\x3A\xE1\xCA\xBA\x9E\x97\x22\x7A\x60\x56\xD4\x57\xEB\x60\x60\xF3\xC0\x63\xEB\x5C\xD5\xC3\x36\x1A\xAC\x6C\x84\x11\xD8\x42\xB7\x57\xAE\xC3\x15\x08\x82\x55\xB5\x66\x85\x9A\xDE\xA5\xB2\xA0\x06\xD5\x9F\x42\x0D\x28\x99\x47\x49\x78\xF4\x22\xBC\xA3\x66\x94\x2C\x8D\x42\x0B\x98\x60\x7E\x64\xC7\x2B\x8C\x57\xAD\x33\xD2\x43\x26\xA9\x06\x80\x84\x45\x0E\x0B\xBC\x54\xDF\x08\xE7\x03\xA6\x80\x45\x59\x2D\x72\x35\xF6\x63\x50\x82\x8F\xC9\xBD\xB6\x3A\xA6\x03\x6B\x37\x82\x7B\x95\x1A\xC1\x40\x2E\xAE\x08\x93\x7E\x19\xB9\x35\x7E\xA7\xC2\x33\x8D\x62\x22\x2C\xAD\xEB\x90\x6A\x44\xD4\x76\xA1\x40\x06\xC8\x95\x63\xBD\x4B\xC8\x11\x51\x03\x12\x64\xD3\x5D\x18\xD1\xBA\x9E\x67\xBE\xAF\x36\xCB\xA7\x67\x4C\x70\x71\x28\x04\xA0\x3C\x14\x5A\x2F\x71\x28\x34\xA0\x30\x14\x25\x4C\xFA\x65\xE4\xA6\x0C\x85\x00\xCF\x34\x8A\xC0\x50\x08\x10\x69\x28\x24\x4B\xA9\x43\x21\x92\x33\x0E\x85\x4A\x59\xDD\x31\x2E\xC5\x87\x6D\x6B\x46\x2A\x6D\xDC\x46\x45\x7A\x35\xD0\x84\x24\x6E\x3B\x9B\x38\xB9\x21\x95\x83\xA0\xE0\x55\xCB\x0E\x66\xC5\x80\x85\x09\x05\x03\x4E\x03\x66\xA4\xCC\x81\x04\x64\x1C\x30\x37\x58\xD9\x30\x37\x32\x63\xC0\x03\xA5\x90\xB0\x33\x71\xB2\x8A\x75\x04\xF2\x58\x5E\xC4\x86\x8C\xC7\x71\x3C\xEF\x88\xC9\xBE\x88\x6A\xF6\xAF\x1C\xCF\x4D\x93\x14\xC9\x2A\x66\x19\xC4\x2A\x76\x99\x2D\x78\xB3\x99\x4F\x13\xC9\x2A\x01\xD1\x00\x96\xB9\x01\x42\x51\xA7\x1B\x03\x92\x52\x1D\x18\xB0\xC4\xA9\x41\xC2\xB0\x07\xB1\x9C\xD4\x6C\xF2\xD4\x48\x56\x79\xC0\xFC\x98\x08\xAF\xA8\x41\x8B\x66\x12\xAC\xDA\x05\x56\x1B\xEB\x95\x41\x0D\x50\x6D\x0C\xB3\xD5\xAC\xEA\x40\x2A\xE0\x5B\x46\x9C\xA2\x51\x7A\xBD\xC6\x88\x25\xCA\x00\x22\x08\xB2\x64\x2E\xC2\x00\x48\x82\x01\x60\x28\x8F\x8A\x16\x1C\xB1\xB1\x55\xA2\xBA\x00\xAF\xA9\x9B\x6C\xD7\x00\xC5\x71\x92\x5A\x65\x1B\xD8\xA8\x65\x30\xB9\xAC\x0B\xBD\x46\xF4\x72\xB6\x33\xBF\x46\x06\x60\x08\x4B\x2B\x0A\x54\x58\x89\x05\xFA\x81\x27\x77\x04\x3C\xE3\x39\x70\x10\x47\x3C\xDC\x04\x22\x88\x3B\xA4\x15\x02\x8E\x83\xAC\x70\x5A\x1B\x17\x09\x07\x22\xD2\x94\x06\x36\x32\x0A\x16\xF0\x86\x40\x0B\x66\xFD\xF8\x09\xE0\x49\x4B\x6D\x06\xB8\xF8\x66\x87\x89\x04\x25\x79\xEC\xDB\x84\x69\x9E\x56\x4D\x24\xC4\xE5\x61\x03\x11\xC0\x61\x04\x0C\x79\xD9\xDC\x40\x42\x58\x13\x36\x63\x98\xBD\x2B\x8C\x03\x7C\x4A\xE0\x3C\x0A\xA0\x40\xBE\x25\xC2\x95\xD5\x6E\x08\xC5\x74\x02\x0C\xC2\xD5\x8D\x5C\x3E\xF4\xD9\xE4\x15\x51\x00\x03\x8B\x60\x48\x1D\x11\x5E\xEE\xC2\x81\x70\x20\xA4\x15\xD2\x66\xAB\x97\x88\x1E\x9C\xCA\x40\x24\xB3\x26\x5E\x6C\x91\xA5\xEC\xDD\x41\x1A\xF5\xF1\xBA\x4D\x6E\x2D\x65\x5A\xDE\x46\x85\x50\x40\xC5\x1A\x70\x9B\xBB\x48\x6F\xAD\x02\xEE\xC2\xA1\xA0\x81\x9A\x8E\x56\xF3\xB0\x65\x08\x9B\x36\x0D\x42\xB3\x4C\xA5\xC3\x20\x5F\x6B\xA0\x80\x15\x38\x10\x90\x5C\xA2\xE9\x22\xB7\xB6\xEC\xA5\xA3\xB4\x99\x59\x20\xA6\x19\x39\xA3\x88\xD0\xD6\xD9\x43\xC1\x02\xCD\x21\xA3\x40\x53\x4C\x96\xDF\x0A\xE7\xA0\x4C\xAC\x24\x24\xCB\x04\x23\x23\xC2\x67\x28\x60\xDC\x56\x7B\x49\xD8\x80\xCD\x24\x38\x34\xFA\x35\x02\xE4\x01\x90\xDC\xB0\x17\x50\x12\x82\xB7\x54\x69\x70\x7D\x61\xAE\x1D\x8D\xBD\xC3\x66\x63\xA7\x25\x87\x9A\x7A\x9B\x4C\x0D\x52\x9B\xA9\x65\x6C\xDD\xD4\x32\x1C\xF2\x3B\x19\x03\x18\x8C\x06\x01\x18\x0C\x8D\xBE\x7D\x30\x3C\xE3\xBB\x95\x06\xAC\x4A\xFF\x08\x65\xD4\xA0\x7D\x83\x6C\x50\x5F\xA7\x66\xB5\x66\x83\x6E\x34\x57\x83\x62\xB2\x57\xF9\x92\xA5\xC1\x60\x12\x8B\x16\x8B\xA5\x04\xEF\x42\xF0\x6E\x2E\x13\x9A\x51\x6C\x01\xC7\x24\x77\x89\x62\x12\x5C\xE6\xD2\x22\x79\x96\xEF\x5C\x24\x17\xD0\x8C\x92\x0B\x38\x26\xC9\x4B\x14\x93\xE4\x32\x97\x36\xC9\xD3\x96\x32\x4B\xC0\x6A\xFE\xB4\xD1\x31\x4A\x9D\x1A\xCA\x29\xA5\xBF\x26\xB3\x02\x87\x3C\x5B\x41\xE1\xD1\x61\xE8\x17\x66\xA5\x0E\x06\x78\x13\x7F\xB0\x62\xBA\x32\x6D\x0A\x88\xCA\x46\xC9\x77\xF8\x95\x41\x1D\x01\x30\x64\x03\x74\x60\x62\xED\xDF\x96\x1F\x1A\x4C\x3D\xD5\xE8\x54\x68\xC2\x1B\x0D\x68\x50\x7D\xAD\x13\x51\xDF\xFE\xA8\x30\xF3\x34\xB5\x9B\xAC\x41\x00\x54\x6E\x80\x56\x93\x71\x34\x6B\xFF\x36\x93\x35\x98\xBA\xC9\x74\x2A\x34\xE1\x8D\x06\x34\xC8\x64\x3A\x11\xD5\x64\x85\x69\xB2\xC2\xC3\x2D\xAF\xA6\x02\x28\x62\x05\x56\xC3\xCD\xF5\x17\x84\x02\x92\x10\x8F\xFB\x9B\x68\x88\x38\x20\x11\x56\xD5\x5A\x49\x34\x18\xB0\x22\xFA\xA2\x3D\x08\x07\x3B\xC3\xBB\xAF\x26\x14\x90\x44\x9C\xC3\x97\x2C\xEA\x08\x7A\xF7\x18\x1D\xC1\xEC\xAC\x40\xF9\x40\x5C\x34\x88\x07\x94\x89\x55\x7A\xF1\x13\x82\x96\x35\xB3\x02\x2E\x37\xCB\x9E\x69\x2D\xE2\x29\xF8\x36\x54\x4D\x24\x33\x6A\xF3\x38\xD5\x86\x29\x1C\x3A\x6D\xE5\x8F\x4D\x78\x4D\x74\x82\xBB\x3A\xB5\x31\x40\xA8\xF8\x1A\x7E\xB9\xB6\x0B\xF5\xD5\x41\x62\x47\xF9\x51\x85\x0B\xC5\x6F\x5D\x30\x41\x6B\x91\xEB\xFD\x03\x7B\x7B\x43\x4D\x06\x80\xEE\x60\x16\x14\x1A\x7D\x98\x01\x84\xA3\x08\x07\xA0\x68\x4B\x7D\x20\xA6\x74\xA0\xC4\x88\x51\x9D\x93\x83\xDC\x50\x2B\xB9\x00\x07\x74\x32\x81\x7E\x60\xE3\x1B\x7E\xC8\xEC\x88\x1D\x68\x82\xC3\xD3\x46\xD2\x18\x95\x62\x0F\xB3\xF2\x46\xB2\xA6\x60\x33\xF7\x00\x63\x4E\x44\x37\xB9\x84\x80\x52\x3F\x4A\x6A\xA4\xD5\x19\x1E\x46\x68\xE9\xDF\x36\x18\x20\x9E\xD9\xC2\xF5\xAB\x86\x46\xF3\xC2\x04\x8D\xB6\x85\xD1\x61\xC3\xEA\x27\xFF\x85\x7C\x5E\x85\xE9\x4E\xCE\x2A\x0D\x48\x19\x8A\x90\x1A\xC2\xB6\x01\xA8\x31\x1F\xCA\x5B\x84\x8A\xDB\x86\xF0\x09\x0C\x25\x74\xD4\x41\x55\x41\xF5\x70\xAA\x80\x76\xAA\xC6\x21\x54\x11\xC5\xC1\xB3\x13\x11\x86\x4D\x45\xD4\x66\x15\x03\x09\x79\x28\xEB\xF3\xDC\x8A\xDB\x36\x1B\xC6\x9A\xC3\x43\x7B\xC9\x10\x54\x7F\xC1\x02\xEE\xCF\x77\xAF\x21\xB8\xEA\x09\x02\x44\x75\x05\x95\xA8\xE2\x0B\x2A\xD8\x18\x57\x0D\xA2\x1E\x51\x06\x22\x0E\x6C\xB4\x68\x33\x21\x56\x83\x23\xD9\xCE\x52\x1B\x68\x08\x80\x3F\x96\x38\x9E\x27\xD6\x14\x24\xF9\x6E\xA0\x53\x8E\xB7\x86\xDE\x0E\x14\x44\xB4\x21\x28\xB6\x87\x51\x2B\xEB\x3B\x10\x72\x62\x56\x1F\xBC\x68\x47\xE5\xA7\x9C\x20\x3C\xEE\x8F\x30\xAC\x39\xAD\x0F\xBF\x62\xDC\xBC\x2F\x61\x80\x18\xF7\xEA\xE0\x28\x14\x6F\x5A\x17\xA7\x2F\x67\xC4\x26\x49\x3A\x23\xCB\x91\x6E\xC2\x35\xC5\xBD\x84\x6F\x9C\x4E\x41\xAC\x76\x59\xA1\x57\xEE\xCB\xBF\x4D\xE5\xB1\x0C\x52\xEB\xEA\x02\xCA\x47\x9A\x95\xD2\x55\xBC\x88\x1D\xA5\x60\xA8\xB0\x5A\x20\x5A\x8C\x00\x50\x30\x40\x24\x3C\x39\x3A\xCC\x24\xDA\x79\x48\x71\x61\xC1\x6B\x82\x42\x42\x12\x23\x42\x02\x28\xD3\x89\x38\x56\xED\xE3\xFE\xCC\xEA\x64\x1D\x3B\x65\x46\xA7\x05\xE7\x18\x4A\x50\x9C\x85\x34\xBC\xC7\x9E\x1F\x25\x59\x4E\x0C\xB3\x20\x80\xC7\xCD\x60\x02\x5F\xD8\xC1\xBA\xDF\x58\x31\x99\x7D\x1D\xF0\x94\x33\xAB\xAD\xF8\x42\xA4\x39\x63\xEF\x6B\xDF\xE8\x6B\x1A\x55\x52\xD5\xE4\x1A\x4C\x38\x7B\x1F\x69\x2F\x6D\x97\x09\x35\x0B\xFF\xD2\xC6\x0E\xBA\x29\x05\xCE\xBA\xC2\xF6\x9C\x1B\x82\x76\x40\xC2\xCC\xA1\x79\xDC\xD3\x74\xAC\x0E\x74\x39\x40\x95\xCA\x54\x93\x52\xCF\xB0\xE2\x89\x31\xD8\xAB\x51\x10\xA8\xE7\xF3\xE5\xA4\x2F\x21\xD8\xBB\x9B\xE6\x01\x80\x87\x92\x2D\xF8\x81\x40\xBB\x20\x30\x8E\x52\xF6\xC1\x48\x7A\xDD\x06\xE3\xB5\x48\x55\x83\xA3\x24\xF9\x96\xA7\x80\xA4\x1A\xA0\x9E\xC1\x04\x88\x50\xCA\x8A\xAD\xCD\xFB\xBC\x28\x4D\x0B\x6F\xAA\x5F\xF5\x6D\x6E\x3A\x91\xE0\x42\x67\xFE\xE7\x45\x9C\x04\xF8\xEA\x8A\xE0\xCB\xAF\x97\x2F\x32\x1A\x5C\x5D\x79\xDE\x2F\x57\x57\xB7\x28\x0B\xFD\xD2\x45\x5F\xFA\x07\x44\xBE\x5E\xFA\x5F\x2F\x9F\x37\x08\x45\x5B\x11\xA4\x21\xCD\x18\xFC\x95\x8A\x50\xFA\x75\x42\x2A\xF0\xF3\xE2\x3F\x3F\x89\x33\xFA\xE4\xEB\xE5\x6F\xCF\x26\xE6\x75\x75\x85\xB2\x2C\xDC\xC7\x5F\x2F\x5F\x70\x92\x9C\xF5\xD3\xAF\x97\xBF\x2D\xA6\xF2\x42\x7C\x1F\xF1\x10\xB3\x5C\xAF\x0F\x71\xD1\x35\x8F\x8B\xE1\xC7\xC1\xD7\xCB\xE7\x51\x12\xEF\x97\xB4\x43\x92\x62\x52\x80\xBF\x5E\x3E\x0F\x70\x84\x29\xFE\x7A\xF9\xE2\x3E\x09\x83\x45\x9D\x71\xB1\xC1\x59\x4E\x65\x9C\xF9\x28\xC5\x3F\x49\xF0\x09\x4E\x17\xE3\xEF\x5F\x2F\x5F\xAC\x28\x20\xCA\x1D\xE9\xA5\xD8\xF3\x69\x73\x09\x6F\x98\x3E\xEA\xD2\x3C\x3B\x78\xB7\xC8\xFF\x56\x07\xDE\x6F\x17\xFB\x28\xB9\xE5\x05\x75\x84\x58\xF2\x71\x49\x97\x32\xAF\x7B\xEC\x17\x8C\x26\x37\x95\x09\x65\xFA\x31\x52\xC7\x4A\xEE\x33\x09\xD7\xA7\xAE\x71\xBA\x0D\xE0\x8A\x07\x30\xC2\x27\x4C\xAE\xAE\xD8\x3F\xCE\xA5\xC8\x36\xA2\x2B\x1E\x51\x36\x38\xEC\xD9\xE9\xEB\xE5\x0B\x36\xAE\xE5\x88\x14\x13\xD6\xD3\xA7\xDB\xF8\xAD\x7C\xFC\x26\x9F\x64\xCB\xB5\x01\x28\x59\x1B\x07\xDE\x29\x2F\x54\x08\x9B\x7F\xAD\xD9\xBF\xCA\x8C\xFF\xA3\x4A\xF9\x5B\x42\x58\xF9\x80\x59\x1F\xC0\xB7\x91\x5B\xF1\xC8\x2D\xB8\x4A\xB8\xF9\xC5\x8A\xFD\x22\x31\x9C\x74\xDE\x06\x6D\xC5\x83\xB6\xC8\xFA\xDB\xE6\x11\x2B\xF6\x88\xE9\xD3\x7B\xB7\x1D\x82\xCD\x59\x36\x67\xD9\x6A\x81\x33\xF3\x8B\xA5\xB6\x55\x37\xA7\x58\xBD\x53\xFC\x28\xFE\x29\x9F\xD0\x81\xA7\x76\xFB\x83\x61\xD9\x21\xC6\x27\xFA\xF5\xF2\x05\xFB\x5E\x4C\x99\x87\xD8\x19\x1E\x13\xD8\x65\x41\xDF\xD4\xB7\xDC\x18\x73\x20\xDD\x45\xDC\x89\xD3\x66\x25\xD5\x7F\xCD\x1C\x7F\x26\x85\xF9\xF6\x9E\x09\x5A\xB6\xEF\x30\xF5\x0F\x5E\x48\xF1\x51\x76\x0F\xB5\xDD\xF8\xA4\x61\xC6\x77\x1E\x7E\xAD\xAF\x3C\xFC\x36\xD2\xC6\xE1\xD7\x3A\x49\xED\xEC\x8E\xD8\x16\x85\xA5\x76\xCA\xCD\x17\xC6\x54\x9A\x0F\x4D\x7D\x25\xD3\x6B\x2C\xDD\x6C\x29\x77\xE8\x68\x4C\xB1\x33\x64\x4D\x03\xF1\x16\x73\x8A\xBD\x3A\x9B\x47\xEC\x0C\xD8\x07\x26\xE1\x66\x2A\x63\xDF\x8E\x56\x33\xD0\x81\x0C\xD8\xCE\xB2\xC5\x96\x16\x7D\xCB\x93\x82\xAE\xE8\xEC\x68\x9D\x0B\x32\xB7\xBA\x36\x23\x00\x73\x84\x8E\xE3\x70\x76\x87\x9F\x5D\xFC\x7A\xF9\x82\x6D\x0F\x75\x3B\x7B\xA1\xF7\x95\xCE\x46\x58\x49\xBB\xCB\xC3\x0D\xD0\xBE\xAD\xE5\xBE\x19\x6D\x27\xF3\x58\x36\x60\xEC\x5A\xEA\xEE\xD3\xCD\xC2\x86\xA5\xE8\x85\x4D\x2B\x92\x9D\x79\x42\x67\x9A\x13\xEF\x88\xC2\x98\x71\x46\x14\x97\xFA\x57\xB6\xA8\xD3\xAF\x3A\xAD\x3B\xF5\x7D\xB4\x95\x90\x8B\xF2\x2E\xA1\xBD\x0A\x23\xAE\xDC\x01\x0D\x39\x76\x1D\xB6\x33\xEF\x66\x74\x74\x11\x3D\x37\x95\x04\xAA\x6A\xA6\x57\x74\xDA\x3B\x2F\x6F\x9D\x85\xE5\x5B\x38\x7D\xD8\xB5\x37\x15\xA4\x4E\x9D\x8B\xA2\xE6\x98\xAE\xC4\xCC\xD3\x17\x0F\x8E\x06\x75\x4E\xC8\x0B\x1B\x6C\xF9\x8C\x6C\x37\x40\x4B\x4A\x76\x1E\x06\x3D\xE9\x15\x4E\xCF\x8A\xB2\x23\xA2\x3E\x78\xA1\x24\x08\xAD\xDF\xE0\x01\xA1\xD5\x8B\x3F\x12\xB0\xDD\x17\x94\x2E\xC2\x15\x65\x30\x21\x27\x6D\xB4\x8F\x38\xB6\x23\x65\x3A\xD2\x63\xF6\x40\xA3\xAA\x1E\x7B\x8F\x8C\x8D\xB5\xB2\x96\x5A\x42\x0A\x3F\xD4\x5B\x1B\xFF\x79\xC9\xED\xF9\x0D\x3F\xCC\xB2\x2C\xDA\xAC\x04\x33\x87\x78\xF5\xF4\x95\xD4\xA3\x14\x8F\xA2\xDB\x68\x9D\x52\x8B\x32\xE5\x71\x42\x02\x4C\x70\xE0\x1D\x51\xCA\x24\xC4\xC4\x2C\xD8\x1A\x75\x28\x24\x6A\xC4\xD0\xC0\xF8\x2E\x47\x91\x47\x13\x0D\x85\x92\x1C\xB7\x1A\x84\xF5\x3E\x2F\x7B\x18\x15\x6E\x31\x19\x68\x0F\x68\x83\x63\x1D\x4A\x3F\x2F\x13\xDA\x2E\x8C\x03\x49\x97\x17\x82\x34\x2E\xD9\xE7\x91\x95\xA0\x8A\x9E\xDB\xD6\xDA\x7A\xB7\xD6\xD4\xB1\x62\x8E\x5B\x2E\x80\x19\xDD\x35\x45\xA1\x25\x41\x4F\x66\x21\x95\xF1\x3C\x56\x2A\xB9\xEC\x50\x94\x61\x29\xB4\x67\x63\xDF\x36\x12\x8F\xAB\x8A\x29\x7E\xDD\x26\x49\xB4\x15\x34\x5B\x41\xB3\x1A\x7B\xFC\x64\x05\x8D\xE7\xE1\x23\xFF\x1C\x47\x1E\x87\x77\x39\x13\xC3\x43\x64\x9F\x99\xA5\x5A\x68\x4E\xE0\xFF\x81\xF5\xD6\x2A\xE4\x7B\xE2\x90\xBF\x97\x98\x49\x7F\x14\x9C\xE1\x79\x3E\x91\x2F\xF7\x55\x20\x04\xEF\xF1\x89\xFF\x56\xC1\x7C\x63\xF2\x71\xCD\x46\xDB\x14\xB4\x4D\x41\xEB\xB0\xC7\x4F\x36\x05\x39\x3E\x53\x3F\xDB\x9C\xBE\x1E\xE4\xE6\x81\x9F\x27\x76\x93\xCD\xEA\x9D\x8D\x2D\x55\x2F\x3F\xE2\x3F\xBD\xD7\x6E\xA9\xFA\xE7\x4B\xD5\x80\x81\x1C\x93\x57\x21\x9A\x3D\xA9\x09\x2B\x58\x86\x4D\x31\x19\xA3\x7D\xAF\xCE\x4E\x31\xC3\x62\x22\xDD\x96\xC6\xE6\x5D\x1A\x7B\xD4\xBB\x95\x9B\x5B\x2D\xE6\x56\x2E\x79\x61\x1B\x9E\xC5\x86\x47\x5D\x6A\xD8\xD6\xC1\xD7\xA2\xC0\x56\xD9\x6E\x95\xED\xA3\xA9\x6C\xB7\x75\xF0\xB1\xE4\x7B\xB2\x2D\xD6\x8C\xB6\x58\xB3\x4D\x76\x6B\x51\x60\xF3\xE7\x6D\xB2\xDB\x26\xBB\x6D\xB2\x6B\x99\xEC\xA6\x5C\xEE\xDA\xA6\x85\xF5\x29\xB0\x4D\x0B\xDB\xB4\xB0\x4D\x0B\xDB\xB4\x00\x4E\x0B\x3D\x56\x16\xB7\xA4\xBE\x02\x05\xB6\xA4\xBE\x25\xF5\x2D\xA9\x6F\x49\xBD\x75\x61\x2B\xC6\x27\xEA\xA5\x24\x3C\x82\x2F\x25\x6F\xC9\x7D\x8D\x0A\x6C\xC9\x7D\x4B\xEE\x5B\x72\xDF\x92\x7B\x6B\x72\xDF\xF2\xD8\x96\xC7\xB6\x3C\x36\x57\x1E\x23\xB8\x90\x67\xB5\xA7\x6F\x84\x97\x68\xEC\x77\x75\x58\x87\x70\xAC\xA5\xE0\xEA\x5E\x9E\x2D\x5D\x6D\xE9\x6A\x15\xF6\xD8\xD2\x55\x93\xAE\x3A\x2E\xFB\x6D\x91\xBB\x45\xEE\x16\xB9\xAB\x8B\x5C\xDB\xCC\xBE\x85\xEC\x16\xB2\x5B\xC8\xAE\x23\x64\xB7\x4A\x78\x0B\xCE\x35\xD9\x63\x0B\x4E\x21\x38\xF9\xBB\x1B\xCD\x5D\x12\x61\x56\xC8\x1D\x06\xD0\x85\x9C\x00\xC6\x6D\x92\x44\x18\xC5\x30\xD8\xED\xE2\x4D\xA5\x9B\x70\xF9\xA6\x02\x21\xF8\xF2\xEA\xEA\xD3\x3B\xFE\xBF\xAF\x97\x2F\xD8\xEF\xCF\xCC\x74\x1F\x43\xEC\x63\xE5\xF1\xBC\x41\xFF\x90\xD2\x30\x89\x33\xED\x56\x34\x23\xFD\x1F\x9C\x81\x01\xAF\xF5\x22\x50\x05\x5F\x35\xB1\xF1\xC2\x53\x19\xDC\xC5\x7A\xDA\xBD\xA5\x62\xB3\x9F\x1C\xD3\x30\x02\xAE\x7A\xD3\x29\x34\x26\xF8\x88\x08\x0D\x51\xF4\xAF\xA2\xF9\xBD\xBB\xB1\x8D\x63\x70\x4D\xF6\x02\x67\xE9\xB6\x39\xF6\x4D\x09\x48\x96\x9C\x5D\xD6\x5F\xD1\x00\x05\x86\x7D\xD0\xF1\xC2\x56\x11\x99\x0F\x92\xD9\x54\xA6\xB7\x57\x6D\x1D\x1C\x5E\x66\xED\xC2\x2F\xB3\xF3\x2B\xBF\x9D\x64\xC1\x28\xFD\xC3\x4A\x43\xC9\x61\x03\xB6\x3F\x27\x4F\x72\xA2\x83\x3D\xD5\xF6\x3E\xF5\x49\xA8\xBD\xDE\x58\x44\xE4\xD6\x62\x63\x16\xA9\x56\x6C\x16\x36\x19\x2F\x59\x27\xCC\xC6\xBB\x43\xC5\xF5\xB3\x3B\xC5\xFC\x83\xB3\x94\x43\x0E\x28\x3B\x67\x1F\xA3\xAA\xE6\x64\x97\x50\xCC\x62\xA1\x59\xAF\x00\x70\x9B\xED\xF5\x62\xC3\xD2\x6F\xD4\xA2\xD6\x89\x8F\x50\xDC\x5A\xF0\x79\x3D\x04\xD5\x60\x86\x4A\xC8\x80\xDA\x5E\x03\x39\xF2\xC8\x4C\x3C\x26\x76\x80\x45\xEF\x9C\x68\x51\x78\xF4\x23\x6B\x0B\xD7\x6C\xFE\xB0\xF3\x6A\x5B\xF9\xB6\x46\xB3\x6C\xE5\xDB\x56\xBE\xAD\x60\x70\xB6\xF2\xCD\xBD\x7C\x33\x9F\x59\x9C\xD9\x6F\x95\x09\x6E\x46\x33\xCC\x38\xCD\x2F\xAF\xF2\x13\x73\xB1\xD1\x6D\xBF\xF5\x7C\x4B\x8F\xAD\xDE\x78\x2C\x53\xCB\x56\x6F\x6C\xF5\xC6\x0A\x06\x67\xAB\x37\x7E\xF6\xE5\x22\xFD\xE0\xC3\xF6\xBC\xBE\xBC\xF4\xDB\xFC\xB9\xCD\x9F\xDB\xFC\xB9\xCD\x9F\x8F\x69\xFE\xDC\x9E\xD7\x7F\xC2\xE7\xF5\xAD\xBC\xD8\xCA\x8B\x33\x9C\x49\xB6\xF2\x62\x2B\x2F\x56\x30\x38\x5B\x79\xB1\x95\x17\x5B\x79\x61\x2B\x2F\x16\xBE\x1A\x6D\xAB\x3D\xB6\xDA\x63\xAB\x3D\xB6\xDA\x63\xAB\x3D\xB6\xDA\x63\xAB\x3D\x7E\xB2\xDA\x43\x74\xC5\x05\x6F\xF0\xDB\x6A\x90\xAD\x06\xD9\x6A\x90\xAD\x06\xD9\x6A\x90\xAD\x06\xD9\x6A\x90\x9F\xB8\x06\xD9\x66\xDC\x33\x4E\xAE\xDB\x8C\xBB\xCD\xB8\x2B\x18\x9C\x6D\xC6\xED\x32\xE3\xDA\x2E\x1B\x6D\xEE\x6E\xD8\x85\x71\xE0\xA1\x28\x2A\x25\xF3\xB2\xFC\xB6\xE5\x76\x0D\x6B\x8F\x38\x3F\xDE\xE2\x42\x8D\x07\x2F\x8C\xA9\x5B\xAF\x2E\xD7\x74\x18\xC9\x68\x37\x77\x18\x31\x9D\x2E\xF3\xB0\x2A\x89\x08\x41\x4E\xAC\x8E\xF8\x98\x61\x27\x2B\x94\x57\x4F\x38\x60\x36\xAF\xA1\xB2\x9B\x45\xDA\x5F\x78\xD5\xEC\x69\x6C\x14\xEE\x1A\x89\xFD\x43\x42\x84\x56\x81\xBC\x74\xE7\x48\xFD\xDD\x2E\x07\xC9\x77\x79\x14\x91\x3C\x76\xF2\xA5\x62\x68\x68\xE2\x8A\x5D\x0F\x8A\xF7\x3D\xA4\x07\xCF\x47\xA9\x6B\xAF\x23\x0B\x2A\x57\xEC\xB2\xAD\x0C\x44\x1C\x74\x93\x0D\xA5\x29\x8E\x9D\xFA\xB4\x5C\xDC\xE2\x12\x4B\xB6\xFB\x5A\x8C\xFD\xD5\xCF\x50\x6F\x65\xDB\x59\xCF\xD0\x5B\xD9\xB6\x95\x6D\x2B\x18\x9C\xAD\x6C\x1B\xA9\x6C\x9B\xEA\xD2\xE5\x2D\x7B\xAF\xD1\x2C\x5B\xF6\xDE\xB2\xF7\x0A\x06\x67\xCB\xDE\x13\x64\xEF\x51\x2E\xDE\xDE\xD2\xF6\x1A\xCD\xB2\xA5\xED\x2D\x6D\xAF\x60\x70\xB6\xB4\x3D\x56\xDA\xEE\x7F\xF9\xFA\x96\xA0\xD7\x68\x96\x2D\x41\x6F\x09\x7A\x05\x83\xB3\x25\xE8\xB1\x12\x74\xB9\x6A\x1D\xA7\x24\x8C\xE9\xCE\x7B\xE6\xDD\x57\x7F\x5F\x34\x7F\x3E\xF3\x62\x9A\x20\x6F\x97\x90\x23\xA2\x52\xFB\x8E\x26\x48\x6A\xC0\xAC\x41\x40\xAF\xF6\x71\x18\xEA\x33\x2F\xC9\xA9\x47\xF0\x7D\xF5\x93\xA3\xB3\xBF\x2B\x3C\xCC\x00\x9C\x30\x16\x00\xBB\x92\x05\xDB\x13\x60\x52\xEF\xBE\x93\x90\x62\xB1\x21\xCD\xA9\x2F\xFC\x66\xFF\x6F\x7E\x7B\x17\xF5\x9F\x62\x23\xDF\xC6\x10\x77\x34\x76\x04\x73\xB2\xC5\x1F\x17\xBE\x06\x2F\x5B\x4A\xC1\x08\x6E\x10\xCA\xBF\x2F\x32\x4A\x68\x12\x3D\x0B\xB3\x2C\x45\x3E\x6E\x7E\xA2\x28\x3D\xA0\xE6\x67\x9E\xA6\x98\x5C\x1C\xF1\xF1\x98\xDC\x63\x91\x83\xD8\xC4\x84\xF0\x8E\x69\xE0\xDD\xA2\x0C\xA3\x20\x10\x34\x64\xCD\xD9\x21\x21\x14\x68\x2F\xD0\xC3\xD8\x27\x50\x7B\x96\xDF\x82\x64\x8E\x79\x04\xA1\x5B\x9A\xAB\x71\xAB\x49\x04\xE1\x3D\x84\x1B\x84\xF7\xC7\x04\x94\xB1\x84\x14\x3F\xB9\xF6\x2A\x64\x57\x79\xA4\xDA\xCE\x1C\x00\x40\xCF\xA9\x4E\x44\x35\xC0\x21\xDC\x51\x50\xA9\x12\x52\xFC\xCC\xC3\x98\x7A\x7F\x61\x92\x28\x50\xCD\xA4\x95\xED\x3C\xD1\xFD\x7C\x42\x7F\x15\xBD\x33\xA6\x5E\x10\xEE\x2E\x85\xA6\x8C\x06\xDE\x2E\xA6\x40\x0B\xA3\x1A\xC6\x21\x2D\x7E\x79\x29\x22\xE8\x98\xD5\xC0\x8A\x12\x6B\x08\xE3\x7B\x80\x46\xD5\x6A\xA4\x53\x21\xD4\xB4\x76\x49\x4E\xBC\x8C\xE2\x54\xA1\x26\xB7\x37\xF4\xBE\xFF\xCA\x8A\x38\x08\xBE\xC7\xF4\x1B\x26\x31\x8E\x14\x60\x16\x9E\xEA\x1F\x17\x4C\x00\x03\x4F\x1D\x56\xB4\xB4\x74\x57\x58\x1B\x10\x24\xD9\x8B\x86\xBB\x0C\xD3\x32\x1B\xB3\x31\x2F\x7E\xEE\x22\xB4\xCF\x8C\xD0\x18\xEF\x11\x0D\xEF\x8D\xDD\xB3\x14\xFB\x21\x8A\x34\x30\x0F\x7B\xB0\xD7\x1D\xC1\x59\xF8\x97\x91\x22\x26\x24\x21\x26\x60\x10\xEE\x43\x0A\x4B\x7B\xB7\x0B\x63\x14\x55\x84\x69\xE2\x65\x7E\xC8\xCD\x50\xFC\xCD\xF0\x2E\xE4\x9F\xAC\x5B\x91\xB6\xB8\x0A\x20\x30\x46\x31\x04\x10\x62\x57\xEB\x51\x59\x0C\x84\xDE\x19\x08\x1E\xB3\xEF\x09\x09\x20\x48\xD1\xEE\x95\x7A\x2B\x60\x06\x6A\xD1\x2E\x8C\x77\x85\x1B\x68\xE2\x08\x01\xDD\x34\x2A\x91\xAE\xD2\x6A\xB2\x83\x00\xE1\x29\x91\x8D\xCF\xF7\x90\xFA\x87\x02\x16\x3C\xC4\x02\x2D\x1D\x56\x79\x17\xB3\x86\x19\x9C\x26\x59\x58\x3B\x9F\x82\xC2\x8F\x71\x98\xFB\x06\x0F\x31\x3A\x86\xBE\x17\x20\x8A\x18\x5A\x39\x5D\xC1\xFC\x21\xA0\xC4\x9D\x8F\x8F\xA2\x14\xB7\x88\xD2\x58\xE7\x51\xD1\x34\xA5\xCF\x2B\xA8\x55\xAB\xA6\x85\x06\x15\xE4\xBB\x90\x42\x41\xA1\x58\xF0\xF5\x13\xBC\xDB\x19\xDB\x05\x6A\x9C\xCB\x11\x9D\xA0\x1E\x4D\xBB\x41\x3A\x09\xAE\xCA\x57\xF8\x1D\xC5\x7B\xAC\x4D\x1E\x0C\xA0\xFA\x46\x9D\x46\xCC\x00\x55\xEC\x2A\x51\xE8\x3D\x78\xB3\xA6\x66\xC6\xD2\x45\x6D\x50\x39\x5B\xD4\xBF\x78\xC2\xD8\x85\x27\xAF\xF2\x0C\x05\xE4\x1F\xB0\xFF\xCD\xC3\xA7\x54\x01\x72\x1E\x6C\xBE\xF4\xC2\x98\x1D\x61\x86\xBA\x57\x65\x8A\x02\x53\x8C\x54\xF2\x37\x35\x8B\xBA\x69\x00\x79\xA4\x4C\xFD\x1A\x60\xAD\x90\xCA\xAD\x01\x68\xFC\x64\x50\xED\x09\x10\x50\x98\x22\x14\xA8\x91\x15\x37\xA3\x86\x0F\x59\x58\x41\x61\x3F\x51\x9A\x46\x0F\x1E\x49\xF2\x38\xF0\xF0\xC9\xC7\x59\x06\x51\xAA\x03\xE2\x02\xA0\xAB\x86\x82\x02\xAD\x26\x5F\x8F\xC4\x01\x88\xA0\x3B\x1F\x40\xA0\x29\xB0\x8C\x34\x80\xA8\x33\x73\xE2\x43\x9A\x61\x4A\x49\x98\x46\x92\x4F\x35\x8D\xBA\x64\x98\x7A\xE1\xAF\x97\x9A\xC6\x65\xB3\x11\x5F\x89\x29\x6E\xE2\xBB\xC2\x28\xB9\x1C\xE0\x0A\xA0\x49\x02\x2A\xA4\xA4\x28\x5A\x44\xAB\x3F\xA5\x66\x61\x0A\xA8\xDB\x80\xA1\xA9\xB0\xF5\x84\x20\x74\x51\x07\x63\x5F\xAA\x29\xCB\xC8\x61\x7E\x92\x3E\xA8\x32\xB1\x36\xC3\x90\xD5\x30\x35\x49\x96\x10\x51\x0D\xB5\x50\x17\x5B\xEB\x2E\x5A\xAB\xC9\x4F\x22\x98\x6D\x09\x53\x1E\x02\x2E\x0C\xD1\xA3\x24\x2E\x13\x96\x32\xE2\x00\x86\x21\x3B\x19\x30\x75\x63\x1D\xCB\x00\xF7\x8F\xE9\x45\xF5\x87\x96\xB4\x04\x24\x2F\x43\x47\xEC\xA1\xE0\xFF\x8A\x6C\x0D\x35\x36\x9A\x15\x24\x85\xF0\x82\xE3\x1F\x1C\x00\xD0\xE5\x4C\x1E\x67\x71\x38\xA1\x8F\x31\xEA\x61\xBB\xA0\x58\x2D\x4B\x58\x93\xC6\xBD\x6A\x04\xC8\x2B\xF8\x3A\x40\xA9\xBC\x8B\x66\x6E\x50\x94\x6A\x88\xBC\x7E\xAF\xC0\xDA\x00\xA1\xD4\xE4\x08\x15\x48\x9F\xA2\x50\xFA\xAC\xC9\x61\x9C\xA1\x36\xF3\xDF\x41\x93\x7E\x62\xB4\x68\x62\xB0\x66\xD5\x43\x56\x38\x21\x8A\xBE\x0D\x9A\xA0\xEE\x1D\x0A\x02\x60\xAD\xA1\x6A\xD6\xE6\x36\x11\x20\xCD\xAF\x22\x00\x30\x14\xD0\x4F\x07\x8A\x8B\x28\x06\x48\x96\xDF\x0A\xEE\x15\x83\xFE\x1D\x03\x15\x91\x88\x5E\xEA\xA3\x75\xA8\x2D\xA5\x21\x2B\xF5\x51\x09\x87\x22\x2E\xD6\x07\xF4\xA4\x8F\xFB\x09\x1A\xF8\x93\x79\xE4\x4F\xA6\xA1\x3F\xC1\x63\x7F\xD2\x06\xFF\xA4\x8E\x3E\xB7\xEE\x6D\xA6\x5A\xB0\x68\xD2\xCD\x57\x23\x1E\xC3\x38\xCF\x94\xB6\x34\xAA\x9A\x18\x54\xA5\xC8\x1B\x75\x9A\xBC\x5D\xAD\xE0\x1B\x0A\x72\xB5\x5B\x30\x51\x29\xB3\x36\x9D\x30\x6B\x56\xE8\x36\xDD\x65\xB2\x28\x08\x34\x0B\x04\x01\x64\x81\x40\x4F\x59\x01\x94\xB2\xAA\xDE\xA5\xEB\x2A\x6C\x64\xEE\x75\x88\x09\x1E\x5C\x79\xBB\xEC\xBE\x52\x88\x4A\x7D\xE5\x74\x5D\xF5\x96\xB8\xD7\x6C\x64\xEE\x41\x78\x5F\x22\x07\xE1\xFD\x45\xFD\x97\x20\x91\x27\xE3\x55\x22\x35\xCD\x6A\xDA\xA8\xBA\xCB\xA9\x81\x11\x68\x92\xA1\xC2\x49\x4A\x2F\x55\xAB\x92\xA5\xEA\x66\x20\xB3\x28\x3C\x15\x88\xB4\xEE\xA9\xB6\x37\xCB\x8F\x10\xC4\x8B\x35\xD8\x11\x78\x3C\x10\x01\xE2\x9A\x47\xDD\x4E\x09\x0A\x23\x56\xBD\xAB\x10\xF0\xD9\xA0\x81\xD6\xC3\xA5\x89\xC4\xE1\x31\xFE\x6E\x00\xB1\xC7\x9B\x98\xE2\x13\x05\x10\x80\x35\x55\xA9\x33\x60\x64\x0D\x47\x33\xB6\x84\x01\x18\x5D\x87\xDF\x1D\xF3\xC8\xC3\x27\xE4\x9B\xE4\x28\xAD\x63\xEB\x6F\xEA\xD9\x66\xB9\x22\x34\xAC\x8A\x19\x99\xF2\xA2\x8F\x26\x1E\x5B\x2C\x20\x08\x12\x41\x5B\x0F\x6F\x20\xCA\x8A\xB8\xD0\x45\x5A\x13\x97\x99\xD6\xD5\xA3\xAE\x48\x3D\x53\xEA\x20\x39\x82\x81\x81\xBF\xB3\x63\x43\x11\x2B\x0F\x4D\xA4\x03\x0C\x01\x6A\x74\x9B\x3B\xB3\xC7\x48\x20\x28\x52\x25\xA0\x1E\xAC\xB5\x48\x7A\x58\x82\x83\x7D\xC7\x76\x4B\x14\x5B\x14\x6D\x40\xEE\x03\xF6\x5B\xEA\xEE\x3A\x51\xF6\x57\xE1\xED\x97\x5E\x84\x2F\x45\x74\x61\x6F\xA6\xC1\xAC\x36\x2B\xD4\xF6\x72\x47\x47\x61\x07\x59\x5C\x11\x05\x80\xF8\xA0\x8C\xDF\x00\xAE\xBB\x18\x92\xA5\xC0\x64\xEB\xFC\x8A\x30\x52\xDA\xFE\x06\x98\xE9\x1B\x24\x01\x6B\xE4\x34\xBF\x27\xE4\x5B\xF3\x74\x5E\x02\x10\x41\x34\xCB\x6F\x91\x47\xB0\xD8\x43\x8A\x33\xDE\x22\xC4\x57\x89\x52\xC7\x15\xFB\x2D\x2E\x4D\xC5\xD0\x6E\x5A\xD9\x2A\x91\xAE\xDA\x04\xE2\x35\x5A\x4D\xBE\x6C\xD1\x49\xA9\xCA\x56\xCD\xFC\x19\x5A\xA2\x16\x53\x0F\xE5\x34\xF1\x93\xF8\x3E\x89\x72\x95\x57\x4C\x3D\x18\x22\xBA\x51\xD9\xE4\x13\xFA\xEB\x85\x34\x52\xD0\x40\xB0\x76\x68\x30\xF8\x7E\x8C\x61\x40\xF8\xBE\x94\x6A\x7B\xDE\xAA\xD8\xBF\x44\x95\xC6\x80\xB5\x41\x82\xC8\xE3\xCC\xBC\x4B\x76\x3A\x96\xF1\xE5\x50\x95\x13\xBD\x9E\x9D\xB5\x80\xD6\x31\x80\xF0\xD6\x91\xD4\x60\x07\xC8\xA8\xE9\xD2\x20\x8B\x38\xA9\x99\xA4\x01\xD6\x03\x39\x3A\x3E\xAA\xF5\x62\xD1\xA4\xD7\x8B\x45\xAB\xB6\x6E\x7F\xD4\xEB\xC5\xA2\x51\xC1\x11\x93\xA4\xC2\x55\x31\x76\x3D\x7F\xAB\xF3\xB1\x00\xB1\xF7\x03\x52\xA8\xD0\x4F\x96\x54\xB0\x94\x71\x2A\xD6\x9A\x4F\x76\x54\x4A\xF2\xD8\xB7\xA3\xEC\xA2\xA4\x7A\x98\x32\xA1\xF8\x38\x8C\xAC\x18\x45\xF8\x84\x71\x91\xBE\xD5\xA5\xFF\x06\x10\x25\xFB\x5F\x9E\x73\x7A\xF8\x94\x26\x84\x7A\xF9\x2F\xFF\x50\xC7\x50\x80\xC8\xDD\x41\xB0\x60\x3D\x05\x22\x3D\x02\x83\x30\x02\xC2\xA4\x11\x11\x60\xF2\xEC\xCD\xA0\xE1\xD1\xA4\x84\x00\x11\x94\x16\x5A\x35\x4E\x0A\xCC\x0C\x01\x1E\x9D\x61\xBA\x20\x5C\x9A\x7C\x3D\x9D\x43\x03\x2F\xAA\x2E\x88\x3A\x50\x77\x2A\x0B\x7E\x52\x4A\xD5\x32\xAC\x0C\xAD\x53\xAC\xD2\x5C\xE6\x58\x15\x99\x25\x59\xB9\xB1\x62\x2A\xE6\x55\x49\x1A\x09\x20\x17\x19\x32\xAC\x9A\x62\x94\x1E\xCA\xCC\x6C\xA0\x87\x82\x80\x26\x00\x45\x36\xFF\x19\xFA\x64\xF9\xED\x8E\x24\xC7\x0B\x7D\x52\x30\x2A\xD0\x1C\x22\x80\x95\x68\xE0\x55\x4D\xA3\x83\x41\xA9\x2A\x88\x6E\x01\xD3\xBC\x65\x13\x0C\xB2\x86\x8C\x51\xEB\x7E\xD4\x0F\x87\x68\x9B\x93\x0A\x98\xFB\x68\x19\x23\xE2\x89\x0C\xF1\x00\x8A\xD8\x4C\x09\x8A\xB3\x34\xC9\xB0\x97\x26\xDF\x2F\x65\x18\xC4\x41\xC6\x68\x8E\xBA\x68\xFD\xE4\xA3\x1E\x06\x51\x34\x10\xC8\x52\xC3\xAA\xD9\xC2\xFD\x2D\xAC\x15\x75\xE5\x9F\x82\x60\x0A\x20\xBB\xCB\x11\xC1\xAC\x11\xEC\x98\x7D\x67\x87\x90\xA3\x1D\x49\xBE\xC3\x18\x20\xD9\x2A\xC4\xD5\x66\x1E\xE2\x1A\x72\x11\xE2\xAA\x24\x95\xCB\xEA\x12\x88\x23\xAE\x03\x85\xB2\x0F\x82\x1A\x89\x96\x47\xDF\x20\x50\xA5\x0D\x04\xE2\x1A\x81\x9D\x98\x56\x3A\x40\x7F\x61\xBF\xE5\xAE\xC7\xB2\x87\xC3\x8B\xA4\xF2\xF5\x08\xE6\x37\x97\x9A\xF3\x7D\x0A\xD1\xBF\xDF\x7E\xFD\x7B\xD0\x83\x74\xD5\x4F\x67\xE0\x79\xFE\x09\x79\x69\x4E\xB0\x77\x1F\x12\x9A\x4B\x35\xE7\xC4\x87\x53\x65\xB2\x33\x7F\x06\x4D\xB0\xD7\x63\x56\xB3\x0A\xA8\x47\xAC\xE2\x4C\xBE\x5A\x45\xD5\x7F\x3D\xD2\x11\x5C\x4A\xBF\x39\x03\x71\x49\x1D\x59\xBA\x7D\xB4\x1A\xCE\x14\x84\xFC\x9C\x1F\x34\x03\x16\x9D\x39\x1F\xF1\x92\xC4\x89\xC4\xE0\xD7\x0E\xB5\x88\x31\xBD\x35\x9C\xC4\x98\x35\xC0\xDC\x24\x9A\x2B\xA5\xB9\xDB\x67\x9E\xE0\x74\xB6\x4E\xF1\xC4\xB7\x12\x69\x66\x0A\xEC\x5A\x18\x5F\xBE\x40\xCE\x87\x2E\x56\x5B\xC0\x83\xDC\xC4\x5A\x24\xD4\xDC\x45\x9B\xD9\xCB\xDD\x04\x5B\xD8\xC1\x7E\xAB\xEE\x09\x5C\x54\x82\x55\x0D\xC9\xF4\xA6\x48\xF3\xEC\xE0\xDD\x22\xFF\x5B\x2D\xD2\xBC\xB1\x03\xF1\x9F\x2B\xA5\x98\x74\x9F\x27\x38\x21\xEE\x33\x45\xA0\xE7\xA1\xAC\x70\x35\x0F\x9F\x28\x26\x31\x8A\xBA\x64\x7B\xBE\xF6\xB3\x5E\x11\xE7\xF4\xDE\x41\x86\x9C\xC5\xC7\x87\xD9\x71\x9E\x48\xE8\x2F\xE3\x5C\x33\x16\x13\xCF\x59\xAA\xF5\x1A\x6C\x39\x53\xCD\x38\xB9\x5B\x24\xF8\x79\xC7\xC5\x4F\x8E\x29\x22\xC6\xE7\x75\xCB\x83\x47\x8B\xE8\x15\x82\x7E\x93\xCC\xD9\x6A\xC2\xB3\xCA\x7D\x12\x06\xCA\xB5\x1E\x59\x42\xA8\x74\x87\x44\x84\xB3\xEC\x25\xBB\x2B\xDA\xE7\x97\x46\xBF\x7A\x52\xFF\xF9\xF4\xD5\xD7\xCB\x17\xC5\xBF\x4D\x8B\xAA\x9F\x81\xC0\x6F\xCF\x04\xE1\x18\x50\x93\xE2\xEF\xC3\xC5\xB0\xFF\xD5\x2E\xE4\xA2\x06\xBA\x4D\x92\x48\xC1\x0C\xE3\x0C\x13\x1A\x26\x31\x93\xC2\x0B\xE3\xC2\x51\x22\x4C\xF1\x1C\x02\x2D\x6A\x8C\x05\x79\x5F\xAC\x6D\x20\xCE\x23\x74\xD8\x35\x30\xAF\x43\xFA\x99\x22\x8A\xAF\xAE\x3E\xE6\xEC\xF6\x97\x1A\x4F\x9B\xB9\x5C\xAE\xC6\xED\x4A\xB3\x2A\xE5\x07\xC8\xA2\x5F\xF4\xD8\x95\x18\x4F\xB5\x4A\xAF\x2F\xE4\xE1\x33\x46\xC4\xB7\x77\xED\xC8\xAA\x1F\x93\x66\x07\xB6\xBF\x8C\x1F\x49\xB2\xBF\xBA\x7A\x77\x4C\xE9\xC3\x4D\x84\xF6\x59\xFB\x47\x0D\xDA\x64\xAF\x39\xB7\xD0\x69\x01\xF3\xD8\x95\xFF\xB2\x7E\x0F\xA1\x8F\x1F\xAE\x41\xD6\x56\x3F\x5D\x4C\xC8\xFA\x86\xA2\x75\x88\xD3\xC7\xC9\x57\x23\x7C\x51\x80\x1E\xC8\x6A\xC4\x11\x62\xFF\x23\xC1\xBB\xF0\x74\xED\xFB\x38\xF2\x6E\x48\x12\xD3\xEB\x38\x78\xCD\x57\x7E\xCA\xB9\xBB\xE5\xF1\x60\x2D\x3A\x09\xA9\x90\x6B\xC6\x25\xA9\x24\x1B\x2C\x91\x40\x59\xFB\x48\x0B\x6F\x66\x9F\x85\xF9\x9F\x30\x0E\x7A\xB8\xF3\x52\x66\x3B\x1F\x83\xB5\xE7\xCC\x37\xFC\xFB\x42\xE4\xEA\xEA\xBA\x78\x38\xFB\x3D\xCE\x68\x95\x1C\x3A\x4C\x0F\x56\x2A\x62\x4E\x6C\x43\x6C\x2A\x18\x57\xC1\xFA\xE8\x25\xB8\x7D\x83\xF7\xA3\xFA\xF3\xEB\xE5\x0B\x25\xDE\x7F\x14\xFF\xF0\xE6\xB6\x5E\x5D\x24\x03\x39\x7E\xC2\x7B\x7C\x4A\xAF\xAE\xFE\x44\xD1\x37\x4C\x5E\xB2\xB6\x1B\x82\xF6\xAF\xAE\xAE\x7E\xF0\x36\x51\x10\x57\x6C\x40\x2E\xF5\xE4\xDA\x60\x82\xBD\x49\x89\x05\x76\x80\xEF\x72\xCC\xCE\xCC\xF2\x0E\x05\x22\x0B\x2D\xA1\xBF\x79\x59\xC1\xD2\xA5\xBA\x9D\xF1\x87\xC8\xC3\x69\x4C\xFF\xBF\x5F\x9E\xD9\x11\xDA\xFB\x77\xF1\x8A\x37\x88\x56\x69\xA4\x10\x5D\x08\x73\xFE\x53\x2B\x62\x9B\xAE\x9F\x69\x31\x95\xAB\x7D\x79\x56\xFB\x4D\x55\x42\x8F\x8B\x1E\xE4\x78\x7C\x14\x54\x8A\xFF\x87\xF4\x3A\xA2\x72\x9A\xA8\xCB\x0C\xB5\xAD\x23\x37\xA3\xCE\xFF\x9B\xE3\x0C\x30\x58\x6F\xA5\x5D\xE8\xA9\x5A\xBF\x4F\x52\x40\xEB\x51\x48\x0F\x30\x28\x97\xAA\x97\x11\x94\xAE\x3D\xF5\x7D\xFD\x40\xF1\x27\x14\xEF\xB1\x58\x6E\xD6\x7F\xF4\x1E\xA1\x0E\x64\x55\xC1\xED\x5D\xDD\x4D\x5B\x7D\xCC\x4E\xAC\x4B\xBA\xA8\x00\xF5\x57\x65\x95\x71\x54\x0A\xEC\xF1\xF3\xCF\x30\xA0\x75\xFD\xC3\x5A\x3E\xA4\xBD\xE4\x69\xA7\xA6\x4A\x67\xE9\xE1\x64\xC1\x37\x28\xA5\x39\xC1\x7A\x34\xF4\x35\xA9\x13\x41\x55\x8B\xA6\x93\xFB\xD8\xB7\x31\x32\x66\xAA\xD7\x78\x1F\xC6\x95\xF7\x75\xAD\x15\xFE\x1D\xFB\xC8\x3F\xE0\xE0\x53\x1E\xE3\xC2\x8B\x3F\xE7\xBB\x5D\x78\x92\xA3\xB0\x59\x93\x07\x1B\x9B\xEA\xB6\xAF\x8D\x27\x13\x62\xAA\x38\x9D\x42\x60\xE8\x10\xBF\xF2\xD5\xEA\xF2\x32\xEA\x90\xF2\xF1\xD5\x21\x71\x12\xD8\x2F\xE3\xD7\x0A\x6D\x61\xF3\x23\x8C\x69\x59\x02\x31\x67\x79\x55\xDD\x0F\x5E\x48\xFA\x0A\xBA\xB8\xBC\xFD\xEE\x7F\x37\x76\x22\x05\xF0\xB2\x7E\x1B\x99\x31\xD9\xB3\x3B\xC8\xE1\x4E\xE6\xBB\xD2\xAD\xF8\x1D\x2E\x9F\x9F\x4B\x49\x37\xC1\xBB\x18\x65\xC0\x65\xEA\xED\xC2\xB7\x7F\x3C\xDF\xCD\x70\x69\xF1\x14\xFB\x3D\xCC\xB0\xC7\x9E\x8B\x49\xEE\x53\x8F\x1A\x3F\x17\x4F\xF3\x34\x32\x4B\x57\x76\xD2\x74\xE6\xBD\x4A\xA1\xF5\x87\x5A\xB9\xF7\x1C\x02\x3E\x31\xF4\x2B\x3F\x4D\xDF\xED\x4B\x7A\x5B\x32\xDA\x92\xD1\x96\x8C\xB6\x64\x34\x51\x32\x12\x47\xA6\xE5\xFD\xCD\x2D\x29\x6D\x49\x69\x4B\x4A\x5B\x52\x9A\x37\x29\x6D\x11\xBD\x45\xF4\xDC\x11\x6D\xF9\xE2\x93\xB6\x1F\x15\x04\x7C\x19\xE2\x13\xF6\x73\x92\x85\xF7\xFA\x22\x87\x65\x11\xC9\xA9\xB7\xD2\xE7\x26\x8C\x03\xF3\x8A\x4A\x3F\x01\x3B\xAF\x1D\xB9\x53\x1D\x71\x09\xDC\x89\xE9\x92\x55\xC9\x56\x8A\x6C\x89\x6B\xC1\xC4\xB5\x0B\xE3\xC0\x2C\x46\xFB\xBC\x3F\x62\x94\x8F\xB5\xE4\xDB\x2B\x7B\x39\xF4\x51\xF9\xB4\x24\x55\x73\x06\x6F\xEB\x68\xDA\xED\x7A\x57\xB8\x58\xBB\x5C\x6D\xF8\xD6\xA9\xE5\x53\x1E\xF7\xD8\xB1\x13\xFA\xFD\xFB\xCB\xCD\x0B\x5B\xDF\x29\x78\xFE\x81\x68\x18\xFF\xD2\x97\x6B\x9B\xC4\x00\x67\xEF\xC5\x73\xEF\x97\xE7\xBB\xDD\x6E\x07\xEF\xFF\x77\x23\x9F\xC7\x98\x26\xFE\x01\x91\x91\xA5\x1C\x65\x4C\x5A\xF9\x3A\x79\x53\x2B\x95\x6D\x0A\xDE\xA6\xE0\x6D\x0A\x1E\x6D\x0A\xEE\x9D\x36\x26\xD8\x2F\x1D\x29\x2F\x6C\xEB\x87\x5B\xC6\xD8\x32\xC6\xB6\x7E\x38\xCD\xFA\xE1\xC8\xA5\x57\xE7\x87\x89\x41\xFC\x5C\x96\x5E\x3A\x52\x1F\x77\x09\x66\x20\xF3\x11\x9F\x0C\x3B\x3C\x33\x0C\x33\xB3\x2B\xFD\x95\xEA\x3A\xD0\x85\xFB\x6A\x3F\xCC\xCD\xD4\xC7\xB2\x5E\x23\x08\x13\x99\x70\x98\xDA\xA4\x9E\xA7\x24\x6B\xB5\x5D\x3F\x8F\x70\xB3\xE6\x80\x61\xFF\x83\x55\x80\xD2\x32\x47\xEF\xF8\x6D\xA3\x35\x95\x13\xD8\xF8\x1A\x17\x07\xDC\x15\x6F\x7D\x9F\x40\x25\xFD\x31\xC9\xE8\x7F\xC2\x2C\xAC\xBB\xF1\x17\x39\x9E\x6A\x3D\x6D\x3F\x81\x17\xB7\x80\xE3\xE5\xF3\xB3\xB6\x9F\xAA\x9E\x9E\xBF\xED\xA4\xF5\xFC\xD2\xAC\xD0\x37\xDA\xE3\x74\x2A\x19\x46\x4C\x4B\xD3\x9B\xC9\xFA\xB6\xCC\xFC\xE2\xD8\x5F\x6D\x99\x5F\x1E\x73\x7A\x9C\x41\x16\xE3\x9A\xE8\xF4\x7A\xAB\xEF\x18\x2C\x10\xBF\x8E\x2B\xFA\x0B\xD8\xC6\xE5\xBD\xA1\x65\x0D\xD6\x75\xA9\x7E\x5E\xE9\xBA\x6F\xB4\xCC\x31\x83\x59\xDF\xC6\xD1\x5E\xDE\xE1\x7F\x81\xE2\x34\x65\xB2\xB4\x82\xD0\xEB\x35\xE8\x5E\x7C\x84\x59\xC8\xFC\xC2\xF1\x50\xCA\xCD\xFC\x76\x83\xC2\x68\x6C\x1E\xD5\x3B\xBF\x9F\xC3\x63\x1A\x85\xBB\x87\xB1\xE9\xFF\x9E\xF1\xD7\xE2\x8B\xE9\x4F\x75\x2A\xE0\x1D\xFD\x31\x59\xBE\x63\xD7\xBC\xCE\xC0\xD0\xF2\xDE\x74\xD1\xF4\x7B\x5C\xDF\x45\xD7\x1A\x52\x86\xB0\x1C\x2E\xDB\x5B\xEC\x13\x6C\xD8\xFE\xEC\x45\x1D\xBE\x24\xA0\x2F\x29\xE8\x1A\x81\xA1\xA1\xDF\xFE\xE2\xF6\x68\x91\x69\x7F\x48\xE9\xCF\x67\x40\xE1\x3F\x06\xD3\xB7\x09\x2D\xCB\xD6\x89\x18\xDC\x84\x71\x98\x1D\x34\x12\x93\xF0\x6A\xB9\x03\xA0\x17\x0B\xA1\x68\x74\x4E\x74\x72\x5C\xFE\x1E\xAB\x71\xD9\x91\x50\xC7\x04\xEB\x4E\xDD\x79\x12\xED\xA9\xF9\x9B\x24\xF6\x91\x0D\xBF\xF6\xF0\xAA\xC7\x47\x44\x32\x5C\xDE\xE0\x35\x8C\xB7\x9E\x0D\x7B\xD9\xC5\x90\x61\xBA\xD0\x92\x1E\x3C\xFA\x5A\x52\x2E\xA9\xF4\xF9\x05\x32\x60\x7F\xC7\x90\x48\x96\xCF\x6D\x9F\xAB\xAF\x1D\x49\xF7\xF1\x75\x1B\xC3\x96\xF9\xDA\x35\x6E\x9C\xC8\x74\xAA\x11\x5C\x29\x77\x8E\x98\x4E\x1A\x8F\x1A\x2F\x9D\x38\x9B\xA3\xA5\x83\x45\x5A\x62\xA5\x9D\x12\x18\x29\xDD\x2C\x38\x66\x9C\x74\xE2\x3C\x42\x94\x8C\x5F\x64\x8E\x71\xC5\xD0\x78\xAC\x67\x60\xDA\x6C\x25\xB3\x0B\x85\x26\xBA\xAF\xC8\xF3\x50\x10\xB0\xEF\x0A\x78\x3E\x4A\x91\x1F\xD2\x07\xC7\x0B\xA0\x46\x30\xA6\xFD\xCB\x88\x6A\xCD\x23\xD5\x77\xDD\xD7\x75\x81\xEE\x53\x6D\xB9\x38\xB0\x1A\xB0\xFA\x2B\x53\xB7\x1D\x04\x06\xAB\xD5\x4E\x0F\x1D\x46\x22\x82\x52\x1F\x52\x1A\x1E\xF9\xC7\xBD\x7A\x52\xB8\x89\x10\xA5\x38\xEE\x4F\xA0\xC0\xCE\x29\xDB\xC0\xFC\x17\x32\x6C\xFE\xD8\xE9\x54\x8E\xFE\x09\xDF\xE5\x21\xC1\x01\xBF\x83\xF1\x26\x21\xEC\x1A\xC6\xE9\xBF\xC9\x26\xD4\xEE\x7A\x56\x29\x95\xF8\x92\x94\xEB\x36\x2D\x0F\x5F\x3D\x9E\x0D\x0C\xFC\x3E\xE1\x7B\x4C\x32\x3C\x1D\x5B\xF1\xC8\xD2\x12\x77\xB3\xD9\x83\x61\x3B\x88\xB7\x1D\xC4\x5B\xF2\x20\x9E\xE5\xB5\xBF\x8E\x97\x76\x6C\xFE\xBB\xF9\xEF\x6A\xFD\x77\x4B\xBC\x9B\xE3\x9E\x8F\xE3\xAA\x97\xB3\x2D\xF2\x74\x06\x7C\x59\x6D\x19\x39\xBA\x4D\x43\x8B\xCB\x68\x4B\x35\xCB\x08\x27\xBA\x6C\x96\x46\x21\xF5\x6E\xF3\xDD\x0E\x5B\xA8\x6A\xDF\x0D\x71\x90\xE4\x29\xF0\x15\x41\x33\x32\x3F\xB4\xFE\x13\x5A\x66\x47\x92\x98\x6E\xA6\xA9\x4D\xF3\x64\x73\x19\xBB\x5D\xDA\x1D\x46\x7A\x97\x6E\x69\xFB\xA8\x73\xD7\xCA\x52\x0E\x30\xAB\xAD\x4D\xC2\xAE\x8F\x5D\xAB\x95\xDE\x5E\x74\xAF\x4B\xEC\xC5\xDC\xD6\x71\x3E\x58\xD2\x6F\x5D\x45\x5C\xA9\xE3\xF6\x10\x7F\x15\x9E\xEB\x2A\xF7\xBC\xAE\xDB\x71\xBE\x9E\xDD\x6F\xBB\xCA\xB7\x26\xA7\x1D\x20\xFB\x72\x1E\xDB\x55\xE8\x85\xDC\xB5\x5B\x19\xB5\x9C\xDB\x76\x94\x73\x95\xEE\xDB\x5F\x87\x15\xB8\x71\x47\xE1\x85\x8D\xD1\xB7\x37\xD7\xEC\x7F\x55\xE7\x8F\x24\xD9\x8B\x1B\xD5\xE0\xE7\xAF\xFA\x9E\xD4\x1E\x83\x99\x2A\xFB\x0F\x4E\x0F\xD8\xF3\xD5\xC1\x6D\x1B\xBC\xAC\xC7\x9F\x09\xF9\x76\xF7\x25\x79\xC3\xDE\xFF\x94\xBE\x2F\xD6\x80\x45\xA1\xB5\x56\x6D\xA7\xBA\xA3\x81\x26\x11\x40\xB7\xCC\x04\x6C\x96\xFC\x84\xEE\xA4\x8A\xCD\x60\xBB\x06\x4F\xA6\x0A\x9C\x32\x32\x1C\x85\x98\xC1\x7B\x6D\x71\xD3\x5D\xEC\xCE\x17\x50\x35\xCC\x18\x9B\x92\x41\xFB\xA5\x52\x70\x3F\xA5\xF5\xBF\xD9\xD2\x36\x00\x28\xAF\x74\xB3\xE6\x67\x81\xB8\x78\xC3\x10\x00\x7B\xA1\x0B\xA3\xDE\x2A\xD4\xD7\x9C\xDD\xD2\x4C\x0F\x06\xE2\xEB\x02\x03\x68\x34\xEF\x09\x0C\x77\x9C\x5E\x37\x14\xB5\xF9\x91\xDF\x7E\xEB\xD0\x12\x2E\x65\xBE\x82\xC6\x49\x9A\xA7\xE2\xC5\x31\x36\x37\xB4\xF6\x05\xF7\x21\x7F\xF6\xB8\x1D\xEF\xF3\x44\x9B\x6B\x0E\x72\xCD\x6D\x20\xD6\x33\x10\x93\x7F\x3D\x67\x1B\xA2\x61\x43\x24\xDA\x64\xE2\x6F\x8A\x6C\x43\x35\xDE\x50\x2D\x6E\x93\xD6\x6B\xF7\x3F\xA7\x88\x64\xF8\x33\xA6\x5F\x5E\x16\x43\xFC\x8A\x9D\xF0\xC6\x84\x0A\xE7\xE2\x2D\x1F\xF8\xEB\x4D\xC2\xD0\xD1\x27\x18\x51\xEC\x85\x71\x80\x4F\xCA\x31\x78\xA6\xDD\x75\x10\x7C\x49\xFE\x37\xC7\xB9\xF9\x51\xC9\x5A\x74\xEB\x8F\xD6\xC3\x28\xF6\xD5\x5F\x10\xE0\x53\x1E\x33\x66\x1F\xE2\xD7\x0F\x9D\x9F\x00\x8D\xA2\xF9\xC2\x49\xEC\x73\x92\x72\xF8\x98\xCC\x2B\x34\xE4\x52\x9F\xF2\x98\x45\x21\xC0\x56\x0C\xFD\x51\x5D\xA8\x2B\xC3\xB1\xED\xDC\x8B\xFF\xF4\xE3\x33\x82\x90\xE3\xAF\xD2\x88\x42\xE1\x0C\x53\x46\x5A\x22\xFA\xE9\xCF\x3F\x12\xFF\x1B\x26\x4D\x5C\x1C\x92\xE4\x5B\x76\x75\xF5\x4F\x4C\xDF\xDE\x5C\x33\x29\x58\x2F\xD6\xFF\xBF\x93\xE4\x9B\xF8\xC2\x88\x33\x65\xC7\xD5\xA2\xCF\x18\x11\xFF\x70\x73\x73\x23\x1B\x8D\xB5\x7E\x44\x04\x1D\x33\x46\x8D\x3B\xA8\x60\x84\xDF\xE3\x28\x8C\x71\xC0\x11\xFF\x48\x92\xF4\xE5\x0E\x45\x19\xE6\x56\x51\xFF\x7C\x65\x25\x7E\x31\x25\xF1\xFE\x21\x38\xBD\x54\xDD\x5D\x77\x5A\xA1\xDC\x5F\x7D\x9B\x54\x8C\x6A\x65\x6A\xAE\x01\x68\x0F\xA6\xB9\x44\xE9\xB1\xE2\x3C\x97\x43\x58\xDF\x69\x95\x32\xC9\x97\x89\x32\x49\x79\xB6\x7A\x92\x44\xD2\x4E\x7B\x81\x3C\xE2\x2A\xD4\x9C\x69\xC4\x41\xA6\x19\xB2\x88\x83\x14\x83\x93\x48\x37\xEB\x4F\x99\x43\x3A\x49\x32\x67\x0A\xE9\xE6\x0C\xCE\x19\xE4\xCB\xA8\xB5\x48\x21\xE4\x54\xA5\x48\x27\xDA\x33\x05\x6B\x37\x99\xA6\x0A\xD6\x6E\x52\xF4\x0B\xD6\xDE\xD6\x1F\x3D\x58\xFB\x4A\x32\x79\xB0\xF6\x76\x06\x53\xB0\x4E\x6E\x97\x61\x0F\xCB\x55\x06\x19\xB5\x06\x69\xB4\x18\xBD\x04\xE9\x42\x7A\xF6\xFC\xB1\xE0\x5C\xDF\x49\x88\xA1\xD9\x63\xE1\x99\xBE\xA7\x20\x33\xE6\x8E\x71\xE6\xF9\xA9\xAD\x32\x46\xE6\xF8\x32\xC2\x3A\x88\x92\xF4\x46\xAD\x3D\xFA\xD2\x9E\xF4\xE9\xA5\xB7\x50\xCD\xFA\xEA\x94\x5C\x8E\xF8\xE8\x1F\xC8\xF4\xD6\xAD\xAE\xAE\xC5\xBB\xF0\xC4\x6E\x65\xF1\x6E\x48\x12\xD3\xEB\x38\x78\xCD\x0F\x37\x97\xA7\xF2\xD8\x66\x92\xBE\x7C\xD8\x21\x51\x0E\x93\x73\x9A\x19\x64\x80\x13\x8C\x38\x85\x0C\x71\x12\xD7\x39\x64\x0C\xEB\x8F\x33\x89\x8C\x20\xC9\x34\xB3\xC8\x18\xCE\xE0\xFA\xB8\xF8\x65\x84\x05\x27\x50\xDE\x71\x8A\xBD\x9E\xA4\xE7\x4F\xD8\x4E\x93\xFC\xD0\x7C\xED\x56\xCD\x0D\x49\xD7\xAE\xA6\x5D\x38\x5B\x2F\x5B\xEE\xF7\x77\x80\xA9\x73\xF5\xB8\xE5\xFE\x08\xA6\x9F\x30\x53\xAF\xA0\xDC\x1F\xC1\x11\x9C\xF3\xF4\x08\xCB\x7A\xF2\x43\xC1\x44\x95\x75\xD7\x35\x93\x5E\x39\xB1\xF3\xFA\x58\xE7\x9C\xD8\x63\xED\x65\x89\x9C\xB8\x9A\x25\xD4\xFE\x0E\x30\x49\x4E\x9C\x6C\x01\x75\x04\xD3\x8F\x9D\x13\xD7\xB5\x7C\x3A\x82\x23\x0C\x58\x02\x19\x64\x95\x51\x96\x40\x46\x58\x3C\x85\x94\x18\xBB\x9C\xEE\xB8\x3C\x35\x30\x4B\x4F\x54\xB8\x76\x5F\xE5\x5A\x36\x47\xAF\xA5\x6C\x5D\x6E\x91\xBA\xBF\x73\x0C\x49\xD0\xCB\xD6\xAC\xAB\x5A\xA1\x1E\xEE\x04\x23\x67\xE7\x79\xD6\xA7\xAF\x63\x14\x3D\xFC\x85\x39\x69\xD7\x11\x18\xEB\x0C\x68\x2F\xE6\x93\x9E\xAD\xEC\x25\x91\x4B\x60\xA8\x84\x2A\x4C\xFE\xB5\x81\x8F\x21\xF6\x31\xF4\xBE\x80\x19\xDC\x8C\xA9\xE9\xAF\x52\x5D\x7E\x55\x77\x99\xB8\x45\xC3\xA8\x8E\xA3\x0E\xB2\xB3\x29\x2E\x84\x19\x84\x1F\x36\xAD\xD0\xB5\x17\xFE\xE5\x37\x1D\x7C\x14\x45\x5E\x12\xFB\xE0\x3B\x29\xCF\xEF\x93\x08\xD1\x30\xC2\xA5\xC2\x65\x3C\x55\x7F\x7E\xBD\x7C\xFE\xF5\xF2\x45\xC1\x5E\x88\xB5\x42\x18\xE0\x25\xF5\x9A\x8F\x97\x92\xE4\xF4\x20\xBC\xC3\xC2\x3F\x57\xDF\x41\xFE\xAB\xAB\xFF\xE7\x3D\x7F\x22\x0E\x11\xBB\xE4\xE0\xC9\x93\xEA\xAB\xFC\x95\x28\x4E\x2F\x48\xAD\x45\xD6\x66\x6C\x87\xDD\xDE\x30\x83\x42\xBF\x9C\x91\xF1\xDD\x64\x3D\x23\xE3\x5F\x9E\x91\xF1\xDD\x64\x1D\xCD\xF8\x02\xE2\x5B\x56\x08\x08\xF4\xDE\xDE\x5C\x2B\xCC\x9A\x6B\x4B\x9C\x3A\xB6\x1F\xBA\xE7\x04\x78\x66\x16\x08\xF4\x9E\x5E\x04\xA2\xFC\x53\x3A\x6D\xE6\x50\xE9\x3D\xD5\x26\xA2\x96\x49\xC7\x6D\x64\x1F\x8B\xB6\xEB\x2F\x05\x1E\x8B\xA5\xE5\xB7\x60\xF8\x41\x2E\x14\x46\x39\xC1\xEA\x3B\x30\x8E\xAF\x16\x82\xEF\xA0\xAF\xFB\xB5\xCD\x21\x9F\x4D\x58\xB5\x32\xCE\x77\xE8\xAF\x51\x8B\xB3\xF3\x22\xE1\xA5\xC5\xF7\x45\xD7\xF7\xDA\x94\x29\xDF\x21\x63\x44\x12\xC2\xED\x9A\x10\xF4\xF0\xB2\x41\xFF\x72\x20\x18\x05\x4F\x5F\x5D\x5D\x11\x9C\xF1\xCF\x2B\x89\x4F\x69\x46\x92\xCE\x97\xE3\xB8\x50\x30\x4C\xB2\xC6\xAE\xDA\x8B\xCD\xCE\x5A\xC1\xEF\x45\x77\xE9\xEE\xA8\x76\x57\xC2\xD2\x3D\x3E\x03\xA4\x32\x98\xB2\x2B\x45\xD5\xF5\x7E\xBC\xB7\xDC\xFB\x06\x6F\x1A\xAA\xFD\xC4\xB0\x53\x3F\xC2\x24\x48\xD3\x12\x43\x02\xA6\xF9\x7B\x4B\xAE\x82\xB7\x89\x0C\xE7\x8C\x89\x85\x77\x95\x93\xAD\x0A\x71\x62\x77\x75\x9D\x60\x1D\x66\x75\xAD\x48\xFE\xCE\xA2\xB9\x88\x90\xEA\x18\xA1\x87\x48\x1C\x88\xAD\x95\x4A\x09\xDE\x8E\x3E\x64\x0C\x5B\x3E\x6E\xB8\x52\x33\x74\xFC\x40\xE2\x4A\xB5\x90\xD2\xE6\x4A\x65\x14\x8A\xED\x77\xC7\x94\x3E\x94\xDF\x30\x6D\xAB\xEC\x15\xF2\x86\x6B\xFD\x27\x71\x5C\xFB\x77\x5F\xA6\x61\xD9\xFB\x13\x2F\x93\x8B\xE3\xF8\x35\x97\x69\xE4\xB0\xDF\x7D\x2B\xBB\x9A\x2B\x53\xE3\xE5\xF8\x22\x8E\xE5\x53\x1B\x67\xAA\xA9\x74\xAB\xEF\x23\x54\x15\xBC\x85\xFB\x31\xEB\x69\x1C\xD0\xF6\xAF\xA1\x4C\xA3\xAF\x5A\xFC\x7C\xA6\x38\xED\x34\x07\xF5\x98\xAD\x3A\xCE\x22\xC0\x8C\xB9\x46\x29\xA1\x52\x72\x8D\x72\xAE\xB8\xEE\x10\x2D\x37\xDD\xC2\xA7\xDA\x45\xBE\xA2\xA7\x4F\x91\xB9\x98\xB0\xDA\x52\xCA\x62\x92\xAC\xDA\xE7\xD7\x60\x20\x30\x3D\x2C\x3A\x5A\xDA\x81\x2E\xE7\x33\x5C\x2B\xD0\x60\x91\x47\xE2\x25\xE3\x5C\x9D\x01\x97\x0F\xF4\x95\x4E\x1F\xBD\xDD\x5A\x0D\xD1\xFE\x84\xCA\x03\x99\x03\xA9\xC0\xDD\xFB\x9D\xBA\x14\x68\x71\xCF\x79\xBF\xA2\x9D\x3A\xCD\xB3\x80\x05\xF4\x33\xD3\x60\xA9\xF0\x3C\x33\x33\x35\x0B\xD7\xFA\x9A\x7F\x18\xD3\x57\x85\x12\x96\x63\x8A\xF5\x1F\xE0\xF5\xDF\x63\x51\x34\xD0\xB1\x5C\x2A\x3B\x1D\xD3\xCF\x98\xBE\x3B\x85\x19\x0D\xE3\xBD\x40\xCF\x44\xC5\x24\x4D\xDB\xA6\xF5\x28\xCC\x0D\x5B\x11\xF3\xAF\x08\x39\x7E\x15\x6B\x72\x31\x86\x7F\xF4\x6A\x4E\x11\xBB\x7E\xD3\x6A\x72\xD9\x66\x72\x25\xFB\x92\xDB\x7C\xBE\xD4\x22\xC7\x2A\x9C\xC9\x5D\xC6\x05\xBC\xA9\x45\xB8\x29\xDD\xC9\x6D\x59\x73\x62\x5F\x72\x14\x62\x39\x47\xEA\x2E\xE0\x5C\x5E\xE4\x28\xD9\x2C\x2E\xE4\xB4\x62\x3C\x97\x2B\xB9\x09\xB3\x02\x97\xEA\x2C\xE8\xEC\xAE\xE5\x26\xA1\xF6\x38\xCA\x8B\xFA\x0F\x31\xFE\x88\x32\x87\x1D\xD9\x79\x0B\x7B\x71\x39\xF2\x0C\xC4\x15\xFA\x0D\xDA\xE5\x3E\x17\x7D\x8F\xF8\xE8\xA7\x0F\xA2\xB8\xBF\x67\x8D\xA8\x9D\xCE\x8A\xC1\xDD\xF5\xF1\xD7\x19\x00\x4B\x2C\x2A\x92\xF0\xF8\xD1\xE3\xA5\x35\x98\x66\x13\x9D\xF7\xD8\x2F\x42\x53\xD0\x8C\x0F\xA8\x29\x86\x15\xC4\xB2\xC4\x08\x99\x30\x5F\x2F\x5F\x88\x69\xE2\x3B\x41\x29\xFB\x0E\x8E\x4E\xBD\x74\x97\x57\x86\x15\x1A\xA5\x51\xEC\xF3\xA4\x45\x33\xD7\x93\xFF\xC0\x60\x35\x6B\x8A\x30\x8A\x3A\xC1\x9D\xA3\xE9\x80\xA9\xF1\x2C\xD5\xE8\x36\xA9\x9E\xA5\x8A\x65\x7E\x3A\x73\x2D\x1C\x8B\x8A\xB3\xD4\x4D\xAF\x78\x29\xC1\xE5\xE6\xCB\xA7\x3C\xC6\x9F\x50\xBC\xC7\xC2\x54\x54\xB7\xFD\x81\x8B\xA4\x62\x2D\x93\x6A\xDC\x52\xB5\x00\x67\x94\x24\x0F\x8A\x6E\x05\x3F\xE1\x83\x62\x0A\xD3\xF2\x73\x62\xEA\x41\x8E\xB5\x0A\x79\x1E\x96\x6C\x9F\x60\x3E\xE1\x3D\x3E\xA5\x57\x57\x1F\xD9\x94\x4D\x11\xC5\x57\x57\x1F\xF3\xEC\xC0\xDB\xAB\x32\x88\xFF\x6A\xCE\x20\x40\xBD\xFE\x85\x1E\x6E\xF1\x9B\x24\xF6\x11\xE5\x65\x8C\xB8\xC0\xA8\xF7\x2A\x2B\xB6\x41\x62\xBC\x39\x20\xF2\x26\x42\x59\xF6\x3A\x0F\xA3\x00\x93\x02\xA3\x98\x30\xAE\x6F\x93\x7B\xF5\xDD\x82\x61\x8A\xBE\xC5\x3E\xC1\x3B\x71\xBD\xBB\x33\x3D\xE7\x02\x6D\x98\xA4\x10\xD6\x87\x74\xDE\x51\x78\x93\xC4\x14\x85\x71\x36\xEA\x10\xBC\x49\x8E\x69\x4E\xF1\xE7\xF0\x98\x46\xB8\x6D\x24\x7A\x3A\xA3\xCC\xF1\x3A\x08\x8A\xD0\xFA\x92\xC8\x24\x26\x60\xEB\xFA\x96\xC3\x64\xAA\xBA\xFB\xF7\x1F\xEC\x63\x91\x51\xB3\xF7\xC0\x7D\xE0\xC1\x8F\xF0\x4D\x12\x31\x83\x39\x0E\xBB\x46\x69\x70\x84\xC0\xB2\x4D\x19\x19\x06\x6B\xE8\x11\xA1\xB4\x0C\xB0\xB4\x46\xBB\x70\x53\x9E\xFD\x95\x0D\x9D\x41\xD6\x9A\x3C\xB5\xBB\xB3\xB7\xE5\x08\x3B\x0F\x9E\x2C\x3E\x00\x83\x3E\x82\xB7\xD9\x88\x4F\xEE\x78\xCE\xCC\xC7\x31\xE4\xDB\x84\x76\x5D\x67\x68\x27\x35\xB9\x91\x0C\xAC\xC6\x36\xC9\x88\x31\xDF\x4A\xB4\x6F\xB0\x7F\xC2\x29\x46\x14\xF2\x17\xE7\xB3\x0C\xA3\x94\x30\x23\x48\x31\x43\xB9\x33\x96\x94\xFA\xF8\xFD\x13\xD3\xBA\xD1\xAD\x9C\x1C\x41\x96\x73\x78\x56\x99\xD7\x7D\x3B\x3F\x0D\x8D\xED\xB7\x9D\xAA\xD9\x4A\x04\x1A\xD2\x30\x89\xB5\xD0\x9F\x3B\x82\x87\x8B\x31\x4B\x08\x8F\x24\xE6\x38\x31\x3C\x5C\x98\xC7\x11\xC4\xA3\xBA\x70\x8F\x28\x1E\xD9\x77\x17\x0A\x63\x89\xC5\x9F\x28\xFA\x86\x49\x79\xD0\xAB\xF8\x21\xEC\xB3\xC8\x15\x8D\xAF\x7E\x11\x79\x71\x71\x7F\xF0\x5F\x90\xE5\x86\xA9\xD5\xF9\x91\x7A\x74\x76\x13\x31\x02\x4F\xCF\x17\x64\xD8\xA8\x31\xB2\x2D\x21\xAF\x22\x3F\x77\x3B\x46\x3F\xBE\xA1\x5C\x5E\xD2\x69\xF7\x9A\x61\x23\xDD\x46\xAA\x07\x11\xDB\xED\x0D\xA3\x0C\x14\x74\x07\xC5\x12\x4C\x7B\xA6\xE2\xB7\xC9\x1F\x78\x47\x3F\x22\x82\x63\xFB\x36\xFB\xC0\x72\xA5\x03\x9F\x49\xEB\x91\x0E\x72\x88\xDB\xFC\x63\xAA\x34\xFC\x89\xB7\x03\x67\xE1\x90\x00\x4C\xE9\x3F\x98\xD0\xD0\x47\xD1\x6B\x44\xDA\x1E\xCF\x47\x5F\x79\x82\xB8\x0F\xF4\xB2\x16\x75\xC6\x77\xA7\x0E\xF6\xEB\x3B\xD8\x1D\x58\xBC\x4D\xDE\x24\x51\x84\xD2\x0C\x03\x6B\x51\x03\x78\xB8\x2D\x49\x5B\xB9\x0F\x1D\x59\x1B\x69\xA9\xDB\xEF\xF1\x70\x59\x47\x57\x7E\x84\xA7\x17\x3B\x83\xF3\x7E\x22\x71\xF6\x9C\x4E\x53\x9B\xF3\x08\xF3\xA4\xF6\x81\x5C\x47\xAC\x7A\xA2\x40\x0F\x25\x43\x3C\x05\x5F\xA5\x85\xB2\x87\x53\x79\xDF\x49\xD6\x0E\x4F\x39\x76\xBA\xAD\x93\xC3\xA7\x70\x7F\xA8\xE7\x99\x96\xC4\xA3\x65\x90\xFE\x84\x5D\x03\xAE\x8B\x78\x3D\xF3\x22\xCC\x62\x68\x20\x6B\x54\xCF\x3D\x7A\x35\x85\xFA\x86\xAC\x7D\x48\x3B\xFA\xBE\xBB\x7F\xF4\x9D\x9A\x6F\xC2\x38\x64\x77\x38\x8E\x19\x1C\x66\xA2\xAE\x81\xE1\x2A\x56\xCF\xA0\xD0\xC9\x0F\x0D\x08\x89\xE2\xB9\x07\x83\xA4\x8C\x7B\x20\xF0\xE3\x32\x7F\x60\x14\x84\xF1\xBE\x2E\xB0\x0D\x4F\xF1\xDD\x9F\xB4\x3B\x92\x97\x8E\xB5\xF6\xA4\xE1\xE8\xBB\x43\xA8\x7E\xFE\x8E\x5A\x43\xF6\x06\xF9\x34\xA9\xE7\x76\xBE\x7C\xD6\x6B\x2A\xFF\xED\x99\x76\x99\x7A\x79\xFC\x90\x75\xB9\x21\xE8\x88\x5B\xBC\x90\xE1\xD4\x6B\x3C\xF8\x98\x46\xC8\xC7\x7C\x9D\x27\x8B\x92\xEF\x5E\x8A\xE8\xE1\xA5\x24\xDC\x93\x5A\xBA\x27\xAF\x54\xC1\x05\xD8\x94\x5A\x4B\xA0\xBA\x46\x7A\x9F\x70\x16\xBD\xE9\xCE\x23\x2F\xAF\xED\x56\x2A\xA4\x46\xEF\xF7\x63\x1A\x5D\x5D\x7D\x4A\xF2\x38\xB8\xEC\x4B\xDB\x37\x7A\xE8\xE7\x34\x0A\xFD\x36\x17\xE5\x48\xCC\x47\x27\x8D\xA5\x56\x03\xFC\xFA\xC8\x0D\xE0\x3E\x35\xCC\x69\xF5\x5F\x56\x62\xF5\x55\xA7\xDA\x7E\x5F\xDE\x58\xAB\x06\xBA\xF7\xAD\x5B\x74\xE1\xB9\xF5\x0C\x12\xA8\x3C\x1D\x95\xB7\xC5\x03\xE5\x0D\x58\xC3\x9C\x9D\x7E\xFA\xEA\xDB\xD9\xA9\xA0\x57\xAA\xE7\xA0\x82\x54\xB0\x9F\x83\xC0\xCE\xEB\xC1\xE7\xA0\x8C\xF2\x40\xA3\xAF\x24\xB8\xEE\x32\x9C\x83\xB2\xE2\xCE\xD8\x39\xC8\xDB\x5E\x68\x9D\x83\x16\x8E\xB3\xDE\x4A\xAA\xE6\x3E\x07\x71\xCF\x58\xAD\xB6\xA3\xC0\x67\xA3\xDA\x75\x10\xDC\x24\x51\x80\x6B\x75\x40\x85\x8D\x37\x10\x9E\xA5\xCE\xFD\x26\xFE\x55\xA9\xE0\xBE\x00\x7B\x36\x2A\x55\x7C\xDE\xE3\xEF\xA2\x22\xB2\xBA\x7A\xED\xDC\x79\x56\x5D\x8B\xD2\xA3\xD4\x43\x6B\x51\xC6\xB5\x44\x58\x8B\xBC\xC3\x4A\x84\xB5\x68\x71\x3E\x3B\x27\xE7\x60\x4D\xC7\x82\x6B\x25\x0B\x66\x72\xD2\x2C\x5F\xE8\x13\xCF\x69\x75\x5D\xF3\x3E\x3B\x9D\xBB\xED\x23\x9D\x8F\x92\xA3\x4C\x0C\x6B\x51\xC6\x75\x62\x58\x8B\xBC\xC3\x26\x86\xB5\x68\x21\xA4\xB2\x51\x0A\xFC\xBE\x8F\x41\x23\x32\x1F\xF1\x41\xC5\xAC\x47\x75\x95\x99\xE1\x6D\x0B\xCB\x16\xCB\xCA\xC5\xEB\x3E\x74\x8C\xC8\xBF\xE3\xD0\x4F\x02\xFC\x4F\x92\xE4\x29\x70\xF4\xB8\xAD\x1A\xAF\x81\x06\x3B\x08\x1D\x3F\x53\x44\xF3\x8C\x39\xEF\x2E\x8F\x22\x92\xC7\x78\x69\x39\xFC\x03\x22\x34\x59\x83\x24\x2A\x97\xAB\xAB\x1D\xBB\xBB\xEE\x45\x73\xF7\x91\x3E\x47\x54\xE7\xC2\x57\x21\xFC\xEF\xD9\x7F\x50\x14\x06\xFF\xFE\x72\xF3\xC2\xF1\x7A\x3F\x95\xCE\xD2\x6A\x14\x13\xD9\x31\x5D\x5A\x8A\x8C\x92\x08\xC7\x4B\x4B\x51\x65\xBC\x7F\x4B\x6C\x6D\x24\x2A\x4C\xE9\x83\x0B\xCE\xA5\xE9\x70\xEF\x51\x92\xCA\x08\x04\xD5\xEC\x30\x8B\x39\xDA\x92\x79\x9F\x09\x62\x05\xE2\xDB\x16\x89\x57\x61\x57\x4B\x7D\xB3\x9C\x70\xEF\xF1\x9E\x1F\x5D\x5F\x87\x38\xD7\x41\x60\x5E\xB7\x6B\xF8\xCF\x2A\xEA\xF9\x2C\x90\x08\xE9\xE7\x4B\xD2\x5C\xE4\x24\x7D\xEB\xCB\x96\xD6\x8D\xA9\x6E\x44\xC2\x6A\xCA\x93\x6C\x5E\x9D\xAC\x2E\xFE\x7E\xF3\xA6\x18\x4D\xE4\x53\x4C\xCC\x73\x92\x59\x08\xA7\xFC\xDB\x4C\x81\xEF\x32\x1F\xA5\xD8\x99\x91\x42\x49\x0D\xE9\x45\x95\x92\x87\x6E\x51\x51\xC0\xEA\x77\x2C\x53\x43\x35\xFE\x68\xB4\xA7\x95\x3C\xCC\x50\x94\x1E\xD0\x14\xA4\x47\x8E\xD7\x36\x27\x52\x93\xB5\x95\xEC\x53\x6B\x66\x18\x76\xB5\xC9\xD8\xD2\x48\x9C\x86\x1C\xCE\x98\x4F\xDA\x11\x6E\xB0\x9A\x49\xAE\x5E\x97\x60\x8D\x2D\x9B\x6B\x9E\x1C\x91\xA5\xFB\xE4\x37\xF6\x18\x2C\xF6\x9C\x37\xA3\xA2\xE2\x73\xED\x2C\x0C\xC5\xC7\xF9\xF9\x93\xD2\x12\x33\xFA\x7A\x52\x47\xFF\xA7\xD4\x25\xA5\xD7\x1F\xB7\xE6\x93\x66\xC6\xB5\x96\xF9\x83\xC1\xF1\xFE\xDB\xE2\xEF\x8F\x98\x44\xA6\xCF\x8D\x8C\xB5\x52\x3A\x90\xF9\x58\x2B\x9D\xE3\x88\xF1\x06\xA5\x34\x27\xF8\x3D\x3B\xC1\x6F\xBD\x3A\x64\x14\xAE\x50\x77\xF7\xDB\x4B\x86\x8A\xE0\x58\x15\xB4\xD2\x71\x9E\xEA\x5B\x29\x0D\xAF\x8B\xBB\xD9\x7D\x92\x62\xB7\xFF\xD0\x77\x7C\xCD\xBA\x53\xBC\x18\xEA\x9B\x91\x1E\x4D\x26\x11\xA5\xF0\x28\x9A\x14\xDE\x35\x17\xC7\x66\x3D\xEA\x16\x65\xA1\xEF\x65\x8C\xF8\x4B\xE8\x93\x19\x45\x9B\x47\x09\x0A\x69\xF6\xB2\xFC\x52\x86\x69\x11\x4C\xF8\x90\x06\x4A\x53\x2C\xE4\x57\xA7\xEF\x29\xCF\x3A\xE6\x2E\xB7\x11\x4C\x22\x83\x25\x19\x4D\xC2\xCF\x9A\xB4\xA6\x71\x67\x5B\xD8\xAB\x37\x8C\x2F\x2E\x94\xFC\x9E\xFE\xE2\xE2\x38\x65\xD6\x45\xA4\x1C\x6D\x0E\x99\xDF\xE9\x96\x9E\x6B\xA6\xA9\x81\xE6\x16\xD5\xE5\xF2\x92\xB9\x65\x52\xAF\x76\x59\x5C\x24\xE9\xC2\xF8\xE5\xA5\x19\xF9\x01\x6D\x79\x85\x46\xBA\xF8\x7B\x56\x45\x18\xEA\xEF\x31\xC5\xFB\xF6\x55\x9C\x75\x98\xD8\xF2\x25\x8B\x59\xE5\x1B\xB0\xFC\x3D\xF1\x70\xAE\x64\xDD\x75\x7A\xD7\x68\x59\x85\x99\xE8\xF1\xC4\xB0\xEA\x3B\xF1\xA0\x4E\xB8\x63\x3C\x9D\xE4\x4B\xAD\x01\x2E\x93\xF8\x07\x5F\x7C\x5E\x69\x52\x91\xE2\xD7\x44\x5F\x5D\xFD\x50\x9B\xA0\x6B\xD3\xDA\xEF\x49\x77\x25\xEB\xF8\x59\xCB\xD9\xEE\x13\x07\xBE\xEB\x3A\x1F\xEF\x6E\x1F\x63\x5D\x44\xAE\x75\x0C\x92\x78\x8E\x48\xFF\x88\xBA\x4C\xEF\x69\x27\xEE\x4F\x4B\xDE\xDA\x77\xFA\x15\x34\xF6\xA9\xFE\xC7\xAC\xBB\xF4\x21\xF9\x9F\x42\xF9\x27\x3F\xED\xB0\x3F\x71\x19\x74\xE9\x29\x7C\x09\x0B\xE8\x5F\x09\x5E\x30\xF4\x81\x79\x62\x59\x79\xBA\xCD\x1D\xAB\x91\xD5\x36\x9F\x2C\x2B\xE4\xAC\xEE\xD6\x9A\x6D\xE7\xF6\xB7\x76\x81\x56\xE4\x70\x9D\x84\x5D\xCC\xE3\xDA\xA5\x9C\xDE\xE5\x9C\xE7\xB8\x59\xFC\xCD\x5D\x9A\xA5\x9D\xAD\x97\xA4\xF3\x7A\x9A\xBB\x88\x33\xBA\x99\x6B\x41\x31\xAF\xBB\x39\x4B\xB5\x1A\xB7\xEB\x23\xF1\x42\xEE\xE7\x2C\xAA\xF0\x12\xF6\x47\x92\xEC\xAF\xAE\x7E\x8F\x33\x5A\xFC\x3F\xA4\xD7\x11\x95\x8D\x5D\x2F\x6E\xA8\x6D\x85\xC6\xC8\xB3\x90\x7A\xFD\x40\x31\x7C\x20\xD7\x1F\x44\xB7\x3C\x39\x24\x52\xED\x4C\xE3\xDD\x31\xA5\x0F\x7F\x86\x01\x3D\x54\xF6\x62\x2D\xD5\x12\x7E\x67\x7A\xFF\x42\xD4\x3F\x88\xDF\x2C\xB7\x62\xBF\x4F\x52\xC0\xC8\xAD\xDD\x6E\x50\x18\xD5\xEB\x02\x00\xE6\x8F\xE2\x1F\x71\xD1\x88\x37\xBF\x65\xDE\xF8\xF6\xE6\xBA\x52\xF5\xED\xCD\xB5\xF0\xDE\xAA\xD6\xB5\xFD\x8C\x02\xEF\xF3\x21\xA5\xE1\x31\xFC\x0B\xF8\xEC\x97\xFD\x38\x0B\xD8\x9B\x2F\x9D\xA5\x6C\xBD\x0D\xD3\x2F\x2F\x0B\x86\xAF\x98\xF2\x98\x50\xE1\x5B\x72\x7C\x0D\x4D\x5D\xE8\x34\x08\x04\xD8\x48\xC1\x69\x83\xB7\xDB\xA2\xF0\xF2\x7F\xA1\xB4\xDE\x1E\xF8\x17\x22\xDF\xF4\x13\xE8\xAA\x30\xCE\xBD\x1C\x6D\xEA\x48\x4F\xBC\xF3\xA2\xBB\x08\xC3\x2C\x60\xD3\x1E\x93\xBD\x61\xD4\xEC\xA8\x7D\xAD\x23\x10\xB1\x9A\x04\x66\xE6\x6A\x07\xA1\xB7\x45\xA3\x4F\xD8\x4F\x22\x76\x79\xFD\xD0\x41\xD7\x28\xD9\x74\xB3\xB1\x75\x54\x50\x23\x61\xD1\x92\xFD\x21\xEB\xE2\x1F\x10\x91\x77\x1E\xDB\x06\xDF\x8D\x88\x93\xAC\x43\x18\x1C\xF1\x31\xC3\x52\xD6\x28\x3F\x07\x53\x92\x13\x53\x9A\x5B\x70\xB4\xD2\x02\x73\x99\x01\xB3\x53\x1A\x06\x69\x88\x7E\xE3\xC6\xCA\x3A\x51\x74\xB0\x4E\x13\x32\xAD\x04\xDC\xCC\xD1\x7F\x98\x45\xD2\x37\x11\xA2\x14\xFA\xA4\x90\x8B\x89\xE5\xCE\x02\xA0\xF0\x85\xCF\xB9\xEF\xE3\x2C\x4B\x48\x73\x36\x82\x4D\x82\xD7\x84\xA0\x07\xB1\x1C\xB4\xC1\xB4\xFB\x79\xB4\x86\xDA\xDF\xA0\xAA\xB2\x59\x06\x34\x61\xF4\xA5\xF7\xBC\xBA\xF8\x47\xD1\xA0\x99\xE2\x4D\x1A\x38\x31\xB0\x0F\x93\x74\x0F\x35\x84\xA0\xDD\x2D\xEF\x79\x59\x42\xE8\x4B\x5E\x63\x3C\xFF\x7A\xF9\xE2\xE9\x13\x76\x08\x1F\x1E\x98\xA2\x3A\x09\xF0\xE9\x3F\x28\xCA\xC1\x7D\x47\x47\xFC\xDF\x1C\xBB\x3C\x7D\xE5\x22\x88\xCD\x5B\x00\xBC\xD9\x55\x75\x8F\x8C\xB7\xC9\x31\x8C\x79\x98\x29\x1F\xF7\xDC\x22\x64\xBC\x08\x51\x26\x32\xE3\x88\xBC\x3B\x86\xF4\x8F\x30\xA3\xAE\x83\x01\x5F\x19\xD6\x3C\xD1\xB4\x3C\xE3\x36\x88\x4B\xDA\x47\x9F\x5C\xFE\x3B\x8C\x69\x26\x5D\xA7\x30\xA2\x7A\x96\xA2\x40\x9D\x80\x80\x6A\xC0\x38\x47\x59\xE7\x66\xB9\x97\x46\x74\x9B\xA2\x46\x70\xB0\x67\x86\x1C\xF9\x19\x5B\x9E\x6C\xEB\x3F\xD4\x13\xB8\xDB\xC0\x8C\x37\x30\x60\x20\x6D\xA6\x1D\xC7\xB4\x9D\x8A\xE4\xCD\xE8\x23\x1B\xDD\x9A\xF5\x37\x6B\x8F\x9C\xD6\x7B\xAF\x59\x6E\x23\x31\xC6\x48\xE8\x0B\x16\x9B\x5D\xC7\xB0\x2B\x9F\x1B\xB7\xE7\xE4\x05\x9E\x93\x9F\xA9\xDB\x43\x9A\xFD\xFF\xF6\x33\x0D\xC0\xF9\x0F\xE8\x16\x46\x8B\x84\x11\x60\xF5\x90\x4D\xD0\x61\x12\x33\xFB\x7B\xBF\x6E\x23\xB0\xF9\xFD\x63\xB3\xFA\xB3\x92\xA5\xC5\xEF\xC3\xD8\x4F\x8E\x69\xF1\x94\xB0\x0D\xC5\x16\x00\x8F\xCD\xEA\xDB\xEE\xCE\xD2\xBB\x3B\xDB\x26\xCE\xC2\xEB\x69\x9B\xF1\x97\x58\x57\xDB\xAC\x3E\xF9\xEA\xFC\x66\xE2\xB5\x2D\x61\x6E\x23\x32\xED\x52\xE6\x66\xDF\x89\xF7\x62\x7D\x82\x11\xC5\x5E\x58\x14\x36\xCA\x21\xC1\x6D\x14\xE6\xDC\x11\x7F\x77\x0A\x33\x1A\xC6\x7B\x21\xF9\xB8\xEE\x87\x6F\xC3\x33\xEE\xBA\xBF\x60\xDE\x47\x7C\xFC\xA7\x6B\x65\xFD\x93\x98\xC2\x5A\xE7\x3E\x6E\x1B\x00\x55\xE7\x63\x56\x78\x8C\x1A\xF0\x31\xDB\x47\xAF\xC8\x1E\xB3\xB6\xA6\x63\xF5\x73\x9E\x7C\xB4\xBE\xEE\x30\xAB\x20\xCB\xB2\xBF\xBA\xFA\x7F\x45\xD7\x26\x2B\xBF\xE0\xAB\x78\x0A\x9A\xF2\x3D\xCD\x33\x91\x16\x1E\xE4\x8F\x04\xEF\xC2\xD3\xB5\xEF\xE3\xC8\xBB\x21\x49\x4C\xAF\xE3\xE0\x35\x7F\xAD\xBA\x5C\xD0\xB5\x5F\xE4\x3C\x11\xD5\x22\x07\x1C\xC8\xC5\xB6\xA9\xB2\x96\x85\xE6\xED\x78\xC4\x63\x3B\x1E\xC1\x3A\x7C\x7A\x77\xC9\xDF\x75\x76\xBD\xDB\xAF\xC0\xFF\x90\x16\xF1\x97\x49\xAF\xB8\x8B\xDE\xE0\xA3\x28\xF2\x92\xB8\xA0\xA0\xD7\xF6\xCF\xEF\x93\x08\xD1\x30\xC2\x25\xD9\xB2\xDC\xAD\xFE\x2C\x8D\xC0\x6D\x50\x95\xC2\x53\x8A\x3B\xD5\x75\xF8\x59\xA1\x77\xC7\xEB\xF0\xC7\xD2\x8E\x43\xA7\xBE\xBA\x71\x64\xA1\xDD\x3F\xCF\x30\x81\x99\xB8\x56\x57\x57\x5F\xF0\x89\x56\x13\xA9\xF4\x29\x98\xB1\x78\x8A\xC5\xED\x14\xA6\x33\x7D\xE5\x60\x12\xD7\xFA\x84\xEF\xF2\x90\xE0\x80\x4F\xF7\x52\xD9\x33\x49\x58\x09\xC9\xF2\xA9\xE6\x98\x4F\xA7\xD6\xF6\xF7\x58\xBB\x17\x77\x12\x3E\x45\x29\x19\x46\xF8\x4B\x52\xDE\x11\x51\xE6\xCD\x19\x52\xC5\xFB\xFC\x58\x5E\xF6\x91\x4D\xA7\x66\x59\xDB\x66\x1F\x62\xFC\x91\x5F\x51\x3E\x3E\x23\xE3\x39\xDA\x7A\x6A\xF2\x52\x92\x9C\x1E\x84\x45\x41\x9A\xA7\x51\xF5\x81\xDE\x91\xC4\x28\x0A\xF4\xE7\x4F\x9E\x94\x6B\x88\x75\x44\x76\xCC\x74\x9F\xDE\x5D\x8E\x33\xC8\x23\x69\x25\x8A\x86\xEF\x31\xC9\x70\x73\x97\x49\xFD\xAC\x31\x79\x39\x30\x68\x6C\x8D\x72\xB3\x47\xAA\x27\x8A\x15\xEA\xD9\x1B\x18\x4A\x38\x70\x65\xFA\xC6\xF8\xFD\xC1\x87\xD6\xE5\x02\x6E\x08\x5D\xBB\x49\x06\xC6\x75\x9C\x1C\x84\x1E\xA2\x51\xD9\xB7\xB2\xC5\x86\xFB\xA2\x96\xF4\xE8\x43\x2A\x36\x4F\x92\xED\xEB\x87\x58\x0D\xD9\xF3\x8E\x28\x2D\x25\x29\x1E\xFF\x10\x99\x49\x98\xB5\x19\x22\xC2\x59\x36\x3D\xF7\x4A\x02\x4A\x72\xEC\xB4\xC1\xB2\x94\x71\xAA\x9B\x55\xDB\x3E\xC3\xBE\xBC\xA4\x3E\xF0\xE1\xF7\xBE\x61\x08\xAC\x35\x4E\xEC\x08\xF6\x48\x04\xE4\x59\x8D\xBC\x2C\x60\x9A\x21\xE8\xE7\xD1\xF3\x2A\x30\xD0\xA9\xE7\xB7\xB6\xE0\xD7\xDB\x4C\xB2\xCD\x24\xDB\x4C\x32\xE3\x4C\xB2\xD5\x6E\x5B\xC4\x6D\x11\x37\x6B\xED\xB6\x45\xDC\x16\x71\x5B\xC4\xCD\x39\xC7\x39\xDF\xFC\xBC\x3D\x35\x6D\x4F\x4D\xE7\xF7\xD4\xB4\x3D\xF8\x6F\x2E\x7C\xE6\x2E\xBC\x65\xE1\xCD\x85\xCF\xDC\x85\x5D\xF7\x4D\xDE\x26\xD5\xB7\x08\x9C\x37\xD5\xAE\x63\xFF\x90\x94\x82\x2B\x1B\x62\x4F\x35\x5C\xB2\x17\x4E\x8F\xE8\x1F\xF1\x6C\xB6\xDC\xC4\xCB\x45\xD7\x25\x99\xB2\xDF\xF9\xF7\xDB\xAF\x7F\x0F\xDA\x76\x3D\x57\x23\x7C\xD3\xC7\x51\x20\x4D\x2D\x63\x23\x2C\xBA\x4A\x1E\x14\x6B\x8D\x96\xD2\x03\xA6\x1A\x69\x20\x6C\xD6\x64\x4C\x76\x28\xE7\x98\xAE\x55\x3A\x87\x7D\xEB\x55\x8B\xCE\xF7\xA9\x3F\x63\x44\xFC\x83\xF0\x55\x94\x16\x47\x6D\x01\x73\xA2\x9A\x44\xD5\xCB\x69\xD4\x3F\xFC\x4F\x18\x07\x56\x59\xD5\x63\x44\xC0\xD9\xFE\xD5\x7A\xAC\x66\xD8\xE6\x34\xCD\x3A\x8C\x2B\xBE\xB2\x72\x06\xE6\x7B\x1D\x52\xF6\xF1\xA6\xCD\x7E\xFD\xEC\xF7\x7E\x45\x71\xBD\x6E\xD3\x69\x65\xDA\x47\x44\x68\x88\x22\x26\xE4\x7B\xF7\x59\xDC\x3C\xB9\xB7\x4D\xD8\xCF\xD6\x57\x3B\xC8\x1F\x9B\xFF\xC1\xFF\x70\x3F\xBF\xE4\xD8\xCF\x50\x8E\x80\xBD\xB5\x07\x58\x8E\x9B\xC7\xD5\x27\xD6\x1A\x4D\xAB\xB6\x3F\x70\x96\x99\x1F\x36\x64\x5C\xC7\xE7\x22\x88\xA9\x76\x6C\x04\x38\x9E\x3A\xD7\x71\x7C\x88\xB5\xFB\xE9\x46\xA8\x37\x70\xAE\x8F\x99\xE3\x16\x45\x28\xF6\xB1\x87\x76\x14\x93\xF2\x25\xA0\x97\xA0\xD1\xBC\x5B\x94\xE1\x97\xA5\x95\x5E\x99\x4C\x2B\x61\xE9\x8F\xFC\x26\xBC\x01\x07\x0F\x01\x6D\x87\x1F\x10\x05\x4E\x0C\x0E\xEC\xBD\xFA\x01\xE8\x2E\x3A\xC1\xC7\xE4\x1E\xAF\x4C\x58\xF7\xB4\x26\x75\x53\x20\x55\xFA\x00\x3A\x09\xA0\x67\xE6\x44\x67\xEB\xD5\x45\x44\x23\x37\xE0\x30\xAB\x23\x33\x7B\xBA\xBE\x0E\x82\x22\x31\x7E\x49\xF8\xA4\x65\xFC\x16\xA4\x03\x7E\xB7\xD5\x89\x0E\xF4\x9C\x14\xF9\x4C\x11\xF9\x18\xE5\xD9\x07\xF2\xBF\x39\xCE\xEA\xD3\xD9\x1C\x5C\x7D\x20\x53\x68\xD2\xDF\x7C\x90\xDF\xD7\xE9\x91\x00\xC6\x17\xC1\x98\xF2\x66\x65\x6C\xF6\xBD\x82\xA9\xCC\xAD\x23\xE9\xD1\x25\xD7\x0D\xB3\x66\xF9\x00\xA2\xEB\x12\xF0\x4D\x12\xFB\x88\x7E\x20\xD7\x11\xBB\x4F\xA1\x79\xB2\x33\x33\x50\xDF\xB2\x36\xB2\xF2\xEB\x17\x36\x7B\xC4\xDA\x5C\x82\x49\x2F\x9A\xCD\xCC\x5B\x42\xBA\x41\x3E\x4D\x6A\x96\x61\x12\x2B\x6E\xE3\xCC\x61\x1D\x23\xEC\x3A\x3F\x2C\x62\xEB\x99\x98\x2E\x35\x10\xE0\xC4\xBE\x80\x1F\xB4\x54\x43\x73\x49\x64\x34\x46\xEF\x08\x5B\x81\x37\xD5\xCC\xDE\x27\x3C\x73\x9C\xB3\x32\xCD\xA7\xD3\x3B\xCC\x64\x7E\x8F\xEF\x51\x37\x6F\xE4\xA6\x58\x1B\x7F\x57\x7E\x43\x18\xFF\x11\x52\x4C\x50\x24\x16\xC3\x1D\x46\x6A\x66\x76\x4E\x85\xBC\xCC\xF0\x3D\xFE\xFE\xE6\x80\xC8\x9B\x48\x58\x64\xAF\x1B\x26\x28\xC9\xDF\xDD\xE5\xEC\x12\x28\xEB\x38\x36\xAF\x20\x7E\x49\x52\xE7\x1E\x7D\x19\x4D\xA8\x03\x54\xAC\xF4\x12\xAC\x25\x33\x77\xA1\x09\x4E\x36\xEE\x04\x9A\xF5\x01\xF1\x06\x20\xA9\x8B\x79\x99\x90\x63\x95\x8B\x84\x9E\x87\xD2\x14\xC7\x2E\x45\x87\xBB\x78\x82\x6E\x5D\x7C\xA7\xDA\x2A\x9D\x49\xBB\x6A\xB5\x7C\x2E\x76\x2E\x2E\x3E\xA3\xEA\x45\x4C\x2C\xA0\xBA\xED\x50\xED\xB4\x42\x68\x11\x67\xBE\x97\xE3\x59\x46\x49\x84\x63\x47\xE4\x8E\xB9\xCB\x42\x49\x4F\x55\x56\x19\x27\x3D\xB7\x74\x75\x55\x1B\xD3\xEF\x70\xCD\x4C\x07\x63\x59\xF3\xE9\x1A\x15\x77\x15\x10\xC8\xED\xDA\xED\x17\x72\x9D\xF0\x27\x8A\xBE\x61\x52\x5E\xAB\x54\xFC\x10\xAE\x6A\xD4\xC3\xA1\x2E\x43\xC0\xBA\xB0\x0B\xAB\x1F\xFC\x97\xB8\x8A\xF5\x26\x89\xEF\x31\xA1\x45\xD9\x92\x7D\x49\x5E\x3F\x50\x46\x46\xBA\x2F\x52\x15\x63\xE2\x01\x79\x3A\xCB\xA5\x4A\x04\x67\xE1\x5F\xE0\x36\x56\xD1\x99\xF3\x59\xB7\x89\x48\x1E\x63\x9A\x14\x6D\xAB\x16\xD3\x39\x5D\xAE\x54\x7E\x31\x49\xAF\x54\x44\xF7\xA2\x75\x7D\xB7\x2C\xB9\x2E\xEF\xAF\x5D\xF2\x45\xD6\x04\x56\x68\x94\x1E\xCB\x0C\xAB\xD3\x61\xC5\xB1\x6E\xB3\xDD\x4D\x42\xD8\x35\xA9\x73\xDB\xF0\x0C\x0C\x56\x2F\xAF\xBC\xCE\xC3\x28\xC0\x84\x2F\xD5\xF0\x83\x30\xEA\x25\xB7\xAB\xDD\xFB\xEF\xA4\x44\xFB\x9C\xD0\x97\x5C\xCB\x2C\x3E\x86\xA5\xD7\x75\x40\x04\x54\xC9\xBC\x82\x57\x62\x09\x81\xD1\xC1\x24\x26\x8E\x9F\x98\xEF\x5D\xDF\x26\xF7\xB8\x39\x1E\x70\x3E\x8E\x0A\x8A\xDF\xC7\x45\xDB\x08\x75\x75\xCE\x4E\x76\x5D\xBB\x5B\xBE\xC7\x7B\x5E\x78\x0C\x34\x8A\x44\x47\xAA\x7F\x9D\x59\x3A\x0F\xA8\x44\xE0\x7C\x4E\x4A\xB6\x28\xF2\x28\x3C\x48\x58\xD7\xD0\x31\xFF\x89\xA9\x98\x02\x7B\xEC\x6E\x3B\xD2\x94\xD6\xCE\x85\xC5\x0E\xBE\x94\x71\x75\xF5\x43\x6B\xEB\xB3\x0C\xD2\x85\xB0\xFB\xE3\xD6\xE7\xF0\x98\x46\xE1\x0E\x3C\xCC\x56\x4A\x24\xAD\x6B\x3A\x2C\x06\xE9\x2D\xE0\x93\x81\x8D\xB3\xF9\x28\x53\x77\x79\x01\x53\x4E\xA5\x60\xE7\xD3\x70\x33\x08\x30\x1B\xEB\x26\xAA\x03\x7C\x97\x97\x39\xAA\xA0\xC7\x5E\x35\x51\x57\xC7\xAD\xE9\xD0\xD8\xEB\x79\xB3\xAE\x1E\x04\xDE\x2D\xF2\xBF\x79\x3E\x4A\x91\x1F\x52\xF0\x58\xE8\x94\xE6\xED\x72\x3A\xC6\xC1\x31\x47\x72\x9D\x36\x9A\x43\xA8\x89\x69\x9B\x0D\x31\x4F\xD9\x53\x8E\xF3\x0F\x91\x93\xBC\x26\x8C\x22\x9C\xF9\xB8\xCA\x84\x6F\x92\xF4\x41\x1B\xCB\xD6\x95\x1B\x95\xCA\xE7\x43\x42\xE8\x7F\xC2\x2C\x6C\x3D\x53\xD0\x87\xFA\xC7\x24\x73\x26\xDE\xDE\x02\xBD\xE4\xB2\x1E\x51\xB4\xF1\x41\x71\xD5\xE4\xBE\x69\xBF\x98\x01\xF5\xE9\x67\x71\x03\xBE\x4D\x0C\xF6\xB3\x5E\x0D\x3F\xA7\xD8\x1D\x56\xD1\x17\x1A\x56\xFE\x6F\xEB\xAA\x27\x78\x0E\x73\x5E\x4B\x4A\x15\xE5\x82\x83\xD8\x5A\x20\x0F\x0B\x73\x79\x78\x3A\x1E\xE9\x19\x1A\x21\x5D\x73\xE6\x30\xFA\x23\x1D\x55\x1B\x49\xB4\x16\x0F\x1B\xA6\xEA\xA0\xD3\x6A\xA3\x29\xD8\x33\x19\x0D\x1D\xE5\xF1\x32\xCC\x30\x49\xF4\x29\xAC\x7A\x72\x1A\x56\x32\xA9\x54\xC6\x2D\x99\x54\xEA\x1F\x09\xEE\x92\xE9\xE4\xD5\xFE\x0E\x6C\x96\x2B\x2C\x16\x13\x65\x05\x56\xD1\x7C\xA9\xFC\xAD\x2D\x5C\xAB\xA9\x7F\x35\x92\x5A\xB2\xBA\x9C\xB8\x5D\x12\xC0\x9C\x5A\x39\xE7\xC7\xC5\xFC\x73\x48\x2A\x9D\xD7\x92\xD2\x54\xBA\xE0\x20\xB6\x15\x6B\x3D\xC2\xAD\x4D\x3B\x37\x12\x3D\x5D\xCD\x85\xF8\x64\x2E\xD3\x99\xB9\xC3\x54\x39\x6E\xD2\x58\x15\xFB\x1E\x6F\x70\x2E\x20\x65\x8F\xF7\x60\xE7\x91\x72\x70\x8C\x4C\x62\xAE\x81\xAF\x57\xCD\x6E\x3A\xC3\x1A\xE7\xAC\x36\x9B\x22\x07\x4D\x22\x68\x9F\x77\x9A\x17\x09\x86\xBE\xD3\xDA\x58\xE2\x58\x9E\x93\x7F\xC8\x0D\x83\xF6\xA4\xDC\x68\x77\xF7\xF6\x1F\x72\xC3\x20\x19\xDD\x68\x3B\x5F\x58\xBE\xDC\x26\x82\xAB\x8C\x0B\xEE\x64\x3D\x3B\xE2\xE3\x31\xB9\xC7\x2B\x91\xA6\xDB\xBB\x42\x2B\x91\x74\xAD\x43\x2B\x06\x41\x96\x46\x21\xF5\x6E\xF3\xDD\x0E\xDB\x09\x6B\x87\x27\xDC\xE4\xA9\x5E\x45\x4A\xF3\xEC\xC0\x84\xA9\x32\xA2\x09\xFF\xC9\x93\xCD\x50\xDE\x8E\x24\x31\xDD\x2C\x65\xB2\xD4\x93\xCD\xA1\x3A\x99\xC9\xC9\x9D\xA4\xCF\xCE\xAE\xC3\x5C\xBC\xF6\x59\x7F\xBA\x02\x26\xCB\xD5\x8A\xDA\x6D\x26\x3D\x07\x35\xEC\x55\xDE\x4A\xE5\x5F\xD8\xB5\xDD\x27\x98\xE5\x7D\xBB\x83\xAC\xAB\x76\xEE\x7E\x7A\xAC\xC8\xBB\x3B\x28\xB0\x84\x7B\x77\xAF\x0A\x16\xF2\xED\x1E\x82\xAE\xCF\xB1\x87\x29\xB1\xB4\x57\xF7\x90\x7E\x51\x97\xEE\x5C\xC1\x2D\xED\xDA\xDD\x05\x5E\xB1\x8B\x0F\x52\x66\x35\xAE\xDE\x5D\x0B\xEE\xF2\xFF\x7F\x00\x00\x00\xFF\xFF\x9B\x75\x3E\x10\x44\xCD\x03\x00") diff --git a/vendor/github.com/open-policy-agent/opa/internal/compiler/wasm/opa/opa.wasm b/vendor/github.com/open-policy-agent/opa/internal/compiler/wasm/opa/opa.wasm new file mode 100644 index 0000000000000000000000000000000000000000..6840e639bdcb4f71f9b12fc52dc8f321b9c40114 GIT binary patch literal 421950 zcmeFa3%F%hRp+~&`*GGjr*>7UQbj80+MC?3lQwCiZ3i#nrB;0=p(tt82K%;OcZaC) zwx@zEH(NKUu6aTyC(;v7tU{Ce z;l)V5CwkYq=XvPI08s<#U%z9$w?FAdcHJPq=E*lso^svQSAF*l-+%4#?|<^8NxJ;# z>yl*o(bp&AN&EcR%IW@ubM#T)wf4f6_!(sr^a#wM!7~eLt*xz5e!`vq@do+Vdx;I}UOi7kS2Ur< z^v9lu+VnJb2m^>#K;5OotW0-u1$gB!o0RFUGM$u@Nx3Vdos4ILq|AzfV#AC_lO!+G zggffy{Fdc8<8hwv-m~Z2WSp)EdT|(k%4(RTgJC`xuMEcHq8KOnP|x#pJWkVbGNDNP zlNCv!KUp#4(Qud#hr=|_lVJ*VfFl_XvccZLAkXu`fQ!9@ys9cL^(4>JYE-3rSCT{> zWZ)Pu*3y)_G~;hITuFw*WOY>|rg%1>hk_bOo^V?LV?NrKrh_cqv7;;oqhVHL{CP+T zq!ihkWQNlIKj)Nh+X$o*+0G3e;RJR>DA% z7HKhFv7UxNlw}Ybg6RKZZ#o?91z+HBSR{-m%T`v$aK{8{ltPrq1J@+Um_b6Rtjxzr z38XXwuEl9-vcjumyi$&fF;7=g>r54kGQBQ+%wyogBpd9=HqvHuv#2(b=AJu-)$rZf zuECS8dD2sk{a}_%#q!tOc=feceb4cyUiIW_ejvGcMUS3z^|jYN<$IEUckaUb>#sR} z)%QR7x*LusmyF_z1D|LK~mue<8Hz;h? z4Ntl*dBUn5tY80>CtqdvkKYp;G)UCb_g;Pc>g3Y>Tb^I@A@@|9Jk1{FC{o@>dtXQruDeRPpxWmBp_XzgE1X_|xLAiVqj}6(27?RlKT+B18c4#%BPE{xWDjQ9ILru@JNpAO(_C8y>PpdkQFDsg#t~<1K zR=o+ix^bFev;+9h>uX4_G^?h+{516{Fq6xGF>jLTUkD967_CqLTyGg;3jCL^Ce5$k za&j}7K3`A6(Cu+DE1F`K<>1Q3#RxBC>Nv9UD^?RA{%utfdKht?_ly0d`kQ`4{{l!x zzq70^nsj|$sMrWBfunr-AuEwh(j}-T3s6rMpq^}n+F=~cOT7Z}4(g)GjtlU4#w#e8 z^#*a}Y5~;tWC>o#KhFTsV4am!1ijn}cOMO(2hycqB+}jcS&^=T`U^$67T(Vd>HYzI zvH>ThE-AK7GUJTPSD9k7jCsNwo2BA9O)NOAPE`Q*vPts>lQ0{PS!JGgk7t_oni-lP z%nXwsni-Q!^5eoxjBfqR3+L-B_TsH%k+*Z>?SALE*u3AZ9$1IMPu{XN^EUKdIBTcX z=VR#UqrWIa_ioR8=r3EOx}-`FcoMF%o18Aw_nBNb8jmDd#0KaGRWPbY!ia7NowP1Q zkKa=*T^4h+OAs!of?U*x30l%mf5+RGJmCsS`hGp&wt7~!z?eq!1t2jyy=4Sg+N5Z( zEQ)nxqu(2p$ajE-ca4XEvZ&LZh|jKEb%ln+w&#~5Fm`4-WR^y*VYnvCt>ILvay!|p z8Id1QchWi$Tzc?(YuPS~dD=sa8jX@y$T5xa>n&=k52R^+cB;_{6!e?MehU2tXg2c- z*LnsZ$eViSrtUJ=X}Qxb;sFe_h0=iciF8u$DJXne_D$(pop80bK7Er$)FfBNB1>B2 zk!dPPgwRTaKmq){%mf~pxd0^afa&nxe%49gcFP0s z+!hd*clJ;6IR)g_Teb(}dCxurK>qV~%(e|e`o@gt7LkH|)JMv*`>+^Fph;^GEg&JQ zKF57O>umad+af1q!7@#z#sxE2FkGcV2INhc?R9DbfBHd5FG|=i*ywSKw4QMbh@ph@ zONM1V{Y0N%cx8M8Q(>u1nzvqcbgpWlqvkbzCW`ojiLyQuy>n4gERO-%6UI>32&7MI z$?jlEClORAR2O|v|G_{FmN2ijfSNm~YO%S!4{8$lJ!k$vcSlqaeua=*OwchUdn3uK zFFZAFS!9;2L+n%IZ=TtkNvER5yG~CH4c>X{OgKvT3uH2d7Davp0F3zCF1Zrg-`k z1Na3Ez?BRTxsO*$2c`=*@_41`)`J0rPO^J3cQTVcMf(elGJ*;-CW6f)Wr;Psq%2z> z6v5C|Mmua<8U4FH2cp({%&MO!HnoxUocz3X@|nonk0jTyqAt`8kYnKh8^e0D=A7O$ zBg3@|gp2ZIULM?df&~Ri;sD9RbIG#0G)I3v__x%`>j6*e0S;rVHJT4%t%27Xsn(z_ zy+Zl$d{mdSff^dzbZ5XTn4&}dqx$tT4%WBMt-|2WCO_U3$7U&bHcvVlZH{L-_%p_$ z+9k1yc?rZan>t}wRNv#YPM0wx$}U;94Hh)mP^TyGQ|L5x;<{hso#xF`-G|x0hq3O` z>Jwrp2(Yb=2pZxn*nbBm?esS_QKV19l@zK!6R{x~#NT-BcJto`{nNXf{++$~ zu>Je@_ODqU>VvjFw)~_n_AfR@F&=$%`&O#I`4S$vkL^?=cf3SgKxl?3lq?}2jNHmq zH@JQ2ycEpBMY_64;DqW^=C6dmIf4&yG5u>prV-0b1RnLEoXJ-Tl8ecceo5o_zGeQI zEP3Nd3dDD(aLCDa$jaBcq8{iXMrN(HbRH=1f!h`1RrY|8JokU3l z3;wf52}jfS3kBz)8+?#kNJZh$^kNHO!QzZtT5vQi+juX?I#=V}NlWn-F|MDBzJAt? z-kUmh5uG&`u1{YfuJIvCaY}pQ(@yWuW3bnM7K&;BASAmaxo{3PFG&&?t2pLOIxBq9h~X*Gz&RmPycTs14J{!-L7c;d*GOnUC%|m_Gyl0%IdX&_D

i4vtiVHhIggF1Z~8`89YdAHk!?w>*1X@)Wd_rC(M)mj{tRPvn#$h z_|y{zVfOv=wn%D4hW;YmKO15}5bS1%(5n)I4>WVs!RCO5>PJ`7DhiwW$h^ll44Ob0 zh7PqXdS!%TT!O6mj`cMpME8dU7TA8{vScSe1>>g{24Ry375Qwy?U27CXF^(-j~R)Y zY+m)6lbZ<$-gms&cSBQ6Uo3GlUkO}WQ|(F^@c#5DRgSKX>lLqYgeoi7f-+tm<}qAK zUqC%n&2%<&mdhWrI@X=IZ+#{+Q{z#g5e@u{EBP5U@V{t^u_=I4ekf}&BdB$qgb67g zvij6{^lB;dipbq%i9(9SgcXtO;KsuZ2U86^YaX($2cSTvF^f)SAzKS3695Ae&BT|( z<`Ixsq9WB*DePatkM)nSnhB0c4I~W1h&WrVVOM}N(DPZwZ^~b$7-7Y`g;vdaL$ck` zhaq{s((DXq)*QUYOZqS9ZQns5MO;APpdQ!75%5wRKgRq3H1SqB8wY$E{2~#fkp3Au zW?=<6C##3Svhl+CHFNz6n#-H>*JopZ6<5eZT`|t6#2E;zK4+v$oM97~Dp%5rx=i7U z6+kmSIs$Cs3cWha<8eK7t}vw>B=w4hv?7u*T1K)W5k!G8mEj1n7#v~H!6U{I@Q}lc zAQsZ+jKgOnjeRQvGEO-EvLsRs7WKt^3a1HRI zTMHR?4Zkq{sXvwDgLsM$Ed+%vt!jfN?qSg!|#T2$}XYC2`)i8VrjxG70pQVFB%snU}BTGAv+= zV6eF9vc+aOtEMp!AarN5NO+veW;sV(mU?vl@(^6#(|gv%>KC6rCY0URFAFz*^^My# zvi;Na(V^^D%T1e3-z&3kPF};qGg^f;!c{f4PAILb7voUm5fmpI4oA>r@{c`#q zwZ{Y~rtj3%42G{$Edapx$~<7k%bQ^O(sVtXzDA|atA|*kGv^5tamn=ey}EdpWir31 zXr;}ANUB{)>jJ5{Nkhs}fVya}CL|Y`30yrEa{xz(rsz?Z++fMIH{7Cqd$&BeW$6Kr zPTHgDg^MxpG-Abdj<6HFtPRT|DYCpen1qB5$;mqTx+HDyGilaLwPcd>4-S{yvfD%S zseL^CpL#Nr#SsM@KGeeU$deabr~p`OJ^&?)aH+S+db9c3lbgx(^%`JwvR){I1Yld9 zRM)-kvg%}+j-tSLUUp<&L{S2%noUx!vgu3BNow+0iEu1<4uh2I^Fecy?5TC; zt^sFwtEXvvIykPUJ9w1g`|!BQF)-?pRuM4+WIBK!Np}Meu;mM_;q${{y2UxpLs9?&q2Dg$)k>TujKb?ejVq%@jX?srO&_dSko_-d9my`QkvIawe-4hj|`3w z=r_qg${rIYi_OsO0PD?H16)uvmj>8Uc;6!Ks_F3IK(KVP1YGJPJKj9fDk|tyG~fJ) z1T9rFdS#&rI6W0Iz;Xrx;kSaRDGewD^7gY}vUm3->lu>$!84inc6Lm5aX)O^3nz7& z=N)2NB@Ct21wv9#%f9Guq4H$Cb@ZyOZ^1W%^vruJh_l zPtX?agJgP}-gr5`cF7yB^Kc>NXOYYx6TP&yDym9Kuw+D|GCFichcZ00_I~@*Ppa+d zvrH`!@3DLX%G?_ROVzC;f^XOu9TfwF=JZbl(~fHTM=le1ExvUY4SKcBxHOv=zXx%K zJPnZ@3$>kk5lo>srm@7MRgbxgSf=VrMnS%mM#Px6OCwTCl99PYE^~uctIFY@rL?6$ zE}#S4z}kk{)%t){9hT^d`OuR<=31lH_^6U^86n=kVnqQBw8a$kXi3eCQ+qbl`N-2* zY(|Yd<4IG^mj7HQsI-loeD|&$Xw|h6Go`kzCWNVpxdzpP$Hr!DAnQEmQ?9T*4hdM6 zY2%ZDv;ev4Yqoiy<#rTtW&au z4TLZ(!Vn9?0ywBe(7P_NaklTDGMSo+%{ySYjB+Knf&~B}Iom)mi;drhfATC19tCn-ek*Rb1>5iqel&Nf=btA=O4P z)w-r5H4}kD)sDD^RN&ynq$w>^t9qMK!(jtpFv85`S~qh?Xlw=tLzM)GTASyA7zNn% zy7>W(fi_efZT~(SdD+abAQ(|LHG_2i06BnGIucT>P}-uI!YY^Q`#-L)JwDMMmiTAk zCO$mXuNT(8v6^4A=SF1BB2Jssg9g1vNd4>xHm-t|Tnf9-5o=r25uUvm_iO1vw)`OP zJurOA$uQK3dkHNGf3w_rHzG$APn)y`9nEKw!~3N&5oI7o6pMJ1q7t+zYH8XG8v=_~ zGOcbYW$rN`@K5_wXVgV9<$*v>rvAS+Qfw>m{_ojz5dMtgh5fE9+--9+WUEp5gKV`w zD3GdR50d>UtAR;sre%lZN%Ni`Ik~AZ71L*$5YY%V0KAyOGW4?&>y*|{5$e#)ZUr`bFpirK{p@0YMCbyR9m?#0Fj!2V?YxpO zGur}2E4@zrwX|xeELOir;nR+$YnGz$%d$<(I*PO&E!>S4?p7A=s)f7B!rkf)7wKsf z*oa>Afw*vbxA)0VT0q5i;;As0C4th302eB0_0}PJrwy<4*GB538AzUF!R%tLi;Y88 zh%aO%_H1L_7ml*#1Z!WAJ^`7`86wm*fUM6)U`^h~5O8dbqP@qL7`ZRv@9|Dpl8P?L z02mZp2Mh>cVOzDw=+wG(lb8TkWsBGInm%R<6DR~|^IRy0CP=%13v?Yuq zABIOFJ{_BkrGp~CVT+gaYT`IfdJt3720w^-R(M|b%#)blX3`3lk*oD4v3z@$f`pTR zi~zBWgWASb=SsGGO>c0`YhKYdXV%cfFiZfCSl;xtD(T~j;B>!cGa+^X*pvXf9os+b zR6O%PUO1&f7Ak4yB%3G)pwUM((N&|dO;8NOW}+($;9;|(E4;U1v#Kj13&Uott8UZ_ z)NFqHw%&l@^>u}v7id|U*J(IoK_S$JvEYT}+7s~{p`q!|3R5y9mk>wbuh*e751it2 z!q}wFm|DOMlSB@|K-DXSQHvvyhoKJcP6)mw)$?RH!o>s?m;Yl*+^i+cG|LP2ftI7u zXgI2-zb_CaB|JhbAsVkFi`63`5$>`3A6Q>2+1Q(;urxTM!(VlvbvOMrzYabhc4k&> zrugGN22+f|p~XK;-2p-HjFr@YeDGc(4CXUIvrtX|1~Az6Xg0A1;CmY5t5{c}7U%&o zuYNftO6L$P6;r^)$1iZ;^mhmsT}KV0EwVIqQ+iwQ!oy}2DY#__=ACTYGN#hm4BjIv5TCZ9L5s;U$tfsf zfV&O0mBMSzvg$Q96uKnRr|DMeC4IVTqyIS}DL`z+@ND->ZUZ(1YW6g_hYM^0EtihBG$J}DM1ZKXafiN*hBhdf3vs;h zDdW>1m&!tNY>NI<{B5nWY1kTbJJ`0r#gVFcsw@nc2&yxSaFsP(CSh6q3P)0ya?%JH zvxFkUT+6VNOv92HJN=(Q#ae(>5>K5t9}#lXApmgF90*grVfVSS5*EVo`5>^;uoa-@ zxs*byW!2G~$QT7^X?$dM1xUqZLYDB_$(r`~$Rtd{`!N2zuLTrkcAK3@6qAdN zhb-a4>_j@g^<17z)E0d7lXF0b+*qd-&WD8%&~t zCTzrpvgkaR4nn~*!|Sy!3V~|RirK;>FzgbO6riwc^JB1($1=9FW)-)gdFv*YEWtBRY(bf* zWNUL0fa#$d+YF1d)hDcqQvRQu(Zl+-mIVw^TI^nHaZM230;_UeVq%6;aI{?Wz+!X3 z*TbYVN&97k-Ok-Rtx36L$y<=dmQIMISgfwMT(KJ()G55{C)fP8{Osz#a7gGzbMw|U z3?k8tUJK8lGxf5#Y+bH4iJg z;USKxUSm_CMsULy;+TYMdTuyTO(E4BKX=247~_VgeLFyM#_&}Tzcle8>p+(xt)+_En?FjLKTaC5>+q}J<8f5Ni)M9 zrH4l86)3I1-B`U+19ukL0JD*%TDxjXwUpKhR{ON|rB_Z7xKm6q^cpJ#N{7Im zLW~N&Tfl`-U7$25a2>#pDiRFCmxlIGn$^h#-^6Di!?t}itXB07jsh2wPyDEnuV9u0 zztds1xp*0~FsD^Ph}FeT*mje!gcwdirZYopSm3qI!M?=uWy7!}O`Q?L=F4bBB5Q5G znn-eFQ&$YPE6oj@rqf^1I$_7QCdv@~OkESltX8`~Ko}p7%ho;7-vzd_Ao@MDjh1f+ zXKfYzpZf^xsYJiD5(44WqF-7loUX$1dy8LCpam_V3l5e*Py{xL{*QHbrPfdkUxk_J zhOeTz&?LX){zZ>SuotZf@Y&M=jf>`<_^y6P!Sj$Vw7iu%Gm$by^v5%H*k7V7nMh9U zo3jXT-t1e~qNy-HUAOJRb*>-9i?#Kh8o#TFWfZ*0j~-x{)Ax!C2(TM>5c3~Qe^pO- z^M@+Vwn8QNxMX8YF%}qfMhMcI&RAUU(b)#n>kKz#F%^2!QyU(qXOpaDd6RA455cn4 zzBLd^>)AJvi_NvZ$TV~Vg^Rf=jLbJ$os&!}-aUf4i z&`ji6T8Hi?F{59a#Qk+>pTow^v|Au_@AKPx{nlXg%bJsUj$fAD;RBRZ`&4L~OB)%rxqM&u z)P~_tf2n(FF#FRNcTWvKfBLBairVeKc&B$Pi0$vwKAWh(o2e{)D0 zHf_q|%{^+)l47!EH$e?`=(bfh#5RV6kjUp5 z5HzfGqg<9M@_?#j8RYb!dg$b^se;5j+~(b=#?Y>Moyb8)2_I>WBgGIP>X`E})T^5I zRGd_T0&z%ldL=OE!%o~4XqI?HBbaR704=v*3*KINu!rW+xmPmE4O+GfYAEEWwG&}Z z?k&&;)?p-uSewao!ae)SZbb0TvjaGmG}gT1zn+Xs$e|%CtwCsHb|Z2)I5r-odeSZ< z3omrm&1Aj$$08kv9UKen0E*^dmu&!B7{i_{For$p7=wl5$QYdDv&0xsQ~TubT?+9B zJD>48IbTCNwB{p)Lr|M3r*Vh?peD0TMZBa2r^Uz`!A8=5IMZw|5<`temy{v$aI8Bj zt~0vYPEj7)>OQw3Yk@xUhB@2=galz^E_muZG`6bbd}za_@rWxJOFKZ1kkld$lGAJY z1h8~IiRH`+5`cztc+W#P_XO6k24m*W*qJ5az(RMU#kG0{F9W8=tlqakYV*>1+ZIw> zz_Wo=%xHjtQgyWjDr0_wY5`vz&u$Ba^5M@MU+nDodbE5U_EUnh1gwfS$O5pD9@;a_Z22{h*52zg8|*gV-z1D>%x*_4dhs4hB#ayIzt5|TulXhbqX5aOw! z5`CIG@0R8QJo9P3QKjJ;dpHA-5cC1Yt|G0dv{$q@MYWfQZABmqud9dA^v}eD8F~fW zO0~Sy>btt#<_o5k?5CfZzDK?XjshcIsBb4aywO}04>=<46bMcGhSuxTu5FI?OWbTz zOP3_Sr-6g#_vm-}8qq>yxGejC9yFh_$FB{KF(!ZP4n4LKxA~=?>L+jTrFu}?y9p4a zUYiHK&oh8lJC*xmx14P0N^@l586@J_7{aYa0W4h@ARNyj&X70=C-PO*$FnT&KEZN# z(we1UD^uwZ)(wZS!gCtkBVD}ceay*9lRO^ro5|{(kZi(#2xZ>_uV-YfIKR%B!e%m4 zGz>Fs9?2=|!W6ACMIH&5&BEd1i62LR3=1bA8hko_wKyn?cXZ+)8}aYo@tus1*?y-J z1%PSTzNaDq&jiFURu>?ushhu(o?wF60!>p(b^KLt&IIac*c3M^dxHk4glPoz9lct0 zX8R|bUw8(M$FtMZ=I*$enV{)&7nv%JOht?0Cs@^>6|yHa0E!7dDC`(j=+pwY@to&& zcBm4yr*HP6kFWLSpNN{q)Nr_@Px(*lb>IB2p{KmoS_hixA}Mei@oyAzJOuK<|> zb7Y+6xb~KPK+tUjKoX~keU>8}`7Na65s80gm?^qa%{!jcqf&POzA~HOS38iB77Uu) zk4QBFk+^6?3M@j|J}7pa{;4{NALSuFA&x{YPoE>u=S-L5K3&XSlJ}Q4%n_&mb5+Lj z%|w6AkL^J3Sk@*S>U^aS*bgt_Xt4>}9r-Mi3~c!$t6tSRgKAp}iCrZ!s3w+m+C^;D zK+XFA8Kmd~bXrWsb4dvm2va#(!;UFr-ZG|=W<_RjU;+ zFqLrFZvW)Vbj!T%v2sKfv-7kVdH*&HxZf?$i zCF3ph5@I@Yj(K6zHn~D5;&p3lI z+`1@?(!Lg4JO{m#EvM3kAMYa-hKRP$b978Wl#N+nh-JI>MR|8hcPCL?deByTez{q6 zxuk{!MSGGt0>QtI5Ke*R*3LNh;)FAzOku*haOlj-I2m5v)81j{Dq4}wSq@^GZm!E@ z`oC~s*rr0mpX`;#o+-jPJIT~e-k+agPqfab>@28u<~`V1>1WY{;8&c)kFuBj7tBDz zWcvcFeNe*#Lp_cC;gr2$F^nJ_Aem&-VnpT#_!nvK=LPGHL~l3=NpT{bMXZvX08CuW zmiZi|3J+6L4Csq9;TtNTDZ-p{l6q^X9p)Jh=%kE50#(v4ueTPhMzo57l(E zZ`WCK`yFi)3(ftwHTToKx0`6M*qOaO9iSl4O!}<)L0rFx7I6X2>)Q-4Uu|E)k$vJX zOO$tHE=3JfIhn}~G-7_Dok&`Klm7P)pr~MI*yxv5uOvxL2i1dKSwD6}NlTT`Z!fED zFi81h+nvFCB%0FQw09uq`^Qj@v&d}ExJtAm!7|Bx62fejO$S`zYWhisR|HddvxqDt z-_zk`Pb>lccvzTol^tx`#3QeaIoboYWC+1CoU-LB<0xyI^(?-H-imn2l@8=u{NAJ% z!dVzlZl!{@amRgR&4WJm!l&?(F^`(`Gi6zRg`QPg{hS40~7xVJXsN3y6&4YlzwOp_d%=lkbG z;{&csk-_uo>U@9rTqHqfAAyKOaJ!{pxSIZ>(a`pwvdI!0cuZ0w^gcaE0eQoDWs*rs#c&}aA;vPhPB619S#{N^ zAS0&t4?Q^qC9D!THwAhQPeaco5CT0*(X-?e3DA?YJif3DJt4ErF9BA)DfQDcnQZMm za{6`TM83B-W&Z}tmq_)7HdAB5~p39@>X_2Wr$inDg+j8|^`=NBXIB zI|mV2S@jKs;44{@9Ec5Ks)ShSZ| z(SL12-H6c~vSr%2ugQe#9S)YzV;>qU$OMlF?W9(0-h@&-l&?dW9S|B9RU*Rly4NZQ z`rbr_P!Hz!0Q3q&-}5L?+N1DGX+3P#WSzbX(vCNd#vSRZe^m zJfjD@58_)RH-}xUy!E_L=F{(_?Qw#jBw8VcT|s516ACCQ5~M0U}jH9k!*KtDCZfZ2{e%vSH|#Vzb%vspxH+gzN@K@*>%O`EoiHn5x$ zA52?(I_qT)ED&qUaKs9xAx}LbKnqi#qm*%l~5M+!eF^ke6-($Ls= zf+sqSVL@VP?|3_}Oia0GQf-}WqCNPi zTj%+SI)A!Cv_B)YXntd+)$Yxg-MWdwksjZWO@H%EcgAfJfFZPSR4HGgO!OH_L24hv zWK@?Z=hTbHw^v4`7HvT%c7E#_XS^6XI;osB8Cq7QLoqv3j3e@Oz#z$(d`)FcVYLhs zc?UPCEg=z5g1M?FseHuhS!t$kiP17R=j3#48##zvpG#o`65K$lGvYn-x@C-)k23-D8kbOe@&tl*y#wEd(XiZy@@Ws%6Aw<@$unR zCVVJ#!GV|8gEzd?QhaitpKC=PkyDp1_fri0N-aYGp~st}EuG%mbv07=)_}wC6ED)A zH$7?=djibSDZwFl3>YteDHM(6FF5duWvIwisz(&EuU@q`4qF-_Ic;olqr@k@-0AHw*lwO_#?&CoNxw5ZwW?Nh^kG#p}J-hA61 zeC@UHlA`<}(OWi{y<5?X9+M$$>y>NY01ahu4*yRv+H8!Nt$tMpo^#VTo&6&r0X@<5 zGn2na(7UF^a>F4@u#OKs10M{>fSr*Q%@?k?rD<)day@4Ov#2-dMTRVH9J<8J207>< zBw&3EG(G*$C9^A((k_&mzvJmX`m%=s#K<-IX@~#kvTMvf2#HIen6Fa#t z12){&byN5Hr|=r*#j~qO2xMKxz%LE_rKZob_G%kN@JTbV6Iy^CH2@v`Cbeq&0o+y-q z^inNtA-LA+N{!i8Is24|s#`IbUUXx6MyNntKUToYf>#t0+mg7s4gRbvXr5(+qxSc-PPe=jz0U`WwYUz{A@`%d;n`M!fU zpE&raXObv2Bs;2k5zyR;F@lmIYu55Z+m#S8Bd5O|=a@}fGX(Zb%k{evR7v}(+1=~Y zXNE))3`5RzPMT-;7wgmJ$6j|*NzQmh)Tg*7FD1fZsPNJLk}zaU@SgSru4j-MrmbPX z&A}2$>cqY_lY_TrkH#4`)+^8fDRhE*?k`wr2=Blwxb^fS|8@)$G3yfi^6J*i0>KZ% zJ>Kkpy>|)OTiC>5Y-2nh$j7gk^qooEod#h}1DdF{xk7SB}&p z$th%Gc3>H8D-mrn36!Ahjk ztPt7RS$x7trej$X78UU!NF}*Q_%aAiRLn|LJR+M6Uo0rbU>}3Z-S&cxWKx@;qD(Ar z9U^E^?Y7erl(@uH^R98Y3c?av1hDPgae=3Gb<3n`a9DWaTQZM02&;`$O8+~^WWp@1K$?j?a$o! zUIRDW$`sV_!ls^R-iO1nsy9N5l}&*)i7P6 zykHE(!(f&|IhgU=XL3Pp%qC7wH$4UBh_I?jT˓Yc^?osZ*ZzRR2H@wcxcLbd8d ze;BWPbV}i0fj7r^<1-mR0LRke*4e;e5pgysjNp90RhZLM%R7Uis4m|cj^C;#F{6CZ zeXgsY48hXGLt7}`>J1s8+)nzPp=5CMg2^y8HWq;$XC_4{d5;C+Pq3F!Wv81OhF47l z1%^!Y6^FqttV!J<3gsfWE)KQ?N&?+Z%b2~+#{SJlX+g!!gmRWHgmU`VdY#W;?2Sg@ z8T!hj?s7Ju2cuL}U!ocX1NxWEl;6r8j7W8!LzMQ?ZWv}GGQUpL(J`I!MHWmb8w|WG zwqC%iyq;6*l@6~+!c*Xt=YxP(!e7t;^bWx*HU{4rHZ;x#l74dIB3`i|9It%;3|uye zeX!58Y_J9#qd~Csl~m;BWpv(}e(nU9^A1aoPH|i$+ziaaybm0@HS7anVzi%g40|HL z^Jyh$B%^6;9K}|CVwj-3opT+n0OO_b(`4zk29u#E3n-#cJK^K!M^wDJQ_GNmqv;h* zF->zl;;%s|R+n9F;>>IiO3Rj|4kTRH$2M^GAd+qw_RGX+c2>K6`HPo7u-@2Lt}^XD zp<^bHFc!{r^@KE@eQVoEXq*7n#)E2K&N>Z>acUwQ;;`HhB+0mpB_v}+RN5knt3WsE zrqy$l+SCcMppnd?PZ7b{zlKxuR%3$*NY&0A9YMc!)=mG1$x8%x@fH5+i-Szda7X1bV!>Z3t|G2#H&1G8`ywF9lUd2L1Y z_PjfIquVMG8K$S7O-4x4EpQMNu|Zl-MqtD(2-6-*K#g1LUH)${RFq^-CZ{+ zY4O&&e*=dL_emxU0FC2_mg#PVT*Y@pWa2HD?f-jF@rh2wxG3~RrsB_k$x-nFH`I{H ztpqpwH+~4-#Md^3Ow2z}A?=)Q9;lFZI6q(^E#E(2Aw>Zmu#mLIVxgC%Ee<4by8l{? z_=$adc}QPx@-N2my|~|WMb-TJ?I#I}4cq%yKU?>M_Wt*OeDVHAHg#XN?_YLu@%?k` zo{e-S;Ee#%Y|Z9%NBP=$enrKpH3W!h(YR;n@6&ZU8zIb?Rkoi}Cy z+~&+1v)p!X3=kbjd_vj=$VVBE12qw0yRxQzIj8zpBG1b4h5*?$7+B5bJCpTe&8P3j zhG?m_-|jmq?yuf9a55Icr)&U?*r-?v8-r#uJqiXA+Z`hMhd(Gf@bN7y;ynEEqW1;bTY2=>JIN_Yo zxEceZ85Rc0L{}(aU==U|#O<`Cd8vKSoAr455qNykdfyFh+BpDsQ@}FI2F+Ldl6|F; z;VZ6Q(&>LDO?)_X-s{FIzvT4&1GkX%5vq$J+v>W}qO$tNBI>{B@F(-L;Y1HX{U}ln zj?>x%%HNY8Jr+^PqfgMzy#bN9CQe?XilGlg+GOmZ6yh67y_T^aJp4TWM*68y>WJ!% zB9H?fX>q;7ah(a*_M!1e6=(2M;`#yFW|;Re#I)b0ZiK_zjn#TMOhHaWF4*XhtDE+P zkjohW0u$s;gxp=4R`(_WyCD3SVg8s!Dyp}n;7xrd#&Y^JI?~;^nTI>VLp8RU9@Eir z0&s96*x@eG*FC*@_&Pl=V^P_}94v4nJg%a#VtkDNRF+Gx0+XG)T{lAa?JBhGbuARt zkEA_GZX&>PY-$lvK8U;*n-<0#Swn6J-N??vA8{#q0(qJr&?xYMVJ?QI!LrtBWsF}f zZ-k1N!mxu` zNA{{M#eUk0z2~i34s7Xxwwp-Rxe7Gfbf>gyr&Z)ZogAJIAr4z|kOKF`7qwF2LWSm2@u?8F`S!3G`~*R@7a zC;BPCJT^KV;fD_Q&t(drxvJ-XoLWaR3~PG4fP4_GAgm zV79YBsgP;)_XTT?+09K73wxEe8k1pdB%3Tri_Aa3Wl5d#ujJ#UJ}Yp@Ku$;j@WceA z9VFA&%e+FfSrUTZX)#tg68+3fi6n(*?4zfAX2rrb))9L|XJ^xqH%@rXC`303Da6Jf zw>r=V7KISz*YzNh&ESI2C6s(%7LJ-<{l$;o_Ef!8^#Yly&9mT9`7)k!V&k(;A^f@T)n5dwbV+Np%@1mB?bUza=1z6K?%> zbaq(HRUe2q!%^l~#nE3Lyun1Y0AoqAR>_WHyZh62fw2aGn(PD|?7M+VnXnDT0*}7{ z*K#A1yklawaLJ=byI2WJlw2sG1?<{BfDU^y^{o?Rvn~_2ue`3+SK7J`(9}L_253JV zhWg^CpWF=V{>?`9ku3MsqAn_zUEt{r%xaAqmgzc68s(4)|MUh&H#o`QCKM1QM4>Fn zB)Y_YHUaxx(kgq@KyvO8Si-Oqlnp^Q1CTUc-5~nB$VEAswOYK<`mDAH{X4E~1{8OO z4V5&gu?I?({?>jL=#*NptWPIRI8jqXvIED|u5T86z}CPC6KR{)j7V`-GfH`QZOkuc zDTj2OY-_qs#G!s&_&HToKTvg_ri$QgiHi9c(I%ho)Krj|X?j%9eq@V(YsFifD;NU- zWd3u2fMrHO46UPqdC{SCDwIz!jQ86EF5mq;(GyA0rsnS@j(byx4_D`D=9HGKYZsE-(x+ZED%UgRAul2Mf~Ls<@??t?_qMebW2_(N*y-Ht^z5_xlRi(l(B1kI6fpk{bz8S2eww#r0af!j<@u4-b21MOEe$6!D8xneRHV;7KA zaJI@c@p3%y3)V$ z(U=Sqkfnu8kY#JU?RGLjEvJbx5n=cZ9R4aw$xN`9w7lJZd3~FM4u=w#w#`N5$q8k6 z$^l|526Y2{h$T4ic@OvizKzl>0W|E{3}QYu)O$0F!!W#PV@DVbzhg$vF@MdzfOb^o zU3^#p5O2wJh?Aj0JLx$11T1klE<7vJkkThC5odABQO1uJLJ7ZUlxB{I49+^vuPeS2 zT6TrD7hweGkQ9<^c*0GloB7x$kMDW$8e?Hi@93G80izmua6G}aq==@1=p$2xKpdr- z>*RH2sBkE3TOCA4Dd) za(!emK%w-AwtkXBJVa!Z1WWbPoSfU^};Ne)Z8T|-FTH|wZl zIvj}>g8++$-U=#*dV4_y@S#d8j$wq1zh^v%kiY89EDpm4sSs2c4Zq4AUKCWtf}k=~ z!2X)cV2N+h$wx3j#R!Y}2=bamCqb16l}RV4gwuHzMb#QuS+gwyLm8)9%5D7>J!wT% z(HB)-12JRN1B585*kWp;imHYjYyycYeSn@V1tzMr^VCF@w1#AA5LHfJm%dC~eb2ft zs_Yx~0xpOuwtuBjRE;FAT$s418uUe#apl0Kpb@f=s9Gp7(o`R4ZVa{0TB1r!LT>j~ zQKiqJEsCllh^m3CPZL!N`dSr*!p2l>m@n*}%UW3g(ml-}abnps@t<11Mx(=5v|K-bM^+(#aN{xm2YB6Dl-KmzH393^gg{^ zaSFO?UerZq5e;9H11!jC-tzR5F>>@RN%$k7(NE2K%RIj`WO%&f&9A#K2IDzr{zfge zcK3V{V&Q(@jqzQ6u)gii{>%K}h06~vU4HP5x7~St_3^?2tNnDx?92kIpehBmGGTS9 z0lzT0!lrOpiU{}u#auTm+J{=7pCa%okVL&6yo#1N#IqFnjB-O@!DPy8nXx<}IZctm z79+NgvCfB5eP%Agnv!?$BI_Dl6B!Kgb##t5?P$W}G!KriOnE?Xme4!pqZS>&)J$w8 zAmG;J*1cM^VDFX@gf>n$0&n6>M_`*i&vXQ-QT+5H@YNh20kWYVVE$%L(N3WX6E4vg zGd!`t()CfNA?1LT>!1JglbeZzg$WEa35$*Y$8dEczj;M5QSWq_GSsa43_7 z5svIE0U=fS_<+*f2KgZrp=qKgf=RyQ49vR6rxS~3w!&=Zm1it^ezJFSc<_AV?UnH zTZ%mn*&KXEc8tsv8ZiTd-O^@Ys8fh7(HYFGNif6&M|flNQM;Ml5RzO?XjvrL#u&QO}JIcMr7l zkRm3MppW7fh$Ko$B!Mj#Nzx9Iok1iyS}5uw32CJ(Si?n>F?IEpUWQ1V!D0661<%g>gI(5IT(&BG)vzAiHDl8!0jPAg znKUO4#b# zzS+5yU`6d*9p>Y0*XP!IEqO0eA#}3WXfYbt`XMWt{UtCbW+2TT$$IH(g=p=cg0(yz z6my!c&uvMwXI18ky?|myx(Eg)1CI_Ht_;CyYW@&60nV!*QlEEdzGwe@uY~^IyJq`A z>-jbbh)Nr;aAlUHIk|SdanuoeL}`p)x#k81yXop zXxdkw8wCHkbh}pXfBfwHnh>A_KVixfbiG$Ly$yC`Kvea)2>v~S1;Ky534Re!qYHw6 zj|9JN-2>kv&><+g%Utd~RlP^ci96)4?5L-YpX~-nF0>Ol0YP#NK@PmB0OB6HE5m{l z3$EyyQtYOQ<9envXNz0!uB83~Dcu#Ly!8$xNrH2SKZrq>8TQG;AnPRJ%mS0j?&~jv zkn~sK`io?j7@KsH+9ZJl)fSUle4%(tr@u%RLqvpJzAgGo5*+=aD$x|=8Y9Atf@o5^ zH7OdK389x@(Dj!o`Br}^HG&T1ip`P$3o8q>IdPNPq26A90eq-p?Mr6RjkZgk@gTw> z8Q#p|Fl>;L7GX5}YTaYeq#i8jFGB@VQ8EcEk-8!a*i33h7|t~kN@OE0>o4Jy{fUZ+ zFv<3UmA#fdHH6nu5~j42=m9u9X*JlOufe>;yrG<=G= z2HO{1mp$P5p^P_Zuyag^GLFi5yjdttZ$)w#@gGq@VVF8fvYtUdp5W38;C7Qib9D7Zs znqXJRTy%ob5%Ds36Z63lPF%^YVv2ISi<#nV$B=?B%S}!DeL(Ok0XrU|m#Kt^P=wB# zhC>irU>OM~qn9{6b4t$PfUOrAE1o{?uYs?u34yO{(}iIYQxU&$a-vDViNs!5qEi&a z^>J4HR8F$9PqO~M9zm0F9deFgSM}OtSCZAm)%0@x3jS1+LGLa{4uvOzQN8ob+;t1* zLc^E{qGD1RSwnpWmm%9#0m0uD3ce=-B^)axprNg`jm~lGgLx2WmGkIc_I7m>%i#55 zvIPCPD8xU9Q$oszV3LB%)gP!=XZD3#o|~ZnfggNlA?g1#o%y(cw*`rT2GgXvb8Rpr zDme(z$t@|%CP5My$Iq}7MkEi52ECTl0FJoeVPX5;^qJxAnQ6EOEZBxY^JZWfFHId% zGNe#Jjg~;M9AI7Os4GvN)D<3dcs6OT2&;r=9GVxd_SN~p(ZhSi&v(ia)H8d_?|P9< z-^^iS_n?}eATK`_1BTh<28Ev?A22{)NQz&xWKhb9s8Y0{c=uTfQQi2%_<|B4mMAY2 zQb{1i6Q@WV1vFEP7Q(0BxlB`-ni*}2Ro&b&!o*y1=1Bpaw60-+RvQXWdb#ADWgr>W z%VGvl0eoD&3EjdS6cU}2m>I-v`$%}pvNqw?`qScl@mOcM@eA7 z^s#B(%5=j-v+|t9HC=yWY0;B~wCJ(CJ91cbj#EODc$8yCji&I?&31uCPm3n%Sg)cW zEvJx;MTl>q!}PHw$JnhYNaCL2ZkSHIh(_EYMxZ*UX^8FmemshmY+YWPK`D~+(H*nt zJoNE^>;p$IVCGfelfXW95Te@p3j6$1VIL>Cw7KMk8A(c9wIHyYPw~?qW)*e$G04oJ zRJ3QMQGa0(@YamYtLSv!yriW!Hw_jASyUXF)96e$$(U2Dx}mtVtZ(AoBt5DU;YyFw zcS>nAlc4u3u?l4r9A#5<#@)*uYR`c!_aLrq2_@(%TO(++wNusLeC#Fb7Q;qT#*|kQ z{A7yR(x=)$`R9ZGKMHsFYo?1=jB9WLHAExx<9dWA39)n4#P$q9*bw+&)D^hVZHoe*D+Q5#C;WsRqc)fon()Es$5rIK5~0vSS-y89OlNlT7eEvcLs+dF9SZex5~m38+4&XOjiEj*DJa40eA_1KEA zZPj|zq8Kp9w?IkS{OOdWjDWiAl%%#2?LtYK7i+2}HsXF1G?CWaC9O$Z-L$6l=v#(N zYwoVI;Kj<(37d%w4-*VLUw}hH;VA~Kxl3Aem$c?Cuz=RI!9_2Yr38G=p4J2(s5_x2 z21aEcaT^1pNUK6xQ>mIq#DfGmbc|P^ZBK882^*}&)WiyD`I-RZdc`LcTqGf5>v>{q zxY68^AnSa+NKZLkN*Mq(ChkR9LBy0;t>*+(=WA$m`)Ru9wLC?1DJ5viAZWDKWjxUn zH1R$Pns^@tO}y_58u&wkMh6{1Y6%)dubpJ44=$GMnkzsu!vK?8+! zse3A};_TkPoaMZcRX?{q`i5UYAE0wRP2(^+yV1Hm^BTOpZ< zS(*?QP_FWf#K|E42zSeDNb4+E1g7ovwXfxZZ3)|^@)i$TGyzXF|jpU%5Gm~!N&2fGJ8q?sHN%e znl4ML|0RXrR>-|%uA{IfL`AhQ9s{nH9|Zq|#c!>w4bUgDfK7$*8!!@#+_3FURV;Q1 zTDfLf*NpODI|i4IpER82j%gOIur&ROnhYg;y)@)-p~ZSs!~LU>p7p6F7zv%rC1EXA zR9l8I$GTF`y+<{!Nsen#Vm>bWm_#Sm-5f#4DN&7;)fQCTa_;;~&W>E&yLC0xT&(^$ zwIvU;3H;OpK4S@HYQ{;b0gLv0?wDRce8-MOWJdxKjK(tY85EbK+)5`0cFZI4!wOEv zj`-dY{bF3RLZazzu-MKYbFASm5WTo)4>@enJ`Z`Y#{AOW+9?FI0$(P0wMW)@8fID2 zq%s+k>Z_J4`h^65blEwe(y5(;oz|5EK;p1emniePo!CC36aSoymRiC@v)Brm))$=4 zeK=|IB6A-j&R8%mawJal#OF^^{iN)@mO-o=rnML15Ftw@u~S?GLP#PZ79D*L3`8+p zm`9?jc$7&Qcr=NmWZT60%$OtaXgiO@%X4kkU?zvWY^JrSz&M{OtaL*#*C&aySaB3d zKl3*dU-l45^}v?Q?8=V2uq#F%-cH-A-R;%B_Ue3Sz0|SUd0&u@H?u4~p^q)T5kZaK z5IRW{Opn6_$wJ`idzgtWZn7qTxSd-FK6-AU;W`gkBg3B=h-gT^X<9z3roFkrBuY7+ zkaPH9G^poko<^w|dJxcYYysUrz>IZolR2YnIVOUUxJ$Yd1+kfcMB;>X4k9T|C7=F^ zf-0Mv3N|h-EeJ*+P8&n6vzB^|gmrFJM)LVCQj?%TTFJ?s8~ZtnSHGt(voyJT+v4-E zTBsv?0a&MKjuhI`JTisCP>I0_R@>`L-}EJ%F;d^6n&9OLJ%X^Z_8=-@Gwqi{^YJgXf&x@pcUO0|p^>MGiGj1QC`i~%_v&4X zqB>}<-PJGUHO)RsA1K06hE(gTlu>B=_@>WUn;@OIi>?QXC3wO7neDcUx& zt&tO1k){(LC0gSb3EJZr6i$Vb?Z{&8rzdy~d}vHI9p+fr(2Ou>+RQrbEzKdydAgp-t*KcS_X=mFvx^Ml#Fnw16)aG0U829TbBD7jCj7I9VwJX_$AuO#0&GssHfRslaZOk z)S-!m30V|{K9LzFBr~ezMNLU!n{tY5Dq1#yS`%)3HHEQZSqIC@>@R6XssF|A-5aOq zgmD1|1l)(^?v&Prc2?xWm0=50s9o1iB^Q0qM1MVeX%XmQwB69s9e3QiNm4{3yrFd(vfVJ3o_!#fVBDjKOX4v+aoqjlgwnOn1{6BC_F#St&5}2A4 zM;zo#^f`0qC}$xU=RmRaM!o?w{V9V3_*8@fkl!jze^M{BO2V8amsoVQTte(61K~)M z-GFtF|weDUQ}(X!R-8R|`MXhfd>g0AU`Ee+>X~b0Y_eJRpD^ z9o_3&3)5!|4J2+ZG=!zi3={5>q-G1H!;m%?xGG!#gSKyfWj9hWN%IY%g-fi3i+e3x zx_t`^@UjT-ppnwi48aFcoMuY*k|;ICGfyyg+d#_>+9L{HZIqn+%CC-MzmA+hpC)W# z(tofAsSF9_cCI(q9i5lc7mKurOTND=g@*kC*(9HAVbAu~>(jjHacMZps-J zipX6iJet4$*^aWHU}45tQWOJZW}+t-W|!b3dBvuf;gvoZ00R`J+dgRV6jO}(Q_ie~ibra4o!o95h{%``_U$eLS#$Xrku|ARgBDETI7p5L1a z(z8@^5%=LN{qd*4M@e=6&=z}4Og5-oT3ZyAZZ?A38^ui`oaBkJ9%c7j zZUGE4Dv*Z8&Jb3|#@ceS{ClgFho|l$Bswm`k_th@Nfv##;!v##-} zv##;a&vxKnbG8*8wO9cuz~r;9A@nBJ+-EG4)C50%qa=f7q~xK=lj;`--S?a><=b^` zI5n5LqLCYmoc$}=2{Te#q{}Ymhwm%t?&9_7NA)TwD>7d$P&!&Y&(sC~9No|CRq+Nx z%D=8i6!O1o>YGkVr+=;beD&OQrimY~prdW*jmck^7v(i(oP=G;>9MX*bskdAp)2}N za^AF+eW#GcQ-*ErI*h=earJ3R zjI*1L&a3^{r>vWsdBKw#>uNs`#_vH{f{@aI2uBJ&Ysd@7Hf|)x8zhl$Sw)m13X_Ac z!+L!!83G!;MNM<*=!ZeNVWc5YK$GZm%k`kZk9td3*CC_+fTZNBz zBo`*n)q~3t%fBb?8Z1F!&axzx^DiSsB%H4>DDe$9z#DZnxKR-bNFh)l=zN__Gm&o` zQr1lcYy9)7eL3z+GuOKtYfeE9}VMp-SWx!9*i5 zKzpxvD;m(-uv)_Uh4dxkAO|##V_&QxPVOFG*gkG9UH4hSk=E{EOltT*QH4ajS#U>; zkay1UmoY06%G%m>*%O51HDrA(5&R8%nAVEBNp*^Q;=q z%E=rH`2nAV*#R|9Kwg#k6rc=@ndV2p1;(w8X)gHshgSdrX51=<*Een@tstvSex1QA zOP6+_vn*Z9!;_sN6PJjNED@5OO`H%zz$3J8L}S|W6keBUo66X<_H0Ey0aTofh_GN_ zw&SdeunHqvhz=rmnXx6B& z_Uru8L(@?t_uUDfx^#LwmohhuNrd26;hGL-P(fFqN7^Sose6ioT+Ckv6D~bsrkbKr% z3;}&AUdWj(c13ewPVgBp&WrtjA^9IF+{*8FuCJY24pX+$Ed41gU9D)IPIw?shB{i_ z0AL?5$UUF%q&aH%bZf@Ea_z?62}3Zp&jQ;If={Q(2)`MSW=f0MdBz0LB9XYylA4rw z)w*{gIJ1>^c!JE;$oo8X-p>v|1(ztAH2X|w@VP}zD8~%V6=Db_%A0(c&iSJD0C6z4 zXhYU$#uov8VULeo&tbzG<|Db3hxYR&7`_DqNijRP%mELu8JthJ|+ zoF3c!#S2bq@oHl3gOYk`XlmwXk6dPo8Kt6g^d%~GwW@L2DQ@t8Hd9<7f-!tAyWw1qA1l5gq(pR)4u zpYZb9e?eRK`dwlIqZ2CIb4Q&Wk$(cCq|_$bMsMdg$z%3U|JHO~Z>nbvVGIRlc@Kn6m@< zfhB4zII|aK7tTvEIwW90)CHWE!q3GVao=}i2YN^bP3y)E5Zg4b%t=3UQw1cdSth~3 z3*uj&r$_vk`xR}p#gCfn(#ukT45A?Z@mem1ZV2=X`Eg|+3^`=@u=NNRDQWJ0+DVr) znfbL~-{L`f-~4f+RG*~zFV75R*j^rW#i&^kXOvPCMs#YF?xI?lj%CNpO%Z&}kz7>M zEw-d*E~?^vt&7U(;7YY4Rufks<)Wfm%|(@8(7C8(=b|dmVT0c{AIdqk4?+Mj4LQ5Q zK=z*K3y~)|r}&22P!xPigPbB5Ye(d_-Y`eO>P*JKu6UZAb2>EV)VAmiV*~DthJk8J z7|y4hQ$A&7=yC2Pu4yarLy(k(Tqi{++EEeab>|d9$vFjA9GIY;<{`Yp)-keu)slA_U7sXoy4d*A8_K#2B8I7^m!Eist+9WgvoV8qy7c5Fus zmmPr^d7l`Z?!A>5AyOb>#6j56d*m0J;ccVRyg@wLhr373j0xQ%kDXriB~KnqN=Sr! z(+0@1B+=r0`68339mpb?G%_Pokh+$^7{sgeY+QO_no5RZPLG;PqGaaC1&Lvzh7-OK zhg5ILn>W=1&Ayc&8Kl*s2-FEZ7Lw)}B2a2gMqVxvXyIxTflL`o?KBaHlj7~q1TzFedfOObuAFcFs;@DBI^|n^~2vKMsPNME>AM(Ua`|$A&s<%*5E@IztZ%ngqZnBSH zqT^l)jZv7`2ouZ7P3H@;Zxr>2D@~Zfld;jJq(HrOk6bfFa?=sAMM% zTh?YwD`6_eT$Ab#))ox0>;h@*jdX^)KkWZnKiiCgV`_ILZ00 zhPmAWNje4N%H#u|1c7;5^GnIsm|sIoE=njfk+dlI+RA#vv?Bu?^X~!!Aut5M`DDO} zqca214K9o1*aFj-&Se9x@D~VXY-`uy+QHmZVvuw{Wuhfnk`W^dZH(Zu#iF5|Pd1Fg zZWWt`Do}|-Vk8NVASh>lk?Q$a^euw=-*0S2{Y~b}{|`?7a)LWmj3} zyVqL#b@o1wU5`K|6`OVTh~vhFcr(WJCbVLwRvAbtQJdSN9k+X2dtCZLd-a?$R#jpI z!KyF%@TiDXgAXox5sBIgqBbbvqeVq7*!XB0e1h2K{{H`OuC>?RXP;CB z3WnIJN^0*t*IaXc^PAs$esfMKRo$D)ao8!wHROUsh$a%IIV@Y&ba6z2QD+hR+$Cm9 z#Lc9TjN2e;8HsA)rFN=uRz#n>33``V-;kEfdQo9-z^F;tsp%ZOZVm3r`!+PqWFo_*qLNl}KHlv2B0i9J;hN(qDPtVuxNHWSBB}#fS%v+;ni!E1T<5j>9FWrtq$rm z(XLJs(ERv;C<<8Qdg**oet5G5pRu=MxWyk)816MMD`^tl5$XFtYG#ISHFA;SK<%j= zhNqYKfMB)p!4rjA`7Mbxh>2sRteODFTF z*2#2RuUx{6?O28h5xGvQO)MknWL73a!p#_fl}j814T4FoS1#8$1*~R*r7M^7I@#E) zS1zrSb&iBGC+ru{vVNKdlTJp$g-+%avKqxeO!Wnwys!Y!uXm%(%S>~(v=!qFL(7Olp)H)T zqi(M2JSj96Jv!Ms3G{eYHqWkAZ*$MEo}8M}sjgmlCV-tVVo4|+QZIyzzb&61UQV|Y zy)H#bNVZK7_?~%`?wGF%~r5`jd9#J4rS9I z0_qS0U*`jpzNiK2HY}W`XCi>jBruRz>Ow@o1hA8$j(b?xvlUP*vdRbC#&T|)PSOm2 zHk{g|dV9{^*OYuyqT_=as;49&9Y=7hN8tte@`arz>*I^EC&fp}$Ldz(AaxO#dIZuP z%G4``^XaM=T8b$K>u&-P;Fa5?^kHT7>MtV8ZV*X;`^fXz>F_?@i!!T;Cn`^je@Oo0 z6x7|Bf0R;rRh!OyRtC5+OP_B2K^qdREDJ3s70&Y&Bx0A(&oK7^(i(lBS{Ea0=;GKr`SBs zZKryXNLRB&IJ9P@21gG2U2-piwLw$U*eA)A+IY1K;c$g0Lx+#;g7soENWOk+#sw>y zzX7*JG<2exn9%cu=@s76Q>0>xsH-=B+gY5!W+#>Qw;+cs@nss*TP@Y|MvaYMzLKri zZ1`@n%aUIRzY)xbRU*qbmsC1J;q?n;*Hgx-&0M__sIbcv#GpFKyjm7fYucd2R~Bh{ zr?Alc4X&2;DrNA3kYlJTgneN+Z-gcPVU*|dv)#rzZQqt7^`V>rx*!+z1lIA9y* z%V!9o?HXGo!?b=OR|mB;=*uC}|D*cccbdLhy6L*kK?QkEPFI|4d@be!KkqJA*dPIJ zliPvAX+@)4W8OT}yhlp11hEs0XMh0WH!DRr&6$pYHliAU1Px4~0js@vNQU!}u#>F? ziQFHlyi14V^=(LSd^C9tAc1l2QR*|%Gr`Gf5;P{o7wcr3) zI~rn4Zd0x$!dMEPxIpE2-MA6v<3^Z|1-)mxRw-lRE5>?L;H-2R5MU+=BJ=znec4}_ z@5@Nq-5w*7uJK3NYP~IOG-!GgV&yW518QLcCDxJz&oMo%4Np*A=V-@k*Uw&ia(XQu z9GN}1wQeu3-d(RDj{n_SHmXI_PByc(P)6p6&g4x}ww3cevh>3XfO-2#Avtm)46Cp9 zi)fQ{o{4R7nNX5V{X%?(EtPgh59(IJlJAOA@YlF%SW!}Z4y@)TiF&~`TS=~&JX$oE ztTz(zS*LSE61k>{tRtj^7V%jkJSvxQ78|9s<%eOe9p}7dK2A)B3P16kO<5D)rb!|O`sR2dzSC6LwV^HZx<<;ZLC#|8gdpZvtmnN&yrd(6@DXe;8MmqDF z!E6pu0|U-DrH!FJhA9<1{9MMUW5$?Pe9thY;d}oopGo8eTtN{R3lipMU&yWj(e4yy&bc)yG;-)(Kf2dSHu%1j3XAezB6bokHHSxqV^g$)4;D-EJ zObUQ7`^-Fnh3Jx{BCTpW(nF+$3W}rX9m>ox~1RcIiQ(-}6D84;w4gocy7h)!@HC09?=T(S3r0z@xmY z>SI*;>!<{kQ4ko_yB||AvAp5rGu#Z8(l!J~+O*f#Y}!j%p{{3ZK5@ z80*^We_77p13ppL(IOuwHYG1g!lqupkH{#X{;fd8 zl5)&+N)#U*rgUT7U%0zTHF#2uID=@R5sWJZE8U20^%&@i$71{?oJG|VbDfr$>x7u= zRE?OEUr>v=P9x^V@62@UXa{(cckMS}$AtH=(&A#$i?k)!2J4cKi=_#_BI zY@&vDMk&8EC=8T^N%D``9(NPt?^;%;|L29&ZVoJN|Xns zfJ_;?KDyDT56K9K%}?K2G8goH`}82N#oa6wD4TG{dVQgB?GyrnL) zgjSi8xg@R}sZMGtXOOd|KzMcwawm7)vXG^tw&PSo!a*D;*y?rG43!!^+8N)Yo#aL+ zNaUE2jEYJ^Iq4jC!3YqVJUYT;&`FIIYcrxk#o7J1U~eLNtQ!VZ45521aDVrE-u}D2 z<1}}OedTwipJkE0+{qe7oQ_oXBrItGhaM54!q)5T9w##0yxwkk;Za_ z{X5g03|2v*H&%UhEQCF<%S4!sF|tbPIF?4NOBLkZRPG!H!1NSFui z!+mTXOkJ9Xe$~HA#?jr1h%B3ut?DQK{C7V393(ljV?t-$P1z^VLVx_D+96$&xfwt> zX6h3DQH#dBcoDRV4T3?C%yDd3Gi_`Jku+ktbP%)!r|2Lb^YVcAWkhQ&%>?Pg66>Q0ZHAXvYD~^3ptXBnX7&!n3dqQE`j6KXAfaNUS?bHiZN!nnn z6^HEx93mqD!EZ%vAFFaI+~)Wh!Nn!W6&7wQPLN76dfXW<@e*#F%aFk)Eoc7AUnXM z2@&AL_`q<~j#u#3TJe)m1ga8Hkm1xzJKgIQOGVBx+kBk~;KOSCIFs&E$>N#=4nWf* zMyAFX5yB*Fi28SZ3}*e?YNP&9AA{+&I99VQ;NqGCO=PaAmg+I?>X*WN(VC=WJd}}} zT6yg2+43?)%{f#l-JV)zU!z%*0U&vTu@L%{AJTMu*fMknR;iH@rBj)7q#XK$cW`^v zhECE59aQ)rO{Ru4BXg-1`8a{#b_J+d3XJ5=5NqL_I5TgmY)e12l_0K=}>KfMFgopW1pm=jbuL2;Kc26g&G*P9Z#hKWdB~K>h z1f?m06=NDCBLAdtUgv) zV~P)P%{(6mIWj|iAJRVur~9{oMYCM)gN@XiSga8xTvA4l(zers<$gWd4>YsYSj0aL zmYxjD{Thh&8wAD~2)C6IoiyEu(Gt%wh~T*aSDk4*PAKsZe7z|UBrU^%O}+A9@Z1{f z3H!xw`@u832EZX8C!1ppMRSIY!7my8I5qupBAd%*2SF{_Cj6%&PrD1e_(fH{F42ZR zSv$!MW15q|&;nzsm>bGI(ON-&BVpa_8nHQyY$P;6U5lv6FWA?Pr7my@-6%6(J;B21 zm3)39e>b@tNDWDdS7&@8afrR1(^r6aFcZ!5)cL6@tKOK!n7WY>B^bjy9Qn{KByeN>*VwiAW z3iUD*l^RT$5+98C*aIfSH#r&ti>Z*b<@gZ50|{mQia1U-`Qgcqmyi^K5ehg?T2LbV zL?fAorGW*F$U1LB$t(+~)dot~NJ7-%c^(-`u~w`Q0FB85ISH!jOcF{1Lcuo?4e1MQ zF3K>a8q#Rp93o6vhk1oVNKJt%PI*2H-Uta1yrHucyb)&Oh#7=6Snnis z7lCf+N1?lBNjY1J!SE3rDFz@3!sxa&0Z~wv(2dwLx_jVkmend@Xo3KjsyVqwxDo-2 z8)_^u`!$wK03nNz-=XjtMn2T=vD30?5f&B`VnqlNC~uuOZjveVD0=B=+(LR9JUWvH zj0`=%7ef_ni;sJ~Di$@vCKpbm%{%y#yXWa#Hp`5;O!H)r1g!%MCSPp{&<;u<=C|

1(t*}R^$T3&dx%+(ZH z0ITh4mIcAvpmGMJBe!c$6HK%`O@8tA%+nN}YB2C{n1B2U=`mt-X5V=?CzI!y-I>I4 zzp(?AA1gaB@*=gQp$o1hHTx$N&$goGobugl^Ky({is4kV77GNyS{aM`bVH;;*1cNv;Q(nI zk=iLVY_5uV+9Lt*)suofXO{F}HEfopushGChSyZP7rdfN+UqN@VPN!uT)0HwVf>f` zxyM|byH zJXFwM_ENZ?cvMv)10vi{sDN<4BlCj4a%tLDwbVN5j2UEv4&#h5F9^E&HD3Em1dhmFPr2Py3BCDn;X z+V7$2qIqnr$^R=imZ4kUw|!VQ2nH8?*!nQK1Uf+z^g*N#b!C9sU}jXwKhcSG^X@A> z*Zzs&0PBqx1!}ouBVKVd1{qa*QQF=%`E5Jh*x9x(2QPPjv+gt^-H3UPkiEQM!$Kud*{pgwh zjjHQqkE?0_cdad&FL;j$G({E0tZ(hDPq!pT8T^R%MZ_3G+wK>Da@2`QQgk7? zQm!#KmV!wlGd^~stI(kjEiWrj+`aYi;2leYu+mQNg1c zgXC!rbq4*+dlHG%&SZDyr1WOa<<+4@mh*a!Elno zuCf;R-44uR2ZY(I?XouS*JQ0bii^5csFrfbXu6>nH{===vTONAkxw*Uxq+jNTA_ki zVivRI(rSW0x`Ihf;}z8lD1O6v+u|8P(2m#A5@LhaQq+=P0vy&@RRiIg1lVIQWt4y{ zu|kkp;Ao)83FRFimd8(3$QjT!k%SSn=7W*SPATFVav}!Ef!*yN#wil(V5nQSq$qbc zkiwxj08Se)n+A?(E!S#Gt|O)gY>nI_0mK?44+6bJJY2x#fCGZnJ%x7@o{qE}`UT#M zDWk$t=(4#X;Jq#zg3Ut2uUzL?HAueWrz`5aE{XPko~+)+O3YQq#$)|V->>JIH;eX8 z^$nb*TRk2V2vrRMlw2nGeFRxgIwo6;O%FUU?cM*@ql#0Xc){DILmre@JoguWn`T$e;gPJMBuDq3 z1McDdW$%8$UqgH3+^cA;ry{{HVBCKE13&-lcdxwSXK#Ao<~Kb1+dt>$PX5wO=N`DZ z`#0bF@|VB-xA;nfQSat|{!f2z|3`m@dcRxz;`co3X>V7(fA!Moeb*fORegQwSLn}& z_bZ0BT6^PvLaNzp z!&o`nI4D_#iLV=V*H-Z>5Z`KvNHu_lbXP`OHEWS23MWxUj18k zyk(qR@H5Oo&MI!)4r@sjSy#jAb5Zn=94T+) zW67)0aUM{*#Zx4z4YU=+DIKTkBg`@^EnE6}%}xDd3BxEO5Rjk3_(nh|gx9&1^aXI3 zS5NqQR`<%0Vr;TVz~+F4H%D@DYLEoa=c}zj82tuad0;z3V5ciy$!4%5(2j*zqBaf{rML?tKH#<)g@eEkw z9aXQSseYdECkG2)L@(XROQ?Mzgq`=iBuY8SOF~2aQb#X!96jK%<6VBN+Fy5)nI5oE ze{#CUwF7txoqx{GBDI1Rt-x!!{&eYy!3C81dbxh|rWa55J+IvN!p(i?;qqkl`vhSB zirJOhF@zBDV7IF4%hhrpe@X{!I<=k^1)ZWnm4C68@#_i4 zV2BZ;*-yjRP?!R1VR`pTVmqZg1z?}~vfYz9r5f&DQN_aqt?Pmi5is}g@n0q`2myO3GLS@;cK>%(Zkd-2)-8`VN z#v0`TaLIy^nl zCP?-CF=qbs>)KsBf3j|6XzPRV!6}lxKuXA?y`JIS?Uzhf~Sj!2`q-%q0aeg^^cHr?dVP&Y>De17h}WA><2$nnBxLD~U_a3Q@$5%Vzyi zow*amkuy-Gi5Bb%dgQ)p>rY({-6a9HW6{>OPQIxEHM9PsU7O+Ny#^Caefr6`I>U`hK!sb%GxW@c5J$Ju5tR@IG~%Z;LnE*-=VXC( z{zr>yGaX8Lx&h~b4AaFG5QEjw0ZItcDd>G5Se9Q#+-f{&5C?V4P%vnZPMS=>C|^zU ztE0!rlj1{K2KU)_)92-ZZO!SdkxvgEL4ZjMEJzH+c}wCuDMmI%8v?*`?iqWBa|~jT z$7`!$0hYntHRacoXcK(ZBL({~mZmGL^D7jo0B>IwA4Z={aEgpR1$CEDPu;R70$D7& ziy7q9sbkUBXMpZl3{K*E^OU?!TY92HF66}WuFX3pS!xF|$ZhEm&%VC8k?QMVtPQG- zlQwUReUZdO<_!yS|fqg)T`BVlJ>!7xfE<^XD$iij<4oUYUFK zOeqUNLL)`D?c^ni0cuqhZkScaMQyS_`otSpW4)_7^z{pkG60cG&=esj_aN7B{J2i8 zDQ0U%hgRWtB}swH&iEsP*tNjXlq^0b3v zRZkF0vJxbFlGVdIC6HIEC+OrRRJC{YYY`=d`YE9%fa0jvt@p~6-A!@)tjP0`n&M!9 z)Rf6$R87K4reTOdz(LLds{Bv7Q3f&Xo<>eF>rX>e@hiLKu)t9pRHhv1z!$V2nTxgS=YJqd!3UiPA>~kBfuej4O9)dk`Gu)n4?5L#c82?O z_(WPCv53#t(qBc{OWVx-4#HofPzoco%CIEdP9`%{QeOsNT&AFFyS6e|z|u z(pGQ)OS}-4^tQ}4%J_j0ioJv~wCnL6^fW?Jg9C&cE9r;X73(ss@dhgZ;#(O~Gu%QAtxhdu!H(@|=*5c8X7p#0C{30f{`}it z{^jaffZx42{rYL76Qm`E#Qu@iuqX>pRHDokRk`fmFX7kxekNN@Os&02o)7kfY)l>H zTV)*I14=}*Lq!FvyOqa=!9#rPFszJ&(W|YUUNuPP3qb|CD1kR^S@o(X{>t)s<8|(p6{4M; zXWM?G-HmeP&G*acxvM(&!LvJ^EusX|$ph1U>D`Z1&n)Zjoz7R8F!ueBeJ|jS;55O` z(U0HXyz>5=Bo#^>QzD=5$N#NJ&KVpldPOL!TYUdb&lI7-K6uur(9o$x0K;X0Ggbn? zdLTDFv&!x~32oB|W)Y!OwT)rMt`*v5+#qtlNSlCCHM(QE&g#K-b^R%sS_J!%J}Q@` z&#EpSA0FzMCFYUhl?V}OhjI;;5k`$md7Pu_AIZGMj13ENZHp|j{nXR@l4@+4F*UgR z1M{{Wv0W8_3G89nersj#7CIfljG>QKu_|c^KD1w__$@;kZ&jb)K+VNs8g1Cw3{fK% zcB;|W0;Y#(XT~y84#mn=b-<%kxE47;XCp}j4UrLv#YLnxa``QTlU?(g8jb*%(LWVy zdmlfAhZ5xDu=Wk&_0$JhnCK#YqzJ+cJHv@=jkOQpLI6%HUB%lNx;=ihx&neH0;Ti_ zUrT>YCa-$ZKI=4vOj7KU$9U2=fgpP~L6Qy!;+GZ7iXdi!si@(luSrd`)uhN!OqKLZ zNOF@z3JKu-%*?a)o0XKwC#ztcmd@-n_AV%p9VJx3MixI%u8b*Q*API5$ulX3HY5pn zhFP1v*sej&p)lquj~lah0eG50fe5JD?3K-e-;it;;TpvvT*>IwwDNt?=$-2;VRg}L zSulFruY5{a6=*ZFMMm$;d+v?z84a>C+4ET9eV#rCtwr9J9n7d&7B0g_HT+-tsLftk zKg`FXVu@&~*7Q7i-B&E=l=ICa8&Y`KOeiPS*HKk+s@tmZ@%Q~o_YSo zzk1UH<>=@a+o#?DS(5bx zg;(D^9pC)8uXxj2et0`S@<1}e#y6jQ)pJ(<*-J-99%v-&(ze!TZLQ0MK@&tMjj&DVX&)qnhf8?L(Pt3=2v?sn(w=Z9wG&D#*^sM9+)HxPt`CrT0*Ni3vg4>yjT zR30Vm_AD(ePN>HrAK0BTkk%Vg;7qeufgBxn_zH&J5_7t0UPRC4$!!+gHp)JSrEioI z4mUL8Pxq0&Llm@n`0Gxlxcq*)oB;Ixl!U8=ln-_8MCR;on_4$_b;nxr;*p$OM${PdPo z!LVFc#D@jcZMwsRd;;I5pp3-%!?aXeKpO;D`JaFm_0@sUxL)fK%C4)P2>9d_NXoBA z6?UUi6_46=Edke)#YGhqbT3K3Im>vzMAsnNPp-eTC6++#`p;EgZ)D)2EsYpfB#l53 zJy{Rz6X_6Jv)`6vpvXfqa2+7|ZW-8eFz5P8vALOp$uMS)e{Ej3U#X;77640SdD{k0 zWktJ8;7A5S5OUq-W#C}^)-GE+(Ta-vHqJUn3r6fnf?duUYKZ4@t!yzc{vOQN#E$$m z>C(B_k!>wWU1m_>6PZ(n0dE3eXmO0czvNAiM6b0-JC*GoF@fiHJ-T!3Xe?bCwv~0M zf-(^W)R#Jh6lrn%mhqe@X>JSwyhKbBLc$HxSW>L01)^1DDxL=(pr9k^Z0N{90fa*A zFwYz%30tBhMNjx{gv$b0f}|R+bB6znk}f-51saq@2&tx|@uX9od&OCjiNEKb9o*rL z3Vvd-;8*4fHagY$H|UiYqU{uWCd5>KAL+aAVi^1De^!<8K~!s$A~CZNCgW8{rY67b z8qe=$W6lEe4tP=`J4(L?|7;?}(VN^W*dROEwjyK`bGP~m)-AIRk?$&F>8Iyh9ewpw zo9EpUG9Gd|6o#Uh>(vLZ{Cx0rENs(cagUI`vFySFTbew@6RwF@F zH-KgyZWh(`+O9V>5F-yjHYLkX$fTV)`zov`9;$#Pij`V-H>~nC zFEf5`qmQ$YKtrS=b;i#xbO2PIEg8)8CHZU_Y?hz0$dsab{NxOv=87w zk#LwjF#>kQ3(W}uni|h6QyO?6{R0il@ZxChr z=1llp$7q>@fct*}Y|xsSqD^385_mc@;V~;nkuzK4-zc(vq-=X`VJMvsSKafHv$g(^ zgoe-(iJ;WSP!4JRmkKb3^eu}NK zO0`cJ3);2{hGYxTn6{NVT>|u=D~_Fvy*#jvc0DkAqtcTIJ9;AX3R%YTvdD&ON2;Cs z=;CRq2koOowXyvp&N)F3r=R01_9R8vN9KFi3?{(a1$TeLQppaM|9D{1qbV@q`c zO#1C(BQpdar$=J`%C;J~&))b0dB%q2Mq(g*Yr>V})jVomahhn4`;+=ADRqY!i==u2 zgrpwQkHSXyEquH9wAvIQyf=F-# z;Nll0F(k4Ym~fLBYb=jnMP96MoPp3+yyk4fF*Mf^aF`FwZOsbgV?%RvRI@@?E%LfJ zkK8I0Tjks;R&5fE0f|UMx`-xgNc>*?Qt?Z^Szg z0<#MZq&-C-8C<7f>Le3k*+12|p#csxj@DA+Xe&3)i6qs1uW80zW#bQZO?4I*p0KW@ z$+yhvw1tRvRM(=#tPb6Y4!xh%w|a>ZC1|{IS!P%%P92h^A^eheDC`WFCvq7AWKIbs z8q8eBe64^>rr8{&`tj`|uRwpIPd>Cb$%l+C>HCG-%p}KF{J;1dX z(pZ7_(!`sW%Tw<}xL4?1aek-+Xc>h_K^QbT_=xT4kU`74X^Xc}h&P-Zw**}wc0ojH zN{2_fz2>)8zGvc1N6PVg5Aim09mhYCXT!@S+$2V%K}d%|Ye|JjYa#TGB~4 zh%M==i@ov539=-`1j|tOY$aH<6?T|lu@~fZo;iUPa$1XoiAv^W6`W6!(W|nq_Css$ zjM)=QVUBq_n?6gcxM0nI{J=}$uxqALk>Bip1xwx2)4_R}Xc%e7yp3t_yrcMfY)>kO z7tS6v#5>fP79sR>9()I}2I+)hMw>eQ!ah#YgH_l0 z&?c?%Y)otC#miDIF{Y#k8s=b33Zg*vU~5ZYlxzQ#o|2&>i;&ox&vh8v>5+`(?z6PN z`hF(I<7SE&$l}z<$3gVwcS*w=nbHZH$odOD*fAUzc4c=kgt`}EB4V;If##;_BgzWt zyntF~%ihlUY3zJN+ZOn%@GCtqR~>zb#uyjSAWz0WvDWJYV!{7wbx{5#-R)`r8c{&U zlLh{5>gW&wHTNXdAL}goh}M54izHcQ9%F>LoI@++eqQs1wFh zwpDc}ZwFQW%tPQ$%aOIU7djiak_f89ac>4tDQIjbOmdre5}m_ zNvuHv2MxE6P@dsK{GE|YuBa2``xTv59pYyPHH^(G^yB2_Z)=p(!{vT!Qw|4bh-=5U8;aIk6yi^@=fX@BP;`C) zl4kK&fm*-G&_?CK@=CL$BEf|Sh7i8p5k^v!R|tiJ3sX&X5@ECFLXdM(XOsKrRY>R)H&~--=}FL^6@=QcE0=Cv$E7 z;Rjy4)7f`c$^hJifYvo>7>m1}BICPTp*YF`+O1!m{qD0nwm19L`*ah4-Z^_eEmrrw zfhU}y^j#AES5@D04?nBVt1x|2jmb;niJ|a2llxp!AC>u`k3b;zLSsP z(kSKot0l;dj})k}@vJVQy%P)V@!RdmE|zheTJ6IK{BFVmGBWtxgbmvLyU9LRG`oMy z2ZD=^<`xKT%p?b?2GtuidB7Q(Sf*1mL=kCMq9OiPXlR+r+LK)U`&Tt6R44)W>UPVG zSx8@NGV9$=mKP+HceTJduI_fonRTW(;EEme9TmUwOsgE_)2%+t&c|`}vS+u-r8%pY z{3pNgksofqk$DmYByc>p-RZe+>#Lpj|Jid|4Y`Ad&VI3;JoJg9ttakc=fVfNdbIPN z?|*LV8H2r#+_ zG3KE%EDzP?pn!4@lX4H!Qc&(O=GPo>9nP0^3tp+ zoG3$z$x||;b^v13NVAEsxdrdIl^_L^UzWHX|JpcPvsVrq{L!n=&dF9&Xd(I9m3&1l zYtnocKM8xNI@ZYRzZBN825RqXyek?nYvrhh zI5nnk?5r3&1YTWflq!vNZD-$TT`^i+L9$6Z;oXi;szG^4{;bBF(7Lw6vqCxat0TYa zpb;);*iP=Av48+EyT*+Ap!et5&c&)Qs}X-NtAQQ77eINo)P)aRl~&OtunN^!4q6Cg zia7f{EC*s|x3N5j=Ep6QbIfyAinQ>klr_h<^R(v0nbzO^6r22*i=onubFdcF&pYTjLnYD275p>{#%itE938mf-H80$?))V-QRuun)>$y%*Ed&7_?Ny`WFkR1 zu|#21U_!SQos0{7=du^e=C0CP?^)r&c}2tPh0N5I(097zP!lKKZOXwR7yn^b1dBK2 zV6h0kUI|)hp3LkD#yLq!)}6L#t*Oq`6){Oc1`A!!5#H>XyPu=K*)ztDgpgl3Bg5t? z&#>0=%{9fS*El5to)X8V09A@V)l7tRaJHxtxg^E8y02rdDo5-6PR5@Oy(#M5xdo}o z*YQ1Bo;zl^G%Qtz3z!k~`v#{?zwoH*yiw;>!^<*%{+=(tfBxm)DtZdcDQ?<5Q{5|vgvki?H0>(-#!{&1pAd)_a&zJz za+IV_I{?hOy*x5cZ9#+pJOj6#^8N)n?lHAprWQkj$G3wzfQcfxe@rdfI!&!~vZrZo zYBNu5QBQ5=sqG(bp{+fQ)Qf5C$Ci9u6EoGS4w&pSjRn(4)B?l_D*{CLR9r|={9VP4 zuf#%^Zt#w26CsUS{bvwtrfHMe!H69dQb$abW~k}L(}omSC}i4p7Xr}kLQk841#1D$ z%J|QVuDFq`h0_eO8&eUJgO;cO!CdN<#4a@y4?)Se9w9f>3}j2_prgfieeg~&wh!$k za6x`7D|h%^j1F1opR3y_GsQhuj~(8+pkB^N7QhV`?kW4Rme8x48S*x79`mvSk`Q@% z!V$|JL6-4_>DtbD9pkx399Tz~j1>q1dz3#I4;GfN^z|&X3A$Pslx|HWEiX#RGPd>&`20!; zE?{_mW75D0#Exb{9VpWx{Ba$AuN4x2eIiEoiW&E}Hee#g;UWmdp8 zlJ6FRl1D2Lwv;GDjJSfIK_U%cyNIrd-bvAF8A%^L)RK&AkriQD?i^x*Ic`b>wU4h| zesuX^1&vkc{0ai$`(1IrC}DfbZYe5KlqzmOgQRtDyAEGOfVC7>BHz9eN{Wt5N%b>D zi5N{vr@-o#yl#?Sm=8EAX6FFe2L~BvAv9fKPEopme&5BFL&68*;|hH|Zi}EkZpB@j zhd>Tl^^r1BKAzDAT8ttsma1?5KCVHFN6T{Cm4^1pK+BaLEk$Xx+8T-7Ue_^37V8jD zukpbb@F9He!`w)S;7ckTmWO12vn5P(q?@rMn2Rsdg#}QV#-#tb?vLx5%PZuKY+i|+ ze>`WHNTlWEcs7W?DN zt5O&}(2|cij*Lk^SYCk*E%^`Y=zlP@&d@Qw>`$VlVH*qsZSGo)$;O(8*g%!;{jwuB z@wuvsmv9LN`w#3S=p;{NRvn&B@NXHHl20aF?ZN5G&TAd(38`be4d)H-=ENGFZ0PWP z9-OWpbkhUcBE3gdbCsZ!w;9>p+i|$sDr^a_vaZKC$K#zW)ADz@Om>9GNNG~R9@XF6 zk8F{zT0MVf=Xx#ckd5oKY8g#vTe36n(@iWAW>sfH5)=1l@?#2!fKv2KJ1kyhmmLDM z^P_~|9h_Z>0_8N!(*ypWox@qAfIJfP4w3-QOM<5SoUCH-orDgIpcy)P2_3a?6#Ez78b4W?)t3UmWZc_ zw)ial>^xQ$)4)m5C_3PE|DqN8Z@O+k5UofD4GhA14FdCPYG{FwwP{73>mGo}`h*_= zNxSNR1NXb0X5^McYsf7WR}f90hr2BoBoi~Fz!m~(AJ7SrSOQK^?S`+xq$mT5NwiMq zY;18nMUo;J^|1oIoDLM=o~L*xe%7iere6d7U2rRMUy*PkkAx~18LBN4LS*4)p^gif zmL4G$=lX9~RI^-CQzK)s&9>OVUSSLWvYy+u)S%4%DMGKtXTvtb@w+y%QJq1=PRnla z6EoZ6t+Zwz6wtjdx32TKG>az*XIAl zPSEC=LG@x^elU9+PNN%wWqfSfG-bM-byDC)TWG+3RhIH-OB1oE!O*TU=k*#1Cci;)|L z`rc2k_Wdf&0#>&kI{b89r>9@Sn}7B7>o4kG1O57V{cEUORW(_8Ad5+X2Icd?P;~$= zQC>j5QhmsPbKjiKcLNQJ`g35}-dyO|_rqM~nF5P8PL3EAS2 z2wCw}?P_wwGbQuL>k`EDT%_)Zj=+fzzM5zUxPlT`wZR#xMw9Ooi7WakGuIWSnP+Up z^ng&}z5>3QCw!?(j~vrn@s`KcEp~j{ix%GE8`zj~Jwi!Mpu*_h;W}q-03)hSK#{|D zfXZy*SS2d6oQq0emz=i0`Y#W(II89}zZY5)f*dOySxLpV2fwlBrQ9WOw zbyI%483kCU(Z3;aE4msCICU#E!?+wQBsI`tp`7W^maPRA@_}_L)d~lS5S3#26w4PZ z-eMC^Yys|YI^L!E5VETsvQj4R(@#k$V|G<&IjC)dges)X=!)%wAX%28v_4@Wdu`ej zkq#kHXBD~a+>_j?p6&!;0A{-^CxR(30x`2&oe_|t{50l*UA8*JeyZ#;9>@5pW@rPj7IfJZ(W`G1nRcZE~ zj$>+IqF6AQj&-k>U{Zw{)Y6g=dHbQ?`G;1+GZgqw>-|9q)&rf%yRE(u;SkdAnvj^2 z4$+cCS29a$l@7tQuXD(+5wq|rQUcmzLT=%9f-0N5MeSTAt+2YO$MyScSNo%b>=jP3 zHk-Ut6`9+L1R>}(o4h?0C}7K3m9oY7Hmq3BL@;2TrOaT&V_sdCPF2h%@6c;BQzG*6 ziWANG{zSfO2XbC{%dUiX*>5)O`AU#kf0nP=jGa$JHMm}#UR>%*)E6MN=)S}>0D zmk9Bt=M?*ST4t_aFXAz!e(IIaQ98Uy5qdXHsh_4L_qu6zXsK&T{Z!$flqpfS&fZ7W z_^{5nUvYe>inNQm3ARb9!Ut0eZ1pC&W7fa6k??qSp43p{Kzjfpw-s|w(FX8nQ)=?_ z!gNG9wHl$>`_`S=?0sJ6tbX0tk@&2W2-CEt86!q2F6RG)42b9@YY}jElIbjl(aPV348zZ&Pk;#{2jjbG(zmJE(nkKI0wnO3gdu zI+~Yeyi+v1(~qGCssZAS@9TU~$MTtMkNil5yeAG5+*?%Fm&5u55-eSTw+WTktRco= zAeX=z=r;*2qD4{!kAgMUxLI4UC)Oazd!99T?xV>PYw&!IHD<4yHP*D`KVyyA%Vv!= z<#NneWA@anv9_BvX04kw)^@SRtO>Km+5&6L9-B4Rma|5j{SmAYX?gew5RB#P#e|fo zWI;?JOdp#I*68NN)T5pYIbxm%4tS_g6Nq~FAstFAa#jAmD{M1U;9R^eb=0sAk`GIF z80^FM0k0(W5{d{Zn`@TU8gkNy7n1ayI7p@)b9*qhGRC2>r7pvwz7M_|DtCb(P0}*! ziIrHVN?a802vmSu`l8hjFPg{vqS@=BFIq3PJ2c;2^ffE2c6VmTi@s(})_%=O?4qwV zDy|KvwA-baqP3^#fD#!qoYru*#y2^SJ zveZRrK~dDvwVOUIiiHpehI6m5Ju)spmxQ+RS`h0}9JEt< zV`IPi+-OZ8=mF*$DNV;P-QLwYlpr3cEA3|!T-Cc}an(9GSweUrGIZIfTw-DIwK_BY zM7I-`=KQ}vaC&QfWjN^foLlTe)Ol4y?gZI)gclgDgHfCEvGRnucvNPz;wNCv;uI-sf#JFO&;KEwXvX*W+Hzy^-pxXkV@3;ARL@T$ehmY9KM&7Tf5u6OFAdc=S%0Fa&cTW5pHYS7&kl2uycv{+sG< zGs6TVXJCGS-cA#I$Y8|E#mRK&kJ$B~lo}3`^wQ#+dR%q{e6DA6%dA?6HGVHu3(Cp|LZrdV;q1&q4dnsV-jHj$e<15HyWT@29I`>O^_!dl zezRXIuWJyhyZta{N8mWtMTDnRG;8Hhyu}%kbroiNzDgNY*lk}|Ap_J6uT-PD=o%G` zZ;WMskh`j-7sd8kE}RCx3NIUb`f*Uz1^6mpBxW8`s~ zN89W(9{-fsSHKe*n-;p>gBZFnNuaIB<|Yp77oJ7}*494Qz&xj|2koonq`lW@S-uN%Y zBzp0Y=Uc2@(+Qjp+qEZunlF&Ce$SJ!5{Gg{wC5-}N~Id=#t)T#caE5M*D9md96i^y zsyUK&sl<$ABU`SxZI{I8@c||?2=1d8gI$@{qrc;& zcL<-erR^*td_u4Ygxuk!1d_{q%a;&H&`)|?ng%8HYFe1)5DNptINi{I_&cy(TrJJt#RUMMyF@QpeOU^H0Yl1<2#SEp)Ud&W-4rfO^`EWXih; zwG$Zb+~Q>Kr9_YPCeDW*nT-QEC))2XrF|^an)bUO3RFScQUOptO_}nAVabo6eM-O7 zJ7~1*Z`zOK%}84a@>e)CrF|gWS3#e~s;qg~m(v9Gvfjcg%CQvPoC=RXeejT5Y6bB-}NllRptB2W3784VZ|SF=TNb zn`obfb&c!UIjX(bfBv=~`QeP)VG=@1NL-*n^rqlrBf0+IAHe85uux2l)Hwo~tY47U zjPn<|HIp*i%vhkvqVyodjFjON>)QSHsxor1PDsz3T5Tl7wbKFk4xv z<&Ab$3Tg<#*p8{B*4fiYa-G-vUG#pJ)r4wJ??%YTKL*m(&x+m(%B((wm$5wS{S&8AByXi9$vq3GG4z2!;J9bcwug8sP;r z5W?vezCkbg*0eV|f$&Ah^qzK#mXpCCXkA19tWI8|K}iYfS1sBl*lW@*t>5*SB_s`* zZlZ-%aj8V@E)Og0oVo=i%q>{lI-+JqG33nbvbcM2o=$U~=e#Iw)6t=LcaGY2*D9md z9Ou=w=Bdr4qBhP5A%8&BHpP+l);Q+KJ_*avriH7hR$E;`6oDZVYfV}TP*@Lks+}@q zAdA;!r+TP7wb^mpLvznF#VU9JzQ^PD_xDn9nx@*EQ3^>;S|McWY0bimlC2&xb+C>*cg6 zr7YZDVSBh0*8f#PBBT~Yw*RkE0a{>{5#X*YA)zR=S06d$BDaZm8EOVabjI-o)cZ_VTuzmkL~;JJzhDH>FTMY7T4U5o!nRwb!U-#<^~O5s5?K8rAx2RJ_Z$)|ZS6 z9V?N?aOaG3nTfK@NoT@nsvf!?L5~PUu3yPGsQw$i%AYUx&ho2EzCiNWFso#@Zm5CW`+*7HszM1nAC`0 z5PA^~*hT;3nN~XwbvBV&>YLO9A8r#rrrj?`B_=-bT)X+b1sl!d_=+eL%Lp#jfin8Ts8LfYAZu<82^QpUlR{Ed(x>{HG)`p6`I)gbUS3l6rdM&w{&zy5H zg$?uS<*z;4T6Uo`d+hIe&8$=*g#4mZMvca_XMLGmba`N)8a(5xFo<2tJYSIB^SG21hGS$yl%OYfs*R4P{IuPd;yX@~IM@Jcq`vGV48Q zr3lqa>&_f3HK+h@#5Fz(yemSf$-A;|)lBaAr?a^gh)?&_+I*OwTA2@XQ@hv1mS}de zdVj|F8&a*3i-x!1TPi1~4F+OMaVJsU7) zX|LVQVId$Ss3eP(kKvR&_Ypimt%!yqJN~o4R5XN?%rER9JgDwyxWpo)c{L9pGfsSk zBM}P7nC;1jJhmdtBaRt82&llrx2qd?NN(l##4GL9$_WR-sin09R|f89%MO=(Dl^<3 zS_sNb1G;0HT)c{{a6S?)_7$F<4$rfbpyXiatQ;z}0mqS%dQ_QKum=u*B>&JZP4i~T zgczO=&$8C5xfCKhmK9K`%B`4~vO9jbSndtf^MyXC&c5z!TWh7&u(sB!=W|J11az;` ztJv1?*R$HzTF%e_Pi#7~Nh2W)vRWwu%$bEqnSEY9^x9X1ge(?BSsM-~qchx%X@QjZ zTL1Ai>A$N0!W!b!v!(~dEUf7*i4;I#jn>%P&5(JHvLeiZsEoxv;RR#I+E6i={w%%# z+4i?@L589-z+{6jN7ELtG-tLv2aO_{hf_XiEz9*CN;tEjKs&5sR6Ds%ap4)4q>~fA zAOTLS40yiC>~RD{1!1L zfF!wjIr-4cr7r6%5E>^SWr9OYPX3GNfb12|2?yvB2;`Dc-XfaSmzgyXf@E>*Jxk>8 za?^B=ipy4>lv~UBJLrMgob_VPF1nU%a|J_6%z&B2=9VQZ$&`lcz-0w#CD~U2$s8_P zi4^smr8ywG$x{K7z1|uC^Cv8j*-Y|HI5AHy6+%qj13jl_gwSjMTjq0talEG6$*<5h z&j^h3u@D%~6&S~?PciTajORMekwS2mn9@$A8gaUY)YvV2lXhGPjOR~-p=72hc|E5XS?^J;lWF=9S5yHi%DTdZx9&w_BGP0u=< zI39n6)2TGMkkC|1M-sv!C;6(zzt)lJLa!(>qY({hOY@Ldc2O{G~n&gOFAh+u%2nF!aQX6I#kFgM9ps7tk=U24B^6R#lrLR~Sarhzgo5 zCv0dsaMm{SLX)GknQ2kUqzy_@Pk9~9);DhGadoeUAnSGPym7$&OpJmobAX#?h9tIx@+=)PokoQgBnBJD%knTX2kJL@ zX7@z&BiAixMnXgGn@H&rC?`=3E6uG#W);xUf*%9wjz7|$iw@4*`$pnE8b51qzd22_ z9Z)f4<9`SZE!pp|aQ+}r?GlxIqL9>ffUXwPZ@U@N_GK*pHUG^B$rxJ)bx;qt8 zCXY@Wz-?lWK8GR5Xl17uW-C%@fQuySRilXudf9`?NRv+$CeI$BG)+{joGuHciMNs2 zu`cT1{*tsv5IoU|6m`T9%MLSHn63?31XU=6a6OV4D?yhGoU4@f5Wgc*=*jye=Vk$D z;93H-O6P$F#s#oDfl_I*Il#l*4$EvW`}+(87@2NXFiFcaGvevIM7MSpiz@T+YxigS zTd~sKH7QmaqzJM&^H0B3kQpZFs6@b)A5|r_BqL=#P~Rm z%0Dxv)_d~V7NuR@w|i|#r)}soOZ{};8|4tvp($m6XsAx&*d%k?_g1hMl~+#CZdS2j zi$mkt)qi3-iuRzpx*aY~i+1*@owW0$ySY#8po!TI&guNoj(!fg*^G7q5Yg|fosDQ` zf2*D0=6%4i-e&)%OQLOjU>mx4yx`M;~4sZe)$DwY;pQNrJ>1Z%(HCTELUq-X; z)@Bp5oLZex$KF=1_wuQ|;(;?;ST>|{bMPTXG8xxWbSSK5Khqj%CL;Mu^I|8v($kc( zB6^>-c_`9d<_z53waPt`V$3^p$&h}RnE*=gKVe9l`(UzkuM^{)_*T~(x`k^c-psAc z7wl;SgL8;NKi7nIG%b>XfzQwyS2Q3Q5(A zHPp2bw0^=(1K&{I+)ymuDjY;1{Mss1_==k1y;iBh9h8cXTBY*-C>2k%O6@svD77}s zwu+TxMP*ZJUCa!bG9>&~r(R1~$7f{v;W>>-2Ms|#IYzH=4eF42Tq0vh17g5m5N!yC zkYcHIQSxkT`qdhgezr!VpRHl&XB&S17K=4ar#$&)B9g+xNw)D*oj5rik+UKh&%i3p zzmnu6XBpLo{X(eT#{8}%Ih~#(uT)y3PLktn5exw=kNk+Hf2leg37=e5vTJ2^P90{U z4yENUo+byt{FOR{<(WY`u7ID?kuh6ZVR`RtqDsk!7DIu~7z-cq9&c3#VKx z z>|^Rq`-b)&_^z>v%;H^wg(kV~Ov0!Ws{UAUwY;y&@yYi9&hvO5S=UOzMli!exd{FUjt{=sTQX-Mo{G z^NYUIYrT^!_=~>NZ@p8S`4@d@(0(cT0+0L>R_@F@W1DX6pV=4Y7}mz;*}z87*ySj* zWVPt;C?rEPLvCN=@1%&2qc)9G@@e#U$W(86^U3?c=~7}?3?zJdv`l11$?R>spvN2_ zEI|y!OL2hSKAD}?{%)mPg$Gpjk1ArkL%P`)@SS}$Ps0Bth45MhEb;k>wmgL|Ow%pJaOe+fJr08oOKqm2c{Bc4~9U9nLM@J`MrpwL^fBJV;HdQTpeAWj?&MM-{tMrTOq> z$RddsSBVzgxdl9{w;lHWpH;bkBQ6D1TqKRM(9Jcg_$quTdQSEoBW247ch#!-y}<2m zT6i03-P;7P)UJWWi8Ly8WCDvE{pwnuaLNa$Ab7RSJ-ptcF-u9g>`8nBdAB}V?Ol0k zui_vPJMF)IS3b9|rhi@MU0N!H{ZceLGWo!UfHzmJa?pbhW$GYq$iYj&__Vn%jzzL- z?wJu66SmfsS*u3kt=xa8DV9fQpO!W6rBR4DyY6C92n)wc2+F%PV&y_PJ}uY8*{2l{ zn(Bx{P#QsbLJ27-uiIZinF{RC>Y`pQ>Z5R29n_rPiV;M>cst&Y!OabfNjdXdCQ-wt>7) zt_WUIWf$B9vn^f^cls^K6^|FQ>eMTLCj-fMbHe-G+PgbsH;jSzE4Sy;d|V zA4eWZ7!Iv~9(0PrJXxTWv^q$bp_{oqB79G!*+TkSLV{b~??2PSgSb zrpvp1Gqm0YR7Vrgf*1+Q8AVxK)Lk;;?WuZlN7{R3f=%X>5qR-??qa0o|TA&X7M~)<&CE~H zw6>23a|Q52Rl-S{7U`qKTGCSQ&bPu~82?-`w}qouM^b4+Ld@&U#46Rh4`?8{)H;`+ z!;@9-F7H`m{t44M1y?EF=JMvSD+&}pEsOSv4r00XTav?0Z|ZB_0N=+Op{d2;s;D@y z6{$L?EhJ7HV2s<5{n2Vp$!|@qvO^_OyW$i8I?9iPYI@YUp&l5^S2U2JQ=~~@x`w%? zg%t7pk~rJ2DVg7TbeWdK=K|s%Iq+D_u(nkmiy2F0g| z7&H;x)4JB7X#%@}i~pp@>L!MpWN_w@DR7|NSA*e9fqmtEHL`yJjtPzIpKD|_(jS}@ zu)jP|H{##X=#*1BwG`e!vX4eEy<3n>BKN`h00-T(7x(=83O71Eu+YeU8aXi6$iYZ_ zh)jUTaTRDnYZP!Ps0MV!LNfp1{rlK0F;46)+7YbmYh8@rCM#3R_3x5$pzru&rmMF{-t3+ zg_&n2!(O=n^R>!7aseE&-jxfW{1q93Ih24|ctH3KnkA&O{}IA~Ciz(|fSHJ#-^~n+ zPRe~?^b#Y02lk2p?w74DEuE!s9t_e7WnEGPkTzPcTv7yZKX+ryqIs7V0o?B>SRI?Z zvd{94_g9}p1aN=#SOgF%O35gXMF3Oi0TFt+Qy^ZDc`r;@>D>M1156@Sfh2`_ z+dOTLTWXt2A|dbRPFq8)Xg`b6uaO8y=3??!a(%TiBK`hr{T_e((+Y&aNUQ9`A(tNF zqjG1f%LnI29hd`jMuy)g;V7UJ4#G9Chmzb&D)P|RX{GTsRrRK;W_QZpXgW`c0i=0`_$!MKbr{|93>lj0@t+rdti!Th#zV8~C{uKB?$vei1j z^dZiZ?eAJ%{W0=xwNUnYKDEI5Alg!hX9-0gv10m=w%q?DelL_x^#9a!rBRP7E%i7| zIWPdI)U4Da(NqnfSI1zp2*}T3u*7F&=XKk5o?@`##JIk< z+1<g(Qf0ZgQduG= z@NwGg-T4jB@2~-!_b9~JQ6|{!#iyv6iZfF>Hz^}G9-!!{mzd|Yix{y z-B8WAn3+#;T+DA`K8TuNlX)2ECW>!dVROAPsE&->m#WG5mN22!qQx;?U&AI-8gp+ z2^^?*igLAnhh<-vX$L)GTDO(VwE?8gW)S^PH?|dSW(79^gbtPcLYkFtnCGXKB`4(u z%T!(#lBn6(LumRKh>ayOMq$TV(6LQNl4ZJYaV1x!+CEe-+T*K|5vKS%OZFQM0vcx$ zk}2vysg~?BD%DH&8=8qIh=d0@BGI^b}W!gnWuzqKV zws2i&z_n>3y|OX+Ayo#SsFYpvmNJl(xTI_(6st0keD;}Ak4mK;0XL4M@_e}PcJjAs zZ8qZA342|KBP6Y3HB7dRcjJ6j$b!XFJ*yW_No3l+csfLLA#+(el~yO=XrWH9I!mt{ z&AeOGsRyV^kEM`vNl1QNJ*6R^^_P^)!$@a?;S`L;1zf=ZxQ0S1Wj7RRnv-(48%eU2 zp4Qj^RjzM~iLhn(pR5FB3$`L6=}Z91w4j=@5}2G0EKvCPiL#ZIU`qYuB-$J;!ylq$ zDgC*u1k*BusA<_F-e6h_I7wGb)voxAmMK#N4P{@`vNsXM?>Z|1!nR=$){im}B#(S& zM;7~wOfZvV;bLqKP2wA6oX;P*asTKfVhX&KdZbCH z3sC*>w?x7^w8qi}uB^o2G5d;A{NAl$Jf!mydJ^cP z8}k2{eoUsMCcPv5fT3^5e<}SqbVL5`^cCj4A^$ag9If*ze~fvs;m@ZHe?Ixfgkb9H zSzYly>~U56Rz6knUOxGDPCmgaRYs3zRVQU({sku__+`>C|Ag<(!~8HptQ>D6R`l*! zz5BOWPS}iJ!>wCD+DHHW*(OjzC3a(JfQ&naEh=~hztgtatEZ6WvqCmBVyV#NaOwHV z3AUN8lxv6e!QHOVw=%=y-L2{?wLo+SC$7L179@~>WDM>2%*d$cS6nlGt3jU=$;)MN zoYP5Q3>03ok9X+-HD{{Rbqx#&Qs_xG8du*`J+CfB2f1{9&CUnZJh%GBy4JxZwaU`9 zo>M&=qo8i2`iAQ6`lV~4G$eQwZjt;dVYw>QOUEzC)*8SkNMmYTG+PTBU2<`4Q7)=f zd;M#<$VzyxdPheYWPo~HLL8D8*(*Z@tEmS@mNn(8csE+-uAjHZ!CT@ zQkW1oPexXqmj8V2H4Tr7dAJKkx};#e6pNRJnc)KIH!O}&NRGewjziD#uNAjqNy!9d zW~mlNd`A@?;vziJFMz$H zKP7inE^Dv47xUN;ONzE>;!h=XucC~n*YYX{%3Fj{QOYgeK%=;CV~>~qPA{crb(xh_ zswV(G%#G6va7sEo81QQ(H8dIv?K3YQk9Gygt@`J|!E_C$>sH3GR{gZ|diiRWWNXc0 ztimk0CYD@VVs@LP*_-3X_PWG>4fjn~13?JVHwZ*?&8lPGZve9T%0u2p+q?n&1?$wD z>%8jJyNb<15w6hY8rZ0vSAA<7a&oIE!$*4!mEIcemouBy60~F+mjitDK5kRGjm(gB zu}elHbxAsy)s{{6;&2mgI*Tv ziTkA3A($Wzu~jJwPNgOzgc3G?U34}m`smN1-GAk;&2dkWGQ2jwmgE5p0l$q~(7aL8 zB3OS2c{bOD74qH)(@h;k8A#rbK4bblObXL<0@_amz$b$Zo{z7TrS9G<=Z z#XFtqd*8qljL`40fT(`&9)4C&;r^Mr#^j~(WHc+eFG_ZD6nH59Fa)c6KcvP6aROtQ z*fv!?W12H&)IF!vr}@f1-Sg$>3&4G0XEt97?%9_6|Ji#RV9TznzVn`Q z?!E85d*6F~`$Kk1-QaVtHC3&YR46qnTjOzsKAj+2Ho;D&m?BlA$~Ez1@N31WC81&p z(cRb`qnQkeAhO4R4F;REY-nQ)nQ{mp!a!nz$1&I@9)=HP0>%bs@C1`U0Be5#|JwVU zbML#ayJguJh}_n@_nxyq)?RDvwbov1?Y%$e^kPsFvcbGs`@Y4Px;id4gL}~wmWjA4 z>hS?)irC;_GsRqzUz>MX!S-b05-xyFmc&(YflsvW!cDz1Ec32c`nX9UVoxR_>m?Cm zArQQf_~gP}tjsfVB$QYAWL01M7sMx@`VCJ{iPr}Te{Fjb5Y5*w9Dy-_o%utOe}^vbn}k$+Bx zs!ct}YL8aK?&0pU8y<})F3b*|eI}cumj~{0iVXbp+Tsb+_QX>c{Y8&Jc8Mp&mtsOJ zn{1B{#Td)tDfLd(jh;ZFkPNti4LD%8XKW4Za=nrrONn(mv?7w!hh1*4wtsucB->sv z0QMq0u2+TKinqIDJM}KNr=4J7Zo-`1NTgWawXt`zToi-pw;V0Da**>~Yxj_@x?;U= zk~k1Hs2GuNu}^fdWxxGm@{&1hDX*k9(uq>|+6h;&pqa40YI-7XEp?UE zXc0I;_UB0qg6b`;R|nYB5GchN>`9cn$nk^)eo-xASt`qSw0D*>>%?*PcWi4zn`W;T z*90X}Oo6+vQqx+hE%nJ%TkQ=ml;}P2bnUhLB3drJ0aat6dCYkqN+gG_up|3Mn+DrR zhR&1pQ&xgKaKNyu;FF{pK3afaTN}GoH6$EbXBXNjF4wYq;?fOzuDP z=N;>d)uoa-KgH=D-@m)CJl!#>?{&D2O__KvAOrvH+F7w~OC}lOv8hmqh$A;Qg?2E& z1N$qq9Uz9S@y_jl1fz~hY!Z=V*Dufe@s~9RUwLj^-@#X0Yr?c#?5_G|^mM-4Vu54v zz&G&oZxf$2Cz0Ki7)za{2neY)~8SOxM4D9UUEn+dK*kmfMM^QEfZQyd1#a}2lx4Vq+ zjfG~k;{t9G(CSG-X|Cpk?-$F8SEv z>9UW_b(c;&++V7$U24h4mP;-B*!)y66I75y^Q7IvsAVXEBd{g^#g4-=Z0lo02qRVz zh5jRSrKA={EEY6ZAlfRXTI0}_TqB`%26F_RxB>#2E9@q-0Sp;bhus7ZB?Lx#Q+t?M~dwH&h$kMAqJw_XsB0C`>i01aTC{4a)}mvzTcu zoa7ubzQ@_re37BYF5-Qn&IX{)K-OI6M@}SCg1%GeSZ^@YxzAfRtFK8$vzV9RBr6iX zJDj?IUI)k9yuxCQQft-@J|xoiH@7bb-|K=nes=QN{h1I=|0AjOXHd492d#?@4sR@FzuWSCP+M zD2Z!pM-o6sJ1aino^n5JEe?EEn3O;ndozF=jK)78xER-gkp1-&B9ji-!HA3_$k>M< zZed*{Qu1eeh2Eeb*fL6Rls{#H-ievWbA$KdwYdVE10M(FUR1D(g%LADFL0Eom%y-Y z8N6Nf6a{N*d{b9r>G^BoIk4z7q5ZTWj$wgC{fD(K{{$L* z+FQW!Kv`zu9kPR1b`Iv4x3PfJWzXU-A zH5=>Jo=(m8njP5oE#Y_;OE@`%5cc z3z#IK>Auw^Nyu;bNNcFRtt^Z4r7q+tCAkZW?O`jnN5MyA0mP+6LyPTR>EZds@G_R{ zXipa*-hJgv{knSi&>1-;?|p4Ox^k&)DlQQPmeiDL&^r85HFmwj)dB*4X#jCqxf7$+ zCaYeP)ZYe{7CMEH#rC-s0mpT;H4wtVI_Rso`^rNoGx{#-lJ5ey=(WhgRxOqpNM#Og zPNyFjRwd?qkv*)p_pEseSGwrxLvOv*=58@4mXMs!*}UUh`cK75443on%Sg#8(WMQ6 zFrZ-;HN}Jcy>g~Hzu53Kx~!<{jmrpT%`vddmzQ!)38L98v7IsrMcm&=@?pp{Rw=Vb zBytvL0$PBq4P!#leXE{k!I$g0(5lEutwn9-OIsUI;QqdJ5<}g-7mJ&ENDeZ&ZLeq4 zQXr$|LqlkyDiL>n_y>8z(mBPQKlx4kY;L)pxby7w5}3%Fws~?L-Mr=HQu8wNKNmgJ z^zcaraPz(a;116{y!WY(;oTcup69H`ODp_AepM|FF}*>&`RrGFe8#gUpR8Wm3?Ja2 zH}vv-P-@)F9^}u(i`+~{Q08KJw{O=W$7FaP%8F7C>Ta5iX4bwa5|9q z-oYwFosuv*um;kPm+lPZL}$awgG!!(^ErzFRdqLj0f64vbx94oQ$DXO$E3vwN>gp( zsV2|1z~w)(iM0@2Duj_TSbCsIHjSdx#E&Z>!&N9{+%Y}h#**Ms`n3feyI+_)@sDS= zcSzwQNb+vY2SqqIajz-5pW2;T#!p91d$7xPs@0WLIK2;|KTLNu_(RcP;rIE~PEA4^ zdpKn0kgqG-8^^h;XTuu+I8_l4+6K7Qq)rXY2h?3(O)e>c>T0k9B5YZ*C{~FDhf#<8 zk#BMK!veA1ns1Rgw&@vWK8HGB zWuC_@|19+YfM?bWEuaf~87w-C$<-+Waj9FvJGeSHouURBO zWkm#zty`0J{5I)U_-G$C!Ex*#IDpHB3j6m|XgU4jLCs8;&c#lR3^-;S-+xW)KYhF` z3SDt)@8Kv?2>8jOrFZ%QKf7~7)K+_@U*VNN)89`jvz{rvbZ&?N$d$Z_1kV72i=7i) zd->J&0)Lw#k^Ngk>=-mV`a;N_YYHCFB$e{_y^To|8Pubg4mvIaecPi-_P9w&G85OlA~dYS z^^JX$&S!mGhP{*W=`nWtvX5$wkN)bhwQI!}IyY+H|)2qf8`bilz1Ut}G z>0z-3Gfu?;nM1JxdJ|K8hl#VEAJaN?*;Q>U*|)i-nthuX9m1lb^HN(~Pz&~R?LW`x zjoq)t?w{vj#I@jI5n{0>5do9@iG%}$gAv zv-`-e*F+n-S$qf((RLi{=U~ZoA9cDWSkm9ezZ7kduFGcC^oW)Z2`#fr(x7_Os=w}$ zY(*n4J^HqyJL;Cz_%JU&MXuxWMU;HpQVyP9Nh9-+%=D5!U{@N!k8l!0<)uP@9k=MS{KP8y(QCvSv|F9~D7mA0hrJUTLRPpk(sC?BM z?6o*CE8y-vN-W|DkId06?qUq*7J>(@8rJVov z^A4LWy++L7k&%bk29i31Pgq~Fqj=i<-rFzry(vDBa?t$!w=F&0Hz4fURSQG|#o12h zX+UW>TDaQUaKHT7#NEWhf7R=vuHI7@@0tBCk{xedjQ`qT@c5HYKKZ-NE0x^B*!4vJ z`&9FH+V6wG-?eTUf4|)^9vc^43nz7TVtRR_*vtar`Yey0tpU!yiDf-d{^&a{_0VHr zD%nzrx)5b;j84zn7#%;3EjCuDD?c#p543-Y5h@U{NXw=~U1w%&vW$Yd{AsDHFaf3! z8?uzsA%C&lPo^%;VLBUXbD^o5#9T6lnhW!-2w9`pjSzbi;0__AM3#lj zUR!l2bWa3Ys|^TIo@ytG5`(SfYTQ{~CTLGqwuOu)79{72r_Cu&FRH}r@N zACo0x*8?-nv=ExeZ;(UL5$|k@X{oCO`<^MLr8B!xZH%Gg6nx^0o0LR`%Yh-QO-}kB zGjIS?#%Y|3FBbtj1Lx%I1!u&?#y*QZiwFV)T;VJn;GHou&>#;t1v5OsV7c{(2Q+FY z&X%6?5J({|^&osg{E8<4Qb60B9^J&<72*+5+Dd(+-63^=Qa71>lDg=&U#a)>c)d0K zp2Jmv;A|%=+I%CF3>_l2n%`Td)-X@a_xGrEP*JU3R)iTvhe^~Vsj@iz*1CMp_bxWS zJw14$`8#5l&3AadXnB9}>x}?2=B7$L@aju%X#OsRbRGu@vWTlEx!A??!N$W=zQQAn z2k%^)u1i($#yU9o1%vi1Iqj<>Wn?MSV&mYm1^(ai9mn5B~UbO78Sw($X>!^Tedl2_z`q=lOhQt7f#xn;iq@>kc#|`b z7I?y!af(4JS@g4p@%kC6wMK)XmaS13N@tzGIcXeS!rFB*RGepE4Wgk%0Gl zDs;y?t<;>MB+bELIdKgVLKKDpKAFT>;^-Kca z&8n|mRLSgZMhMau5FD7_7AC}tWHf)yFhNlLx-c<=SHQ#!GR$;c)nc|56Hz8%f{UvX zCUnI(n9zg3L~gFd1cE_$kiD-V9+F7el`7punxJ2qA7TsLj3r07!iZ{-6ahr3^rhUB zCW&%%^ZhN@4r^`XNV@0(-6DcJNG8!@dnHp@zn3$Wx5-rAMm989`ne%^*-xc{^0VW$ z>n>#~?=MwHDI93I-pRJDemp{#y>j!hJL3ys-AY9_z(Yc zuOWA!`yhDp_#d6=y>S=5|Cq~86+V=wRC$y{Ng4`Jo8k>I2* zXq=F`ARQXK8|{1*+NptRQNV#p99h8{I~GnG;a4X6fbAA_K{*Hm=~8QOf(S4VjL(U= zG2r{t1ha%5d0@Q*POHx@%2??z>+I+NevDbfJ!&OqsVw-R1f9X%bRsS)Qm7Gb zyk9q|AYcGa{8;ZL?_6c}Y?DJu=T zs(S{rlES-LkH{qw)u46t0Erxn%?Q4;SyQRGUYZjnXC*VjZp>*ut})zKBbR5!tA?O( z4KZ4@gm)W7Fng=aEjOa9w|=)`L%~mVfQ{>0np zhwZb5y0^`=;5;{DL zt%Qqd!XC+z=xY*__drEAjB#_Ghs#PlBhBNEl*;19@dGlP>&#kX zL)jAvScXBqtFu=ZY|kv9Bss8h#*$ra%H4zyXT=n88(s&Rad9}1sncg%_Q?k0N_j6a z?<8SMR79K^(JWDgb26;} zeQrhK3+`?wK^kndl3-p;TCSe-xOyVCr0&_pPRrHu)3sckswJ*wLM(AL)px-?)ep5? z&9liqu1@16wJ%|JCo%UhVRr;;t0|lqxF7P!MBFX$cTBR8-XX_RU+iQhvSI#qyEqgn z9T=SUjiEyDj-#n}nX>5-WY_O7( z{u28;?n72Ea#L!)pxt1Er`UAy()CSS^%m3D56kg|>O2D& zyvmuc&4({v>S+&N5boI9t%^Z`N&Rha>%Rm}S^3S!J1ZH5)38d>{A1lUd)$&11Ha*l zmp*Ps#;qOmT0vWJjy7#DX~hi@;WAj)c5Ly7-ov)Ss)W+nPS&$8j7uL+7tulDWyk96$O;bs%iOG zNWK2VXt?E#l%6ofnhDeVyrx1c8)ky-_0o!1#syCKi#%b*SF<;0Vn_}RP40rXIeC^q zh>~aZh~!!FhMYWG;dH)xF(Tm_xkd7_=>ppz&mrPr-rQIbahDFWRQsWS1!P?*7&=j&~R0&6c`zmx^hte(XqG>C)( z`=2y92J9Nzc)%lN6sqcC^Vyj~CuPq^9_ryUif9~q?c#7dZUsm=|{Mm^VL9%P|5^HinF)K9v37g{sx$K?fC{Kz>)=-qula%@$D$XfmdJuK_ zhMmdv_2>5TBQ?jqSv7~1QI=A}%N_PBB^Y=FSR~E6iyan|^AU;LIfoIE*7`}R^?s5n zdoezX4JcjnGrnqz%RWB|Q@{0-Jgpw;)WglJUcN_F5i4bnYzvO^LIG;=sJ;Q2)XFFlvk45chE5@!>gF4bP9EG;iSHfE#^} zcT`o zKPU9(B!A{N$~KVx;{f12%BRx$aiag@WbX%^AD1#cF0sZr(m*3DMGDaT_;SCrM0xi6 zs*~edw5b}pjKP4IFOq1jzX&D5RQsZOXdf1cslQ0oc%V&*^W&Q1HH17H68MYeH_GoH zqpqI~LiZLo$_1@Krl-gK^aM|3GoGk#(gzM3jrlk>5Qusga22Y3ejb|y- zZyeC$=lt1>&d%liAbuDJ?v1&JvJm+XWiP|+|@O1m^#?Vf;cIK>%q0gDuF#-;U z+e4om)jT}Pe4{lcolQbkI(t}*J9>utiB*T>%;sJgSJU^6H!5#wF3#(uqG?u}k6gs5 zmeQ%2D@AK24futDBa87{mdq=aYD(=~Za1Q_qO&~{R%56(J7DDRtxn*SL+BwkHt_=S zjdg-;N=f|!r3yGLd8T*8Y?P$noC zD^MXfoaIM-Lvyhf>rU=o96zzRF?+_3qM43hJv?iEg4!n*&#G1brjUFnt)!z*BGxLj zb6Zbmoc9KSUw~nx;f~U9dEy;Rc$!v61}!|Yn`)9tp^VqAy{_w%U019wgf`*M%{`le z>ufgEwZ*nh!VH_%ey)mvqHhvvP6TRB)>Dh>ld22o!i-ch*m%~7#mO|v6F^YaM?H9e zIyw7pIVNt>s(%TB8-+{yxQTYKsB}dE4eoher8gDuovd#_(LT${Q`aKRxhp}7cFZhd zkxb!zTQY=M#%U>Q~WXpO_8T4w{E#5b&@0#X=bzzgXO} zVD-#WiEb|b*3;=Gs>9r0Hy;2gUeX8RkzB>HW6I{E=jTOvIt}BGKSpnMBNOmeh-qaOX-@;qE0_$6NtShj76z}W`tjEK< zy8`RGcu!YgeG5O@6<9xtpXds#@8Yku#9ZIQUG4kz9ejEFetrGFd5!n~efxerd|%bR zU*G>%x9`{2{~y}->*0H4`+j}&AJhfgWL@9)Xl8b-!CcUM&F~Ey z4R+MtBQ=lq=z@?Rq^0-bK|w^0(h01vBFsZvl4k+nZSq*(CcxrxlNc04g=lvc>WwXX(j;xtP1`W7d23Fsw&h;|K5cLNJ_;uPR1|?SgyeG z_^b%wQ+&?iCXdOn8O0O;8ut+{A3N<6307JwX>oz@u@O$-fB+p`Dx9BCKBfgbQFThz zFf%C+BYYW{*TB;`WZ;j3@Nn3%j7YrAs1yRTfnG$i%T0lUm{6VtHG2$o&K4~Z*fQ3T zN93l7W0&amv5d?XkIPsz1sz*67Ww&LtvDJ~WWBCD0dOSh4aEJt!OlhqLF1H?g@nLD zXNYOS2Yh5aIPrm3tnC03CbKYNuIL%`c(piQwZZVHqeUXwCiD_LMtypaEAarhA;wCI zmr^H!9wXC(^L9w>07R(1mFPh}NgVqFav)NG%ofPyF;K`v+gPFxe6AD=Moz3X1O#0J zyc`7k@O%tGp_w`R9xXLz-=_&M{hl%{%7pyEq(%<@TrA=nDE$;Y7c7>7n1 z(Th=Fi*i&CBsm+UY;?l{8v#OpF+kW(_<#-)n4&8Tqx0?{a@GqxB(Z6)`HXLN40s82 zE6^t1Sqy|cMpzf()$c5=MS2KG^ShNS=EIH!lK_qcmRgJ$i+i=M`{8J`T*r(O;~g9F zt2_-0N?P#;R)^ojEyIjH30!1~v8Sd<)!BzQK?TYa{tS#+%qqfyFVE@|9S%J;aJny{ zvID*^ba#WP%GqOPrxhNHhp$Tp1F$5?40Vu%!XJm_qM$U7i5Njju^>J1jJyv9&VTTO zPMROSbg@BUPu_O6k53>enWCfFo5j?Nfs%|bFq0^c##t6LZ{H!0l*M$RePmL=O7+&7 zJ@Y+X7xjRSS?DjuyW{bAKpcGxO9a(3QYaT2%zQVCioj3=G@SIh!N@*v!$)wTd z;@nK4a^t9_;~XWgRoSJOGH$zVG)-)?K-3eOolQlcKYAm*MeSHn!}e-`^s%qNLUc*> zkV<->Td+BC0a2T{4+=_*G0GzkIAA{yE})$%Eq)ZohbiA>JSPW7>0WHy(fqfB=YxGz zMADZEY~-p)qH!t^;H#kt4Nct;XhcKMtePN6nGaM}I~1}Xq83_ACAfJ_gPiCCT*@d+ zTFNNx(3swIXr!dZZ5;52q%t9enR&%#0J5Px4MYBswtNMKq)xz>QL+zFnF|tGW3*#$ zes9Tz$5w;f!1T6ykv(<#=x_(UczXfDgc85DRd}*AS=eAGA0wpT?pFEkq!iUp^XBIWKl(>nm|iKrYfB?QO=sN4 zPEGH=FgFJIdkAjCGM+oi!#7%-wTPDCMPimbf3O5H$7r}HaX}IoNpF}Kyiw(#i}B6T zfa#%Kjto^u@A}hQHu`ZFY=6cKAbhl1>L(DYO)evnTE#`t7xtvO9}8*h z@8n)9_GMxSYzt^Aaod$tbs|`v!^**l@y6&oh_zWtZ_C*`HAk)PCx){w@ zL+gZmmNsElFwI1J%`qJpN(Y=9DiUz8B_DA3R#Z!q33!W|vMN#pvFtx5ldL^hBr#Ji zMlNG&DOvJeT{rLDTId;H7MFgK`M%&?xNO38kHO0YAT)!{}L-;EdUTBM88O%)TZBHEf~VwJ!V*-km9La zAIvJsm3vxsdmI$k3Ms?rW=8Ww8>(#t6X#*&WCB zz-)jR2y}>m9fhLDL=(L=gPh&p8Y;2gkqMG27-O(?)U$<$t*z>`$HlVIY3^FwvnK5DCg8mTh@jMOf1Dp6-`9ctPBxWXYntfDFPbb7>;*G z`jx4zIV+7g#C#bs)6*NvbwPl$AXIBtfEJ=Evx7yo3&arAkgihGF)JX@Dz+6x8YJRQ zi1uI@ebh=;oB0Y(R{>Bat;6sci%L$RlzZ`OjHUeBaQOEQrDV62m_{!8ihd) zUy}P%PcCR^(~D(t97bX8wCB|w25qQW?N!&P`K5P!h9(rWM| zklo2wxbO`4QbY88mKS#QYCGnh(R!+UF=OB`7tVg6myPC2&s}*PHYpYAS~rCLJ#9B! zY}`_DUe;^R4E`ptqCo1cOFDth#F)LSLy-@J1=~UfLdOdi4Bxr7CsgJtB^oYZ$TYa2 zF@>bu6s#cBLX%W|9c@7KXR}}gpqFc}6X2W$ug%7}MhS4uKY074*3&JhPiaC3W=1UQ z*32JvCCBQ-dT&t%gerLfL~9^ay)$8*`G~xOx&X{1%M-c#OqhE)O0s~WMi4MDU4V)` z{2N+^ZVab!Pa=bI?&X8{(GAyNv8{sV_jX4`Su(k_=)#bCu=xXdX3vRlRRA6_j}4o# z96QbX9_DwCopMDfyrlV*J#_@Q`NM2M z4*~vFsxsZhPqJgoNfLyXxWHU9?AgCT3~5{VITrC&KAfPaTP5kLH+tDdeo?Hv2#poA z89wNd;|~&&#VDD{bczl{FkYMgc%@^3c49NFnLh{2=;6|g7SUry54#yHw`!izCH!Pr z4L3bY<~m%I&)GUvo`efAezrM_z^IY-jA^0Y$1U*|5mkX1<>*xwhy>r>Ul(3zp@E&W z=`McAREV|~pc%UxP;56^(#v~ToV6BIO|+)EFPd$g(HIbD?F4;TSqcE9g-JP~L(&3S zle@xIF{T+wSz9vjuP(|EI5g=^k(H;9GU6HVT{|q}mA$us@rh)YLU2S7jvrmVgBw%FjrWii@iZR)ViN}*eyOh`XiJB&@b z!g7FDg{}!2_!aX-cWIz1RYus`Q#o2kjhy5I4SbxB=?ezR+=649!Hc-2@dkd#Dc~Q+ zN@?W~*E$z^z;9E@xZa*4D0O0DAQxZgLwhcT+ys%@IRzrEECP`#hiQSG4LH1$#dY&N z&6na=>PHysY6 zxMs@ng;B%>W@pnK4eV=?Nx^)Bt8uWuN&Z!&vKMALr|LH)tTaTLstGs86x;_8*A@tx z#RLR0H2#Rj)QQxzNoR6=r0mYt_#j@`jhjVY%mY}YdNI=c*j_NpMsor$`tt_7YEJ1H zmcs_y8}he1+i#C^K96yyHcGC8sQHEAtCBV75qolO5-mYisg2tE~T^Fjy!6%bO?=By?0 z8L^I2%>^y?D9W4W&aihm=S*@=Q7ar<^#~b7?kVx9dMHuFyLJ|8Ur3rBs8IEIs=ULO z>mUYYt28tPi>X*#Mps*`+##-#BTI$YWAAVYJ~+((11i9OtsxzQ32R!0phOnS@fX%* z$6IeI;4Q}VzK>t>g*yS;&7oV0*+(ux&NSups`+SS?@9GAW=$%lXp#CX=dg>VG~;_Y zKcN2e%9m5oswx!&3`U>d-m(Af?EerG-Bb^z|GppyM=KxiK^MpM8$eH38sk82G`T|V zYWf(P2r(8v;sb&(OvP>BkO%jiZEnAZ-?ygohG*Zcj#THkPKQ6JJSKa=2xEhZ zo^wBe&kMi-p|&Uw5lR_$#hya{%{v^fk^7M@GZ|cyM?O=Hj=b7p@T~IW9L5rxeKZeu z1}w9Gs$U7^-!aTA8ZHja1bb#iW=+jqRG(+E_*3+Nt>C<5{%**)1~SN)U{ZFZqIqM9U=Wew_X zDTm_@~i$R91{>cKKwm??rK z8dzoae`-o++B<4!%z{1WGKhB^FTd zL@WRZkzW%26!jiZdSz-`)C_y&%r_xNXXi}<#-Gq2UM7^?d)pf*Gt~|^>duLKq9~ zzK>A_9Mez4NeO`j@aDPD;gPlKfp|>PdiIOLBG90;ALzrtla`5F!jKMhV1r)OCetFq z&JAM>Dv1v41Q&m3QYYH^DMObe5pSj`T4gX6o0m1miBH;?O|~#jD79lE7xm_u`NNl; z;k0>1T59&%ms1C+q=~N~m?_imMWJe1w+fp^Wsy+{3eq&$a6{bLx0$xsUxH%O|E};& zGoMT3PXE?wo7coIMvV4n3}zolWD{aDzAQsM3U-B;_UphI9mLd=m!;33vtMvKL+jQw z-R$X}e&s4`IH?Y#eZn?yeK5g=jkN-oGmKszZa1^Ck6v7vd3_%+`6Vq|=GpkhVd#CW5){%Em@HXPNGr_?3ksekK~by|6q4v@SVs`d zWlA`Ju{Jn9BMXW`fw zkg#XBG+I&fq-aBbo%@Wv$Fi7;j2@HV<*0tV24PC)-w&<)ww;|{awL|2E01%|ik%s! zr8HU@YrsgZSWL}}Ui+!MTsGiaSw?{sHT2aoA^3bHA0Hp=$$TG7e;Z$yjzx01g@=|k z#JCyY06&9Pmn;CLJFts1<7pX4^8D!I88`b_GFyeiMVAt;E-$e28G(30=wPLmnL*OR z8{&DASYf=56(~2HOYBHz)TF2=kikMi?0{k(XLf+^9wdXsTaCsyyyHg3-pVvqND`O#z80_S*PzIzzg+raPm2xl3uq!I z3`WE=0er*D&Feq(FMn-x55n^}2gBT-em0MN`;UM68}qMPmy8}z@_uUW^lI|+kbb7g zR=rA|Zmc;A?+bfeX>T!)g9qttzOAc!b!7vP4M1ra#G>86kNnL@CAmi{Uj!D0aCJHm zt>HMmK{j-SIwz0KymL#1s{b21a@EtYsZ&k!`P0TiR0Ahe6d-)u8>tgCV%!f>Be6n` zMzjfd;JH72$3|70XNmTWs5x*sZZX8wEDzRY#@-4p$9-Hbm%JbM1-NW$@)KN^_lE`r zQZb~?t73#6{^qL|Tj7n}m``|*LtZ1{<6aKMwpJuqo>0o{)e)#TZwpNaVTDH()ht2D zq`-~NFT!hTgpYd_3+9Ly&>>Y0M#s4Q<(}}>gqlX`3xb?m^t+M0C@O*Owjc3%uOFJ7 zo>R+`qRTB8)8=?AMp-ryZI%?bj472|7~dQq2I<^cV?-2xNe98#n?L-h2CZC#A71SI zZAP)Fg$UlM2l!LnjrnJ2<@35PrMs~4?)BfY*#4`7!Nr&T?OLCM!7C&c>g~7t3~q*t zv#&IQg$^Jmvu(>Kn;Bg?u2;V9tvD-eYxbQocf)Bn#0h2yIYdOWfRE(%;w0Jt9e97Z z)&LF$L$PweP-vVUryAF*Vpai0Q$y7Yd;inf2TWb^H#7k<6hT2HaR7AmE*UnN{i=YZ z!NgQX%wnQ(aBRvs4~3VISCbC*Qlp8nGTamub1f#~r45Q9tXQAg=OrVJbDNN6n+~j) z-*j_{9OT(1401%ToSbqlq%h;V0-Y0p}gNm_5d zV1U$irRnQ7D^4|Kwmw(#anRIDr?5yy+!2D8b`k6OY`Wii<0Tfmaj+JW2ZxgcSvmVz zACJ{oz@_VFfsL;_rARyYyj>xuS6nY zPFlWl=J4z4$(gO!U8<*N3jS{W*F|}zkgka{E(Y>n*%NnsZ}wiH2%Z?5A6L+W*Eyqy zA(Qt(_8E@x_=hAG$`(89gcWBUf*_6Kq`O3<=DUn^3D$UOkZEj33-z{+5+SvsbBlWs z#+vV}wvakRyg1IqK3oxw#%8_7)}qJ-=EQ1s6y;9CZtb=&2h*S*XKxK349gyzYnp74 zumrW#N1w`GL`nB=CEcGqC3sAn$7Q>NH7dG4n;y{!0upp_ngsM$Uf?)LJR3XY z%H|FopIlG#LmF{nBW&P(G#~Y8jy$dKcV${LJjV1wgi-Cr-|_t|L-hc~K$H}3#k%#p zD&hJQSNiuIW{O3bpy9ZupAkRFCn;9S=u?Wn%cGoSm4#)b8El_UIzXpApLsNWPnq^W ziM|2>e2ZVS_Vz3&PfKZGp6E&1EVJLE1eh|8&89kD5<7YJF9e_`vXg8WhwRj4A3*Cp zf(usGND^KGiKLA*XzRPrT)fkizu;IsKmg{BF68QhRQwwCVCwum49NtzPrD|@pa`}_2% zWA*sS1(^OxDAqyZy5QfT`*@qEiCR!JS=p~oT)xy=8KxZppX6Vr5G!J4hg+dxCdHz> za^}h6wJ2i!Wb<15L?b`Hnnisp;t61I#F-Uw3Yh3lkI+4$r>2_z+VA>PLfBkVO|=NW z@_A|x3(i|pE`9j1Y9_qYPH54+&zJ zO&^<@-jxk$y4~VTOpU4d0ql%RHRocc0VNpHKsdh}(=G<0g_$h!hIul3x&yFIp*3Qz zU2Ac2CW9zF2%uFMColCy!%Q9&x=;r%2my;?Pb2J63RjGt97HVf?$ah#G}2rju@u<{ zR!)7(v@!H0hJ-<9IPuNrb@_5082?I-M2w}S(GvV|09B7h_TF?@U1hTChC67q(wb<* zYO~}cC1E&VigjIDh`(}ih>>!rE4Aavtq~(<_`qbt!Y`R-TshR*@TH8@&_9r0h8Y^+ zmDF&EqAq3j(FEQt=fP-x14lQ%^PKGo3wnrXMb`^JhtKP9Yt$TaXUH8UXr89o_XWqb zK1LmxnHPnv%V~BQm=Di&LM?+md|Ahl)U$cpIzyb98%DZR#e*&QOMBIK(m(jMO_jOM;k1OZfTD8a@+qLUP-zZ;-!43#0T&& zkR^yXDmzH#MVHJ33>TxWC7J2=;uHQ|wT{caTVEz7IzKgyb5uU04G~Y90!+Md5KlXi zG)65Klh^(RT^+jkz$w__UrQdi^yznkN;*~w(ZDit&_YrRpfMxO#J`I?FqMbb` zi!^)mQiSV~t?ZGlH+ypEsO-o#o&KX;yp)^>Oe`JE*Ka1dbQPah`v!`Z1O1LX8U5~{ zd2iADdSPYuflE?P$n0*eV3X11GP;LCcSSmv(OpGW7EzNH-7K(;ZmraZy}6Rsld=Om zSJH1*jJ>#?=SmyYt5KI3-Bl0W3~L11E*8yC0(&bX?W}d)$1))p#5zKh2NS}sh=JM1 z3zj`kN}L!W#>PEDBIc8G*Dv+;j`Na4G5LTe&8r?{38E|%_CY0!>Z!}&>*vSm z^x1=D?@L`c&F{eHCs5ZqB13aV-UvBS4&b` zTI1rl6$5s+EZB*l_Igukf0BS>p`GcY4DZg+LA3RBZ6CVEwxfKk6prnM14%ES* zbH8b|3wFu&ejy*ENZjcKudKGEL8s&kGB}=Bo;s7l-tq&2NI>OQBpax=wh04wb^^T#%J7G zTMch1zZ|y_EB+Ugmf%Zv@Wm<^s$mZKP$q~Cr(iqUcp_)BPJeo41Fy>AmnS$SVk5XP zP?A`8$w%o~uh-Z2&nOoE-Y_UOx2oZ2ym4%2GTl9N(G8`-sh@Wk=O~|Bs&^ImV0go8ZLwO& z`i?l&K>36f*lIqg*7(G6vXPM-kGsYwcT4rbg893BRG^J|*@#3~bisRSnUML5caHFc z-BqM_ixHLXtGsGKp(M#Z0Jrm|_M5V619moab%0^$Zr1=gzdpD?`G3IORp|g^wzzA} zfdv8^fL9^j2k&8?kYJMF-NaBNCb%V?k2MVRx+YTJaaRJELa!xrZK25LgmMu<;}I-a zPqTjL%d50Pz%vxV0#B;20tl0Z>FY+fWS!DxsF=xRXFaD#t}4w5rP?|rJjFK&&RB5~ z@(C%CmH_GZf-@I@e8HvZIUF%?%$ zx+VPNvk&;q4ox#xzih{bHyic!nJz2P0RSaoaxK0aPG1-_zw_9oo`V&^uDUi3>up7% zI38Go)r|GVlI>l|cKImx*KZL)8^l=kvol@`BF~77jkr##@3A z4wi4QFfrECuTKT8R%YCWV0{}j60mR4XNZXE-uU_!M9)Q$@bM&HXp~pZ%z_MNTgs55dv6i8o?UmNu3`=v<8@@5IhEtOXA~2Nrxuv1q=2N%bw1R zx}5gAqzgQgW3FoZ=SK5LH^En?@FPVXB7!2Uf_~DJVT%?I8WEu?M+i zhT@ID#>S?Hr8v}&%(QqX=VShMs{K~C-;TH6X6?7l^v(UfUk6n_g*$i>mth*FGodAa#RZRXWZAOeAB9{$FM&TL+|%(wsa znv0JQzLK3olB>!B1!WIo;cM0OQ4NdSUOay^I#SMls`RMnGW!K_8iJ*2 z;`cB5w@>$Yl*OG=U@RxWcnH3#c?2T07LF}>^}@IR}DC>X~*>k zvk$3#_67clFjH5zK_8|tAsck(EFAb-x*>00EYC+ek=CW%J&T6#k%A$DmJ#V--;&QUo0kc!4?I{g+>;Ye@f(gX|6}dLuYyxGB@>WUFBs^s3<^HX zHhn_3kWH3j7bU*_ifFzDnySe-pK1RCzgZFwns3 zio0~B%j}Odw*Z(9Wd2Kd@5MxywpMi#_L3-Mb%7R#Y!r6Gs1*qv9PJJBA8n2-&~n>d zDBNC=i*q-3?@5K{u0$@fo0N*iP5{>P0x@0ufvr_TcC6&t>m-fDBEvNQH-h_phZ$hR zQ~&bFqSFBQ?xmrv4s2d{91gvBzo7avijBvq49&;heyKGyYfWZmnU<6fC#YmZB*7&A z335^sU2y-esRXzW5S7zNk&<=N%_ePKP5s_ZLGsmve32UVOuG6&JK6af_=QyS@?gZCo(}J7P zg0;{hiIh)0*j`L@L;c3muG5VZmtXtWJfel&5$1xt_;pujcf;J6R(o5(= z?D3PT`Z%^{Lb>O>2ZA@oJkx-krl$)|kM_T&R5DtZUd>8~>96$jT_nSbj1S>^vsa~$7o`jj~-qH_HH*ZdtHeqD68*__4b{WZG)Fd*bErLG0@oO4o;U! z{Ru7{KY}&WrMVSnllf{xa%sFF>o24fDp$Zzw4K%M4icui2wY^0DNEbHAgJ6L8mwG!_IMT}X9^C6CtI9Ah6 zN_+x7B!Oce#~!~eg^9Qa(@+D8X+TN>%1FNvD6IhL-s=W%XaB=zGxJ&Ne!LYaql2r| z->ER)IHl80tJ_WY~IM9%%}NctD8f`8Fl6D&5TFgI=m4!y%+ z7!zO{%8WpZT3N%MH7~=nD?V&Nh0Bh~soR?%wBfQKgzYhhwYD2vW3reCwtMLMR<*u} zOt#qMFQF>nJ%ueFpc@ai$(7@Sn?CnRSS(x#jF)ySS z^+wy0{e;kT#)mENIOD@-32@%HguPa>NMMTkuoS^YW!lxp3!hhVY4GfqE>X~rXTREg z{S|!$n<|dS1t8eN2fo!riBLvR6g)!nvw3fC_NdJIIZtjI%(N3SE->-(fk%hrqNWM= zO(7X16yov=4oo*w>Dh6&$3_A@9^N{jFI0NX5=d5qf7c91IAdf%;gLaDW27sdXk>C8oHFOHJoJrn1yO_fd`4~{IsP=pn_19|pI^N~>yma? zMK#@J2&l(m_CX`pKa2L~6NS#ib$uyu;M!ai`XH0}ESB1WU?=NAm=f@_^fFSEkh)O% zPp*{9moJsRYyMKC=FzC~awBlf#FnRmWff%9#6c~wG+S<6u3sI>BDEQ9mjRH9QdiOJ z;w<8{;oip!_e%-lHc)GT?R}G!X%Rgg4hDniNmV?+V2zi=P%R^ERcTorGBRE|0$0+8 z1Zdg4{FPCXmBHi%@S1$rZYQ5V787ktA=V0 zv>Ew)WBWMP;k?o-u%*UWf%mtM`*&~qxPP1Tb745rzbhtymgcl5ZS_T9*)S9GESSiJ;*c{_+#b`mQq;19gfRSMAVr5IQgVT|l z2U64x9*D9MMwtUcyhm}Bfhwk)OdLwGtl4;?xl+k|44er5#Rk?V2%-QALvx>UNNUZi z^0as@j=)3a;-J$0R&8?q8y**KS5+`iBND{~{9liZML#gSC=y;c+k6=KFm;V#0j>$( zQCE+b?#l3qK^q!l8YSC08R*qUrKJg7((qFoM6o&uKWH-omQmCVi}SJ{-Y#~s>k64) z+$_$f_xw*dnL+E1*o=m=oA1{~F>W5W_yE{o_$(-bNbJZ|fo%xLgP0hLk&Zss#O5L- zmWNnP=l`Ba$>F3@kGQ(Xj1f*zjW}V_y`JjFN2KvstgMY6YGw=+R)Or-l5tkhV$D4l z#PVWpDqPzgJGa2f#$W7SutK${e29x(r=EpccbiIDC37Zd+%*feHh^a8Y~Tj4&Z67+ z1$Wt_W|om4W9Fv=bY*ABRrfMNWn)V3nBiEL?D1o}IT=#6tF3CF)S)slpqB(?D|`5j zYft{Y#GuYukaBB(AJcoUt)o6}PwB(Wt$T24$1}|L4D*TkU=}!E2Wx8@pqeBNO76eO(g0@Tu@pyg zilYPY{7BO9)N;j!Bl_TZ@=9qku>9#UC@Ko}-T zEB_yJY+DHDNEzNZOfgSM#tuMPH5qc=0THsr zGm9^LUA3UdL|H!ZuxPEw=|3Fe+$>KojVcLnP3SPqiDXEM6LsNcr6noO72%i$;Im?) zONwLNVxsk>xC7ASH`t_<6vw23ae{CN8|07{BuCA^GGh!OLPB@XJJ z6C5Nvr&)^IdUtJA!vJojxR>BPzoXnmylf{d4|ER|uRxEHtK#i3IW9Wb~ z+QTehO{NLW6@a?MJDJ{!mgG}q%O5Zf z`W@#YG>uZsVTO|MpVD(=O7j*bm8)?q?wC>|xPyaynE<#ol8*0cs~S+X+@W;`zw_b) z`2%+-docB;ZHWACxnn5q70N+!5AhpF0p5o_Rg)@N7+s zDN|ZuFCX5F_rMH3@MC5bwK@iQMqt1Tu*5S0<27i*!Soq{@r=N@s=#>j(5Lk(a)P3g zukLnCwtP6XT!DIquxWl~ps!kyH}CJhwjmpz=a&l*|MH43B%1PGsr5M7Bj|}gR!)2^ zLjzGZTj24Pa=%OWA{NaZlDIPH?1I!6YUz>Hzm>}ssVzlLhW5$5@xnt3a@A^dNOk7I zSEBC*ito|AcOvuZlwPT`W7N~<&3LdLK$u@SbL>3nU%d*=&B+qQ8i;(06*I+JufwOKtGtu8tHZP zldKT>QN$)GYm{pmO92e_tQwSK!d%Xx`KeKRUFf!tO8_O(Kp64b{nGr&cU{W49e$2i zfD)plbtSZ=^fqUWlkL@f=R21QZ%d~c(K3>kJvY?C#UF2n``;$_9WStg@YdSO{4)kv zu&`mP0%;l$%k==huI6~bWF~49%?rilF${)u=W4apI`|_o{luhqy1M6$D}gh7nX>Ly z{FvA{bFl49VY;}z#@X}r$u*gO_xzZCAC^Wcoc);XF__^n4pZ0U(_(I3FxuRn>uYtU zvC50-0SDkOOx}rWj@7NvfQNg#1_qvRq^F<-eI3hh5*^y^&@6vc@e7c>?S9&lLG`ekqr*o0`& zc}AGlCQ(O7E$MX1&K%lK1)B3MMma)8#AXH*`nEl{OPDbc^E1?nj4XDw6hGDsofaKO z-_S7Ux(Rr3j7I@^M*)b@rlFy~giqa5I0bsm>S@j89So)3-J8#h0>+rz(Vic%$x=H! zbURz(S10mQgSMVZB`Rnp)HBXKQgAtIt6D_cs*kedn>}#f>|^rs2*xaOwT!!2>TggN z(?6_++5);9Uro{h69X${CQCyLc-%}{qUk6b9o$Hk7YG}WouQOVR(-SIQco!xKn__Y zW2wk3Rn00`Vi8}r#00Wxp^+~Xc3(^-bMDytfyK7XkTVysW8g5Saxt6g?Qi4(-O5>! z3=<@(^r5rTNWOjoNki^m$VsWxIB`bmKwCTokj})6&=%|MbBLys^MPI=l1RxM z*4SA}vlgB6@2n#7@2n#7?@$pSc8<>)?ttH?o}9Jv?!+!KwdR9f7pV&TgAJnTIri*O zi*tn@`6u*nPaMb&NI)HPXEQq|RCIxqj)gNc5@~SIh#ai!^=I~R`}lIBT6!na52l|U z5?}!x;#39JnqTAsFnXzE(MzFB9t42F&LyoO) z5~~Ujl+F#e4SicuugA6yvej(7^(x$QjvC@jgBa)H4r5(`t@BYKAs_d0SFrh)fJyWn z{zQu#`zlm*z9R`h$3|MEtD*+}ojR!M9V`>cOFT1bnU8zz*r;)8`h_TB<3&X95u$z4 z&@=`_^Rqo_@rb!92N;wl_mDp*C3Ej$*Kth!+)V@&y(ve}`ldg6D>or14_^|Jd&7ag z-Fjm&kdhrpQpLTi^5IZPKB;~k{Hi(|%|4x7lr=m=)(av!@Z&d|(|2!`yBv%h_yLYp zzS98AcgX4(0B$W!Xg*|%xaY7!c)}cm%#pJozT46-l$65M8i_{fp~~P7z8m%UZtYXM zsNg%!9ivnDPH|C2<)c`Y5|{YyfXXPiwg;~>)NSH3h9nK5-Kv(2{GPswuwLRkY3XyZ z4RC@DMG|gs-t=DW#S>RxW+cuF&nb^0GO4O@=w2X=jUbutQuCvFmw9VF9qn?-b2`;} zCk|T{O%v8PC5@H@0-%moxDH$&3Zg3l&c0Bhx#m2GrBSlX;zQ~a>rS_1YwM`@mPYtm zP6@qDUd(V=wVcOZ=>^wIV>SfmsWg66wSl3xRHr^%bAPC4v{lczr@^kZ7E^4Etz(gwvUi z{MCUEiPq;}5Qjt`d3!$Jmh%gfe;0I-7|40c@UH2j2co(dJg75HheW4?a%@Qq7h}>| z5bRv`?Xzy~*7T%zc%nNZ`dUD^#)f)&Rd5`bZEDNAs~r)|Q3D1!dWR2jkAL$eRK0X* zu(Q#!7-a0#nn>PY(@iX)<3;LQeK&@4S?iuuJH4 zn?RZ4)5+uTOW(ItYJGEZP5j_;bG?FXuQKa_1uL(WE!J{kHJK7g+-aH9yydaBUkfnm z*aPypg$LVbEj;3`4jyEK2*t*p%#74x*6;Vp=(&@3pgrGNf9m4^(ugMXLxBz>!ol>X z!$~OP)fO@OlLRm_qY9Zhkg-&y<64M)u(uGS&q!TLh?HViaVk`@fyp=@RE&iEv9%JV zl{R13X2+8-n-~)B86a1)Pso_N36<))tBBWAczCwJ_%bsu*xsOhT}w=$60u=p+Xv9& z&||$oK}An_i5-#Iz}q?on!!p+No;_X;T2s#*#j%s)Z+pTxCdBp(Uz7{q>_`5jsB^l#}PngtKKuB00hAxQbU_9_K1v8DP!D z@~*h?F=`zPwYnuQ3A-Q$%3-?=Rw~qsSjoMd^ny#Q#8Zw{Osu2@8qJts6A^671oJr+ zy8KGskKv#!<&B_kRZicw0RIsP7d@#swe3Ixx7`E#_cgv>M$rk)X9v)ugp)pBJ{|p2x-G2Ay+Gf@sZJ^)AkT5IxAoz}zVy&jG_%CJp2?ltQT(y{vC2pqSfPs#KU1&+{ z;+9S_FGKS1loe;oE;ZHz;s9Z_8FWRTir~8S5@BYMq5aO-VswTXsXS&RlCFn?^(J>M zw#Za&CZvV)6IMn@L1hvT9`xEA0yG6<+6T=SDG)OA$HO|pZ1-f8pegS+_l@_0VE z`#wo)5HZ6v6h53a2e0bN<)&_Cw0sdCPd&Q-1fDeRlh7RV-XBRdQW_J^WH9O#pCq-w zetKTa8$Ft_C|^1bGxDDSSa1A-C6ZIU%O@iFp{W`HBd(j!Pv%aOGsv351&crdkhU zgXc@!#^~7hC2k{5+;SUrBeO-^7WNTvwjQJ5^hYX!A;G~PE9Z!&fgQQ8W4#yKT_2)&eeNT9K3+LuK+Q!zYTYl`i&8T(RUa|{{ESJW_C#syYt3go0TA7P zQaFmslPBp-Gq`Q=sqg*fxWvJy$&fUARs(-wGr30=Qs96kUGP75m@~C}M8=MR(H%o> zlNY=EcZmOV5VP>GNkrs=H$&l-#v#guA?$S5UiW(UMJHkpB4;fdW82C3NA+}l5bEmg z#bME7Fh9b7+iDdnXmh?pt9%(Qj;K`@s=3u8ZL8bf23H^LT18bd1`Mj%zZV|hf{BJl z$YHmhC1-N#S-aGDf7tK0>m&6}dM|Yw4042;9Zrd}AYHuiGy!YZ_hAm`xsOH|o7pq< z>2t*o6Il6aTjVmkdp*BMQN9!0-hg4{0bmU zd`%3%x0`&Nh;0CoeYV4c&cZ08f!Z>Z$fwse1QV#O=(%Z+9jFHs=~3P&xE9^vNa|e& z?OPdjHQFcQOGYI)Wjz+>vIzeF%;W!jnnduB6^&Ry*wTj#Hp^f}%kyK{PD|s_LrGt_ zODoTD%gr$gZ8p&&MD`n|xl6=>G~xtH-%ZI!FHeZ1(j~(6ypXE>EW&1p;YnuiO|iH5 zO%cQ-hI8lgVG{`JgmEcSnzI-N=bCAuQdbMK0L#1JN6iRv^kj`Vj#!QNfdZmTut{cV z4Vk%va4S#m7su686vrD9nwC*3;uze8I5rXwg*aw2UWnsE?R>y5uKCat$DxwIGA*x) zW6v^)W4>Hn9DA1a#BrR$)fdMx*>@4hSsbUHWe0+_{o+__)+=+-e!iB&gSR%J_zU`hLyWanudsX=~tHIhU(d}7g!sotA>%ZyS2gc(JX2>AAk zZen4{g`xonu_g4$qMIbEnyH*bxA;U<;sHh0i#l*=1G8;Pvna8Iq*iH}d%bM2Ea$Gq z7EI(gqJ7UBc^dNp=cY?=!;&wkgZlvMH-RIgOM*S#};pFOyy9 zAp;(Fdx%|@tUwYj6+F#y1RsR0K+YOVPN5|8<#y>2T#DOjitT>u?IP&Gsr+< zNYW=d=Q<1|t>r^x+B`a7c{-F`B!SE1M7HJSDWZ&>8Z!NG`XDe$5^#-scVovQ)EDfq z0HyLU7#~0=*G}@m%>oihPSoy=3RNzFK+$sD@G|+lnsJNax)-e49%|xwAMwJ&YF9Ju zs+--X0%K`dfElYRAhyBxh3Grm&%f`9^|984kr8ohQMHH?BZ>AyxuPhQo4J4Z__^ zE(6P1xePivyR}Zv?uMLAbM+>gNz#s}%#w}jm7tPF>@wiX)ofJOd6ED<8^$O8-$q@Qvtgrd^;`zDA!irUQ7(fHq1BR{m4Cx=*26d$6M^fZJf5 z{OC2Rn4@qHw zDX*w2nQtLmE2|7hu2q3*bSvuCX`mf!3U>B2hsvtytP-p^LPfONVjY~B3ZExF%tH4+ zd6xZ7c3KSnLi&x{xI9Zgd2-o%fdvNEr9T!#7>_UTg*HFl-CeAZ=5eGd+Gcn*IXLhm ziS``GaEfSaHG7lxpt&Wi0ix@(-z4lIF{Ug&d#+`)(~JDoUr2BDShiTG9y>8)6Z2$A z^{~i(##{Xqz12?f>3gfCd|GdHdfU@Ch=J@LfdXQ_Zt)xbd|2%B?5)n8YP4a}A5uln zxT_J`2lzbxqPna9(31b%+y8~Qt6Lqwx=A{qB3Y=@0T|rJz3-{71N2ZCCJ6MiG8!bx zLkE!jEgirW0oDO(iCh_0P)P@%?X$yNUu9H$!JXtiE%1z!{33?*z6Ho<=OnkB?>ot* z1zIP0db?f=z=nUOo#e3oGhXslPAM^+p7D~Ca`H!3Aa9r!v!#?1i#n~l*1E|-o^wmI zu*&}=IGr<(fMw|z(m4WxVWAVMKhAtzw$SwuLSUgIohO#gDH7pa(UU}8LOR>qigSYp zVWCIZ%);M}crDi|t*x&Tx$Jr1b3`Y_G~>;ZOUQu}yiM2RdXFVSUm|nw5$+o7b7>XF zg5)IU%TVXUasCWBl4()C-7d;^!m&x9T^dO)q3lSOOGp_PJd#7-d11>=VeQmh9A+;^ zYIWAO3aR9%INY^*kn4SiWq)?9r=C;T@{mkjJa4n?K9&&c2&9H-(*cm|jz^u-!>aq8?2#4sd$sIQ(W-3OsUjzo*-$Qk9Es6-P7A;6RN;5a zyU8apP*+_nkL^scGBl<;GgHbj)6-wZ4};z3e)k@- zp8H*!`|M4TE6F}}r3{Fl!hJSa$_^XkDDFD<87n7}F0^E+v-^GH8t(Vjuf60Eg1Y-% zLYX#RdsTv2j-b{9)c5}w=GM2~!kipGX*)oK41km*QTwM3?i7io1)sdqbcz*`6FH*O zNaHx5)0Drl+G)fcvVZ4KETsQwE2QoGvT*K*3IO3$U(OX|PPGZfCOW>5ET-96E-@Y6 zNrq=f%2mbmkVCbW#WabsGNM<+G$*C4i0OnK@u9+|*AUbx$zL;LMNr3^{eqe>>E~Qf zYd0T)dM8ixopz%8PC(>gCsLERl=pL=ptes|(L4KOuO_HHv3i1fzwcf`Q{LH~>lW0V z?_PqsL+G-ok)XDyIY3ZH=p@HT%Qb>pQxZ?OnQ(pjMY$}u9l+^4;mOa283Ed}(AEM> zKxd)N?^b9hnb()G3<3e>MWK*Q7G1`=8bxT6r5>K}vZbXZ9j$-f;@tw7&i(#Y9w~N?+>BoQ7tQ=^)lQO1bj%G z@}bT5BAoP2ne|SYefaCLJ#}^_GMED{_88pt$#8etm*EoSoecM!3>khvtSHIw)lNfs zP2V{5@k9`b36fm%n~MmNq&b7%xz3>@9d{fdcuryxH)MVfm3`absEhz`tuUF?hD2$} z)(!0}THTgW=#e*wx0^DyXkHmT7|p6OY^$>oX|=kruj=|tdpfpWsu^%mTtG$*T+<*o z8jcbOmV*&I`x*sI(mW8f>^|Ta{rfz#*)5O`*`{H1&NkpLZ@8*eMcSy@Y&c5ugeZ<{ zKJQeDT8$q4rZl6DN1zJm&1q9IczIe}PxAt8sRx=8R4eW0oCsMg@(&MW#qbG@}ou3a80U2Af&j`zX5DXZK{}}wol|0&}6x$Ka&|nVk zy8i?%XvmRpS8v@mIHC<+d>)1m*faw#-{%cP)dQSo^XMloUGzC>+BgOXMe|60a`>CP z6=e=WRO{ptI_MclYWk`bLI?EQ_X3mkWcr@6C^Cw>t0@G&hfuWJ=is+EwFJ_CifLu0 zhwXJrH9kVV`BW8uJI*t7Y4fR4hh%f}%y$dC*$>)E<2jco&V9+!(`xiB>ly{o{YF21 zosfPyL;5|-jdBn097fhun`@Hy&L8=2K%8HI0$#fWBTL)$3x)~4Z>e5!ro1rwcN+Ok zunI@ee9iQE*Q{B&=1R2}l_f{rEO)tp`Ac^2wEZ&phQ8^SlVB_Q=C;VL9j4b$rd)Ht~v-7TRU(PUOreL2y7%7iyROfA?(134 z@_MIuC{A!cZ@=cK!E2ZA{ZMlK0XJ1A%R(-0Y)@4QQ1e0^qc^x`{>W3&DhAZu^YG_n zGG_U5VYX&7r%T}M2d_UPI3=O#{-#L|&SQ0jZ_H>Gi*9*xA2Ag?f2SnFqs8qzz+^c_ z^pVOB{XKJfeS&SsR&%iUoS;k`*FJNW0uLAe);jH+n3FE%goVLtX3?T!a&vM5B*Yn~ znzrv7CY+oxly|Xh*dm}KAEPq@K`JbpLwBBI7?b}8RUcZ_v6A!)qm{g}m0ON{=kep= zP~2qx>*LB0Kq@i;8$)e**u?bnT49l(4dv5TkSqd_kZ&*-`UDFFq}aM!L|-_29S=d?Eorw zS%w6q$bsiSusO{3e7`78r(WNDy1dFjCjVPgl2z@a!GBDugMLSg2x;BY|BZ zW3W?)*gsU%H|7|UQSg0$LDv0fNFtQ`Ns#&`5lY`!-Rj)jxks06S<(}_Q_^jzp9^_I z=1E2&?5>4cy48_>asvIZx$D)T0j1V(_D2dX*n!&)bj_>eL`Z$;{+ zQ0*eMkN4*5WTL+?Mkk}RS3_|vb)_wnP8(r)5LVegdd--jSeuj0vMRcUc>T6m{E<8I z(h=9irTgI#Aa}4RJ4qc28YL$tS`l>HJ-5yxTNed!HAF242%-W@)~EG4L6mivw?)Kt^+Ca{TYs$B_yJDcAWuivb&v4UXU z*3#OD!{IFy<5D5k%*hnkPM38C0}U;>vTPY6}5z?*omFV0-D5VwR^l>xh{ zN7uVh+CZ^CSq7oS-rCZkuHz)hP7*X=z#Om^^0}A6h2UoET8V>ax?lg?=Pr}0K8}4H z(WFjDQbd(Qk84v73x753hN5{g6f_yxR0LAgwlrz$F^qxL3KuPPZgsbp@esHq<~e|N zP#uxW6nGu}TyQ07JW_COE6b_QK2_Q`2?L1{Z* zWy-{KWNz7}?ez<4(nANI0jnGA7OAmZyUSRvt;cdhNMz=jSc-xzZahIfBw1)K_!oV~0n|0CJzq&#i4xkhLPCQw~!eyf{$_8)( zj}o(aVPX{`HtH0Pdg+bH>&|8I;fNOVyjfjRPLOui$@ANN*jOFx zIPS9ct&Uo|^hYpw^(SC=QebZktkPed)j^|vfGEI1YkS%@y-|?@eeoBX`cP2CdV9J7 zP!wy_f?={SqVC%^vizJMj1zP`U)UNv&p<8(as~BEcN@rg5HJQ7tF^nU;q#KUPs3-I z@>vVLx_mR2124yU#~e8O;BCRIc!1Dr>nGheUa;C5Z32Z199KV>wX@Q~XlD(-IIz)9 zztqDDUJmmtsI5+R;U>0r-@aZ!=ef5`9kfP)wk8cG9e_t|OIH`6I2GoF#xxMDan3@R zX(Ko;ExywY_Al$42O_-~YcHO3UukQ?S!D(9X|eL{f8|O1|2SMh-e;yIaHJ*k8ciqr zdt*;`+>Q5@jmUOV?27q=BNQck7>Ao_w@!$)I5FcFXI0^|cEb{@@1D?Wj7%P*>K|8j zu$0Aedj7IQ!YP%J?vH-tPH*hORzU#0DrnWe4=;D$P>dg`qGURgk{_4)0@q{aubMpJ zh{B?&oFj@=>~YJ!X?5ytatBYt??H&keLnm;|9J~iBUahskC*PwqrA*+0yRzXc}m+` z3KhFarFp4%D-~>x4g^V>eUl1VwSJeHmUg(v4oe=c+;qHP%TX((0oj?7t-^mS-0Q`8 zn$)-I!?cKjNgZQmHuHjA8uEhiZO97-kj@FmqzXG}GI9$~C@r3eWyqRQx0x+$d19_G z2KLygang4ymtq2S!Xzs9*E8CXBzit^FZKLzN>*=;BdweEHyF3aZUhy-Kf!+Wzf76*mH|NEi%{jm>0bL&YtVNh-^%8Gya`Gh!huJghE~Gx@16ydLwrg0Vv( zeSN)uEA`{Rb?fH0_A_}BoDdMj<^tTCUe*2g(l%%OK|S|ZqS7HiR>1Z3a8S01g&|BtUu zlS7NtW~z0UdAM3M?Qu4muw#bkWBb_jafs$NkW*8JZ{5X`I^M-H8IQ+q2cIia%0jL* zVUWy%h8=x|o)z2BWk&9GzpbB2k{dY{V1Oq8D~B(d^xNk7v8_Dei~$0gpvpjYvjlJK z24>v3zU*%Xd#g1JiTi$zye4Wg|5l8pR-8aX_$0N4znEdvV1m15r&DrXsZa@2y zpZbnFYbBcqvTBX18R=%$u<7~M!}W3!eH`_Jqr~rWE|^dNP{ek`&tPlqN)~Ej*jl%8 zW-uXg>^qC7_;1bC<*S8?(8+SaB?~D0Zm9fTQG_H5cR>ytG|`xYN$W2(@)YiIY9J4W z90K_~7nFB}rY_J#4Vj)3pPFvIl+r|dFpsX!+IN-(> z`>Jv@Stq>`Qh5BBJ^~+PhXtkBxz!H(>I4@VZGjJ?)hPz-DtQKo^=q5&Jd1<)#N<(? z6hVoWznoT!?ESmL#X;IPV0N=H@amBtgxhc-4hj@a?=u6UWT}QOt}(ovK4n>KHY3GW z<{4>mh3mvl)O{<@-D&th!ucr=?L1H)*XRP4c9jo!Rd$r*c;yE@kmQY@A2xV>Hd-IQBLvG$L{)DTaa_%9@C_nbGJ8Q~kP>m5?m8G|fdbF3v z@}Mkq)Az~l51Vg#e>Mi?q)^eq4IaumRX0jy)0#yXNyn91dpRdG&^8fe zjWtAuRfq(B5b0NUz4NJQ+5HO4O*9n?2@jauTD;F$fegImmc;v7ID|i+C=MQ8!_=P8 z4r?hHQqEYP)6(h+W z0lOX`NW9wJ#ft)d~4SeuPEuD8PIXt5*?c1!`>FnhNGbC=#)RmH3J zD4qNER+$?4p<7X7(^v;lmI~rF2#}zTKv3dd5sTR%bvCAdenN%9%@`1{Wenv7(g*8+ zn#1{Hgm?jVnz`n6iCak*d{}#_(>|vhkA`!$#80NxIv+6qK|^y|K-TT?AUnVet{@#P z-WyJaFA)#h)T}_Vo9cA`^-mJS(+P>`eTzS78_`OvBZZTM zQ%1>s@j_9oBg9Fa+<%5ZEExT0@vT;VIpuGUZ(fgBTOQ6oj#@lypkKV9Icx~`8|Y*W z6<(x--*vtFcu{$fNNz4VTXz39cH;<3DJ^U@=k~z4n5LoN(ChgdM^XmkLV$8ldfR1E zm~L#m6q{|~1RFbGcc0$pJ)S^L$8X#B>c<0=HpZh%9o(2KzTGv;SE83tdHL}f(G1A` zRz!d`c;z;%4R{q?!l7G887uZhM}P%HP;KN6oBsfdeiOhj|7Vj8bVW1>_jBiX#T8#BVSW3)5+j=w#hO*hEKy=Jqzslix=A?o&iReLNk38z`$JU!ZEO!Rv${DM{$s@7rH|%)|j~?E0+rhMwYt?(CgZCZ`@2~p% z>iRcst0~I!@cw>(-{1e9cFRNr@_}nFDZu3ysO)I<>QM%Dp_hY5br1Gt$dD_4i1t~w zNX8$uKVd{Q$31pn$hO*6OB2Iy zu!C6Bel_<%41Oo|n!S02D!&Kp*aI>6Jzz&&!@0(eJrINU19r^B4zGFuJJ@Jzs$P#B zY?O)}XvqV1MD?3gpi}rlIE0#+N`*rTf~clZYmFc;GK?1)#)}MtqY6ths2~v5B>y7A z2p`#H82`)B)VrWhdyDDQqrqZ|-$r8yJ!5z^P7`a5?`_8GD$*G2QJurc|!!DsK`F zIBIEIp=S9q*PvvW_#)QlU~mXqXf^d{2p%`W7r0uJG zIfBXS-V7Xc-1Tj_t4t5^!JVW>9TKLSe$9P8`2#n}WLh9Nd5YxGnK$3AwugsLoxP+U z^6jO|OODw<{PZv_C)p8P{e)TfS)Xb5W~nf!;a`mfszU&0wx;IEI+#HJ#zSW>qp>7* zB`%)ZwU$@j&ixJc@-ml9+g6Bm5BV`+sh(V2TfU4|xdnbif${TnfQn2Bg1MG2UA|;_ z&*FQ_`{)BAuGK{nani4l z74k6T<)!IMubmou(wsuo>+_rS4Q7=4igi>JiKguyIFX~ z(PIb3S@!DtWjsk-9Qy~TmW(FbI+K8p31&H(BWQOyG$`@20}H`h791vIDq<6 z43>^?yyzqf&K+{|S}2h(k|LPd7OV#;b~U(iNO-4*%lI+o2c%3B{a3DZrN~LqjPXj{!Ev^+SQIeGB@oQ2D_*P^||QXK;#Hv=m4f}?xA=U@hImJv*gImrDX67e#xDa*YuE`%Y4^TJ|&v6p>znMMvEG4{+8MqM%Fi}(#> z4w8TU=McC#b#+CPe|3-J%+`o^r;f(c>13L0agbEDTsLnob+F4TL8#@G$4=!rSGol4 z?yLYnSTg&O(BPKYivz}&_||=2;@8p^Y%#ZNkTl?7Xi;iWj+6nSn!F5WazbXr;0DOs zuPfXzk#r0XueRIHR$JZcQmjB3MbZ(HavA|fDIBgcqiT-d2sF4bU+DO&sF8WRaz(i1 zd-HG`gmfXA!UNcCdH-rX#=1d-H z-a}7z53#r?9xq|m?wTAeUqTaEoSqc^evr)b)U z{k)*W^8k>I2__DD{F+ke*G62s9%eo5p16t1HSuFI6`SCs{=)^ks$AT!&~DiP0KDfI zrVtV|KuAEX?v)7$Colv@@V$_AA5C$ZZL)Q*`Rr4dT|-_&mf-eg?7Zd59Z+I}o>}FDCAkYQ!B(0Q2L|YRi2Q zac9tIXN6O4(lS=jmsL+Sj=mDU1NL-dIA)ve=_IpUv#;^<&+s>geH|#8I3%EW14}kp zShQf-a9B~s4fid^^gJswj2kYrD+gVYf$dHNo24x{u-%?HZLnPsz2{h-nhS{8TWVt- z#Hq9f4~!U6w9H)s@|4%PSG2#IXMHjlmaaDAC!;YnsA(E}?wB5T%jHzIRJT|9B0pSg zFT3Vr&6<~M5%Xo8E5QO6%{2o`hyZglY8w3)Hg+tK(Qjb)LJfc+z){RzF?|9WL=%{N z><@n>-$~mWEi8`sx1dHqBCndfWrFdfa?y@=Dy`q5H!jB_OF5d__`@{NvMi2)I=^}7 z4#v6N0iX9L0{OmNQ||8G0B@DX%oeR8`jsOY4$$)&i>uyVrNgoh*T*Ayf8#C#xUn&S zuc`sO620Jp0lab;z$m4sQ`el7r4XwdV2=mxgeI zob}2+`%VMuW=S`20NPim2fwHKi{$dJ9*ZR%D<_{tTHeAN|iWLNh1&$?G)_IE1ncZSi`mu50|QD!HuLr5H_HO1v1&vCj`jekb4sNV-)+ zJla^=FEwrei32SB)q(7;2xR7bN2_CKgXi6|`sT*i-ClgpI2k=ZU<>&0j*akoj4U(! zaaM^BLT%cQP)UH@e&dSB7k1)RL`m^PyTLT2Mv)J{&rN%qeDM2$OblQHht??nWd#xh zH29Rp1)aTHtLd$?3B#GTd=%ov;3&Z#k}x@qO*A*S)3fhf&LDpm8W!Wc<+g}k=S@5k zJ2U*#aN0C2XpOnmJI~Kp0A45855(9w2$HE)kbdZ7AR_69-vbdzKl~nuNc!RT(4q9h z?^!

4)F@>)#O3cuho750tI%Yeke!BO>($)VPrwQ`qADex(io=DjHeQ7YCv`T^g@ z83v$eWAXkww{Ih0{eed!KgNyE&;Q-H(qRz6zW82Wz1+%xp(EaxDqS|!S_T2v>e}MF z-%T-b*OG`lO=)@3C6pC}BfrS*VcLZ8;LD2#2@? z=Kte_EE!;oX1Xs|1yMtIX3;Q!0s+~7_$H>wlik=hXFfXEe#B|G`Q({46ii4LO@RS*z+>8RS_hBl`)?Q?wQ-Bp(%4KAvw*g8}|$Ds+R zP-xN4G|YcFdChoY(!wQk$%|SXl-T)4CeaN#VEL=x@$TDZLCL^B{%`KE-jgn$w0rV) zhN@*pFB5nH#h&5LSRR+#a(_Ac8X2hAfnwn(W`q}_2Qy+o2PJb(^*(~0Z7B{YbHnn4 zF@KGW7aWSWPVAu(A&XS@U7>CL%iPigL`NA%d-KwCHlE74;}84>^C{%#a&$9OMpCoL ziKH22xludEc?JG&l!$49hvL7g0J)7qXS6<=|JN0;<2Z`{0aL9rNap*bj&QkwZuIs1 z`_GM@Z1_&tLlM+!viQ?t&&!un5Ug}xyMoc9@(N#?7qiM$MJptcWVs2-ph@)O?@Vpr zRlS_q^_M*nr$*ucNyD`O!guR|=$yj+1)`4|$5vk*r7Kbc~s_gV$j!`#X zjR0ONPa9xOV-2kgMZSBSycw?G&~${vxO?y7$*!*KWT5OS)sTj&ti`0m&$9`)p@|-D zvM2xt3dW}jd38q(CX&#oW5_bR0+TZx+Moa334F6(zn*o-9cJB2jb*py$%+Q0dQjPH zj1%xzBoW*6D!o%g#mkar5Mx4l%%F;&xK#C~`-3;Tq=~2(6oV2Dphw`}Lr@_13&Vuy z1IIN(kmDizUNbgk2d2Xw!i2RIDZ=1^QenT?mlC4Jc=0`=h6HZfQvOM-#kTN}>%nle zjDgl+ar8YU8PjJl&U~*}FjUIohhHow@?+;I&v@Y6*`zTJUB_qw}ar-1F_0jvn#@QqBj$ls`uFU!c%gwa{s)3WIc`3WL zIg${g>h;xN)-#wjdOWHfb*A$@20MTM9{Ap3Wyu#hyFMJVs8wrOE4yDjz!Ly)>XYRn zv6lRC-2Ku)a%=_pAa|Dl9P5!UN_3hEe$>s;qQVBsP6}mrYc0iEE>7-Q&p~I3QgIk@ zb6-)PkyftpzT60s$DXkotJx3{Y!MM7cnT52oq~wQ@Cc}+LM_oGgI*(|XS!h6n~2z3 zt@{Haa%ldrPNM9Kh-k@dlxHLj5g`o&z8IwEK;ZdNocTmJfd3Q02ig#*&|ODnm-N2> zW}W|6j<;`q5c<|VcjrGgE90i1d!5jwv*rwTMb-MB5!}2?tzHmjT#=+f>ge67} zuYWo)FmavgbZ`E?V-UV!2ZwO>dhC#m26SpIQwf014p|HjAA)%Ny2^!=tE29tK@og!=4 zW(wfWz*%~oK}4%bOZMyJ7sRqDVpvTGW<@4lS(PhFio)3NRDqS3VUKm%$D<6 zcl^kGHEDW@3YrPhFs!pds~Ye(D8yb=a)58VUo}||AdpYN-vY;?PaPZ}UCn50vr=Si zrIet;HRJ(uhgOK3XEZzt3LfIRx&W)FP|F+MjsxTKKrBI(@qt8}>)#M7!G!~2Vc50x z9}vq`9TwApI@+tOLCyE%%C}}s)qCZ7p|1m0QP^)~qNf#S94ZdVw)wI`JjV%lj%sH$ z*$u)q@i`4YFX5tO_abGB9K`3N_`Tpf3N7A{l)WFFP&i z$lcN&OCo@hiTr7U2W81)mXNoDV_e zD(wyTO%QsiUfFa=jA}Y52MTx~1Y6tSNqiV(CDam!NFF z``yd{*Wvg3=_(l*!b&XnQT3M)vB=kNajmKHhxEp96!2q>(b^nz6uiU$uBn3m`)h)$ zBEw>D5^r2Av~U9ZK~sd|u(vtc8?1lX6$nr8>T-*zy%wIDF|tp$mzVt!}C?#8m3T#teQ zf=!FXOdp;guSCDSj=a~pU*L3;$uv2Er=3@Sh~pB~vXKx32tYMjoH9CBX>0c}iL+5u znNf!;q=y4wwvSlL%XTumw`!WFzix$Fjx;^ZCt&TWh9QB$9a{o0<;*LGvxY-+>?riA zOgwQ6KZ6{_sp|qWBc!y)i*2|;nx|=^_)<&8%peQtYPPZncU;sO;z-wThTCg*)>0D; z6A*SrrZJ%K4?1up1?7NfHM=w&jSPCI3GzXSczKY%6gA3&y&SE)b5r~aX zaIMGA_;&5);=h%F2n~^KRNag^oYXU>@iv+1sCz5cC_o%_?|{t-lq0?q4RY*vm8J%a z08;*TbgSEYmmV$+EdwclS>=j9V()H`s-a$q7#nkJ+YTTxLqh$G&^bSVsF(vL8X748 z%d7GoX8_-F_D_3;Z5b=?^ zy-SW?034x9;3|qlgK%pyy`8x-qrP>brI;+RrvpA)AL+Qgk}ld?zFt%S`t1u0r|`B~ zfZ6s1hGThh-!q{N9-c71Z(w!%B>hH)nm|#MS8j7C(3x{$ln*@t@sf_1Grw!u+v1Nd z?M-0_;AGa)rC`#BiyxHOnl)d<0Si7{d?JtGA6YXHIkf1<;R-|K{zocT3FJpZCAX!& zB8Glj6iRr)o!D?K1hIqS*_*8+olzbdQnQetC7JRnv~f7Rzv?((HWlg2lXNF7uZ-J( zZSkO+K5`$E!9hX?pdtt#?^iI5=VI6xaUqLl#I7+VkP`T(wVbpM3y>#+4aH=Dei#jF zkv0N_!!w2xX2#pT^t2mp1e(vkb*)z02O($G)gEB`dbC|P-nzg4!EgI}rXkA^;}yii zngX~xNvMOTVxJI(zh^NNS3a%}5e#w04^8Ncy8rFH0?hKltMB)eOtC4#KAEQ!ofyav zio{pUO1B+_(Ta~?TF?o5{Aga#!5BJ_z!>XWHBFf^E<3-^;s*!vh$otcEAPvB539wA zLMw??_vOu#l94*X;=m9COc}^4SI(dNrtL_$M~&1coUTz@ZzWnIv!20@9R}1{kqFt* zk2hvd(a_kkBLnc6LweDY@N5_BkG4K}IGA0&o_*IpN@lOsW=i-woji@Q#z0hcL)-s4rOW3r%IS{SfGPM;h*jZbPEVlzJlh zkV?$oE}QJU`NPz(XcHUaD)9u}czO7A_uil5fK^O)+=Poiu58He8sp!OJ_o$U$s9K; zGbo;CUKVeUwdA3k%2A)0XZTTQHm10C$vL8y1Fa^=P*0lD1NF18&QOC52_c!Alr~&5 zDBtTxCucJqf*!d+44kOf$UUU`9lb^#L3dUojx9DQa1yIL%g&4pnu0u9 zIF6xc)EAFg!<;vQZxYv6?DJ6hj)EoUn~ANoCHFBSS@t<~q3AVkS2t02MBEd(f?;~Z zAM18gvmSr#>0V&y)*0^)yXm`Dml5T&7WTHkRQf#Bp()1Em+0vn0JC*%!xCdduUfKNvE=(pEBx5sXkrTiyJ(dAv< z@sofeSWKUL`VPDQ=5?lxW4B5UI%A_9DJz}c`!sgtVt29ged4p zl$+{RJG?80n$>uFh#Gj8^D>3K;kA8hnS^euwUZaUr3Qd5D z?^*qgYn*)}^Zg+esz5p~-s{=!08*(FN+tG&K zV9PDc;s=`qJy%DJgb_n;0vZO)hZE*62nK=R(u$@@O+OKH6b}QvwDcs_@)EAh_!D8` zdnn?^Jj7JsDU+D#T9n}Yt? zBJ7X5uFNn!p0v>v*--J8hp~$_on4`=kO(%mQZTod;G$4$a9TkU*;`%_kGKRvJTg3U z5F4p9THSMNb+07CfnyvRJ&G4jIAh+E-Sl0nmv91lZx9;@fq_I`5};m^L|)R@pYe4d z5ia4lw7eP;dA`wIN*zqQT3&|0d84cxqgdX%K_W~>v0>;lAGyNg9`KwK+HHRWQhyW2VyAti{@q zHw6_Lr4b|Ipq`;x;zkzBTqSYgnbi@+R>$x`@g0Zi*Q`$HfWy}yAAu~tkG!w=1JFoa zi(_tgIS?gK0$f{3NeI( zh7zqKqKiw@vBJYm61b8vbZCbTmF*+4?&VNDDC%=%AkX4_Qg4AjX;`Zf7Uxfgf&?r8 zV&FHq`#&-swax}PGAC@yel1V(V2~%1ZcgOLX8A6j@hWf7 zm%m$aJ)WBlFDHG&2_a;heY^8NeK6f8AH?Prw@rKK$f!XxL@L%fRDPDxrh^|y$4ESY z^>JMoWD2eEW6Vljd%C;!E>D8rZxev^X93`kwK90BN2}wnoJ!5)Gelz(nX?WURMA@L z)X)+mWFWl7-;jTZkV&#*uZzDPmg4?8yyN}|xF3lP@!8L@(iG&A?xDug`7njr5p2oWcYmeTIRR2)3UXHrY z^YwhSu;HP|kan7CYI$^n(L5h_vx zba#OGICAZOrmbtmniU5Q8QT-|HAmkU8_7HEl>V>p|CZa@#Yf#AzkTO6upu#f>*L)Sm{nfKVq6#GYp!*D6fVEvroMx}pt zNXb)K%HO(tW|uk#a=wZgIavI8$JEkq_vZhQT)+KHc1)8=sFH!>1p|X&j6wpYF=1jOpAy`oo8=pC ze#IDe;Xl*Ge=o@3Xy^%UCxhv@>}i1N0Gw|~bXaAY+FLiprm#)La}o!~MpFdxZ8qhZ zf2Y^beF}$${n!VrruoNPaMWPPgjH@#V?zJR#f&Y}z&jBICUe=DMhKQ2g?<_ZlT1XI zY&DjN=p2AnON(1KmWiwztKnMNuv=;q0kTUOPsRi{#4zfVD)7 z8R@m0y!P;-0!yezdD2vw(4Rh2Afe0o9K4ck(B2!!^gE!)kRyKpSyN$3Ik24?z+u5; zLXXy$ZjiImfYWje$3Rd4Qw7jgHwz%iQy^2sdGTm2TqwYt{};M^{N|;{Cg6{MMxXDW zBMTn4())ZKj1EWh(G1Z+l6%AspNsb>Q~gFC9m)CCo()#)6p;Ml?h-ujF3zu_J(7>Y zP?N_YTPD}JLKI+{Xf%|8S(MV5^qU8L72%TeyzwbC^I|T%Ob3e+p9bUPqM`WfXWR{F+lXk-@On^C?8*7Ew9fo)zr)1oMaHKFg#qL0=z;4C@L+s+Arn=%L#{DLTlU za&~lMuFKdkP+w3J1B^Ls`T4*LXq}Ddx;o^`VNM;U z?qLV46AE+KV0}2i8qEFy*>lN?H2O6jl*f!m$jp} zIvRY)?8IP|JGYW5=y^S7%mXfqkvF=Nkn!MsU~=3*nd=?>*feGcO3*O1H0@#SwskqJ zu?(6NdeopvMY?jiUo~IW>fTdlTnTbx^4!0;b9;1l*{?tMD}FUyRD33iN%_}V>RGq` z(|}Xu%mp~zKMkW}@GXGCwz!}; zLMrihQ;K6qHhF}vMDo!N?L~N#N3jcN$9=Jw2VU93+66mDXDZz>oZ(xKo|5?n-N;pm z-c_5d`biQsrz6X-*b_4f(IM{m$5wcqnc+W)O_?dYTCt0x_0fFn`GM+tmDG#k1!)(F zw7AJ?o#AE^rZj{}h8X0TP)U6`mOlg}MA7Wyed{f+a=HwTb%hDdqxcXHBkYz>kt4`y zj?+_mMIrp8?EBK>;FxJ7nhvO8R(8+*@*NhIXJ=O(1jC50tGwthk*2ORGx8-&7nhQ!`TJ&a(rOUn`IQ0lur0##k2-x!vKTKJ~5RM90pcWvs#}@Mx0u$^fODPTa zs=Q;m_+f#~Q4DwZgj z2(4Hak|w)4oj2Kz+0!UxK%^;UAYB|uQ*o>xmu0WlVvQPCum^({SM}sv1ulNJ)xP3t zgD+|~3>lAwu9}ca>1{ie`Bz81 zrV8ANK`_JK2?88FI&z{k0+Ri8Ne3_kNnO+_&mk>NyWdTVEX_n&@q4;S^Om9a^4vy& zj-9N%W&T=?x_HV#ud;WvXR%P&D%8Sp1Slv7Lev&r1Z{x)-fRnW@22jB98e+KSOhLF zo)hyKX!TQjSABQVM(?CU+Ae;;vM(`FjlR z*h?=e@Qky)>EBHrZhTU6j+PPTCVo6U(wpq;q_?V^E*RHlgn=zdwNX^*i=X&;J?oXp+Ysg+J;3Gd7wJPY4p^@9^iK z@~DArvK{pS&YI?!2P7LD1ec%6Gv5E}ya$lS&NcD z;a*2-zE%jpSR|Y1Y^lK=u_9+&HEa`<98*x@JCf)xJYbU80Uh777T+BAG~1wXz;`On zbx@A#IO&)Z-BYbiCSYfFf?^$tZdk3NNAWP(GCf5(LMD1ujupd?bG=-Lv-zEIF7W^y zFMO*$WHK7Nx9lXZ%&ekXwG}lRLwc@a;cI-27LWl1cl$ttKn~QYe-k> z7;uZn@FX$?=u!^Ujo+Cg)l6~zFYljDQ5**Y-88M!HpVHrQ<+0anyDXO1^cS#SUM@M z46^yj)gJL|4=q#V21VHJp(=UQB&h2=$Tq|NR~k>d+;_KOUk-)VyC2tla+~yrc~*)x zKsmS&_%?dYcm$we(71sT{~)ks#GfMYvHiRF;devQi@)fxMN%XnE5ZPQ=Ll+@`h?XM zpf6ZiEpA0uv;l~LS=_dP2PPI*mFXkEGYvOmNRCX_*mjQKW&H-=5O)W6u~Qt+vX(s; zmGaLlt+XPO1w$gihi@yW#OxOE0*_(Uf7lKa>B_!@!U0v`QJef`;w%dw%(O9{OyW^2 z@?tz-mMxU~0=rO;GNy816In~BbAe7JAjv`oz$FekFP9^58e2dngUMqoG+|aBD2^)7 zDr~<1zCBFzCxX*bW${<6a5m;ao7Gx9my0z%+a=dg4~H-0b9wN&^r9~opm^QnNVd3 zuE9QjyK?x7r$hPQ~8 z!?^G1E`hI$Ckel;@wXA*t(hw||G>|B(j0uOyNv{rGC3fy**pp7cbkQzuLT~P?W;k2?&4SQN(}2Kuv|pVC`a^O;bVr_M1%hmYtzgJ3 z7y1W0Z(c;6{Wuu?Rmc!j_4wOwEB@0SLzeEmnU(x7h=4>`(Xj(UyS}B>@TQtb&+ShZf58c!UOEvaHl`4nvf>(H z!n$Zgj;u3}m;*H{aE6d1Fdtrs7izCNdSje^Jxx@ST?$MTwg_6hV>%v^i5*UtNH*B$ z-~Oqu(EWz@;SoN@2$Td<9lss@5fv8J3QnUomE=CDzLeV<3#P{goG0A_C9NqdT9Vw% zvkA)sr~y^TsgG`{2?IuE4;7*bqcuNT-QsrEUGgmb_{NETmeJ_7DRFpmbMbcy2@YNV zHx*FCgw-WVbDBh-+y;AyXkj223~0gv39r7wZmC2Zy7b2{ocGWPnaAGyz3>L13+h_ole#ownaD(#AuoOCo4sgqG ztWw_$>thCjP$EobuPNATZtPedsvW!6pI)g^^XSp0EkPc;%rWj=_5>1gN#KQ?5p>vL zp%;}B?Xh=YTTLLO*|urE+`YCQV5;j8-F3ViG1s=m z4S>tPQ8VH+LI}mHtDy)y2wIM!fQ+dpgOSyOxNie0M-~kX`zWGWsm9NESKfh>XD|k&5-cE;&f3)l%QiXa#xdfygy#h-pFSGm z-^fXkDc*`z92ANq7X;wMNO`I9xIj`?SlZxeq`=kGn)I%PV(9{a!KLosPtlBr|pg~3g_pOjp&R0WS) z_h%qd$lts^st>$rDVV*cg$K+{hBd=^&UXSiZkd#XEEV7Q^#nz)$fcS+?Ir0-a<(|0 zww5hB{Wv8-f(s0~IJCcn*=6q<)@!nyF)hX{Wfy`J2gZFOt!TaR474tN)OhZ0;egQ} z_6<*>!9?L<=aF$T(qj(^q8vTr=t$yvz=(;Cjq8K$@H6&(OVNtI%ynv!MIIhtVM4l` z#rRtGi1sN>ge@Et1yWz&Nq^HI^8+4PWff1G7=GiEnc&nIerwOOZ#w+;PL&F`srsjFl7OhS+bRRZnpL9TpKUr86 zbJ&XS>H&k7{lsyWm4~814VQRDGddym{lIZE;TeHH6R>`HpoTf_KK(mtd1_US3S0a= zSya637rp)azD*A=A|ET8u(lVGPlZ~23rZ0)C&dkL@J)Jd0J8BIesIwE0n7@63r3Oy zf|YB)p9e}}UO)^dTL`L>3w3&|een;(Lhv{UAND3KElDzuw+ODDk`tN9WbvcH*rAZ2 zf_?s-)K`gSbg!74xbY&Tz$#5?+u2f5UZreqRAv=b@bcN;Z&3R%@51@k{@arIK(2=xwU|1GtxDg_NQl_o_A z%2VBHhmG8dUJ)qz;%&lv<9VIQUbDfb@$q1NTwk>h(|(N?rS`Ra`E16Z8L$E<1#Hc1 zpal0>$r+(X*k~yTpbmHuwGYN^&cNWsK6etV@xs%>eLgj`9Qq-mnqJWU)T+ge<7oJz z#pGd-BnZLY{D2UQh67Qh_JI}?f2pi%T10UpRHbrKy~r>IY4+mv<1aFdMjoZ1!)77Y z^-|vBmWyMJM9(Gnni|qIN``W$vL*`?(NS7B75D@BzJcayaJx^rzm|LE|k*67f}X=M7PXk;&q{GOeW z%t)?#5uE*^MmzB7b$&kUPI{g`%ELV(LP-tAw#skz{`!l< z*ZCg;E-Wh8^Y+D^J3$wJysiJ?)mvj4Vi<1$1la^yr_kXV&HnA<7rlM7fBWe9xAXTL zarm_z65w3Y`1;@`aReKevTA%Fex_w*piG$Hvrl6eo1%Y3{E;?`OSViLxc37v^h3I* z&BN6MDP%wsDrpYYV10Dx`Td=9?N#gov=i+Gymd+`s0-6uW|TWalPKv>i8pl5|k|;?LkB3YOWJV5+E9%x` za%TdvYd6f{Q^3o@UJ~P?*8@Ct=le6Pj|TB=s%!Ma9xm}z$E%}ZTp$x*vWzz|>Tsqd zz-qABYLOURa}u^ofOlG~WznzY!U*kz0!heOh2)K@mB}I{25}pvPOyimuRkA{8oyE6 zR2xr^E-pvw!!;@B5r`#B0ZSI^+#2PzFtgb@6urbJlmUxgg(1f6q`l;=&_~HW1Esuv zh$x(#g*ozoC7qza^()DZSWlX(U~e>;a=zLM&j4wbvPaaRz}{q({ou(qg2EPEc%mLN zB6YK2M1>!Y2;_7`!O>wv7#762P@#hhCe3c)Bl@Ctc3}b}xJH8RqKo>jHo|L5W8G^V zbHC`OJj&{(fWkQ)yX9Epu|_^<=UZLj(T9gT#N)M*{<#TPG8D;#Y+w}3c{zT=iVV$E ziJjf#&Iw@}`5gSKk&@Jl%4P+t!%1+cV2YiKo z0>*v9PT9}0b<&4&CCZCO?4(FnEffr=w|G-gc3}py^R;-CT5+qp9do3uu-6?MnkP%@ zUhX$o!I-IxLvj5SrstdE(dZFa#So3Aa0DpNJLvV0j~{!=JrBps+=0zIGjV*nPkYvX z`Q&o%srh$L*&(pa!iWFq2ks0Eji=m!#yS1&XSbhnDBh6t?&m(bQ@9@gtL?{NE_ngE zcYk=Za2A_DJbK6ly-sV?eWI=)zyaTVvcBT?R$+?fG;J9bfyg0>*e4DR$8g0zp8WEh zv3{3+bC0|z&Ae_r-C(I{q2m{O*UJ29)0WR7L9NMW@i{Ngg~%I+24t-Py=l`v50bnV zChy&%xT4gSATC7*9r*Oj?K{t)G8YUvAjBJ3(vZSpWbseN-S6<1ynhvg-=>|ldil%8 zQ*L1S#@xbEvm6gF5;sSXk)0t>pf{{b`sIu9i|O1GaXJ6L*L@SA2>ZH5kE;P6vn_4QDeYr~AJan+^673$+gAYl3R8%%%Nc?0Ux@LMlQj+a zB3K31>%l4AJ}Ar6=SC*)3s&?HR4VL0eQOQG*4}a7ZuPeb)^xxf&5U*pN)(uOM6LC? z1$O$*w%N3MHrJ{I-s}b8QmTP+&{-oSvjf;9fy`hJ7~5)}(SZR$XNKLDfKvj(KSs#2 z;!ePFaCi78ecE~WC&2wY{1F(&suujS4xlR(lo7K2nSbOexGUqyGHlF(5gU};0N!vY zz+N3Np*9N=6F3XA1QE4!+-70s+q+=V@x@q_G(+cSK_oKn6lVbeCaa5qNH-=QJY11a zuJv}}lMi!EjFie_h*S;3eIhBW_VwG3Lp3#z#4rpUgsDSm((y`KQ2LF@2jFom_pA6a zfg*hLv6dc-bLBE4_Ml`&BY2PO@xyhWCFep`$>hVa7&8Dg#s;rQGo(W`85W#zv1F@% z;Ed+SB8K&I;f(D1!5MRZ#A)9Edx7A5vkC}9*eCm&G(eU)2g98y4rV{4+)2DFXFR1( zLJZ`LoWxpQ+Te^7|Ck(+FpLlTiJAAo!(}21hTNOL#zv3H*tnVj#>%dz7ge4^t3XdR zaWxh^pCBKT>gnLm@!`i0jhG1|4g==V2icbR@ZtR5O}PHYA}k~R;J^o&*?qVPBG-}U ztfP8a6bw+g)=_PPAM}hU=+}|2)}b^m1G=PX7EzjQi-@(|eXLq{f!WqQ2Fsa@9GoDShcZwe^5fRx1GPOci`fRZLobpe1~XXm`A6>|D`>OIpW_Od z@pZ|Sh6Y!I=w1GWvADxmaZ$yo|N4Hu-lm`TE%T}s_AT@@eyt7W&$C}SZ8xjrBp+}; zo45k}O;A?jV~p)_U^@l2un(rG5kpGO>1aAeBC~6UF@ip{U$?QY4i`Ty82F94e#ENx zb==15{?144Y^s+>cxqZ0{{aDn8u5qS&yTdj5BSVKZzM9Gr@!W_UsF#zEYKd1LYPNS z+^i1aKnxE0;J$hgT9~3XH|lcC>Lfhhf2Vf7e4#iXqbHg4>z9){P@0D*QJ;X_s!V;7 zQ`D-Uv)6hr&@I1fXdK{#_EM1?gm*Fc;g9Y_Bq!`1P|fIO-$X_79Cfhq8?Aj4&eso`3sKJzBs zMgPp%IoKZ)iT>H{HELPZ#*6>?4lCGUo(zj8C+q&wh6bPYeQ-4D9s+jLz-|KOG+>;? z0{rA5p}K)(&Tq|7&b5iOX^iyA4{a2c>i;tKS2M{Ex_{qN^plr_8XT?x7uo^WY*$@q zd$q=0t-g7zHW*r}c48quE7R$%_&8M;!HcU|Jpk{YK?!Eu?;mu zT9V<{1YdXzQQJuf(aMVn14uyBeLzrYi6^dW3y^Xm4ru~=TUAH z3}60OmI>j+BhJ3bSa*y)l34k4$Jm3k&zo<{17=tdjhRi>Iw3|rQ-wddKqIGvjMB(4 ziLf9HsADSR2IMa4K+OC8Na{G@`{5wEB&b0PISO@z=7=@zsY6#u(?XJ`b{gu)`BA)V zH4w{W^Ay@t_aH=_)T2FBB#_D@B#m^4XeUUjKT>pfrb^b-3n+$TXIOA+3L>}&3$CY` zFEIrSh}!5tZ$fYv??tU+pt*&Io0to~Xqb!IvPL#}${N`W6A;(V2?7z}R>n5f`Oj^WY(cQc^xG(6AG z5bAjXeqr=SROs05)w!M;bUlHesJZMYH0>Py?$ex_F(7vFPZTq7BSZx9)1U?O03@=r zo}!6wJw?%Z?(pVyX@4~Tu6xHr`i^H9aktG4-aTE-K7mTbc8}XLQEyTCs)_+vUk9Jy z<1K!m3chOcg!T!8!G_pq__^JFU&q)SW_0rj%fI4YdEugbMOVFvI!F*=8w@i1P1Wp; zp;;&W&@f!o_y;(OQBC;jhAs5>9JGYM??B+HEHWzH6WmpwL= z71Uq^G6siqxlg_4;Ua_oF@g_Ex{1MiZ)cohe2yq+1Hp6IY#^4}Ct&+{nG@(x2KalWy@6T z5^Er6SiRegg3GhJh+?Irjw7gCrnfZ$TeWlQi>zlFBn?UDOhvlQF_PstXNH#jISSrw zMbYw`Y|z>QpZ(rD1E%?Od7v%uJ%4%q6!S77yV}1N4q4ytUjYnxu=oF*jiwLt7zmG6 zVg639B+#%eNbxsh=oVjw2P4on6@nYA)b6;P){Tw*%CG8zzpq@4F50wwr9g}fex<_^ znmrV3=m9#;+Xr^v@|*V$-(>&~Zw%n^8o-+_9>AN10laCq3p;pSKY-WmzO4bgr53Os zb3J_9T}Jh`jZyu98r9PmkBZD~4J=RZ_UowL-jC|-yCCaskDEfacIW-dZy#Uoy;xlL zw|`w-2DV^Jk?TO1(dedLe48TV<@TuvYIf;Q{>3$ZIYUT7ydg<}O>imteai!}^RKM& z&=(8z4Bsaof4Gq~EUf>)&MT!c-%K3?1=;Fx-jtWe^nS=I9>-4)+1VTxu#OWNr#U@c z!?>j?is1YywA>xI!v42r)TZ~j>!{JoSqHd7wj?T3RLxYV1#+Z!abEF#5oTJYP4DDk zeDg|0XrQ9$)k@6jVfmZ2Es2(n^`f(Gt#9k4LgPMeS`H9Jee0XXb%k*C!W+~2woatZ z#$FuG9&_S@RWvvzdnc}i;&?tq^suJHL1%mk*{QqqIEyH*>?{qkkS#;jlP}=iean+w zgtCO^tAYV6bpIdr7+64sZ0(Cm!%Y||?d4s1-5P!^z5Z8Ah+Uu-{1N~I@5iYLwt0aI z>ALVyUOE_!#^dp1hGAw_@V#dM2`F~@;OK$?Aj?n{0|PBVuFvHa!s2o`?PVKXFn6d->N*%KnM{B&NYzOQ2bi%4&*!t zthDMOjm_Tu%?%#wxS&DHNP~RV%vbl|i7!nq^>IEYH&)RBK=4))SKz4oMq#_5RoKWz zo0Q4GxwjS?+gWL0w6lg^9N03F5V4q1^SGx(OeR<9yxn!^7f|f&zJ0xd8+8ym0mwis z5PvF~bz?7Gol{vhCw(+8eTsI{L_c+y#^D-Wq#X7^_#glxt11hJ3t}>{u)5GaDwd7C zb_>!y5?O6W9CknX(VzGz_kv+h(@VW!m7ZSRySz94)}Qb_>%+%SxXE&~u953YjvCj# z7n0`E6Q!Enx4ZrBB@HFnC=VIKS*}U|2KQ)ROqcgsY`Wy6m+n}x1Y=~%GNTl(j%)OX z^6j*!Bh3o|S4T-!7XcBb4$6sO2PL?Us@}0M10k7Fl;plQU)O&v^{sucq@g@5+ZXQ^+zYp-siIM|Py_l(7jFWw(HS-R` zjYJAIleL)jORVZ~m}DYVnW(6hTWv{vDX2`$=2_UvdRjNDvL|_uZu%=yczNORgW%B- z&j}#lJ-dnT3R)dCH~v)Nu+|6`FaXoUf!~W>IXI5|Xl5k`{}V<9QjuvAEscA+G9Wte zCs!d|QQTdj&wI*n6RqzV9Qw!Je#8=0PTYILd~#%Tm+NDN^&YbN@l`FqGa|;0KpZ$SsfQYUi5s$Yc%ExaR9PBjt7Hm zt-CDJp@D>a$SG~;%v1`7m+DIc>KAcK3$Cm6a#1Hq6<^)6G&)frE?;n(!zAUGG^|&; z>CZK_-HGi;yq&=egZQoJSpc#i1%+J^$IDG14^vgdDa4lJEC84S0pvl??M=X(d7Tq+ z+ig-NAS-M;C*;0(NNoUhSf!F`4f~?ZA3#J+s)2nwQZ3U>9f*t{hYfrG>b){5n39YNHVf?HVNzBt21bRkqjaP-r6c1P!E&~|G<7p=R9X~sL7`qh@IXh9xyZ*oBn4DL*tMTp4F75B{ zkH(WJ`)U?l;a`eQbLE>f+;Ni3h@}Mjy4&`eV8(xn;$dJDHpgxa*;ILxi49_0t)VR6 z0b@nnrb(_vp~<>8SUA|E-mT3_TSXm4-Lc;Ho^efJYH{VXwrF!0-YYnYdqk5|IOOBV zE;uN)-377pH9?l4O>o^prpP~RnK&!_pa1L*aCvz}bvb<+<}J>1PXYmqzC=NR#1z}T z;eGCVJ7~ROt=HV~cdr3vf@H@7m>6;^T9D^KDW1b3;q-e29AAj?h2*K^G^trT@10gH z(HGr-C`38I8QA^X1ft6ACX0XWWHVQSoxWb?QGvM>`=Q?h|Fl0GjeEXYAy$n!snjMt z9EeFJK5KgjY>D+Sn6`J>` z8AdL`q2lNk1rKl4y;8RGbG8Fqn6`;z^DS+)7bK)JI?AlcxoZG4KpZ&5HXYWwpV>+xZTIrkI76i5{GXq$PdMU*KDZu9 zNm&iaU~&QmiT**6aGL@>Mfq!Z6TsgF(^)EDX=`m3k%iI1=mV{P{?b)deV6xE9_Fsm_>4 zwIeT)cCpLGjm}tWRX3`mGv-3z!N%RsNHC4PJ(@)Rb))*)#>EM|wsB9QGsea}>2*eK zuU_kn*tkbT!mBf0s*%ugbVe@CRl1x2^LBeWmf^D*UQ2!~0GCQvr6`M!u~YsPI3 zhT20cMC1gXV!#%J%;m^oG@wehzk)L3Fk&@+fC~;ffgO)IuBS3t_(BS-3vkd9=)At? z4hJTniDf}Ckf*iatJAja!jmldK8VpTIU<94JH1=?;)ZvrV7K5!W2o*k%+~$j<~wf? zWz?y$o>-1q0{K?0{3!x>6*%v$JOUVa=C+F#Rk;IA>2-^u_jl)FiHj+Ef1qeHgEVNm zoEqU96wNV`1ZVDm)0L4jN8}p&uChAp8Jo@a*nVx_Avc>`E3ao?u-}5yB&+iB zmlm66yBj-kVT+}-Odq!V_8#1U??Vi3k*Nq!~ZOqsjQ1HUtWv={oAudD? z+#N7FXi5gdTJg~~Tlag1zI%BqqcK(VD^bsiC~39~-^GC~+!@1JuE1E-Gm1KA3@#Bg zv_8uPqgqE%@3R81g!Cyf}(y$bfv2yn(c&I^GU z$zYjFhhy(2B5kHp)-A)$>U=0B@hkLB_-$nS8Hz z45DDm`OZ5A*{Hzw0a|NlVg%N%v5rCB$1!+H0A3Hx&%-fj0LivvFj^gpl9BX|L6yDc z7-TtxrYkrfD2W@w&pQU=;Tm=@M60-ERr2&se(7+kTo%6poWecxn~zh={bm+RR%g&f z#d=k{7bG+7a z|BUO}GEM>u?~(l2gsy(O#Sn|IJCHF;ovejn6kh3#q(vHv-h$H@8NsRj5#}3ENx0HW$mq z*)}}0!gwOT>yR|J5GC724l<9;wsFxB=^@(#gWv#;XQ{W2Nb{IWZ2HH;5osP@TR-lP zNGp;B461mhS1LJD8DpnGGvJtX+&)P^OS=IzGQp(Xn69H?@F{j?fyiOMg0-he zD&@vDxciJ7?K8S^-=lP1Ni%Td5)~etqaF1hLjKo4o}eXUR9Fy|*UvfKrC6I7EpDde zz{yDuJNZMzWg2oS1KKzfjY1o9-@@4TxJi z3x##WVz^MAiE^Q`V62Y|MNekE0yxtswmk_XPT^dF4zX@$J|1sjKGXHtd`yHd|9o1` z{x+EvY9V)w$OcR%$CpRBL{H_Ash`wT<^7M42WY&McE6(1PuyKIp*@kNy5t5t71nE_Q|=k?IwqfsRN!{&)rLQ75EUxvhxD1 z)v=7x1r655)OO8uv@=pSrtUJXY+i_-zxwz9lWj$QY{^nhJ zq5Yk?r^>`CHO+Gnkd3pEgx)g$>(j}M98OI;VIN}Jl#>g$%|S-@kvUaFxb7BE?O^&a&p3xfD_esFqqAFP2F{#umfMg7n-~So zA9_`9tt7SgyLU(d?9GZrIuqEXovNNhN7-l=!c8MI5x;{>J>x>7(41Y)sOhdAEFlMN zNbxne*yexYa3hZ5+!BJV5c>MRB`b3aYxdIqY_=Iq!J=-@lywq2_j$gvp~YTW@M5Qm zG+9>j)>hfs7+7G_n;0~0){egWgrWmp9T#h-t*xStV;xT6R$h&ac9U!rbSR*=##)K< z0Tp@*H^A;quEWmfP7?~7%rdy1iS=;;FE6GL5WrU?1<>4(3R-VL7X=AK%65M|D44>j z+K@@2Q1dT?ZHSX#hqH<@=vK61J(%RqhAu05myinC-qwUu^umw|X-y21VmGR=s4Zw@ zYkWov0e+X2U7foRV5hZ#*u|=yW$lT5pLVlTlu1eK0LUPbbHvVb+i$TlTjP`1?*c#P z$y%|G+DIL8XEq^th4IdkN$}a255dnYNS|bxIh5cJHwk{$6MVLV-84obh2Y(q-h^rq z{EUkzivjlpk6#9tDd|zzsIK4v;LrAQIEF)pp=c+>;}$M5ek55U2X3(3oBxF)l2tZ7 z@X)}@Cl7Z>=pX*zx7_BC9y4cJ)9waV7G9{-!jo~Oz`F%M2M!5NgmDA^cqdp$;sRm9%0a)PxOn-^F;JXHKt4z~6+Iq^DF#k0=IL#xu*+bU7=6aISj z=W8pm_+TCDl$*2Aytu1(hLWaRk#}mAT;c1J6X53Z zch@@@AKvDK=5>L0Y`|HG=Co5lY29ahx~3vZODSdNi{CvD$FJ(JtV z-uG_GhznUcU8UEYKJ&T!m{&cACGhlq4;)ivQNUjYbn)d+nIoc~TJQMUN;)xthj;VG zDDU|>@*MX!M$bAB_EpRB>mg>VnG-3_$A*ZwyU;0N}U4dXgw zKXD=}8oTVZ+HDzl_=?d7zvs^F(N~V>N?`EdW9tVl!Yf9fSU)fc_$x*~yne9c!B>oa z#0Re#l{)2A9$)+^rx>#E_nQ}g)~P`)QD$#Uu7CcQy*+gEdYoNrkJvj%D!#@qXbS~C zQwnQfb)5*%-!9|s7&!4GvJb%G4&>d+d}_PQ{F}z0Zjo|0z|;9f@(jpE|CH~@Frf@g z$9!9oqKrR__%~QWf9=(emx6O(6@{)++!M`WRHnf<*iQTE+AV%bxGhKXr><3 zGIL5W&%MdR!uajQn){jEpE>F(yOwX|NgH*)3=)Q1{wl4lne>gx6K8;haTm1d@^Z-2}1sq2x;1nAD=M78p-3UmdE|G4&D`~`i+YozPW$+6P9cgRppO}qt3etdlBhIZlPJPw*hc8n$yc*)-KZN>P@DlXrPIF-thdX*ZKHh9Vx#=0&y_zf zO*ZCqPDX9L1C@%qYz%!ge>4-5RWM4{pp7v<+JI~3L|>=6sZ1=_0%iKi%-^`%Pz}B{ zlpi)!a%E`i{tMgs$CI^^@IDOv{p*F8m_7yMha4}d{g*TR3UJ_2_ug@33^~*DdqgQM z(oWTV`jmN9?$J^8EG7#9wgwL5a4oxJZ@6#qyxn9?XBS zT(buo#C|YRkiZ&g;OC}t-g762?APW4{p-?`*}28T^3Sw@`lq1!IT_#;BBW!Bj5BJxV4!W zJdW$wN>^z@qE@Y@>N5O8N)pnS+6K+Cb-U-4&|EAKZHq6@nF0dIcwb9Nica<>ODD%O z>VsL)uPo<@N}V!9-R;OgSQSv1Y^0~-c=q^7PFb+3ETiHbg?PHW0&p<96#R(z#h=sh z6cyRAfD%B&7jnvO!n~qtF(GgQ&i12`1*7gIDm*y3IHd9D86j-QFuGu@Uk6%nXz$gd z%XBz;gE4Jn{a{md|Izz^!TzI9lE(!}usr0n zAYJ^jSiQOgeT>ddF;D#frw$2#;Cfw|xq6gDbD8RkcUv4796UI>%+ap{%ZoQdX>n(# zG`K_Q*PjwqKo?V&a-aU_gga_C265B|5f`A_5(ewE0p#liW8jkXw8|dE1l5K{Pic|e z;_jQ4kCDhFb|jxKPfUr8dBhE0Y8s-yN7s z;CXlE!et6_gUn(X@&qZW!VgON;Rrh~{#05DqfqIkoe&O6teiC23?MsDRaI4gZ!kD^ zoht6Wan0I|IPOKIC9YXEH7hBrz!=2);r!}oIfnZ9k3mfrzhcjG_~V=>Hzwcp7f_Sd zKi{99uVL;mkMTbB#;+N}0GAVXpKRsTBN|&az2?$QOHsH(*C`EW{)ut)gF=1jW<(g( zr7L6NMfhXbNd6=TI#P(KDS0PA;Q8hHvQFp>HIU`SS$#om=ED8l@{{&lD{d&wQ-aj^ zC;2xP?-vLZY|?Wy|KUA6;=qZ?J+k5sB7;^@yQho_IlY1%om2{bo~z8~hEp5?@XC=z zWzUs(#**?h&JXXkI2vovTPUENFYpmHGW*i(YUpm=vcUA9d4TV<|4H63Isuj6Q2QE4 zI60=T8HK-StSYH>p1?3!D?U1pqaoFwzkB#m!$>iY2DAPtOS#?r47kfY2e3gU}cPX!{S0k_~NB%GW z2~Glos6SO~%HsG$B7u%u*e%i(c02Tzz1rQI+zQxLo*X({Y|Bp19flPS`UN!l# zVB6rOadDc395h;vsU8b?{@wezKZW%P(y~)Jpi&`j;azxLxK$E#Wh;<6Q04;m5!B*4 zgm{`0H9Y!S)D$~Gf#OXVC3tK0dUg`QuW`^St<^L6c@gsldeMgHWt5AEQs@;H$geH* zl2JN>v5dIb5Y7_O%SP?jIl%r4y}}{~^s=c`Hc%pBNjSKl4#WZXng9a`XpH(yZne)` zW@lqW2Ga~UmZQg3duY4-%uU8^Ofp4+tl%+#Y)PtcdZ`KTWym}}TsZ;5LUgOacs+#_ zg45{%EvADi1+}a7DymY;Hs(T5FhoE}+RR=O=NZSHC8VVa&A%`DDmkba^}qED!vwO_ z;R;`}rA}*KlBJdlG_dwdjcadnshz{H)L|Q2ORey!BcVpdtfK7%^9_F*Ej}lJ=JQV< zK^kCdRw}C%6p!`UqqHjz%XBoHk;DuMesDKaH#$R#)cG%(aQR&Z)_^zWwXU@yKqYk% zzI2UB$H07(PDlo|N3HDOqnyv{^VORaS^!fZ|U-qbO*=E{J2Q z|JqAshTzjyr&DS`gg$h}&^Ag~6e%nDHlY|U5DOw3cB-e?A5jWU*hueN5WRiJD9ZB- zyDse^dKvuOu1X}fRCH12gtq-;7WPM1&UJ5N^{8A6r?Mm7TS=T+Ih;X9K~<^Y@M>gN z5JCpQRP>(+p+NTw>#_9hyRBo*zi_wdw6%np@mkXi2OE}9{m_Zm+EIa#Gd!O4s|UP` zROlckLoLw&-+qCjwXD~4SiZJ+SykpORpls!ixLCE!2+0#MRV~*8$K^syn-k-ej}iK z!NmiAdp!A)ES?v*h+e%tvG<5f0g!^uGA%e20GZE!V6veM>_;w*7XMCeSi&MIh#H-_ zYbVnXm9&itn!C*d?*ETC=Yh?_(+exQa?WrzVd1&PyZM3_wY%IH;5!**$%8Oh(_{VFmw3bl8VSa(m)H6N} z3^(p{vxME&N#{-2?OMvft|$!Y`S!)qcD$)dD21vdW^+guw1qTSSd)x8KiRU9Ho}S% zj_}iI>^xieLB!bu7&oj*>|~`DgCB`YMx0Q=J$O0=X@uAF&rR4bKL&A7WP1xD)1s6N zM3t>1mbj?PvbPvZX{Tn0sp`zZ`jIxHQ)yx}~L`p7_@k?Iqp$K{NL*CEwB0aH}SoRbxjz$a>s%ehz`1*p| zO?JE$QS~M17PSk#R3blZuNtvFeoF-UmYlS{#SZ74UqfYn={5AaT$iUP{L9~3GeR_Y z#+SyRjmv|l>a=Sv$5~9vnQzpK5FgiyAJ8MMYD3-6ep$;Em4tGn>=_a;oX1=V_9cy9V6i=d9`v7WnYKPH@jr20Bwc|QBoBYJyg z`i|L^Xz$3-*Bzywah=L99U|cYAu(}u{PGLXkS@XK_|!y zIsr|nW<%Z1s~Hzx8t@%zh%TT{o%se4L$IytODQ2!F>v4;?m0RaU|W>d(ON^x2E9c( z&@ZRoB2Do5d<~qggoI?3_EamT2lW(c0TYGRX+;{RY*z7G-l$k!3QfO8cTX%hff$a} zp+*Rz$Jyl#x_>G*R;7xjTZvhDQ%t(bsp&#{v8*@jPYiT7T8hfuIA&Fj-^q4 zs~e#cI)0{3=;YU&!hb|OLroY~o#~u?i9*(na+Br^Cg>FLKuWV?6XtKqGMqLP-}N|? zrRJsUYo)Njt#D|hu8iKy-%@y6 zvLl;LQZBu~cGId4&kyq0LpOGd7bnqcUbjyGsTwp&tqN8$js4x|Efknr`~!;aTYo&D z=+eo7dGZ1ZJ)0PO<}U`%2;RBuUN+RrCKk+0%5?g{#9q8tjQB_+Fdh~ce09b9!$iPY^znAT>Xl#HQIfO{Bx`%ykvAmvCp9M(2bRyY*D#1RM1}j znPs~{Z-;`Sc5t6e%M`RAT~JsV4Pz54%oLPr3!Y_0;MF%GRZw2NVi_Xk!mHOUYA{KV za_Jw@>L) ziX%&h7SAzt-MChJ{NQvFR6GQxQxi#E2?MYEem101b4 z0#7i&;kz2(6hs5umcWiG-IHw)cK1m}D7qqNJ6lzByE#f@heE3htAZIeWUhG9KysDX zjOEyFB*BM9xfLgSA-YD%Qohz2n>M7gY@+ubp%IX-(YlntY^-w`>yMwAN*btEE2wf2b zuDz^;l3)vX#IvVZbj%^KA%uw^Cuqr% z6_=_|%HQxZUCmb5l|G1%1^FA9!tcoFuI^)bE#qB1JBqA+r%I}t;~~R%OkIOUq4vC; z)9JVq7tXa34L6ViN;(-tgLsUpCh9<@9m~qK+S~Cb(j-N@<53Z!WUk5EcRDYeX{@G>_*)7jR|cCBN-QQ&sgwoR!Kky@@HU|E{IpKS`TWF8nu|&DV&BXI9E8xy_ zJ}WM9#!^7D4mIKVf{mj*=+ayjPa=Z*SeX^?o%D;W!+SuF8_1RV__5owNEH-Bt7stv zJu4BjKi2Sa4c^NhLDHfx@=6WX^vJn-FA_JlQy#IE!=l6m?rX4mrqX7gr{>-3l>E&| z)%824tlEQYT6wX^<N*D;YRtljGd81efy@yO#4pg?45!_>64iw(k(mGoIX(6W(q_<;j#ud<{}e z;+C;8xywgsus6F{bbfD_&voy5RPpHV@#yh(9^)qH87*_tQnbka1b1g1Qb!85xX4(q zO@%q$SRY7q>I9eqp0KT-scia{@{y?&o(l`=yc}tS4++{jk4I1NlbFh;RhF%E;vs_QGO_Qkx75blaJy!=PmwX-24))B} z!T*^WP_69BjIGzC?9c`DV+qBjxqH~I#jxdheAj#F25BKXI~2Ag6TIpz=4y{E0Hmei zphY=7BEpEx<1O~+1pIKF!rn0ZsSO6R9TD32!aG=qf(jy+m$9gK>l@8_F$+Ugr+m^x zj4_XB38}V}I*W@y+sYgzVL-2yp^x)W<=x}rm&YS_U#f>WnJ?-2zGo;`R-Z+ah#M*) zF<4arVu#(xWA9+u+pf0syJ|gnwVOF4I`Cak_y^sDZ8e5*OLb`nrt9M+-_qp6Y|IS_ zbUSf4$!uQmBVN-o;adG@l8#5LRATpiWJY|%?E{I}WOU3UabLP$U0+q3%6_>x6zGS^+GWG{69@>kwF^!9vY~00UFH*0S#|wGYYDDb;hb`Ew7)pLIuUILr0u#&WAqvpmV# zdI}D+fW}?+AhnB(eL<` z%-M$ZoJL3&vp~(;yyJ3sjg&&T||rT--S1O$BZol9Z-|&`?8{t)b?>a)bao zPCC}!m&9}JVB^Z|2b<-hWJ7~ZNokwB9c(@Qt~p;5Vl%s1mZOeHf91`MsEoN6eN)3E zX1o6H3!;VS<5kFj>@4R75~~P?rdAPAt6<@6fLidO_wm~pF)Y)SA{0waBE62rv3*rG zvKM{IzCMAscE?$U+b4w>jB1=)knIv> zMXZ;*6669MPawFbR8&OJih#8z%#L1`8S>&V#e0NB&_AtFFK=a< zLNk(WLto3*?-@zBT6^4`?l0k*WfrqFOIJj_M?y5DuUV`1qB_1-%`y+D*QtHe3DBdn zYSz9x4{n&18Fi6Ntwz(eYU?A9Uw5cZdmp5Ga*Gabx>-p)|Ivw`C_ zlBX@KJa22?@N0)4z1u%c!PpnG@cVOZy`gJ$Hzg-JiA)-3zkRLoCnJ<33VM@YGg3=& zySKpYh(VakkqE&Z=sav>i7{GiqcQrdtfEm^E*_~tNGA$Z+_~bBcEd4Fnny<6hGXnL zv;>@9^-2O69*N0F-)bwvkfbG7h*=g%9@mC{$TNY@e3IG_cw z0;c){xd==TaI~(jCynK#JgX*ku5Q9?Vy9UqX95(&(kmU5LvFTf{lfndW!skeTAI6x zjnY!i=3gGu{_=;Yqebe2S274YP3Xy;2G74vFOaF|ounD%Xg?*{VKpzWixbQns}w=+gGl-z??~dn*-o^z zOZnmagszm2d~@;(r#~H{+wzJHs@tGAU3wV_29L~>ouefl8n+LQQSEOvD6ODPt7YAG z7V6*LNo{5C6zAmc?oRv^z6rv(PnR614=E!RW)Sy_x$zfsx21!Z+@WmoZg1t(dpO@< zX1f}oZf=ln!q*L!?;Q67X(H^qwLQ1c0JhD8denZ-8|#!7-PqgtTw|n#$t!C0sw|QV z2xyv<=-F@5s$CcxGZq`#;(t9#x4rFlW z&X(dM@e8}_PhYQ~i@4b+ZX3RcAdM2;$O%?FIVdrPgzccWyBAP&Q9AqCb6jgTDTn9w z5YJ8Dug5@svnwcl7ffZiUv+8AZPQN|Lhsmu9jY3v5r{k?t>&exfrJ^eS_WEc4|1kD zIlH^loT0xaD3zaqN7l7_LDj1lt#PpumX=e6<(J%W!htjiqqXB&f@NZ1dLYD^`?Lm) zQ)2Y5Qa90A4>yD2TC_=^){S&8|3tKviCDV8U3x1QL984eUG-6W?sAm|54*?gq@jmc zC9+dQa_37*#g#c7Os>?o?4FBabEX0HBq$Lz&&j}oEzNEYBZwv8zZ{ z_Q6S7%d#x4)mC;}XfCs?)E1BnGI}ZhhIxuk7h;E!-9H)n-R@$dkW_?rSgL+qb(CaY zT8A8EGhkVYkE-oX8g72yuk2bp z-LJGBpipEZ-LJH&Pq!tdwaf5bX#!W=qFt`GO$s|y1E6-M1%gGM)5@fG*xGdMAhtc0 z%2#=5AROL%D^XQBOb^;z{jh6z-0DlKROwfaTf`6BACzc#G^q23h8b#zOm#42!I8vo zU=$J_>}a%`v(%4JQfQ-*evwf|mui-9o2Rw8ZErg%z!Lu&(K92wSn;DvWhpZ%qi2R< zA{l0=*@Gc!=c`@qXUD!gd&-<1vOwoyH!Hh(^qNo=Q`$o!NwW?o4>3bBE<1I&UZZ-Z zKN7Cg+<#K32wAZ*gnPSE78fj8Yo`XZ(?5>#vWP@Pi?lE~8;mwS@MXR{qvwBfqB_+P z$x}A%*yh`Kwt{3sRBMq_mL!~-3S-4*Z5q(B7tW)5z~iVlsdC5;O+c;`8F9hI(t{aX zhro|=WU4jj2aMRVaOA?Cvtun$#)1B!?4*>W|cFSFD|#)+R3l7Bxtjj z%tJvP$ZkN|`si`u$het@7S76~g$0I5cFU#k!|>_5Mb-qSO&EDvoCR_GmTD`l$fAK% zfNl`!HO~psuKI2N@;iF$XO^`Pg@-u<(lRw0dOM{pAzNK4uUC|$dWsAAU6z`EWO*v* zd3|YPoQI)o)x=~-^`e?&m}_UhZc&ao5oNNCS|&pZZH8t?4l5|9=|CZpS6-y~r69A% z-!cq@M0)`-wRncfcqxTD_=K3`kk83CdE&EQh1UE<8FX`AA)rh6zC${??#oLNdLd9! zGKo-FiT%49?UIgq)7o0s*{kehSS7ZCR4dd$$#T_}BzTtv-y*6L`YbJDt4))PHp(Iw zC{(PAjZ6BCY0b?cdF&x3h8~%)YEp$$N~({WR~VCs9-|jT+a}6*t!R<)T8YDZDU0Sp zDi|#d*Og0IMU~k0R4c&n?MiFATiB~?QSAGp^L0s}leI#y3K%W!XLgRZr>i`BdKv}IT4=^5%gPLrnBO=wcJlNY)OmX^K?A}Zl!{pEUQy6?p8t0*!< z>#OR49N64aF0Qx5OK%wWrJ*-cI9ma(cnlVdnUt&y0}rcRw`W1?EM{7Ih)c+Ji(V?J z?-ER{C>vnUIKc6Rtby={LS{)-$o7i5aSv}~->dyLL z3X>0=SjrP$$Dx7E^cDGn{MfJpDSNEazzut&ujJgrcm$3io zh*sRV7{S+Vz%OsYUN9xwA=y#7LlZ_^6SCX%tDCSm)P%4i+Rim$k#91!9;M4}!s6^E zL=h5dLRy%HN;8TtZ$dSoN~jSsZ+0=#G7=lhHoIgmiSw}m`Ouz7gt5ND%eAPrS6K$M zfG(*`$6C-x%_`m=n)CD`3@zw)B~F5V){(e!qqYhL%SV$45|DZ;m>e`{{G?fOomeUH z=?=&quo-JyZ%^1%qOuan`k#GYn%7=P-`?>@YqU>Td%P{VwW)etkeE7&v3qidJ8Dab zT~Z^1;#yUJ8bMW{$G(CR*m+tZQI_ncR&F+cM#{usinMYjmmt{KrK1xbZKB#klla}TOG63|O6Mg} zTGeGB^WSYNyI^*4pPbyDJ(ZoFIXOM;k%NYwRWujrBc$%k=7ZI_7AvqzZp@n@C{V5n zlzndr!S>d05WY*5mW4|sL8w4JA!IpQ3?)ZecB5N%TDl4PBUIjE>^@ue1w}i(HR%%> z(VQDczbwsk{-Cacvt#Ed&r?OozLI_jXka!?6ZRJv0X0ntf0>;>(roJJ48}p_-uw-%Bj9Aht+h?gs0x)i)f8q z$aTKrcV3e!yXunZh-_0trbtzkZ4s4br?)S08am?JMoni`x$(KPdtwry!exH=lEQOy zH4|E!cvJ$&)oy4}pkP+kVvI3oB@7f zWOk0}F$U&Er4T!uQqV6SO<;4qiT>}dkosW1@ zINC98XBW%YOYXL<$0QMyW2b4LS_)F36DggY%o_GKS;@Hdl3mzQhl*imlN>sZ&sETA z7*YDk^%@CPOz$eIl|FaK6p_Cy3wd9Xh2TaB+xWqa7>~LUyTiBZjsrPCliBGJ;ecJqnxMBXU+`*;Q4#GQ@4| zK-HwLrM#?MPwF$KaaQ9&^&z@wFQe-bY~S?zQ4&Sl1*0H$Yw}Gws5TrUU@p@PY|5^}K*appE2d}0_;B1u1LpWC$6xg3_2uxn#jVEQR$&4bi{Y}(@N zZf%GBW4D9^(SkdO0iqDWd$}EVuu`uY%e7-95|8pqyL3shkO!q#uW?WCMxEm`5U3dv z5#}PA-R|Vst`zC)>+R!Tp8a$Ks)^GX)OIHKY=Uhs@~#CW`vfKZWUB8{TN*olr~{|J zrT*GwQjXGVd`u?Rpb@dix}s61JO!otr5B?5R{tlTcJ#XsJ6c|#IWJb?@1yp0JFw|( ztX=Z#3tHm2p{i36+|pRnOlCj*S7tzM=`%nVN2YcepulRlTxXMVW)_O%>LrI3-RTdT zBJ{^NZGt_3)|0RdTf@PcW}niE*>OwlFr99vA=QwQ#vLu0yC{gSFT9`xcohYj@7pmd z!ac5zHqxmRVVq7Zuy2{!KKZCk*}Z(;C2qxN2B5~C`X((r2lZkiI!_kNXa_=uKJTr_ z3;>n{L64>RF}NB1V*?leB868LMS2D1rOa!QT~2NHArT&gBBx?}q?CB=LJ2XxQZ)U= zY_KoByRChf3uE|o4uf{nH>xJvi=C#&YsBraQBz0UChW&x@s~zTI^upi>wO4tQ)1-(c484vEufn6_Lc|QrA3g1_A;Z4IW=BygnyV==D_^#x}#AfoniJD;g>)adGyw@NnjK zTs+t9Ktudtpz9v&Q)1o;b8(ax3g$U5YiL(8xd9ia9b}xe?Uu)mxuD%7q0wr&zSS_V zOS7uGz|5s{78fsYJLCc`u8jxfg&8iMM{Wx2@*OUoH-`~}k5(dwo{z?8R9J~$zag#Z zA%W;rR(83>=2c^`NgfUt?<@?sc;;$dhDIsP#kG47f0~QyB`o-@c|lE@2Ls2S=Hhxa z3;r}0&r@I4H$Qyt@?hZj(_B18quJcT#i`DKi=&zs7x#5r;JCPzg2HWX`>ZBZqz0dU zn2~3st|dW#pjX)rV0(lW_@j15r3(+Hdi$|DrUten1sx+AOz$zG+Uea+C$u`>#o_)~ zTGMm9oml7*F4wG0=onVpZ>_7y@S5Ipc~U0RF|_9*J-th}BI^s%V2r2Y=@e{T4Q^|V zHZi7PRs)3D{=trzV3tR-kI1K&Gt^(yJz|%rI@yI|{kN^$vDrO65=x z^p4Ecx{O?4))tjv_>A2ouSI+5rw!I@PA=;L<8{Zj2+jo#0LTH^#2l)@f_z%N~_RRE$|1gaI zFpThwvIE0t#~_ATMzevPP~5Z>k4mDj>oQi7Bv6h|9E_<<6Rv$7rj}{&WLY@eGlEg( zXnxdM+CB)OiCUI%CMk5>Y?kHi!0b>d+!o=b4DrZt!%A4YM9N|b*4*R(b%cF#L<&E2 zFOhbpmi6x8Eq>E}lAOdlde#_Dx}L=^Iwhut7$2!ktnWQHp4(O)<0`^s_aKd^O(Spn zG!pOIFTiS2(dTWsyr?75wJ|ogEuLo+D9I$CQ9ovZEJVT6;E=OU83a_z8sO9s+pdoyLlxtlYJarpGi2CGl9K*ARm4v3vJQ^ z5{1t~lNEd@qouG?>PVY<5#!+2^qA!^9iaLjZGbmE%Dl9Z9{#$v8As5T(eK*X#SmR6b5GE%@Yc=_b8@m}7EV*+jONz5 zdetI%XltPw>)YB=ZFu7~{8BR`*1yhm8#}%{W>#;{&X#SXb5m`mC6%uE_)Hp}jMsNi2dS2LsWkv7If zvL)V>%ESq$)>PjztD`}k`JcxK-^T>4)85h6*fJ|#-(H(+W&8#CFm?4!jnpdlZ)}g# zd*jJ?I}w}0anj?dwkf5tz(5NB9NdD!+tlkq$xmu%u6MQI+PpT^+?s4-d|A$9t0zg@ zHq^i~IvQJ>7HAw0HO;2NQ5su1?NIuX-&>o$NVP`As# zLz9X39)UEl4%bNSVov$&@((OiP(+qT)%J@hP)U%2tUhQ+thV3|OnJ*G9$) zahjWPG+(Ia9Uj+zaa?$NJDoiAOLE2zKkSfh`mz&}Of_b5eO>*`WMdt@$#l{y<428& z&%+IO){fL10wyQhVE8TO=+4%r6m`SqrTTWbrEa%n?4UfR z1x#y5&VyH(sZDh?)ylX(Ken@F7FXjN@0>&%EY$fsYm*M9~B=J-*?}5i+XRfYSRT) zDlp?86So1>pLjLp>^QO$?mOz>nvSD}vz_8|e!t0un&19x2xd1eqsk0jRKdrH@POj#elXWuQhBliMnhhCo8L;i$9O^=h zTa{%NPDi|KOuUAn-PB}egwB(p^OKX!jZLOiBVt}-TSsTIiFRyBs+MaKGg57hvl?j? zb4HVy(qv9(qPx|1)Hl{lPA;gQVHPR?m-G}9I|r8*jGrX*+KnZjgFPSn+yfaj4# zdpfhF(M*O@(T>z#LvprG#m_@(n4xB&eW!w@ePggC*gDo zPPV1m8+1Ccv#q&v4kwdSbxILFl5DP-nBwOA)-s#s@xIgR&ub)bQbT85az6ECXre}Q zDmcB-R4eTT$%dK}2yaAmOwMYcjcQYM^+Rmmm_w)UNHlfC&p$uDv+&8uj@kxlw!J-B zb851|w71oc3(NshpsBC)qU3PEGK+wj?92xs{a6N%d6itaLxC zX{mP}&&qcuTg)_-qedTa;6Vo;a_C{n8MUw}Q$5p+pP6RJp?O}?DSgX}KI1##FZ1e6 zM_Xrodcw5xm6=jMzeWS%1bzuPsxrfjZDuB$+Uu#brbZ?{PCDtkIypW`f6>V_rVDp= z1|4Wd^1NhQ&9u3l%zviV&iQ-|)w)Ems-T+`lBb7osA)p7QSoplLym9LC5Zb3)=NfHIj)b=F* z$2U-)bL#7TTW99cPM97&F`Y{0rWbo_!g`iu`du7lp|em=RgZ4RdsQ(ReV?5svIO$(Znc#mULGU+yH zwzC1uWOe#s8mT35YE5;mnM~@n+w%^))M{k8G@{n*GlQPeoNQT;G*jCe`Omxov#L2M z)c}jY4|8GSo}bjH!`Il>(AkkhT)#ilyRtAP7ine5(E^lGNNiv%!XSFkV7+2+*q>`pmwQSd}=FD@< znTMZaJKN!BbDB9j=iEBxsSX<&Iz1;7nmIeCm7e3GX3oxG*;C(Uug{#F)7eTT+rZ4( zImsqio(;^Lom1P=(bi-GGiT?_XiB!sv4NShbCT_~jg9W|^w~L7Vrzp9%$%Lm-kPkf zw}F|nbJQ(uSo(wvwKgPOiow}AXN@@9G&iawE-`JwcpxS=UoAW3loL+yF%Q^($W9U(IeO^!k z5<1K%ifVP&>prgW_4K=PWgy%4U# z>3l1yI~mT2?F6>_Kn>a8l%}dUD)5BGKH^)iX-=No+E!m1NQ6>b1b)>pnQoe)eDGg) zwslZx8ude^sYRK!;LOUiRU9kPx`0vH0i#_2U#G7y2wlU>i2Y5Ov@`TFlsmAZA#T_X z4T4$VFrn4K8?t{v&(Kz1xBX8;kVLFxG{nmyC!qvLb__L*?X4P{qldJm=C==J?rdsT zUPCIa(Zl2Jw;j@e|Fn&l0w$D<5(s%)h8i!f4{7E!I~QIe;+Lk>P<4+X4ULqr0eQ!2 z_lC$PxcMIB`Tpa#o zhfJlVg`v`R(Tt5w#aTaOMkb^Kv9N7OGTo*@Qxh`ZjtJGrs8fzSa@4^?P3I0FZ99a} zv3CedSMBJ74x$uPu&rq4s||ZtSJ7#Qj3yR6d}iahP6L?Y-aLSy|JzRQ(!{l|xCaBgO3Rk_tyd{yA#M%%)_=2%jubsR>p64$-ujLNUB!%uJ@ZxV>Xe`uf^A z>DZ>Mu@zp!T6PE{q|UJf7+fnV%bJR#RHs84>!{UqBwGT@HIuF_bRlyJEXg(vULPeK z6wFL?nT%?fnDre)%-&9l7r!bG^jLPJm0QIp+8(ReJ3JnE=O^1}?O6n5RvTu!BNOgd zA=~|GLf0k6jX!$AF+R3M!dfGQmKf|RrtD$lBoTlcM zRO{Tf_KwbZ^UqyyUgn;|M*go&ucAijy;&SdqcksKKIoP4A8jJr-7??wA|!4uGxcVv z>DqncZtQ@GN6e7;Q2vjM*Dwv3N>dqJmEh_#dzCUMUL8}$+^@7|9q{2{6k=dQN762XAC8HyOF}tLp%m)QH%;#|wVM2fJ zmpd9ipVCfWr!)@3e{5FR@%ZOwg`JCE&~b%Na{>Mfv*InpFZ_Zph$*hzF8sn13gD1? z5q?3(&z)l~#($X&%arYM{DO`bonx-Re`Qv@tMFg#u^IiPH#ojM>34TFG(1(}ezsrVY z?z;@XpyQX$G0X9<$cncTzn~+C_kH|#XT|#gejyz19{g*v;;qGhuaEbpi*D}2|3Fr} z2k{F!-g1ul5&nm=;yr@@(X6m__=Q9Lb#kA?FX+%v*XdLI6Bq{F{wMP^CxSklr?~p< z8T^8d;M!;L3;NyvWPZwt5We;~{6hHJ=kW{SYk!7c2w(dG{$GJGk9Zruu$GJUi`-x1 ze?KeU2lxdYL7qRvFNE*=5q=?j-@oJk7=-Wp5B!3TCH~%@;1_f}>%%_9U#c61_*xvl zpyNUxuN?mYS@90XFX$MyzjKeke`Hp?G57@?XZ!Sy!Y}9;;;$WxzY2^E{J07H<7~Xl zSQw9A&@slRcQk%M#{}n?3HSva6P;s@!9Uf8Wzwt0FO>TG$~_UkFvy3=Jqf?Crw@}m z4ZkqL=R@wv_=V9vOzx@pPXh<|FuAAW7Y_Dea?ivs9OT2~o`qkSdY3DN-0Ap*aC%An z!XzJ0ZY_Rcluu7?J$_+`k0*B~e&J{zCbt25XllwS+VUrJ&`vm^yv*P^>zp&oNllubx7r|pbOzung1sy$|V>aLy z9{2I&zKmaZ#)rv$6~FMZ50m>Ee&IPECiivx!lyn=?i={u2jRBdj9*yg@Q?IiaUWKRe=6AM)0bO~zXpWMJ01T;AY4~h<6oB*wjRHr zqqlR+WB6apiuV$J;RAnPxf}40%+*DiIv9ojR2!xxjXRR~F9#d^wQ`r1Ain?|(+4^C z8T^8dAnXPFLO5&#{?EbsLKm+8Tk&^ig)Qw%e*ro+eA~I3@qg^Yg1-AH{;eR~*Sg4) zaN{4GtRnX!{K8E>Ozy?_h1-0X+)MCZ3cl;Za%GUa82^ufj;g=AJU)tlT~^qF!K~Rp&ebli+zapvzb|%Sa<9iP9O~m)cTd&} zAl$yU;uoIsZ7KJ=_yrw_-?+Ri#oukiGJV68GX6kN?{a(M?+tVW^W65iC+kZosC}zht=R0!eP_#&jR6lG~iEV#cRbs7aZ>{w{9E$PH?IZlRFRp zd{F1ZjQ`tN@%|hC zlB{@3@h{Jcw*vn?S@G83Uz-*0KKw$yyR~(Hg#RI+Bgo5-@jvXt9`Jqt5&VLVPkh)< z@C$Ff+uf~jxliC1!uNO<|MNgcFn9e7zn~*XZv+09LAb8o!2hO?7v%4k_}}tjxB0wm z#J>rIul*fFZ|ZuQ|@&9 zDIbq1!yc{p7lO?`p4?0DUuxrJc-RAbGYNu@{}WkZPvRGJ1Y_ zFX(vIx5u>A%;#YG7p{)ww&Gt0DjQsw+%iVu;dV5lC$cHPqwt>r!ZNzaiJ1L>jv+qY z5d2fY8=fD?U4#FXtaxwW-;x!tt0-db1v;wif8=5OAA#I$oC&@e z1cuz~-!I@dJH&C%4#nI4bh9qw*t^O)=5*&)EpqOaPdIn>r`GMdbVR~%Z0J*K-AiBp zwsq@Ycp9V9_Lm<8s5W-u2l1ebuTz%}4zprz35;7*`! ze>b=f$df!&d{K~$Ro##b_?JVCb2@mJ}Vf@+of0?l0P(AMob_2Tu z)p>sq90NGfoIVgpeo%YZB@Ax(2>ShAK9H&39%?ro`#Q($=iOkut4+g)#(>&beY*mT z0BY~x(C^~kqrhmO-<6+Aa~Seg0b{F|i9_-yuo%`rNvwo_#1^SGD>=xyoZ91xT_T2| z*_|DJNz}#LmH4qBGVbOQ7e71vT2fc~apH%^^3k>-spl_* z6t9fcf_{(tRYMuIayVS>OfwyRaUoZ@aw{DjLE1t2o&Olpm}!6CoqQ?3PiB>`(C0Th ze7TRmf%qD?!Eu|9pB;XRkN*bob)Vok-p9`luk!JKm6iTMK7MxiFdzT7S@GjOes=id zy5tg!tk1>hSr`4v+Zw{fQrLuYdUTv%_!l@#9(fxyq-X9sYfv|4QPA z$J0`ues*}URtfseeX)pn1iZ;yK8&$>0Pcg_>%@K$b0&C<^CZ8YSQs&v;oc4&Wo)m{ z&6GyG<1lbI2zZC)kl;9y6U`?_fw7u5e78bs6qIfWkk;-R94`aRtpe>IfqNe`YlQ4)N!~-UKe~KAy1oKKxu< z;R?coJqU`kkT4;8&w=(C2-;_GG45r6KfABsa!y|4{0dy{DNs6B;a&}1Ap9EKYkm0j zxPs>XMYzIqoG-=|p6C1q+#7+?xd~TLI>DZRmpH$fFd@nLw{Qi`_20%7G;jVlu5AAN zSv!AV>kn-FiZ8pq@G9qFoBk5Q1hv^x+`9mO)^5LylV5Va99K4b<#Q#jpz!bG-VOM( zcKILp@OyCA0L5R6doOsCIQQW`;KLup72YEJN4T=N^JfBEyUy3~PYB!A75)_e1f?%K zy0DG&XK)3D2exy(+u6*YBS292^ZXMO|7W;@(s=<_HgScAZQ+3poS&@S`vV{6LtH`O zAMsC6I)RN_;s4;Du!QhWa0S`3KgE?T8^_wOAsu?w$Xh!zm_zFD$WC2=vdBgAWSyUIOno`3Z5Kz=sO}ZmLB*G}kmwDn2@mX^an5D$90gQwqj7~PEZPph z6%>9Tt}vPKgK)?A@S|{rqX{32Tjj$OxZ?!E$KxL7!;i-mjwE~%ZeV8(+gGO(J_9KK zwYb9G#IM7Zy;b3XomJ=S2@}-TfnAj>cZcn%fgM%x1N*7^PuOl6*h{C=KLb0d&I9|X z&ck-must-egUSXP*gw^0!gkLtt{1ZHoMHQ>?3#k?nX+#R%5&JZ8MbMLZJA*kW?;LN z-7>IQ>O5?#4BIHTvrRI%rvu$Li7On=`3zjyDit2KQ3ke2*(I0qyKIt*BL>q0s2+Oa z_6D+jl;8%o$NroT_Th2d{ea>R!5!(tD{&9;;fLbNrl@qPa3}ciV{v6eRQQRwg3_6W zE9g9IE0lduQ26P%g04LiSJ3&{xPs0D8=%go6E+Lz+6LT|kJE}f7aUJ`*v1#O?VZc- z3qTif!nVBg3A+GXNO%|S6+r1FyqeD}0CZ(YOLbY&#p)biz`ggz#3}z-Dy`=a&MK?+$EKdk|+k z+Z2i>*`&l+U`QSGFI82R0v_2euxa2R0s^2euvMA+YJlo)g$|6eqCZC{AFzk=-V+*(grfRukB0 z6du@SWJd{XGP1`6wiwxA0vn9%FM;hv@dKNS>@9(ssJ7QJ_%HV)4@5Q z2BeRE5wiyv42FS9Fcz4jBj%W6aPxBb2w-7eue?3-20?p5@w~BliM-?UPR&c@&CUB& z-mmkz^QwC+?6I`x4|_h)^O>Hn^xV|*-Jab&tG7J1<&`aCwpMRlwDrcV&urbib?esF zZ5`X@ZTn(daR2GW5py9he?#8{azQC51La_EkNt3kpwVXzr|0D@yw?y0$_=U$zAUGAdXzFYcjF@+Ix5I78s1!mx+fz<=g zAGmbcC(BGj#2h+1bFI?Yfd8*xGx!*+ANbM0F9vqYOkj_${7dpL%fB{%QU2omoAOuW zugYJOe=qbv{zLhXG_+U#r=!>7sh+X_m7W=PmkBdXU0=;jFO*rZ|mOn zS@)LiM@d_F!K3m(@(7kbLj8mGmm}sp5HZCTaomMKQt#GoNxmhx%vifF+(e*Ty89ppKEF(?0i{(UODFjm09U?MQ94qjEYYVxY;Rj02y zYgNsvrF}o{`)S{;eOp(rTfL@t#5@mJn0%VaLsmVo5_PtWm1ebh$UKbI>UHQ7=%MI4 z(ceY?6#YxIJK7QJh+PO>23-N=(so7AK&TX|g4B*`!@B(X{ISJji(l>iM(-^pTT0&B z{ja+p)bHSad?mSmF7(QvR|ai@{xs+>gZ?(?qd^}J`ee}O5HFaI7uz;(jd#Q^jCaMa zk1vg{i<`22%JwT8S~k3FRM`P#lgp--)s$UTc1_vhvVHfi+WYvur|;dppBdUdY~HY& zhkbikH;*HqG-C6}EhAa5RpvoGE8~@wm4{RwSvj`y*2>!|mqJ@Azo=Y1YRRbequw1A zA6+wg;pmq}cSGMep!sSTz>WP$`yyL7_oB1%9o%sSD(3h_UbvSkJ>tR zYst0&+bXx6wC&_=Gq*Ktn?vvH-WKVOcOTJR)qQ4neRnJL-R>pb8@qqs{m1UVb$`~q zwcEBq1iJv}0S*9%fU#gPxD9LoUw}>Y$t~bFsfc+O1pS)`=3`)9j+lMHP%snB2I}{h zfJ?!(U?ung_#@c(^N9Hfm{%g^R4^T!3(POznc!4#8~83*0{(}`UAW7@O7MN~1Mm=d z*yB;$C%{u6_f^UT3P3NgE7%nJrhg1>;x;0qw-W;^HtZ-9;9Jz!9F7J?Em5mbZI!E|s1SPbq4 zYrtAyKBFDLdEj~Q0+{wW@j)|a1s&jAa31IamxIM%1y}H?j9)UBf>b{@L*E;qcjaOgv(4*^BG7FRGim*!lNGcvbp(3OLR4IVc5TxerOcSY5*CCf%^8@Ww;QMB(QedJf? zpORmjKQn(}{>ByEE6jk1$piTwMYw&zZeVxN9~=Y@0f&Mkz%ig2oDNKX!a(|9j+hNv z!TF#ITnQF~Tfl8#2}mE65i{aI`WetXW0)8A0~_J1veB77$M!kC&*VPUeNOH34d{$M zXZM-j=gK};_qndm;y!Eo+}r2=K0oU7lRoRA$NN0h=h;55^?9RDN#8zwckA1~?-_m1 z?t6aUuD-YQ?W*`z#nOsp6)P%!P_d@sfr=kh{G?)i#nTnfR=ilTq2jfQH!6Nz5m}wL zI)8QX>RndvzB;~o-_=7_k6695<8&E z!$%CS9L|ovjnpUT3HpP9pbYE<_6GZc{Xu0cVvYx}qI1(HUCW2$ZBrq9Ndz^+VoDFKhOwa^cK?ewq)>wP2Gd4eVUhIO{Lg-THa_Gv~ z)v;@#>tlDs?u>1UZH{%v_8q+c;IjuO2hSWld+<8Ql+qbNKb^Yof zFLJBk&DVpa9_HuBM<6&ZD!8QJvV!Xh78Tr3a8tpGf>i};3hphqui*ZIhYB7kc&gx; zf}a*VU+_Y~254i!y9K{5_@Lmkg6@Lg`sst=J75-Q0JFjMU@^D>+ywp`ECnk;q##z1 zTToEYyP%|?Z^4j){R=7!Mi(4Va8SX)1&0 zo0*F~0D2no?PXCEDp7ObgHclj2LC8(;^2-Kqh<}*_#VH1O8DovCT5QB%i?aA?p;Py zj;tIt>iSV;^q!-qkG^p9^p)qYG~d7T^);_Q{rdB-Z+U%b_fNVX?|!EHx$c*`f8PCi z_bpRhpcms0RD#1myx)N#*%DC)p7SC2JNOuE0o@=tZpeKmcSG)G{8r`dl}n_WFp)LPU2EyL;3Qy;V44^FQAOIP zfYU&Y?h{MJF3Y(*XHwoNd9Bci`6uVE*<<}48wUM+&@TpU9Q2z(e;o8z={gFYLy ze(-05w+!wcTomsW-w@v%-@me=@|Mc)RGMwQx5c+rY}-md6E-3X2uCnBgjp{~Oe6Rv zxENdpt^upS-M~bOd6u=gu9YvW+_3V^mA_v3&Po-jtHjy# z1NpkE=-Q&|ixwB%ShS*ORndJ#_ZK}>^hnW@MNbz!Tl8Gf3q`LMy;k&VXk*d4MZYh4 zzvzRa4~v4jk$Xqw=apM4yDK-3>K?UW^tRFFfYT0`*`4a1+kLxhQ|ecS)Vj5l37C}? zD@Q`k;2>})I2?=x$AW6`4e(p~svyJltJhNAesA~N*zZsM zHuw9i-^Kym15BTY*#-0kM}alq&tNn737EPM*d6mZ%`|8^1Kkg+L2d%9Q!18QO?!OiEb@;dhYbxD}A5*kguH{pVwm7;jiSq z3jHqc_j!NIdq3~a1eiwkcoyt(j}!tWH`27R}1N#UJ^-z!{ExT^36g%1^OfO3n9 zii(R$iux1{DcZm2u%a>0H;T?EnqHJF>VhuYNuPeS=uzK~H`zX%)2pagaj%kIeR>_< zYYa36I=R=Wy=rnXDab@x7;sc7SijOJ2 zsQBXIOQFk)uP(l}_|D?HpykDDitjDHulP}DeesjUPeZ>dey5nFY457u6MG-myQ%k; zy|3+UO7<@q3ssk#QgTMg*(FJ+zGP-eOG#@H*CIS_gCt=o+wkz?uO+9bop@eUJDaC-1Qw z${l#XzzOI)E*ZFV;134gGw{a)e>cz!+H26{K~o2vG3bIpk{bhgCbM4rhg&adj^Fd^ z&hd|w{sem5&h?oW;eQ+cA+mK?4y&w~Rgr?e zj=VQ=#K;l3tY1e}9>i>XWMwcfiKl)6Pn8}<^Qd$&;;-rXdg-;_+wi@u-!pd=Eh|}8 zx@`Yt!u%c*1 z`HHF)?JF)`ao>s$SA4Q!3$%5`wiN{{_gPuF@|cyHt1n%7uU!vsT>0CTzhC+O%FkDB zTWMAeK@Tx*)zPaat(v}S-m3GU3s>E!HV?w^qHqYU8RuuljJ+ z-&gHSrmb9k#_F?HH?D47{hQTcxgafvY{}aq-5C?-meMUlw(P&td300UoZsl-2-zZ7z4(F$)FmX4Z6VRa+#ZN151I`Cm}nRf(lRt z#)0ENHRuFgKzrquf+xVU;3c5-(Cr}5GY2~6VBHbur%WC?pdO?RP65-wEbyCXu|-1#SS^h<7Xq^y`6cU2@=FpsN@;7_0$nf$5FD0qo~di8}#E7xp9w)@?E5 zoI}7R;2y96{1L=TPhc12$ZCW7NZ4M>76a1VGAJO!Td=-LfE z3U~v&1j9z;L=T#nUxQl>cEtLm)f_zX2D#0-x zJ6o4s#cx`R2^;rh+<-Nu4uKvetv9K3dW5tcrU99CHu(VOfIzQr=1||D3XBJ}U_Q72 zECN+6$nn6WXire>aTD$mun}wm?|PWli239S;$P|HUvnL01d_SK>zY9RZe7co>0W++ zBAQeFJFE|NM7n*?qrm@)iaS(4Dv`IQ~Lu;TPLJvR>MSdK4G_o%8 zH1t#GXV8n#2I%L|FQC_;H=tiazlyvac?a4Ay$k&g`UCVQ=soDK(EE|ik&mE{q0gW# z&{n7$ibNyP7?cb3fbyY2s29{5+6C$h?FRLO20#O$!B8nw2JHpy1MLS5g@!>D&`4+$ zbU^g5=#kLaXjOC^G!Z%ungmUTrbcgy-W_p^KnvpzEMT&<)T{(9O`dphg7&|T1S zXeG27x*NI&S_|C=-48toJp?@rJqoRZ9*3TUo`#-F5FFLxoT;s5i6= z)EC+f>IV&g210|OQm72t3)%o^?>rBLZ}zi8`=fx3+)E=gJ6k!4ul3nrBE5P7qkzwA2bvi230^Kp;6EQ&_U24 z&|%OK&=_belz_%V6QGIEanK}aGBg!B5t;^_0-Xk(4xI^|4b?zNs1~Y+WveFE`lzBE`zRsu7a+Cu7egqH$XQ*H}84-o=c!RpgWxvZikjYcR+VScR|acmC$PFZs;CpEp#7rKlC8<5cDwg zD6|fG40-~33VH_mDfB$_0`wB}GV}`cD)c(^CiE8cHuMg(33?aZ(O;myK_6x5j}prg z%f>?!po!3N&?IOwG!;4#ng*Q$od%r_oe7-{)j&z87OID4L5!BN77`*rkmzuw#97uZ+9 z6cFs!n8o?U;8ySe__BQ&FY()-LF`qx591JA?X#EzE(c4%kHIV8Z=eTjrE+itI2EM8 z)j<0)!t1I{_`~a}fvl6l>#9@mhu2k$@Q2q`KgF-Ll@RQ8Toc{VrlP$^=Rv2lb58q= zWD^N&9$|aOTr3*LpnI2I{R=GN(o2u;Kfb^8%;%y@9*$jD`wwnKZyc8Q&mVrlaOq^7 zzBSkDTaUm7_+9j@+bVZky4%uE(We$ID?*35aM^9>QKds|g|37?UH;i}?GO4r)Vit_ z?dIF7K0>#7*6JEo2k0^VULdu1u&)2I^?IwomW zX#4qUyU+a4-au#TIjHQ=GTBbPj@?9i{*Ltak|o&jWiRReN_)w!OLw<6lTWaj6sBz^ zE!a$CFIlvF(eh8AE&qm{L^hH??9@h*S~d5d+DU%p?Ia)W)K0SgOY9^6u3aRs%YWtm z3v3<#C+#EE-ahjGx;^AS_K)rNoBiKn|M-vH!|fmackLe8_6~y`BW&mR-)7H1pD+FX zm+JNZhWrfvUw@wE-__$w_SNxJibN$HBxdrBXHeg=I3x!6vh z@Ag~$S`WiX1schL@}Qnj0aOGPLnTljXjf==s6Vs^GzdaP6^TRT(B9C#&=6>UXgD+i zs)R;E2SNu!he1a`W1z860vZoZfF?r6LnlB}plawO=w#?r=o`=(|6hAw0vA=){y(dL zfJ|kgU^#>f7l{zf3>7h#v|LDSUxon&m>OornZc#E43`#dv$B1o-crOCGczkwY_GS) z%C~(XE3>k+GFyfJ_c_bVy?16%-=yC6_iw(Rhs!zl+_RtOJm=i!ITtt!I0q;L%7IFt z3aA0j1w4Qk@B;xL1k3xTfscR|U<>dWuoc(_d zfS-V0%)grVn2)lQ0p&m?PzBTg=K>zU3;2Nm5CUcZvw-t~3xGL5J+KJ40=NoT3|tE= z0d4?p0&W3r1MUDCfV+WvfTchq&;;BEJOHc&RspMlHNaZnabO*=9%!~~v^)(w3p@|J z2>c6p1=s|<0c-}|0^R}M13myg0$PAAz-Pc#U>ooiupRgo*a>_O>;iTJKLfu4zXPoR z4j`mg8CDAbb0TXZ0A*%t3eW|x0S5zJfo=d94`l#70orV~14jV8fxbWvkPGAi`2gux zs`cbZzfJ0r;b0ExW`|nj~3IM1#A3g{e3=9E^fMGx}Py!qWj0VO61I_@>0?q-- zfO4P`r~+z$a{&+F1^hq&2mv#IS-|Ot%ks+d%JVAossT4}9#EU-&kF!`z?{7Lym`O^;1b|6U=eTya22o^ zxE5Ff+yLAJ+ydMN+yOKIcLVnTOMyn93Ahh<09XmE0#*ZSfVIHmz&c<(&wxvZQ@}IJo?rGN@FPHa3sTl%SFEqe z?p^91)&h~AY zPweDib|G)U&i><0pLP18)3#1ub^4~$K?w)J^yY+wr3ot&zE1cdVGqzPu}5NNVs>Jm z#D0lN;*p7`C!Q&6@y$s^NyCzglO`vflyquRIqc=;!tQ$=us&?#{R*@3-U9pXR)Fk; zV%Y-uJ5TRi2aw&;I$*uZMrjM|lYWJ5k}=twJPasK9-VxA^0?#^lP4yhoP26>d2&Va zJYaqDQ_0UIzmWV=@+--21BMh!N^#2Ql(8w}Qcg&jm~wK;DJkVC6)AI5=EHVmUCR2D zXEgRBTVOk4NHwPpO&yk6oH{ynZ0fkw6H+Ipo(hzwR;12NouB#=@LKBT)VF|lQa?`p zKJ|yx0bK@k8Puhy%kf<%b$JL_-DO=D(g*6E)+6m`;27Xopg3(r+Vr#qX_us3ns#N{ zjcGThtxJ0j=x-~wonkxPc8;ygR&J}bRoiN9(`_N!JlmDFtASf=x7*G>#Bs>M>0Q$c zVP`Z2cI<)lVEXFxH(^EcVR~}+)b1B|zoh$eKuUGde5?JK@nK`(;2Z4zuZ@zdKU&43Q``5-LIqU?6k;HS3JAzPi#L0dt+q038l zT`Ri4&KNW{4Z3+C19Wmtpdp~KAUa!M9%v=#agYJJwkA+N5ZT3#0gVL(1=Zs^5A-mo z8T2CPWzegj*FbsDwLJzzwm(E4gUJ5J0Nq)#dn8*cvSlm=#jSaeW%};GsstBcQdQX3z(q4?zXpz+*uxK#zjtIUaf7y+xn`=)mgjU5nvYBIr0= zM}b}ky$yN~)B^fU5SGJ+&Y*#yV?jef#h~K_orLR|phcjYK}SJ1R-OYf0{>qNS_WDU zS_2vY-PggOB0~#*c8oW|?+572?gr6$7P~+N&~Y6Mnv8uPqLViI3J3yPD6Z#ftH0F#GK+-Oc^vOw=ob-l&28DlNyCca5C&TWj zBDD%OMxd6~^SX3RImol7vQ%*UM4469eNdwsdH33Ev?Py+C| zWCCWAQ!$5ZNZuHpM_y+0h%v>SG7|I1Jj@&Gfo9Ab&ojHje=)m56XuNK)DkgYOknfH zJj@lZrM@1XBZ&831oXG%*#-bZfT6Zw;QObBdH#Ic0^6mwD+JHK&9)|eP5S!uXVafg ze<3})N8cU_)88D>V_=UVJ$hthfWM!daVp>hDl#fFYBK^E7V!9^fuX?g%#oR+!0+cV zUVm5S(#&OhKP5g&M0_+KR0tXfI$BT* zu3JDR@JHgO#79SfMuPT*pA!E(22=tX2^u4a_~&K18z`jF6be_rsl2FSc*53~;S8t4sB z8>k+1CrIw!6EBy1oa_|BKPE3<+QIU&3Gy;Y%)^pKCXGv)2t?%OxlC@}2)X$cAR;fD zn7lkvXFDH}m*+8gc_WjTUw}pbE1e_qvXRNlBa%lXkA%!Tk;%)agyrSt*T2_)nzAsxq)j~_{YH2ul+=YSVu%F0Ld z=+`5sM4N0!wArE zpwS?b&DHYx!*H(!JuYYiuFasQKqR-X!aPE9`#R8*pba3k{5}x#MiFSZpb@xAd7fnY z`vtAU^$`%s_)^aAg*oO3P#;h~5S`z56y}*@K*xee{-^V8hJi@tCn8y&XfsI4`|sm_ zlKF|$^8OIaEwSYNQ{aC(NXq>;;QyOIH-ic=$4L2qIR2+KfO;L!1oukND$v8AM?m|t zE|BM%T#tE+h}HyhU9hX2b%B08AlCuaueX0L#(#_aAD;iv{-Z&qpwmGHytK3cop27G zt$UB|+@`*%dl~L|K+;~Y3~MDf$P4m;AagclR4l6*KTa7WUvq+gP%I{R>T&5q7h$-d-3^33F| z$=@eerT9{2rOaXHfSi_k4%RHQQn#k=NG3m+=n zP-tJOEZw?v$I^nvqZ>ChzT0S6p2W^T*}43e<#di7$vd{T9&BG!p3_!h#2G4}MWD+; zpJ4xw&S$e@uOk~YQV^Y)MD#l7-=GHUr7j1ZmdMYCnT_j3pk<&3LGpZ>ZjhBSK_o{# z32FwZ<*Aa+{9Ka@ah(IY9dsv%)_prb@;n!L&dc&-#1Hxvd$(;Md0tK#_EzY8&co6m z(}GR{O#?j(qO$_!Iga$brC&gF&fo~pCeXW}Pe8js@_dw1@Wj(Wn?Ub^WIt|3Uwsi& z03J3J)B@i1JxHF*CC`(T>&pg=dpggk8TPZAL5AM^{4v?jOR*<#I_ON06I2iR^aheq z-{9xkmA#4f0m<{?N6%Q2*Xrzn1xv{ZsajDzrJB;~B60_ov;D7480|uH8?^9HDP_eLKf%N7;Vi z_7G=^5a0Ny$HzSk=!0j1&IUO_%{UilPeNOQEHCk#GR8YD%Tn_S;s+Fl#(4-7Z#>I> zDdSV~O5%&e3wMGS5+5WU=m!tX1>YlnM`sSbhkR=J7@cJq&Bx;LF5+1-ZMsVSREIpb z$DjUc{1G&=jq_hro~;CCR&f8~7x<^-BA=vYYQj$mza$hS z7PCE1I#bpU>;Y!r+`65>%#@ia3xHjzyHY22ncU@!E@yVJq#c~r6(~v@o^~;?DQ$Dw zyJ_#GonV`clW2dll^s%Z2tK^n)!WV6?N+p%JeziN`djJmq`#ML=-#>e=pJKw(7DA8 z8A~%7Gw#ngBeRU1u{^Zrh@NxM*3CVi#|gG?_O#ji*$eEW>|^XF*{9miv0sQYYp=wa zwTaou*-Cb9c1^#!e$V!MuHV*vJNnI6Zc^S+-dBEC_9(yP{hrr~^EUlQ`j0f^=jRvX zPsWM0Q=oum##yxo6?84=h4X3)3WfvxtlHlST4B$fx-@NR5SRmekMp5JjWZixZG5fq z=f+jJU2QI_@kzen|3yR*Yx#@uXX3vI$JA%8sJ=h-6=)6EQbT2Au(t^99Wr&k5Ryb=}*bk&t0Vfu#IB41B4W zcoX;%(P+%|bl&I|K_jqc8VkAwbQ|an(A}Vs;9X-tv_C%`^pn{4Z-cBuB+pru@;sgG zO>_)oouQ!7f^LM&Lwj>{E-(?DRZ4W}dyq>(QRl(qxiKKU{G`{jEkm1;TpB(vMKB2Z zj^zk@iG3O{uys)D#Ma5J)F&kWI1(Q@SR}?ajjK*rx!CN7+SBvX^D}^={Gs_JK+l4n ztWQ&a9?)9YdKy6eFSRYTO}073sQ_mK#8-x<4aF!Po^~6?=~(+%jIKEtU02zYvXd~v zk}<+^v)%pN{muhw0Ut0O2m*D$bHEqCm%!J+H^2^HfwDlk1h@=X1Y7}J1uO=x1(pCe z05<{e0`CKRfb9J2{60WGKmn2pk_vEST|p|41{?$&3Zw%#@Sz|RI2>cScu(=3?ycQh z2LnTZB48L$43q%J0i%I&z!|_k_weCvePQg&APsjZGyr;g_*fa??APigC-;{C(g zhP63?O5*wIUkCI5zxnzfp3BN%6#Qw;8I{Aa5-=#a5ZoZa2;?x za3k;z@E-6xa76wQ`MrU@Kn`#mkO;{KM;8~Q060do0Iwh*2X!sz26P89fWv?hID0?6 zHNEv{;27XoU??yg7y*m~MghkIrvqmKX9M_r3C`vp%48V_B$;x6x^;arKW9$LA9CNM z0PkH%mZI}7zoqjp#jm%aH@=hIALjuU8_ogBfD3>*z}<$W25cr7?gJh$JY-l6tOexv z6rM6YYkGryQ{ijVH^2^jSD_B4TQ9&j6?OyjEb}b%U64+l67hwWnmO$d z;Lx=6w2ZV~X@h{lzz|@4+OuiTr@fr^YT9dQ^o@o)uot)(XOP!+tL^rF`upj0PSHm= z8`qh6Ze|GIB6+gclf5?d+Su!rUa#SN?2mhWgEO-C;C$@S_R;n!!1?wI?2GJI*c;fl z8jM-ytWH^pSxH%4fo@sdalA)%Rv(}rpa3*M6=e<0D#;q1bprdI!{n?*?0XIg*(upM z+5NNmcO7o){bg@MpXz=~`aRR{`F@}G+u3iPaQl<(1~2l&?O^ zn!f*F!S^2yE;s~Vf3Oz}#`hnJ#rGc;54dK)4dUAmbOv>9;qir~KpDOQv8ZrmVRIq> z2E^ifp1a4etlKg>z6G%m--5_#?B6&T--DRhG`mTD3j$gBZ56(Y^EIUdwmSmu3+u?1+KnZXba1J293Gq9=2_e7#FdKYzA?OMaoi8rG2ho84iR5=7Wo%{Elh}7?}{s3(S{r~iR z2m|y>X{}w4cbBiyp16wN-EFwna5eZ6@g&KAeiQuXm*iiPiQf?4pz%MXaY*Cudw$;| z$3RQlCvBhO8z?fq579?kK(hU0yVW1W*J)(mdl5+f>ahlsdUB*6NBxA>IZ^sq;sMeX zjdeD&UVoI{OMH)g1G(QM(~~^C5uE%%K=SbRcy~PhNZ*=K^AE{0h+jya@M+tpZNL41 z`T)^S&Mk6Ip}FKCK+YjFPtg3J=e^y*XNjMl2rSN8l66bgtyvAgKY^vd1Hh|UuVuXf zYzB5^wH;y4&dR<3xHbFX><_X($$qrYqkT34Klf?t;%%Ivi^go?H^>e2oOFti`$DI#Pz-%}{H6LD|b#vBjS$AgLm34R4Jz4i=Ez7z; z>%pv7vNmPCp7kd1@2nrNJfk_YSN0KX-uy@Qh1s`cug-oydy8fcUDIc6pXNSK_4%of zp|7{Ezc0GH_m4vdE9QW8ybN6ZuQ*^_RjVkyJ9~IZ~#zVC@`R{ zNtZo`-BIlN@S3p5R5@g6)L!^ke**MW;W|O*-yr-?Gik%Gg+D!iG4e{+X1HmwXg?9} zd2l@nXR?!j$pqLq;VOUc;@nqtx}~^JU&*0&_T*nC+%kNT$WIOY>Apu?&+n{_r(Qfq zc1aX&u`Zsq!oLCjyuo=2q;H_kPYdo-es^1i|3qFdJT$(StP5`w_xq>t@Hud+<6A4r zl?Q(+r~GB4@^FK6;TJ(3q~}ZEPu~}=$CW-(xl`1`3vjFB$r1G| z;S5t@9{_)ch`#{tM7ZV8rtANuxKGbtBl<%b-0FH*g1hSY{UW|hU45<;@odtKi)Q#+ z5&tM%x;x=d_49)+ykQzo?+THBneRm5KSuc5;I_+~xXu*qvKap1k8#1jNQ7Ste@d?z zSIQ6R(5TC`4gTaFuPfJ1_|sQMGerIPPli4@+=f$lySatG9d7k_Ow^6*GTf(j*)GB_ zhMUHP{B0Ha-vod5a`#l;ZmUGQm%y!#|0`Yo>v5mT_o*1~i{Vz6?>AjNH{d??>x&VJ z!ZpJk`G_p~r>KX|M7%pi_$9h@ZKv^cYf^bUcDTDDK0mJ1@93Ko>U^ve;mdU6s9E@L z6ytXj-0E@no-Upi8IQ>4PPl2j%U_o8FF2i#|59A3eg=znDHCoP{u5pNe%z=0(>Fae z`4`WzAC{Hfo|-}NG0=`Iz|U7_>$!=L!))gqooxGBE-#I>_1 zZ!_E#sNm87bYOA+nnWJ1EQmcJxJyF3|M!sPg8K-HXg1^zTyg||D7i>Gem978{qR2> z@wZ$C{aS>-PS<}D7xDD!b^bQ^Q@iJja@pa20&d%t_}&Vxjk@+4x0uJXQqG$QL%&kt zUn9!}H`R;$-7Ch$1@Ncz@HaNat}m?ykY!8p1VS3ms@& zcZqpw(-Lj_-l2>CGu)@@P8IQQhg&_qChO8QT+hqlmh&l|rJoJ{^qg16&vW2U@h9lg zEfC?q67^6D_ZtY;a1-A9fulMY0$;f0DtQ5 zYeo1ML^yb{AK6Axd@b-Fh3n4Sd40aFYcFixvi7`K7rzbul*Q{LE&DEYc8%0 z23{`w%YH5YXWC%zRB{n>Vp_PN(Zx?Oet8M^wh<380xjqIO1Swji@sa%ih z!k3Eprs=|;Cj7hT!k3A7F4g(_MfiJl{Yu)Yvjl-lS3wsV*H1)#-dU}U$0o|#qVr!Z^Do~vsr|b z{}sUZ#l>Y{})gMS5DB5uo{MGH;Pn37Fh*!q< zwP>#v_^aiI5hDEW@K@JEqOKp>p5*x)AnHxVYZv}`BAx=c)%n?>Yu^$P{%l?SlnVcD zA|4rk8T=_<1;T%p2q*8qD#l|y{MGBt&7%Lva1FRm`Lc*~8--im|3KvPKArz1qJ3Aw zUtMp_^x(*rI}?(dv1^5R$sOlM_tkmR2c!GyHCe3tC3;VtyI$v3JBv^kL*8KJpM$h! zfjl&q_X5${<(sy)nS*hK|8C@DDCLLxB$bKsO?k6}NR*@YTfYPV6bR}pT8sF9r z{r?i^ZXTCrsK4PRFxwxkhL>&`N?D*aGUbovB;up=ED?Q&k*@S!ny!n*6&wfem9S^n zW;0NkX^mjG@l}(>>#TJdzj(#8-fBqk`JImPI=3g}_7)h`t^p>stI(`=4Ya6TgRGq? z7H7FH5OQ=%W&c-KJ1Ts2-cUlabO!5c6Vs%_AMjPyRfLklcWa%qI!7FCZ!$~3iD-g| zH6?t@>G7mS9NyVox`@m>YF)MEu0UFv);-Y1^Am7~YLLU=K|Bl089)Xr5AMS9#5^io zZqFgA0978JFVIyL!0)LG9;ymZ>-N?KyB!>fp|%bc>G#Y|w?+Ia-80;kuI{RQ1YET} zM1X*EHY&vv(r@-|g?io@N2v-Vo)kA{6Y)dO}65-{G3xk7btr_qwL%gl|-bl<mVIW;i{LiqI^?G4sf7l;5yV&=qpHz5cpTexme0%0hPw4wY{9B!#PV zhMWU3qr6?-P++#-hx`|&Yi?HKnSd*Z)*P7Wa8&sM6)uOn)=wRr!ihc>9K;$mb$jR!WyS%IIC6WCwz)o$^nd3cdg6e3Iu$C5$SRe zRR^5@8q|cdqQ+V7ag}tZ(zB;Mu36+?DT+Qa#+_g&;PzG@$KuyKGm7PxdGX%FLyqnm zj-7jleCN5mV8DtsOf3hnlJOYJcOP{$Y1pApz(_1Fv>+pTe4<7OdM>ff4Htoyt04jmznG_(K5)2HR;`ALRUW#uxZQXRrZY6>z!E zWE=whj^a9tcY8j0IV#XcT@=^ZF}xAsIV=gx5Jg2W){cf_Ml|HOm0+K@M<{ z@~Rr?8!s0&h@$;k?W9lqMk7YWhz1qa-&dN%M&+GES zU0LZ%oF1$zC%4}+9g`M56^A(_fKXnHXkRF3o(^7TnU0yG&N3Yn$_y4W&r;A4aC)m< zGY@iLRE27Slv&ghh!GiUyt|B@ zS&=78GpZUY#$`;8Naz;jm(2TgLl9*qhBf zjl6ckp6Hs(b&>L6*u*mx4(P4(cvkT4%>8INaUZW(bZrt{9Kkxs5hR7)&udBgb76;u zMCbt{n{IV?!Vf+e`@vAvkd@jyT-I^a;m$+aOn96@j6V3U;?*LPLM5^mcv$NVF2)*R zb(A-2hevpk`Lsp_U!(S@aRzH1Ai>bUxgHZk9gG34$9c=hDIhSL zn8*{Hckp-+Xq}PG0h)M4H+VAEE#4Q_^J>yQ#aezttcQ3Gn|TdtA41RAXk?R~$R`!{ zDRt>dgm{|wC;ljST6L)A8UBknm&OCN`f`}zLMVO3co;>;lO?$rEf3!4pDj{&)yU+r z823>buSJ>=Q|TsS7oIoVaa1E?uk+y_{p=gO$MYK@%%MSd$UOt=DvuA{-u0&Oa2`>- z0GoLYbGgw8-r%kLH=hD%NWP_Mq9BBvw|P59>cqjsCXC9;clbybx0o=5DtcGV9UW*> ztZymwdwfLjeie!2eLfel{CvPWGS3@L8j#2QAMz^ajrNf-fvOy}^s(T}s4p3^McpMu zuBbkdsZV%A@|bBeU<*%**AhuJBwBo`?zntfXa1k@EMOsjUY-AQo^UPd4t)TTyF*)f zWYUX=`@&f51s5wS3b+b$f}zTyB8Ou@Q4!W%?g|I5g<)6_1@i2Xo8)$&Q6Yt~d&TOT zqTYQ0ehc<-SX&1}eU)v-N>>myzWnHVGG|}<(sf?6;>HF4zStU9OYYle~u@_eVH(xqwFvqrtuC@ zL>ND!k(HgsRNkrQ&?QIt&UgrSa~7LJj$Gw?V+QLeOp50PLjPd*a+M#9nan?c)r{MV zE{l$YyDZsUWtXw1xXW{-yZs{{$1HZpj}J|RZdUPrK*7rOG^30xI0N#lM$!r3;!dGG|x@~{NMcakIIu8cKA0x4a@9N zRq5&k7PRDTVSeoI0MyKRK%LOwal%9cpGv0D8{Z%>rcu>-@eMP#`7bb z?oUd#sAvwQl%xD+yjnGZ>17V;Y5%7LtfC?xwjW6t&rzb9U*Yrn@5ZT-_J|?cv>QjB zJ=~|Iw7{=>jCXd5L@Oq=zXk05+FX8Hwuv zO4)h#aE8&Se`GQKCuA>@M4YK265xPVQ6$>vWntvr0ZmffDv>0Z>=khyz!YgKCK7RY z-ide};IyK6guX@jhC_#>uP*6d+@Kv_(UBVL*v=Exp!>F~63N~H>|ko9rrK2LkhCDk zcTBwEea3v#n_VIaFabT{{Np`6tm-5UKA_Rx9Hz<3trY9`2Dd+8y~(u;07IDplz zokBZwiu>~mgl(?{CZwE9{C+#(UD=2)qNSv|h$0R@XpNkOk4HYlMg$K|HZjv3N*g(=zLtApJK)OB5fD@{qvMM`v6nNl2H z=u?x)518uJCaa?g&c(R8aJ|Nq$b!Ng>snJn6_5KmQ>Q9s2C>9st%5mH#q}mDvxU3C z#B{aU|2LXCaa)F)Ocv66yxG*Lg2@H9m=c6J!>uN4dIv&2&us|ns;#IA+|K{0_20q& z@z1`~lz>~LJKw;(pd)t|dkhMfcO&m)Sb%vIb~%;*G<8usL%w@Vsmvo-10^~HxL3F- zyxUt5SgLv+bILN4E&M#3RRLdZqdEpy4J=nZ0cVr?Hmu%On9{;Az*+07yieuB&Cu@G z`ve{kJ}jGv)lu=F>0p&f1zDv*wP~fPi!iGY9uJvPtGqC{@OfwWJar-0D$_yacGiV_ zVZVn>mWn`V;A&GM-Kum~6+U7j9V7>wYfQ=XpW8cwT#q6)UtIumtjkaCwWfpE9kt(M zCT8j}ofH+hR^a2N&Xg%=5(g{XPlyH}7u3GJ)$2@MWj&%kqr6jW zFfp4E(J`7$DHIIaQ7q&}QzAKGl`-=v(?RSG)F`1IhJ-3-d!OdH63>#)Gu(%z%}fe{ z&+^1-gXH%duSs{%?S*K8E#K#Pw%kFl)BA$F$&3o2t@*~#(1vM<#~;lbUKXfps~7%heL((e7-J9jtZva z-Z0r%ap9ENc{AeW_i=sN&8Dt06bkRGh0Z3K|NT20x1-AAtPZ}#-ISdQpR20sZSI~9 zLmlRRhu0dZbb45(gYWVXR8h<#>^<-lUVM1G&npU97j^MLIPc7+?nBvGSZ|@iek2k^ zrNd6}eAmZfG{J(vfx@pGxIX1+ zvL5LkSooQUVmg$_9aQ@!%->TCH><}@*0%@m= z$?5-&_Y_zoeJ>s+^Cj@3AJh+fcZr8#uk@pgr#!fucgN|>O!p^II-^9Ie}F+{{P){5c{9_q{GV@*&dl0?rfF8!LQmxr4#EagmFn>y_v^#9(?AS zlUZEgC~(a)bB;`Rh{?=1^N|hb0yD4i>AbQrv{|Qvq29$J5R{xRF>?k>7F=i#XGQ2z zGh+yvr!F%y_Nnokwb0Bs)pSh%-ik%S!R(zcH#4R*os6NcFf%4gz18id@?UA@EDw!I zwy<>r245w7gmo-@t~N*JA@qgC<}MCaK=7LLD5h)3!-*HmsB$N4NGq;2AFTDDy0{LZ zs0Gkh&T|F4F3%D(?^8@!9RbMD%vl?}-ppHvdCwfk);l+tQ-T;ol6T)|wh`}Ed);L2 z!c?KbGhnOcn^}0X*%pMVVZg<_nb$4mgM%}f=A;Lc-^?KMyw%(%BAGeBhGronup)Bl z-p}1;hD9iUdu|7#*4@3`42vpp7fjFveQfqJOS!`gyM)MNXP&Kl@J=&KK{O8{2!FHz zwdkmr<#hOAOW>H{4uqgNd6#*kF_K@j;DJ(L*J{Ie=!#pp+q}M`3Lz_Vc*mCDpXNXZ zwE~|vwY>y-6;SS1?&+{Zx$Q~JMDI1v?x2#2MC02nC`--f{%L6|%gj|Blo%NzLx`_9 zkC_lCjphX%)%zU|XOP4*%pFj>hZNGD(ZceMYgv|w_R?ArMh@(&V|nS!u7_ z51PlSZ_|1VX7Uxf`5n3^u%d!m4eFImzx-m)$yLO3tgJNmVSAv*xkFRQP^PG;6a$;; zU(>?ML*`svxM_jeQ(ewLMU23!%;)I>i;&EBDqB^Sx?L46p-?CGuVUSx>g6vvqXY3C zqC9NA!pI^ng)whY5l>2%hwBc-in;x~@VG{LE3geAQs!P0$tdVVY;MQR&ua5wEbdWU zr3t21Bk3H7@i3*MdQ~1VXXx(FkwJ5mHD-m~oI)y7{-UB&ogN65!7E7xu<`X$IQKT8A4YGsKZh#D36P-5{dDoIv2_m&Kqwh zfGa7hzV||XL^rX3mE(TWXtoGKM`|0kDsr83upX#m* z)$pt^_hi4a!Q59Da+K2_g4n6fF*QiDIa3$#I9IjX%d@6zG*8mqJsAffVZB>9rOxZ3 zGE8Om^bvMRlfTMZkvn-ursSS7ueGWNFYTQE5sR`g10=IZVH=5b{-Ekv{8-g_*BUg5 z_aw0%RX@ZEtTKs<6`mI$@)X-%jt&@G!zyODs4ZmuTOR%fJNjAmj(W5{L9Fol%!l{k zqXTOS5WkR7nP+=!u}u*7p=MPIJTbBq>B-?9B{iR)Hm^}xL+W0Y{&CHj=*)1r;D9K+ z#E(4E&GdcZ?;g!ItW5_5r66N!;GC)&kOcCG(^HI^+4TnB?+SnoCn1$H!qIp0&PfUOu z>>Z$g<|Xst(a$)B6gUPHRN$%spGkX2bY zCUCuCK1TOYX~5O)BfN~c%B$uq-IEiT_PeNOCOy4oo~R2`>I=4?BUR>9{|fnfllf>} zd_n@`)etTpmvF*UhVr`TL*s*7j3%K{TPGrcH-!8t1CH@lYN8gw-ZY=c^3jfHOp@_< zI_8mf2~>gAwJ*BMW^;F4sYd%~2?R5uf19&)x6t`pt)boFPJo6NLAA<1&?8h!`#A#eZ&= z@WRV0W|UZrj`H-$pHxkK_r037n6Lj+swteMy;YN3+;fwgWve+q zl7>1a+{DxPtJ-#<2fi@>q^WAg%lBd~7)gQU67jOF({_&2iGD)oGY(8c$W^RxEMg`e zw=*0gA^9ML0!Aet*y=l~O61vQd3=_X~U?0VEAK$7l#0jm*Qt+!|yBev5_2M z)iuqBjng`@4XNqv(LMNU^C^*MvHoGhXtI|US*pAcrjQ4unIO>fDo8a^Lhp~fe#Q8PhV{9UIqm6K9tn{bFP5(Rd#QjX4ZOFxq z@B0|ZPm5ptAND|FjQ)F`<%Ke4;rq;#+DpMzsE{Fe%be!Puy$6^K>JB<0sxK zI3pXsBe4RXeZGs2%DCgC%{-UCx>NB99(cdr*ij)kUsL}DKUAqAbYl9z7)G7h!(sEdkvI|^v;E*u`xD{US! z;X|3%u_+ITpYr^UO}Q|B$_p&L!j%DelqsNvn+E^nDi>SsihlDO4zgceNR`rlzx1W< z7r%R5Vi7$$N?@pf_H&31`Nx6ev@@w(YFQF3uJWx1mOm5?@3+NymO7sUBvQp(;a+Cp z?_(!X`uGzW9l6Rvi&`EUNneZLQYs5Ptp;VpS(J20EooJTr~ zQuWSMR9x*y%&HXGvQ@Xmb%o_3R%a|26T^7!rmNd0UIEa)Oq61OSTwP^mx|wrk*vy7 zu5zWNEK*9SS(VPQW6_vF= z8Y(PXJn{YJ95m>!weZyepO|>v!yiKii;MA5s|nT*;IA-EYQcM>o@{(7@ z)A4>c91F1#l`|5muX3B^TFo>bf+ZcF$HFf##q1DvPk>%yJN-8tj^!JNv4;Y_*(zml z2aZg4!R&($pJufdejvPirQB}e??>gE5Ud%3E+$NHoj>l+Tblo$tK4B3$D*rjJ{+WGrcbCNJe4 z!QScV5Wb|~^YT|vao>?K*2_dn3Vei$==`O_btg9G=8%M~?Awx>E(?rQi!YWx0jFnw6Q-#1K}D zY_im-2X*{+sFZR$!;WHSor^S-G4$wNh)$?nVHvKeIjcDgnWn(f)A(yh%We$Mf>W0ER4W>fsyn%EquBIV?x7rWQ(if z5f#f>$tuf2qi!88rUCW@R4i(-M;7S;kvSl8pS5LV2bsNG3)1IR=rbLHhq5}%&BdY| z&EHpSeDI14`zjBE+4E@iWf~7wgyBPGh=u!8K6TdB<(i5&d=>2#j0Zr;a+THllB_2GHcXPP~{zkFEj;@!}t5}nF850(IHr`P#(1$ zA2T47zAJe|tk3FMq-HqsUcJ`hWihdFcQXG%)$y>alnc1Xngp3(A)_M^H)7;k^#F0S zx?ryIm|$JwgQw6JK$%Vz3l@ui$9Vp6%W2FP1wC1n(RhjTFyZwWt3g$)ni2klnD|sV zfxZlM#i$--@l`ja7q1anz;za1bOKUXAU-{66lE6U)+MyvwNb`SOUTQbHjpVkX*rIi z$l7oW|6~sHh#Rpc$Gl6ex17C~Y|Bo`M~XIX)uXwwt7U`e@0#wTu9i|)z{7iORNj?l zvE3s&jnJr{1dbb(8M(07DEf&ey?CKD)vE3jnj5mdpR#bP6jtBkAw1vVA8yJwbd8!}_PC^-uH@ zo8qJw$FfMZ?5Etmu6a)pDP+WE=?x3FMAGMo?4mRUf78M(k@Nw_;&hK3WwYfV#t8Ub z`U_oE%fTwYIo$It%b_aI$$oq@oA%z7w>9NtrHjqy!*g5(rW)?ZRo)TIj#^=&(=Q~V zqN0h;zYQQo!mWNAl*qA%O#DDPRaV%~-hy3d0%{Ha3U7juk03YKD|!YFT&w|}6i zI8w32**fC+9OXj`w@Bj^ha!hUSdu9pS?)He>WRuRwGKPSL43bYZh84hPKa!*${AI^ zS*zh-QJ(?;RdAk*h4T9e&1ptuB^>vL`Nl7F87kDT;*dSMd^!7Uyuv$ z4C<f zSSGU$(_VDi%8zs%b)6tghZaTo#IjW1wMYBdcQW}Hi&NHkd5_g`_P1mHmv-+hmgm&l z-eYRrp$T{y!Wml3#*S4PDmFqBKX-MGnrt&NleC;Ki6+T0KE_N4ru~aVt9b-vPcqHtX=(q~TS(up@ z8ZtclvbTa;EsM=zAH=SbN;v}>^nZy1#q8WVF10UVI+l}tVfk0|WZ-71mfrDY@=EBHzcG|w;_30lJ#bBIr` zhv>t-k#Gm_fHJl^r#JhQ9*q7-yivl8%(Qk!1Ihs3r;B>Oe>->hXA0Tx z2;oXT6?I?aLitfB5r}|XMP3)zI3%5;?6#cECU#A_adX48LTPUkhcZRyMSWN1&^uJSiIti2;%{Kl>b&Jg2{G@;dXiB+sPM1AojG-caMM9QUB zt%i}*Epkorab=_ZitRFehSUeg@sh@`F+K+@w2szP8`mDxVjS zxys7bKBxKodkuTF^~{JwrQX5}w?*W3t@9ud(tY8)e}p)zyl(iNPG@TB|T(L${rQeH3CO!D&8v`xMNf4HVWPWIIJN zdH_Y#T;)3JLcSX-vp5BZXt)EeO1=-O+5tY0$qE$(O-IhnRhC%iwU?Q(Q2k>D8XZ#? zQg7mV(ZRy~JN(8{?~|k45I3w;qq4GuCEsb~YUHw;Glg$=6%Isc zu%4tE@L@F>l^>a*QGv$C)-qOy6cf{3qDEwDygJm{JQQbG(dTX}fA=kei+3ViDIE&t zvQ2$XtRgF1vT`E_5b+0dm3w3t(}j#%HR`-8_gc?l?oqm@+dT$MB&{9FS{G+w%2F$@ z0DbhMoL=drnHsc_D9fx|V_JrlwJ_QzeKb$&rj1;sQPiSrdHH;FEiMhYAUV(q(GXEl#8KMP#u48o8sa|dAQna+y>v&0wmK^1e(SV$AB!oeNP$Gt zm}>nz;wjzUb0LRA_5^O}sQW4Ok8(>|-XX>x(l6A?W2twHyqcBqXW>VJ<~Z~iOt7E>WF zDsyZ9)QSsh3&v7cjLO?%R<&_o`#UvWg73Y->q%>36+Y-k9tj5<4QVKGh|YQQ`@nErfuw^> zhJ^fpkG>3MwC2;rP?_Y1M_q+05HzOXU(#1|RUTPB+n924d7U>@XRp9kRQ2!y1M>^= z2OQbIe_=u4fB^-C0|yiyJ#fIF{>Kcgbh^q1Rvuk&Ox2MC=3rlIz#LVOAx96%A29I$ E0q@gP2mk;8 literal 0 HcmV?d00001 diff --git a/vendor/github.com/open-policy-agent/opa/internal/compiler/wasm/optimizations.go b/vendor/github.com/open-policy-agent/opa/internal/compiler/wasm/optimizations.go new file mode 100644 index 00000000..fed367ca --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/compiler/wasm/optimizations.go @@ -0,0 +1,274 @@ +package wasm + +import ( + "bytes" + "context" + "encoding/csv" + "fmt" + "os" + "os/exec" + "strconv" + "strings" + "time" + + "github.com/open-policy-agent/opa/internal/compiler/wasm/opa" + "github.com/open-policy-agent/opa/internal/wasm/encoding" + "github.com/open-policy-agent/opa/internal/wasm/instruction" + "github.com/open-policy-agent/opa/internal/wasm/module" +) + +const warning = `--------------------------------------------------------------- +WARNING: Using EXPERIMENTAL, unsupported wasm-opt optimization. + It is not supported, and may go away in the future. +---------------------------------------------------------------` + +// optimizeBinaryen passes the encoded module into wasm-opt, and replaces +// the compiler's module with the decoding of the process' output. +func (c *Compiler) optimizeBinaryen() error { + if os.Getenv("EXPERIMENTAL_WASM_OPT") == "" && os.Getenv("EXPERIMENTAL_WASM_OPT_ARGS") == "" { + c.debug.Printf("not opted in, skipping wasm-opt optimization") + return nil + } + if !woptFound() { + c.debug.Printf("wasm-opt binary not found, skipping optimization") + return nil + } + if os.Getenv("EXPERIMENTAL_WASM_OPT") != "silent" { // for benchmarks + fmt.Fprintln(os.Stderr, warning) + } + args := []string{ // WARNING: flags with typos are ignored! + "-O2", + "--debuginfo", // don't strip name section + } + // allow overriding the options + if env := os.Getenv("EXPERIMENTAL_WASM_OPT_ARGS"); env != "" { + args = strings.Split(env, " ") + } + + args = append(args, "-o", "-") // always output to stdout + ctx, cancel := context.WithTimeout(context.Background(), 10*time.Second) + defer cancel() + + wopt := exec.CommandContext(ctx, "wasm-opt", args...) + stdin, err := wopt.StdinPipe() + if err != nil { + return fmt.Errorf("get stdin: %w", err) + } + defer stdin.Close() + + var stdout, stderr bytes.Buffer + wopt.Stdout = &stdout + + if err := wopt.Start(); err != nil { + return fmt.Errorf("start wasm-opt: %w", err) + } + if err := encoding.WriteModule(stdin, c.module); err != nil { + return fmt.Errorf("encode module: %w", err) + } + if err := stdin.Close(); err != nil { + return fmt.Errorf("write to wasm-opt: %w", err) + } + if err := wopt.Wait(); err != nil { + return fmt.Errorf("wait for wasm-opt: %w", err) + } + + if d := stderr.String(); d != "" { + c.debug.Printf("wasm-opt debug output: %s", d) + } + mod, err := encoding.ReadModule(&stdout) + if err != nil { + return fmt.Errorf("decode module: %w", err) + } + c.module = mod + return nil +} + +func woptFound() bool { + _, err := exec.LookPath("wasm-opt") + return err == nil +} + +// NOTE(sr): Yes, there are more control instructions than these two, +// but we haven't made use of them yet. So this function only checks +// for the control instructions we're possibly emitting, and which are +// relevant for block nesting. +func withControlInstr(is []instruction.Instruction) bool { + for _, i := range is { + switch i := i.(type) { + case instruction.Br, instruction.BrIf: + return true + case instruction.StructuredInstruction: + // NOTE(sr): We could attempt to further flatten the nested blocks + // here, but I believe we'd then have to correct block labels. + if withControlInstr(i.Instructions()) { + return true + } + } + } + return false +} + +func unquote(s string) (string, error) { + return strconv.Unquote("\"" + strings.ReplaceAll(s, `\`, `\x`) + "\"") +} + +func (c *Compiler) removeUnusedCode() error { + cgCSV, err := opa.CallGraphCSV() + if err != nil { + return fmt.Errorf("csv unpack: %w", err) + } + r := csv.NewReader(bytes.NewReader(cgCSV)) + r.LazyQuotes = true + cg, err := r.ReadAll() + if err != nil { + return fmt.Errorf("csv read: %w", err) + } + + cgIdx := map[uint32][]uint32{} + for i := range cg { + callerName, err := unquote(cg[i][0]) + if err != nil { + return fmt.Errorf("unquote caller name %s: %w", cg[i][0], err) + } + calleeName, err := unquote(cg[i][1]) + if err != nil { + return fmt.Errorf("unquote callee name %s: %w", cg[i][1], err) + } + caller, ok := c.funcs[callerName] + if !ok { + return fmt.Errorf("caller not found: %s (%s)", cg[i][0], callerName) + } + callee, ok := c.funcs[calleeName] + if !ok { + return fmt.Errorf("callee not found: %s (%s)", cg[i][1], calleeName) + } + cgIdx[caller] = append(cgIdx[caller], callee) + } + + // add the calls from planned functions + for _, f := range c.funcsCode { + fidx := c.funcs[f.name] + cgIdx[fidx] = findCallees(f.code.Func.Expr.Instrs) + } + + keepFuncs := map[uint32]struct{}{} + + // we'll keep + // - what's referenced in a table (these could be called indirectly) + // - what's exported or imported + // - what's been compiled by us + // - anything transitively called from those + + for _, imp := range c.module.Import.Imports { + if _, ok := imp.Descriptor.(module.FunctionImport); ok { + reach(cgIdx, keepFuncs, c.funcs[imp.Name]) + } + } + + for _, exp := range c.module.Export.Exports { + if exp.Descriptor.Type == module.FunctionExportType { + reach(cgIdx, keepFuncs, c.funcs[exp.Name]) + } + } + + for _, f := range c.funcsCode { + reach(cgIdx, keepFuncs, c.funcs[f.name]) + } + + // anything referenced in a table + for _, seg := range c.module.Element.Segments { + for _, idx := range seg.Indices { + if c.skipElemRE2(keepFuncs, idx) { + c.debug.Printf("dropping element %d because policy does not depend on re2", idx) + } else { + reach(cgIdx, keepFuncs, idx) + } + } + } + + // remove all that's not needed, update index for remaining ones + funcNames := []module.NameMap{} + for _, nm := range c.module.Names.Functions { + if _, ok := keepFuncs[nm.Index]; ok { + funcNames = append(funcNames, nm) + } + } + c.module.Names.Functions = funcNames + + // For anything that we don't want, replace the function code entries' + // expressions with `unreachable`. + // We do this because it lets the resulting wasm module pass `wasm-validate`, + // empty bodies would not. + nopEntry := module.Function{ + Expr: module.Expr{ + Instrs: []instruction.Instruction{instruction.Unreachable{}}, + }, + } + var buf bytes.Buffer + if err := encoding.WriteCodeEntry(&buf, &module.CodeEntry{Func: nopEntry}); err != nil { + return fmt.Errorf("write code entry: %w", err) + } + for i := range c.module.Code.Segments { + idx := i + c.functionImportCount() + if _, ok := keepFuncs[uint32(idx)]; !ok { + c.module.Code.Segments[i].Code = buf.Bytes() + } + } + return nil +} + +func findCallees(instrs []instruction.Instruction) []uint32 { + var ret []uint32 + for _, expr := range instrs { + switch expr := expr.(type) { + case instruction.Call: + ret = append(ret, expr.Index) + case instruction.StructuredInstruction: + ret = append(ret, findCallees(expr.Instructions())...) + } + } + return ret +} + +func reach(cg map[uint32][]uint32, keep map[uint32]struct{}, node uint32) { + if _, ok := keep[node]; !ok { + keep[node] = struct{}{} + for _, v := range cg[node] { + reach(cg, keep, v) + } + } +} + +// skipElemRE2 determines if a function in the table is really required: +// We'll exclude anything with a prefix of "re2::" if none of the known +// entrypoints into re2 are used. +func (c *Compiler) skipElemRE2(keep map[uint32]struct{}, idx uint32) bool { + if c.usesRE2(keep) { + return false + } + return c.nameContains(idx, "re2::", "lexer::", "std::", "__cxa_pure_virtual", "operator", "parser_") +} + +func (c *Compiler) usesRE2(keep map[uint32]struct{}) bool { + for _, fn := range builtinsUsingRE2 { + if _, ok := keep[c.function(fn)]; ok { + return true + } + } + return false +} + +func (c *Compiler) nameContains(idx uint32, hs ...string) bool { + // TODO(sr): keep reverse mapping (idx -> name) in Compiler struct + for _, nm := range c.module.Names.Functions { + if nm.Index == idx { + for _, h := range hs { + if strings.Contains(nm.Name, h) { + return true + } + } + return false + } + } + return false +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/compiler/wasm/wasm.go b/vendor/github.com/open-policy-agent/opa/internal/compiler/wasm/wasm.go new file mode 100644 index 00000000..77b0150a --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/compiler/wasm/wasm.go @@ -0,0 +1,1783 @@ +// Copyright 2018 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +// Package wasm contains an IR->WASM compiler backend. +package wasm + +import ( + "bytes" + "encoding/binary" + "fmt" + "io" + + "github.com/pkg/errors" + + "github.com/open-policy-agent/opa/ast" + "github.com/open-policy-agent/opa/internal/compiler/wasm/opa" + "github.com/open-policy-agent/opa/internal/debug" + "github.com/open-policy-agent/opa/internal/ir" + "github.com/open-policy-agent/opa/internal/wasm/encoding" + "github.com/open-policy-agent/opa/internal/wasm/instruction" + "github.com/open-policy-agent/opa/internal/wasm/module" + "github.com/open-policy-agent/opa/internal/wasm/types" + "github.com/open-policy-agent/opa/internal/wasm/util" + opatypes "github.com/open-policy-agent/opa/types" +) + +// Record Wasm ABI version in exported global variable +const ( + opaWasmABIVersionVal = 1 + opaWasmABIVersionVar = "opa_wasm_abi_version" + opaWasmABIMinorVersionVal = 2 + opaWasmABIMinorVersionVar = "opa_wasm_abi_minor_version" +) + +// nolint: deadcode,varcheck +const ( + opaTypeNull int32 = iota + 1 + opaTypeBoolean + opaTypeNumber + opaTypeString + opaTypeArray + opaTypeObject + opaTypeSet + opaTypeStringInterned + opaTypeBooleanInterned +) + +const ( + opaAbort = "opa_abort" + opaRuntimeError = "opa_runtime_error" + opaNull = "opa_null" + opaBoolean = "opa_boolean" + opaNumberInt = "opa_number_int" + opaNumberRef = "opa_number_ref" + opaNumberSize = "opa_number_size" + opaArrayWithCap = "opa_array_with_cap" + opaArrayAppend = "opa_array_append" + opaObject = "opa_object" + opaObjectInsert = "opa_object_insert" + opaSet = "opa_set" + opaSetAdd = "opa_set_add" + opaStringTerminated = "opa_string_terminated" + opaValueNumberSetInt = "opa_value_number_set_int" + opaValueCompare = "opa_value_compare" + opaValueGet = "opa_value_get" + opaValueIter = "opa_value_iter" + opaValueLength = "opa_value_length" + opaValueMerge = "opa_value_merge" + opaValueShallowCopy = "opa_value_shallow_copy" + opaValueType = "opa_value_type" + opaMemoizeInit = "opa_memoize_init" + opaMemoizePush = "opa_memoize_push" + opaMemoizePop = "opa_memoize_pop" + opaMemoizeInsert = "opa_memoize_insert" + opaMemoizeGet = "opa_memoize_get" + opaMappingInit = "opa_mapping_init" + opaMappingLookup = "opa_mapping_lookup" + opaMPDInit = "opa_mpd_init" + opaMallocInit = "opa_malloc_init" +) + +var builtinsFunctions = map[string]string{ + ast.Plus.Name: "opa_arith_plus", + ast.Minus.Name: "opa_arith_minus", + ast.Multiply.Name: "opa_arith_multiply", + ast.Divide.Name: "opa_arith_divide", + ast.Abs.Name: "opa_arith_abs", + ast.Round.Name: "opa_arith_round", + ast.Ceil.Name: "opa_arith_ceil", + ast.Floor.Name: "opa_arith_floor", + ast.Rem.Name: "opa_arith_rem", + ast.ArrayConcat.Name: "opa_array_concat", + ast.ArrayReverse.Name: "opa_array_reverse", + ast.ArraySlice.Name: "opa_array_slice", + ast.SetDiff.Name: "opa_set_diff", + ast.And.Name: "opa_set_intersection", + ast.Or.Name: "opa_set_union", + ast.Intersection.Name: "opa_sets_intersection", + ast.Union.Name: "opa_sets_union", + ast.IsNumber.Name: "opa_types_is_number", + ast.IsString.Name: "opa_types_is_string", + ast.IsBoolean.Name: "opa_types_is_boolean", + ast.IsArray.Name: "opa_types_is_array", + ast.IsSet.Name: "opa_types_is_set", + ast.IsObject.Name: "opa_types_is_object", + ast.IsNull.Name: "opa_types_is_null", + ast.TypeNameBuiltin.Name: "opa_types_name", + ast.BitsOr.Name: "opa_bits_or", + ast.BitsAnd.Name: "opa_bits_and", + ast.BitsNegate.Name: "opa_bits_negate", + ast.BitsXOr.Name: "opa_bits_xor", + ast.BitsShiftLeft.Name: "opa_bits_shiftleft", + ast.BitsShiftRight.Name: "opa_bits_shiftright", + ast.Count.Name: "opa_agg_count", + ast.Sum.Name: "opa_agg_sum", + ast.Product.Name: "opa_agg_product", + ast.Max.Name: "opa_agg_max", + ast.Min.Name: "opa_agg_min", + ast.Sort.Name: "opa_agg_sort", + ast.All.Name: "opa_agg_all", + ast.Any.Name: "opa_agg_any", + ast.Base64IsValid.Name: "opa_base64_is_valid", + ast.Base64Decode.Name: "opa_base64_decode", + ast.Base64Encode.Name: "opa_base64_encode", + ast.Base64UrlEncode.Name: "opa_base64_url_encode", + ast.Base64UrlDecode.Name: "opa_base64_url_decode", + ast.NetCIDRContains.Name: "opa_cidr_contains", + ast.NetCIDROverlap.Name: "opa_cidr_contains", + ast.NetCIDRIntersects.Name: "opa_cidr_intersects", + ast.Equal.Name: "opa_cmp_eq", + ast.GreaterThan.Name: "opa_cmp_gt", + ast.GreaterThanEq.Name: "opa_cmp_gte", + ast.LessThan.Name: "opa_cmp_lt", + ast.LessThanEq.Name: "opa_cmp_lte", + ast.NotEqual.Name: "opa_cmp_neq", + ast.GlobMatch.Name: "opa_glob_match", + ast.JSONMarshal.Name: "opa_json_marshal", + ast.JSONUnmarshal.Name: "opa_json_unmarshal", + ast.ObjectFilter.Name: "builtin_object_filter", + ast.ObjectGet.Name: "builtin_object_get", + ast.ObjectRemove.Name: "builtin_object_remove", + ast.ObjectUnion.Name: "builtin_object_union", + ast.Concat.Name: "opa_strings_concat", + ast.FormatInt.Name: "opa_strings_format_int", + ast.IndexOf.Name: "opa_strings_indexof", + ast.Substring.Name: "opa_strings_substring", + ast.Lower.Name: "opa_strings_lower", + ast.Upper.Name: "opa_strings_upper", + ast.Contains.Name: "opa_strings_contains", + ast.StartsWith.Name: "opa_strings_startswith", + ast.EndsWith.Name: "opa_strings_endswith", + ast.StringReverse.Name: "opa_strings_reverse", + ast.Split.Name: "opa_strings_split", + ast.Replace.Name: "opa_strings_replace", + ast.ReplaceN.Name: "opa_strings_replace_n", + ast.Trim.Name: "opa_strings_trim", + ast.TrimLeft.Name: "opa_strings_trim_left", + ast.TrimPrefix.Name: "opa_strings_trim_prefix", + ast.TrimRight.Name: "opa_strings_trim_right", + ast.TrimSuffix.Name: "opa_strings_trim_suffix", + ast.TrimSpace.Name: "opa_strings_trim_space", + ast.NumbersRange.Name: "opa_numbers_range", + ast.ToNumber.Name: "opa_to_number", + ast.WalkBuiltin.Name: "opa_value_transitive_closure", + ast.ReachableBuiltin.Name: "builtin_graph_reachable", + ast.RegexIsValid.Name: "opa_regex_is_valid", + ast.RegexMatch.Name: "opa_regex_match", + ast.RegexMatchDeprecated.Name: "opa_regex_match", + ast.RegexFindAllStringSubmatch.Name: "opa_regex_find_all_string_submatch", + ast.JSONRemove.Name: "builtin_json_remove", + ast.JSONFilter.Name: "builtin_json_filter", + ast.Member.Name: "builtin_member", + ast.MemberWithKey.Name: "builtin_member3", +} + +// If none of these is called from a policy, the resulting wasm +// module will not contain any RE2-related functions +var builtinsUsingRE2 = [...]string{ + builtinsFunctions[ast.RegexIsValid.Name], + builtinsFunctions[ast.RegexMatch.Name], + builtinsFunctions[ast.RegexMatchDeprecated.Name], + builtinsFunctions[ast.RegexFindAllStringSubmatch.Name], + builtinsFunctions[ast.GlobMatch.Name], +} + +type externalFunc struct { + ID int32 + Decl *opatypes.Function +} + +var builtinDispatchers = [...]string{ + "opa_builtin0", + "opa_builtin1", + "opa_builtin2", + "opa_builtin3", + "opa_builtin4", +} + +// Compiler implements an IR->WASM compiler backend. +type Compiler struct { + stages []func() error // compiler stages to execute + errors []error // compilation errors encountered + + policy *ir.Policy // input policy to compile + module *module.Module // output WASM module + code *module.CodeEntry // output WASM code + + funcsCode []funcCode // compile functions' code + + builtinStringAddrs map[int]uint32 // addresses of built-in string constants + externalFuncNameAddrs map[string]int32 // addresses of required built-in function names for listing + externalFuncs map[string]externalFunc // required built-in function ids and types + entrypointNameAddrs map[string]int32 // addresses of available entrypoint names for listing + entrypoints map[string]int32 // available entrypoint ids + stringOffset int32 // null-terminated string data base offset + stringAddrs []uint32 // null-terminated string constant addresses + opaStringAddrs []uint32 // addresses of interned opa_string_t + opaBoolAddrs map[ir.Bool]uint32 // addresses of interned opa_boolean_t + fileAddrs []uint32 // null-terminated string constant addresses, used for file names + funcs map[string]uint32 // maps imported and exported function names to function indices + + nextLocal uint32 + locals map[ir.Local]uint32 + lctx uint32 // local pointing to eval context + lrs uint32 // local pointing to result set + + debug debug.Debug +} + +type funcCode struct { + name string + code *module.CodeEntry +} + +const ( + errVarAssignConflict int = iota + errObjectInsertConflict + errIllegalEntrypoint +) + +var errorMessages = [...]struct { + id int + message string +}{ + {errVarAssignConflict, "var assignment conflict"}, + {errObjectInsertConflict, "object insert conflict"}, + {errIllegalEntrypoint, "internal: illegal entrypoint id"}, +} + +// New returns a new compiler object. +func New() *Compiler { + c := &Compiler{ + debug: debug.Discard(), + } + c.stages = []func() error{ + c.initModule, + c.compileStringsAndBooleans, + c.addImportMemoryDecl, + c.compileExternalFuncDecls, + c.compileEntrypointDecls, + c.compileFuncs, + c.compilePlans, + c.emitABIVersionGlobals, + + // "local" optimizations + c.removeUnusedCode, + + // final emissions + c.emitFuncs, + + // global optimizations + c.optimizeBinaryen, + } + return c +} + +// ABIVersion returns the Wasm ABI version this compiler +// emits. +func (*Compiler) ABIVersion() ast.WasmABIVersion { + return ast.WasmABIVersion{ + Version: opaWasmABIVersionVal, + Minor: opaWasmABIMinorVersionVal, + } +} + +// WithPolicy sets the policy to compile. +func (c *Compiler) WithPolicy(p *ir.Policy) *Compiler { + c.policy = p + return c +} + +// WithDebug sets the sink for debug logs emitted by the compiler. +func (c *Compiler) WithDebug(sink io.Writer) *Compiler { + if sink != nil { + c.debug = debug.New(sink) + } + return c +} + +// Compile returns a compiled WASM module. +func (c *Compiler) Compile() (*module.Module, error) { + + for _, stage := range c.stages { + if err := stage(); err != nil { + return nil, err + } else if len(c.errors) > 0 { + return nil, c.errors[0] // TODO(tsandall) return all errors. + } + } + + return c.module, nil +} + +// initModule instantiates the module from the pre-compiled OPA binary. The +// module is then updated to include declarations for all of the functions that +// are about to be compiled. +func (c *Compiler) initModule() error { + + bs, err := opa.Bytes() + if err != nil { + return err + } + + c.module, err = encoding.ReadModule(bytes.NewReader(bs)) + if err != nil { + return err + } + + c.funcs = make(map[string]uint32) + for _, fn := range c.module.Names.Functions { + name := fn.Name + // Account for recording duplicate functions -- this only happens + // with the RE2 C++ lib so far. + // NOTE: This isn't good enough for function names used more than + // two times. But let's deal with that when it happens. + if _, ok := c.funcs[name]; ok { // already seen + c.debug.Printf("function name duplicate: %s (%d)", name, fn.Index) + name = name + ".1" + } + c.funcs[name] = fn.Index + } + + for _, fn := range c.policy.Funcs.Funcs { + + params := make([]types.ValueType, len(fn.Params)) + for i := 0; i < len(params); i++ { + params[i] = types.I32 + } + + tpe := module.FunctionType{ + Params: params, + Results: []types.ValueType{types.I32}, + } + + c.emitFunctionDecl(fn.Name, tpe, false) + } + + c.emitFunctionDecl("eval", module.FunctionType{ + Params: []types.ValueType{types.I32}, + Results: []types.ValueType{types.I32}, + }, true) + + c.emitFunctionDecl("builtins", module.FunctionType{ + Params: nil, + Results: []types.ValueType{types.I32}, + }, true) + + c.emitFunctionDecl("entrypoints", module.FunctionType{ + Params: nil, + Results: []types.ValueType{types.I32}, + }, true) + + // NOTE(sr): LLVM needs a section of linear memory to be zero'ed out and reserved, + // for static variables defined in the C code. When using imported memory, it adds + // a data segment to ensure that. When not using imported memory, it would ensure + // that a zero'ed out region is available by adjust the __heap_base address. + // Since we control "imported/not-imported" memory here, we make these adjustments + // here, too: + // + // a. the __heap_base exported variable is read, + // b. the __heap_base variable is removed from exports and globals + // c. a data segment filled with zeros of the proper length is added + var idx uint32 + var del int + for i, exp := range c.module.Export.Exports { + if exp.Name == "__heap_base" { + idx = exp.Descriptor.Index + del = i + } + } + heapBase := c.module.Global.Globals[idx].Init.Instrs[0].(instruction.I32Const).Value + + // (b) remove __heap_base export and global + c.module.Export.Exports = append(c.module.Export.Exports[:del], c.module.Export.Exports[del+1:]...) + c.module.Global.Globals = append(c.module.Global.Globals[:idx], c.module.Global.Globals[idx+1:]...) + + // (c) add data segment with zeros + offset, err := getLowestFreeDataSegmentOffset(c.module) + if err != nil { + return err + } + + c.module.Data.Segments = append(c.module.Data.Segments, module.DataSegment{ + Index: 0, + Offset: module.Expr{ + Instrs: []instruction.Instruction{ + instruction.I32Const{ + Value: offset, + }, + }, + }, + Init: bytes.Repeat([]byte{0}, int(heapBase-offset)), + }) + + return nil +} + +// NOTE(sr): The wasm module we start with, compiled via LLVM, has NO memory +// import or export. Here, we change that: the OPA-generated wasm module will +// +// a. import its memory +// b. re-export that memory +// c. have no "own" memory (in its memory section) +// +// (b) is provided by the LLVM base module, it already has an export of memory[0] +// (a) and (c) are taken care of here. +// +// In the future, we could change that, and here would be the place to do so. +func (c *Compiler) addImportMemoryDecl() error { + offset, err := getLowestFreeDataSegmentOffset(c.module) + if err != nil { + return err + } + + c.module.Import.Imports = append(c.module.Import.Imports, module.Import{ + Module: "env", + Name: "memory", + Descriptor: module.MemoryImport{ + Mem: module.MemType{ + Lim: module.Limit{ + Min: util.Pages(uint32(offset)), + }, + }, + }, + }) + c.module.Memory.Memories = nil + + return nil +} + +// emitABIVersionGLobals adds globals for ABI [minor] version, exports them +func (c *Compiler) emitABIVersionGlobals() error { + abiVersionGlobals := []module.Global{ + { + Type: types.I32, + Mutable: false, + Init: module.Expr{ + Instrs: []instruction.Instruction{ + instruction.I32Const{Value: opaWasmABIVersionVal}, + }, + }, + }, + { + Type: types.I32, + Mutable: false, + Init: module.Expr{ + Instrs: []instruction.Instruction{ + instruction.I32Const{Value: opaWasmABIMinorVersionVal}, + }, + }, + }, + } + abiVersionExports := []module.Export{ + { + Name: opaWasmABIVersionVar, + Descriptor: module.ExportDescriptor{ + Type: module.GlobalExportType, + Index: uint32(len(c.module.Global.Globals)), + }, + }, + { + Name: opaWasmABIMinorVersionVar, + Descriptor: module.ExportDescriptor{ + Type: module.GlobalExportType, + Index: uint32(len(c.module.Global.Globals)) + 1, + }, + }, + } + c.module.Global.Globals = append(c.module.Global.Globals, abiVersionGlobals...) + c.module.Export.Exports = append(c.module.Export.Exports, abiVersionExports...) + return nil +} + +// compileStringsAndBooleans compiles various string constants (strings, file names, +// external function names, entrypoint names, builtin names), and interned opa_value structs +// for strings and booleans into the data section of the module. +// All are indexed for lookups in later stages. +func (c *Compiler) compileStringsAndBooleans() error { + + var err error + c.stringOffset, err = getLowestFreeDataSegmentOffset(c.module) + if err != nil { + return err + } + + var buf bytes.Buffer + + c.writeStrings(&buf) + + if err := c.writeInternedOPAValues(&buf); err != nil { + return err + } + + c.writeFileAddrs(&buf) + c.writeExternalFuncNames(&buf) + c.writeEntrypointNames(&buf) + c.writeBuiltinStrings(&buf) + + c.module.Data.Segments = append(c.module.Data.Segments, module.DataSegment{ + Index: 0, + Offset: module.Expr{ + Instrs: []instruction.Instruction{ + instruction.I32Const{ + Value: c.stringOffset, + }, + }, + }, + Init: buf.Bytes(), + }) + + return nil +} + +func (c *Compiler) writeStrings(buf *bytes.Buffer) { + c.stringAddrs = make([]uint32, len(c.policy.Static.Strings)) + + for i, s := range c.policy.Static.Strings { + addr := uint32(buf.Len()) + uint32(c.stringOffset) + buf.WriteString(s.Value) + buf.WriteByte(0) + c.stringAddrs[i] = addr + } +} + +func (c *Compiler) writeInternedOPAValues(buf *bytes.Buffer) error { + // interned `opa_value*` for these true/false booleans + c.opaBoolAddrs = make(map[ir.Bool]uint32, 2) + for _, val := range []bool{true, false} { + opaBool := ir.Bool(val) + v := byte(0) + if val { + v = 1 + } + c.opaBoolAddrs[opaBool] = uint32(buf.Len()) + uint32(c.stringOffset) + size := 2 + n, err := buf.Write([]byte{byte(opaTypeBooleanInterned), v}) + if err != nil { + return fmt.Errorf("write interned bools: %w", err) + } + if n != size { + return fmt.Errorf("short write: %d (expected %d)", n, size) + } + } + + // interned `opa_value*` for these constant strings + c.opaStringAddrs = make([]uint32, len(c.policy.Static.Strings)) + for i, s := range c.policy.Static.Strings { + c.opaStringAddrs[i] = uint32(buf.Len()) + uint32(c.stringOffset) + size := 12 + b := make([]byte, size) + binary.LittleEndian.PutUint16(b[0:], uint16(opaTypeStringInterned)) + binary.LittleEndian.PutUint32(b[4:], uint32(len(s.Value))) + binary.LittleEndian.PutUint32(b[8:], c.stringAddrs[i]) + n, err := buf.Write(b) + if err != nil { + return fmt.Errorf("write interned strings: %w", err) + } + if n != size { + return fmt.Errorf("short write: %d (expected %d)", n, size) + } + } + return nil +} + +func (c *Compiler) writeFileAddrs(buf *bytes.Buffer) { + // NOTE(sr): All files that have been consulted in planning are recorded, + // regardless of their potential in generating runtime errors. + c.fileAddrs = make([]uint32, len(c.policy.Static.Files)) + + for i, file := range c.policy.Static.Files { + addr := uint32(buf.Len()) + uint32(c.stringOffset) + buf.WriteString(file.Value) + buf.WriteByte(0) + c.fileAddrs[i] = addr + } +} + +func (c *Compiler) writeExternalFuncNames(buf *bytes.Buffer) { + c.externalFuncNameAddrs = make(map[string]int32) + + for _, decl := range c.policy.Static.BuiltinFuncs { + if _, ok := builtinsFunctions[decl.Name]; !ok { + addr := int32(buf.Len()) + int32(c.stringOffset) + buf.WriteString(decl.Name) + buf.WriteByte(0) + c.externalFuncNameAddrs[decl.Name] = addr + } + } +} + +func (c *Compiler) writeEntrypointNames(buf *bytes.Buffer) { + c.entrypointNameAddrs = make(map[string]int32) + + for _, plan := range c.policy.Plans.Plans { + addr := int32(buf.Len()) + int32(c.stringOffset) + buf.WriteString(plan.Name) + buf.WriteByte(0) + c.entrypointNameAddrs[plan.Name] = addr + } +} + +func (c *Compiler) writeBuiltinStrings(buf *bytes.Buffer) { + c.builtinStringAddrs = make(map[int]uint32, len(errorMessages)) + + for i := range errorMessages { + addr := uint32(buf.Len()) + uint32(c.stringOffset) + buf.WriteString(errorMessages[i].message) + buf.WriteByte(0) + c.builtinStringAddrs[errorMessages[i].id] = addr + } +} + +// compileExternalFuncDecls generates a function that lists the built-ins required by +// the policy. The host environment should invoke this function obtain the list +// of built-in function identifiers (represented as integers) that will be used +// when calling out. +func (c *Compiler) compileExternalFuncDecls() error { + + c.code = &module.CodeEntry{} + c.nextLocal = 0 + c.locals = map[ir.Local]uint32{} + + lobj := c.genLocal() + + c.appendInstr(instruction.Call{Index: c.function(opaObject)}) + c.appendInstr(instruction.SetLocal{Index: lobj}) + c.externalFuncs = make(map[string]externalFunc) + + for index, decl := range c.policy.Static.BuiltinFuncs { + if _, ok := builtinsFunctions[decl.Name]; !ok { + c.appendInstr(instruction.GetLocal{Index: lobj}) + c.appendInstr(instruction.I32Const{Value: c.externalFuncNameAddrs[decl.Name]}) + c.appendInstr(instruction.Call{Index: c.function(opaStringTerminated)}) + c.appendInstr(instruction.I64Const{Value: int64(index)}) + c.appendInstr(instruction.Call{Index: c.function(opaNumberInt)}) + c.appendInstr(instruction.Call{Index: c.function(opaObjectInsert)}) + c.externalFuncs[decl.Name] = externalFunc{ID: int32(index), Decl: decl.Decl} + } + } + + c.appendInstr(instruction.GetLocal{Index: lobj}) + + c.code.Func.Locals = []module.LocalDeclaration{ + { + Count: c.nextLocal, + Type: types.I32, + }, + } + + return c.storeFunc("builtins", c.code) +} + +// compileEntrypointDecls generates a function that lists the entrypoints available +// in the policy. The host environment can pick which entrypoint to invoke by setting +// the entrypoint identifier (represented as an integer) on the evaluation context. +func (c *Compiler) compileEntrypointDecls() error { + + c.code = &module.CodeEntry{} + c.nextLocal = 0 + c.locals = map[ir.Local]uint32{} + + lobj := c.genLocal() + + c.appendInstr(instruction.Call{Index: c.function(opaObject)}) + c.appendInstr(instruction.SetLocal{Index: lobj}) + c.entrypoints = make(map[string]int32) + + for index, plan := range c.policy.Plans.Plans { + c.appendInstr(instruction.GetLocal{Index: lobj}) + c.appendInstr(instruction.I32Const{Value: c.entrypointNameAddrs[plan.Name]}) + c.appendInstr(instruction.Call{Index: c.function(opaStringTerminated)}) + c.appendInstr(instruction.I64Const{Value: int64(index)}) + c.appendInstr(instruction.Call{Index: c.function(opaNumberInt)}) + c.appendInstr(instruction.Call{Index: c.function(opaObjectInsert)}) + c.entrypoints[plan.Name] = int32(index) + } + + c.appendInstr(instruction.GetLocal{Index: lobj}) + + c.code.Func.Locals = []module.LocalDeclaration{ + { + Count: c.nextLocal, + Type: types.I32, + }, + } + + return c.storeFunc("entrypoints", c.code) +} + +// compileFuncs compiles the policy functions and emits them into the module. +func (c *Compiler) compileFuncs() error { + for _, fn := range c.policy.Funcs.Funcs { + if err := c.compileFunc(fn); err != nil { + return fmt.Errorf("func %v: %w", fn.Name, err) + } + } + + if err := c.emitMappingAndStartFunc(); err != nil { + return fmt.Errorf("writing mapping: %w", err) + } + + if err := c.replaceBooleanFunc(); err != nil { + return fmt.Errorf("replacing opa_boolean: %w", err) + } + return nil +} + +// compilePlans compiles the policy plans and emits the resulting function into +// the module. +func (c *Compiler) compilePlans() error { + + c.code = &module.CodeEntry{} + c.nextLocal = 0 + c.locals = map[ir.Local]uint32{} + c.lctx = c.genLocal() + c.lrs = c.genLocal() + + // Initialize memoization. + c.appendInstr(instruction.Call{Index: c.function(opaMemoizeInit)}) + + // Initialize the input and data locals. + c.appendInstr(instruction.GetLocal{Index: c.lctx}) + c.appendInstr(instruction.I32Load{Offset: 0, Align: 2}) + c.appendInstr(instruction.SetLocal{Index: c.local(ir.Input)}) + + c.appendInstr(instruction.GetLocal{Index: c.lctx}) + c.appendInstr(instruction.I32Load{Offset: 4, Align: 2}) + c.appendInstr(instruction.SetLocal{Index: c.local(ir.Data)}) + + // Initialize the result set. + c.appendInstr(instruction.Call{Index: c.function(opaSet)}) + c.appendInstr(instruction.SetLocal{Index: c.lrs}) + c.appendInstr(instruction.GetLocal{Index: c.lctx}) + c.appendInstr(instruction.GetLocal{Index: c.lrs}) + c.appendInstr(instruction.I32Store{Offset: 8, Align: 2}) + + // Initialize the entrypoint id local. + leid := c.genLocal() + c.appendInstr(instruction.GetLocal{Index: c.lctx}) + c.appendInstr(instruction.I32Load{Offset: 12, Align: 2}) + c.appendInstr(instruction.SetLocal{Index: leid}) + + // Add each entrypoint to this block. + main := instruction.Block{} + + for i, plan := range c.policy.Plans.Plans { + + entrypoint := instruction.Block{ + Instrs: []instruction.Instruction{ + instruction.GetLocal{Index: leid}, + instruction.I32Const{Value: int32(i)}, + instruction.I32Ne{}, + instruction.BrIf{Index: 0}, + }, + } + + for j, block := range plan.Blocks { + + instrs, err := c.compileBlock(block) + if err != nil { + return fmt.Errorf("plan %d block %d: %w", i, j, err) + } + + entrypoint.Instrs = append(entrypoint.Instrs, instruction.Block{ + Instrs: instrs, + }) + } + + entrypoint.Instrs = append(entrypoint.Instrs, instruction.Br{Index: 1}) + main.Instrs = append(main.Instrs, entrypoint) + } + + // If none of the entrypoint blocks execute, call opa_abort() as this likely + // indicates inconsistency between the generated entrypoint identifiers in the + // eval() and entrypoint() functions (or the SDK invoked eval() with an invalid + // entrypoint ID which should not be possible.) + main.Instrs = append(main.Instrs, + instruction.I32Const{Value: c.builtinStringAddr(errIllegalEntrypoint)}, + instruction.Call{Index: c.function(opaAbort)}, + instruction.Unreachable{}, + ) + + c.appendInstr(main) + c.appendInstr(instruction.I32Const{Value: int32(0)}) + + c.code.Func.Locals = []module.LocalDeclaration{ + { + Count: c.nextLocal, + Type: types.I32, + }, + } + + return c.storeFunc("eval", c.code) +} + +func (c *Compiler) compileFunc(fn *ir.Func) error { + idx, ok := c.funcs[fn.Name] + if !ok { + return fmt.Errorf("unknown function: %v", fn.Name) + } + + memoize := len(fn.Params) == 2 + + if len(fn.Params) == 0 { + return fmt.Errorf("illegal function: zero args") + } + + c.nextLocal = 0 + c.locals = map[ir.Local]uint32{} + + for _, a := range fn.Params { + _ = c.local(a) + } + + _ = c.local(fn.Return) + + c.code = &module.CodeEntry{} + + // memoization: get + if memoize { + c.appendInstr(instruction.I32Const{Value: int32(idx)}) + c.appendInstr(instruction.Call{Index: c.function(opaMemoizeGet)}) + c.appendInstr(instruction.TeeLocal{Index: c.local(fn.Return)}) + c.appendInstr(instruction.If{Instrs: []instruction.Instruction{ + instruction.GetLocal{Index: c.local(fn.Return)}, + instruction.Return{}, + }}) + } + + for i := range fn.Blocks { + instrs, err := c.compileBlock(fn.Blocks[i]) + if err != nil { + return errors.Wrapf(err, "block %d", i) + } + if i < len(fn.Blocks)-1 { // not the last block: wrap in `block` instr + if withControlInstr(instrs) { // unless we don't need to + c.appendInstr(instruction.Block{Instrs: instrs}) + } else { + c.appendInstrs(instrs) + } + } else { // last block, no wrapping + // memoization: insert, spliced into the instructions right + // before the return: + for _, instr := range instrs { + if _, ok := instr.(instruction.Return); ok && memoize { + c.appendInstr(instruction.I32Const{Value: int32(idx)}) + c.appendInstr(instruction.GetLocal{Index: c.local(fn.Return)}) + c.appendInstr(instruction.Call{Index: c.function(opaMemoizeInsert)}) + } + c.appendInstr(instr) + } + } + } + + c.code.Func.Locals = []module.LocalDeclaration{ + { + Count: c.nextLocal, + Type: types.I32, + }, + } + + return c.storeFunc(fn.Name, c.code) +} + +func mapFunc(mapping ast.Object, fn *ir.Func, index int) (ast.Object, bool) { + curr := ast.NewObject() + curr.Insert(ast.StringTerm(fn.Path[len(fn.Path)-1]), ast.IntNumberTerm(index)) + for i := len(fn.Path) - 2; i >= 0; i-- { + o := ast.NewObject() + o.Insert(ast.StringTerm(fn.Path[i]), ast.NewTerm(curr)) + curr = o + } + return mapping.Merge(curr) +} + +func (c *Compiler) emitMappingAndStartFunc() error { + var indices []uint32 + var ok bool + mapping := ast.NewObject() + + // element segment offset for our mapped function entries + elemOffset, err := getLowestFreeElementSegmentOffset(c.module) + if err != nil { + return err + } + + for i, fn := range c.policy.Funcs.Funcs { + indices = append(indices, c.funcs[fn.Name]) + mapping, ok = mapFunc(mapping, fn, i+int(elemOffset)) + if !ok { + return fmt.Errorf("mapping function %v failed", fn.Name) + } + } + + // emit data segment for JSON blob encoding mapping + jsonMap := []byte(mapping.String()) + dataOffset, err := getLowestFreeDataSegmentOffset(c.module) + if err != nil { + return err + } + c.module.Data.Segments = append(c.module.Data.Segments, module.DataSegment{ + Index: 0, + Offset: module.Expr{ + Instrs: []instruction.Instruction{ + instruction.I32Const{ + Value: dataOffset, + }, + }, + }, + Init: jsonMap, + }) + + // write element segments for table entries + c.module.Element.Segments = append(c.module.Element.Segments, module.ElementSegment{ + Index: 0, + Offset: module.Expr{ + Instrs: []instruction.Instruction{ + instruction.I32Const{ + Value: elemOffset, + }, + }, + }, + Indices: indices, + }) + + // adjust table limits + min := c.module.Table.Tables[0].Lim.Min + uint32(len(indices)) + max := *c.module.Table.Tables[0].Lim.Max + uint32(len(indices)) + c.module.Table.Tables[0].Lim.Min = min + c.module.Table.Tables[0].Lim.Max = &max + + heapBase, err := getLowestFreeDataSegmentOffset(c.module) + if err != nil { + return err + } + + // create function that calls `void opa_mapping_initialize(const char *s, const int l)` + // with s being the offset of the data segment just written, and l its length + fName := "_initialize" + c.code = &module.CodeEntry{} + c.appendInstr(instruction.I32Const{Value: heapBase}) + c.appendInstr(instruction.Call{Index: c.function(opaMallocInit)}) + c.appendInstr(instruction.Call{Index: c.function(opaMPDInit)}) + c.appendInstr(instruction.I32Const{Value: dataOffset}) + c.appendInstr(instruction.I32Const{Value: int32(len(jsonMap))}) + c.appendInstr(instruction.Call{Index: c.function(opaMappingInit)}) + c.emitFunctionDecl(fName, module.FunctionType{}, false) + idx := c.function(fName) + c.module.Start.FuncIndex = &idx + return c.storeFunc(fName, c.code) +} + +// replaceBooleanFunc finds the `opa_boolean` code section, and replaces it with +// a simpler function, that's returning one of the interned `opa_boolean_t`s +// instead. +// NOTE(sr): We're doing it in this crude way because LLVM 11 doesn't let us +// differentiate that some unknown symbols (opa_abort, opa_builtin0, etc) should +// be created as imports, and other should be ignored. So, we're having a stub +// implementation in wasm/src/value.c that'll get replaced here. +func (c *Compiler) replaceBooleanFunc() error { + c.code = &module.CodeEntry{} + c.appendInstr(instruction.I32Const{Value: int32(c.opaBoolAddrs[true])}) + c.appendInstr(instruction.I32Const{Value: int32(c.opaBoolAddrs[false])}) + c.appendInstr(instruction.GetLocal{Index: 0}) + c.appendInstr(instruction.Select{}) + + return c.storeFunc(opaBoolean, c.code) +} + +func (c *Compiler) compileBlock(block *ir.Block) ([]instruction.Instruction, error) { + + var instrs []instruction.Instruction + + for _, stmt := range block.Stmts { + switch stmt := stmt.(type) { + case *ir.ResultSetAdd: + instrs = append(instrs, instruction.GetLocal{Index: c.lrs}) + instrs = append(instrs, instruction.GetLocal{Index: c.local(stmt.Value)}) + instrs = append(instrs, instruction.Call{Index: c.function(opaSetAdd)}) + case *ir.ReturnLocalStmt: + instrs = append(instrs, instruction.GetLocal{Index: c.local(stmt.Source)}) + instrs = append(instrs, instruction.Return{}) + case *ir.BlockStmt: + for i := range stmt.Blocks { + block, err := c.compileBlock(stmt.Blocks[i]) + if err != nil { + return nil, err + } + if withControlInstr(block) { + instrs = append(instrs, instruction.Block{Instrs: block}) + } else { + instrs = append(instrs, block...) + } + } + case *ir.BreakStmt: + instrs = append(instrs, instruction.Br{Index: stmt.Index}) + case *ir.CallStmt: + if err := c.compileCallStmt(stmt, &instrs); err != nil { + return nil, err + } + case *ir.CallDynamicStmt: + if err := c.compileCallDynamicStmt(stmt, &instrs); err != nil { + return nil, err + } + case *ir.WithStmt: + if err := c.compileWithStmt(stmt, &instrs); err != nil { + return instrs, err + } + case *ir.AssignVarStmt: + instrs = append(instrs, c.instrRead(stmt.Source)) + instrs = append(instrs, instruction.SetLocal{Index: c.local(stmt.Target)}) + case *ir.AssignVarOnceStmt: + instrs = append(instrs, instruction.Block{ + Instrs: []instruction.Instruction{ + instruction.Block{ + Instrs: append([]instruction.Instruction{ + instruction.GetLocal{Index: c.local(stmt.Target)}, + instruction.I32Eqz{}, + instruction.BrIf{Index: 0}, + instruction.GetLocal{Index: c.local(stmt.Target)}, + c.instrRead(stmt.Source), + instruction.Call{Index: c.function(opaValueCompare)}, + instruction.I32Eqz{}, + instruction.BrIf{Index: 1}, + }, + c.runtimeErrorAbort(stmt.Location, errVarAssignConflict)...), + }, + c.instrRead(stmt.Source), + instruction.SetLocal{Index: c.local(stmt.Target)}, + }, + }) + case *ir.AssignIntStmt: + instrs = append(instrs, instruction.GetLocal{Index: c.local(stmt.Target)}) + instrs = append(instrs, instruction.I64Const{Value: stmt.Value}) + instrs = append(instrs, instruction.Call{Index: c.function(opaValueNumberSetInt)}) + case *ir.ScanStmt: + if err := c.compileScan(stmt, &instrs); err != nil { + return nil, err + } + case *ir.NopStmt: + instrs = append(instrs, instruction.Nop{}) + case *ir.NotStmt: + if err := c.compileNot(stmt, &instrs); err != nil { + return nil, err + } + case *ir.DotStmt: + if loc, ok := stmt.Source.(ir.Local); ok { + instrs = append(instrs, instruction.GetLocal{Index: c.local(loc)}) + instrs = append(instrs, c.instrRead(stmt.Key)) + instrs = append(instrs, instruction.Call{Index: c.function(opaValueGet)}) + instrs = append(instrs, instruction.TeeLocal{Index: c.local(stmt.Target)}) + instrs = append(instrs, instruction.I32Eqz{}) + instrs = append(instrs, instruction.BrIf{Index: 0}) + } else { + // Booleans and strings would lead to the BrIf (since opa_value_get + // on them returns 0), so let's skip that. + instrs = append(instrs, instruction.Br{Index: 0}) + break + } + case *ir.LenStmt: + instrs = append(instrs, c.instrRead(stmt.Source)) + instrs = append(instrs, instruction.Call{Index: c.function(opaValueLength)}) + instrs = append(instrs, instruction.Call{Index: c.function(opaNumberSize)}) + instrs = append(instrs, instruction.SetLocal{Index: c.local(stmt.Target)}) + case *ir.EqualStmt: + if stmt.A != stmt.B { // constants, or locals, being equal here can skip the check + instrs = append(instrs, c.instrRead(stmt.A)) + instrs = append(instrs, c.instrRead(stmt.B)) + instrs = append(instrs, instruction.Call{Index: c.function(opaValueCompare)}) + instrs = append(instrs, instruction.BrIf{Index: 0}) + } + case *ir.NotEqualStmt: + if stmt.A == stmt.B { // same local, same bool constant, or same string constant + instrs = append(instrs, instruction.Br{Index: 0}) + continue + } + _, okA := stmt.A.(ir.Bool) + if _, okB := stmt.B.(ir.Bool); okA && okB { + // not equal (checked above), but both booleans => not equal + continue + } + _, okA = stmt.A.(ir.StringIndex) + if _, okB := stmt.B.(ir.StringIndex); okA && okB { + // not equal (checked above), but both strings => not equal + continue + } + instrs = append(instrs, c.instrRead(stmt.A)) + instrs = append(instrs, c.instrRead(stmt.B)) + instrs = append(instrs, instruction.Call{Index: c.function(opaValueCompare)}) + instrs = append(instrs, instruction.I32Eqz{}) + instrs = append(instrs, instruction.BrIf{Index: 0}) + case *ir.MakeNullStmt: + instrs = append(instrs, instruction.Call{Index: c.function(opaNull)}) + instrs = append(instrs, instruction.SetLocal{Index: c.local(stmt.Target)}) + case *ir.MakeNumberIntStmt: + instrs = append(instrs, instruction.I64Const{Value: stmt.Value}) + instrs = append(instrs, instruction.Call{Index: c.function(opaNumberInt)}) + instrs = append(instrs, instruction.SetLocal{Index: c.local(stmt.Target)}) + case *ir.MakeNumberRefStmt: + instrs = append(instrs, instruction.I32Const{Value: c.stringAddr(stmt.Index)}) + instrs = append(instrs, instruction.I32Const{Value: int32(len(c.policy.Static.Strings[stmt.Index].Value))}) + instrs = append(instrs, instruction.Call{Index: c.function(opaNumberRef)}) + instrs = append(instrs, instruction.SetLocal{Index: c.local(stmt.Target)}) + case *ir.MakeArrayStmt: + instrs = append(instrs, instruction.I32Const{Value: stmt.Capacity}) + instrs = append(instrs, instruction.Call{Index: c.function(opaArrayWithCap)}) + instrs = append(instrs, instruction.SetLocal{Index: c.local(stmt.Target)}) + case *ir.MakeObjectStmt: + instrs = append(instrs, instruction.Call{Index: c.function(opaObject)}) + instrs = append(instrs, instruction.SetLocal{Index: c.local(stmt.Target)}) + case *ir.MakeSetStmt: + instrs = append(instrs, instruction.Call{Index: c.function(opaSet)}) + instrs = append(instrs, instruction.SetLocal{Index: c.local(stmt.Target)}) + case *ir.IsArrayStmt: + if loc, ok := stmt.Source.(ir.Local); ok { + instrs = append(instrs, instruction.GetLocal{Index: c.local(loc)}) + instrs = append(instrs, instruction.Call{Index: c.function(opaValueType)}) + instrs = append(instrs, instruction.I32Const{Value: opaTypeArray}) + instrs = append(instrs, instruction.I32Ne{}) + instrs = append(instrs, instruction.BrIf{Index: 0}) + } else { + instrs = append(instrs, instruction.Br{Index: 0}) + break + } + case *ir.IsObjectStmt: + if loc, ok := stmt.Source.(ir.Local); ok { + instrs = append(instrs, instruction.GetLocal{Index: c.local(loc)}) + instrs = append(instrs, instruction.Call{Index: c.function(opaValueType)}) + instrs = append(instrs, instruction.I32Const{Value: opaTypeObject}) + instrs = append(instrs, instruction.I32Ne{}) + instrs = append(instrs, instruction.BrIf{Index: 0}) + } else { + instrs = append(instrs, instruction.Br{Index: 0}) + break + } + case *ir.IsUndefinedStmt: + instrs = append(instrs, instruction.GetLocal{Index: c.local(stmt.Source)}) + instrs = append(instrs, instruction.I32Const{Value: 0}) + instrs = append(instrs, instruction.I32Ne{}) + instrs = append(instrs, instruction.BrIf{Index: 0}) + case *ir.ResetLocalStmt: + instrs = append(instrs, instruction.I32Const{Value: 0}) + instrs = append(instrs, instruction.SetLocal{Index: c.local(stmt.Target)}) + case *ir.IsDefinedStmt: + instrs = append(instrs, instruction.GetLocal{Index: c.local(stmt.Source)}) + instrs = append(instrs, instruction.I32Eqz{}) + instrs = append(instrs, instruction.BrIf{Index: 0}) + case *ir.ArrayAppendStmt: + instrs = append(instrs, instruction.GetLocal{Index: c.local(stmt.Array)}) + instrs = append(instrs, c.instrRead(stmt.Value)) + instrs = append(instrs, instruction.Call{Index: c.function(opaArrayAppend)}) + case *ir.ObjectInsertStmt: + instrs = append(instrs, instruction.GetLocal{Index: c.local(stmt.Object)}) + instrs = append(instrs, c.instrRead(stmt.Key)) + instrs = append(instrs, c.instrRead(stmt.Value)) + instrs = append(instrs, instruction.Call{Index: c.function(opaObjectInsert)}) + case *ir.ObjectInsertOnceStmt: + tmp := c.genLocal() + instrs = append(instrs, instruction.Block{ + Instrs: []instruction.Instruction{ + instruction.Block{ + Instrs: append([]instruction.Instruction{ + instruction.GetLocal{Index: c.local(stmt.Object)}, + c.instrRead(stmt.Key), + instruction.Call{Index: c.function(opaValueGet)}, + instruction.TeeLocal{Index: tmp}, + instruction.I32Eqz{}, + instruction.BrIf{Index: 0}, + instruction.GetLocal{Index: tmp}, + c.instrRead(stmt.Value), + instruction.Call{Index: c.function(opaValueCompare)}, + instruction.I32Eqz{}, + instruction.BrIf{Index: 1}, + }, c.runtimeErrorAbort(stmt.Location, errObjectInsertConflict)...), + }, + instruction.GetLocal{Index: c.local(stmt.Object)}, + c.instrRead(stmt.Key), + c.instrRead(stmt.Value), + instruction.Call{Index: c.function(opaObjectInsert)}, + }, + }) + case *ir.ObjectMergeStmt: + instrs = append(instrs, instruction.GetLocal{Index: c.local(stmt.A)}) + instrs = append(instrs, instruction.GetLocal{Index: c.local(stmt.B)}) + instrs = append(instrs, instruction.Call{Index: c.function(opaValueMerge)}) + instrs = append(instrs, instruction.SetLocal{Index: c.local(stmt.Target)}) + case *ir.SetAddStmt: + instrs = append(instrs, instruction.GetLocal{Index: c.local(stmt.Set)}) + instrs = append(instrs, c.instrRead(stmt.Value)) + instrs = append(instrs, instruction.Call{Index: c.function(opaSetAdd)}) + default: + var buf bytes.Buffer + err := ir.Pretty(&buf, stmt) + if err != nil { + return nil, err + } + return instrs, fmt.Errorf("illegal statement: %v", buf.String()) + } + } + + return instrs, nil +} + +func (c *Compiler) compileScan(scan *ir.ScanStmt, result *[]instruction.Instruction) error { + var instrs = *result + instrs = append(instrs, instruction.I32Const{Value: 0}) + instrs = append(instrs, instruction.SetLocal{Index: c.local(scan.Key)}) + body, err := c.compileScanBlock(scan) + if err != nil { + return err + } + instrs = append(instrs, instruction.Block{ + Instrs: []instruction.Instruction{ + instruction.Loop{Instrs: body}, + }, + }) + *result = instrs + return nil +} + +func (c *Compiler) compileScanBlock(scan *ir.ScanStmt) ([]instruction.Instruction, error) { + var instrs []instruction.Instruction + + // Execute iterator. + instrs = append(instrs, instruction.GetLocal{Index: c.local(scan.Source)}) + instrs = append(instrs, instruction.GetLocal{Index: c.local(scan.Key)}) + instrs = append(instrs, instruction.Call{Index: c.function(opaValueIter)}) + + // Check for emptiness. + instrs = append(instrs, instruction.TeeLocal{Index: c.local(scan.Key)}) + instrs = append(instrs, instruction.I32Eqz{}) + instrs = append(instrs, instruction.BrIf{Index: 1}) + + // Load value. + instrs = append(instrs, instruction.GetLocal{Index: c.local(scan.Source)}) + instrs = append(instrs, instruction.GetLocal{Index: c.local(scan.Key)}) + instrs = append(instrs, instruction.Call{Index: c.function(opaValueGet)}) + instrs = append(instrs, instruction.SetLocal{Index: c.local(scan.Value)}) + + // Loop body. + nested, err := c.compileBlock(scan.Block) + if err != nil { + return nil, err + } + + // Continue. + instrs = append(instrs, nested...) + instrs = append(instrs, instruction.Br{Index: 0}) + + return instrs, nil +} + +func (c *Compiler) compileNot(not *ir.NotStmt, result *[]instruction.Instruction) error { + var instrs = *result + + // generate and initialize condition variable + cond := c.genLocal() + instrs = append(instrs, instruction.I32Const{Value: 1}) + instrs = append(instrs, instruction.SetLocal{Index: cond}) + + nested, err := c.compileBlock(not.Block) + if err != nil { + return err + } + + // unset condition variable if end of block is reached + nested = append(nested, instruction.I32Const{Value: 0}) + nested = append(nested, instruction.SetLocal{Index: cond}) + instrs = append(instrs, instruction.Block{Instrs: nested}) + + // break out of block if condition variable was unset + instrs = append(instrs, instruction.GetLocal{Index: cond}) + instrs = append(instrs, instruction.I32Eqz{}) + instrs = append(instrs, instruction.BrIf{Index: 0}) + + *result = instrs + return nil +} + +func (c *Compiler) compileWithStmt(with *ir.WithStmt, result *[]instruction.Instruction) error { + + var instrs = *result + save := c.genLocal() + instrs = append(instrs, instruction.Call{Index: c.function(opaMemoizePush)}) + instrs = append(instrs, instruction.GetLocal{Index: c.local(with.Local)}) + instrs = append(instrs, instruction.SetLocal{Index: save}) + + if len(with.Path) == 0 { + instrs = append(instrs, c.instrRead(with.Value)) + instrs = append(instrs, instruction.SetLocal{Index: c.local(with.Local)}) + } else { + instrs = c.compileUpsert(with.Local, with.Path, with.Value, with.Location, instrs) + } + + undefined := c.genLocal() + instrs = append(instrs, instruction.I32Const{Value: 1}) + instrs = append(instrs, instruction.SetLocal{Index: undefined}) + + nested, err := c.compileBlock(with.Block) + if err != nil { + return err + } + + nested = append(nested, instruction.I32Const{Value: 0}) + nested = append(nested, instruction.SetLocal{Index: undefined}) + instrs = append(instrs, instruction.Block{Instrs: nested}) + instrs = append(instrs, instruction.GetLocal{Index: save}) + instrs = append(instrs, instruction.SetLocal{Index: c.local(with.Local)}) + instrs = append(instrs, instruction.Call{Index: c.function(opaMemoizePop)}) + instrs = append(instrs, instruction.GetLocal{Index: undefined}) + instrs = append(instrs, instruction.BrIf{Index: 0}) + + *result = instrs + + return nil +} + +func (c *Compiler) compileUpsert(local ir.Local, path []int, value ir.LocalOrConst, loc ir.Location, instrs []instruction.Instruction) []instruction.Instruction { + + lcopy := c.genLocal() // holds copy of local + instrs = append(instrs, instruction.GetLocal{Index: c.local(local)}) + instrs = append(instrs, instruction.SetLocal{Index: lcopy}) + + // Shallow copy the local if defined otherwise initialize to an empty object. + instrs = append(instrs, instruction.Block{ + Instrs: []instruction.Instruction{ + instruction.Block{Instrs: []instruction.Instruction{ + instruction.GetLocal{Index: lcopy}, + instruction.I32Eqz{}, + instruction.BrIf{Index: 0}, + instruction.GetLocal{Index: lcopy}, + instruction.Call{Index: c.function(opaValueShallowCopy)}, + instruction.TeeLocal{Index: lcopy}, + instruction.SetLocal{Index: c.local(local)}, + instruction.Br{Index: 1}, + }}, + instruction.Call{Index: c.function(opaObject)}, + instruction.TeeLocal{Index: lcopy}, + instruction.SetLocal{Index: c.local(local)}, + }, + }) + + // Initialize the locals that specify the path of the upsert operation. + lpath := make(map[int]uint32, len(path)) + + for i := 0; i < len(path); i++ { + lpath[i] = c.genLocal() + instrs = append(instrs, instruction.I32Const{Value: c.opaStringAddr(path[i])}) + instrs = append(instrs, instruction.SetLocal{Index: lpath[i]}) + } + + // Generate a block that traverses the path of the upsert operation, + // shallowing copying values at each step as needed. Stop before the final + // segment that will only be inserted. + var inner []instruction.Instruction + ltemp := c.genLocal() + + for i := 0; i < len(path)-1; i++ { + + // Lookup the next part of the path. + inner = append(inner, instruction.GetLocal{Index: lcopy}) + inner = append(inner, instruction.GetLocal{Index: lpath[i]}) + inner = append(inner, instruction.Call{Index: c.function(opaValueGet)}) + inner = append(inner, instruction.SetLocal{Index: ltemp}) + + // If the next node is missing, break. + inner = append(inner, instruction.GetLocal{Index: ltemp}) + inner = append(inner, instruction.I32Eqz{}) + inner = append(inner, instruction.BrIf{Index: uint32(i)}) + + // If the next node is not an object, break. + inner = append(inner, instruction.GetLocal{Index: ltemp}) + inner = append(inner, instruction.Call{Index: c.function(opaValueType)}) + inner = append(inner, instruction.I32Const{Value: opaTypeObject}) + inner = append(inner, instruction.I32Ne{}) + inner = append(inner, instruction.BrIf{Index: uint32(i)}) + + // Otherwise, shallow copy the next node node and insert into the copy + // before continuing. + inner = append(inner, instruction.GetLocal{Index: ltemp}) + inner = append(inner, instruction.Call{Index: c.function(opaValueShallowCopy)}) + inner = append(inner, instruction.SetLocal{Index: ltemp}) + inner = append(inner, instruction.GetLocal{Index: lcopy}) + inner = append(inner, instruction.GetLocal{Index: lpath[i]}) + inner = append(inner, instruction.GetLocal{Index: ltemp}) + inner = append(inner, instruction.Call{Index: c.function(opaObjectInsert)}) + inner = append(inner, instruction.GetLocal{Index: ltemp}) + inner = append(inner, instruction.SetLocal{Index: lcopy}) + } + + inner = append(inner, instruction.Br{Index: uint32(len(path) - 1)}) + + // Generate blocks that handle missing nodes during traversal. + var block []instruction.Instruction + lval := c.genLocal() + + for i := 0; i < len(path)-1; i++ { + block = append(block, instruction.Block{Instrs: inner}) + block = append(block, instruction.Call{Index: c.function(opaObject)}) + block = append(block, instruction.SetLocal{Index: lval}) + block = append(block, instruction.GetLocal{Index: lcopy}) + block = append(block, instruction.GetLocal{Index: lpath[i]}) + block = append(block, instruction.GetLocal{Index: lval}) + block = append(block, instruction.Call{Index: c.function(opaObjectInsert)}) + block = append(block, instruction.GetLocal{Index: lval}) + block = append(block, instruction.SetLocal{Index: lcopy}) + inner = block + block = nil + } + + // Finish by inserting the statement's value into the shallow copied node. + instrs = append(instrs, instruction.Block{Instrs: inner}) + instrs = append(instrs, instruction.GetLocal{Index: lcopy}) + instrs = append(instrs, instruction.GetLocal{Index: lpath[len(path)-1]}) + instrs = append(instrs, c.instrRead(value)) + instrs = append(instrs, instruction.Call{Index: c.function(opaObjectInsert)}) + + return instrs +} + +func (c *Compiler) compileCallDynamicStmt(stmt *ir.CallDynamicStmt, result *[]instruction.Instruction) error { + instrs := []instruction.Instruction{} + larray := c.genLocal() + lidx := c.genLocal() + + // init array: + instrs = append(instrs, + instruction.I32Const{Value: int32(len(stmt.Path))}, + instruction.Call{Index: c.function(opaArrayWithCap)}, + instruction.SetLocal{Index: larray}, + ) + + // append to it: + for _, lv := range stmt.Path { + instrs = append(instrs, + instruction.GetLocal{Index: larray}, + c.instrRead(lv), + instruction.Call{Index: c.function(opaArrayAppend)}, + ) + } + + // prep stack for later call_indirect + for _, arg := range stmt.Args { + instrs = append(instrs, instruction.GetLocal{Index: c.local(arg)}) + } + + tpe := module.FunctionType{ + Params: []types.ValueType{types.I32, types.I32}, // data, input + Results: []types.ValueType{types.I32}, + } + typeIndex := c.emitFunctionType(tpe) + + instrs = append(instrs, + // lookup elem idx via larray path + instruction.GetLocal{Index: larray}, + instruction.Call{Index: c.function(opaMappingLookup)}, // [arg0 arg1 larray] -> [arg0 arg1 tbl_idx] + instruction.TeeLocal{Index: lidx}, + instruction.I32Eqz{}, // mapping not found + instruction.BrIf{Index: 0}, // check data + + instruction.GetLocal{Index: lidx}, + instruction.CallIndirect{Index: typeIndex}, // [arg0 arg1 tbl_idx] -> [res] + instruction.TeeLocal{Index: c.local(stmt.Result)}, + instruction.I32Eqz{}, + instruction.BrIf{Index: 2}, // mapping found, "undefined" result counts + ) + + *result = append(*result, instrs...) + return nil +} + +func (c *Compiler) compileCallStmt(stmt *ir.CallStmt, result *[]instruction.Instruction) error { + + fn := stmt.Func + + if name, ok := builtinsFunctions[stmt.Func]; ok { + fn = name + } + + if index, ok := c.funcs[fn]; ok { + return c.compileInternalCall(stmt, index, result) + } + + if ef, ok := c.externalFuncs[fn]; ok { + return c.compileExternalCall(stmt, ef, result) + } + + c.errors = append(c.errors, fmt.Errorf("undefined function: %q", fn)) + + return nil +} + +func (c *Compiler) compileInternalCall(stmt *ir.CallStmt, index uint32, result *[]instruction.Instruction) error { + + instrs := []instruction.Instruction{} + + // Prepare function args and call. + for _, arg := range stmt.Args { + instrs = append(instrs, c.instrRead(arg)) + } + + instrs = append(instrs, + instruction.Call{Index: index}, + instruction.TeeLocal{Index: c.local(stmt.Result)}, + instruction.I32Eqz{}, + instruction.BrIf{Index: 0}) + + *result = append(*result, instrs...) + + return nil +} + +func (c *Compiler) compileExternalCall(stmt *ir.CallStmt, ef externalFunc, result *[]instruction.Instruction) error { + + if len(stmt.Args) >= len(builtinDispatchers) { + c.errors = append(c.errors, fmt.Errorf("too many built-in call arguments: %q", stmt.Func)) + return nil + } + + instrs := *result + instrs = append(instrs, instruction.I32Const{Value: ef.ID}) + instrs = append(instrs, instruction.I32Const{Value: 0}) // unused context parameter + + for _, arg := range stmt.Args { + instrs = append(instrs, c.instrRead(arg)) + } + + instrs = append(instrs, instruction.Call{Index: c.function(builtinDispatchers[len(stmt.Args)])}) + + if ef.Decl.Result() != nil { + instrs = append(instrs, instruction.TeeLocal{Index: c.local(stmt.Result)}) + instrs = append(instrs, instruction.I32Eqz{}) + instrs = append(instrs, instruction.BrIf{Index: 0}) + } else { + instrs = append(instrs, instruction.Drop{}) + } + + *result = instrs + return nil +} + +func (c *Compiler) emitFunctionDecl(name string, tpe module.FunctionType, export bool) { + + var idx uint32 + if old, ok := c.funcs[name]; ok { + c.debug.Printf("function declaration for %v is being emitted multiple times (overwriting old index %d)", name, old) + idx = old + } else { + typeIndex := c.emitFunctionType(tpe) + c.module.Function.TypeIndices = append(c.module.Function.TypeIndices, typeIndex) + c.module.Code.Segments = append(c.module.Code.Segments, module.RawCodeSegment{}) + idx = uint32((len(c.module.Function.TypeIndices) - 1) + c.functionImportCount()) + c.funcs[name] = idx + } + + if export { + c.module.Export.Exports = append(c.module.Export.Exports, module.Export{ + Name: name, + Descriptor: module.ExportDescriptor{ + Type: module.FunctionExportType, + Index: idx, + }, + }) + } + + // add functions 'name' entry + var found bool + for _, m := range c.module.Names.Functions { + if m.Index == idx { + found = true + } + } + if !found { + c.module.Names.Functions = append(c.module.Names.Functions, module.NameMap{ + Index: idx, + Name: name, + }) + } +} + +func (c *Compiler) emitFunctionType(tpe module.FunctionType) uint32 { + for i, other := range c.module.Type.Functions { + if tpe.Equal(other) { + return uint32(i) + } + } + c.module.Type.Functions = append(c.module.Type.Functions, tpe) + return uint32(len(c.module.Type.Functions) - 1) +} + +func (c *Compiler) emitFunction(name string, entry *module.CodeEntry) error { + var buf bytes.Buffer + if err := encoding.WriteCodeEntry(&buf, entry); err != nil { + return err + } + index := c.function(name) - uint32(c.functionImportCount()) + c.module.Code.Segments[index].Code = buf.Bytes() + return nil +} + +// emitFuncs writes the compiled (and optimized) functions' code into the +// module +func (c *Compiler) emitFuncs() error { + for _, fn := range c.funcsCode { + if err := c.emitFunction(fn.name, fn.code); err != nil { + return fmt.Errorf("write function %s: %w", fn.name, err) + } + } + return nil +} + +func (c *Compiler) functionImportCount() int { + var count int + + for _, imp := range c.module.Import.Imports { + if imp.Descriptor.Kind() == module.FunctionImportType { + count++ + } + } + + return count +} + +func (c *Compiler) stringAddr(index int) int32 { + return int32(c.stringAddrs[index]) +} + +func (c *Compiler) builtinStringAddr(code int) int32 { + return int32(c.builtinStringAddrs[code]) +} + +func (c *Compiler) opaStringAddr(index int) int32 { + return int32(c.opaStringAddrs[index]) +} + +func (c *Compiler) opaBoolAddr(b ir.Bool) int32 { + return int32(c.opaBoolAddrs[b]) +} + +func (c *Compiler) fileAddr(code int) int32 { + return int32(c.fileAddrs[code]) +} + +func (c *Compiler) local(l ir.Local) uint32 { + var u32 uint32 + var exist bool + if u32, exist = c.locals[l]; !exist { + u32 = c.nextLocal + c.locals[l] = u32 + c.nextLocal++ + } + return u32 +} + +func (c *Compiler) genLocal() uint32 { + l := c.nextLocal + c.nextLocal++ + return l +} + +func (c *Compiler) function(name string) uint32 { + fidx, ok := c.funcs[name] + if !ok { + panic(fmt.Sprintf("function not found: %s", name)) + } + return fidx +} + +func (c *Compiler) appendInstr(instr instruction.Instruction) { + c.code.Func.Expr.Instrs = append(c.code.Func.Expr.Instrs, instr) +} + +func (c *Compiler) appendInstrs(instrs []instruction.Instruction) { + for _, instr := range instrs { + c.appendInstr(instr) + } +} + +func getLowestFreeDataSegmentOffset(m *module.Module) (int32, error) { + + var offset int32 + + for i := range m.Data.Segments { + + if len(m.Data.Segments[i].Offset.Instrs) != 1 { + return 0, errors.New("bad data segment offset instructions") + } + + instr, ok := m.Data.Segments[i].Offset.Instrs[0].(instruction.I32Const) + if !ok { + return 0, errors.New("bad data segment offset expr") + } + + // NOTE(tsandall): assume memory up to but not including addr is taken. + addr := instr.Value + int32(len(m.Data.Segments[i].Init)) + if addr > offset { + offset = addr + } + } + + return offset, nil +} + +func getLowestFreeElementSegmentOffset(m *module.Module) (int32, error) { + var offset int32 + + for _, seg := range m.Element.Segments { + if len(seg.Offset.Instrs) != 1 { + return 0, errors.New("bad data segment offset instructions") + } + + instr, ok := seg.Offset.Instrs[0].(instruction.I32Const) + if !ok { + return 0, errors.New("bad data segment offset expr") + } + + addr := instr.Value + int32(len(seg.Indices)) + if addr > offset { + offset = addr + } + } + + return offset, nil +} + +// runtimeErrorAbort uses the passed source location to build the +// arguments for a call to opa_runtime_error(file, row, col, msg). +// It returns the instructions that make up the function call with +// arguments, followed by Unreachable. +func (c *Compiler) runtimeErrorAbort(loc ir.Location, errType int) []instruction.Instruction { + index, row, col := loc.Index, loc.Row, loc.Col + return []instruction.Instruction{ + instruction.I32Const{Value: c.fileAddr(index)}, + instruction.I32Const{Value: int32(row)}, + instruction.I32Const{Value: int32(col)}, + instruction.I32Const{Value: c.builtinStringAddr(errType)}, + instruction.Call{Index: c.function(opaRuntimeError)}, + instruction.Unreachable{}, + } +} + +func (c *Compiler) storeFunc(name string, code *module.CodeEntry) error { + for _, fn := range c.funcsCode { + if fn.name == name { + return fmt.Errorf("duplicate function entry %s", name) + } + } + c.funcsCode = append(c.funcsCode, funcCode{name: name, code: code}) + return nil +} + +func (c *Compiler) instrRead(lv ir.LocalOrConst) instruction.Instruction { + switch x := lv.(type) { + case ir.Bool: + return instruction.I32Const{Value: c.opaBoolAddr(x)} + case ir.StringIndex: + return instruction.I32Const{Value: c.opaStringAddr(int(x))} + case ir.Local: + return instruction.GetLocal{Index: c.local(x)} + } + panic("unreachable") +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/debug/debug.go b/vendor/github.com/open-policy-agent/opa/internal/debug/debug.go new file mode 100644 index 00000000..b89c6e71 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/debug/debug.go @@ -0,0 +1,36 @@ +package debug + +import ( + "io" + "io/ioutil" + "log" +) + +// Debug allows printing debug messages. +type Debug interface { + // Printf prints, with a short file:line-number prefix + Printf(format string, args ...interface{}) + // Writer returns the writer being written to, which may be + // `ioutil.Discard` if no debug output is requested. + Writer() io.Writer + + // Output allows tweaking the calldepth used for figuring + // out which Go source file location is the interesting one, + // i.e., which is included in the debug message. Useful for + // setting up local helper methods. + Output(calldepth int, s string) error +} + +// New returns a new `Debug` outputting to the passed `sink`. +func New(sink io.Writer) Debug { + flags := log.Lshortfile + return log.New(sink, "", flags) +} + +// Discard returns a new `Debug` that doesn't output anything. +// Note: We're not implementing the methods here with noop stubs +// since doing this way, we can propagate the "discarding" via +// `(Debug).Writer()`. +func Discard() Debug { + return New(ioutil.Discard) +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/deepcopy/deepcopy.go b/vendor/github.com/open-policy-agent/opa/internal/deepcopy/deepcopy.go new file mode 100644 index 00000000..cf8cc1cf --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/deepcopy/deepcopy.go @@ -0,0 +1,27 @@ +// Copyright 2020 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package deepcopy + +// DeepCopy performs a recursive deep copy for nested slices/maps and +// returns the copied object. Supports []interface{} +// and map[string]interface{} only +func DeepCopy(val interface{}) interface{} { + switch val := val.(type) { + case []interface{}: + cpy := make([]interface{}, len(val)) + for i := range cpy { + cpy[i] = DeepCopy(val[i]) + } + return cpy + case map[string]interface{}: + cpy := make(map[string]interface{}, len(val)) + for k := range val { + cpy[k] = DeepCopy(val[k]) + } + return cpy + default: + return val + } +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/file/archive/tarball.go b/vendor/github.com/open-policy-agent/opa/internal/file/archive/tarball.go new file mode 100644 index 00000000..6b8ba48d --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/file/archive/tarball.go @@ -0,0 +1,42 @@ +package archive + +import ( + "archive/tar" + "bytes" + "compress/gzip" + "strings" +) + +// MustWriteTarGz write the list of file names and content +// into a tarball. +func MustWriteTarGz(files [][2]string) *bytes.Buffer { + var buf bytes.Buffer + gw := gzip.NewWriter(&buf) + defer gw.Close() + tw := tar.NewWriter(gw) + defer tw.Close() + for _, file := range files { + if err := WriteFile(tw, file[0], []byte(file[1])); err != nil { + panic(err) + } + } + return &buf +} + +// WriteFile adds a file header with content to the given tar writer +func WriteFile(tw *tar.Writer, path string, bs []byte) error { + + hdr := &tar.Header{ + Name: "/" + strings.TrimLeft(path, "/"), + Mode: 0600, + Typeflag: tar.TypeReg, + Size: int64(len(bs)), + } + + if err := tw.WriteHeader(hdr); err != nil { + return err + } + + _, err := tw.Write(bs) + return err +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/file/url/url.go b/vendor/github.com/open-policy-agent/opa/internal/file/url/url.go new file mode 100644 index 00000000..aacc5718 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/file/url/url.go @@ -0,0 +1,42 @@ +// Copyright 2019 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +// Package url contains helpers for dealing with file paths and URLs. +package url + +import ( + "fmt" + "net/url" + "runtime" + "strings" +) + +var goos = runtime.GOOS + +// Clean returns a cleaned file path that may or may not be a URL. +func Clean(path string) (string, error) { + + if strings.Contains(path, "://") { + + url, err := url.Parse(path) + if err != nil { + return "", err + } + + if url.Scheme != "file" { + return "", fmt.Errorf("unsupported URL scheme: %v", path) + } + + path = url.Path + + // Trim leading slash on Windows if present. The url.Path field returned + // by url.Parse has leading slash that causes CreateFile() calls to fail + // on Windows. See https://github.com/golang/go/issues/6027 for details. + if goos == "windows" && len(path) >= 1 && path[0] == '/' { + path = path[1:] + } + } + + return path, nil +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/future/filter_imports.go b/vendor/github.com/open-policy-agent/opa/internal/future/filter_imports.go new file mode 100644 index 00000000..b0fdc8c6 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/future/filter_imports.go @@ -0,0 +1,20 @@ +// Copyright 2021 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package future + +import "github.com/open-policy-agent/opa/ast" + +// FilterFutureImports filters OUT any future imports from the passed slice of +// `*ast.Import`s. +func FilterFutureImports(imps []*ast.Import) []*ast.Import { + ret := []*ast.Import{} + for _, imp := range imps { + path := imp.Path.Value.(ast.Ref) + if !ast.FutureRootDocument.Equal(path[0]) { + ret = append(ret, imp) + } + } + return ret +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/future/parser_opts.go b/vendor/github.com/open-policy-agent/opa/internal/future/parser_opts.go new file mode 100644 index 00000000..804702b9 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/future/parser_opts.go @@ -0,0 +1,42 @@ +// Copyright 2021 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package future + +import ( + "fmt" + + "github.com/open-policy-agent/opa/ast" +) + +// ParserOptionsFromFutureImports transforms a slice of `ast.Import`s into the +// `ast.ParserOptions` that can be used to parse a statement according to the +// included "future.keywords" and "future.keywords.xyz" imports. +func ParserOptionsFromFutureImports(imports []*ast.Import) (ast.ParserOptions, error) { + popts := ast.ParserOptions{ + FutureKeywords: []string{}, + } + for _, imp := range imports { + path := imp.Path.Value.(ast.Ref) + if !ast.FutureRootDocument.Equal(path[0]) { + continue + } + if len(path) >= 2 { + if string(path[1].Value.(ast.String)) != "keywords" { + return popts, fmt.Errorf("unknown future import: %v", imp) + } + if len(path) == 2 { + // retun, one "future.keywords" import means we can disregard any others + return ast.ParserOptions{AllFutureKeywords: true}, nil + } + } + if len(path) == 3 { + if imp.Alias != "" { + return popts, fmt.Errorf("alias not supported") + } + popts.FutureKeywords = append(popts.FutureKeywords, string(path[2].Value.(ast.String))) + } + } + return popts, nil +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/gojsonschema/LICENSE-APACHE-2.0.txt b/vendor/github.com/open-policy-agent/opa/internal/gojsonschema/LICENSE-APACHE-2.0.txt new file mode 100644 index 00000000..55ede8a4 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/gojsonschema/LICENSE-APACHE-2.0.txt @@ -0,0 +1,202 @@ + + Apache License + Version 2.0, January 2004 + http://www.apache.org/licenses/ + + TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION + + 1. Definitions. + + "License" shall mean the terms and conditions for use, reproduction, + and distribution as defined by Sections 1 through 9 of this document. + + "Licensor" shall mean the copyright owner or entity authorized by + the copyright owner that is granting the License. + + "Legal Entity" shall mean the union of the acting entity and all + other entities that control, are controlled by, or are under common + control with that entity. For the purposes of this definition, + "control" means (i) the power, direct or indirect, to cause the + direction or management of such entity, whether by contract or + otherwise, or (ii) ownership of fifty percent (50%) or more of the + outstanding shares, or (iii) beneficial ownership of such entity. + + "You" (or "Your") shall mean an individual or Legal Entity + exercising permissions granted by this License. + + "Source" form shall mean the preferred form for making modifications, + including but not limited to software source code, documentation + source, and configuration files. + + "Object" form shall mean any form resulting from mechanical + transformation or translation of a Source form, including but + not limited to compiled object code, generated documentation, + and conversions to other media types. + + "Work" shall mean the work of authorship, whether in Source or + Object form, made available under the License, as indicated by a + copyright notice that is included in or attached to the work + (an example is provided in the Appendix below). + + "Derivative Works" shall mean any work, whether in Source or Object + form, that is based on (or derived from) the Work and for which the + editorial revisions, annotations, elaborations, or other modifications + represent, as a whole, an original work of authorship. For the purposes + of this License, Derivative Works shall not include works that remain + separable from, or merely link (or bind by name) to the interfaces of, + the Work and Derivative Works thereof. + + "Contribution" shall mean any work of authorship, including + the original version of the Work and any modifications or additions + to that Work or Derivative Works thereof, that is intentionally + submitted to Licensor for inclusion in the Work by the copyright owner + or by an individual or Legal Entity authorized to submit on behalf of + the copyright owner. For the purposes of this definition, "submitted" + means any form of electronic, verbal, or written communication sent + to the Licensor or its representatives, including but not limited to + communication on electronic mailing lists, source code control systems, + and issue tracking systems that are managed by, or on behalf of, the + Licensor for the purpose of discussing and improving the Work, but + excluding communication that is conspicuously marked or otherwise + designated in writing by the copyright owner as "Not a Contribution." + + "Contributor" shall mean Licensor and any individual or Legal Entity + on behalf of whom a Contribution has been received by Licensor and + subsequently incorporated within the Work. + + 2. Grant of Copyright License. Subject to the terms and conditions of + this License, each Contributor hereby grants to You a perpetual, + worldwide, non-exclusive, no-charge, royalty-free, irrevocable + copyright license to reproduce, prepare Derivative Works of, + publicly display, publicly perform, sublicense, and distribute the + Work and such Derivative Works in Source or Object form. + + 3. Grant of Patent License. Subject to the terms and conditions of + this License, each Contributor hereby grants to You a perpetual, + worldwide, non-exclusive, no-charge, royalty-free, irrevocable + (except as stated in this section) patent license to make, have made, + use, offer to sell, sell, import, and otherwise transfer the Work, + where such license applies only to those patent claims licensable + by such Contributor that are necessarily infringed by their + Contribution(s) alone or by combination of their Contribution(s) + with the Work to which such Contribution(s) was submitted. If You + institute patent litigation against any entity (including a + cross-claim or counterclaim in a lawsuit) alleging that the Work + or a Contribution incorporated within the Work constitutes direct + or contributory patent infringement, then any patent licenses + granted to You under this License for that Work shall terminate + as of the date such litigation is filed. + + 4. Redistribution. You may reproduce and distribute copies of the + Work or Derivative Works thereof in any medium, with or without + modifications, and in Source or Object form, provided that You + meet the following conditions: + + (a) You must give any other recipients of the Work or + Derivative Works a copy of this License; and + + (b) You must cause any modified files to carry prominent notices + stating that You changed the files; and + + (c) You must retain, in the Source form of any Derivative Works + that You distribute, all copyright, patent, trademark, and + attribution notices from the Source form of the Work, + excluding those notices that do not pertain to any part of + the Derivative Works; and + + (d) If the Work includes a "NOTICE" text file as part of its + distribution, then any Derivative Works that You distribute must + include a readable copy of the attribution notices contained + within such NOTICE file, excluding those notices that do not + pertain to any part of the Derivative Works, in at least one + of the following places: within a NOTICE text file distributed + as part of the Derivative Works; within the Source form or + documentation, if provided along with the Derivative Works; or, + within a display generated by the Derivative Works, if and + wherever such third-party notices normally appear. The contents + of the NOTICE file are for informational purposes only and + do not modify the License. You may add Your own attribution + notices within Derivative Works that You distribute, alongside + or as an addendum to the NOTICE text from the Work, provided + that such additional attribution notices cannot be construed + as modifying the License. + + You may add Your own copyright statement to Your modifications and + may provide additional or different license terms and conditions + for use, reproduction, or distribution of Your modifications, or + for any such Derivative Works as a whole, provided Your use, + reproduction, and distribution of the Work otherwise complies with + the conditions stated in this License. + + 5. Submission of Contributions. Unless You explicitly state otherwise, + any Contribution intentionally submitted for inclusion in the Work + by You to the Licensor shall be under the terms and conditions of + this License, without any additional terms or conditions. + Notwithstanding the above, nothing herein shall supersede or modify + the terms of any separate license agreement you may have executed + with Licensor regarding such Contributions. + + 6. Trademarks. This License does not grant permission to use the trade + names, trademarks, service marks, or product names of the Licensor, + except as required for reasonable and customary use in describing the + origin of the Work and reproducing the content of the NOTICE file. + + 7. Disclaimer of Warranty. Unless required by applicable law or + agreed to in writing, Licensor provides the Work (and each + Contributor provides its Contributions) on an "AS IS" BASIS, + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or + implied, including, without limitation, any warranties or conditions + of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A + PARTICULAR PURPOSE. You are solely responsible for determining the + appropriateness of using or redistributing the Work and assume any + risks associated with Your exercise of permissions under this License. + + 8. Limitation of Liability. In no event and under no legal theory, + whether in tort (including negligence), contract, or otherwise, + unless required by applicable law (such as deliberate and grossly + negligent acts) or agreed to in writing, shall any Contributor be + liable to You for damages, including any direct, indirect, special, + incidental, or consequential damages of any character arising as a + result of this License or out of the use or inability to use the + Work (including but not limited to damages for loss of goodwill, + work stoppage, computer failure or malfunction, or any and all + other commercial damages or losses), even if such Contributor + has been advised of the possibility of such damages. + + 9. Accepting Warranty or Additional Liability. While redistributing + the Work or Derivative Works thereof, You may choose to offer, + and charge a fee for, acceptance of support, warranty, indemnity, + or other liability obligations and/or rights consistent with this + License. However, in accepting such obligations, You may act only + on Your own behalf and on Your sole responsibility, not on behalf + of any other Contributor, and only if You agree to indemnify, + defend, and hold each Contributor harmless for any liability + incurred by, or claims asserted against, such Contributor by reason + of your accepting any such warranty or additional liability. + + END OF TERMS AND CONDITIONS + + APPENDIX: How to apply the Apache License to your work. + + To apply the Apache License to your work, attach the following + boilerplate notice, with the fields enclosed by brackets "[]" + replaced with your own identifying information. (Don't include + the brackets!) The text should be enclosed in the appropriate + comment syntax for the file format. We also recommend that a + file or class name and description of purpose be included on the + same "printed page" as the copyright notice for easier + identification within third-party archives. + + Copyright 2015 xeipuuv + + Licensed under the Apache License, Version 2.0 (the "License"); + you may not use this file except in compliance with the License. + You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + + Unless required by applicable law or agreed to in writing, software + distributed under the License is distributed on an "AS IS" BASIS, + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + See the License for the specific language governing permissions and + limitations under the License. diff --git a/vendor/github.com/open-policy-agent/opa/internal/gojsonschema/README.md b/vendor/github.com/open-policy-agent/opa/internal/gojsonschema/README.md new file mode 100644 index 00000000..53fdef96 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/gojsonschema/README.md @@ -0,0 +1,481 @@ +# gojsonschema (Details of library residing in OPA's internal) + +## Description + +https://github.com/xeipuuv/gojsonschema was duplicated into `internal/gojsonschema` folder and modified to make it possible +to set Rego types in OPA, using schemas returned from `gojsonschema`s `Compile` utility. + +The modifications done are as below: + +1. Some of the private fields in `gojsonschema`'s structs, `schema` and `subSchema` have been exported (publicized) so that OPA's type checking code can access and manipulate the values returned by `gojsonschema`s `Compile` method, + +2. Also, other changes in `gojsonschema`'s code include fixes to satisfy OPA's lint and format checker scripts. Hence, this `internal/gojsonschema` is in conformance with OPA's code style. + +3. Modernize usage of Go in `gojsonschema`, using conventions like type switching and language-built-in map accessing rather than helper methods. + +[![GoDoc](https://godoc.org/github.com/xeipuuv/gojsonschema?status.svg)](https://godoc.org/github.com/xeipuuv/gojsonschema) +[![Build Status](https://travis-ci.org/xeipuuv/gojsonschema.svg)](https://travis-ci.org/xeipuuv/gojsonschema) +[![Go Report Card](https://goreportcard.com/badge/github.com/xeipuuv/gojsonschema)](https://goreportcard.com/report/github.com/xeipuuv/gojsonschema) + +# gojsonschema (Details of original parent repository from README) + +## Description + +An implementation of JSON Schema for the Go programming language. Supports draft-04, draft-06 and draft-07. + +References : + +* http://json-schema.org +* http://json-schema.org/latest/json-schema-core.html +* http://json-schema.org/latest/json-schema-validation.html + +## Installation + +``` +go get github.com/xeipuuv/gojsonschema +``` + +Dependencies : +* [github.com/xeipuuv/gojsonpointer](https://github.com/xeipuuv/gojsonpointer) +* [github.com/xeipuuv/gojsonreference](https://github.com/xeipuuv/gojsonreference) +* [github.com/stretchr/testify/assert](https://github.com/stretchr/testify#assert-package) + +## Usage + +### Example + +```go + +package main + +import ( + "fmt" + "github.com/xeipuuv/gojsonschema" +) + +func main() { + + schemaLoader := gojsonschema.NewReferenceLoader("file:///home/me/schema.json") + documentLoader := gojsonschema.NewReferenceLoader("file:///home/me/document.json") + + result, err := gojsonschema.Validate(schemaLoader, documentLoader) + if err != nil { + panic(err.Error()) + } + + if result.Valid() { + fmt.Printf("The document is valid\n") + } else { + fmt.Printf("The document is not valid. see errors :\n") + for _, desc := range result.Errors() { + fmt.Printf("- %s\n", desc) + } + } +} + + +``` + +#### Loaders + +There are various ways to load your JSON data. +In order to load your schemas and documents, +first declare an appropriate loader : + +* Web / HTTP, using a reference : + +```go +loader := gojsonschema.NewReferenceLoader("http://www.some_host.com/schema.json") +``` + +* Local file, using a reference : + +```go +loader := gojsonschema.NewReferenceLoader("file:///home/me/schema.json") +``` + +References use the URI scheme, the prefix (file://) and a full path to the file are required. + +* JSON strings : + +```go +loader := gojsonschema.NewStringLoader(`{"type": "string"}`) +``` + +* Custom Go types : + +```go +m := map[string]interface{}{"type": "string"} +loader := gojsonschema.NewGoLoader(m) +``` + +And + +```go +type Root struct { + Users []User `json:"users"` +} + +type User struct { + Name string `json:"name"` +} + +... + +data := Root{} +data.Users = append(data.Users, User{"John"}) +data.Users = append(data.Users, User{"Sophia"}) +data.Users = append(data.Users, User{"Bill"}) + +loader := gojsonschema.NewGoLoader(data) +``` + +#### Validation + +Once the loaders are set, validation is easy : + +```go +result, err := gojsonschema.Validate(schemaLoader, documentLoader) +``` + +Alternatively, you might want to load a schema only once and process to multiple validations : + +```go +schema, err := gojsonschema.NewSchema(schemaLoader) +... +result1, err := schema.Validate(documentLoader1) +... +result2, err := schema.Validate(documentLoader2) +... +// etc ... +``` + +To check the result : + +```go + if result.Valid() { + fmt.Printf("The document is valid\n") + } else { + fmt.Printf("The document is not valid. see errors :\n") + for _, err := range result.Errors() { + // Err implements the ResultError interface + fmt.Printf("- %s\n", err) + } + } +``` + + +## Loading local schemas + +By default `file` and `http(s)` references to external schemas are loaded automatically via the file system or via http(s). An external schema can also be loaded using a `SchemaLoader`. + +```go + sl := gojsonschema.NewSchemaLoader() + loader1 := gojsonschema.NewStringLoader(`{ "type" : "string" }`) + err := sl.AddSchema("http://some_host.com/string.json", loader1) +``` + +Alternatively if your schema already has an `$id` you can use the `AddSchemas` function +```go + loader2 := gojsonschema.NewStringLoader(`{ + "$id" : "http://some_host.com/maxlength.json", + "maxLength" : 5 + }`) + err = sl.AddSchemas(loader2) +``` + +The main schema should be passed to the `Compile` function. This main schema can then directly reference the added schemas without needing to download them. +```go + loader3 := gojsonschema.NewStringLoader(`{ + "$id" : "http://some_host.com/main.json", + "allOf" : [ + { "$ref" : "http://some_host.com/string.json" }, + { "$ref" : "http://some_host.com/maxlength.json" } + ] + }`) + + schema, err := sl.Compile(loader3) + + documentLoader := gojsonschema.NewStringLoader(`"hello world"`) + + result, err := schema.Validate(documentLoader) +``` + +It's also possible to pass a `ReferenceLoader` to the `Compile` function that references a loaded schema. + +```go +err = sl.AddSchemas(loader3) +schema, err := sl.Compile(gojsonschema.NewReferenceLoader("http://some_host.com/main.json")) +``` + +Schemas added by `AddSchema` and `AddSchemas` are only validated when the entire schema is compiled, unless meta-schema validation is used. + +## Using a specific draft +By default `gojsonschema` will try to detect the draft of a schema by using the `$schema` keyword and parse it in a strict draft-04, draft-06 or draft-07 mode. If `$schema` is missing, or the draft version is not explicitely set, a hybrid mode is used which merges together functionality of all drafts into one mode. + +Autodectection can be turned off with the `AutoDetect` property. Specific draft versions can be specified with the `Draft` property. + +```go +sl := gojsonschema.NewSchemaLoader() +sl.Draft = gojsonschema.Draft7 +sl.AutoDetect = false +``` + +If autodetection is on (default), a draft-07 schema can savely reference draft-04 schemas and vice-versa, as long as `$schema` is specified in all schemas. + +## Meta-schema validation +Schemas that are added using the `AddSchema`, `AddSchemas` and `Compile` can be validated against their meta-schema by setting the `Validate` property. + +The following example will produce an error as `multipleOf` must be a number. If `Validate` is off (default), this error is only returned at the `Compile` step. + +```go +sl := gojsonschema.NewSchemaLoader() +sl.Validate = true +err := sl.AddSchemas(gojsonschema.NewStringLoader(`{ + "$id" : "http://some_host.com/invalid.json", + "$schema": "http://json-schema.org/draft-07/schema#", + "multipleOf" : true +}`)) + ``` +``` + ``` + +Errors returned by meta-schema validation are more readable and contain more information, which helps significantly if you are developing a schema. + +Meta-schema validation also works with a custom `$schema`. In case `$schema` is missing, or `AutoDetect` is set to `false`, the meta-schema of the used draft is used. + + +## Working with Errors + +The library handles string error codes which you can customize by creating your own gojsonschema.locale and setting it +```go +gojsonschema.Locale = YourCustomLocale{} +``` + +However, each error contains additional contextual information. + +Newer versions of `gojsonschema` may have new additional errors, so code that uses a custom locale will need to be updated when this happens. + +**err.Type()**: *string* Returns the "type" of error that occurred. Note you can also type check. See below + +Note: An error of RequiredType has an err.Type() return value of "required" + + "required": RequiredError + "invalid_type": InvalidTypeError + "number_any_of": NumberAnyOfError + "number_one_of": NumberOneOfError + "number_all_of": NumberAllOfError + "number_not": NumberNotError + "missing_dependency": MissingDependencyError + "internal": InternalError + "const": ConstEror + "enum": EnumError + "array_no_additional_items": ArrayNoAdditionalItemsError + "array_min_items": ArrayMinItemsError + "array_max_items": ArrayMaxItemsError + "unique": ItemsMustBeUniqueError + "contains" : ArrayContainsError + "array_min_properties": ArrayMinPropertiesError + "array_max_properties": ArrayMaxPropertiesError + "additional_property_not_allowed": AdditionalPropertyNotAllowedError + "invalid_property_pattern": InvalidPropertyPatternError + "invalid_property_name": InvalidPropertyNameError + "string_gte": StringLengthGTEError + "string_lte": StringLengthLTEError + "pattern": DoesNotMatchPatternError + "multiple_of": MultipleOfError + "number_gte": NumberGTEError + "number_gt": NumberGTError + "number_lte": NumberLTEError + "number_lt": NumberLTError + "condition_then" : ConditionThenError + "condition_else" : ConditionElseError + +**err.Value()**: *interface{}* Returns the value given + +**err.Context()**: *gojsonschema.JsonContext* Returns the context. This has a String() method that will print something like this: (root).firstName + +**err.Field()**: *string* Returns the fieldname in the format firstName, or for embedded properties, person.firstName. This returns the same as the String() method on *err.Context()* but removes the (root). prefix. + +**err.Description()**: *string* The error description. This is based on the locale you are using. See the beginning of this section for overwriting the locale with a custom implementation. + +**err.DescriptionFormat()**: *string* The error description format. This is relevant if you are adding custom validation errors afterwards to the result. + +**err.Details()**: *gojsonschema.ErrorDetails* Returns a map[string]interface{} of additional error details specific to the error. For example, GTE errors will have a "min" value, LTE will have a "max" value. See errors.go for a full description of all the error details. Every error always contains a "field" key that holds the value of *err.Field()* + +Note in most cases, the err.Details() will be used to generate replacement strings in your locales, and not used directly. These strings follow the text/template format i.e. +``` +{{.field}} must be greater than or equal to {{.min}} +``` + +The library allows you to specify custom template functions, should you require more complex error message handling. +```go +gojsonschema.ErrorTemplateFuncs = map[string]interface{}{ + "allcaps": func(s string) string { + return strings.ToUpper(s) + }, +} +``` + +Given the above definition, you can use the custom function `"allcaps"` in your localization templates: +``` +{{allcaps .field}} must be greater than or equal to {{.min}} +``` + +The above error message would then be rendered with the `field` value in capital letters. For example: +``` +"PASSWORD must be greater than or equal to 8" +``` + +Learn more about what types of template functions you can use in `ErrorTemplateFuncs` by referring to Go's [text/template FuncMap](https://golang.org/pkg/text/template/#FuncMap) type. + +## Formats +JSON Schema allows for optional "format" property to validate instances against well-known formats. gojsonschema ships with all of the formats defined in the spec that you can use like this: + +````json +{"type": "string", "format": "email"} +```` + +Not all formats defined in draft-07 are available. Implemented formats are: + +* `date` +* `time` +* `date-time` +* `hostname`. Subdomains that start with a number are also supported, but this means that it doesn't strictly follow [RFC1034](http://tools.ietf.org/html/rfc1034#section-3.5) and has the implication that ipv4 addresses are also recognized as valid hostnames. +* `email`. Go's email parser deviates slightly from [RFC5322](https://tools.ietf.org/html/rfc5322). Includes unicode support. +* `idn-email`. Same caveat as `email`. +* `ipv4` +* `ipv6` +* `uri`. Includes unicode support. +* `uri-reference`. Includes unicode support. +* `iri` +* `iri-reference` +* `uri-template` +* `uuid` +* `regex`. Go uses the [RE2](https://github.com/google/re2/wiki/Syntax) engine and is not [ECMA262](http://www.ecma-international.org/publications/files/ECMA-ST/Ecma-262.pdf) compatible. +* `json-pointer` +* `relative-json-pointer` + +`email`, `uri` and `uri-reference` use the same validation code as their unicode counterparts `idn-email`, `iri` and `iri-reference`. If you rely on unicode support you should use the specific +unicode enabled formats for the sake of interoperability as other implementations might not support unicode in the regular formats. + +The validation code for `uri`, `idn-email` and their relatives use mostly standard library code. + +For repetitive or more complex formats, you can create custom format checkers and add them to gojsonschema like this: + +```go +// Define the format checker +type RoleFormatChecker struct {} + +// Ensure it meets the gojsonschema.FormatChecker interface +func (f RoleFormatChecker) IsFormat(input interface{}) bool { + + asString, ok := input.(string) + if ok == false { + return false + } + + return strings.HasPrefix("ROLE_", asString) +} + +// Add it to the library +gojsonschema.FormatCheckers.Add("role", RoleFormatChecker{}) +```` + +Now to use in your json schema: +````json +{"type": "string", "format": "role"} +```` + +Another example would be to check if the provided integer matches an id on database: + +JSON schema: +```json +{"type": "integer", "format": "ValidUserId"} +``` + +```go +// Define the format checker +type ValidUserIdFormatChecker struct {} + +// Ensure it meets the gojsonschema.FormatChecker interface +func (f ValidUserIdFormatChecker) IsFormat(input interface{}) bool { + + asFloat64, ok := input.(float64) // Numbers are always float64 here + if ok == false { + return false + } + + // XXX + // do the magic on the database looking for the int(asFloat64) + + return true +} + +// Add it to the library +gojsonschema.FormatCheckers.Add("ValidUserId", ValidUserIdFormatChecker{}) +```` + +Formats can also be removed, for example if you want to override one of the formats that is defined by default. + +```go +gojsonschema.FormatCheckers.Remove("hostname") +``` + + +## Additional custom validation +After the validation has run and you have the results, you may add additional +errors using `Result.AddError`. This is useful to maintain the same format within the resultset instead +of having to add special exceptions for your own errors. Below is an example. + +```go +type AnswerInvalidError struct { + gojsonschema.ResultErrorFields +} + +func newAnswerInvalidError(context *gojsonschema.JsonContext, value interface{}, details gojsonschema.ErrorDetails) *AnswerInvalidError { + err := AnswerInvalidError{} + err.SetContext(context) + err.SetType("custom_invalid_error") + // it is important to use SetDescriptionFormat() as this is used to call SetDescription() after it has been parsed + // using the description of err will be overridden by this. + err.SetDescriptionFormat("Answer to the Ultimate Question of Life, the Universe, and Everything is {{.answer}}") + err.SetValue(value) + err.SetDetails(details) + + return &err +} + +func main() { + // ... + schema, err := gojsonschema.NewSchema(schemaLoader) + result, err := gojsonschema.Validate(schemaLoader, documentLoader) + + if true { // some validation + jsonContext := gojsonschema.NewJsonContext("question", nil) + errDetail := gojsonschema.ErrorDetails{ + "answer": 42, + } + result.AddError( + newAnswerInvalidError( + gojsonschema.NewJsonContext("answer", jsonContext), + 52, + errDetail, + ), + errDetail, + ) + } + + return result, err + +} +``` + +This is especially useful if you want to add validation beyond what the +json schema drafts can provide such business specific logic. + +## Uses + +gojsonschema uses the following test suite : + +https://github.com/json-schema/JSON-Schema-Test-Suite diff --git a/vendor/github.com/open-policy-agent/opa/internal/gojsonschema/draft.go b/vendor/github.com/open-policy-agent/opa/internal/gojsonschema/draft.go new file mode 100644 index 00000000..dac1aafd --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/gojsonschema/draft.go @@ -0,0 +1,122 @@ +// Copyright 2018 johandorland ( https://github.com/johandorland ) +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. + +package gojsonschema + +import ( + "errors" + "math" + + "github.com/xeipuuv/gojsonreference" +) + +// Draft is a JSON-schema draft version +type Draft int + +// Supported Draft versions +const ( + Draft4 Draft = 4 + Draft6 Draft = 6 + Draft7 Draft = 7 + Hybrid Draft = math.MaxInt32 +) + +type draftConfig struct { + Version Draft + MetaSchemaURL string + MetaSchema string +} +type draftConfigs []draftConfig + +var drafts draftConfigs + +func init() { + drafts = []draftConfig{ + { + Version: Draft4, + MetaSchemaURL: "http://json-schema.org/draft-04/schema", + MetaSchema: `{"id":"http://json-schema.org/draft-04/schema#","$schema":"http://json-schema.org/draft-04/schema#","description":"Core schema meta-schema","definitions":{"schemaArray":{"type":"array","minItems":1,"items":{"$ref":"#"}},"positiveInteger":{"type":"integer","minimum":0},"positiveIntegerDefault0":{"allOf":[{"$ref":"#/definitions/positiveInteger"},{"default":0}]},"simpleTypes":{"enum":["array","boolean","integer","null","number","object","string"]},"stringArray":{"type":"array","items":{"type":"string"},"minItems":1,"uniqueItems":true}},"type":"object","properties":{"id":{"type":"string"},"$schema":{"type":"string"},"title":{"type":"string"},"description":{"type":"string"},"default":{},"multipleOf":{"type":"number","minimum":0,"exclusiveMinimum":true},"maximum":{"type":"number"},"exclusiveMaximum":{"type":"boolean","default":false},"minimum":{"type":"number"},"exclusiveMinimum":{"type":"boolean","default":false},"maxLength":{"$ref":"#/definitions/positiveInteger"},"minLength":{"$ref":"#/definitions/positiveIntegerDefault0"},"pattern":{"type":"string","format":"regex"},"additionalItems":{"anyOf":[{"type":"boolean"},{"$ref":"#"}],"default":{}},"items":{"anyOf":[{"$ref":"#"},{"$ref":"#/definitions/schemaArray"}],"default":{}},"maxItems":{"$ref":"#/definitions/positiveInteger"},"minItems":{"$ref":"#/definitions/positiveIntegerDefault0"},"uniqueItems":{"type":"boolean","default":false},"maxProperties":{"$ref":"#/definitions/positiveInteger"},"minProperties":{"$ref":"#/definitions/positiveIntegerDefault0"},"required":{"$ref":"#/definitions/stringArray"},"additionalProperties":{"anyOf":[{"type":"boolean"},{"$ref":"#"}],"default":{}},"definitions":{"type":"object","additionalProperties":{"$ref":"#"},"default":{}},"properties":{"type":"object","additionalProperties":{"$ref":"#"},"default":{}},"patternProperties":{"type":"object","additionalProperties":{"$ref":"#"},"default":{}},"dependencies":{"type":"object","additionalProperties":{"anyOf":[{"$ref":"#"},{"$ref":"#/definitions/stringArray"}]}},"enum":{"type":"array","minItems":1,"uniqueItems":true},"type":{"anyOf":[{"$ref":"#/definitions/simpleTypes"},{"type":"array","items":{"$ref":"#/definitions/simpleTypes"},"minItems":1,"uniqueItems":true}]},"format":{"type":"string"},"allOf":{"$ref":"#/definitions/schemaArray"},"anyOf":{"$ref":"#/definitions/schemaArray"},"oneOf":{"$ref":"#/definitions/schemaArray"},"not":{"$ref":"#"}},"dependencies":{"exclusiveMaximum":["maximum"],"exclusiveMinimum":["minimum"]},"default":{}}`, + }, + { + Version: Draft6, + MetaSchemaURL: "http://json-schema.org/draft-06/schema", + MetaSchema: `{"$schema":"http://json-schema.org/draft-06/schema#","$id":"http://json-schema.org/draft-06/schema#","title":"Core schema meta-schema","definitions":{"schemaArray":{"type":"array","minItems":1,"items":{"$ref":"#"}},"nonNegativeInteger":{"type":"integer","minimum":0},"nonNegativeIntegerDefault0":{"allOf":[{"$ref":"#/definitions/nonNegativeInteger"},{"default":0}]},"simpleTypes":{"enum":["array","boolean","integer","null","number","object","string"]},"stringArray":{"type":"array","items":{"type":"string"},"uniqueItems":true,"default":[]}},"type":["object","boolean"],"properties":{"$id":{"type":"string","format":"uri-reference"},"$schema":{"type":"string","format":"uri"},"$ref":{"type":"string","format":"uri-reference"},"title":{"type":"string"},"description":{"type":"string"},"default":{},"examples":{"type":"array","items":{}},"multipleOf":{"type":"number","exclusiveMinimum":0},"maximum":{"type":"number"},"exclusiveMaximum":{"type":"number"},"minimum":{"type":"number"},"exclusiveMinimum":{"type":"number"},"maxLength":{"$ref":"#/definitions/nonNegativeInteger"},"minLength":{"$ref":"#/definitions/nonNegativeIntegerDefault0"},"pattern":{"type":"string","format":"regex"},"additionalItems":{"$ref":"#"},"items":{"anyOf":[{"$ref":"#"},{"$ref":"#/definitions/schemaArray"}],"default":{}},"maxItems":{"$ref":"#/definitions/nonNegativeInteger"},"minItems":{"$ref":"#/definitions/nonNegativeIntegerDefault0"},"uniqueItems":{"type":"boolean","default":false},"contains":{"$ref":"#"},"maxProperties":{"$ref":"#/definitions/nonNegativeInteger"},"minProperties":{"$ref":"#/definitions/nonNegativeIntegerDefault0"},"required":{"$ref":"#/definitions/stringArray"},"additionalProperties":{"$ref":"#"},"definitions":{"type":"object","additionalProperties":{"$ref":"#"},"default":{}},"properties":{"type":"object","additionalProperties":{"$ref":"#"},"default":{}},"patternProperties":{"type":"object","additionalProperties":{"$ref":"#"},"default":{}},"dependencies":{"type":"object","additionalProperties":{"anyOf":[{"$ref":"#"},{"$ref":"#/definitions/stringArray"}]}},"propertyNames":{"$ref":"#"},"const":{},"enum":{"type":"array","minItems":1,"uniqueItems":true},"type":{"anyOf":[{"$ref":"#/definitions/simpleTypes"},{"type":"array","items":{"$ref":"#/definitions/simpleTypes"},"minItems":1,"uniqueItems":true}]},"format":{"type":"string"},"allOf":{"$ref":"#/definitions/schemaArray"},"anyOf":{"$ref":"#/definitions/schemaArray"},"oneOf":{"$ref":"#/definitions/schemaArray"},"not":{"$ref":"#"}},"default":{}}`, + }, + { + Version: Draft7, + MetaSchemaURL: "http://json-schema.org/draft-07/schema", + MetaSchema: `{"$schema":"http://json-schema.org/draft-07/schema#","$id":"http://json-schema.org/draft-07/schema#","title":"Core schema meta-schema","definitions":{"schemaArray":{"type":"array","minItems":1,"items":{"$ref":"#"}},"nonNegativeInteger":{"type":"integer","minimum":0},"nonNegativeIntegerDefault0":{"allOf":[{"$ref":"#/definitions/nonNegativeInteger"},{"default":0}]},"simpleTypes":{"enum":["array","boolean","integer","null","number","object","string"]},"stringArray":{"type":"array","items":{"type":"string"},"uniqueItems":true,"default":[]}},"type":["object","boolean"],"properties":{"$id":{"type":"string","format":"uri-reference"},"$schema":{"type":"string","format":"uri"},"$ref":{"type":"string","format":"uri-reference"},"$comment":{"type":"string"},"title":{"type":"string"},"description":{"type":"string"},"default":true,"readOnly":{"type":"boolean","default":false},"examples":{"type":"array","items":true},"multipleOf":{"type":"number","exclusiveMinimum":0},"maximum":{"type":"number"},"exclusiveMaximum":{"type":"number"},"minimum":{"type":"number"},"exclusiveMinimum":{"type":"number"},"maxLength":{"$ref":"#/definitions/nonNegativeInteger"},"minLength":{"$ref":"#/definitions/nonNegativeIntegerDefault0"},"pattern":{"type":"string","format":"regex"},"additionalItems":{"$ref":"#"},"items":{"anyOf":[{"$ref":"#"},{"$ref":"#/definitions/schemaArray"}],"default":true},"maxItems":{"$ref":"#/definitions/nonNegativeInteger"},"minItems":{"$ref":"#/definitions/nonNegativeIntegerDefault0"},"uniqueItems":{"type":"boolean","default":false},"contains":{"$ref":"#"},"maxProperties":{"$ref":"#/definitions/nonNegativeInteger"},"minProperties":{"$ref":"#/definitions/nonNegativeIntegerDefault0"},"required":{"$ref":"#/definitions/stringArray"},"additionalProperties":{"$ref":"#"},"definitions":{"type":"object","additionalProperties":{"$ref":"#"},"default":{}},"properties":{"type":"object","additionalProperties":{"$ref":"#"},"default":{}},"patternProperties":{"type":"object","additionalProperties":{"$ref":"#"},"propertyNames":{"format":"regex"},"default":{}},"dependencies":{"type":"object","additionalProperties":{"anyOf":[{"$ref":"#"},{"$ref":"#/definitions/stringArray"}]}},"propertyNames":{"$ref":"#"},"const":true,"enum":{"type":"array","items":true,"minItems":1,"uniqueItems":true},"type":{"anyOf":[{"$ref":"#/definitions/simpleTypes"},{"type":"array","items":{"$ref":"#/definitions/simpleTypes"},"minItems":1,"uniqueItems":true}]},"format":{"type":"string"},"contentMediaType":{"type":"string"},"contentEncoding":{"type":"string"},"if":{"$ref":"#"},"then":{"$ref":"#"},"else":{"$ref":"#"},"allOf":{"$ref":"#/definitions/schemaArray"},"anyOf":{"$ref":"#/definitions/schemaArray"},"oneOf":{"$ref":"#/definitions/schemaArray"},"not":{"$ref":"#"}},"default":true}`, + }, + } +} + +func (dc draftConfigs) GetMetaSchema(url string) string { + for _, config := range dc { + if config.MetaSchemaURL == url { + return config.MetaSchema + } + } + return "" +} +func (dc draftConfigs) GetDraftVersion(url string) *Draft { + for _, config := range dc { + if config.MetaSchemaURL == url { + return &config.Version + } + } + return nil +} +func (dc draftConfigs) GetSchemaURL(draft Draft) string { + for _, config := range dc { + if config.Version == draft { + return config.MetaSchemaURL + } + } + return "" +} + +func parseSchemaURL(documentNode interface{}) (string, *Draft, error) { + if _, ok := documentNode.(bool); ok { + return "", nil, nil + } + + m, ok := documentNode.(map[string]interface{}) + if !ok { + return "", nil, errors.New("schema is invalid") + } + + if v, ok := m[KeySchema]; ok { + s, ok := v.(string) + if !ok { + return "", nil, errors.New(formatErrorDescription( + Locale.MustBeOfType(), + ErrorDetails{ + "key": KeySchema, + "type": TypeString, + })) + } + + schemaReference, err := gojsonreference.NewJsonReference(s) + + if err != nil { + return "", nil, err + } + + schema := schemaReference.String() + + return schema, drafts.GetDraftVersion(schema), nil + } + + return "", nil, nil +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/gojsonschema/errors.go b/vendor/github.com/open-policy-agent/opa/internal/gojsonschema/errors.go new file mode 100644 index 00000000..f7aaf903 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/gojsonschema/errors.go @@ -0,0 +1,366 @@ +// nolint: goconst // String duplication will be handled later by using errors.Is. +package gojsonschema + +import ( + "bytes" + "sync" + "text/template" +) + +var errorTemplates = errorTemplate{template.New("errors-new"), sync.RWMutex{}} + +// template.Template is not thread-safe for writing, so some locking is done +// sync.RWMutex is used for efficiently locking when new templates are created +type errorTemplate struct { + *template.Template + sync.RWMutex +} + +type ( + + // FalseError indicates that - + // ErrorDetails: - + FalseError struct { + ResultErrorFields + } + + // RequiredError indicates that a required field is missing + // ErrorDetails: property string + RequiredError struct { + ResultErrorFields + } + + // InvalidTypeError indicates that a field has the incorrect type + // ErrorDetails: expected, given + InvalidTypeError struct { + ResultErrorFields + } + + // NumberAnyOfError is produced in case of a failing "anyOf" validation + // ErrorDetails: - + NumberAnyOfError struct { + ResultErrorFields + } + + // NumberOneOfError is produced in case of a failing "oneOf" validation + // ErrorDetails: - + NumberOneOfError struct { + ResultErrorFields + } + + // NumberAllOfError is produced in case of a failing "allOf" validation + // ErrorDetails: - + NumberAllOfError struct { + ResultErrorFields + } + + // NumberNotError is produced if a "not" validation failed + // ErrorDetails: - + NumberNotError struct { + ResultErrorFields + } + + // MissingDependencyError is produced in case of a "missing dependency" problem + // ErrorDetails: dependency + MissingDependencyError struct { + ResultErrorFields + } + + // InternalError indicates an internal error + // ErrorDetails: error + InternalError struct { + ResultErrorFields + } + + // ConstError indicates a const error + // ErrorDetails: allowed + ConstError struct { + ResultErrorFields + } + + // EnumError indicates an enum error + // ErrorDetails: allowed + EnumError struct { + ResultErrorFields + } + + // ArrayNoAdditionalItemsError is produced if additional items were found, but not allowed + // ErrorDetails: - + ArrayNoAdditionalItemsError struct { + ResultErrorFields + } + + // ArrayMinItemsError is produced if an array contains less items than the allowed minimum + // ErrorDetails: min + ArrayMinItemsError struct { + ResultErrorFields + } + + // ArrayMaxItemsError is produced if an array contains more items than the allowed maximum + // ErrorDetails: max + ArrayMaxItemsError struct { + ResultErrorFields + } + + // ItemsMustBeUniqueError is produced if an array requires unique items, but contains non-unique items + // ErrorDetails: type, i, j + ItemsMustBeUniqueError struct { + ResultErrorFields + } + + // ArrayContainsError is produced if an array contains invalid items + // ErrorDetails: + ArrayContainsError struct { + ResultErrorFields + } + + // ArrayMinPropertiesError is produced if an object contains less properties than the allowed minimum + // ErrorDetails: min + ArrayMinPropertiesError struct { + ResultErrorFields + } + + // ArrayMaxPropertiesError is produced if an object contains more properties than the allowed maximum + // ErrorDetails: max + ArrayMaxPropertiesError struct { + ResultErrorFields + } + + // AdditionalPropertyNotAllowedError is produced if an object has additional properties, but not allowed + // ErrorDetails: property + AdditionalPropertyNotAllowedError struct { + ResultErrorFields + } + + // InvalidPropertyPatternError is produced if an pattern was found + // ErrorDetails: property, pattern + InvalidPropertyPatternError struct { + ResultErrorFields + } + + // InvalidPropertyNameError is produced if an invalid-named property was found + // ErrorDetails: property + InvalidPropertyNameError struct { + ResultErrorFields + } + + // StringLengthGTEError is produced if a string is shorter than the minimum required length + // ErrorDetails: min + StringLengthGTEError struct { + ResultErrorFields + } + + // StringLengthLTEError is produced if a string is longer than the maximum allowed length + // ErrorDetails: max + StringLengthLTEError struct { + ResultErrorFields + } + + // DoesNotMatchPatternError is produced if a string does not match the defined pattern + // ErrorDetails: pattern + DoesNotMatchPatternError struct { + ResultErrorFields + } + + // DoesNotMatchFormatError is produced if a string does not match the defined format + // ErrorDetails: format + DoesNotMatchFormatError struct { + ResultErrorFields + } + + // MultipleOfError is produced if a number is not a multiple of the defined multipleOf + // ErrorDetails: multiple + MultipleOfError struct { + ResultErrorFields + } + + // NumberGTEError is produced if a number is lower than the allowed minimum + // ErrorDetails: min + NumberGTEError struct { + ResultErrorFields + } + + // NumberGTError is produced if a number is lower than, or equal to the specified minimum, and exclusiveMinimum is set + // ErrorDetails: min + NumberGTError struct { + ResultErrorFields + } + + // NumberLTEError is produced if a number is higher than the allowed maximum + // ErrorDetails: max + NumberLTEError struct { + ResultErrorFields + } + + // NumberLTError is produced if a number is higher than, or equal to the specified maximum, and exclusiveMaximum is set + // ErrorDetails: max + NumberLTError struct { + ResultErrorFields + } + + // ConditionThenError is produced if a condition's "then" validation is invalid + // ErrorDetails: - + ConditionThenError struct { + ResultErrorFields + } + + // ConditionElseError is produced if a condition's "else" condition is invalid + // ErrorDetails: - + ConditionElseError struct { + ResultErrorFields + } +) + +// newError takes a ResultError type and sets the type, context, description, details, value, and field +func newError(err ResultError, context *JSONContext, value interface{}, locale locale, details ErrorDetails) { + var t string + var d string + switch err.(type) { + case *FalseError: + t = "false" + d = locale.False() + case *RequiredError: + t = "required" + d = locale.Required() + case *InvalidTypeError: + t = "invalid_type" + d = locale.InvalidType() + case *NumberAnyOfError: + t = "number_any_of" + d = locale.NumberAnyOf() + case *NumberOneOfError: + t = "number_one_of" + d = locale.NumberOneOf() + case *NumberAllOfError: + t = "number_all_of" + d = locale.NumberAllOf() + case *NumberNotError: + t = "number_not" + d = locale.NumberNot() + case *MissingDependencyError: + t = "missing_dependency" + d = locale.MissingDependency() + case *InternalError: + t = "internal" + d = locale.Internal() + case *ConstError: + t = "const" + d = locale.Const() + case *EnumError: + t = "enum" + d = locale.Enum() + case *ArrayNoAdditionalItemsError: + t = "array_no_additional_items" + d = locale.ArrayNoAdditionalItems() + case *ArrayMinItemsError: + t = "array_min_items" + d = locale.ArrayMinItems() + case *ArrayMaxItemsError: + t = "array_max_items" + d = locale.ArrayMaxItems() + case *ItemsMustBeUniqueError: + t = "unique" + d = locale.Unique() + case *ArrayContainsError: + t = "contains" + d = locale.ArrayContains() + case *ArrayMinPropertiesError: + t = "array_min_properties" + d = locale.ArrayMinProperties() + case *ArrayMaxPropertiesError: + t = "array_max_properties" + d = locale.ArrayMaxProperties() + case *AdditionalPropertyNotAllowedError: + t = "additional_property_not_allowed" + d = locale.AdditionalPropertyNotAllowed() + case *InvalidPropertyPatternError: + t = "invalid_property_pattern" + d = locale.InvalidPropertyPattern() + case *InvalidPropertyNameError: + t = "invalid_property_name" + d = locale.InvalidPropertyName() + case *StringLengthGTEError: + t = "string_gte" + d = locale.StringGTE() + case *StringLengthLTEError: + t = "string_lte" + d = locale.StringLTE() + case *DoesNotMatchPatternError: + t = "pattern" + d = locale.DoesNotMatchPattern() + case *DoesNotMatchFormatError: + t = "format" + d = locale.DoesNotMatchFormat() + case *MultipleOfError: + t = "multiple_of" + d = locale.MultipleOf() + case *NumberGTEError: + t = "number_gte" + d = locale.NumberGTE() + case *NumberGTError: + t = "number_gt" + d = locale.NumberGT() + case *NumberLTEError: + t = "number_lte" + d = locale.NumberLTE() + case *NumberLTError: + t = "number_lt" + d = locale.NumberLT() + case *ConditionThenError: + t = "condition_then" + d = locale.ConditionThen() + case *ConditionElseError: + t = "condition_else" + d = locale.ConditionElse() + } + + err.SetType(t) + err.SetContext(context) + err.SetValue(value) + err.SetDetails(details) + err.SetDescriptionFormat(d) + details["field"] = err.Field() + + if _, exists := details["context"]; !exists && context != nil { + details["context"] = context.String() + } + + err.SetDescription(formatErrorDescription(err.DescriptionFormat(), details)) +} + +// formatErrorDescription takes a string in the default text/template +// format and converts it to a string with replacements. The fields come +// from the ErrorDetails struct and vary for each type of error. +func formatErrorDescription(s string, details ErrorDetails) string { + + var tpl *template.Template + var descrAsBuffer bytes.Buffer + var err error + + errorTemplates.RLock() + tpl = errorTemplates.Lookup(s) + errorTemplates.RUnlock() + + if tpl == nil { + errorTemplates.Lock() + tpl = errorTemplates.New(s) + + if ErrorTemplateFuncs != nil { + tpl.Funcs(ErrorTemplateFuncs) + } + + tpl, err = tpl.Parse(s) + errorTemplates.Unlock() + + if err != nil { + return err.Error() + } + } + + err = tpl.Execute(&descrAsBuffer, details) + if err != nil { + return err.Error() + } + + return descrAsBuffer.String() +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/gojsonschema/format_checkers.go b/vendor/github.com/open-policy-agent/opa/internal/gojsonschema/format_checkers.go new file mode 100644 index 00000000..1e770464 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/gojsonschema/format_checkers.go @@ -0,0 +1,368 @@ +package gojsonschema + +import ( + "net" + "net/mail" + "net/url" + "regexp" + "strings" + "sync" + "time" +) + +type ( + // FormatChecker is the interface all formatters added to FormatCheckerChain must implement + FormatChecker interface { + // IsFormat checks if input has the correct format + IsFormat(input interface{}) bool + } + + // FormatCheckerChain holds the formatters + FormatCheckerChain struct { + formatters map[string]FormatChecker + } + + // EmailFormatChecker verifies email address formats + EmailFormatChecker struct{} + + // IPV4FormatChecker verifies IP addresses in the IPv4 format + IPV4FormatChecker struct{} + + // IPV6FormatChecker verifies IP addresses in the IPv6 format + IPV6FormatChecker struct{} + + // DateTimeFormatChecker verifies date/time formats per RFC3339 5.6 + // + // Valid formats: + // Partial Time: HH:MM:SS + // Full Date: YYYY-MM-DD + // Full Time: HH:MM:SSZ-07:00 + // Date Time: YYYY-MM-DDTHH:MM:SSZ-0700 + // + // Where + // YYYY = 4DIGIT year + // MM = 2DIGIT month ; 01-12 + // DD = 2DIGIT day-month ; 01-28, 01-29, 01-30, 01-31 based on month/year + // HH = 2DIGIT hour ; 00-23 + // MM = 2DIGIT ; 00-59 + // SS = 2DIGIT ; 00-58, 00-60 based on leap second rules + // T = Literal + // Z = Literal + // + // Note: Nanoseconds are also suported in all formats + // + // http://tools.ietf.org/html/rfc3339#section-5.6 + DateTimeFormatChecker struct{} + + // DateFormatChecker verifies date formats + // + // Valid format: + // Full Date: YYYY-MM-DD + // + // Where + // YYYY = 4DIGIT year + // MM = 2DIGIT month ; 01-12 + // DD = 2DIGIT day-month ; 01-28, 01-29, 01-30, 01-31 based on month/year + DateFormatChecker struct{} + + // TimeFormatChecker verifies time formats + // + // Valid formats: + // Partial Time: HH:MM:SS + // Full Time: HH:MM:SSZ-07:00 + // + // Where + // HH = 2DIGIT hour ; 00-23 + // MM = 2DIGIT ; 00-59 + // SS = 2DIGIT ; 00-58, 00-60 based on leap second rules + // T = Literal + // Z = Literal + TimeFormatChecker struct{} + + // URIFormatChecker validates a URI with a valid Scheme per RFC3986 + URIFormatChecker struct{} + + // URIReferenceFormatChecker validates a URI or relative-reference per RFC3986 + URIReferenceFormatChecker struct{} + + // URITemplateFormatChecker validates a URI template per RFC6570 + URITemplateFormatChecker struct{} + + // HostnameFormatChecker validates a hostname is in the correct format + HostnameFormatChecker struct{} + + // UUIDFormatChecker validates a UUID is in the correct format + UUIDFormatChecker struct{} + + // RegexFormatChecker validates a regex is in the correct format + RegexFormatChecker struct{} + + // JSONPointerFormatChecker validates a JSON Pointer per RFC6901 + JSONPointerFormatChecker struct{} + + // RelativeJSONPointerFormatChecker validates a relative JSON Pointer is in the correct format + RelativeJSONPointerFormatChecker struct{} +) + +var ( + // FormatCheckers holds the valid formatters, and is a public variable + // so library users can add custom formatters + FormatCheckers = FormatCheckerChain{ + formatters: map[string]FormatChecker{ + "date": DateFormatChecker{}, + "time": TimeFormatChecker{}, + "date-time": DateTimeFormatChecker{}, + "hostname": HostnameFormatChecker{}, + "email": EmailFormatChecker{}, + "idn-email": EmailFormatChecker{}, + "ipv4": IPV4FormatChecker{}, + "ipv6": IPV6FormatChecker{}, + "uri": URIFormatChecker{}, + "uri-reference": URIReferenceFormatChecker{}, + "iri": URIFormatChecker{}, + "iri-reference": URIReferenceFormatChecker{}, + "uri-template": URITemplateFormatChecker{}, + "uuid": UUIDFormatChecker{}, + "regex": RegexFormatChecker{}, + "json-pointer": JSONPointerFormatChecker{}, + "relative-json-pointer": RelativeJSONPointerFormatChecker{}, + }, + } + + // Regex credit: https://www.socketloop.com/tutorials/golang-validate-hostname + rxHostname = regexp.MustCompile(`^([a-zA-Z0-9]|[a-zA-Z0-9][a-zA-Z0-9\-]{0,61}[a-zA-Z0-9])(\.([a-zA-Z0-9]|[a-zA-Z0-9][a-zA-Z0-9\-]{0,61}[a-zA-Z0-9]))*$`) + + // Use a regex to make sure curly brackets are balanced properly after validating it as a AURI + rxURITemplate = regexp.MustCompile("^([^{]*({[^}]*})?)*$") + + rxUUID = regexp.MustCompile("^(?i)[a-f0-9]{8}-[a-f0-9]{4}-[a-f0-9]{4}-[a-f0-9]{4}-[a-f0-9]{12}$") + + rxJSONPointer = regexp.MustCompile("^(?:/(?:[^~/]|~0|~1)*)*$") + + rxRelJSONPointer = regexp.MustCompile("^(?:0|[1-9][0-9]*)(?:#|(?:/(?:[^~/]|~0|~1)*)*)$") + + lock = new(sync.RWMutex) +) + +// Add adds a FormatChecker to the FormatCheckerChain +// The name used will be the value used for the format key in your json schema +func (c *FormatCheckerChain) Add(name string, f FormatChecker) *FormatCheckerChain { + lock.Lock() + c.formatters[name] = f + lock.Unlock() + + return c +} + +// Remove deletes a FormatChecker from the FormatCheckerChain (if it exists) +func (c *FormatCheckerChain) Remove(name string) *FormatCheckerChain { + lock.Lock() + delete(c.formatters, name) + lock.Unlock() + + return c +} + +// Has checks to see if the FormatCheckerChain holds a FormatChecker with the given name +func (c *FormatCheckerChain) Has(name string) bool { + lock.RLock() + _, ok := c.formatters[name] + lock.RUnlock() + + return ok +} + +// IsFormat will check an input against a FormatChecker with the given name +// to see if it is the correct format +func (c *FormatCheckerChain) IsFormat(name string, input interface{}) bool { + lock.RLock() + f, ok := c.formatters[name] + lock.RUnlock() + + // If a format is unrecognized it should always pass validation + if !ok { + return true + } + + return f.IsFormat(input) +} + +// IsFormat checks if input is a correctly formatted e-mail address +func (f EmailFormatChecker) IsFormat(input interface{}) bool { + asString, ok := input.(string) + if !ok { + return true + } + + _, err := mail.ParseAddress(asString) + return err == nil +} + +// IsFormat checks if input is a correctly formatted IPv4-address +func (f IPV4FormatChecker) IsFormat(input interface{}) bool { + asString, ok := input.(string) + if !ok { + return true + } + + // Credit: https://github.com/asaskevich/govalidator + ip := net.ParseIP(asString) + return ip != nil && strings.Contains(asString, ".") +} + +// IsFormat checks if input is a correctly formatted IPv6=address +func (f IPV6FormatChecker) IsFormat(input interface{}) bool { + asString, ok := input.(string) + if !ok { + return true + } + + // Credit: https://github.com/asaskevich/govalidator + ip := net.ParseIP(asString) + return ip != nil && strings.Contains(asString, ":") +} + +// IsFormat checks if input is a correctly formatted date/time per RFC3339 5.6 +func (f DateTimeFormatChecker) IsFormat(input interface{}) bool { + asString, ok := input.(string) + if !ok { + return true + } + + formats := []string{ + "15:04:05", + "15:04:05Z07:00", + "2006-01-02", + time.RFC3339, + time.RFC3339Nano, + } + + for _, format := range formats { + if _, err := time.Parse(format, asString); err == nil { + return true + } + } + + return false +} + +// IsFormat checks if input is a correctly formatted date (YYYY-MM-DD) +func (f DateFormatChecker) IsFormat(input interface{}) bool { + asString, ok := input.(string) + if !ok { + return true + } + _, err := time.Parse("2006-01-02", asString) + return err == nil +} + +// IsFormat checks if input correctly formatted time (HH:MM:SS or HH:MM:SSZ-07:00) +func (f TimeFormatChecker) IsFormat(input interface{}) bool { + asString, ok := input.(string) + if !ok { + return true + } + + if _, err := time.Parse("15:04:05Z07:00", asString); err == nil { + return true + } + + _, err := time.Parse("15:04:05", asString) + return err == nil +} + +// IsFormat checks if input is correctly formatted URI with a valid Scheme per RFC3986 +func (f URIFormatChecker) IsFormat(input interface{}) bool { + asString, ok := input.(string) + if !ok { + return true + } + + u, err := url.Parse(asString) + + if err != nil || u.Scheme == "" { + return false + } + + return !strings.Contains(asString, `\`) +} + +// IsFormat checks if input is a correctly formatted URI or relative-reference per RFC3986 +func (f URIReferenceFormatChecker) IsFormat(input interface{}) bool { + asString, ok := input.(string) + if !ok { + return true + } + + _, err := url.Parse(asString) + return err == nil && !strings.Contains(asString, `\`) +} + +// IsFormat checks if input is a correctly formatted URI template per RFC6570 +func (f URITemplateFormatChecker) IsFormat(input interface{}) bool { + asString, ok := input.(string) + if !ok { + return true + } + + u, err := url.Parse(asString) + if err != nil || strings.Contains(asString, `\`) { + return false + } + + return rxURITemplate.MatchString(u.Path) +} + +// IsFormat checks if input is a correctly formatted hostname +func (f HostnameFormatChecker) IsFormat(input interface{}) bool { + asString, ok := input.(string) + if !ok { + return true + } + + return rxHostname.MatchString(asString) && len(asString) < 256 +} + +// IsFormat checks if input is a correctly formatted UUID +func (f UUIDFormatChecker) IsFormat(input interface{}) bool { + asString, ok := input.(string) + if !ok { + return true + } + + return rxUUID.MatchString(asString) +} + +// IsFormat checks if input is a correctly formatted regular expression +func (f RegexFormatChecker) IsFormat(input interface{}) bool { + asString, ok := input.(string) + if !ok { + return true + } + + if asString == "" { + return true + } + _, err := regexp.Compile(asString) + return err == nil +} + +// IsFormat checks if input is a correctly formatted JSON Pointer per RFC6901 +func (f JSONPointerFormatChecker) IsFormat(input interface{}) bool { + asString, ok := input.(string) + if !ok { + return true + } + + return rxJSONPointer.MatchString(asString) +} + +// IsFormat checks if input is a correctly formatted relative JSON Pointer +func (f RelativeJSONPointerFormatChecker) IsFormat(input interface{}) bool { + asString, ok := input.(string) + if !ok { + return true + } + + return rxRelJSONPointer.MatchString(asString) +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/gojsonschema/internalLog.go b/vendor/github.com/open-policy-agent/opa/internal/gojsonschema/internalLog.go new file mode 100644 index 00000000..4ef7a8d0 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/gojsonschema/internalLog.go @@ -0,0 +1,37 @@ +// Copyright 2015 xeipuuv ( https://github.com/xeipuuv ) +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. + +// author xeipuuv +// author-github https://github.com/xeipuuv +// author-mail xeipuuv@gmail.com +// +// repository-name gojsonschema +// repository-desc An implementation of JSON Schema, based on IETF's draft v4 - Go language. +// +// description Very simple log wrapper. +// Used for debugging/testing purposes. +// +// created 01-01-2015 + +package gojsonschema + +import ( + "log" +) + +const internalLogEnabled = false + +func internalLog(format string, v ...interface{}) { + log.Printf(format, v...) +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/gojsonschema/jsonContext.go b/vendor/github.com/open-policy-agent/opa/internal/gojsonschema/jsonContext.go new file mode 100644 index 00000000..f131ba58 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/gojsonschema/jsonContext.go @@ -0,0 +1,73 @@ +// Copyright 2013 MongoDB, Inc. +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. + +// author tolsen +// author-github https://github.com/tolsen +// +// repository-name gojsonschema +// repository-desc An implementation of JSON Schema, based on IETF's draft v4 - Go language. +// +// description Implements a persistent (immutable w/ shared structure) singly-linked list of strings for the purpose of storing a json context +// +// created 04-09-2013 + +package gojsonschema + +import "bytes" + +// JSONContext implements a persistent linked-list of strings +type JSONContext struct { + head string + tail *JSONContext +} + +// NewJSONContext creates a new JSONContext +func NewJSONContext(head string, tail *JSONContext) *JSONContext { + return &JSONContext{head, tail} +} + +// String displays the context in reverse. +// This plays well with the data structure's persistent nature with +// Cons and a json document's tree structure. +func (c *JSONContext) String(del ...string) string { + byteArr := make([]byte, 0, c.stringLen()) + buf := bytes.NewBuffer(byteArr) + c.writeStringToBuffer(buf, del) + + return buf.String() +} + +func (c *JSONContext) stringLen() int { + length := 0 + if c.tail != nil { + length = c.tail.stringLen() + 1 // add 1 for "." + } + + length += len(c.head) + return length +} + +func (c *JSONContext) writeStringToBuffer(buf *bytes.Buffer, del []string) { + if c.tail != nil { + c.tail.writeStringToBuffer(buf, del) + + if len(del) > 0 { + buf.WriteString(del[0]) + } else { + buf.WriteString(".") + } + } + + buf.WriteString(c.head) +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/gojsonschema/jsonLoader.go b/vendor/github.com/open-policy-agent/opa/internal/gojsonschema/jsonLoader.go new file mode 100644 index 00000000..65ada35c --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/gojsonschema/jsonLoader.go @@ -0,0 +1,405 @@ +// Copyright 2015 xeipuuv ( https://github.com/xeipuuv ) +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. + +// author xeipuuv +// author-github https://github.com/xeipuuv +// author-mail xeipuuv@gmail.com +// +// repository-name gojsonschema +// repository-desc An implementation of JSON Schema, based on IETF's draft v4 - Go language. +// +// description Different strategies to load JSON files. +// Includes References (file and HTTP), JSON strings and Go types. +// +// created 01-02-2015 + +package gojsonschema + +import ( + "bytes" + "encoding/json" + "errors" + "fmt" + "io" + "io/ioutil" + "net/http" + "net/url" + "os" + "path/filepath" + "runtime" + "strings" + + "github.com/xeipuuv/gojsonreference" +) + +// NOTE(sr): We need to control from which hosts remote references are +// allowed to be resolved via HTTP requests. It's quite cumbersome to +// add extra parameters to all calls and interfaces involved, so we're +// using a global variable instead: +var allowNet map[string]struct{} + +func SetAllowNet(hosts []string) { + if hosts == nil { + allowNet = nil // resetting the global + return + } + allowNet = make(map[string]struct{}, len(hosts)) + for _, host := range hosts { + allowNet[host] = struct{}{} + } +} + +func isAllowed(ref *url.URL) bool { + if allowNet == nil { + return true + } + _, ok := allowNet[ref.Hostname()] + return ok +} + +var osFS = osFileSystem(os.Open) + +// JSONLoader defines the JSON loader interface +type JSONLoader interface { + JSONSource() interface{} + LoadJSON() (interface{}, error) + JSONReference() (gojsonreference.JsonReference, error) + LoaderFactory() JSONLoaderFactory +} + +// JSONLoaderFactory defines the JSON loader factory interface +type JSONLoaderFactory interface { + // New creates a new JSON loader for the given source + New(source string) JSONLoader +} + +// DefaultJSONLoaderFactory is the default JSON loader factory +type DefaultJSONLoaderFactory struct { +} + +// FileSystemJSONLoaderFactory is a JSON loader factory that uses http.FileSystem +type FileSystemJSONLoaderFactory struct { + fs http.FileSystem +} + +// New creates a new JSON loader for the given source +func (d DefaultJSONLoaderFactory) New(source string) JSONLoader { + return &jsonReferenceLoader{ + fs: osFS, + source: source, + } +} + +// New creates a new JSON loader for the given source +func (f FileSystemJSONLoaderFactory) New(source string) JSONLoader { + return &jsonReferenceLoader{ + fs: f.fs, + source: source, + } +} + +// osFileSystem is a functional wrapper for os.Open that implements http.FileSystem. +type osFileSystem func(string) (*os.File, error) + +// Opens a file with the given name +func (o osFileSystem) Open(name string) (http.File, error) { + return o(name) +} + +// JSON Reference loader +// references are used to load JSONs from files and HTTP + +type jsonReferenceLoader struct { + fs http.FileSystem + source string +} + +func (l *jsonReferenceLoader) JSONSource() interface{} { + return l.source +} + +func (l *jsonReferenceLoader) JSONReference() (gojsonreference.JsonReference, error) { + return gojsonreference.NewJsonReference(l.JSONSource().(string)) +} + +func (l *jsonReferenceLoader) LoaderFactory() JSONLoaderFactory { + return &FileSystemJSONLoaderFactory{ + fs: l.fs, + } +} + +// NewReferenceLoader returns a JSON reference loader using the given source and the local OS file system. +func NewReferenceLoader(source string) JSONLoader { + return &jsonReferenceLoader{ + fs: osFS, + source: source, + } +} + +// NewReferenceLoaderFileSystem returns a JSON reference loader using the given source and file system. +func NewReferenceLoaderFileSystem(source string, fs http.FileSystem) JSONLoader { + return &jsonReferenceLoader{ + fs: fs, + source: source, + } +} + +func (l *jsonReferenceLoader) LoadJSON() (interface{}, error) { + + var err error + + reference, err := gojsonreference.NewJsonReference(l.JSONSource().(string)) + if err != nil { + return nil, err + } + + refToURL := reference + refToURL.GetUrl().Fragment = "" + + if reference.HasFileScheme { + + filename := strings.TrimPrefix(refToURL.String(), "file://") + filename, err = url.QueryUnescape(filename) + + if err != nil { + return nil, err + } + + if runtime.GOOS == "windows" { + // on Windows, a file URL may have an extra leading slash, use slashes + // instead of backslashes, and have spaces escaped + filename = strings.TrimPrefix(filename, "/") + filename = filepath.FromSlash(filename) + } + + return l.loadFromFile(filename) + } + + // NOTE(sr): hardcoded metaschema references are not subject to allow_net + // checking; their contents are hardcoded in the library! + // + // returned cached versions for metaschemas for drafts 4, 6 and 7 + // for performance and allow for easier offline use + if metaSchema := drafts.GetMetaSchema(refToURL.String()); metaSchema != "" { + return decodeJSONUsingNumber(strings.NewReader(metaSchema)) + } + + if isAllowed(refToURL.GetUrl()) { + return l.loadFromHTTP(refToURL.String()) + } + + return nil, fmt.Errorf("remote reference loading disabled: %s", reference.String()) +} + +func (l *jsonReferenceLoader) loadFromHTTP(address string) (interface{}, error) { + + resp, err := http.Get(address) + if err != nil { + return nil, err + } + + // must return HTTP Status 200 OK + if resp.StatusCode != http.StatusOK { + return nil, errors.New(formatErrorDescription(Locale.HTTPBadStatus(), ErrorDetails{"status": resp.Status})) + } + + bodyBuff, err := ioutil.ReadAll(resp.Body) + if err != nil { + return nil, err + } + + return decodeJSONUsingNumber(bytes.NewReader(bodyBuff)) +} + +func (l *jsonReferenceLoader) loadFromFile(path string) (interface{}, error) { + f, err := l.fs.Open(path) + if err != nil { + return nil, err + } + defer f.Close() + + bodyBuff, err := ioutil.ReadAll(f) + if err != nil { + return nil, err + } + + return decodeJSONUsingNumber(bytes.NewReader(bodyBuff)) + +} + +// JSON string loader + +type jsonStringLoader struct { + source string +} + +func (l *jsonStringLoader) JSONSource() interface{} { + return l.source +} + +func (l *jsonStringLoader) JSONReference() (gojsonreference.JsonReference, error) { + return gojsonreference.NewJsonReference("#") +} + +func (l *jsonStringLoader) LoaderFactory() JSONLoaderFactory { + return &DefaultJSONLoaderFactory{} +} + +// NewStringLoader creates a new JSONLoader, taking a string as source +func NewStringLoader(source string) JSONLoader { + return &jsonStringLoader{source: source} +} + +func (l *jsonStringLoader) LoadJSON() (interface{}, error) { + + return decodeJSONUsingNumber(strings.NewReader(l.JSONSource().(string))) + +} + +// JSON bytes loader + +type jsonBytesLoader struct { + source []byte +} + +func (l *jsonBytesLoader) JSONSource() interface{} { + return l.source +} + +func (l *jsonBytesLoader) JSONReference() (gojsonreference.JsonReference, error) { + return gojsonreference.NewJsonReference("#") +} + +func (l *jsonBytesLoader) LoaderFactory() JSONLoaderFactory { + return &DefaultJSONLoaderFactory{} +} + +// NewBytesLoader creates a new JSONLoader, taking a `[]byte` as source +func NewBytesLoader(source []byte) JSONLoader { + return &jsonBytesLoader{source: source} +} + +func (l *jsonBytesLoader) LoadJSON() (interface{}, error) { + return decodeJSONUsingNumber(bytes.NewReader(l.JSONSource().([]byte))) +} + +// JSON Go (types) loader +// used to load JSONs from the code as maps, interface{}, structs ... + +type jsonGoLoader struct { + source interface{} +} + +func (l *jsonGoLoader) JSONSource() interface{} { + return l.source +} + +func (l *jsonGoLoader) JSONReference() (gojsonreference.JsonReference, error) { + return gojsonreference.NewJsonReference("#") +} + +func (l *jsonGoLoader) LoaderFactory() JSONLoaderFactory { + return &DefaultJSONLoaderFactory{} +} + +// NewGoLoader creates a new JSONLoader from a given Go struct +func NewGoLoader(source interface{}) JSONLoader { + return &jsonGoLoader{source: source} +} + +func (l *jsonGoLoader) LoadJSON() (interface{}, error) { + + // convert it to a compliant JSON first to avoid types "mismatches" + + jsonBytes, err := json.Marshal(l.JSONSource()) + if err != nil { + return nil, err + } + + return decodeJSONUsingNumber(bytes.NewReader(jsonBytes)) + +} + +type jsonIOLoader struct { + buf *bytes.Buffer +} + +// NewReaderLoader creates a new JSON loader using the provided io.Reader +func NewReaderLoader(source io.Reader) (JSONLoader, io.Reader) { + buf := &bytes.Buffer{} + return &jsonIOLoader{buf: buf}, io.TeeReader(source, buf) +} + +// NewWriterLoader creates a new JSON loader using the provided io.Writer +func NewWriterLoader(source io.Writer) (JSONLoader, io.Writer) { + buf := &bytes.Buffer{} + return &jsonIOLoader{buf: buf}, io.MultiWriter(source, buf) +} + +func (l *jsonIOLoader) JSONSource() interface{} { + return l.buf.String() +} + +func (l *jsonIOLoader) LoadJSON() (interface{}, error) { + return decodeJSONUsingNumber(l.buf) +} + +func (l *jsonIOLoader) JSONReference() (gojsonreference.JsonReference, error) { + return gojsonreference.NewJsonReference("#") +} + +func (l *jsonIOLoader) LoaderFactory() JSONLoaderFactory { + return &DefaultJSONLoaderFactory{} +} + +// JSON raw loader +// In case the JSON is already marshalled to interface{} use this loader +// This is used for testing as otherwise there is no guarantee the JSON is marshalled +// "properly" by using https://golang.org/pkg/encoding/json/#Decoder.UseNumber +type jsonRawLoader struct { + source interface{} +} + +// NewRawLoader creates a new JSON raw loader for the given source +func NewRawLoader(source interface{}) JSONLoader { + return &jsonRawLoader{source: source} +} +func (l *jsonRawLoader) JSONSource() interface{} { + return l.source +} +func (l *jsonRawLoader) LoadJSON() (interface{}, error) { + return l.source, nil +} +func (l *jsonRawLoader) JSONReference() (gojsonreference.JsonReference, error) { + return gojsonreference.NewJsonReference("#") +} +func (l *jsonRawLoader) LoaderFactory() JSONLoaderFactory { + return &DefaultJSONLoaderFactory{} +} + +func decodeJSONUsingNumber(r io.Reader) (interface{}, error) { + + var document interface{} + + decoder := json.NewDecoder(r) + decoder.UseNumber() + + err := decoder.Decode(&document) + if err != nil { + return nil, err + } + + return document, nil + +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/gojsonschema/locales.go b/vendor/github.com/open-policy-agent/opa/internal/gojsonschema/locales.go new file mode 100644 index 00000000..384fbad2 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/gojsonschema/locales.go @@ -0,0 +1,472 @@ +// Copyright 2015 xeipuuv ( https://github.com/xeipuuv ) +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. + +// author xeipuuv +// author-github https://github.com/xeipuuv +// author-mail xeipuuv@gmail.com +// +// repository-name gojsonschema +// repository-desc An implementation of JSON Schema, based on IETF's draft v4 - Go language. +// +// description Contains const string and messages. +// +// created 01-01-2015 + +package gojsonschema + +type ( + // locale is an interface for defining custom error strings + locale interface { + + // False returns a format-string for "false" schema validation errors + False() string + + // Required returns a format-string for "required" schema validation errors + Required() string + + // InvalidType returns a format-string for "invalid type" schema validation errors + InvalidType() string + + // NumberAnyOf returns a format-string for "anyOf" schema validation errors + NumberAnyOf() string + + // NumberOneOf returns a format-string for "oneOf" schema validation errors + NumberOneOf() string + + // NumberAllOf returns a format-string for "allOf" schema validation errors + NumberAllOf() string + + // NumberNot returns a format-string to format a NumberNotError + NumberNot() string + + // MissingDependency returns a format-string for "missing dependency" schema validation errors + MissingDependency() string + + // Internal returns a format-string for internal errors + Internal() string + + // Const returns a format-string to format a ConstError + Const() string + + // Enum returns a format-string to format an EnumError + Enum() string + + // ArrayNotEnoughItems returns a format-string to format an error for arrays having not enough items to match positional list of schema + ArrayNotEnoughItems() string + + // ArrayNoAdditionalItems returns a format-string to format an ArrayNoAdditionalItemsError + ArrayNoAdditionalItems() string + + // ArrayMinItems returns a format-string to format an ArrayMinItemsError + ArrayMinItems() string + + // ArrayMaxItems returns a format-string to format an ArrayMaxItemsError + ArrayMaxItems() string + + // Unique returns a format-string to format an ItemsMustBeUniqueError + Unique() string + + // ArrayContains returns a format-string to format an ArrayContainsError + ArrayContains() string + + // ArrayMinProperties returns a format-string to format an ArrayMinPropertiesError + ArrayMinProperties() string + + // ArrayMaxProperties returns a format-string to format an ArrayMaxPropertiesError + ArrayMaxProperties() string + + // AdditionalPropertyNotAllowed returns a format-string to format an AdditionalPropertyNotAllowedError + AdditionalPropertyNotAllowed() string + + // InvalidPropertyPattern returns a format-string to format an InvalidPropertyPatternError + InvalidPropertyPattern() string + + // InvalidPropertyName returns a format-string to format an InvalidPropertyNameError + InvalidPropertyName() string + + // StringGTE returns a format-string to format an StringLengthGTEError + StringGTE() string + + // StringLTE returns a format-string to format an StringLengthLTEError + StringLTE() string + + // DoesNotMatchPattern returns a format-string to format an DoesNotMatchPatternError + DoesNotMatchPattern() string + + // DoesNotMatchFormat returns a format-string to format an DoesNotMatchFormatError + DoesNotMatchFormat() string + + // MultipleOf returns a format-string to format an MultipleOfError + MultipleOf() string + + // NumberGTE returns a format-string to format an NumberGTEError + NumberGTE() string + + // NumberGT returns a format-string to format an NumberGTError + NumberGT() string + + // NumberLTE returns a format-string to format an NumberLTEError + NumberLTE() string + + // NumberLT returns a format-string to format an NumberLTError + NumberLT() string + + // Schema validations + + // RegexPattern returns a format-string to format a regex-pattern error + RegexPattern() string + + // GreaterThanZero returns a format-string to format an error where a number must be greater than zero + GreaterThanZero() string + + // MustBeOfA returns a format-string to format an error where a value is of the wrong type + MustBeOfA() string + + // MustBeOfAn returns a format-string to format an error where a value is of the wrong type + MustBeOfAn() string + + // CannotBeUsedWithout returns a format-string to format a "cannot be used without" error + CannotBeUsedWithout() string + + // CannotBeGT returns a format-string to format an error where a value are greater than allowed + CannotBeGT() string + + // MustBeOfType returns a format-string to format an error where a value does not match the required type + MustBeOfType() string + + // MustBeValidRegex returns a format-string to format an error where a regex is invalid + MustBeValidRegex() string + + // MustBeValidFormat returns a format-string to format an error where a value does not match the expected format + MustBeValidFormat() string + + // MustBeGTEZero returns a format-string to format an error where a value must be greater or equal than 0 + MustBeGTEZero() string + + // KeyCannotBeGreaterThan returns a format-string to format an error where a key is greater than the maximum allowed + KeyCannotBeGreaterThan() string + + // KeyItemsMustBeOfType returns a format-string to format an error where a key is of the wrong type + KeyItemsMustBeOfType() string + + // KeyItemsMustBeUnique returns a format-string to format an error where keys are not unique + KeyItemsMustBeUnique() string + + // ReferenceMustBeCanonical returns a format-string to format a "reference must be canonical" error + ReferenceMustBeCanonical() string + + // NotAValidType returns a format-string to format an invalid type error + NotAValidType() string + + // Duplicated returns a format-string to format an error where types are duplicated + Duplicated() string + + // HTTPBadStatus returns a format-string for errors when loading a schema using HTTP + HTTPBadStatus() string + + // ParseError returns a format-string for JSON parsing errors + ParseError() string + + // ConditionThen returns a format-string for ConditionThenError errors + ConditionThen() string + + // ConditionElse returns a format-string for ConditionElseError errors + ConditionElse() string + + // ErrorFormat returns a format string for errors + ErrorFormat() string + } + + // DefaultLocale is the default locale for this package + DefaultLocale struct{} +) + +// False returns a format-string for "false" schema validation errors +func (l DefaultLocale) False() string { + return "False always fails validation" +} + +// Required returns a format-string for "required" schema validation errors +func (l DefaultLocale) Required() string { + return `{{.property}} is required` +} + +// InvalidType returns a format-string for "invalid type" schema validation errors +func (l DefaultLocale) InvalidType() string { + return `Invalid type. Expected: {{.expected}}, given: {{.given}}` +} + +// NumberAnyOf returns a format-string for "anyOf" schema validation errors +func (l DefaultLocale) NumberAnyOf() string { + return `Must validate at least one schema (anyOf)` +} + +// NumberOneOf returns a format-string for "oneOf" schema validation errors +func (l DefaultLocale) NumberOneOf() string { + return `Must validate one and only one schema (oneOf)` +} + +// NumberAllOf returns a format-string for "allOf" schema validation errors +func (l DefaultLocale) NumberAllOf() string { + return `Must validate all the schemas (allOf)` +} + +// NumberNot returns a format-string to format a NumberNotError +func (l DefaultLocale) NumberNot() string { + return `Must not validate the schema (not)` +} + +// MissingDependency returns a format-string for "missing dependency" schema validation errors +func (l DefaultLocale) MissingDependency() string { + return `Has a dependency on {{.dependency}}` +} + +// Internal returns a format-string for internal errors +func (l DefaultLocale) Internal() string { + return `Internal Error {{.error}}` +} + +// Const returns a format-string to format a ConstError +func (l DefaultLocale) Const() string { + return `{{.field}} does not match: {{.allowed}}` +} + +// Enum returns a format-string to format an EnumError +func (l DefaultLocale) Enum() string { + return `{{.field}} must be one of the following: {{.allowed}}` +} + +// ArrayNoAdditionalItems returns a format-string to format an ArrayNoAdditionalItemsError +func (l DefaultLocale) ArrayNoAdditionalItems() string { + return `No additional items allowed on array` +} + +// ArrayNotEnoughItems returns a format-string to format an error for arrays having not enough items to match positional list of schema +func (l DefaultLocale) ArrayNotEnoughItems() string { + return `Not enough items on array to match positional list of schema` +} + +// ArrayMinItems returns a format-string to format an ArrayMinItemsError +func (l DefaultLocale) ArrayMinItems() string { + return `Array must have at least {{.min}} items` +} + +// ArrayMaxItems returns a format-string to format an ArrayMaxItemsError +func (l DefaultLocale) ArrayMaxItems() string { + return `Array must have at most {{.max}} items` +} + +// Unique returns a format-string to format an ItemsMustBeUniqueError +func (l DefaultLocale) Unique() string { + return `{{.type}} items[{{.i}},{{.j}}] must be unique` +} + +// ArrayContains returns a format-string to format an ArrayContainsError +func (l DefaultLocale) ArrayContains() string { + return `At least one of the items must match` +} + +// ArrayMinProperties returns a format-string to format an ArrayMinPropertiesError +func (l DefaultLocale) ArrayMinProperties() string { + return `Must have at least {{.min}} properties` +} + +// ArrayMaxProperties returns a format-string to format an ArrayMaxPropertiesError +func (l DefaultLocale) ArrayMaxProperties() string { + return `Must have at most {{.max}} properties` +} + +// AdditionalPropertyNotAllowed returns a format-string to format an AdditionalPropertyNotAllowedError +func (l DefaultLocale) AdditionalPropertyNotAllowed() string { + return `Additional property {{.property}} is not allowed` +} + +// InvalidPropertyPattern returns a format-string to format an InvalidPropertyPatternError +func (l DefaultLocale) InvalidPropertyPattern() string { + return `Property "{{.property}}" does not match pattern {{.pattern}}` +} + +// InvalidPropertyName returns a format-string to format an InvalidPropertyNameError +func (l DefaultLocale) InvalidPropertyName() string { + return `Property name of "{{.property}}" does not match` +} + +// StringGTE returns a format-string to format an StringLengthGTEError +func (l DefaultLocale) StringGTE() string { + return `String length must be greater than or equal to {{.min}}` +} + +// StringLTE returns a format-string to format an StringLengthLTEError +func (l DefaultLocale) StringLTE() string { + return `String length must be less than or equal to {{.max}}` +} + +// DoesNotMatchPattern returns a format-string to format an DoesNotMatchPatternError +func (l DefaultLocale) DoesNotMatchPattern() string { + return `Does not match pattern '{{.pattern}}'` +} + +// DoesNotMatchFormat returns a format-string to format an DoesNotMatchFormatError +func (l DefaultLocale) DoesNotMatchFormat() string { + return `Does not match format '{{.format}}'` +} + +// MultipleOf returns a format-string to format an MultipleOfError +func (l DefaultLocale) MultipleOf() string { + return `Must be a multiple of {{.multiple}}` +} + +// NumberGTE returns the format string to format a NumberGTEError +func (l DefaultLocale) NumberGTE() string { + return `Must be greater than or equal to {{.min}}` +} + +// NumberGT returns the format string to format a NumberGTError +func (l DefaultLocale) NumberGT() string { + return `Must be greater than {{.min}}` +} + +// NumberLTE returns the format string to format a NumberLTEError +func (l DefaultLocale) NumberLTE() string { + return `Must be less than or equal to {{.max}}` +} + +// NumberLT returns the format string to format a NumberLTError +func (l DefaultLocale) NumberLT() string { + return `Must be less than {{.max}}` +} + +// Schema validators + +// RegexPattern returns a format-string to format a regex-pattern error +func (l DefaultLocale) RegexPattern() string { + return `Invalid regex pattern '{{.pattern}}'` +} + +// GreaterThanZero returns a format-string to format an error where a number must be greater than zero +func (l DefaultLocale) GreaterThanZero() string { + return `{{.number}} must be strictly greater than 0` +} + +// MustBeOfA returns a format-string to format an error where a value is of the wrong type +func (l DefaultLocale) MustBeOfA() string { + return `{{.x}} must be of a {{.y}}` +} + +// MustBeOfAn returns a format-string to format an error where a value is of the wrong type +func (l DefaultLocale) MustBeOfAn() string { + return `{{.x}} must be of an {{.y}}` +} + +// CannotBeUsedWithout returns a format-string to format a "cannot be used without" error +func (l DefaultLocale) CannotBeUsedWithout() string { + return `{{.x}} cannot be used without {{.y}}` +} + +// CannotBeGT returns a format-string to format an error where a value are greater than allowed +func (l DefaultLocale) CannotBeGT() string { + return `{{.x}} cannot be greater than {{.y}}` +} + +// MustBeOfType returns a format-string to format an error where a value does not match the required type +func (l DefaultLocale) MustBeOfType() string { + return `{{.key}} must be of type {{.type}}` +} + +// MustBeValidRegex returns a format-string to format an error where a regex is invalid +func (l DefaultLocale) MustBeValidRegex() string { + return `{{.key}} must be a valid regex` +} + +// MustBeValidFormat returns a format-string to format an error where a value does not match the expected format +func (l DefaultLocale) MustBeValidFormat() string { + return `{{.key}} must be a valid format {{.given}}` +} + +// MustBeGTEZero returns a format-string to format an error where a value must be greater or equal than 0 +func (l DefaultLocale) MustBeGTEZero() string { + return `{{.key}} must be greater than or equal to 0` +} + +// KeyCannotBeGreaterThan returns a format-string to format an error where a value is greater than the maximum allowed +func (l DefaultLocale) KeyCannotBeGreaterThan() string { + return `{{.key}} cannot be greater than {{.y}}` +} + +// KeyItemsMustBeOfType returns a format-string to format an error where a key is of the wrong type +func (l DefaultLocale) KeyItemsMustBeOfType() string { + return `{{.key}} items must be {{.type}}` +} + +// KeyItemsMustBeUnique returns a format-string to format an error where keys are not unique +func (l DefaultLocale) KeyItemsMustBeUnique() string { + return `{{.key}} items must be unique` +} + +// ReferenceMustBeCanonical returns a format-string to format a "reference must be canonical" error +func (l DefaultLocale) ReferenceMustBeCanonical() string { + return `Reference {{.reference}} must be canonical` +} + +// NotAValidType returns a format-string to format an invalid type error +func (l DefaultLocale) NotAValidType() string { + return `has a primitive type that is NOT VALID -- given: {{.given}} Expected valid values are:{{.expected}}` +} + +// Duplicated returns a format-string to format an error where types are duplicated +func (l DefaultLocale) Duplicated() string { + return `{{.type}} type is duplicated` +} + +// HTTPBadStatus returns a format-string for errors when loading a schema using HTTP +func (l DefaultLocale) HTTPBadStatus() string { + return `Could not read schema from HTTP, response status is {{.status}}` +} + +// ErrorFormat returns a format string for errors +// Replacement options: field, description, context, value +func (l DefaultLocale) ErrorFormat() string { + return `{{.field}}: {{.description}}` +} + +// ParseError returns a format-string for JSON parsing errors +func (l DefaultLocale) ParseError() string { + return `Expected: {{.expected}}, given: Invalid JSON` +} + +// ConditionThen returns a format-string for ConditionThenError errors +// If/Else +func (l DefaultLocale) ConditionThen() string { + return `Must validate "then" as "if" was valid` +} + +// ConditionElse returns a format-string for ConditionElseError errors +func (l DefaultLocale) ConditionElse() string { + return `Must validate "else" as "if" was not valid` +} + +// constants +const ( + StringNumber = "Number" + StringArrayOfStrings = "Array Of Strings" + StringArrayOfSchemas = "Array Of Schemas" + StringSchema = "Valid Schema" + StringSchemaOrArrayOfStrings = "Schema Or Array Of Strings" + StringProperties = "Properties" + StringDependency = "Dependency" + StringProperty = "Property" + StringUndefined = "Undefined" + StringContextRoot = "(Root)" + StringRootSchemaProperty = "(Root)" +) diff --git a/vendor/github.com/open-policy-agent/opa/internal/gojsonschema/result.go b/vendor/github.com/open-policy-agent/opa/internal/gojsonschema/result.go new file mode 100644 index 00000000..8baff071 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/gojsonschema/result.go @@ -0,0 +1,220 @@ +// Copyright 2015 xeipuuv ( https://github.com/xeipuuv ) +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. + +// author xeipuuv +// author-github https://github.com/xeipuuv +// author-mail xeipuuv@gmail.com +// +// repository-name gojsonschema +// repository-desc An implementation of JSON Schema, based on IETF's draft v4 - Go language. +// +// description Result and ResultError implementations. +// +// created 01-01-2015 + +package gojsonschema + +import ( + "fmt" + "strings" +) + +type ( + // ErrorDetails is a map of details specific to each error. + // While the values will vary, every error will contain a "field" value + ErrorDetails map[string]interface{} + + // ResultError is the interface that library errors must implement + ResultError interface { + // Field returns the field name without the root context + // i.e. firstName or person.firstName instead of (root).firstName or (root).person.firstName + Field() string + // SetType sets the error-type + SetType(string) + // Type returns the error-type + Type() string + // SetContext sets the JSON-context for the error + SetContext(*JSONContext) + // Context returns the JSON-context of the error + Context() *JSONContext + // SetDescription sets a description for the error + SetDescription(string) + // Description returns the description of the error + Description() string + // SetDescriptionFormat sets the format for the description in the default text/template format + SetDescriptionFormat(string) + // DescriptionFormat returns the format for the description in the default text/template format + DescriptionFormat() string + // SetValue sets the value related to the error + SetValue(interface{}) + // Value returns the value related to the error + Value() interface{} + // SetDetails sets the details specific to the error + SetDetails(ErrorDetails) + // Details returns details about the error + Details() ErrorDetails + // String returns a string representation of the error + String() string + } + + // ResultErrorFields holds the fields for each ResultError implementation. + // ResultErrorFields implements the ResultError interface, so custom errors + // can be defined by just embedding this type + ResultErrorFields struct { + errorType string // A string with the type of error (i.e. invalid_type) + context *JSONContext // Tree like notation of the part that failed the validation. ex (root).a.b ... + description string // A human readable error message + descriptionFormat string // A format for human readable error message + value interface{} // Value given by the JSON file that is the source of the error + details ErrorDetails + } + + // Result holds the result of a validation + Result struct { + errors []ResultError + // Scores how well the validation matched. Useful in generating + // better error messages for anyOf and oneOf. + score int + } +) + +// Field returns the field name without the root context +// i.e. firstName or person.firstName instead of (root).firstName or (root).person.firstName +func (v *ResultErrorFields) Field() string { + return strings.TrimPrefix(v.context.String(), StringRootSchemaProperty+".") +} + +// SetType sets the error-type +func (v *ResultErrorFields) SetType(errorType string) { + v.errorType = errorType +} + +// Type returns the error-type +func (v *ResultErrorFields) Type() string { + return v.errorType +} + +// SetContext sets the JSON-context for the error +func (v *ResultErrorFields) SetContext(context *JSONContext) { + v.context = context +} + +// Context returns the JSON-context of the error +func (v *ResultErrorFields) Context() *JSONContext { + return v.context +} + +// SetDescription sets a description for the error +func (v *ResultErrorFields) SetDescription(description string) { + v.description = description +} + +// Description returns the description of the error +func (v *ResultErrorFields) Description() string { + return v.description +} + +// SetDescriptionFormat sets the format for the description in the default text/template format +func (v *ResultErrorFields) SetDescriptionFormat(descriptionFormat string) { + v.descriptionFormat = descriptionFormat +} + +// DescriptionFormat returns the format for the description in the default text/template format +func (v *ResultErrorFields) DescriptionFormat() string { + return v.descriptionFormat +} + +// SetValue sets the value related to the error +func (v *ResultErrorFields) SetValue(value interface{}) { + v.value = value +} + +// Value returns the value related to the error +func (v *ResultErrorFields) Value() interface{} { + return v.value +} + +// SetDetails sets the details specific to the error +func (v *ResultErrorFields) SetDetails(details ErrorDetails) { + v.details = details +} + +// Details returns details about the error +func (v *ResultErrorFields) Details() ErrorDetails { + return v.details +} + +// String returns a string representation of the error +func (v ResultErrorFields) String() string { + // as a fallback, the value is displayed go style + valueString := fmt.Sprintf("%v", v.value) + + // marshal the go value value to json + if v.value == nil { + valueString = TypeNull + } else { + if vs, err := marshalToJSONString(v.value); err == nil { + if vs == nil { + valueString = TypeNull + } else { + valueString = *vs + } + } + } + + return formatErrorDescription(Locale.ErrorFormat(), ErrorDetails{ + "context": v.context.String(), + "description": v.description, + "value": valueString, + "field": v.Field(), + }) +} + +// Valid indicates if no errors were found +func (v *Result) Valid() bool { + return len(v.errors) == 0 +} + +// Errors returns the errors that were found +func (v *Result) Errors() []ResultError { + return v.errors +} + +// AddError appends a fully filled error to the error set +// SetDescription() will be called with the result of the parsed err.DescriptionFormat() +func (v *Result) AddError(err ResultError, details ErrorDetails) { + if _, exists := details["context"]; !exists && err.Context() != nil { + details["context"] = err.Context().String() + } + + err.SetDescription(formatErrorDescription(err.DescriptionFormat(), details)) + + v.errors = append(v.errors, err) +} + +func (v *Result) addInternalError(err ResultError, context *JSONContext, value interface{}, details ErrorDetails) { + newError(err, context, value, Locale, details) + v.errors = append(v.errors, err) + v.score -= 2 // results in a net -1 when added to the +1 we get at the end of the validation function +} + +// Used to copy errors from a sub-schema to the main one +func (v *Result) mergeErrors(otherResult *Result) { + v.errors = append(v.errors, otherResult.Errors()...) + v.score += otherResult.score +} + +func (v *Result) incrementScore() { + v.score++ +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/gojsonschema/schema.go b/vendor/github.com/open-policy-agent/opa/internal/gojsonschema/schema.go new file mode 100644 index 00000000..3e6f8c08 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/gojsonschema/schema.go @@ -0,0 +1,963 @@ +// Copyright 2015 xeipuuv ( https://github.com/xeipuuv ) +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. + +// author xeipuuv +// author-github https://github.com/xeipuuv +// author-mail xeipuuv@gmail.com +// +// repository-name gojsonschema +// repository-desc An implementation of JSON Schema, based on IETF's draft v4 - Go language. +// +// description Defines Schema, the main entry to every SubSchema. +// Contains the parsing logic and error checking. +// +// created 26-02-2013 + +package gojsonschema + +import ( + "encoding/json" + "errors" + "math/big" + "regexp" + "text/template" + + "github.com/xeipuuv/gojsonreference" +) + +var ( + // Locale is the default locale to use + // Library users can overwrite with their own implementation + Locale locale = DefaultLocale{} + + // ErrorTemplateFuncs allows you to define custom template funcs for use in localization. + ErrorTemplateFuncs template.FuncMap +) + +// NewSchema instances a schema using the given JSONLoader +func NewSchema(l JSONLoader) (*Schema, error) { + return NewSchemaLoader().Compile(l) +} + +// Schema holds a schema +type Schema struct { + DocumentReference gojsonreference.JsonReference + RootSchema *SubSchema + Pool *schemaPool + ReferencePool *schemaReferencePool +} + +func (d *Schema) parse(document interface{}, draft Draft) error { + d.RootSchema = &SubSchema{Property: StringRootSchemaProperty, Draft: &draft} + return d.parseSchema(document, d.RootSchema) +} + +// SetRootSchemaName sets the root-schema name +func (d *Schema) SetRootSchemaName(name string) { + d.RootSchema.Property = name +} + +// Parses a SubSchema +// +// Pretty long function ( sorry :) )... but pretty straight forward, repetitive and boring +// Not much magic involved here, most of the job is to validate the key names and their values, +// then the values are copied into SubSchema struct +// +func (d *Schema) parseSchema(documentNode interface{}, currentSchema *SubSchema) error { + + if currentSchema.Draft == nil { + if currentSchema.Parent == nil { + return errors.New("Draft not set") + } + currentSchema.Draft = currentSchema.Parent.Draft + } + + // As of draft 6 "true" is equivalent to an empty schema "{}" and false equals "{"not":{}}" + if *currentSchema.Draft >= Draft6 { + if b, isBool := documentNode.(bool); isBool { + currentSchema.pass = &b + return nil + } + } + + m, isMap := documentNode.(map[string]interface{}) + if !isMap { + return errors.New(formatErrorDescription( + Locale.ParseError(), + ErrorDetails{ + "expected": StringSchema, + }, + )) + } + + if currentSchema.Parent == nil { + currentSchema.Ref = &d.DocumentReference + currentSchema.ID = &d.DocumentReference + } + + if currentSchema.ID == nil && currentSchema.Parent != nil { + currentSchema.ID = currentSchema.Parent.ID + } + + // In draft 6 the id keyword was renamed to $id + // Hybrid mode uses the old id by default + var keyID string + + switch *currentSchema.Draft { + case Draft4: + keyID = KeyID + case Hybrid: + keyID = KeyIDNew + if _, found := m[KeyID]; found { + keyID = KeyID + } + default: + keyID = KeyIDNew + } + + if id, err := getString(m, keyID); err != nil { + return err + } else if id != nil { + jsonReference, err := gojsonreference.NewJsonReference(*id) + if err != nil { + return err + } + if currentSchema == d.RootSchema { + currentSchema.ID = &jsonReference + } else { + ref, err := currentSchema.Parent.ID.Inherits(jsonReference) + if err != nil { + return err + } + currentSchema.ID = ref + } + } + + // definitions + if v, ok := m[KeyDefinitions]; ok { + switch mt := v.(type) { + case map[string]interface{}: + for _, dv := range mt { + switch dv.(type) { + case bool, map[string]interface{}: + newSchema := &SubSchema{Property: KeyDefinitions, Parent: currentSchema} + err := d.parseSchema(dv, newSchema) + if err != nil { + return err + } + default: + return invalidType(StringArrayOfSchemas, KeyDefinitions) + } + } + default: + return invalidType(StringArrayOfSchemas, KeyDefinitions) + } + } + + // title + var err error + currentSchema.title, err = getString(m, KeyTitle) + if err != nil { + return err + } + + // description + currentSchema.description, err = getString(m, KeyDescription) + if err != nil { + return err + } + + // $ref + if ref, err := getString(m, KeyRef); err != nil { + return err + } else if ref != nil { + jsonReference, err := gojsonreference.NewJsonReference(*ref) + if err != nil { + return err + } + + currentSchema.Ref = &jsonReference + + if sch, ok := d.ReferencePool.Get(currentSchema.Ref.String()); ok { + currentSchema.RefSchema = sch + } else { + return d.parseReference(documentNode, currentSchema) + } + } + + // type + if typ, found := m[KeyType]; found { + switch t := typ.(type) { + case string: + err := currentSchema.Types.Add(t) + if err != nil { + return err + } + case []interface{}: + for _, typeInArray := range t { + s, isString := typeInArray.(string) + if !isString { + return invalidType(KeyType, TypeString+"/"+StringArrayOfStrings) + } + if err := currentSchema.Types.Add(s); err != nil { + return err + } + } + default: + return invalidType(KeyType, TypeString+"/"+StringArrayOfStrings) + } + } + + // properties + if properties, found := m[KeyProperties]; found { + err := d.parseProperties(properties, currentSchema) + if err != nil { + return err + } + } + + // additionalProperties + if additionalProperties, found := m[KeyAdditionalProperties]; found { + switch v := additionalProperties.(type) { + case bool: + currentSchema.additionalProperties = v + case map[string]interface{}: + newSchema := &SubSchema{Property: KeyAdditionalProperties, Parent: currentSchema, Ref: currentSchema.Ref} + currentSchema.additionalProperties = newSchema + err := d.parseSchema(v, newSchema) + if err != nil { + return errors.New(err.Error()) + } + default: + return invalidType(TypeBoolean+"/"+StringSchema, KeyAdditionalProperties) + } + } + + // patternProperties + if patternProperties, err := getMap(m, KeyPatternProperties); err != nil { + return err + } else if patternProperties != nil { + if len(patternProperties) > 0 { + currentSchema.patternProperties = make(map[string]*SubSchema) + for k, v := range patternProperties { + _, err := regexp.MatchString(k, "") + if err != nil { + return errors.New(formatErrorDescription( + Locale.RegexPattern(), + ErrorDetails{"pattern": k}, + )) + } + newSchema := &SubSchema{Property: k, Parent: currentSchema, Ref: currentSchema.Ref} + err = d.parseSchema(v, newSchema) + if err != nil { + return errors.New(err.Error()) + } + currentSchema.patternProperties[k] = newSchema + } + } + } + + // propertyNames + if propertyNames, found := m[KeyPropertyNames]; found && *currentSchema.Draft >= Draft6 { + switch propertyNames.(type) { + case bool, map[string]interface{}: + newSchema := &SubSchema{Property: KeyPropertyNames, Parent: currentSchema, Ref: currentSchema.Ref} + currentSchema.propertyNames = newSchema + err := d.parseSchema(propertyNames, newSchema) + if err != nil { + return err + } + default: + return errors.New(formatErrorDescription( + Locale.InvalidType(), + ErrorDetails{ + "expected": StringSchema, + "given": KeyPatternProperties, + }, + )) + } + } + + // dependencies + if dependencies, found := m[KeyDependencies]; found { + err := d.parseDependencies(dependencies, currentSchema) + if err != nil { + return err + } + } + + // items + if items, found := m[KeyItems]; found { + switch i := items.(type) { + case []interface{}: + for _, itemElement := range i { + switch itemElement.(type) { + case map[string]interface{}, bool: + newSchema := &SubSchema{Parent: currentSchema, Property: KeyItems} + newSchema.Ref = currentSchema.Ref + currentSchema.ItemsChildren = append(currentSchema.ItemsChildren, newSchema) + err := d.parseSchema(itemElement, newSchema) + if err != nil { + return err + } + default: + return invalidType(StringSchema+"/"+StringArrayOfSchemas, KeyItems) + } + currentSchema.ItemsChildrenIsSingleSchema = false + } + case map[string]interface{}, bool: + newSchema := &SubSchema{Parent: currentSchema, Property: KeyItems} + newSchema.Ref = currentSchema.Ref + currentSchema.ItemsChildren = append(currentSchema.ItemsChildren, newSchema) + err := d.parseSchema(items, newSchema) + if err != nil { + return err + } + currentSchema.ItemsChildrenIsSingleSchema = true + default: + return invalidType(StringSchema+"/"+StringArrayOfSchemas, KeyItems) + } + } + + // additionalItems + if additionalItems, found := m[KeyAdditionalItems]; found { + switch i := additionalItems.(type) { + case bool: + currentSchema.additionalItems = i + case map[string]interface{}: + newSchema := &SubSchema{Property: KeyAdditionalItems, Parent: currentSchema, Ref: currentSchema.Ref} + currentSchema.additionalItems = newSchema + err := d.parseSchema(additionalItems, newSchema) + if err != nil { + return errors.New(err.Error()) + } + default: + return invalidType(TypeBoolean+"/"+StringSchema, KeyAdditionalItems) + } + } + + // validation : number / integer + if multipleOf, found := m[KeyMultipleOf]; found { + multipleOfValue := mustBeNumber(multipleOf) + if multipleOfValue == nil { + return invalidType(StringNumber, KeyMultipleOf) + } + if multipleOfValue.Cmp(big.NewRat(0, 1)) <= 0 { + return errors.New(formatErrorDescription( + Locale.GreaterThanZero(), + ErrorDetails{"number": KeyMultipleOf}, + )) + } + currentSchema.multipleOf = multipleOfValue + } + + if minimum, found := m[KeyMinimum]; found { + minimumValue := mustBeNumber(minimum) + if minimumValue == nil { + return errors.New(formatErrorDescription( + Locale.MustBeOfA(), + ErrorDetails{"x": KeyMinimum, "y": StringNumber}, + )) + } + currentSchema.minimum = minimumValue + } + + if exclusiveMinimum, found := m[KeyExclusiveMinimum]; found { + switch *currentSchema.Draft { + case Draft4: + boolExclusiveMinimum, isBool := exclusiveMinimum.(bool) + if !isBool { + return invalidType(TypeBoolean, KeyExclusiveMinimum) + } + if currentSchema.minimum == nil { + return errors.New(formatErrorDescription( + Locale.CannotBeUsedWithout(), + ErrorDetails{"x": KeyExclusiveMinimum, "y": KeyMinimum}, + )) + } + if boolExclusiveMinimum { + currentSchema.exclusiveMinimum = currentSchema.minimum + currentSchema.minimum = nil + } + case Hybrid: + switch b := exclusiveMinimum.(type) { + case bool: + if currentSchema.minimum == nil { + return errors.New(formatErrorDescription( + Locale.CannotBeUsedWithout(), + ErrorDetails{"x": KeyExclusiveMinimum, "y": KeyMinimum}, + )) + } + if b { + currentSchema.exclusiveMinimum = currentSchema.minimum + currentSchema.minimum = nil + } + case json.Number: + currentSchema.exclusiveMinimum = mustBeNumber(m[KeyExclusiveMinimum]) + default: + return invalidType(TypeBoolean+"/"+TypeNumber, KeyExclusiveMinimum) + } + default: + if isJSONNumber(exclusiveMinimum) { + currentSchema.exclusiveMinimum = mustBeNumber(exclusiveMinimum) + } else { + return invalidType(TypeNumber, KeyExclusiveMinimum) + } + } + } + + if maximum, found := m[KeyMaximum]; found { + maximumValue := mustBeNumber(maximum) + if maximumValue == nil { + return errors.New(formatErrorDescription( + Locale.MustBeOfA(), + ErrorDetails{"x": KeyMaximum, "y": StringNumber}, + )) + } + currentSchema.maximum = maximumValue + } + + if exclusiveMaximum, found := m[KeyExclusiveMaximum]; found { + switch *currentSchema.Draft { + case Draft4: + boolExclusiveMaximum, isBool := exclusiveMaximum.(bool) + if !isBool { + return invalidType(TypeBoolean, KeyExclusiveMaximum) + } + if currentSchema.maximum == nil { + return errors.New(formatErrorDescription( + Locale.CannotBeUsedWithout(), + ErrorDetails{"x": KeyExclusiveMaximum, "y": KeyMaximum}, + )) + } + if boolExclusiveMaximum { + currentSchema.exclusiveMaximum = currentSchema.maximum + currentSchema.maximum = nil + } + case Hybrid: + switch b := exclusiveMaximum.(type) { + case bool: + if currentSchema.maximum == nil { + return errors.New(formatErrorDescription( + Locale.CannotBeUsedWithout(), + ErrorDetails{"x": KeyExclusiveMaximum, "y": KeyMaximum}, + )) + } + if b { + currentSchema.exclusiveMaximum = currentSchema.maximum + currentSchema.maximum = nil + } + case json.Number: + currentSchema.exclusiveMaximum = mustBeNumber(exclusiveMaximum) + default: + return invalidType(TypeBoolean+"/"+TypeNumber, KeyExclusiveMaximum) + } + default: + if isJSONNumber(exclusiveMaximum) { + currentSchema.exclusiveMaximum = mustBeNumber(exclusiveMaximum) + } else { + return invalidType(TypeNumber, KeyExclusiveMaximum) + } + } + } + + // validation : string + + if minLength, found := m[KeyMinLength]; found { + minLengthIntegerValue := mustBeInteger(minLength) + if minLengthIntegerValue == nil { + return errors.New(formatErrorDescription( + Locale.MustBeOfAn(), + ErrorDetails{"x": KeyMinLength, "y": TypeInteger}, + )) + } + if *minLengthIntegerValue < 0 { + return errors.New(formatErrorDescription( + Locale.MustBeGTEZero(), + ErrorDetails{"key": KeyMinLength}, + )) + } + currentSchema.minLength = minLengthIntegerValue + } + + if maxLength, found := m[KeyMaxLength]; found { + maxLengthIntegerValue := mustBeInteger(maxLength) + if maxLengthIntegerValue == nil { + return errors.New(formatErrorDescription( + Locale.MustBeOfAn(), + ErrorDetails{"x": KeyMaxLength, "y": TypeInteger}, + )) + } + if *maxLengthIntegerValue < 0 { + return errors.New(formatErrorDescription( + Locale.MustBeGTEZero(), + ErrorDetails{"key": KeyMaxLength}, + )) + } + currentSchema.maxLength = maxLengthIntegerValue + } + + if currentSchema.minLength != nil && currentSchema.maxLength != nil { + if *currentSchema.minLength > *currentSchema.maxLength { + return errors.New(formatErrorDescription( + Locale.CannotBeGT(), + ErrorDetails{"x": KeyMinLength, "y": KeyMaxLength}, + )) + } + } + + if pattern, err := getString(m, KeyPattern); err != nil { + return err + } else if pattern != nil { + regexpObject, err := regexp.Compile(*pattern) + if err != nil { + return errors.New(formatErrorDescription( + Locale.MustBeValidRegex(), + ErrorDetails{"key": KeyPattern}, + )) + } + currentSchema.pattern = regexpObject + } + + if format, err := getString(m, KeyFormat); err != nil { + return err + } else if format != nil { + currentSchema.format = *format + } + + // validation : object + + if minProperties, found := m[KeyMinProperties]; found { + minPropertiesIntegerValue := mustBeInteger(minProperties) + if minPropertiesIntegerValue == nil { + return errors.New(formatErrorDescription( + Locale.MustBeOfAn(), + ErrorDetails{"x": KeyMinProperties, "y": TypeInteger}, + )) + } + if *minPropertiesIntegerValue < 0 { + return errors.New(formatErrorDescription( + Locale.MustBeGTEZero(), + ErrorDetails{"key": KeyMinProperties}, + )) + } + currentSchema.minProperties = minPropertiesIntegerValue + } + + if maxProperties, found := m[KeyMaxProperties]; found { + maxPropertiesIntegerValue := mustBeInteger(maxProperties) + if maxPropertiesIntegerValue == nil { + return errors.New(formatErrorDescription( + Locale.MustBeOfAn(), + ErrorDetails{"x": KeyMaxProperties, "y": TypeInteger}, + )) + } + if *maxPropertiesIntegerValue < 0 { + return errors.New(formatErrorDescription( + Locale.MustBeGTEZero(), + ErrorDetails{"key": KeyMaxProperties}, + )) + } + currentSchema.maxProperties = maxPropertiesIntegerValue + } + + if currentSchema.minProperties != nil && currentSchema.maxProperties != nil { + if *currentSchema.minProperties > *currentSchema.maxProperties { + return errors.New(formatErrorDescription( + Locale.KeyCannotBeGreaterThan(), + ErrorDetails{"key": KeyMinProperties, "y": KeyMaxProperties}, + )) + } + } + + required, err := getSlice(m, KeyRequired) + if err != nil { + return err + } + for _, requiredValue := range required { + s, isString := requiredValue.(string) + if !isString { + return invalidType(TypeString, KeyRequired) + } else if isStringInSlice(currentSchema.required, s) { + return errors.New(formatErrorDescription( + Locale.KeyItemsMustBeUnique(), + ErrorDetails{"key": KeyRequired}, + )) + } + currentSchema.required = append(currentSchema.required, s) + } + + // validation : array + + if minItems, found := m[KeyMinItems]; found { + minItemsIntegerValue := mustBeInteger(minItems) + if minItemsIntegerValue == nil { + return errors.New(formatErrorDescription( + Locale.MustBeOfAn(), + ErrorDetails{"x": KeyMinItems, "y": TypeInteger}, + )) + } + if *minItemsIntegerValue < 0 { + return errors.New(formatErrorDescription( + Locale.MustBeGTEZero(), + ErrorDetails{"key": KeyMinItems}, + )) + } + currentSchema.minItems = minItemsIntegerValue + } + + if maxItems, found := m[KeyMaxItems]; found { + maxItemsIntegerValue := mustBeInteger(maxItems) + if maxItemsIntegerValue == nil { + return errors.New(formatErrorDescription( + Locale.MustBeOfAn(), + ErrorDetails{"x": KeyMaxItems, "y": TypeInteger}, + )) + } + if *maxItemsIntegerValue < 0 { + return errors.New(formatErrorDescription( + Locale.MustBeGTEZero(), + ErrorDetails{"key": KeyMaxItems}, + )) + } + currentSchema.maxItems = maxItemsIntegerValue + } + + if uniqueItems, found := m[KeyUniqueItems]; found { + bUniqueItems, isBool := uniqueItems.(bool) + if !isBool { + return errors.New(formatErrorDescription( + Locale.MustBeOfA(), + ErrorDetails{"x": KeyUniqueItems, "y": TypeBoolean}, + )) + } + currentSchema.uniqueItems = bUniqueItems + } + + if contains, found := m[KeyContains]; found && *currentSchema.Draft >= Draft6 { + newSchema := &SubSchema{Property: KeyContains, Parent: currentSchema, Ref: currentSchema.Ref} + currentSchema.contains = newSchema + err := d.parseSchema(contains, newSchema) + if err != nil { + return err + } + } + + // validation : all + if vConst, found := m[KeyConst]; found && *currentSchema.Draft >= Draft6 { + is, err := marshalWithoutNumber(vConst) + if err != nil { + return err + } + currentSchema._const = is + } + + enum, err := getSlice(m, KeyEnum) + if err != nil { + return err + } + for _, v := range enum { + is, err := marshalWithoutNumber(v) + if err != nil { + return err + } + if isStringInSlice(currentSchema.enum, *is) { + return errors.New(formatErrorDescription( + Locale.KeyItemsMustBeUnique(), + ErrorDetails{"key": KeyEnum}, + )) + } + currentSchema.enum = append(currentSchema.enum, *is) + } + + // validation : SubSchema + oneOf, err := getSlice(m, KeyOneOf) + if err != nil { + return err + } + for _, v := range oneOf { + newSchema := &SubSchema{Property: KeyOneOf, Parent: currentSchema, Ref: currentSchema.Ref} + currentSchema.oneOf = append(currentSchema.oneOf, newSchema) + err := d.parseSchema(v, newSchema) + if err != nil { + return err + } + } + + anyOf, err := getSlice(m, KeyAnyOf) + if err != nil { + return err + } + for _, v := range anyOf { + newSchema := &SubSchema{Property: KeyAnyOf, Parent: currentSchema, Ref: currentSchema.Ref} + currentSchema.AnyOf = append(currentSchema.AnyOf, newSchema) + err := d.parseSchema(v, newSchema) + if err != nil { + return err + } + } + + allOf, err := getSlice(m, KeyAllOf) + if err != nil { + return err + } + for _, v := range allOf { + newSchema := &SubSchema{Property: KeyAllOf, Parent: currentSchema, Ref: currentSchema.Ref} + currentSchema.AllOf = append(currentSchema.AllOf, newSchema) + err := d.parseSchema(v, newSchema) + if err != nil { + return err + } + } + + if vNot, found := m[KeyNot]; found { + switch vNot.(type) { + case bool, map[string]interface{}: + newSchema := &SubSchema{Property: KeyNot, Parent: currentSchema, Ref: currentSchema.Ref} + currentSchema.not = newSchema + err := d.parseSchema(vNot, newSchema) + if err != nil { + return err + } + default: + return errors.New(formatErrorDescription( + Locale.MustBeOfAn(), + ErrorDetails{"x": KeyNot, "y": TypeObject}, + )) + } + } + + if *currentSchema.Draft >= Draft7 { + if vIf, found := m[KeyIf]; found { + switch vIf.(type) { + case bool, map[string]interface{}: + newSchema := &SubSchema{Property: KeyIf, Parent: currentSchema, Ref: currentSchema.Ref} + currentSchema._if = newSchema + err := d.parseSchema(vIf, newSchema) + if err != nil { + return err + } + default: + return errors.New(formatErrorDescription( + Locale.MustBeOfAn(), + ErrorDetails{"x": KeyIf, "y": TypeObject}, + )) + } + } + + if then, found := m[KeyThen]; found { + switch then.(type) { + case bool, map[string]interface{}: + newSchema := &SubSchema{Property: KeyThen, Parent: currentSchema, Ref: currentSchema.Ref} + currentSchema._then = newSchema + err := d.parseSchema(then, newSchema) + if err != nil { + return err + } + default: + return errors.New(formatErrorDescription( + Locale.MustBeOfAn(), + ErrorDetails{"x": KeyThen, "y": TypeObject}, + )) + } + } + + if vElse, found := m[KeyElse]; found { + switch vElse.(type) { + case bool, map[string]interface{}: + newSchema := &SubSchema{Property: KeyElse, Parent: currentSchema, Ref: currentSchema.Ref} + currentSchema._else = newSchema + err := d.parseSchema(vElse, newSchema) + if err != nil { + return err + } + default: + return errors.New(formatErrorDescription( + Locale.MustBeOfAn(), + ErrorDetails{"x": KeyElse, "y": TypeObject}, + )) + } + } + } + + return nil +} + +func (d *Schema) parseReference(documentNode interface{}, currentSchema *SubSchema) error { + var ( + refdDocumentNode interface{} + dsp *schemaPoolDocument + err error + ) + + newSchema := &SubSchema{Property: KeyRef, Parent: currentSchema, Ref: currentSchema.Ref} + + d.ReferencePool.Add(currentSchema.Ref.String(), newSchema) + + dsp, err = d.Pool.GetDocument(*currentSchema.Ref) + if err != nil { + return err + } + newSchema.ID = currentSchema.Ref + + refdDocumentNode = dsp.Document + newSchema.Draft = dsp.Draft + + switch refdDocumentNode.(type) { + case bool, map[string]interface{}: + // expected + default: + return errors.New(formatErrorDescription( + Locale.MustBeOfType(), + ErrorDetails{"key": StringSchema, "type": TypeObject}, + )) + } + + err = d.parseSchema(refdDocumentNode, newSchema) + if err != nil { + return err + } + + currentSchema.RefSchema = newSchema + + return nil + +} + +func (d *Schema) parseProperties(documentNode interface{}, currentSchema *SubSchema) error { + m, isMap := documentNode.(map[string]interface{}) + if !isMap { + return errors.New(formatErrorDescription( + Locale.MustBeOfType(), + ErrorDetails{"key": StringProperties, "type": TypeObject}, + )) + } + + for k := range m { + schemaProperty := k + newSchema := &SubSchema{Property: schemaProperty, Parent: currentSchema, Ref: currentSchema.Ref} + currentSchema.PropertiesChildren = append(currentSchema.PropertiesChildren, newSchema) + err := d.parseSchema(m[k], newSchema) + if err != nil { + return err + } + } + + return nil +} + +func (d *Schema) parseDependencies(documentNode interface{}, currentSchema *SubSchema) error { + m, isMap := documentNode.(map[string]interface{}) + if !isMap { + return errors.New(formatErrorDescription( + Locale.MustBeOfType(), + ErrorDetails{"key": KeyDependencies, "type": TypeObject}, + )) + } + currentSchema.dependencies = make(map[string]interface{}) + + for k := range m { + switch values := m[k].(type) { + case []interface{}: + var valuesToRegister []string + for _, value := range values { + str, isString := value.(string) + if !isString { + return errors.New(formatErrorDescription( + Locale.MustBeOfType(), + ErrorDetails{ + "key": StringDependency, + "type": StringSchemaOrArrayOfStrings, + }, + )) + } + valuesToRegister = append(valuesToRegister, str) + currentSchema.dependencies[k] = valuesToRegister + } + + case bool, map[string]interface{}: + depSchema := &SubSchema{Property: k, Parent: currentSchema, Ref: currentSchema.Ref} + err := d.parseSchema(m[k], depSchema) + if err != nil { + return err + } + currentSchema.dependencies[k] = depSchema + + default: + return errors.New(formatErrorDescription( + Locale.MustBeOfType(), + ErrorDetails{ + "key": StringDependency, + "type": StringSchemaOrArrayOfStrings, + }, + )) + } + + } + + return nil +} + +func invalidType(expected, given string) error { + return errors.New(formatErrorDescription( + Locale.InvalidType(), + ErrorDetails{ + "expected": expected, + "given": given, + }, + )) +} + +func getString(m map[string]interface{}, key string) (*string, error) { + v, found := m[key] + if !found { + // not found + return nil, nil + } + s, isString := v.(string) + if !isString { + // wrong type + return nil, invalidType(TypeString, key) + } + return &s, nil +} + +func getMap(m map[string]interface{}, key string) (map[string]interface{}, error) { + v, found := m[key] + if !found { + // not found + return nil, nil + } + s, isMap := v.(map[string]interface{}) + if !isMap { + // wrong type + return nil, invalidType(StringSchema, key) + } + return s, nil +} + +func getSlice(m map[string]interface{}, key string) ([]interface{}, error) { + v, found := m[key] + if !found { + return nil, nil + } + s, isArray := v.([]interface{}) + if !isArray { + return nil, errors.New(formatErrorDescription( + Locale.MustBeOfAn(), + ErrorDetails{"x": key, "y": TypeArray}, + )) + } + return s, nil +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/gojsonschema/schemaLoader.go b/vendor/github.com/open-policy-agent/opa/internal/gojsonschema/schemaLoader.go new file mode 100644 index 00000000..d530dee3 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/gojsonschema/schemaLoader.go @@ -0,0 +1,206 @@ +// Copyright 2018 johandorland ( https://github.com/johandorland ) +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. + +package gojsonschema + +import ( + "bytes" + "errors" + + "github.com/xeipuuv/gojsonreference" +) + +// SchemaLoader is used to load schemas +type SchemaLoader struct { + pool *schemaPool + AutoDetect bool + Validate bool + Draft Draft +} + +// NewSchemaLoader creates a new NewSchemaLoader +func NewSchemaLoader() *SchemaLoader { + + ps := &SchemaLoader{ + pool: &schemaPool{ + schemaPoolDocuments: make(map[string]*schemaPoolDocument), + }, + AutoDetect: true, + Validate: false, + Draft: Hybrid, + } + ps.pool.autoDetect = &ps.AutoDetect + + return ps +} + +func (sl *SchemaLoader) validateMetaschema(documentNode interface{}) error { + + var ( + schema string + err error + ) + if sl.AutoDetect { + schema, _, err = parseSchemaURL(documentNode) + if err != nil { + return err + } + } + + // If no explicit "$schema" is used, use the default metaschema associated with the draft used + if schema == "" { + if sl.Draft == Hybrid { + return nil + } + schema = drafts.GetSchemaURL(sl.Draft) + } + + //Disable validation when loading the metaschema to prevent an infinite recursive loop + sl.Validate = false + + metaSchema, err := sl.Compile(NewReferenceLoader(schema)) + + if err != nil { + return err + } + + sl.Validate = true + + result := metaSchema.validateDocument(documentNode) + + if !result.Valid() { + var res bytes.Buffer + for _, err := range result.Errors() { + res.WriteString(err.String()) + res.WriteString("\n") + } + return errors.New(res.String()) + } + + return nil +} + +// AddSchemas adds an arbritrary amount of schemas to the schema cache. As this function does not require +// an explicit URL, every schema should contain an $id, so that it can be referenced by the main schema +func (sl *SchemaLoader) AddSchemas(loaders ...JSONLoader) error { + emptyRef, _ := gojsonreference.NewJsonReference("") + + for _, loader := range loaders { + doc, err := loader.LoadJSON() + + if err != nil { + return err + } + + if sl.Validate { + if err := sl.validateMetaschema(doc); err != nil { + return err + } + } + + // Directly use the Recursive function, so that it get only added to the schema Pool by $id + // and not by the ref of the document as it's empty + if err = sl.pool.parseReferences(doc, emptyRef, false); err != nil { + return err + } + } + + return nil +} + +//AddSchema adds a schema under the provided URL to the schema cache +func (sl *SchemaLoader) AddSchema(url string, loader JSONLoader) error { + + ref, err := gojsonreference.NewJsonReference(url) + + if err != nil { + return err + } + + doc, err := loader.LoadJSON() + + if err != nil { + return err + } + + if sl.Validate { + if err := sl.validateMetaschema(doc); err != nil { + return err + } + } + + return sl.pool.parseReferences(doc, ref, true) +} + +// Compile loads and compiles a schema +func (sl *SchemaLoader) Compile(rootSchema JSONLoader) (*Schema, error) { + + ref, err := rootSchema.JSONReference() + + if err != nil { + return nil, err + } + + d := Schema{} + d.Pool = sl.pool + d.Pool.jsonLoaderFactory = rootSchema.LoaderFactory() + d.DocumentReference = ref + d.ReferencePool = newSchemaReferencePool() + + var doc interface{} + if ref.String() != "" { + // Get document from schema pool + spd, err := d.Pool.GetDocument(d.DocumentReference) + if err != nil { + return nil, err + } + doc = spd.Document + } else { + // Load JSON directly + doc, err = rootSchema.LoadJSON() + if err != nil { + return nil, err + } + // References need only be parsed if loading JSON directly + // as pool.GetDocument already does this for us if loading by reference + err = sl.pool.parseReferences(doc, ref, true) + if err != nil { + return nil, err + } + } + + if sl.Validate { + if err := sl.validateMetaschema(doc); err != nil { + return nil, err + } + } + + draft := sl.Draft + if sl.AutoDetect { + _, detectedDraft, err := parseSchemaURL(doc) + if err != nil { + return nil, err + } + if detectedDraft != nil { + draft = *detectedDraft + } + } + + err = d.parse(doc, draft) + if err != nil { + return nil, err + } + + return &d, nil +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/gojsonschema/schemaPool.go b/vendor/github.com/open-policy-agent/opa/internal/gojsonschema/schemaPool.go new file mode 100644 index 00000000..ed8ff688 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/gojsonschema/schemaPool.go @@ -0,0 +1,230 @@ +// Copyright 2015 xeipuuv ( https://github.com/xeipuuv ) +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. + +// author xeipuuv +// author-github https://github.com/xeipuuv +// author-mail xeipuuv@gmail.com +// +// repository-name gojsonschema +// repository-desc An implementation of JSON Schema, based on IETF's draft v4 - Go language. +// +// description Defines resources pooling. +// Eases referencing and avoids downloading the same resource twice. +// +// created 26-02-2013 + +package gojsonschema + +import ( + "errors" + "fmt" + + "github.com/xeipuuv/gojsonreference" +) + +type schemaPoolDocument struct { + Document interface{} + Draft *Draft +} + +type schemaPool struct { + schemaPoolDocuments map[string]*schemaPoolDocument + jsonLoaderFactory JSONLoaderFactory + autoDetect *bool +} + +func (p *schemaPool) parseReferences(document interface{}, ref gojsonreference.JsonReference, pooled bool) error { + + var ( + draft *Draft + err error + reference = ref.String() + ) + // Only the root document should be added to the schema pool if pooled is true + if _, ok := p.schemaPoolDocuments[reference]; pooled && ok { + return fmt.Errorf("Reference already exists: \"%s\"", reference) + } + + if *p.autoDetect { + _, draft, err = parseSchemaURL(document) + if err != nil { + return err + } + } + + err = p.parseReferencesRecursive(document, ref, draft) + + if pooled { + p.schemaPoolDocuments[reference] = &schemaPoolDocument{Document: document, Draft: draft} + } + + return err +} + +func (p *schemaPool) parseReferencesRecursive(document interface{}, ref gojsonreference.JsonReference, draft *Draft) error { + // parseReferencesRecursive parses a JSON document and resolves all $id and $ref references. + // For $ref references it takes into account the $id scope it is in and replaces + // the reference by the absolute resolved reference + + // When encountering errors it fails silently. Error handling is done when the schema + // is syntactically parsed and any error encountered here should also come up there. + switch m := document.(type) { + case []interface{}: + for _, v := range m { + err := p.parseReferencesRecursive(v, ref, draft) + if err != nil { + return err + } + } + case map[string]interface{}: + localRef := &ref + + keyID := KeyIDNew + if _, ok := m[KeyID]; ok { + keyID = KeyID + } + if v, ok := m[keyID]; ok { + if value, isString := v.(string); isString { + jsonReference, err := gojsonreference.NewJsonReference(value) + if err == nil { + localRef, err = ref.Inherits(jsonReference) + if err == nil { + if _, ok := p.schemaPoolDocuments[localRef.String()]; ok { + return fmt.Errorf("Reference already exists: \"%s\"", localRef.String()) + } + p.schemaPoolDocuments[localRef.String()] = &schemaPoolDocument{Document: document, Draft: draft} + } + } + } + } + + if v, ok := m[KeyRef]; ok { + if s, isString := v.(string); isString { + jsonReference, err := gojsonreference.NewJsonReference(s) + if err == nil { + absoluteRef, err := localRef.Inherits(jsonReference) + if err == nil { + m[KeyRef] = absoluteRef.String() + } + } + } + } + + for k, v := range m { + // const and enums should be interpreted literally, so ignore them + if k == KeyConst || k == KeyEnum { + continue + } + // Something like a property or a dependency is not a valid schema, as it might describe properties named "$ref", "$id" or "const", etc + // Therefore don't treat it like a schema. + if k == KeyProperties || k == KeyDependencies || k == KeyPatternProperties { + if child, ok := v.(map[string]interface{}); ok { + for _, v := range child { + err := p.parseReferencesRecursive(v, *localRef, draft) + if err != nil { + return err + } + } + } + } else { + err := p.parseReferencesRecursive(v, *localRef, draft) + if err != nil { + return err + } + } + } + } + return nil +} + +func (p *schemaPool) GetDocument(reference gojsonreference.JsonReference) (*schemaPoolDocument, error) { + + var ( + spd *schemaPoolDocument + draft *Draft + ok bool + err error + ) + + if internalLogEnabled { + internalLog("Get Document ( %s )", reference.String()) + } + + // Create a deep copy, so we can remove the fragment part later on without altering the original + refToURL, _ := gojsonreference.NewJsonReference(reference.String()) + + // First check if the given fragment is a location independent identifier + // http://json-schema.org/latest/json-schema-core.html#rfc.section.8.2.3 + + if spd, ok = p.schemaPoolDocuments[refToURL.String()]; ok { + if internalLogEnabled { + internalLog(" From pool") + } + return spd, nil + } + + // If the given reference is not a location independent identifier, + // strip the fragment and look for a document with it's base URI + + refToURL.GetUrl().Fragment = "" + + if cachedSpd, ok := p.schemaPoolDocuments[refToURL.String()]; ok { + document, _, err := reference.GetPointer().Get(cachedSpd.Document) + + if err != nil { + return nil, err + } + + if internalLogEnabled { + internalLog(" From pool") + } + + spd = &schemaPoolDocument{Document: document, Draft: cachedSpd.Draft} + p.schemaPoolDocuments[reference.String()] = spd + + return spd, nil + } + + // It is not possible to load anything remotely that is not canonical... + if !reference.IsCanonical() { + return nil, errors.New(formatErrorDescription( + Locale.ReferenceMustBeCanonical(), + ErrorDetails{"reference": reference.String()}, + )) + } + + jsonReferenceLoader := p.jsonLoaderFactory.New(reference.String()) + document, err := jsonReferenceLoader.LoadJSON() + + if err != nil { + return nil, err + } + + // add the whole document to the pool for potential re-use + err = p.parseReferences(document, refToURL, true) + if err != nil { + return nil, err + } + + _, draft, _ = parseSchemaURL(document) + + // resolve the potential fragment and also cache it + document, _, err = reference.GetPointer().Get(document) + + if err != nil { + return nil, err + } + + return &schemaPoolDocument{Document: document, Draft: draft}, nil +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/gojsonschema/schemaReferencePool.go b/vendor/github.com/open-policy-agent/opa/internal/gojsonschema/schemaReferencePool.go new file mode 100644 index 00000000..876419f5 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/gojsonschema/schemaReferencePool.go @@ -0,0 +1,68 @@ +// Copyright 2015 xeipuuv ( https://github.com/xeipuuv ) +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. + +// author xeipuuv +// author-github https://github.com/xeipuuv +// author-mail xeipuuv@gmail.com +// +// repository-name gojsonschema +// repository-desc An implementation of JSON Schema, based on IETF's draft v4 - Go language. +// +// description Pool of referenced schemas. +// +// created 25-06-2013 + +package gojsonschema + +import ( + "fmt" +) + +type schemaReferencePool struct { + documents map[string]*SubSchema +} + +func newSchemaReferencePool() *schemaReferencePool { + + p := &schemaReferencePool{} + p.documents = make(map[string]*SubSchema) + + return p +} + +func (p *schemaReferencePool) Get(ref string) (r *SubSchema, o bool) { + + if internalLogEnabled { + internalLog(fmt.Sprintf("Schema Reference ( %s )", ref)) + } + + if sch, ok := p.documents[ref]; ok { + if internalLogEnabled { + internalLog(" From pool") + } + return sch, true + } + + return nil, false +} + +func (p *schemaReferencePool) Add(ref string, sch *SubSchema) { + + if internalLogEnabled { + internalLog(fmt.Sprintf("Add Schema Reference %s to pool", ref)) + } + if _, ok := p.documents[ref]; !ok { + p.documents[ref] = sch + } +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/gojsonschema/schemaType.go b/vendor/github.com/open-policy-agent/opa/internal/gojsonschema/schemaType.go new file mode 100644 index 00000000..271832d3 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/gojsonschema/schemaType.go @@ -0,0 +1,83 @@ +// Copyright 2015 xeipuuv ( https://github.com/xeipuuv ) +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. + +// author xeipuuv +// author-github https://github.com/xeipuuv +// author-mail xeipuuv@gmail.com +// +// repository-name gojsonschema +// repository-desc An implementation of JSON Schema, based on IETF's draft v4 - Go language. +// +// description Helper structure to handle schema types, and the combination of them. +// +// created 28-02-2013 + +package gojsonschema + +import ( + "errors" + "fmt" + "strings" +) + +type jsonSchemaType struct { + types []string +} + +// Is the schema typed ? that is containing at least one type +// When not typed, the schema does not need any type validation +func (t *jsonSchemaType) IsTyped() bool { + return len(t.types) > 0 +} + +func (t *jsonSchemaType) Add(etype string) error { + + if !isStringInSlice(JSONTypes, etype) { + return errors.New(formatErrorDescription(Locale.NotAValidType(), ErrorDetails{"given": "/" + etype + "/", "expected": JSONTypes})) + } + + if t.Contains(etype) { + return errors.New(formatErrorDescription(Locale.Duplicated(), ErrorDetails{"type": etype})) + } + + t.types = append(t.types, etype) + + return nil +} + +func (t *jsonSchemaType) Contains(etype string) bool { + + for _, v := range t.types { + if v == etype { + return true + } + } + + return false +} + +func (t *jsonSchemaType) String() string { + + if len(t.types) == 0 { + return StringUndefined // should never happen + } + + // Displayed as a list [type1,type2,...] + if len(t.types) > 1 { + return fmt.Sprintf("[%s]", strings.Join(t.types, ",")) + } + + // Only one type: name only + return t.types[0] +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/gojsonschema/subSchema.go b/vendor/github.com/open-policy-agent/opa/internal/gojsonschema/subSchema.go new file mode 100644 index 00000000..d8bc0cb5 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/gojsonschema/subSchema.go @@ -0,0 +1,151 @@ +// Copyright 2015 xeipuuv ( https://github.com/xeipuuv ) +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. + +// author xeipuuv +// author-github https://github.com/xeipuuv +// author-mail xeipuuv@gmail.com +// +// repository-name gojsonschema +// repository-desc An implementation of JSON Schema, based on IETF's draft v4 - Go language. +// +// description Defines the structure of a sub-SubSchema. +// A sub-SubSchema can contain other sub-schemas. +// +// created 27-02-2013 + +package gojsonschema + +import ( + "math/big" + "regexp" + + "github.com/xeipuuv/gojsonreference" +) + +// Constants +const ( + KeySchema = "$schema" + KeyID = "id" + KeyIDNew = "$id" + KeyRef = "$ref" + KeyTitle = "title" + KeyDescription = "description" + KeyType = "type" + KeyItems = "items" + KeyAdditionalItems = "additionalItems" + KeyProperties = "properties" + KeyPatternProperties = "patternProperties" + KeyAdditionalProperties = "additionalProperties" + KeyPropertyNames = "propertyNames" + KeyDefinitions = "definitions" + KeyMultipleOf = "multipleOf" + KeyMinimum = "minimum" + KeyMaximum = "maximum" + KeyExclusiveMinimum = "exclusiveMinimum" + KeyExclusiveMaximum = "exclusiveMaximum" + KeyMinLength = "minLength" + KeyMaxLength = "maxLength" + KeyPattern = "pattern" + KeyFormat = "format" + KeyMinProperties = "minProperties" + KeyMaxProperties = "maxProperties" + KeyDependencies = "dependencies" + KeyRequired = "required" + KeyMinItems = "minItems" + KeyMaxItems = "maxItems" + KeyUniqueItems = "uniqueItems" + KeyContains = "contains" + KeyConst = "const" + KeyEnum = "enum" + KeyOneOf = "oneOf" + KeyAnyOf = "anyOf" + KeyAllOf = "allOf" + KeyNot = "not" + KeyIf = "if" + KeyThen = "then" + KeyElse = "else" +) + +// SubSchema holds a sub schema +type SubSchema struct { + Draft *Draft + + // basic SubSchema meta properties + ID *gojsonreference.JsonReference + title *string + description *string + + Property string + + // Quick pass/fail for boolean schemas + pass *bool + + // Types associated with the SubSchema + Types jsonSchemaType + + // Reference url + Ref *gojsonreference.JsonReference + // Schema referenced + RefSchema *SubSchema + + // hierarchy + Parent *SubSchema + ItemsChildren []*SubSchema + ItemsChildrenIsSingleSchema bool + PropertiesChildren []*SubSchema + + // validation : number / integer + multipleOf *big.Rat + maximum *big.Rat + exclusiveMaximum *big.Rat + minimum *big.Rat + exclusiveMinimum *big.Rat + + // validation : string + minLength *int + maxLength *int + pattern *regexp.Regexp + format string + + // validation : object + minProperties *int + maxProperties *int + required []string + + dependencies map[string]interface{} + additionalProperties interface{} + patternProperties map[string]*SubSchema + propertyNames *SubSchema + + // validation : array + minItems *int + maxItems *int + uniqueItems bool + contains *SubSchema + + additionalItems interface{} + + // validation : all + _const *string //const is a golang keyword + enum []string + + // validation : SubSchema + oneOf []*SubSchema + AnyOf []*SubSchema + AllOf []*SubSchema + not *SubSchema + _if *SubSchema // if/else are golang keywords + _then *SubSchema + _else *SubSchema +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/gojsonschema/types.go b/vendor/github.com/open-policy-agent/opa/internal/gojsonschema/types.go new file mode 100644 index 00000000..df2d4b2f --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/gojsonschema/types.go @@ -0,0 +1,62 @@ +// Copyright 2015 xeipuuv ( https://github.com/xeipuuv ) +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. + +// author xeipuuv +// author-github https://github.com/xeipuuv +// author-mail xeipuuv@gmail.com +// +// repository-name gojsonschema +// repository-desc An implementation of JSON Schema, based on IETF's draft v4 - Go language. +// +// description Contains const types for schema and JSON. +// +// created 28-02-2013 + +package gojsonschema + +// Type constants +const ( + TypeArray = `array` + TypeBoolean = `boolean` + TypeInteger = `integer` + TypeNumber = `number` + TypeNull = `null` + TypeObject = `object` + TypeString = `string` +) + +// JSONTypes hosts the list of type that are supported in JSON +var JSONTypes []string + +// SchemaTypes Hosts The List Of Type That Are Supported In Schemas +var SchemaTypes []string + +func init() { + JSONTypes = []string{ + TypeArray, + TypeBoolean, + TypeInteger, + TypeNumber, + TypeNull, + TypeObject, + TypeString} + + SchemaTypes = []string{ + TypeArray, + TypeBoolean, + TypeInteger, + TypeNumber, + TypeObject, + TypeString} +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/gojsonschema/utils.go b/vendor/github.com/open-policy-agent/opa/internal/gojsonschema/utils.go new file mode 100644 index 00000000..fd0f1870 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/gojsonschema/utils.go @@ -0,0 +1,165 @@ +// Copyright 2015 xeipuuv ( https://github.com/xeipuuv ) +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. + +// author xeipuuv +// author-github https://github.com/xeipuuv +// author-mail xeipuuv@gmail.com +// +// repository-name gojsonschema +// repository-desc An implementation of JSON Schema, based on IETF's draft v4 - Go language. +// +// description Various utility functions. +// +// created 26-02-2013 + +// nolint: deadcode,unused,varcheck // Package in development (2021). +package gojsonschema + +import ( + "encoding/json" + "math/big" +) + +func isStringInSlice(s []string, what string) bool { + for i := range s { + if s[i] == what { + return true + } + } + return false +} + +func marshalToJSONString(value interface{}) (*string, error) { + + mBytes, err := json.Marshal(value) + if err != nil { + return nil, err + } + + sBytes := string(mBytes) + return &sBytes, nil +} + +func marshalWithoutNumber(value interface{}) (*string, error) { + + // The JSON is decoded using https://golang.org/pkg/encoding/json/#Decoder.UseNumber + // This means the numbers are internally still represented as strings and therefore 1.00 is unequal to 1 + // One way to eliminate these differences is to decode and encode the JSON one more time without Decoder.UseNumber + // so that these differences in representation are removed + + jsonString, err := marshalToJSONString(value) + if err != nil { + return nil, err + } + + var document interface{} + + err = json.Unmarshal([]byte(*jsonString), &document) + if err != nil { + return nil, err + } + + return marshalToJSONString(document) +} + +func isJSONNumber(what interface{}) bool { + + switch what.(type) { + + case json.Number: + return true + } + + return false +} + +func checkJSONInteger(what interface{}) (isInt bool) { + + jsonNumber := what.(json.Number) + + bigFloat, isValidNumber := new(big.Rat).SetString(string(jsonNumber)) + + return isValidNumber && bigFloat.IsInt() + +} + +// same as ECMA Number.MAX_SAFE_INTEGER and Number.MIN_SAFE_INTEGER +const ( + maxJSONFloat = float64(1<<53 - 1) // 9007199254740991.0 2^53 - 1 + minJSONFloat = -float64(1<<53 - 1) //-9007199254740991.0 -2^53 - 1 +) + +func mustBeInteger(what interface{}) *int { + number, ok := what.(json.Number) + if !ok { + return nil + } + + isInt := checkJSONInteger(number) + if !isInt { + return nil + } + + int64Value, err := number.Int64() + if err != nil { + return nil + } + + // This doesn't actually convert to an int32 value; it converts to the + // system-specific default integer. Assuming this is a valid int32 could cause + // bugs. + int32Value := int(int64Value) + return &int32Value +} + +func mustBeNumber(what interface{}) *big.Rat { + number, ok := what.(json.Number) + if !ok { + return nil + } + + float64Value, success := new(big.Rat).SetString(string(number)) + if success { + return float64Value + } + return nil +} + +func convertDocumentNode(val interface{}) interface{} { + + if lval, ok := val.([]interface{}); ok { + + res := []interface{}{} + for _, v := range lval { + res = append(res, convertDocumentNode(v)) + } + + return res + + } + + if mval, ok := val.(map[interface{}]interface{}); ok { + + res := map[string]interface{}{} + + for k, v := range mval { + res[k.(string)] = convertDocumentNode(v) + } + + return res + + } + + return val +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/gojsonschema/validation.go b/vendor/github.com/open-policy-agent/opa/internal/gojsonschema/validation.go new file mode 100644 index 00000000..a3c3e014 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/gojsonschema/validation.go @@ -0,0 +1,837 @@ +// Copyright 2015 xeipuuv ( https://github.com/xeipuuv ) +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. + +// author xeipuuv +// author-github https://github.com/xeipuuv +// author-mail xeipuuv@gmail.com +// +// repository-name gojsonschema +// repository-desc An implementation of JSON Schema, based on IETF's draft v4 - Go language. +// +// description Extends Schema and SubSchema, implements the validation phase. +// +// created 28-02-2013 + +package gojsonschema + +import ( + "encoding/json" + "math/big" + "reflect" + "regexp" + "strconv" + "strings" + "unicode/utf8" +) + +// Validate loads and validates a JSON schema +func Validate(ls JSONLoader, ld JSONLoader) (*Result, error) { + // load schema + schema, err := NewSchema(ls) + if err != nil { + return nil, err + } + return schema.Validate(ld) +} + +// Validate loads and validates a JSON document +func (v *Schema) Validate(l JSONLoader) (*Result, error) { + root, err := l.LoadJSON() + if err != nil { + return nil, err + } + return v.validateDocument(root), nil +} + +func (v *Schema) validateDocument(root interface{}) *Result { + result := &Result{} + context := NewJSONContext(StringContextRoot, nil) + v.RootSchema.validateRecursive(v.RootSchema, root, result, context) + return result +} + +func (v *SubSchema) subValidateWithContext(document interface{}, context *JSONContext) *Result { + result := &Result{} + v.validateRecursive(v, document, result, context) + return result +} + +// Walker function to validate the json recursively against the SubSchema +func (v *SubSchema) validateRecursive(currentSubSchema *SubSchema, currentNode interface{}, result *Result, context *JSONContext) { + + if internalLogEnabled { + internalLog("validateRecursive %s", context.String()) + internalLog(" %v", currentNode) + } + + // Handle true/false schema as early as possible as all other fields will be nil + if currentSubSchema.pass != nil { + if !*currentSubSchema.pass { + result.addInternalError( + new(FalseError), + context, + currentNode, + ErrorDetails{}, + ) + } + return + } + + // Handle referenced schemas, returns directly when a $ref is found + if currentSubSchema.RefSchema != nil { + v.validateRecursive(currentSubSchema.RefSchema, currentNode, result, context) + return + } + + // Check for null value + if currentNode == nil { + if currentSubSchema.Types.IsTyped() && !currentSubSchema.Types.Contains(TypeNull) { + result.addInternalError( + new(InvalidTypeError), + context, + currentNode, + ErrorDetails{ + "expected": currentSubSchema.Types.String(), + "given": TypeNull, + }, + ) + return + } + + currentSubSchema.validateSchema(currentSubSchema, currentNode, result, context) + v.validateCommon(currentSubSchema, currentNode, result, context) + + } else { // Not a null value + + if value, isNumber := currentNode.(json.Number); isNumber { + isInt := checkJSONInteger(value) + + validType := currentSubSchema.Types.Contains(TypeNumber) || (isInt && currentSubSchema.Types.Contains(TypeInteger)) + + if currentSubSchema.Types.IsTyped() && !validType { + + givenType := TypeInteger + if !isInt { + givenType = TypeNumber + } + + result.addInternalError( + new(InvalidTypeError), + context, + currentNode, + ErrorDetails{ + "expected": currentSubSchema.Types.String(), + "given": givenType, + }, + ) + return + } + + currentSubSchema.validateSchema(currentSubSchema, value, result, context) + v.validateNumber(currentSubSchema, value, result, context) + v.validateCommon(currentSubSchema, value, result, context) + v.validateString(currentSubSchema, value, result, context) + + } else { + + rValue := reflect.ValueOf(currentNode) + rKind := rValue.Kind() + + switch rKind { + + // Slice => JSON array + + case reflect.Slice: + + if currentSubSchema.Types.IsTyped() && !currentSubSchema.Types.Contains(TypeArray) { + result.addInternalError( + new(InvalidTypeError), + context, + currentNode, + ErrorDetails{ + "expected": currentSubSchema.Types.String(), + "given": TypeArray, + }, + ) + return + } + + castCurrentNode := currentNode.([]interface{}) + + currentSubSchema.validateSchema(currentSubSchema, castCurrentNode, result, context) + + v.validateArray(currentSubSchema, castCurrentNode, result, context) + v.validateCommon(currentSubSchema, castCurrentNode, result, context) + + // Map => JSON object + + case reflect.Map: + if currentSubSchema.Types.IsTyped() && !currentSubSchema.Types.Contains(TypeObject) { + result.addInternalError( + new(InvalidTypeError), + context, + currentNode, + ErrorDetails{ + "expected": currentSubSchema.Types.String(), + "given": TypeObject, + }, + ) + return + } + + castCurrentNode, ok := currentNode.(map[string]interface{}) + if !ok { + castCurrentNode = convertDocumentNode(currentNode).(map[string]interface{}) + } + + currentSubSchema.validateSchema(currentSubSchema, castCurrentNode, result, context) + + v.validateObject(currentSubSchema, castCurrentNode, result, context) + v.validateCommon(currentSubSchema, castCurrentNode, result, context) + + for _, pSchema := range currentSubSchema.PropertiesChildren { + nextNode, ok := castCurrentNode[pSchema.Property] + if ok { + subContext := NewJSONContext(pSchema.Property, context) + v.validateRecursive(pSchema, nextNode, result, subContext) + } + } + + // Simple JSON values : string, number, boolean + + case reflect.Bool: + + if currentSubSchema.Types.IsTyped() && !currentSubSchema.Types.Contains(TypeBoolean) { + result.addInternalError( + new(InvalidTypeError), + context, + currentNode, + ErrorDetails{ + "expected": currentSubSchema.Types.String(), + "given": TypeBoolean, + }, + ) + return + } + + value := currentNode.(bool) + + currentSubSchema.validateSchema(currentSubSchema, value, result, context) + v.validateNumber(currentSubSchema, value, result, context) + v.validateCommon(currentSubSchema, value, result, context) + v.validateString(currentSubSchema, value, result, context) + + case reflect.String: + + if currentSubSchema.Types.IsTyped() && !currentSubSchema.Types.Contains(TypeString) { + result.addInternalError( + new(InvalidTypeError), + context, + currentNode, + ErrorDetails{ + "expected": currentSubSchema.Types.String(), + "given": TypeString, + }, + ) + return + } + + value := currentNode.(string) + + currentSubSchema.validateSchema(currentSubSchema, value, result, context) + v.validateNumber(currentSubSchema, value, result, context) + v.validateCommon(currentSubSchema, value, result, context) + v.validateString(currentSubSchema, value, result, context) + + } + + } + + } + + result.incrementScore() +} + +// Different kinds of validation there, SubSchema / common / array / object / string... +func (v *SubSchema) validateSchema(currentSubSchema *SubSchema, currentNode interface{}, result *Result, context *JSONContext) { + + if internalLogEnabled { + internalLog("validateSchema %s", context.String()) + internalLog(" %v", currentNode) + } + + if len(currentSubSchema.AnyOf) > 0 { + + validatedAnyOf := false + var bestValidationResult *Result + + for _, anyOfSchema := range currentSubSchema.AnyOf { + if !validatedAnyOf { + validationResult := anyOfSchema.subValidateWithContext(currentNode, context) + validatedAnyOf = validationResult.Valid() + + if !validatedAnyOf && (bestValidationResult == nil || validationResult.score > bestValidationResult.score) { + bestValidationResult = validationResult + } + } + } + if !validatedAnyOf { + + result.addInternalError(new(NumberAnyOfError), context, currentNode, ErrorDetails{}) + + if bestValidationResult != nil { + // add error messages of closest matching SubSchema as + // that's probably the one the user was trying to match + result.mergeErrors(bestValidationResult) + } + } + } + + if len(currentSubSchema.oneOf) > 0 { + + nbValidated := 0 + var bestValidationResult *Result + + for _, oneOfSchema := range currentSubSchema.oneOf { + validationResult := oneOfSchema.subValidateWithContext(currentNode, context) + if validationResult.Valid() { + nbValidated++ + } else if nbValidated == 0 && (bestValidationResult == nil || validationResult.score > bestValidationResult.score) { + bestValidationResult = validationResult + } + } + + if nbValidated != 1 { + + result.addInternalError(new(NumberOneOfError), context, currentNode, ErrorDetails{}) + + if nbValidated == 0 { + // add error messages of closest matching SubSchema as + // that's probably the one the user was trying to match + result.mergeErrors(bestValidationResult) + } + } + + } + + if len(currentSubSchema.AllOf) > 0 { + nbValidated := 0 + + for _, allOfSchema := range currentSubSchema.AllOf { + validationResult := allOfSchema.subValidateWithContext(currentNode, context) + if validationResult.Valid() { + nbValidated++ + } + result.mergeErrors(validationResult) + } + + if nbValidated != len(currentSubSchema.AllOf) { + result.addInternalError(new(NumberAllOfError), context, currentNode, ErrorDetails{}) + } + } + + if currentSubSchema.not != nil { + validationResult := currentSubSchema.not.subValidateWithContext(currentNode, context) + if validationResult.Valid() { + result.addInternalError(new(NumberNotError), context, currentNode, ErrorDetails{}) + } + } + + if currentSubSchema.dependencies != nil && len(currentSubSchema.dependencies) > 0 { + if currentNodeMap, ok := currentNode.(map[string]interface{}); ok { + for elementKey := range currentNodeMap { + if dependency, ok := currentSubSchema.dependencies[elementKey]; ok { + switch dependency := dependency.(type) { + + case []string: + for _, dependOnKey := range dependency { + if _, dependencyResolved := currentNode.(map[string]interface{})[dependOnKey]; !dependencyResolved { + result.addInternalError( + new(MissingDependencyError), + context, + currentNode, + ErrorDetails{"dependency": dependOnKey}, + ) + } + } + + case *SubSchema: + dependency.validateRecursive(dependency, currentNode, result, context) + } + } + } + } + } + + if currentSubSchema._if != nil { + validationResultIf := currentSubSchema._if.subValidateWithContext(currentNode, context) + if currentSubSchema._then != nil && validationResultIf.Valid() { + validationResultThen := currentSubSchema._then.subValidateWithContext(currentNode, context) + if !validationResultThen.Valid() { + result.addInternalError(new(ConditionThenError), context, currentNode, ErrorDetails{}) + result.mergeErrors(validationResultThen) + } + } + if currentSubSchema._else != nil && !validationResultIf.Valid() { + validationResultElse := currentSubSchema._else.subValidateWithContext(currentNode, context) + if !validationResultElse.Valid() { + result.addInternalError(new(ConditionElseError), context, currentNode, ErrorDetails{}) + result.mergeErrors(validationResultElse) + } + } + } + + result.incrementScore() +} + +func (v *SubSchema) validateCommon(currentSubSchema *SubSchema, value interface{}, result *Result, context *JSONContext) { + + if internalLogEnabled { + internalLog("validateCommon %s", context.String()) + internalLog(" %v", value) + } + + // const: + if currentSubSchema._const != nil { + vString, err := marshalWithoutNumber(value) + if err != nil { + result.addInternalError(new(InternalError), context, value, ErrorDetails{"error": err}) + } + if *vString != *currentSubSchema._const { + result.addInternalError(new(ConstError), + context, + value, + ErrorDetails{ + "allowed": *currentSubSchema._const, + }, + ) + } + } + + // enum: + if len(currentSubSchema.enum) > 0 { + vString, err := marshalWithoutNumber(value) + if err != nil { + result.addInternalError(new(InternalError), context, value, ErrorDetails{"error": err}) + } + if !isStringInSlice(currentSubSchema.enum, *vString) { + result.addInternalError( + new(EnumError), + context, + value, + ErrorDetails{ + "allowed": strings.Join(currentSubSchema.enum, ", "), + }, + ) + } + } + + // format: + if currentSubSchema.format != "" { + if !FormatCheckers.IsFormat(currentSubSchema.format, value) { + result.addInternalError( + new(DoesNotMatchFormatError), + context, + value, + ErrorDetails{"format": currentSubSchema.format}, + ) + } + } + + result.incrementScore() +} + +func (v *SubSchema) validateArray(currentSubSchema *SubSchema, value []interface{}, result *Result, context *JSONContext) { + + if internalLogEnabled { + internalLog("validateArray %s", context.String()) + internalLog(" %v", value) + } + + nbValues := len(value) + + // TODO explain + if currentSubSchema.ItemsChildrenIsSingleSchema { + for i := range value { + subContext := NewJSONContext(strconv.Itoa(i), context) + validationResult := currentSubSchema.ItemsChildren[0].subValidateWithContext(value[i], subContext) + result.mergeErrors(validationResult) + } + } else { + if currentSubSchema.ItemsChildren != nil && len(currentSubSchema.ItemsChildren) > 0 { + + nbItems := len(currentSubSchema.ItemsChildren) + + // while we have both schemas and values, check them against each other + for i := 0; i != nbItems && i != nbValues; i++ { + subContext := NewJSONContext(strconv.Itoa(i), context) + validationResult := currentSubSchema.ItemsChildren[i].subValidateWithContext(value[i], subContext) + result.mergeErrors(validationResult) + } + + if nbItems < nbValues { + // we have less schemas than elements in the instance array, + // but that might be ok if "additionalItems" is specified. + + switch currentSubSchema.additionalItems.(type) { + case bool: + if !currentSubSchema.additionalItems.(bool) { + result.addInternalError(new(ArrayNoAdditionalItemsError), context, value, ErrorDetails{}) + } + case *SubSchema: + additionalItemSchema := currentSubSchema.additionalItems.(*SubSchema) + for i := nbItems; i != nbValues; i++ { + subContext := NewJSONContext(strconv.Itoa(i), context) + validationResult := additionalItemSchema.subValidateWithContext(value[i], subContext) + result.mergeErrors(validationResult) + } + } + } + } + } + + // minItems & maxItems + if currentSubSchema.minItems != nil { + if nbValues < int(*currentSubSchema.minItems) { + result.addInternalError( + new(ArrayMinItemsError), + context, + value, + ErrorDetails{"min": *currentSubSchema.minItems}, + ) + } + } + if currentSubSchema.maxItems != nil { + if nbValues > int(*currentSubSchema.maxItems) { + result.addInternalError( + new(ArrayMaxItemsError), + context, + value, + ErrorDetails{"max": *currentSubSchema.maxItems}, + ) + } + } + + // uniqueItems: + if currentSubSchema.uniqueItems { + var stringifiedItems = make(map[string]int) + for j, v := range value { + vString, err := marshalWithoutNumber(v) + if err != nil { + result.addInternalError(new(InternalError), context, value, ErrorDetails{"err": err}) + } + if i, ok := stringifiedItems[*vString]; ok { + result.addInternalError( + new(ItemsMustBeUniqueError), + context, + value, + ErrorDetails{"type": TypeArray, "i": i, "j": j}, + ) + } + stringifiedItems[*vString] = j + } + } + + // contains: + + if currentSubSchema.contains != nil { + validatedOne := false + var bestValidationResult *Result + + for i, v := range value { + subContext := NewJSONContext(strconv.Itoa(i), context) + + validationResult := currentSubSchema.contains.subValidateWithContext(v, subContext) + if validationResult.Valid() { + validatedOne = true + break + } else { + if bestValidationResult == nil || validationResult.score > bestValidationResult.score { + bestValidationResult = validationResult + } + } + } + if !validatedOne { + result.addInternalError( + new(ArrayContainsError), + context, + value, + ErrorDetails{}, + ) + if bestValidationResult != nil { + result.mergeErrors(bestValidationResult) + } + } + } + + result.incrementScore() +} + +func (v *SubSchema) validateObject(currentSubSchema *SubSchema, value map[string]interface{}, result *Result, context *JSONContext) { + + if internalLogEnabled { + internalLog("validateObject %s", context.String()) + internalLog(" %v", value) + } + + // minProperties & maxProperties: + if currentSubSchema.minProperties != nil { + if len(value) < int(*currentSubSchema.minProperties) { + result.addInternalError( + new(ArrayMinPropertiesError), + context, + value, + ErrorDetails{"min": *currentSubSchema.minProperties}, + ) + } + } + if currentSubSchema.maxProperties != nil { + if len(value) > int(*currentSubSchema.maxProperties) { + result.addInternalError( + new(ArrayMaxPropertiesError), + context, + value, + ErrorDetails{"max": *currentSubSchema.maxProperties}, + ) + } + } + + // required: + for _, requiredProperty := range currentSubSchema.required { + _, ok := value[requiredProperty] + if ok { + result.incrementScore() + } else { + result.addInternalError( + new(RequiredError), + context, + value, + ErrorDetails{"property": requiredProperty}, + ) + } + } + + // additionalProperty & patternProperty: + for pk := range value { + + // Check whether this property is described by "properties" + found := false + for _, spValue := range currentSubSchema.PropertiesChildren { + if pk == spValue.Property { + found = true + } + } + + // Check whether this property is described by "patternProperties" + ppMatch := v.validatePatternProperty(currentSubSchema, pk, value[pk], result, context) + + // If it is not described by neither "properties" nor "patternProperties" it must pass "additionalProperties" + if !found && !ppMatch { + switch ap := currentSubSchema.additionalProperties.(type) { + case bool: + // Handle the boolean case separately as it's cleaner to return a specific error than failing to pass the false schema + if !ap { + result.addInternalError( + new(AdditionalPropertyNotAllowedError), + context, + value[pk], + ErrorDetails{"property": pk}, + ) + + } + case *SubSchema: + validationResult := ap.subValidateWithContext(value[pk], NewJSONContext(pk, context)) + result.mergeErrors(validationResult) + } + } + } + + // propertyNames: + if currentSubSchema.propertyNames != nil { + for pk := range value { + validationResult := currentSubSchema.propertyNames.subValidateWithContext(pk, context) + if !validationResult.Valid() { + result.addInternalError(new(InvalidPropertyNameError), + context, + value, ErrorDetails{ + "property": pk, + }) + result.mergeErrors(validationResult) + } + } + } + + result.incrementScore() +} + +func (v *SubSchema) validatePatternProperty(currentSubSchema *SubSchema, key string, value interface{}, result *Result, context *JSONContext) bool { + + if internalLogEnabled { + internalLog("validatePatternProperty %s", context.String()) + internalLog(" %s %v", key, value) + } + + validated := false + + for pk, pv := range currentSubSchema.patternProperties { + if matches, _ := regexp.MatchString(pk, key); matches { + validated = true + subContext := NewJSONContext(key, context) + validationResult := pv.subValidateWithContext(value, subContext) + result.mergeErrors(validationResult) + } + } + + if !validated { + return false + } + + result.incrementScore() + return true +} + +func (v *SubSchema) validateString(currentSubSchema *SubSchema, value interface{}, result *Result, context *JSONContext) { + + // Ignore JSON numbers + stringValue, isString := value.(string) + if !isString { + return + } + + if internalLogEnabled { + internalLog("validateString %s", context.String()) + internalLog(" %v", value) + } + + // minLength & maxLength: + if currentSubSchema.minLength != nil { + if utf8.RuneCount([]byte(stringValue)) < int(*currentSubSchema.minLength) { + result.addInternalError( + new(StringLengthGTEError), + context, + value, + ErrorDetails{"min": *currentSubSchema.minLength}, + ) + } + } + if currentSubSchema.maxLength != nil { + if utf8.RuneCount([]byte(stringValue)) > int(*currentSubSchema.maxLength) { + result.addInternalError( + new(StringLengthLTEError), + context, + value, + ErrorDetails{"max": *currentSubSchema.maxLength}, + ) + } + } + + // pattern: + if currentSubSchema.pattern != nil { + if !currentSubSchema.pattern.MatchString(stringValue) { + result.addInternalError( + new(DoesNotMatchPatternError), + context, + value, + ErrorDetails{"pattern": currentSubSchema.pattern}, + ) + + } + } + + result.incrementScore() +} + +func (v *SubSchema) validateNumber(currentSubSchema *SubSchema, value interface{}, result *Result, context *JSONContext) { + + // Ignore non numbers + number, isNumber := value.(json.Number) + if !isNumber { + return + } + + if internalLogEnabled { + internalLog("validateNumber %s", context.String()) + internalLog(" %v", value) + } + + float64Value, _ := new(big.Rat).SetString(string(number)) + + // multipleOf: + if currentSubSchema.multipleOf != nil { + if q := new(big.Rat).Quo(float64Value, currentSubSchema.multipleOf); !q.IsInt() { + result.addInternalError( + new(MultipleOfError), + context, + number, + ErrorDetails{ + "multiple": new(big.Float).SetRat(currentSubSchema.multipleOf), + }, + ) + } + } + + //maximum & exclusiveMaximum: + if currentSubSchema.maximum != nil { + if float64Value.Cmp(currentSubSchema.maximum) == 1 { + result.addInternalError( + new(NumberLTEError), + context, + number, + ErrorDetails{ + "max": new(big.Float).SetRat(currentSubSchema.maximum), + }, + ) + } + } + if currentSubSchema.exclusiveMaximum != nil { + if float64Value.Cmp(currentSubSchema.exclusiveMaximum) >= 0 { + result.addInternalError( + new(NumberLTError), + context, + number, + ErrorDetails{ + "max": new(big.Float).SetRat(currentSubSchema.exclusiveMaximum), + }, + ) + } + } + + //minimum & exclusiveMinimum: + if currentSubSchema.minimum != nil { + if float64Value.Cmp(currentSubSchema.minimum) == -1 { + result.addInternalError( + new(NumberGTEError), + context, + number, + ErrorDetails{ + "min": new(big.Float).SetRat(currentSubSchema.minimum), + }, + ) + } + } + if currentSubSchema.exclusiveMinimum != nil { + if float64Value.Cmp(currentSubSchema.exclusiveMinimum) <= 0 { + result.addInternalError( + new(NumberGTError), + context, + number, + ErrorDetails{ + "min": new(big.Float).SetRat(currentSubSchema.exclusiveMinimum), + }, + ) + } + } + + result.incrementScore() +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/ir/ir.go b/vendor/github.com/open-policy-agent/opa/internal/ir/ir.go new file mode 100644 index 00000000..d907bce0 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/ir/ir.go @@ -0,0 +1,497 @@ +// Copyright 2018 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +// Package ir defines an intermediate representation (IR) for Rego. +// +// The IR specifies an imperative execution model for Rego policies similar to a +// query plan in traditional databases. +package ir + +import ( + "fmt" + + "github.com/open-policy-agent/opa/types" +) + +type ( + // Policy represents a planned policy query. + Policy struct { + Static *Static + Plans *Plans + Funcs *Funcs + } + + // Static represents a static data segment that is indexed into by the policy. + Static struct { + Strings []*StringConst + BuiltinFuncs []*BuiltinFunc + Files []*StringConst + } + + // BuiltinFunc represents a built-in function that may be required by the + // policy. + BuiltinFunc struct { + Name string + Decl *types.Function + } + + // Plans represents a collection of named query plans to expose in the policy. + Plans struct { + Plans []*Plan + } + + // Funcs represents a collection of planned functions to include in the + // policy. + Funcs struct { + Funcs []*Func + } + + // Func represents a named plan (function) that can be invoked. Functions + // accept one or more parameters and return a value. By convention, the + // input document and data documents are always passed as the first and + // second arguments (respectively). + Func struct { + Name string + Params []Local + Return Local + Blocks []*Block // TODO(tsandall): should this be a plan? + Path []string // optional: if non-nil, include in data function tree + } + + // Plan represents an ordered series of blocks to execute. Plan execution + // stops when a return statement is reached. Blocks are executed in-order. + Plan struct { + Name string + Blocks []*Block + } + + // Block represents an ordered sequence of statements to execute. Blocks are + // executed until a return statement is encountered, a statement is undefined, + // or there are no more statements. If all statements are defined but no return + // statement is encountered, the block is undefined. + Block struct { + Stmts []Stmt + } + + // Stmt represents an operation (e.g., comparison, loop, dot, etc.) to execute. + Stmt interface { + locationStmt + } + + locationStmt interface { + SetLocation(index, row, col int, file, text string) + GetLocation() *Location + } + + // Local represents a plan-scoped variable. + // + // TODO(tsandall): should this be int32 for safety? + Local int + + // Const represents a constant value from the policy. + Const interface { + typeMarker() + } + + // NullConst represents a null value. + NullConst struct{} + + // BooleanConst represents a boolean value. + BooleanConst struct { + Value bool + } + + // StringConst represents a string value. + StringConst struct { + Value string + } + + // IntConst represents an integer constant. + IntConst struct { + Value int64 + } + + // FloatConst represents a floating-point constant. + FloatConst struct { + Value float64 + } +) + +const ( + // Input is the local variable that refers to the global input document. + Input Local = iota + + // Data is the local variable that refers to the global data document. + Data + + // Unused is the free local variable that can be allocated in a plan. + Unused +) + +func (a *Policy) String() string { + return "Policy" +} + +func (a *Static) String() string { + return fmt.Sprintf("Static (%d strings, %d files)", len(a.Strings), len(a.Files)) +} + +func (a *Funcs) String() string { + return fmt.Sprintf("Funcs (%d funcs)", len(a.Funcs)) +} + +func (a *Func) String() string { + return fmt.Sprintf("%v (%d params: %v, %d blocks, path: %v)", a.Name, len(a.Params), a.Params, len(a.Blocks), a.Path) +} + +func (a *Plan) String() string { + return fmt.Sprintf("Plan %v (%d blocks)", a.Name, len(a.Blocks)) +} + +func (a *Block) String() string { + return fmt.Sprintf("Block (%d statements)", len(a.Stmts)) +} + +func (*BooleanConst) typeMarker() {} +func (*NullConst) typeMarker() {} +func (*IntConst) typeMarker() {} +func (*FloatConst) typeMarker() {} +func (*StringConst) typeMarker() {} + +// ReturnLocalStmt represents a return statement that yields a local value. +type ReturnLocalStmt struct { + Source Local + + Location +} + +// CallStmt represents a named function call. The result should be stored in the +// result local. +type CallStmt struct { + Func string + Args []LocalOrConst + Result Local + + Location +} + +// CallDynamicStmt represents an indirect (data) function call. The result should +// be stored in the result local. +type CallDynamicStmt struct { + Args []Local + Result Local + Path []LocalOrConst + + Location +} + +// LocalOrConst is a tagged union of the two types, Local and StringIndex. +type LocalOrConst interface { + fmt.Stringer + localOrConst() +} + +func (Local) localOrConst() {} +func (l Local) String() string { + return fmt.Sprintf("Local<%d>", int(l)) +} + +// StringIndex represents the index into the plan's list of constant strings +// of a constant string. +type StringIndex int + +func (StringIndex) localOrConst() {} +func (s StringIndex) String() string { + return fmt.Sprintf("String<%d>", int(s)) +} + +// Bool represents a constant boolean. +type Bool bool + +func (Bool) localOrConst() {} +func (b Bool) String() string { + return fmt.Sprintf("Bool<%v>", bool(b)) +} + +// BlockStmt represents a nested block. Nested blocks and break statements can +// be used to short-circuit execution. +type BlockStmt struct { + Blocks []*Block + + Location +} + +func (a *BlockStmt) String() string { + return fmt.Sprintf("BlockStmt (%d blocks) %v", len(a.Blocks), a.GetLocation()) +} + +// BreakStmt represents a jump out of the current block. The index specifies how +// many blocks to jump starting from zero (the current block). Execution will +// continue from the end of the block that is jumped to. +type BreakStmt struct { + Index uint32 + + Location +} + +// DotStmt represents a lookup operation on a value (e.g., array, object, etc.) +// The source of a DotStmt may be a scalar value in which case the statement +// will be undefined. +type DotStmt struct { + Source LocalOrConst + Key LocalOrConst + Target Local + + Location +} + +// LenStmt represents a length() operation on a local variable. The +// result is stored in the target local variable. +type LenStmt struct { + Source LocalOrConst + Target Local + + Location +} + +// ScanStmt represents a linear scan over a composite value. The +// source may be a scalar in which case the block will never execute. +type ScanStmt struct { + Source Local + Key Local + Value Local + Block *Block + + Location +} + +// NotStmt represents a negated statement. +type NotStmt struct { + Block *Block + + Location +} + +// AssignIntStmt represents an assignment of an integer value to a +// local variable. +type AssignIntStmt struct { + Value int64 + Target Local + + Location +} + +// AssignVarStmt represents an assignment of one local variable to another. +type AssignVarStmt struct { + Source LocalOrConst + Target Local + + Location +} + +// AssignVarOnceStmt represents an assignment of one local variable to another. +// If the target is defined, execution aborts with a conflict error. +// +// TODO(tsandall): is there a better name for this? +type AssignVarOnceStmt struct { + Target Local + Source LocalOrConst + + Location +} + +// ResetLocalStmt resets a local variable to 0. +type ResetLocalStmt struct { + Target Local + + Location +} + +// MakeNullStmt constructs a local variable that refers to a null value. +type MakeNullStmt struct { + Target Local + + Location +} + +// MakeNumberIntStmt constructs a local variable that refers to an integer value. +type MakeNumberIntStmt struct { + Value int64 + Target Local + + Location +} + +// MakeNumberRefStmt constructs a local variable that refers to a number stored as a string. +type MakeNumberRefStmt struct { + Index int + Target Local + + Location +} + +// MakeArrayStmt constructs a local variable that refers to an array value. +type MakeArrayStmt struct { + Capacity int32 + Target Local + + Location +} + +// MakeObjectStmt constructs a local variable that refers to an object value. +type MakeObjectStmt struct { + Target Local + + Location +} + +// MakeSetStmt constructs a local variable that refers to a set value. +type MakeSetStmt struct { + Target Local + + Location +} + +// EqualStmt represents an value-equality check of two local variables. +type EqualStmt struct { + A LocalOrConst + B LocalOrConst + + Location +} + +// NotEqualStmt represents a != check of two local variables. +type NotEqualStmt struct { + A LocalOrConst + B LocalOrConst + + Location +} + +// IsArrayStmt represents a dynamic type check on a local variable. +type IsArrayStmt struct { + Source LocalOrConst + + Location +} + +// IsObjectStmt represents a dynamic type check on a local variable. +type IsObjectStmt struct { + Source LocalOrConst + + Location +} + +// IsDefinedStmt represents a check of whether a local variable is defined. +type IsDefinedStmt struct { + Source Local + + Location +} + +// IsUndefinedStmt represents a check of whether local variable is undefined. +type IsUndefinedStmt struct { + Source Local + + Location +} + +// ArrayAppendStmt represents a dynamic append operation of a value +// onto an array. +type ArrayAppendStmt struct { + Value LocalOrConst + Array Local + + Location +} + +// ObjectInsertStmt represents a dynamic insert operation of a +// key/value pair into an object. +type ObjectInsertStmt struct { + Key LocalOrConst + Value LocalOrConst + Object Local + + Location +} + +// ObjectInsertOnceStmt represents a dynamic insert operation of a key/value +// pair into an object. If the key already exists and the value differs, +// execution aborts with a conflict error. +type ObjectInsertOnceStmt struct { + Key LocalOrConst + Value LocalOrConst + Object Local + + Location +} + +// ObjectMergeStmt performs a recursive merge of two object values. If either of +// the locals refer to non-object values this operation will abort with a +// conflict error. Overlapping object keys are merged recursively. +type ObjectMergeStmt struct { + A Local + B Local + Target Local + + Location +} + +// SetAddStmt represents a dynamic add operation of an element into a set. +type SetAddStmt struct { + Value LocalOrConst + Set Local + + Location +} + +// WithStmt replaces the Local or a portion of the document referred to by the +// Local with the Value and executes the contained block. If the Path is +// non-empty, the Value is upserted into the Local. If the intermediate nodes in +// the Local referred to by the Path do not exist, they will be created. When +// the WithStmt finishes the Local is reset to it's original value. +type WithStmt struct { + Local Local + Path []int + Value LocalOrConst + Block *Block + + Location +} + +// NopStmt adds a nop instruction. Useful during development and debugging only. +type NopStmt struct { + Location +} + +// ResultSetAdd adds a value into the result set returned by the query plan. +type ResultSetAdd struct { + Value Local + + Location +} + +// Location records the filen index, and the row and column inside that file +// that a statement can be connected to. +type Location struct { + Index int // filename string constant index + Col, Row int + file, text string // only used for debugging +} + +// SetLocation sets the Location for a given Stmt. +func (l *Location) SetLocation(index, row, col int, file, text string) { + *l = Location{ + Index: index, + Row: row, + Col: col, + file: file, + text: text, + } +} + +// GetLocation returns a Stmt's Location. +func (l *Location) GetLocation() *Location { + return l +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/ir/pretty.go b/vendor/github.com/open-policy-agent/opa/internal/ir/pretty.go new file mode 100644 index 00000000..a0f2df3b --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/ir/pretty.go @@ -0,0 +1,44 @@ +// Copyright 2018 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package ir + +import ( + "fmt" + "io" + "strings" +) + +// Pretty writes a human-readable representation of an IR object to w. +func Pretty(w io.Writer, x interface{}) error { + + pp := &prettyPrinter{ + depth: -1, + w: w, + } + return Walk(pp, x) +} + +type prettyPrinter struct { + depth int + w io.Writer +} + +func (pp *prettyPrinter) Before(x interface{}) { + pp.depth++ +} + +func (pp *prettyPrinter) After(x interface{}) { + pp.depth-- +} + +func (pp *prettyPrinter) Visit(x interface{}) (Visitor, error) { + pp.writeIndent("%T %+v", x, x) + return pp, nil +} + +func (pp *prettyPrinter) writeIndent(f string, a ...interface{}) { + pad := strings.Repeat("| ", pp.depth) + fmt.Fprintf(pp.w, pad+f+"\n", a...) +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/ir/walk.go b/vendor/github.com/open-policy-agent/opa/internal/ir/walk.go new file mode 100644 index 00000000..08a8f424 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/ir/walk.go @@ -0,0 +1,93 @@ +// Copyright 2018 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package ir + +// Visitor defines the interface for visiting IR nodes. +type Visitor interface { + Before(x interface{}) + Visit(x interface{}) (Visitor, error) + After(x interface{}) +} + +// Walk invokes the visitor for nodes under x. +func Walk(vis Visitor, x interface{}) error { + impl := walkerImpl{ + vis: vis, + } + impl.walk(x) + return impl.err +} + +type walkerImpl struct { + vis Visitor + err error +} + +func (w *walkerImpl) walk(x interface{}) { + if w.err != nil { // abort on error + return + } + if x == nil { + return + } + + prev := w.vis + w.vis.Before(x) + defer w.vis.After(x) + w.vis, w.err = w.vis.Visit(x) + if w.err != nil { + return + } else if w.vis == nil { + w.vis = prev + return + } + + switch x := x.(type) { + case *Policy: + w.walk(x.Static) + w.walk(x.Plans) + w.walk(x.Funcs) + case *Static: + for _, s := range x.Strings { + w.walk(s) + } + for _, f := range x.BuiltinFuncs { + w.walk(f) + } + for _, f := range x.Files { + w.walk(f) + } + case *Plans: + for _, pl := range x.Plans { + w.walk(pl) + } + case *Funcs: + for _, fn := range x.Funcs { + w.walk(fn) + } + case *Func: + for _, b := range x.Blocks { + w.walk(b) + } + case *Plan: + for _, b := range x.Blocks { + w.walk(b) + } + case *Block: + for _, s := range x.Stmts { + w.walk(s) + } + case *BlockStmt: + for _, b := range x.Blocks { + w.walk(b) + } + case *ScanStmt: + w.walk(x.Block) + case *NotStmt: + w.walk(x.Block) + case *WithStmt: + w.walk(x.Block) + } +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/jwx/LICENSE b/vendor/github.com/open-policy-agent/opa/internal/jwx/LICENSE new file mode 100644 index 00000000..6369f4fc --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/jwx/LICENSE @@ -0,0 +1,21 @@ +The MIT License (MIT) + +Copyright (c) 2015 lestrrat + +Permission is hereby granted, free of charge, to any person obtaining a copy +of this software and associated documentation files (the "Software"), to deal +in the Software without restriction, including without limitation the rights +to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the Software is +furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in all +copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE +SOFTWARE. diff --git a/vendor/github.com/open-policy-agent/opa/internal/jwx/buffer/buffer.go b/vendor/github.com/open-policy-agent/opa/internal/jwx/buffer/buffer.go new file mode 100644 index 00000000..ca4ac419 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/jwx/buffer/buffer.go @@ -0,0 +1,113 @@ +// Package buffer provides a very thin wrapper around []byte buffer called +// `Buffer`, to provide functionalities that are often used within the jwx +// related packages +package buffer + +import ( + "encoding/base64" + "encoding/binary" + "encoding/json" + + "github.com/pkg/errors" +) + +// Buffer wraps `[]byte` and provides functions that are often used in +// the jwx related packages. One notable difference is that while +// encoding/json marshalls `[]byte` using base64.StdEncoding, this +// module uses base64.RawURLEncoding as mandated by the spec +type Buffer []byte + +// FromUint creates a `Buffer` from an unsigned int +func FromUint(v uint64) Buffer { + data := make([]byte, 8) + binary.BigEndian.PutUint64(data, v) + + i := 0 + for ; i < len(data); i++ { + if data[i] != 0x0 { + break + } + } + return Buffer(data[i:]) +} + +// FromBase64 constructs a new Buffer from a base64 encoded data +func FromBase64(v []byte) (Buffer, error) { + b := Buffer{} + if err := b.Base64Decode(v); err != nil { + return Buffer(nil), errors.Wrap(err, "failed to decode from base64") + } + + return b, nil +} + +// FromNData constructs a new Buffer from a "n:data" format +// (I made that name up) +func FromNData(v []byte) (Buffer, error) { + size := binary.BigEndian.Uint32(v) + buf := make([]byte, int(size)) + copy(buf, v[4:4+size]) + return Buffer(buf), nil +} + +// Bytes returns the raw bytes that comprises the Buffer +func (b Buffer) Bytes() []byte { + return []byte(b) +} + +// NData returns Datalen || Data, where Datalen is a 32 bit counter for +// the length of the following data, and Data is the octets that comprise +// the buffer data +func (b Buffer) NData() []byte { + buf := make([]byte, 4+b.Len()) + binary.BigEndian.PutUint32(buf, uint32(b.Len())) + + copy(buf[4:], b.Bytes()) + return buf +} + +// Len returns the number of bytes that the Buffer holds +func (b Buffer) Len() int { + return len(b) +} + +// Base64Encode encodes the contents of the Buffer using base64.RawURLEncoding +func (b Buffer) Base64Encode() ([]byte, error) { + enc := base64.RawURLEncoding + out := make([]byte, enc.EncodedLen(len(b))) + enc.Encode(out, b) + return out, nil +} + +// Base64Decode decodes the contents of the Buffer using base64.RawURLEncoding +func (b *Buffer) Base64Decode(v []byte) error { + enc := base64.RawURLEncoding + out := make([]byte, enc.DecodedLen(len(v))) + n, err := enc.Decode(out, v) + if err != nil { + return errors.Wrap(err, "failed to decode from base64") + } + out = out[:n] + *b = Buffer(out) + return nil +} + +// MarshalJSON marshals the buffer into JSON format after encoding the buffer +// with base64.RawURLEncoding +func (b Buffer) MarshalJSON() ([]byte, error) { + v, err := b.Base64Encode() + if err != nil { + return nil, errors.Wrap(err, "failed to encode to base64") + } + return json.Marshal(string(v)) +} + +// UnmarshalJSON unmarshals from a JSON string into a Buffer, after decoding it +// with base64.RawURLEncoding +func (b *Buffer) UnmarshalJSON(data []byte) error { + var x string + if err := json.Unmarshal(data, &x); err != nil { + return errors.Wrap(err, "failed to unmarshal JSON") + } + return b.Base64Decode([]byte(x)) +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/jwx/jwa/elliptic.go b/vendor/github.com/open-policy-agent/opa/internal/jwx/jwa/elliptic.go new file mode 100644 index 00000000..b7e35dc7 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/jwx/jwa/elliptic.go @@ -0,0 +1,11 @@ +package jwa + +// EllipticCurveAlgorithm represents the algorithms used for EC keys +type EllipticCurveAlgorithm string + +// Supported values for EllipticCurveAlgorithm +const ( + P256 EllipticCurveAlgorithm = "P-256" + P384 EllipticCurveAlgorithm = "P-384" + P521 EllipticCurveAlgorithm = "P-521" +) diff --git a/vendor/github.com/open-policy-agent/opa/internal/jwx/jwa/key_type.go b/vendor/github.com/open-policy-agent/opa/internal/jwx/jwa/key_type.go new file mode 100644 index 00000000..076bd39e --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/jwx/jwa/key_type.go @@ -0,0 +1,67 @@ +package jwa + +import ( + "strconv" + + "github.com/pkg/errors" +) + +// KeyType represents the key type ("kty") that are supported +type KeyType string + +var keyTypeAlg = map[string]struct{}{"EC": {}, "oct": {}, "RSA": {}} + +// Supported values for KeyType +const ( + EC KeyType = "EC" // Elliptic Curve + InvalidKeyType KeyType = "" // Invalid KeyType + OctetSeq KeyType = "oct" // Octet sequence (used to represent symmetric keys) + RSA KeyType = "RSA" // RSA +) + +// Accept is used when conversion from values given by +// outside sources (such as JSON payloads) is required +func (keyType *KeyType) Accept(value interface{}) error { + var tmp KeyType + switch x := value.(type) { + case string: + tmp = KeyType(x) + case KeyType: + tmp = x + default: + return errors.Errorf(`invalid type for jwa.KeyType: %T`, value) + } + _, ok := keyTypeAlg[tmp.String()] + if !ok { + return errors.Errorf("Unknown Key Type algorithm") + } + + *keyType = tmp + return nil +} + +// String returns the string representation of a KeyType +func (keyType KeyType) String() string { + return string(keyType) +} + +// UnmarshalJSON unmarshals and checks data as KeyType Algorithm +func (keyType *KeyType) UnmarshalJSON(data []byte) error { + var quote byte = '"' + var quoted string + if data[0] == quote { + var err error + quoted, err = strconv.Unquote(string(data)) + if err != nil { + return errors.Wrap(err, "Failed to process signature algorithm") + } + } else { + quoted = string(data) + } + _, ok := keyTypeAlg[quoted] + if !ok { + return errors.Errorf("Unknown signature algorithm") + } + *keyType = KeyType(quoted) + return nil +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/jwx/jwa/parameters.go b/vendor/github.com/open-policy-agent/opa/internal/jwx/jwa/parameters.go new file mode 100644 index 00000000..2fe72e1d --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/jwx/jwa/parameters.go @@ -0,0 +1,29 @@ +package jwa + +import ( + "crypto/elliptic" + + "github.com/open-policy-agent/opa/internal/jwx/buffer" +) + +// EllipticCurve provides a indirect type to standard elliptic curve such that we can +// use it for unmarshal +type EllipticCurve struct { + elliptic.Curve +} + +// AlgorithmParameters provides a single structure suitable to unmarshaling any JWK +type AlgorithmParameters struct { + N buffer.Buffer `json:"n,omitempty"` + E buffer.Buffer `json:"e,omitempty"` + D buffer.Buffer `json:"d,omitempty"` + P buffer.Buffer `json:"p,omitempty"` + Q buffer.Buffer `json:"q,omitempty"` + Dp buffer.Buffer `json:"dp,omitempty"` + Dq buffer.Buffer `json:"dq,omitempty"` + Qi buffer.Buffer `json:"qi,omitempty"` + Crv EllipticCurveAlgorithm `json:"crv,omitempty"` + X buffer.Buffer `json:"x,omitempty"` + Y buffer.Buffer `json:"y,omitempty"` + K buffer.Buffer `json:"k,omitempty"` +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/jwx/jwa/signature.go b/vendor/github.com/open-policy-agent/opa/internal/jwx/jwa/signature.go new file mode 100644 index 00000000..a0988eca --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/jwx/jwa/signature.go @@ -0,0 +1,76 @@ +package jwa + +import ( + "strconv" + + "github.com/pkg/errors" +) + +// SignatureAlgorithm represents the various signature algorithms as described in https://tools.ietf.org/html/rfc7518#section-3.1 +type SignatureAlgorithm string + +var signatureAlg = map[string]struct{}{"ES256": {}, "ES384": {}, "ES512": {}, "HS256": {}, "HS384": {}, "HS512": {}, "PS256": {}, "PS384": {}, "PS512": {}, "RS256": {}, "RS384": {}, "RS512": {}, "none": {}} + +// Supported values for SignatureAlgorithm +const ( + ES256 SignatureAlgorithm = "ES256" // ECDSA using P-256 and SHA-256 + ES384 SignatureAlgorithm = "ES384" // ECDSA using P-384 and SHA-384 + ES512 SignatureAlgorithm = "ES512" // ECDSA using P-521 and SHA-512 + HS256 SignatureAlgorithm = "HS256" // HMAC using SHA-256 + HS384 SignatureAlgorithm = "HS384" // HMAC using SHA-384 + HS512 SignatureAlgorithm = "HS512" // HMAC using SHA-512 + NoSignature SignatureAlgorithm = "none" + PS256 SignatureAlgorithm = "PS256" // RSASSA-PSS using SHA256 and MGF1-SHA256 + PS384 SignatureAlgorithm = "PS384" // RSASSA-PSS using SHA384 and MGF1-SHA384 + PS512 SignatureAlgorithm = "PS512" // RSASSA-PSS using SHA512 and MGF1-SHA512 + RS256 SignatureAlgorithm = "RS256" // RSASSA-PKCS-v1.5 using SHA-256 + RS384 SignatureAlgorithm = "RS384" // RSASSA-PKCS-v1.5 using SHA-384 + RS512 SignatureAlgorithm = "RS512" // RSASSA-PKCS-v1.5 using SHA-512 + NoValue SignatureAlgorithm = "" // No value is different from none +) + +// Accept is used when conversion from values given by +// outside sources (such as JSON payloads) is required +func (signature *SignatureAlgorithm) Accept(value interface{}) error { + var tmp SignatureAlgorithm + switch x := value.(type) { + case string: + tmp = SignatureAlgorithm(x) + case SignatureAlgorithm: + tmp = x + default: + return errors.Errorf(`invalid type for jwa.SignatureAlgorithm: %T`, value) + } + _, ok := signatureAlg[tmp.String()] + if !ok { + return errors.Errorf("Unknown signature algorithm") + } + *signature = tmp + return nil +} + +// String returns the string representation of a SignatureAlgorithm +func (signature SignatureAlgorithm) String() string { + return string(signature) +} + +// UnmarshalJSON unmarshals and checks data as Signature Algorithm +func (signature *SignatureAlgorithm) UnmarshalJSON(data []byte) error { + var quote byte = '"' + var quoted string + if data[0] == quote { + var err error + quoted, err = strconv.Unquote(string(data)) + if err != nil { + return errors.Wrap(err, "Failed to process signature algorithm") + } + } else { + quoted = string(data) + } + _, ok := signatureAlg[quoted] + if !ok { + return errors.Errorf("Unknown signature algorithm") + } + *signature = SignatureAlgorithm(quoted) + return nil +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/jwx/jwk/ecdsa.go b/vendor/github.com/open-policy-agent/opa/internal/jwx/jwk/ecdsa.go new file mode 100644 index 00000000..30bee46b --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/jwx/jwk/ecdsa.go @@ -0,0 +1,120 @@ +package jwk + +import ( + "crypto/ecdsa" + "crypto/elliptic" + "math/big" + + "github.com/pkg/errors" + + "github.com/open-policy-agent/opa/internal/jwx/jwa" +) + +func newECDSAPublicKey(key *ecdsa.PublicKey) (*ECDSAPublicKey, error) { + + var hdr StandardHeaders + err := hdr.Set(KeyTypeKey, jwa.EC) + if err != nil { + return nil, errors.Wrapf(err, "Failed to set Key Type") + } + + return &ECDSAPublicKey{ + StandardHeaders: &hdr, + key: key, + }, nil +} + +func newECDSAPrivateKey(key *ecdsa.PrivateKey) (*ECDSAPrivateKey, error) { + + var hdr StandardHeaders + err := hdr.Set(KeyTypeKey, jwa.EC) + if err != nil { + return nil, errors.Wrapf(err, "Failed to set Key Type") + } + + return &ECDSAPrivateKey{ + StandardHeaders: &hdr, + key: key, + }, nil +} + +// Materialize returns the EC-DSA public key represented by this JWK +func (k ECDSAPublicKey) Materialize() (interface{}, error) { + return k.key, nil +} + +// Materialize returns the EC-DSA private key represented by this JWK +func (k ECDSAPrivateKey) Materialize() (interface{}, error) { + return k.key, nil +} + +// GenerateKey creates a ECDSAPublicKey from JWK format +func (k *ECDSAPublicKey) GenerateKey(keyJSON *RawKeyJSON) error { + + var x, y big.Int + + if keyJSON.X == nil || keyJSON.Y == nil || keyJSON.Crv == "" { + return errors.Errorf("Missing mandatory key parameters X, Y or Crv") + } + + x.SetBytes(keyJSON.X.Bytes()) + y.SetBytes(keyJSON.Y.Bytes()) + + var curve elliptic.Curve + switch keyJSON.Crv { + case jwa.P256: + curve = elliptic.P256() + case jwa.P384: + curve = elliptic.P384() + case jwa.P521: + curve = elliptic.P521() + default: + return errors.Errorf(`invalid curve name %s`, keyJSON.Crv) + } + + *k = ECDSAPublicKey{ + StandardHeaders: &keyJSON.StandardHeaders, + key: &ecdsa.PublicKey{ + Curve: curve, + X: &x, + Y: &y, + }, + } + return nil +} + +// GenerateKey creates a ECDSAPrivateKey from JWK format +func (k *ECDSAPrivateKey) GenerateKey(keyJSON *RawKeyJSON) error { + + if keyJSON.D == nil { + return errors.Errorf("Missing mandatory key parameter D") + } + eCDSAPublicKey := &ECDSAPublicKey{} + err := eCDSAPublicKey.GenerateKey(keyJSON) + if err != nil { + return errors.Wrap(err, `failed to generate public key`) + } + dBytes := keyJSON.D.Bytes() + // The length of this octet string MUST be ceiling(log-base-2(n)/8) + // octets (where n is the order of the curve). This is because the private + // key d must be in the interval [1, n-1] so the bitlength of d should be + // no larger than the bitlength of n-1. The easiest way to find the octet + // length is to take bitlength(n-1), add 7 to force a carry, and shift this + // bit sequence right by 3, which is essentially dividing by 8 and adding + // 1 if there is any remainder. Thus, the private key value d should be + // output to (bitlength(n-1)+7)>>3 octets. + n := eCDSAPublicKey.key.Params().N + octetLength := (new(big.Int).Sub(n, big.NewInt(1)).BitLen() + 7) >> 3 + if octetLength-len(dBytes) != 0 { + return errors.Errorf("Failed to generate private key. Incorrect D value") + } + privateKey := &ecdsa.PrivateKey{ + PublicKey: *eCDSAPublicKey.key, + D: (&big.Int{}).SetBytes(keyJSON.D.Bytes()), + } + + k.key = privateKey + k.StandardHeaders = &keyJSON.StandardHeaders + + return nil +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/jwx/jwk/headers.go b/vendor/github.com/open-policy-agent/opa/internal/jwx/jwk/headers.go new file mode 100644 index 00000000..cf700ee8 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/jwx/jwk/headers.go @@ -0,0 +1,178 @@ +package jwk + +import ( + "github.com/pkg/errors" + + "github.com/open-policy-agent/opa/internal/jwx/jwa" +) + +// Convenience constants for common JWK parameters +const ( + AlgorithmKey = "alg" + KeyIDKey = "kid" + KeyOpsKey = "key_ops" + KeyTypeKey = "kty" + KeyUsageKey = "use" + PrivateParamsKey = "privateParams" +) + +// Headers provides a common interface to all future possible headers +type Headers interface { + Get(string) (interface{}, bool) + Set(string, interface{}) error + Walk(func(string, interface{}) error) error + GetAlgorithm() jwa.SignatureAlgorithm + GetKeyID() string + GetKeyOps() KeyOperationList + GetKeyType() jwa.KeyType + GetKeyUsage() string + GetPrivateParams() map[string]interface{} +} + +// StandardHeaders stores the common JWK parameters +type StandardHeaders struct { + Algorithm *jwa.SignatureAlgorithm `json:"alg,omitempty"` // https://tools.ietf.org/html/rfc7517#section-4.4 + KeyID string `json:"kid,omitempty"` // https://tools.ietf.org/html/rfc7515#section-4.1.4 + KeyOps KeyOperationList `json:"key_ops,omitempty"` // https://tools.ietf.org/html/rfc7517#section-4.3 + KeyType jwa.KeyType `json:"kty,omitempty"` // https://tools.ietf.org/html/rfc7517#section-4.1 + KeyUsage string `json:"use,omitempty"` // https://tools.ietf.org/html/rfc7517#section-4.2 + PrivateParams map[string]interface{} `json:"privateParams,omitempty"` // https://tools.ietf.org/html/rfc7515#section-4.1.4 +} + +// GetAlgorithm is a convenience function to retrieve the corresponding value stored in the StandardHeaders +func (h *StandardHeaders) GetAlgorithm() jwa.SignatureAlgorithm { + if v := h.Algorithm; v != nil { + return *v + } + return jwa.NoValue +} + +// GetKeyID is a convenience function to retrieve the corresponding value stored in the StandardHeaders +func (h *StandardHeaders) GetKeyID() string { + return h.KeyID +} + +// GetKeyOps is a convenience function to retrieve the corresponding value stored in the StandardHeaders +func (h *StandardHeaders) GetKeyOps() KeyOperationList { + return h.KeyOps +} + +// GetKeyType is a convenience function to retrieve the corresponding value stored in the StandardHeaders +func (h *StandardHeaders) GetKeyType() jwa.KeyType { + return h.KeyType +} + +// GetKeyUsage is a convenience function to retrieve the corresponding value stored in the StandardHeaders +func (h *StandardHeaders) GetKeyUsage() string { + return h.KeyUsage +} + +// GetPrivateParams is a convenience function to retrieve the corresponding value stored in the StandardHeaders +func (h *StandardHeaders) GetPrivateParams() map[string]interface{} { + return h.PrivateParams +} + +// Get is a general getter function for JWK StandardHeaders structure +func (h *StandardHeaders) Get(name string) (interface{}, bool) { + switch name { + case AlgorithmKey: + alg := h.GetAlgorithm() + if alg != jwa.NoValue { + return alg, true + } + return nil, false + case KeyIDKey: + v := h.KeyID + if v == "" { + return nil, false + } + return v, true + case KeyOpsKey: + v := h.KeyOps + if v == nil { + return nil, false + } + return v, true + case KeyTypeKey: + v := h.KeyType + if v == jwa.InvalidKeyType { + return nil, false + } + return v, true + case KeyUsageKey: + v := h.KeyUsage + if v == "" { + return nil, false + } + return v, true + case PrivateParamsKey: + v := h.PrivateParams + if len(v) == 0 { + return nil, false + } + return v, true + default: + return nil, false + } +} + +// Set is a general getter function for JWK StandardHeaders structure +func (h *StandardHeaders) Set(name string, value interface{}) error { + switch name { + case AlgorithmKey: + var acceptor jwa.SignatureAlgorithm + if err := acceptor.Accept(value); err != nil { + return errors.Wrapf(err, `invalid value for %s key`, AlgorithmKey) + } + h.Algorithm = &acceptor + return nil + case KeyIDKey: + if v, ok := value.(string); ok { + h.KeyID = v + return nil + } + return errors.Errorf("invalid value for %s key: %T", KeyIDKey, value) + case KeyOpsKey: + if err := h.KeyOps.Accept(value); err != nil { + return errors.Wrapf(err, "invalid value for %s key", KeyOpsKey) + } + return nil + case KeyTypeKey: + if err := h.KeyType.Accept(value); err != nil { + return errors.Wrapf(err, "invalid value for %s key", KeyTypeKey) + } + return nil + case KeyUsageKey: + if v, ok := value.(string); ok { + h.KeyUsage = v + return nil + } + return errors.Errorf("invalid value for %s key: %T", KeyUsageKey, value) + case PrivateParamsKey: + if v, ok := value.(map[string]interface{}); ok { + h.PrivateParams = v + return nil + } + return errors.Errorf("invalid value for %s key: %T", PrivateParamsKey, value) + default: + return errors.Errorf(`invalid key: %s`, name) + } +} + +// Walk iterates over all JWK standard headers fields while applying a function to its value. +func (h StandardHeaders) Walk(f func(string, interface{}) error) error { + for _, key := range []string{AlgorithmKey, KeyIDKey, KeyOpsKey, KeyTypeKey, KeyUsageKey, PrivateParamsKey} { + if v, ok := h.Get(key); ok { + if err := f(key, v); err != nil { + return errors.Wrapf(err, `walk function returned error for %s`, key) + } + } + } + + for k, v := range h.PrivateParams { + if err := f(k, v); err != nil { + return errors.Wrapf(err, `walk function returned error for %s`, k) + } + } + return nil +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/jwx/jwk/interface.go b/vendor/github.com/open-policy-agent/opa/internal/jwx/jwk/interface.go new file mode 100644 index 00000000..7a7d03ef --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/jwx/jwk/interface.go @@ -0,0 +1,71 @@ +package jwk + +import ( + "crypto/ecdsa" + "crypto/rsa" + + "github.com/open-policy-agent/opa/internal/jwx/jwa" +) + +// Set is a convenience struct to allow generating and parsing +// JWK sets as opposed to single JWKs +type Set struct { + Keys []Key `json:"keys"` +} + +// Key defines the minimal interface for each of the +// key types. Their use and implementation differ significantly +// between each key types, so you should use type assertions +// to perform more specific tasks with each key +type Key interface { + Headers + + // Materialize creates the corresponding key. For example, + // RSA types would create *rsa.PublicKey or *rsa.PrivateKey, + // EC types would create *ecdsa.PublicKey or *ecdsa.PrivateKey, + // and OctetSeq types create a []byte key. + Materialize() (interface{}, error) + GenerateKey(*RawKeyJSON) error +} + +// RawKeyJSON is generic type that represents any kind JWK +type RawKeyJSON struct { + StandardHeaders + jwa.AlgorithmParameters +} + +// RawKeySetJSON is generic type that represents a JWK Set +type RawKeySetJSON struct { + Keys []RawKeyJSON `json:"keys"` +} + +// RSAPublicKey is a type of JWK generated from RSA public keys +type RSAPublicKey struct { + *StandardHeaders + key *rsa.PublicKey +} + +// RSAPrivateKey is a type of JWK generated from RSA private keys +type RSAPrivateKey struct { + *StandardHeaders + *jwa.AlgorithmParameters + key *rsa.PrivateKey +} + +// SymmetricKey is a type of JWK generated from symmetric keys +type SymmetricKey struct { + *StandardHeaders + key []byte +} + +// ECDSAPublicKey is a type of JWK generated from ECDSA public keys +type ECDSAPublicKey struct { + *StandardHeaders + key *ecdsa.PublicKey +} + +// ECDSAPrivateKey is a type of JWK generated from ECDH-ES private keys +type ECDSAPrivateKey struct { + *StandardHeaders + key *ecdsa.PrivateKey +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/jwx/jwk/jwk.go b/vendor/github.com/open-policy-agent/opa/internal/jwx/jwk/jwk.go new file mode 100644 index 00000000..22ccf8df --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/jwx/jwk/jwk.go @@ -0,0 +1,150 @@ +// Package jwk implements JWK as described in https://tools.ietf.org/html/rfc7517 +package jwk + +import ( + "crypto/ecdsa" + "crypto/rsa" + "encoding/json" + + "github.com/pkg/errors" + + "github.com/open-policy-agent/opa/internal/jwx/jwa" +) + +// GetPublicKey returns the public key based on the private key type. +// For rsa key types *rsa.PublicKey is returned; for ecdsa key types *ecdsa.PublicKey; +// for byte slice (raw) keys, the key itself is returned. If the corresponding +// public key cannot be deduced, an error is returned +func GetPublicKey(key interface{}) (interface{}, error) { + if key == nil { + return nil, errors.New(`jwk.New requires a non-nil key`) + } + + switch v := key.(type) { + // Mental note: although Public() is defined in both types, + // you can not coalesce the clauses for rsa.PrivateKey and + // ecdsa.PrivateKey, as then `v` becomes interface{} + // b/c the compiler cannot deduce the exact type. + case *rsa.PrivateKey: + return v.Public(), nil + case *ecdsa.PrivateKey: + return v.Public(), nil + case []byte: + return v, nil + default: + return nil, errors.Errorf(`invalid key type %T`, key) + } +} + +// GetKeyTypeFromKey creates a jwk.Key from the given key. +func GetKeyTypeFromKey(key interface{}) jwa.KeyType { + + switch key.(type) { + case *rsa.PrivateKey, *rsa.PublicKey: + return jwa.RSA + case *ecdsa.PrivateKey, *ecdsa.PublicKey: + return jwa.EC + case []byte: + return jwa.OctetSeq + default: + return jwa.InvalidKeyType + } +} + +// New creates a jwk.Key from the given key. +func New(key interface{}) (Key, error) { + if key == nil { + return nil, errors.New(`jwk.New requires a non-nil key`) + } + + switch v := key.(type) { + case *rsa.PrivateKey: + return newRSAPrivateKey(v) + case *rsa.PublicKey: + return newRSAPublicKey(v) + case *ecdsa.PrivateKey: + return newECDSAPrivateKey(v) + case *ecdsa.PublicKey: + return newECDSAPublicKey(v) + case []byte: + return newSymmetricKey(v) + default: + return nil, errors.Errorf(`invalid key type %T`, key) + } +} + +func parse(jwkSrc string) (*Set, error) { + + var jwkKeySet Set + var jwkKey Key + rawKeySetJSON := &RawKeySetJSON{} + err := json.Unmarshal([]byte(jwkSrc), rawKeySetJSON) + if err != nil { + return nil, errors.Wrap(err, "Failed to unmarshal JWK Set") + } + if len(rawKeySetJSON.Keys) == 0 { + + // It might be a single key + rawKeyJSON := &RawKeyJSON{} + err := json.Unmarshal([]byte(jwkSrc), rawKeyJSON) + if err != nil { + return nil, errors.Wrap(err, "Failed to unmarshal JWK") + } + jwkKey, err = rawKeyJSON.GenerateKey() + if err != nil { + return nil, errors.Wrap(err, "Failed to generate key") + } + // Add to set + jwkKeySet.Keys = append(jwkKeySet.Keys, jwkKey) + } else { + for i := range rawKeySetJSON.Keys { + rawKeyJSON := rawKeySetJSON.Keys[i] + jwkKey, err = rawKeyJSON.GenerateKey() + if err != nil { + return nil, errors.Wrap(err, "Failed to generate key: %s") + } + jwkKeySet.Keys = append(jwkKeySet.Keys, jwkKey) + } + } + return &jwkKeySet, nil +} + +// ParseBytes parses JWK from the incoming byte buffer. +func ParseBytes(buf []byte) (*Set, error) { + return parse(string(buf[:])) +} + +// ParseString parses JWK from the incoming string. +func ParseString(s string) (*Set, error) { + return parse(s) +} + +// GenerateKey creates an internal representation of a key from a raw JWK JSON +func (r *RawKeyJSON) GenerateKey() (Key, error) { + + var key Key + + switch r.KeyType { + case jwa.RSA: + if r.D != nil { + key = &RSAPrivateKey{} + } else { + key = &RSAPublicKey{} + } + case jwa.EC: + if r.D != nil { + key = &ECDSAPrivateKey{} + } else { + key = &ECDSAPublicKey{} + } + case jwa.OctetSeq: + key = &SymmetricKey{} + default: + return nil, errors.Errorf(`Unrecognized key type`) + } + err := key.GenerateKey(r) + if err != nil { + return nil, errors.Wrap(err, "Failed to generate key from JWK") + } + return key, nil +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/jwx/jwk/key_ops.go b/vendor/github.com/open-policy-agent/opa/internal/jwx/jwk/key_ops.go new file mode 100644 index 00000000..36d4dd3e --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/jwx/jwk/key_ops.go @@ -0,0 +1,68 @@ +package jwk + +import ( + "encoding/json" + "fmt" + + "github.com/pkg/errors" +) + +// KeyUsageType is used to denote what this key should be used for +type KeyUsageType string + +const ( + // ForSignature is the value used in the headers to indicate that + // this key should be used for signatures + ForSignature KeyUsageType = "sig" + // ForEncryption is the value used in the headers to indicate that + // this key should be used for encryptiong + ForEncryption KeyUsageType = "enc" +) + +// KeyOperation is used to denote the allowed operations for a Key +type KeyOperation string + +// KeyOperationList represents an slice of KeyOperation +type KeyOperationList []KeyOperation + +var keyOps = map[string]struct{}{"sign": {}, "verify": {}, "encrypt": {}, "decrypt": {}, "wrapKey": {}, "unwrapKey": {}, "deriveKey": {}, "deriveBits": {}} + +// KeyOperation constants +const ( + KeyOpSign KeyOperation = "sign" // (compute digital signature or MAC) + KeyOpVerify KeyOperation = "verify" // (verify digital signature or MAC) + KeyOpEncrypt KeyOperation = "encrypt" // (encrypt content) + KeyOpDecrypt KeyOperation = "decrypt" // (decrypt content and validate decryption, if applicable) + KeyOpWrapKey KeyOperation = "wrapKey" // (encrypt key) + KeyOpUnwrapKey KeyOperation = "unwrapKey" // (decrypt key and validate decryption, if applicable) + KeyOpDeriveKey KeyOperation = "deriveKey" // (derive key) + KeyOpDeriveBits KeyOperation = "deriveBits" // (derive bits not to be used as a key) +) + +// Accept determines if Key Operation is valid +func (keyOperationList *KeyOperationList) Accept(v interface{}) error { + switch x := v.(type) { + case KeyOperationList: + *keyOperationList = x + return nil + default: + return errors.Errorf(`invalid value %T`, v) + } +} + +// UnmarshalJSON unmarshals and checks data as KeyType Algorithm +func (keyOperationList *KeyOperationList) UnmarshalJSON(data []byte) error { + var tempKeyOperationList []string + err := json.Unmarshal(data, &tempKeyOperationList) + if err != nil { + return fmt.Errorf("invalid key operation") + } + for _, value := range tempKeyOperationList { + _, ok := keyOps[value] + if !ok { + return fmt.Errorf("unknown key operation") + } + *keyOperationList = append(*keyOperationList, KeyOperation(value)) + } + return nil +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/jwx/jwk/rsa.go b/vendor/github.com/open-policy-agent/opa/internal/jwx/jwk/rsa.go new file mode 100644 index 00000000..1a5cba47 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/jwx/jwk/rsa.go @@ -0,0 +1,133 @@ +package jwk + +import ( + "crypto/rsa" + "encoding/binary" + "math/big" + + "github.com/pkg/errors" + + "github.com/open-policy-agent/opa/internal/jwx/jwa" +) + +func newRSAPublicKey(key *rsa.PublicKey) (*RSAPublicKey, error) { + + var hdr StandardHeaders + err := hdr.Set(KeyTypeKey, jwa.RSA) + if err != nil { + return nil, errors.Wrapf(err, "Failed to set Key Type") + } + return &RSAPublicKey{ + StandardHeaders: &hdr, + key: key, + }, nil +} + +func newRSAPrivateKey(key *rsa.PrivateKey) (*RSAPrivateKey, error) { + + var hdr StandardHeaders + err := hdr.Set(KeyTypeKey, jwa.RSA) + if err != nil { + return nil, errors.Wrapf(err, "Failed to set Key Type") + } + + var algoParams jwa.AlgorithmParameters + + // it is needed to use raw encoding to omit the "=" paddings at the end + algoParams.D = key.D.Bytes() + algoParams.P = key.Primes[0].Bytes() + algoParams.Q = key.Primes[1].Bytes() + algoParams.Dp = key.Precomputed.Dp.Bytes() + algoParams.Dq = key.Precomputed.Dq.Bytes() + algoParams.Qi = key.Precomputed.Qinv.Bytes() + + // "modulus" (N) from the public key in the private key + algoParams.N = key.PublicKey.N.Bytes() + + // make the E a.k.a "coprime" + // https://en.wikipedia.org/wiki/RSA_(cryptosystem) + coprime := make([]byte, 8) + binary.BigEndian.PutUint64(coprime, uint64(key.PublicKey.E)) + // find the 1st index of non 0x0 paddings from the beginning + i := 0 + for ; i < len(coprime); i++ { + if coprime[i] != 0x0 { + break + } + } + algoParams.E = coprime[i:] + + return &RSAPrivateKey{ + StandardHeaders: &hdr, + AlgorithmParameters: &algoParams, + key: key, + }, nil +} + +// Materialize returns the standard RSA Public Key representation stored in the internal representation +func (k *RSAPublicKey) Materialize() (interface{}, error) { + if k.key == nil { + return nil, errors.New(`key has no rsa.PublicKey associated with it`) + } + return k.key, nil +} + +// Materialize returns the standard RSA Private Key representation stored in the internal representation +func (k *RSAPrivateKey) Materialize() (interface{}, error) { + if k.key == nil { + return nil, errors.New(`key has no rsa.PrivateKey associated with it`) + } + return k.key, nil +} + +// GenerateKey creates a RSAPublicKey from a RawKeyJSON +func (k *RSAPublicKey) GenerateKey(keyJSON *RawKeyJSON) error { + + if keyJSON.N == nil || keyJSON.E == nil { + return errors.Errorf("Missing mandatory key parameters N or E") + } + rsaPublicKey := &rsa.PublicKey{ + N: (&big.Int{}).SetBytes(keyJSON.N.Bytes()), + E: int((&big.Int{}).SetBytes(keyJSON.E.Bytes()).Int64()), + } + k.key = rsaPublicKey + k.StandardHeaders = &keyJSON.StandardHeaders + return nil +} + +// GenerateKey creates a RSAPublicKey from a RawKeyJSON +func (k *RSAPrivateKey) GenerateKey(keyJSON *RawKeyJSON) error { + + rsaPublicKey := &RSAPublicKey{} + err := rsaPublicKey.GenerateKey(keyJSON) + if err != nil { + return errors.Wrap(err, "failed to generate public key") + } + + if keyJSON.D == nil || keyJSON.P == nil || keyJSON.Q == nil { + return errors.Errorf("Missing mandatory key parameters D, P or Q") + } + privateKey := &rsa.PrivateKey{ + PublicKey: *rsaPublicKey.key, + D: (&big.Int{}).SetBytes(keyJSON.D.Bytes()), + Primes: []*big.Int{ + (&big.Int{}).SetBytes(keyJSON.P.Bytes()), + (&big.Int{}).SetBytes(keyJSON.Q.Bytes()), + }, + } + + if keyJSON.Dp.Len() > 0 { + privateKey.Precomputed.Dp = (&big.Int{}).SetBytes(keyJSON.Dp.Bytes()) + } + if keyJSON.Dq.Len() > 0 { + privateKey.Precomputed.Dq = (&big.Int{}).SetBytes(keyJSON.Dq.Bytes()) + } + if keyJSON.Qi.Len() > 0 { + privateKey.Precomputed.Qinv = (&big.Int{}).SetBytes(keyJSON.Qi.Bytes()) + } + + k.key = privateKey + k.StandardHeaders = &keyJSON.StandardHeaders + k.AlgorithmParameters = &keyJSON.AlgorithmParameters + return nil +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/jwx/jwk/symmetric.go b/vendor/github.com/open-policy-agent/opa/internal/jwx/jwk/symmetric.go new file mode 100644 index 00000000..8a073615 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/jwx/jwk/symmetric.go @@ -0,0 +1,41 @@ +package jwk + +import ( + "github.com/pkg/errors" + + "github.com/open-policy-agent/opa/internal/jwx/jwa" +) + +func newSymmetricKey(key []byte) (*SymmetricKey, error) { + var hdr StandardHeaders + + err := hdr.Set(KeyTypeKey, jwa.OctetSeq) + if err != nil { + return nil, errors.Wrapf(err, "Failed to set Key Type") + } + return &SymmetricKey{ + StandardHeaders: &hdr, + key: key, + }, nil +} + +// Materialize returns the octets for this symmetric key. +// Since this is a symmetric key, this just calls Octets +func (s SymmetricKey) Materialize() (interface{}, error) { + return s.Octets(), nil +} + +// Octets returns the octets in the key +func (s SymmetricKey) Octets() []byte { + return s.key +} + +// GenerateKey creates a Symmetric key from a RawKeyJSON +func (s *SymmetricKey) GenerateKey(keyJSON *RawKeyJSON) error { + + *s = SymmetricKey{ + StandardHeaders: &keyJSON.StandardHeaders, + key: keyJSON.K, + } + return nil +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/jwx/jws/headers.go b/vendor/github.com/open-policy-agent/opa/internal/jwx/jws/headers.go new file mode 100644 index 00000000..045e38fa --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/jwx/jws/headers.go @@ -0,0 +1,154 @@ +package jws + +import ( + "github.com/pkg/errors" + + "github.com/open-policy-agent/opa/internal/jwx/jwa" +) + +// Constants for JWS Common parameters +const ( + AlgorithmKey = "alg" + ContentTypeKey = "cty" + CriticalKey = "crit" + JWKKey = "jwk" + JWKSetURLKey = "jku" + KeyIDKey = "kid" + PrivateParamsKey = "privateParams" + TypeKey = "typ" +) + +// Headers provides a common interface for common header parameters +type Headers interface { + Get(string) (interface{}, bool) + Set(string, interface{}) error + GetAlgorithm() jwa.SignatureAlgorithm +} + +// StandardHeaders contains JWS common parameters. +type StandardHeaders struct { + Algorithm jwa.SignatureAlgorithm `json:"alg,omitempty"` // https://tools.ietf.org/html/rfc7515#section-4.1.1 + ContentType string `json:"cty,omitempty"` // https://tools.ietf.org/html/rfc7515#section-4.1.10 + Critical []string `json:"crit,omitempty"` // https://tools.ietf.org/html/rfc7515#section-4.1.11 + JWK string `json:"jwk,omitempty"` // https://tools.ietf.org/html/rfc7515#section-4.1.3 + JWKSetURL string `json:"jku,omitempty"` // https://tools.ietf.org/html/rfc7515#section-4.1.2 + KeyID string `json:"kid,omitempty"` // https://tools.ietf.org/html/rfc7515#section-4.1.4 + PrivateParams map[string]interface{} `json:"privateParams,omitempty"` // https://tools.ietf.org/html/rfc7515#section-4.1.9 + Type string `json:"typ,omitempty"` // https://tools.ietf.org/html/rfc7515#section-4.1.9 +} + +// GetAlgorithm returns algorithm +func (h *StandardHeaders) GetAlgorithm() jwa.SignatureAlgorithm { + return h.Algorithm +} + +// Get is a general getter function for StandardHeaders structure +func (h *StandardHeaders) Get(name string) (interface{}, bool) { + switch name { + case AlgorithmKey: + v := h.Algorithm + if v == "" { + return nil, false + } + return v, true + case ContentTypeKey: + v := h.ContentType + if v == "" { + return nil, false + } + return v, true + case CriticalKey: + v := h.Critical + if len(v) == 0 { + return nil, false + } + return v, true + case JWKKey: + v := h.JWK + if v == "" { + return nil, false + } + return v, true + case JWKSetURLKey: + v := h.JWKSetURL + if v == "" { + return nil, false + } + return v, true + case KeyIDKey: + v := h.KeyID + if v == "" { + return nil, false + } + return v, true + case PrivateParamsKey: + v := h.PrivateParams + if len(v) == 0 { + return nil, false + } + return v, true + case TypeKey: + v := h.Type + if v == "" { + return nil, false + } + return v, true + default: + return nil, false + } +} + +// Set is a general setter function for StandardHeaders structure +func (h *StandardHeaders) Set(name string, value interface{}) error { + switch name { + case AlgorithmKey: + if err := h.Algorithm.Accept(value); err != nil { + return errors.Wrapf(err, `invalid value for %s key`, AlgorithmKey) + } + return nil + case ContentTypeKey: + if v, ok := value.(string); ok { + h.ContentType = v + return nil + } + return errors.Errorf(`invalid value for %s key: %T`, ContentTypeKey, value) + case CriticalKey: + if v, ok := value.([]string); ok { + h.Critical = v + return nil + } + return errors.Errorf(`invalid value for %s key: %T`, CriticalKey, value) + case JWKKey: + if v, ok := value.(string); ok { + h.JWK = v + return nil + } + return errors.Errorf(`invalid value for %s key: %T`, JWKKey, value) + case JWKSetURLKey: + if v, ok := value.(string); ok { + h.JWKSetURL = v + return nil + } + return errors.Errorf(`invalid value for %s key: %T`, JWKSetURLKey, value) + case KeyIDKey: + if v, ok := value.(string); ok { + h.KeyID = v + return nil + } + return errors.Errorf(`invalid value for %s key: %T`, KeyIDKey, value) + case PrivateParamsKey: + if v, ok := value.(map[string]interface{}); ok { + h.PrivateParams = v + return nil + } + return errors.Errorf(`invalid value for %s key: %T`, PrivateParamsKey, value) + case TypeKey: + if v, ok := value.(string); ok { + h.Type = v + return nil + } + return errors.Errorf(`invalid value for %s key: %T`, TypeKey, value) + default: + return errors.Errorf(`invalid key: %s`, name) + } +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/jwx/jws/interface.go b/vendor/github.com/open-policy-agent/opa/internal/jwx/jws/interface.go new file mode 100644 index 00000000..e647c8ac --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/jwx/jws/interface.go @@ -0,0 +1,22 @@ +package jws + +// Message represents a full JWS encoded message. Flattened serialization +// is not supported as a struct, but rather it's represented as a +// Message struct with only one `Signature` element. +// +// Do not expect to use the Message object to verify or construct a +// signed payloads with. You should only use this when you want to actually +// want to programmatically view the contents for the full JWS Payload. +// +// To sign and verify, use the appropriate `SignWithOption()` nad `Verify()` functions +type Message struct { + Payload []byte `json:"payload"` + Signatures []*Signature `json:"signatures,omitempty"` +} + +// Signature represents the headers and signature of a JWS message +type Signature struct { + Headers Headers `json:"header,omitempty"` // Unprotected Headers + Protected Headers `json:"Protected,omitempty"` // Protected Headers + Signature []byte `json:"signature,omitempty"` // GetSignature +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/jwx/jws/jws.go b/vendor/github.com/open-policy-agent/opa/internal/jwx/jws/jws.go new file mode 100644 index 00000000..bfa498bb --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/jwx/jws/jws.go @@ -0,0 +1,221 @@ +// Package jws implements the digital Signature on JSON based data +// structures as described in https://tools.ietf.org/html/rfc7515 +// +// If you do not care about the details, the only things that you +// would need to use are the following functions: +// +// jws.SignWithOption(Payload, algorithm, key) +// jws.Verify(encodedjws, algorithm, key) +// +// To sign, simply use `jws.SignWithOption`. `Payload` is a []byte buffer that +// contains whatever data you want to sign. `alg` is one of the +// jwa.SignatureAlgorithm constants from package jwa. For RSA and +// ECDSA family of algorithms, you will need to prepare a private key. +// For HMAC family, you just need a []byte value. The `jws.SignWithOption` +// function will return the encoded JWS message on success. +// +// To verify, use `jws.Verify`. It will parse the `encodedjws` buffer +// and verify the result using `algorithm` and `key`. Upon successful +// verification, the original Payload is returned, so you can work on it. +package jws + +import ( + "bytes" + "crypto/rand" + "encoding/base64" + "encoding/json" + "io" + "strings" + + "github.com/open-policy-agent/opa/internal/jwx/jwa" + "github.com/open-policy-agent/opa/internal/jwx/jwk" + "github.com/open-policy-agent/opa/internal/jwx/jws/sign" + "github.com/open-policy-agent/opa/internal/jwx/jws/verify" + + "github.com/pkg/errors" +) + +// SignLiteral generates a Signature for the given Payload and Headers, and serializes +// it in compact serialization format. In this format you may NOT use +// multiple signers. +// +func SignLiteral(payload []byte, alg jwa.SignatureAlgorithm, key interface{}, hdrBuf []byte, rnd io.Reader) ([]byte, error) { + encodedHdr := base64.RawURLEncoding.EncodeToString(hdrBuf) + encodedPayload := base64.RawURLEncoding.EncodeToString(payload) + signingInput := strings.Join( + []string{ + encodedHdr, + encodedPayload, + }, ".", + ) + signer, err := sign.New(alg) + if err != nil { + return nil, errors.Wrap(err, `failed to create signer`) + } + + var signature []byte + switch s := signer.(type) { + case *sign.ECDSASigner: + signature, err = s.SignWithRand([]byte(signingInput), key, rnd) + default: + signature, err = signer.Sign([]byte(signingInput), key) + } + if err != nil { + return nil, errors.Wrap(err, `failed to sign Payload`) + } + encodedSignature := base64.RawURLEncoding.EncodeToString(signature) + compactSerialization := strings.Join( + []string{ + signingInput, + encodedSignature, + }, ".", + ) + return []byte(compactSerialization), nil +} + +// SignWithOption generates a Signature for the given Payload, and serializes +// it in compact serialization format. In this format you may NOT use +// multiple signers. +// +// If you would like to pass custom Headers, use the WithHeaders option. +func SignWithOption(payload []byte, alg jwa.SignatureAlgorithm, key interface{}) ([]byte, error) { + var headers Headers = &StandardHeaders{} + + err := headers.Set(AlgorithmKey, alg) + if err != nil { + return nil, errors.Wrap(err, "Failed to set alg value") + } + + hdrBuf, err := json.Marshal(headers) + if err != nil { + return nil, errors.Wrap(err, `failed to marshal Headers`) + } + // NOTE(sr): we don't use SignWithOption -- if we did, this rand.Reader + // should come from the BuiltinContext's Seed, too. + return SignLiteral(payload, alg, key, hdrBuf, rand.Reader) +} + +// Verify checks if the given JWS message is verifiable using `alg` and `key`. +// If the verification is successful, `err` is nil, and the content of the +// Payload that was signed is returned. If you need more fine-grained +// control of the verification process, manually call `Parse`, generate a +// verifier, and call `Verify` on the parsed JWS message object. +func Verify(buf []byte, alg jwa.SignatureAlgorithm, key interface{}) (ret []byte, err error) { + + verifier, err := verify.New(alg) + if err != nil { + return nil, errors.Wrap(err, "failed to create verifier") + } + + buf = bytes.TrimSpace(buf) + if len(buf) == 0 { + return nil, errors.New(`attempt to verify empty buffer`) + } + + parts, err := SplitCompact(string(buf[:])) + if err != nil { + return nil, errors.Wrap(err, `failed extract from compact serialization format`) + } + + signingInput := strings.Join( + []string{ + parts[0], + parts[1], + }, ".", + ) + + decodedSignature, err := base64.RawURLEncoding.DecodeString(parts[2]) + if err != nil { + return nil, errors.Wrap(err, "Failed to decode signature") + } + if err := verifier.Verify([]byte(signingInput), decodedSignature, key); err != nil { + return nil, errors.Wrap(err, "Failed to verify message") + } + + if decodedPayload, err := base64.RawURLEncoding.DecodeString(parts[1]); err == nil { + return decodedPayload, nil + } + return nil, errors.Wrap(err, "Failed to decode Payload") +} + +// VerifyWithJWK verifies the JWS message using the specified JWK +func VerifyWithJWK(buf []byte, key jwk.Key) (payload []byte, err error) { + + keyVal, err := key.Materialize() + if err != nil { + return nil, errors.Wrap(err, "Failed to materialize key") + } + return Verify(buf, key.GetAlgorithm(), keyVal) +} + +// VerifyWithJWKSet verifies the JWS message using JWK key set. +// By default it will only pick up keys that have the "use" key +// set to either "sig" or "enc", but you can override it by +// providing a keyaccept function. +func VerifyWithJWKSet(buf []byte, keyset *jwk.Set) (payload []byte, err error) { + + for _, key := range keyset.Keys { + payload, err := VerifyWithJWK(buf, key) + if err == nil { + return payload, nil + } + } + return nil, errors.New("failed to verify with any of the keys") +} + +// ParseByte parses a JWS value serialized via compact serialization and provided as []byte. +func ParseByte(jwsCompact []byte) (m *Message, err error) { + return parseCompact(string(jwsCompact[:])) +} + +// ParseString parses a JWS value serialized via compact serialization and provided as string. +func ParseString(s string) (*Message, error) { + return parseCompact(s) +} + +// SplitCompact splits a JWT and returns its three parts +// separately: Protected Headers, Payload and Signature. +func SplitCompact(jwsCompact string) ([]string, error) { + + parts := strings.Split(jwsCompact, ".") + if len(parts) < 3 { + return nil, errors.New("Failed to split compact serialization") + } + return parts, nil +} + +// parseCompact parses a JWS value serialized via compact serialization. +func parseCompact(str string) (m *Message, err error) { + + var decodedHeader, decodedPayload, decodedSignature []byte + parts, err := SplitCompact(str) + if err != nil { + return nil, errors.Wrap(err, `invalid compact serialization format`) + } + + if decodedHeader, err = base64.RawURLEncoding.DecodeString(parts[0]); err != nil { + return nil, errors.Wrap(err, `failed to decode Headers`) + } + var hdr StandardHeaders + if err := json.Unmarshal(decodedHeader, &hdr); err != nil { + return nil, errors.Wrap(err, `failed to parse JOSE Headers`) + } + + if decodedPayload, err = base64.RawURLEncoding.DecodeString(parts[1]); err != nil { + return nil, errors.Wrap(err, `failed to decode Payload`) + } + + if len(parts) > 2 { + if decodedSignature, err = base64.RawURLEncoding.DecodeString(parts[2]); err != nil { + return nil, errors.Wrap(err, `failed to decode Signature`) + } + } + + var msg Message + msg.Payload = decodedPayload + msg.Signatures = append(msg.Signatures, &Signature{ + Protected: &hdr, + Signature: decodedSignature, + }) + return &msg, nil +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/jwx/jws/message.go b/vendor/github.com/open-policy-agent/opa/internal/jwx/jws/message.go new file mode 100644 index 00000000..1366a3d7 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/jwx/jws/message.go @@ -0,0 +1,26 @@ +package jws + +// PublicHeaders returns the public headers in a JWS +func (s Signature) PublicHeaders() Headers { + return s.Headers +} + +// ProtectedHeaders returns the protected headers in a JWS +func (s Signature) ProtectedHeaders() Headers { + return s.Protected +} + +// GetSignature returns the signature in a JWS +func (s Signature) GetSignature() []byte { + return s.Signature +} + +// GetPayload returns the payload in a JWS +func (m Message) GetPayload() []byte { + return m.Payload +} + +// GetSignatures returns the all signatures in a JWS +func (m Message) GetSignatures() []*Signature { + return m.Signatures +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/jwx/jws/sign/ecdsa.go b/vendor/github.com/open-policy-agent/opa/internal/jwx/jws/sign/ecdsa.go new file mode 100644 index 00000000..62af72b6 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/jwx/jws/sign/ecdsa.go @@ -0,0 +1,90 @@ +package sign + +import ( + "crypto" + "crypto/ecdsa" + "crypto/rand" + "io" + + "github.com/open-policy-agent/opa/internal/jwx/jwa" + + "github.com/pkg/errors" +) + +var ecdsaSignFuncs = map[jwa.SignatureAlgorithm]ecdsaSignFunc{} + +func init() { + algs := map[jwa.SignatureAlgorithm]crypto.Hash{ + jwa.ES256: crypto.SHA256, + jwa.ES384: crypto.SHA384, + jwa.ES512: crypto.SHA512, + } + + for alg, h := range algs { + ecdsaSignFuncs[alg] = makeECDSASignFunc(h) + } +} + +func makeECDSASignFunc(hash crypto.Hash) ecdsaSignFunc { + return ecdsaSignFunc(func(payload []byte, key *ecdsa.PrivateKey, rnd io.Reader) ([]byte, error) { + curveBits := key.Curve.Params().BitSize + keyBytes := curveBits / 8 + // Curve bits do not need to be a multiple of 8. + if curveBits%8 > 0 { + keyBytes++ + } + h := hash.New() + h.Write(payload) + r, s, err := ecdsa.Sign(rnd, key, h.Sum(nil)) + if err != nil { + return nil, errors.Wrap(err, "failed to sign payload using ecdsa") + } + + rBytes := r.Bytes() + rBytesPadded := make([]byte, keyBytes) + copy(rBytesPadded[keyBytes-len(rBytes):], rBytes) + + sBytes := s.Bytes() + sBytesPadded := make([]byte, keyBytes) + copy(sBytesPadded[keyBytes-len(sBytes):], sBytes) + + out := append(rBytesPadded, sBytesPadded...) + return out, nil + }) +} + +func newECDSA(alg jwa.SignatureAlgorithm) (*ECDSASigner, error) { + signfn, ok := ecdsaSignFuncs[alg] + if !ok { + return nil, errors.Errorf(`unsupported algorithm while trying to create ECDSA signer: %s`, alg) + } + + return &ECDSASigner{ + alg: alg, + sign: signfn, + }, nil +} + +// Algorithm returns the signer algorithm +func (s ECDSASigner) Algorithm() jwa.SignatureAlgorithm { + return s.alg +} + +// SignWithRand signs payload with a ECDSA private key and a provided randomness +// source (such as `rand.Reader`). +func (s ECDSASigner) SignWithRand(payload []byte, key interface{}, r io.Reader) ([]byte, error) { + if key == nil { + return nil, errors.New(`missing private key while signing payload`) + } + + privateKey, ok := key.(*ecdsa.PrivateKey) + if !ok { + return nil, errors.Errorf(`invalid key type %T. *ecdsa.PrivateKey is required`, key) + } + return s.sign(payload, privateKey, r) +} + +// Sign signs payload with a ECDSA private key +func (s ECDSASigner) Sign(payload []byte, key interface{}) ([]byte, error) { + return s.SignWithRand(payload, key, rand.Reader) +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/jwx/jws/sign/hmac.go b/vendor/github.com/open-policy-agent/opa/internal/jwx/jws/sign/hmac.go new file mode 100644 index 00000000..cbf7b9f0 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/jwx/jws/sign/hmac.go @@ -0,0 +1,66 @@ +package sign + +import ( + "crypto/hmac" + "crypto/sha256" + "crypto/sha512" + "hash" + + "github.com/open-policy-agent/opa/internal/jwx/jwa" + + "github.com/pkg/errors" +) + +var hmacSignFuncs = map[jwa.SignatureAlgorithm]hmacSignFunc{} + +func init() { + algs := map[jwa.SignatureAlgorithm]func() hash.Hash{ + jwa.HS256: sha256.New, + jwa.HS384: sha512.New384, + jwa.HS512: sha512.New, + } + + for alg, h := range algs { + hmacSignFuncs[alg] = makeHMACSignFunc(h) + + } +} + +func newHMAC(alg jwa.SignatureAlgorithm) (*HMACSigner, error) { + signer, ok := hmacSignFuncs[alg] + if !ok { + return nil, errors.Errorf(`unsupported algorithm while trying to create HMAC signer: %s`, alg) + } + + return &HMACSigner{ + alg: alg, + sign: signer, + }, nil +} + +func makeHMACSignFunc(hfunc func() hash.Hash) hmacSignFunc { + return hmacSignFunc(func(payload []byte, key []byte) ([]byte, error) { + h := hmac.New(hfunc, key) + h.Write(payload) + return h.Sum(nil), nil + }) +} + +// Algorithm returns the signer algorithm +func (s HMACSigner) Algorithm() jwa.SignatureAlgorithm { + return s.alg +} + +// Sign signs payload with a Symmetric key +func (s HMACSigner) Sign(payload []byte, key interface{}) ([]byte, error) { + hmackey, ok := key.([]byte) + if !ok { + return nil, errors.Errorf(`invalid key type %T. []byte is required`, key) + } + + if len(hmackey) == 0 { + return nil, errors.New(`missing key while signing payload`) + } + + return s.sign(payload, hmackey) +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/jwx/jws/sign/interface.go b/vendor/github.com/open-policy-agent/opa/internal/jwx/jws/sign/interface.go new file mode 100644 index 00000000..2ef2bee4 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/jwx/jws/sign/interface.go @@ -0,0 +1,46 @@ +package sign + +import ( + "crypto/ecdsa" + "crypto/rsa" + "io" + + "github.com/open-policy-agent/opa/internal/jwx/jwa" +) + +// Signer provides a common interface for supported alg signing methods +type Signer interface { + // Sign creates a signature for the given `payload`. + // `key` is the key used for signing the payload, and is usually + // the private key type associated with the signature method. For example, + // for `jwa.RSXXX` and `jwa.PSXXX` types, you need to pass the + // `*"crypto/rsa".PrivateKey` type. + // Check the documentation for each signer for details + Sign(payload []byte, key interface{}) ([]byte, error) + + Algorithm() jwa.SignatureAlgorithm +} + +type rsaSignFunc func([]byte, *rsa.PrivateKey) ([]byte, error) + +// RSASigner uses crypto/rsa to sign the payloads. +type RSASigner struct { + alg jwa.SignatureAlgorithm + sign rsaSignFunc +} + +type ecdsaSignFunc func([]byte, *ecdsa.PrivateKey, io.Reader) ([]byte, error) + +// ECDSASigner uses crypto/ecdsa to sign the payloads. +type ECDSASigner struct { + alg jwa.SignatureAlgorithm + sign ecdsaSignFunc +} + +type hmacSignFunc func([]byte, []byte) ([]byte, error) + +// HMACSigner uses crypto/hmac to sign the payloads. +type HMACSigner struct { + alg jwa.SignatureAlgorithm + sign hmacSignFunc +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/jwx/jws/sign/rsa.go b/vendor/github.com/open-policy-agent/opa/internal/jwx/jws/sign/rsa.go new file mode 100644 index 00000000..bc51dbcd --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/jwx/jws/sign/rsa.go @@ -0,0 +1,97 @@ +package sign + +import ( + "crypto" + "crypto/rand" + "crypto/rsa" + + "github.com/open-policy-agent/opa/internal/jwx/jwa" + + "github.com/pkg/errors" +) + +var rsaSignFuncs = map[jwa.SignatureAlgorithm]rsaSignFunc{} + +func init() { + algs := map[jwa.SignatureAlgorithm]struct { + Hash crypto.Hash + SignFunc func(crypto.Hash) rsaSignFunc + }{ + jwa.RS256: { + Hash: crypto.SHA256, + SignFunc: makeSignPKCS1v15, + }, + jwa.RS384: { + Hash: crypto.SHA384, + SignFunc: makeSignPKCS1v15, + }, + jwa.RS512: { + Hash: crypto.SHA512, + SignFunc: makeSignPKCS1v15, + }, + jwa.PS256: { + Hash: crypto.SHA256, + SignFunc: makeSignPSS, + }, + jwa.PS384: { + Hash: crypto.SHA384, + SignFunc: makeSignPSS, + }, + jwa.PS512: { + Hash: crypto.SHA512, + SignFunc: makeSignPSS, + }, + } + + for alg, item := range algs { + rsaSignFuncs[alg] = item.SignFunc(item.Hash) + } +} + +func makeSignPKCS1v15(hash crypto.Hash) rsaSignFunc { + return rsaSignFunc(func(payload []byte, key *rsa.PrivateKey) ([]byte, error) { + h := hash.New() + h.Write(payload) + return rsa.SignPKCS1v15(rand.Reader, key, hash, h.Sum(nil)) + }) +} + +func makeSignPSS(hash crypto.Hash) rsaSignFunc { + return rsaSignFunc(func(payload []byte, key *rsa.PrivateKey) ([]byte, error) { + h := hash.New() + h.Write(payload) + return rsa.SignPSS(rand.Reader, key, hash, h.Sum(nil), &rsa.PSSOptions{ + SaltLength: rsa.PSSSaltLengthAuto, + }) + }) +} + +func newRSA(alg jwa.SignatureAlgorithm) (*RSASigner, error) { + signfn, ok := rsaSignFuncs[alg] + if !ok { + return nil, errors.Errorf(`unsupported algorithm while trying to create RSA signer: %s`, alg) + } + return &RSASigner{ + alg: alg, + sign: signfn, + }, nil +} + +// Algorithm returns the signer algorithm +func (s RSASigner) Algorithm() jwa.SignatureAlgorithm { + return s.alg +} + +// Sign creates a signature using crypto/rsa. key must be a non-nil instance of +// `*"crypto/rsa".PrivateKey`. +func (s RSASigner) Sign(payload []byte, key interface{}) ([]byte, error) { + if key == nil { + return nil, errors.New(`missing private key while signing payload`) + } + rsakey, ok := key.(*rsa.PrivateKey) + if !ok { + return nil, errors.Errorf(`invalid key type %T. *rsa.PrivateKey is required`, key) + } + + return s.sign(payload, rsakey) +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/jwx/jws/sign/sign.go b/vendor/github.com/open-policy-agent/opa/internal/jwx/jws/sign/sign.go new file mode 100644 index 00000000..ec880606 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/jwx/jws/sign/sign.go @@ -0,0 +1,67 @@ +package sign + +import ( + "crypto/x509" + "encoding/pem" + "fmt" + + "github.com/pkg/errors" + + "github.com/open-policy-agent/opa/internal/jwx/jwa" +) + +// New creates a signer that signs payloads using the given signature algorithm. +func New(alg jwa.SignatureAlgorithm) (Signer, error) { + switch alg { + case jwa.RS256, jwa.RS384, jwa.RS512, jwa.PS256, jwa.PS384, jwa.PS512: + return newRSA(alg) + case jwa.ES256, jwa.ES384, jwa.ES512: + return newECDSA(alg) + case jwa.HS256, jwa.HS384, jwa.HS512: + return newHMAC(alg) + default: + return nil, errors.Errorf(`unsupported signature algorithm %s`, alg) + } +} + +// GetSigningKey returns a *rsa.PrivateKey or *ecdsa.PrivateKey typically encoded in PEM blocks of type "RSA PRIVATE KEY" +// or "EC PRIVATE KEY" for RSA and ECDSA family of algorithms. +// For HMAC family, it return a []byte value +func GetSigningKey(key string, alg jwa.SignatureAlgorithm) (interface{}, error) { + switch alg { + case jwa.RS256, jwa.RS384, jwa.RS512, jwa.PS256, jwa.PS384, jwa.PS512: + block, _ := pem.Decode([]byte(key)) + if block == nil { + return nil, fmt.Errorf("failed to parse PEM block containing the key") + } + + priv, err := x509.ParsePKCS1PrivateKey(block.Bytes) + if err != nil { + pkcs8priv, err2 := x509.ParsePKCS8PrivateKey(block.Bytes) + if err2 != nil { + return nil, fmt.Errorf("error parsing private key (%v), (%v)", err, err2) + } + return pkcs8priv, nil + } + return priv, nil + case jwa.ES256, jwa.ES384, jwa.ES512: + block, _ := pem.Decode([]byte(key)) + if block == nil { + return nil, fmt.Errorf("failed to parse PEM block containing the key") + } + + priv, err := x509.ParseECPrivateKey(block.Bytes) + if err != nil { + pkcs8priv, err2 := x509.ParsePKCS8PrivateKey(block.Bytes) + if err2 != nil { + return nil, fmt.Errorf("error parsing private key (%v), (%v)", err, err2) + } + return pkcs8priv, nil + } + return priv, nil + case jwa.HS256, jwa.HS384, jwa.HS512: + return []byte(key), nil + default: + return nil, errors.Errorf("unsupported signature algorithm: %s", alg) + } +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/jwx/jws/verify/ecdsa.go b/vendor/github.com/open-policy-agent/opa/internal/jwx/jws/verify/ecdsa.go new file mode 100644 index 00000000..e71dc6f8 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/jwx/jws/verify/ecdsa.go @@ -0,0 +1,67 @@ +package verify + +import ( + "crypto" + "crypto/ecdsa" + "math/big" + + "github.com/pkg/errors" + + "github.com/open-policy-agent/opa/internal/jwx/jwa" +) + +var ecdsaVerifyFuncs = map[jwa.SignatureAlgorithm]ecdsaVerifyFunc{} + +func init() { + algs := map[jwa.SignatureAlgorithm]crypto.Hash{ + jwa.ES256: crypto.SHA256, + jwa.ES384: crypto.SHA384, + jwa.ES512: crypto.SHA512, + } + + for alg, h := range algs { + ecdsaVerifyFuncs[alg] = makeECDSAVerifyFunc(h) + } +} + +func makeECDSAVerifyFunc(hash crypto.Hash) ecdsaVerifyFunc { + return ecdsaVerifyFunc(func(payload []byte, signature []byte, key *ecdsa.PublicKey) error { + + r, s := &big.Int{}, &big.Int{} + n := len(signature) / 2 + r.SetBytes(signature[:n]) + s.SetBytes(signature[n:]) + + h := hash.New() + h.Write(payload) + + if !ecdsa.Verify(key, h.Sum(nil), r, s) { + return errors.New(`failed to verify signature using ecdsa`) + } + return nil + }) +} + +func newECDSA(alg jwa.SignatureAlgorithm) (*ECDSAVerifier, error) { + verifyfn, ok := ecdsaVerifyFuncs[alg] + if !ok { + return nil, errors.Errorf(`unsupported algorithm while trying to create ECDSA verifier: %s`, alg) + } + + return &ECDSAVerifier{ + verify: verifyfn, + }, nil +} + +// Verify checks whether the signature for a given input and key is correct +func (v ECDSAVerifier) Verify(payload []byte, signature []byte, key interface{}) error { + if key == nil { + return errors.New(`missing public key while verifying payload`) + } + ecdsakey, ok := key.(*ecdsa.PublicKey) + if !ok { + return errors.Errorf(`invalid key type %T. *ecdsa.PublicKey is required`, key) + } + + return v.verify(payload, signature, ecdsakey) +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/jwx/jws/verify/hmac.go b/vendor/github.com/open-policy-agent/opa/internal/jwx/jws/verify/hmac.go new file mode 100644 index 00000000..77e45887 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/jwx/jws/verify/hmac.go @@ -0,0 +1,33 @@ +package verify + +import ( + "crypto/hmac" + + "github.com/pkg/errors" + + "github.com/open-policy-agent/opa/internal/jwx/jwa" + "github.com/open-policy-agent/opa/internal/jwx/jws/sign" +) + +func newHMAC(alg jwa.SignatureAlgorithm) (*HMACVerifier, error) { + + s, err := sign.New(alg) + if err != nil { + return nil, errors.Wrap(err, `failed to generate HMAC signer`) + } + return &HMACVerifier{signer: s}, nil +} + +// Verify checks whether the signature for a given input and key is correct +func (v HMACVerifier) Verify(signingInput, signature []byte, key interface{}) (err error) { + + expected, err := v.signer.Sign(signingInput, key) + if err != nil { + return errors.Wrap(err, `failed to generated signature`) + } + + if !hmac.Equal(signature, expected) { + return errors.New(`failed to match hmac signature`) + } + return nil +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/jwx/jws/verify/interface.go b/vendor/github.com/open-policy-agent/opa/internal/jwx/jws/verify/interface.go new file mode 100644 index 00000000..f5beb697 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/jwx/jws/verify/interface.go @@ -0,0 +1,39 @@ +package verify + +import ( + "crypto/ecdsa" + "crypto/rsa" + + "github.com/open-policy-agent/opa/internal/jwx/jws/sign" +) + +// Verifier provides a common interface for supported alg verification methods +type Verifier interface { + // Verify checks whether the payload and signature are valid for + // the given key. + // `key` is the key used for verifying the payload, and is usually + // the public key associated with the signature method. For example, + // for `jwa.RSXXX` and `jwa.PSXXX` types, you need to pass the + // `*"crypto/rsa".PublicKey` type. + // Check the documentation for each verifier for details + Verify(payload []byte, signature []byte, key interface{}) error +} + +type rsaVerifyFunc func([]byte, []byte, *rsa.PublicKey) error + +// RSAVerifier implements the Verifier interface +type RSAVerifier struct { + verify rsaVerifyFunc +} + +type ecdsaVerifyFunc func([]byte, []byte, *ecdsa.PublicKey) error + +// ECDSAVerifier implements the Verifier interface +type ECDSAVerifier struct { + verify ecdsaVerifyFunc +} + +// HMACVerifier implements the Verifier interface +type HMACVerifier struct { + signer sign.Signer +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/jwx/jws/verify/rsa.go b/vendor/github.com/open-policy-agent/opa/internal/jwx/jws/verify/rsa.go new file mode 100644 index 00000000..8188ceb1 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/jwx/jws/verify/rsa.go @@ -0,0 +1,88 @@ +package verify + +import ( + "crypto" + "crypto/rsa" + + "github.com/open-policy-agent/opa/internal/jwx/jwa" + + "github.com/pkg/errors" +) + +var rsaVerifyFuncs = map[jwa.SignatureAlgorithm]rsaVerifyFunc{} + +func init() { + algs := map[jwa.SignatureAlgorithm]struct { + Hash crypto.Hash + VerifyFunc func(crypto.Hash) rsaVerifyFunc + }{ + jwa.RS256: { + Hash: crypto.SHA256, + VerifyFunc: makeVerifyPKCS1v15, + }, + jwa.RS384: { + Hash: crypto.SHA384, + VerifyFunc: makeVerifyPKCS1v15, + }, + jwa.RS512: { + Hash: crypto.SHA512, + VerifyFunc: makeVerifyPKCS1v15, + }, + jwa.PS256: { + Hash: crypto.SHA256, + VerifyFunc: makeVerifyPSS, + }, + jwa.PS384: { + Hash: crypto.SHA384, + VerifyFunc: makeVerifyPSS, + }, + jwa.PS512: { + Hash: crypto.SHA512, + VerifyFunc: makeVerifyPSS, + }, + } + + for alg, item := range algs { + rsaVerifyFuncs[alg] = item.VerifyFunc(item.Hash) + } +} + +func makeVerifyPKCS1v15(hash crypto.Hash) rsaVerifyFunc { + return rsaVerifyFunc(func(payload, signature []byte, key *rsa.PublicKey) error { + h := hash.New() + h.Write(payload) + return rsa.VerifyPKCS1v15(key, hash, h.Sum(nil), signature) + }) +} + +func makeVerifyPSS(hash crypto.Hash) rsaVerifyFunc { + return rsaVerifyFunc(func(payload, signature []byte, key *rsa.PublicKey) error { + h := hash.New() + h.Write(payload) + return rsa.VerifyPSS(key, hash, h.Sum(nil), signature, nil) + }) +} + +func newRSA(alg jwa.SignatureAlgorithm) (*RSAVerifier, error) { + verifyfn, ok := rsaVerifyFuncs[alg] + if !ok { + return nil, errors.Errorf(`unsupported algorithm while trying to create RSA verifier: %s`, alg) + } + + return &RSAVerifier{ + verify: verifyfn, + }, nil +} + +// Verify checks if a JWS is valid. +func (v RSAVerifier) Verify(payload, signature []byte, key interface{}) error { + if key == nil { + return errors.New(`missing public key while verifying payload`) + } + rsaKey, ok := key.(*rsa.PublicKey) + if !ok { + return errors.Errorf(`invalid key type %T. *rsa.PublicKey is required`, key) + } + + return v.verify(payload, signature, rsaKey) +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/jwx/jws/verify/verify.go b/vendor/github.com/open-policy-agent/opa/internal/jwx/jws/verify/verify.go new file mode 100644 index 00000000..1bb3bf83 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/jwx/jws/verify/verify.go @@ -0,0 +1,57 @@ +package verify + +import ( + "crypto/ecdsa" + "crypto/rsa" + "crypto/x509" + "encoding/pem" + "fmt" + + "github.com/pkg/errors" + + "github.com/open-policy-agent/opa/internal/jwx/jwa" +) + +// New creates a new JWS verifier using the specified algorithm +// and the public key +func New(alg jwa.SignatureAlgorithm) (Verifier, error) { + switch alg { + case jwa.RS256, jwa.RS384, jwa.RS512, jwa.PS256, jwa.PS384, jwa.PS512: + return newRSA(alg) + case jwa.ES256, jwa.ES384, jwa.ES512: + return newECDSA(alg) + case jwa.HS256, jwa.HS384, jwa.HS512: + return newHMAC(alg) + default: + return nil, errors.Errorf(`unsupported signature algorithm: %s`, alg) + } +} + +// GetSigningKey returns a *rsa.PublicKey or *ecdsa.PublicKey typically encoded in PEM blocks of type "PUBLIC KEY", +// for RSA and ECDSA family of algorithms. +// For HMAC family, it return a []byte value +func GetSigningKey(key string, alg jwa.SignatureAlgorithm) (interface{}, error) { + switch alg { + case jwa.RS256, jwa.RS384, jwa.RS512, jwa.PS256, jwa.PS384, jwa.PS512, jwa.ES256, jwa.ES384, jwa.ES512: + block, _ := pem.Decode([]byte(key)) + if block == nil { + return nil, fmt.Errorf("failed to parse PEM block containing the key") + } + + pub, err := x509.ParsePKIXPublicKey(block.Bytes) + if err != nil { + return nil, err + } + + switch pub := pub.(type) { + case *rsa.PublicKey, *ecdsa.PublicKey: + return pub, nil + default: + return nil, fmt.Errorf("invalid key type %T", pub) + } + case jwa.HS256, jwa.HS384, jwa.HS512: + return []byte(key), nil + default: + return nil, errors.Errorf("unsupported signature algorithm: %s", alg) + } +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/lcss/README.md b/vendor/github.com/open-policy-agent/opa/internal/lcss/README.md new file mode 100644 index 00000000..39b8d82c --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/lcss/README.md @@ -0,0 +1,3 @@ +# Longest Common Substring + +Original source https://github.com/vmarkovtsev/go-lcss diff --git a/vendor/github.com/open-policy-agent/opa/internal/lcss/lcss.go b/vendor/github.com/open-policy-agent/opa/internal/lcss/lcss.go new file mode 100644 index 00000000..8217a345 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/lcss/lcss.go @@ -0,0 +1,197 @@ +package lcss + +import "bytes" + +// LongestCommonSubstring returns the longest substring which is present in all the given strings. +// https://en.wikipedia.org/wiki/Longest_common_substring_problem +// Not to be confused with the Longest Common Subsequence. +// Complexity: +// * time: sum of `n_i*log(n_i)` where `n_i` is the length of each string. +// * space: sum of `n_i`. +// Returns a byte slice which is never a nil. +// +// ### Algorithm. +// We build suffix arrays for each of the passed string and then follow the same procedure +// as in merge sort: pick the least suffix in the lexicographical order. It is possible +// because the suffix arrays are already sorted. +// We record the last encountered suffixes from each of the strings and measure the longest +// common prefix of those at each "merge sort" step. +// The string comparisons are optimized by maintaining the char-level prefix tree of the "heads" +// of the suffix array sequences. +func LongestCommonSubstring(strs ...[]byte) []byte { + strslen := len(strs) + if strslen == 0 { + return []byte{} + } + if strslen == 1 { + return strs[0] + } + suffixes := make([][]int, strslen) + for i, str := range strs { + suffixes[i] = qsufsort(str) + } + return lcss(strs, suffixes) +} + +func lcss(strs [][]byte, suffixes [][]int) []byte { + strslen := len(strs) + if strslen == 0 { + return []byte{} + } + if strslen == 1 { + return strs[0] + } + minstrlen := len(strs[0]) // minimum length of the strings + for _, str := range strs { + if minstrlen > len(str) { + minstrlen = len(str) + } + } + heads := make([]int, strslen) // position in each suffix array + boilerplate := make([][]byte, strslen) // existing suffixes in the tree + boiling := 0 // indicates how many distinct suffix arrays are presented in `boilerplate` + var root charNode // the character tree built on the strings from `boilerplate` + lcs := []byte{} // our function's return value, `var lcss []byte` does *not* work + for { + mini := -1 + var minSuffixStr []byte + for i, head := range heads { + if head >= len(suffixes[i]) { + // this suffix array has been scanned till the end + continue + } + suffix := strs[i][suffixes[i][head]:] + if minSuffixStr == nil { + // initialize + mini = i + minSuffixStr = suffix + } else if bytes.Compare(minSuffixStr, suffix) > 0 { + // the current suffix is the smallest in the lexicographical order + mini = i + minSuffixStr = suffix + } + } + if mini == -1 { + // all heads exhausted + break + } + if boilerplate[mini] != nil { + // if we already have a suffix from this string, replace it with the new one + root.Remove(boilerplate[mini]) + } else { + // we track the number of distinct strings which have been touched + // when `boiling` becomes strslen we can start measuring the longest common prefix + boiling++ + } + boilerplate[mini] = minSuffixStr + root.Add(minSuffixStr) + heads[mini]++ + if boiling == strslen && root.LongestCommonPrefixLength() > len(lcs) { + // all heads > 0, the current common prefix of the suffixes is the longest + lcs = root.LongestCommonPrefix() + if len(lcs) == minstrlen { + // early exit - we will never find a longer substring + break + } + } + } + return lcs +} + +// charNode builds a tree of individual characters. +// `used` is the counter for collecting garbage: those nodes which have `used`=0 are removed. +// The root charNode always remains intact apart from `children`. +// The tree supports 4 operations: +// 1. Add() a new string. +// 2. Remove() an existing string which was previously Add()-ed. +// 3. LongestCommonPrefixLength(). +// 4. LongestCommonPrefix(). +type charNode struct { + char byte + children []charNode + used int +} + +// Add includes a new string into the tree. We start from the root and +// increment `used` of all the nodes we visit. +func (cn *charNode) Add(str []byte) { + head := cn + for i, char := range str { + found := false + for j, child := range head.children { + if child.char == char { + head.children[j].used++ + head = &head.children[j] // -> child + found = true + break + } + } + if !found { + // add the missing nodes one by one + for _, char = range str[i:] { + head.children = append(head.children, charNode{char: char, children: nil, used: 1}) + head = &head.children[len(head.children)-1] + } + break + } + } +} + +// Remove excludes a node which was previously Add()-ed. +// We start from the root and decrement `used` of all the nodes we visit. +// If there is a node with `used`=0, we erase it from the parent's list of children +// and stop traversing the tree. +func (cn *charNode) Remove(str []byte) { + stop := false + head := cn + for _, char := range str { + for j, child := range head.children { + if child.char != char { + continue + } + head.children[j].used-- + var parent *charNode + head, parent = &head.children[j], head // shift to the child + if head.used == 0 { + parent.children = append(parent.children[:j], parent.children[j+1:]...) + // we can skip deleting the rest of the nodes - they have been already discarded + stop = true + } + break + } + if stop { + break + } + } +} + +// LongestCommonPrefixLength returns the length of the longest common prefix of the strings +// which are stored in the tree. We visit the children recursively starting from the root and +// stop if `used` value decreases or there is more than one child. +func (cn charNode) LongestCommonPrefixLength() int { + var result int + for head := cn; len(head.children) == 1 && head.children[0].used >= head.used; head = head.children[0] { + + result++ + } + return result +} + +// LongestCommonPrefix returns the longest common prefix of the strings +// which are stored in the tree. We compute the length by calling LongestCommonPrefixLength() +// and then record the characters which we visit along the way from the root to the last node. +func (cn charNode) LongestCommonPrefix() []byte { + result := make([]byte, cn.LongestCommonPrefixLength()) + if len(result) == 0 { + return result + } + var i int + for head := cn.children[0]; ; head = head.children[0] { + result[i] = head.char + i++ + if i == len(result) { + break + } + } + return result +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/lcss/qsufsort.go b/vendor/github.com/open-policy-agent/opa/internal/lcss/qsufsort.go new file mode 100644 index 00000000..61c51968 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/lcss/qsufsort.go @@ -0,0 +1,169 @@ +// Copyright 2011 The Go Authors. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +// This algorithm is based on "Faster Suffix Sorting" +// by N. Jesper Larsson and Kunihiko Sadakane +// paper: http://www.larsson.dogma.net/ssrev-tr.pdf +// code: http://www.larsson.dogma.net/qsufsort.c + +// This algorithm computes the suffix array sa by computing its inverse. +// Consecutive groups of suffixes in sa are labeled as sorted groups or +// unsorted groups. For a given pass of the sorter, all suffixes are ordered +// up to their first h characters, and sa is h-ordered. Suffixes in their +// final positions and unambiguously sorted in h-order are in a sorted group. +// Consecutive groups of suffixes with identical first h characters are an +// unsorted group. In each pass of the algorithm, unsorted groups are sorted +// according to the group number of their following suffix. + +// In the implementation, if sa[i] is negative, it indicates that i is +// the first element of a sorted group of length -sa[i], and can be skipped. +// An unsorted group sa[i:k] is given the group number of the index of its +// last element, k-1. The group numbers are stored in the inverse slice (inv), +// and when all groups are sorted, this slice is the inverse suffix array. + +package lcss + +import "sort" + +// qsufsort constructs the suffix array for a given string. +func qsufsort(data []byte) []int { + // initial sorting by first byte of suffix + sa := sortedByFirstByte(data) + if len(sa) < 2 { + return sa + } + // initialize the group lookup table + // this becomes the inverse of the suffix array when all groups are sorted + inv := initGroups(sa, data) + + // the index starts 1-ordered + sufSortable := &suffixSortable{sa: sa, inv: inv, h: 1} + + for sa[0] > -len(sa) { // until all suffixes are one big sorted group + // The suffixes are h-ordered, make them 2*h-ordered + pi := 0 // pi is first position of first group + sl := 0 // sl is negated length of sorted groups + for pi < len(sa) { + if s := sa[pi]; s < 0 { // if pi starts sorted group + pi -= s // skip over sorted group + sl += s // add negated length to sl + } else { // if pi starts unsorted group + if sl != 0 { + sa[pi+sl] = sl // combine sorted groups before pi + sl = 0 + } + pk := inv[s] + 1 // pk-1 is last position of unsorted group + sufSortable.sa = sa[pi:pk] + sort.Sort(sufSortable) + sufSortable.updateGroups(pi) + pi = pk // next group + } + } + if sl != 0 { // if the array ends with a sorted group + sa[pi+sl] = sl // combine sorted groups at end of sa + } + + sufSortable.h *= 2 // double sorted depth + } + + for i := range sa { // reconstruct suffix array from inverse + sa[inv[i]] = i + } + return sa +} + +func sortedByFirstByte(data []byte) []int { + // total byte counts + var count [256]int + for _, b := range data { + count[b]++ + } + // make count[b] equal index of first occurrence of b in sorted array + sum := 0 + for b := range count { + count[b], sum = sum, count[b]+sum + } + // iterate through bytes, placing index into the correct spot in sa + sa := make([]int, len(data)) + for i, b := range data { + sa[count[b]] = i + count[b]++ + } + return sa +} + +func initGroups(sa []int, data []byte) []int { + // label contiguous same-letter groups with the same group number + inv := make([]int, len(data)) + prevGroup := len(sa) - 1 + groupByte := data[sa[prevGroup]] + for i := len(sa) - 1; i >= 0; i-- { + if b := data[sa[i]]; b < groupByte { + if prevGroup == i+1 { + sa[i+1] = -1 + } + groupByte = b + prevGroup = i + } + inv[sa[i]] = prevGroup + if prevGroup == 0 { + sa[0] = -1 + } + } + // Separate out the final suffix to the start of its group. + // This is necessary to ensure the suffix "a" is before "aba" + // when using a potentially unstable sort. + lastByte := data[len(data)-1] + s := -1 + for i := range sa { + if sa[i] >= 0 { + if data[sa[i]] == lastByte && s == -1 { + s = i + } + if sa[i] == len(sa)-1 { + sa[i], sa[s] = sa[s], sa[i] + inv[sa[s]] = s + sa[s] = -1 // mark it as an isolated sorted group + break + } + } + } + return inv +} + +type suffixSortable struct { + sa []int + inv []int + h int + buf []int // common scratch space +} + +func (x *suffixSortable) Len() int { return len(x.sa) } +func (x *suffixSortable) Less(i, j int) bool { return x.inv[x.sa[i]+x.h] < x.inv[x.sa[j]+x.h] } +func (x *suffixSortable) Swap(i, j int) { x.sa[i], x.sa[j] = x.sa[j], x.sa[i] } + +func (x *suffixSortable) updateGroups(offset int) { + bounds := x.buf[0:0] + group := x.inv[x.sa[0]+x.h] + for i := 1; i < len(x.sa); i++ { + if g := x.inv[x.sa[i]+x.h]; g > group { + bounds = append(bounds, i) + group = g + } + } + bounds = append(bounds, len(x.sa)) + x.buf = bounds + + // update the group numberings after all new groups are determined + prev := 0 + for _, b := range bounds { + for i := prev; i < b; i++ { + x.inv[x.sa[i]] = offset + b - 1 + } + if b-prev == 1 { + x.sa[prev] = -1 + } + prev = b + } +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/leb128/leb128.go b/vendor/github.com/open-policy-agent/opa/internal/leb128/leb128.go new file mode 100644 index 00000000..24ddc909 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/leb128/leb128.go @@ -0,0 +1,170 @@ +// Copyright 2018 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +// Package leb128 implements LEB128 integer encoding. +package leb128 + +import ( + "io" +) + +// MustReadVarInt32 returns an int32 from r or panics. +func MustReadVarInt32(r io.Reader) int32 { + i32, err := ReadVarInt32(r) + if err != nil { + panic(err) + } + return i32 +} + +// MustReadVarInt64 returns an int64 from r or panics. +func MustReadVarInt64(r io.Reader) int64 { + i64, err := ReadVarInt64(r) + if err != nil { + panic(err) + } + return i64 +} + +// MustReadVarUint32 returns an uint32 from r or panics. +func MustReadVarUint32(r io.Reader) uint32 { + u32, err := ReadVarUint32(r) + if err != nil { + panic(err) + } + return u32 +} + +// MustReadVarUint64 returns an uint64 from r or panics. +func MustReadVarUint64(r io.Reader) uint64 { + u64, err := ReadVarUint64(r) + if err != nil { + panic(err) + } + return u64 +} + +// Copied rom http://dwarfstd.org/doc/Dwarf3.pdf. + +// ReadVarUint32 tries to read a uint32 from r. +func ReadVarUint32(r io.Reader) (uint32, error) { + u64, err := ReadVarUint64(r) + if err != nil { + return 0, err + } + return uint32(u64), nil +} + +// ReadVarUint64 tries to read a uint64 from r. +func ReadVarUint64(r io.Reader) (uint64, error) { + var result uint64 + var shift uint64 + buf := make([]byte, 1) + for { + if _, err := r.Read(buf); err != nil { + return 0, err + } + v := uint64(buf[0]) + result |= (v & 0x7F) << shift + if v&0x80 == 0 { + return result, nil + } + shift += 7 + } + +} + +// ReadVarInt32 tries to read a int32 from r. +func ReadVarInt32(r io.Reader) (int32, error) { + i64, err := ReadVarInt64(r) + if err != nil { + return 0, err + } + return int32(i64), nil +} + +// ReadVarInt64 tries to read a int64 from r. +func ReadVarInt64(r io.Reader) (int64, error) { + var result int64 + var shift uint64 + size := uint64(32) + buf := make([]byte, 1) + for { + if _, err := r.Read(buf); err != nil { + return 0, err + } + v := int64(buf[0]) + result |= (v & 0x7F) << shift + shift += 7 + if v&0x80 == 0 { + if (shift < size) && (v&0x40 != 0) { + result |= (^0 << shift) + } + return result, nil + } + } +} + +// WriteVarUint32 writes u to w. +func WriteVarUint32(w io.Writer, u uint32) error { + var b []byte + _, err := w.Write(appendUleb128(b, uint64(u))) + return err +} + +// WriteVarUint64 writes u to w. +func WriteVarUint64(w io.Writer, u uint64) error { + var b []byte + _, err := w.Write(appendUleb128(b, u)) + return err +} + +// WriteVarInt32 writes u to w. +func WriteVarInt32(w io.Writer, i int32) error { + var b []byte + _, err := w.Write(appendSleb128(b, int64(i))) + return err +} + +// WriteVarInt64 writes u to w. +func WriteVarInt64(w io.Writer, i int64) error { + var b []byte + _, err := w.Write(appendSleb128(b, i)) + return err +} + +// Copied from https://github.com/golang/go/blob/master/src/cmd/internal/dwarf/dwarf.go. + +// appendUleb128 appends v to b using DWARF's unsigned LEB128 encoding. +func appendUleb128(b []byte, v uint64) []byte { + for { + c := uint8(v & 0x7f) + v >>= 7 + if v != 0 { + c |= 0x80 + } + b = append(b, c) + if c&0x80 == 0 { + break + } + } + return b +} + +// appendSleb128 appends v to b using DWARF's signed LEB128 encoding. +func appendSleb128(b []byte, v int64) []byte { + for { + c := uint8(v & 0x7f) + s := uint8(v & 0x40) + v >>= 7 + if (v != -1 || s == 0) && (v != 0 || s != 0) { + c |= 0x80 + } + b = append(b, c) + if c&0x80 == 0 { + break + } + } + return b +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/merge/merge.go b/vendor/github.com/open-policy-agent/opa/internal/merge/merge.go new file mode 100644 index 00000000..16f39350 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/merge/merge.go @@ -0,0 +1,64 @@ +// Copyright 2017 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +// Package merge contains helpers to merge data structures +// frequently encountered in OPA. +package merge + +// InterfaceMaps returns the result of merging a and b. If a and b cannot be +// merged because of conflicting key-value pairs, ok is false. +func InterfaceMaps(a map[string]interface{}, b map[string]interface{}) (map[string]interface{}, bool) { + + if a == nil { + return b, true + } + + if hasConflicts(a, b) { + return nil, false + } + + return merge(a, b), true +} + +func merge(a, b map[string]interface{}) map[string]interface{} { + + for k := range b { + + add := b[k] + exist, ok := a[k] + if !ok { + a[k] = add + continue + } + + existObj := exist.(map[string]interface{}) + addObj := add.(map[string]interface{}) + + a[k] = merge(existObj, addObj) + } + + return a +} + +func hasConflicts(a, b map[string]interface{}) bool { + for k := range b { + + add := b[k] + exist, ok := a[k] + if !ok { + continue + } + + existObj, existOk := exist.(map[string]interface{}) + addObj, addOk := add.(map[string]interface{}) + if !existOk || !addOk { + return true + } + + if hasConflicts(existObj, addObj) { + return true + } + } + return false +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/planner/planner.go b/vendor/github.com/open-policy-agent/opa/internal/planner/planner.go new file mode 100644 index 00000000..f3835f47 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/planner/planner.go @@ -0,0 +1,2117 @@ +// Copyright 2018 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +// Package planner contains a query planner for Rego queries. +package planner + +import ( + "errors" + "fmt" + "io" + "sort" + + "github.com/open-policy-agent/opa/ast" + "github.com/open-policy-agent/opa/ast/location" + "github.com/open-policy-agent/opa/internal/debug" + "github.com/open-policy-agent/opa/internal/ir" +) + +// QuerySet represents the input to the planner. +type QuerySet struct { + Name string + Queries []ast.Body + RewrittenVars map[ast.Var]ast.Var +} + +type planiter func() error + +// Planner implements a query planner for Rego queries. +type Planner struct { + policy *ir.Policy // result of planning + queries []QuerySet // input queries to plan + modules []*ast.Module // input modules to support queries + strings map[string]int // global string constant indices + files map[string]int // global file constant indices + externs map[string]*ast.Builtin // built-in functions that are required in execution environment + decls map[string]*ast.Builtin // built-in functions that may be provided in execution environment + rules *ruletrie // rules that may be planned + funcs *funcstack // functions that have been planned + plan *ir.Plan // in-progress query plan + curr *ir.Block // in-progress query block + vars *varstack // in-scope variables + ltarget ir.LocalOrConst // target variable or constant of last planned statement + lnext ir.Local // next variable to use + loc *location.Location // location currently "being planned" + debug debug.Debug // debug information produced during planning +} + +// debugf prepends the planner location. We're passing callstack depth 2 because +// it should still log the file location of p.debugf. +func (p *Planner) debugf(format string, args ...interface{}) { + var msg string + if p.loc != nil { + msg = fmt.Sprintf("%s: "+format, append([]interface{}{p.loc}, args...)...) + } else { + msg = fmt.Sprintf(format, args...) + } + _ = p.debug.Output(2, msg) // ignore error +} + +// New returns a new Planner object. +func New() *Planner { + return &Planner{ + policy: &ir.Policy{ + Static: &ir.Static{}, + Plans: &ir.Plans{}, + Funcs: &ir.Funcs{}, + }, + strings: map[string]int{}, + files: map[string]int{}, + externs: map[string]*ast.Builtin{}, + lnext: ir.Unused, + vars: newVarstack(map[ast.Var]ir.Local{ + ast.InputRootDocument.Value.(ast.Var): ir.Input, + ast.DefaultRootDocument.Value.(ast.Var): ir.Data, + }), + rules: newRuletrie(), + funcs: newFuncstack(), + debug: debug.Discard(), + } +} + +// WithBuiltinDecls tells the planner what built-in function may be available +// inside the execution environment. +func (p *Planner) WithBuiltinDecls(decls map[string]*ast.Builtin) *Planner { + p.decls = decls + return p +} + +// WithQueries sets the query sets to generate a plan for. The rewritten collection provides +// a mapping of rewritten query vars for each query set. The planner uses rewritten variables +// but the result set key will be the original variable name. +func (p *Planner) WithQueries(queries []QuerySet) *Planner { + p.queries = queries + return p +} + +// WithModules sets the module set that contains query dependencies. +func (p *Planner) WithModules(modules []*ast.Module) *Planner { + p.modules = modules + return p +} + +// WithDebug sets where debug messages are written to. +func (p *Planner) WithDebug(sink io.Writer) *Planner { + if sink != nil { + p.debug = debug.New(sink) + } + return p +} + +// Plan returns a IR plan for the policy query. +func (p *Planner) Plan() (*ir.Policy, error) { + + if err := p.buildFunctrie(); err != nil { + return nil, err + } + + if err := p.planQueries(); err != nil { + return nil, err + } + + if err := p.planExterns(); err != nil { + return nil, err + } + + return p.policy, nil +} + +func (p *Planner) buildFunctrie() error { + + for _, module := range p.modules { + + // Create functrie node for empty packages so that extent queries return + // empty objects. For example: + // + // package x.y + // + // Query: data.x + // + // Expected result: {"y": {}} + if len(module.Rules) == 0 { + _ = p.rules.LookupOrInsert(module.Package.Path) + continue + } + + for _, rule := range module.Rules { + val := p.rules.LookupOrInsert(rule.Path()) + val.rules = append(val.rules, rule) + } + } + + return nil +} + +func (p *Planner) planRules(rules []*ast.Rule) (string, error) { + + pathRef := rules[0].Path() + path := pathRef.String() + + var pathPieces []string + for i := 1; /* skip `data` */ i < len(pathRef); i++ { + pathPieces = append(pathPieces, string(pathRef[i].Value.(ast.String))) + } + + if funcName, ok := p.funcs.Get(path); ok { + return funcName, nil + } + + // Save current state of planner. + // + // TODO(tsandall): perhaps we would be better off using stacks here or + // splitting block planner into separate struct that could be instantiated + // for rule and comprehension bodies. + pvars := p.vars + pcurr := p.curr + pltarget := p.ltarget + plnext := p.lnext + ploc := p.loc + + // Reset the variable counter for the function plan. + p.lnext = ir.Input + + // Set the location to the rule head. + p.loc = rules[0].Head.Loc() + + // Create function definition for rules. + fn := &ir.Func{ + Name: fmt.Sprintf("g%d.%s", p.funcs.gen(), path), + Params: []ir.Local{ + p.newLocal(), // input document + p.newLocal(), // data document + }, + Return: p.newLocal(), + Path: append([]string{fmt.Sprintf("g%d", p.funcs.gen())}, pathPieces...), + } + + // Initialize parameters for functions. + for i := 0; i < len(rules[0].Head.Args); i++ { + fn.Params = append(fn.Params, p.newLocal()) + } + + params := fn.Params[2:] + + // Initialize return value for partial set/object rules. Complete document + // rules assign directly to `fn.Return`. + switch rules[0].Head.DocKind() { + case ast.PartialObjectDoc: + fn.Blocks = append(fn.Blocks, p.blockWithStmt(&ir.MakeObjectStmt{Target: fn.Return})) + case ast.PartialSetDoc: + fn.Blocks = append(fn.Blocks, p.blockWithStmt(&ir.MakeSetStmt{Target: fn.Return})) + } + + // For complete document rules, allocate one local variable for output + // of the rule body + else branches. + // It is used to let ordered rules (else blocks) check if the previous + // rule body returned a value. + lresult := p.newLocal() + + // At this point the locals for the params and return value have been + // allocated. This will be the first local that can be used in each block. + lnext := p.lnext + + var defaultRule *ast.Rule + var ruleLoc *location.Location + + // Generate function blocks for rules. + for i := range rules { + + // Save location of first encountered rule for the ReturnLocalStmt below + if i == 0 { + ruleLoc = p.loc + } + + // Save default rule for the end. + if rules[i].Default { + defaultRule = rules[i] + continue + } + + // Ordered rules are nested inside an additional block so that execution + // can short-circuit. For unordered rules, blocks can be added directly + // to the function. + var blocks *[]*ir.Block + + if rules[i].Else == nil { + blocks = &fn.Blocks + } else { + stmt := &ir.BlockStmt{} + block := &ir.Block{Stmts: []ir.Stmt{stmt}} + fn.Blocks = append(fn.Blocks, block) + blocks = &stmt.Blocks + } + + var prev *ast.Rule + + // Unordered rules are treated as a special case of ordered rules. + for rule := rules[i]; rule != nil; prev, rule = rule, rule.Else { + + // Update the location for each ordered rule. + p.loc = rule.Head.Loc() + + // Setup planner for block. + p.lnext = lnext + p.vars = newVarstack(map[ast.Var]ir.Local{ + ast.InputRootDocument.Value.(ast.Var): fn.Params[0], + ast.DefaultRootDocument.Value.(ast.Var): fn.Params[1], + }) + + curr := &ir.Block{} + *blocks = append(*blocks, curr) + p.curr = curr + + if prev != nil { + // Ordered rules are handled by short circuiting execution. The + // plan will jump out to the extra block that was planned above. + p.appendStmt(&ir.IsUndefinedStmt{Source: lresult}) + } else { + // The first rule body resets the local, so it can be reused. + p.appendStmt(&ir.ResetLocalStmt{Target: lresult}) + } + + // Complete and partial rules are treated as special cases of + // functions. If there are no args, the first step is a no-op. + err := p.planFuncParams(params, rule.Head.Args, 0, func() error { + + // Run planner on the rule body. + err := p.planQuery(rule.Body, 0, func() error { + + // Run planner on the result. + switch rule.Head.DocKind() { + case ast.CompleteDoc: + return p.planTerm(rule.Head.Value, func() error { + p.appendStmt(&ir.AssignVarOnceStmt{ + Target: lresult, + Source: p.ltarget, + }) + return nil + }) + case ast.PartialSetDoc: + return p.planTerm(rule.Head.Key, func() error { + p.appendStmt(&ir.SetAddStmt{ + Set: fn.Return, + Value: p.ltarget, + }) + return nil + }) + case ast.PartialObjectDoc: + return p.planTerm(rule.Head.Key, func() error { + key := p.ltarget + return p.planTerm(rule.Head.Value, func() error { + value := p.ltarget + p.appendStmt(&ir.ObjectInsertOnceStmt{ + Object: fn.Return, + Key: key, + Value: value, + }) + return nil + }) + }) + default: + return fmt.Errorf("illegal rule kind") + } + }) + + if err != nil { + return err + } + + return nil + }) + + if err != nil { + return "", err + } + } + + // rule[i] and its else-rule(s), if present, are done + if rules[i].Head.DocKind() == ast.CompleteDoc { + end := &ir.Block{} + p.appendStmtToBlock(&ir.IsDefinedStmt{Source: lresult}, end) + p.appendStmtToBlock( + &ir.AssignVarOnceStmt{ + Target: fn.Return, + Source: lresult, + }, + end) + *blocks = append(*blocks, end) + } + } + + // Default rules execute if the return is undefined. + if defaultRule != nil { + + // Set the location for the default rule head. + p.loc = defaultRule.Head.Loc() + // NOTE(sr) for `default p = 1`, + // defaultRule.Loc() is `default`, + // defaultRule.Head.Loc() is `p = 1`. + + fn.Blocks = append(fn.Blocks, p.blockWithStmt(&ir.IsUndefinedStmt{Source: fn.Return})) + + p.curr = fn.Blocks[len(fn.Blocks)-1] + + err := p.planQuery(defaultRule.Body, 0, func() error { + p.loc = defaultRule.Head.Loc() + return p.planTerm(defaultRule.Head.Value, func() error { + p.appendStmt(&ir.AssignVarOnceStmt{ + Target: fn.Return, + Source: p.ltarget, + }) + return nil + }) + }) + + if err != nil { + return "", err + } + } + + p.loc = ruleLoc + + // All rules return a value. + fn.Blocks = append(fn.Blocks, p.blockWithStmt(&ir.ReturnLocalStmt{Source: fn.Return})) + + p.appendFunc(fn) + p.funcs.Add(path, fn.Name) + + // Restore the state of the planner. + p.lnext = plnext + p.ltarget = pltarget + p.vars = pvars + p.curr = pcurr + p.loc = ploc + + return fn.Name, nil +} + +func (p *Planner) planFuncParams(params []ir.Local, args ast.Args, idx int, iter planiter) error { + if idx >= len(args) { + return iter() + } + return p.planUnifyLocal(params[idx], args[idx], func() error { + return p.planFuncParams(params, args, idx+1, iter) + }) +} + +func (p *Planner) planQueries() error { + + for _, qs := range p.queries { + + // Initialize the plan with a block that prepares the query result. + p.plan = &ir.Plan{Name: qs.Name} + p.policy.Plans.Plans = append(p.policy.Plans.Plans, p.plan) + p.curr = &ir.Block{} + + // Build a set of variables appearing in the query and allocate strings for + // each one. The strings will be used in the result set objects. + qvs := ast.NewVarSet() + + for _, q := range qs.Queries { + vs := q.Vars(ast.VarVisitorParams{SkipRefCallHead: true, SkipClosures: true}).Diff(ast.ReservedVars) + qvs.Update(vs) + } + + lvarnames := make(map[ast.Var]ir.StringIndex, len(qvs)) + + for _, qv := range qvs.Sorted() { + qv = rewrittenVar(qs.RewrittenVars, qv) + if !qv.IsGenerated() && !qv.IsWildcard() { + lvarnames[qv] = ir.StringIndex(p.getStringConst(string(qv))) + } + } + + if len(p.curr.Stmts) > 0 { + p.appendBlock(p.curr) + } + + lnext := p.lnext + + for _, q := range qs.Queries { + p.loc = q.Loc() + p.lnext = lnext + p.vars.Push(map[ast.Var]ir.Local{}) + p.curr = &ir.Block{} + defined := false + qvs := q.Vars(ast.VarVisitorParams{SkipRefCallHead: true, SkipClosures: true}).Diff(ast.ReservedVars).Sorted() + + if err := p.planQuery(q, 0, func() error { + + // Add an object containing variable bindings into the result set. + lr := p.newLocal() + + p.appendStmt(&ir.MakeObjectStmt{ + Target: lr, + }) + + for _, qv := range qvs { + rw := rewrittenVar(qs.RewrittenVars, qv) + if !rw.IsGenerated() && !rw.IsWildcard() { + p.appendStmt(&ir.ObjectInsertStmt{ + Object: lr, + Key: lvarnames[rw], + Value: p.vars.GetOrEmpty(qv), + }) + } + } + + p.appendStmt(&ir.ResultSetAdd{ + Value: lr, + }) + + defined = true + return nil + }); err != nil { + return err + } + + p.vars.Pop() + + if defined { + p.appendBlock(p.curr) + } + } + + } + + return nil +} + +func (p *Planner) planQuery(q ast.Body, index int, iter planiter) error { + + if index >= len(q) { + return iter() + } + + old := p.loc + p.loc = q[index].Loc() + + err := p.planExpr(q[index], func() error { + return p.planQuery(q, index+1, func() error { + curr := p.loc + p.loc = old + err := iter() + p.loc = curr + return err + }) + }) + + p.loc = old + return err +} + +// TODO(tsandall): improve errors to include location information. +func (p *Planner) planExpr(e *ast.Expr, iter planiter) error { + + if e.Negated { + return p.planNot(e, iter) + } + + if len(e.With) > 0 { + return p.planWith(e, iter) + } + + if e.IsCall() { + return p.planExprCall(e, iter) + } + + return p.planExprTerm(e, iter) +} + +func (p *Planner) planNot(e *ast.Expr, iter planiter) error { + not := &ir.NotStmt{ + Block: &ir.Block{}, + } + + prev := p.curr + p.curr = not.Block + + if err := p.planExpr(e.Complement(), func() error { return nil }); err != nil { + return err + } + + p.curr = prev + p.appendStmt(not) + + return iter() +} + +func (p *Planner) planWith(e *ast.Expr, iter planiter) error { + + // Plan the values that will be applied by the with modifiers. All values + // must be defined for the overall expression to evaluate. + values := make([]*ast.Term, len(e.With)) + + for i := range e.With { + values[i] = e.With[i].Value + } + + return p.planTermSlice(values, func(locals []ir.LocalOrConst) error { + + paths := make([][]int, len(e.With)) + saveVars := ast.NewVarSet() + dataRefs := []ast.Ref{} + + for i := range e.With { + + target := e.With[i].Target.Value.(ast.Ref) + paths[i] = make([]int, len(target)-1) + + for j := 1; j < len(target); j++ { + if s, ok := target[j].Value.(ast.String); ok { + paths[i][j-1] = p.getStringConst(string(s)) + } else { + return errors.New("invalid with target") + } + } + + head := target[0].Value.(ast.Var) + saveVars.Add(head) + + if head.Equal(ast.DefaultRootDocument.Value) { + dataRefs = append(dataRefs, target) + } + } + + restore := make([][2]ir.Local, len(saveVars)) + + for i, v := range saveVars.Sorted() { + lorig := p.vars.GetOrEmpty(v) + lsave := p.newLocal() + p.appendStmt(&ir.AssignVarStmt{Source: lorig, Target: lsave}) + restore[i] = [2]ir.Local{lorig, lsave} + } + + // If any of the `with` statements targeted the data document, overwriting + // parts of the ruletrie, we shadow the existing planned functions during + // expression planning. This causes the planner to re-plan any rules that + // may be required during planning of this expression (transitively). + shadowing := p.dataRefsShadowRuletrie(dataRefs) + if shadowing { + p.funcs.Push(map[string]string{}) + for _, ref := range dataRefs { + p.rules.Push(ref) + } + } + + err := p.planWithRec(e, paths, locals, 0, func() error { + if shadowing { + p.funcs.Pop() + for i := len(dataRefs) - 1; i >= 0; i-- { + p.rules.Pop(dataRefs[i]) + } + } + + err := p.planWithUndoRec(restore, 0, func() error { + + err := iter() + + if shadowing { + p.funcs.Push(map[string]string{}) + for _, ref := range dataRefs { + p.rules.Push(ref) + } + } + return err + }) + + return err + }) + + if shadowing { + p.funcs.Pop() + for i := len(dataRefs) - 1; i >= 0; i-- { + p.rules.Pop(dataRefs[i]) + } + } + return err + + }) +} + +func (p *Planner) planWithRec(e *ast.Expr, targets [][]int, values []ir.LocalOrConst, index int, iter planiter) error { + if index >= len(e.With) { + return p.planExpr(e.NoWith(), iter) + } + + prev := p.curr + p.curr = &ir.Block{} + + err := p.planWithRec(e, targets, values, index+1, iter) + if err != nil { + return err + } + + block := p.curr + p.curr = prev + target := e.With[index].Target.Value.(ast.Ref) + head := target[0].Value.(ast.Var) + + p.appendStmt(&ir.WithStmt{ + Local: p.vars.GetOrEmpty(head), + Path: targets[index], + Value: values[index], + Block: block, + }) + + return nil +} + +func (p *Planner) planWithUndoRec(restore [][2]ir.Local, index int, iter planiter) error { + + if index >= len(restore) { + return iter() + } + + prev := p.curr + p.curr = &ir.Block{} + + if err := p.planWithUndoRec(restore, index+1, iter); err != nil { + return err + } + + block := p.curr + p.curr = prev + lorig := restore[index][0] + lsave := restore[index][1] + + p.appendStmt(&ir.WithStmt{ + Local: lorig, + Value: lsave, + Block: block, + }) + + return nil +} + +func (p *Planner) dataRefsShadowRuletrie(refs []ast.Ref) bool { + for _, ref := range refs { + if p.rules.Lookup(ref) != nil { + return true + } + } + return false +} + +func (p *Planner) planExprTerm(e *ast.Expr, iter planiter) error { + return p.planTerm(e.Terms.(*ast.Term), func() error { + p.appendStmt(&ir.NotEqualStmt{ + A: p.ltarget, + B: ir.Bool(false), + }) + return iter() + }) +} + +func (p *Planner) planExprCall(e *ast.Expr, iter planiter) error { + operator := e.Operator().String() + switch operator { + case ast.Equality.Name: + return p.planUnify(e.Operand(0), e.Operand(1), iter) + + default: + + var relation bool + var name string + var arity int + var void bool + var args []ir.LocalOrConst + + node := p.rules.Lookup(e.Operator()) + + if node != nil { + var err error + name, err = p.planRules(node.Rules()) + if err != nil { + return err + } + arity = node.Arity() + args = []ir.LocalOrConst{ + p.vars.GetOrEmpty(ast.InputRootDocument.Value.(ast.Var)), + p.vars.GetOrEmpty(ast.DefaultRootDocument.Value.(ast.Var)), + } + } else if decl, ok := p.decls[operator]; ok { + relation = decl.Relation + arity = len(decl.Decl.Args()) + void = decl.Decl.Result() == nil + name = operator + p.externs[operator] = decl + } else { + return fmt.Errorf("illegal call: unknown operator %q", operator) + } + + operands := e.Operands() + + if len(operands) < arity || len(operands) > arity+1 { + return fmt.Errorf("illegal call: wrong number of operands: got %v, want %v)", len(operands), arity) + } + + if relation { + return p.planExprCallRelation(name, arity, operands, args, iter) + } + + return p.planExprCallFunc(name, arity, void, operands, args, iter) + } +} + +func (p *Planner) planExprCallRelation(name string, arity int, operands []*ast.Term, args []ir.LocalOrConst, iter planiter) error { + + if len(operands) == arity { + return p.planCallArgs(operands, 0, args, func(args []ir.LocalOrConst) error { + p.ltarget = p.newLocal() + ltarget := p.ltarget.(ir.Local) + p.appendStmt(&ir.CallStmt{ + Func: name, + Args: args, + Result: ltarget, + }) + + lsize := p.newLocal() + + p.appendStmt(&ir.LenStmt{ + Source: ltarget, + Target: lsize, + }) + + lzero := p.newLocal() + + p.appendStmt(&ir.MakeNumberIntStmt{ + Value: 0, + Target: lzero, + }) + + p.appendStmt(&ir.NotEqualStmt{ + A: lsize, + B: lzero, + }) + + return iter() + }) + } + + return p.planCallArgs(operands[:len(operands)-1], 0, args, func(args []ir.LocalOrConst) error { + + p.ltarget = p.newLocal() + + p.appendStmt(&ir.CallStmt{ + Func: name, + Args: args, + Result: p.ltarget.(ir.Local), + }) + + return p.planScanValues(operands[len(operands)-1], func(ir.Local) error { + return iter() + }) + }) +} + +func (p *Planner) planExprCallFunc(name string, arity int, void bool, operands []*ast.Term, args []ir.LocalOrConst, iter planiter) error { + + if len(operands) == arity { + // definition: f(x) = y { ... } + // call: f(x) # result not captured + return p.planCallArgs(operands, 0, args, func(args []ir.LocalOrConst) error { + p.ltarget = p.newLocal() + ltarget := p.ltarget.(ir.Local) + p.appendStmt(&ir.CallStmt{ + Func: name, + Args: args, + Result: ltarget, + }) + + if !void { + p.appendStmt(&ir.NotEqualStmt{ + A: ltarget, + B: ir.Bool(false), + }) + } + + return iter() + }) + } + + // definition: f(x) = y { ... } + // call: f(x, 1) # caller captures result + return p.planCallArgs(operands[:len(operands)-1], 0, args, func(args []ir.LocalOrConst) error { + result := p.newLocal() + p.appendStmt(&ir.CallStmt{ + Func: name, + Args: args, + Result: result, + }) + return p.planUnifyLocal(result, operands[len(operands)-1], iter) + }) +} + +func (p *Planner) planCallArgs(terms []*ast.Term, idx int, args []ir.LocalOrConst, iter func([]ir.LocalOrConst) error) error { + if idx >= len(terms) { + return iter(args) + } + return p.planTerm(terms[idx], func() error { + args = append(args, p.ltarget) + return p.planCallArgs(terms, idx+1, args, iter) + }) +} + +func (p *Planner) planUnify(a, b *ast.Term, iter planiter) error { + + switch va := a.Value.(type) { + case ast.Null, ast.Boolean, ast.Number, ast.String, ast.Ref, ast.Set, *ast.SetComprehension, *ast.ArrayComprehension, *ast.ObjectComprehension: + return p.planTerm(a, func() error { + return p.planUnifyLocal(p.ltarget, b, iter) + }) + case ast.Var: + return p.planUnifyVar(va, b, iter) + case *ast.Array: + switch vb := b.Value.(type) { + case ast.Var: + return p.planUnifyVar(vb, a, iter) + case ast.Ref: + return p.planTerm(b, func() error { + return p.planUnifyLocalArray(p.ltarget, va, iter) + }) + case *ast.Array: + if va.Len() == vb.Len() { + return p.planUnifyArraysRec(va, vb, 0, iter) + } + return nil + } + case ast.Object: + switch vb := b.Value.(type) { + case ast.Var: + return p.planUnifyVar(vb, a, iter) + case ast.Ref: + return p.planTerm(b, func() error { + return p.planUnifyLocalObject(p.ltarget, va, iter) + }) + case ast.Object: + return p.planUnifyObjects(va, vb, iter) + } + } + + return fmt.Errorf("not implemented: unify(%v, %v)", a, b) +} + +func (p *Planner) planUnifyVar(a ast.Var, b *ast.Term, iter planiter) error { + + if la, ok := p.vars.Get(a); ok { + return p.planUnifyLocal(la, b, iter) + } + + return p.planTerm(b, func() error { + target := p.newLocal() + p.vars.Put(a, target) + p.appendStmt(&ir.AssignVarStmt{ + Source: p.ltarget, + Target: target, + }) + return iter() + }) +} + +func (p *Planner) planUnifyLocal(a ir.LocalOrConst, b *ast.Term, iter planiter) error { + switch vb := b.Value.(type) { + case ast.Null, ast.Boolean, ast.Number, ast.String, ast.Ref, ast.Set, *ast.SetComprehension, *ast.ArrayComprehension, *ast.ObjectComprehension: + return p.planTerm(b, func() error { + p.appendStmt(&ir.EqualStmt{ + A: a, + B: p.ltarget, + }) + return iter() + }) + case ast.Var: + if lv, ok := p.vars.Get(vb); ok { + p.appendStmt(&ir.EqualStmt{ + A: a, + B: lv, + }) + return iter() + } + lv := p.newLocal() + p.vars.Put(vb, lv) + p.appendStmt(&ir.AssignVarStmt{ + Source: a, + Target: lv, + }) + return iter() + case *ast.Array: + return p.planUnifyLocalArray(a, vb, iter) + case ast.Object: + return p.planUnifyLocalObject(a, vb, iter) + } + + return fmt.Errorf("not implemented: unifyLocal(%v, %v)", a, b) +} + +func (p *Planner) planUnifyLocalArray(a ir.LocalOrConst, b *ast.Array, iter planiter) error { + p.appendStmt(&ir.IsArrayStmt{ + Source: a, + }) + + blen := p.newLocal() + alen := p.newLocal() + + p.appendStmt(&ir.LenStmt{ + Source: a, + Target: alen, + }) + + p.appendStmt(&ir.MakeNumberIntStmt{ + Value: int64(b.Len()), + Target: blen, + }) + + p.appendStmt(&ir.EqualStmt{ + A: alen, + B: blen, + }) + + lkey := p.newLocal() + + p.appendStmt(&ir.MakeNumberIntStmt{ + Target: lkey, + }) + + lval := p.newLocal() + + return p.planUnifyLocalArrayRec(a, 0, b, lkey, lval, iter) +} + +func (p *Planner) planUnifyLocalArrayRec(a ir.LocalOrConst, index int, b *ast.Array, lkey, lval ir.Local, iter planiter) error { + if b.Len() == index { + return iter() + } + + p.appendStmt(&ir.AssignIntStmt{ + Value: int64(index), + Target: lkey, + }) + + p.appendStmt(&ir.DotStmt{ + Source: a.(ir.Local), + Key: lkey, + Target: lval, + }) + + return p.planUnifyLocal(lval, b.Elem(index), func() error { + return p.planUnifyLocalArrayRec(a, index+1, b, lkey, lval, iter) + }) +} + +func (p *Planner) planUnifyObjects(a, b ast.Object, iter planiter) error { + if a.Len() != b.Len() { + return nil + } + + aKeys := ast.NewSet(a.Keys()...) + bKeys := ast.NewSet(b.Keys()...) + unifyKeys := aKeys.Diff(bKeys) + + // planUnifyObjectsRec will actually set variables where possible; + // planUnifyObjectLocals only asserts equality -- it won't assign + // to any local + return p.planUnifyObjectsRec(a, b, aKeys.Intersect(bKeys).Slice(), 0, func() error { + if unifyKeys.Len() == 0 { + return iter() + } + return p.planObject(a, func() error { + la := p.ltarget + return p.planObject(b, func() error { + return p.planUnifyObjectLocals(la, p.ltarget, unifyKeys.Slice(), 0, p.newLocal(), p.newLocal(), iter) + }) + }) + }) +} + +func (p *Planner) planUnifyObjectLocals(a, b ir.LocalOrConst, keys []*ast.Term, index int, l0, l1 ir.Local, iter planiter) error { + if index == len(keys) { + return iter() + } + + return p.planTerm(keys[index], func() error { + p.appendStmt(&ir.DotStmt{ + Source: a, + Key: p.ltarget, + Target: l0, + }) + p.appendStmt(&ir.DotStmt{ + Source: b, + Key: p.ltarget, + Target: l1, + }) + p.appendStmt(&ir.EqualStmt{ + A: l0, + B: l1, + }) + + return p.planUnifyObjectLocals(a, b, keys, index+1, l0, l1, iter) + }) +} + +func (p *Planner) planUnifyLocalObject(a ir.LocalOrConst, b ast.Object, iter planiter) error { + p.appendStmt(&ir.IsObjectStmt{ + Source: a, + }) + + blen := p.newLocal() + alen := p.newLocal() + + p.appendStmt(&ir.LenStmt{ + Source: a, + Target: alen, + }) + + p.appendStmt(&ir.MakeNumberIntStmt{ + Value: int64(b.Len()), + Target: blen, + }) + + p.appendStmt(&ir.EqualStmt{ + A: alen, + B: blen, + }) + + lval := p.newLocal() + bkeys := b.Keys() + + return p.planUnifyLocalObjectRec(a, 0, bkeys, b, lval, iter) +} + +func (p *Planner) planUnifyLocalObjectRec(a ir.LocalOrConst, index int, keys []*ast.Term, b ast.Object, lval ir.Local, iter planiter) error { + + if index == len(keys) { + return iter() + } + + return p.planTerm(keys[index], func() error { + p.appendStmt(&ir.DotStmt{ + Source: a, + Key: p.ltarget, + Target: lval, + }) + return p.planUnifyLocal(lval, b.Get(keys[index]), func() error { + return p.planUnifyLocalObjectRec(a, index+1, keys, b, lval, iter) + }) + }) +} + +func (p *Planner) planUnifyArraysRec(a, b *ast.Array, index int, iter planiter) error { + if index == a.Len() { + return iter() + } + return p.planUnify(a.Elem(index), b.Elem(index), func() error { + return p.planUnifyArraysRec(a, b, index+1, iter) + }) +} + +func (p *Planner) planUnifyObjectsRec(a, b ast.Object, keys []*ast.Term, index int, iter planiter) error { + if index == len(keys) { + return iter() + } + + aval := a.Get(keys[index]) + bval := b.Get(keys[index]) + if aval == nil || bval == nil { + return nil + } + + return p.planUnify(aval, bval, func() error { + return p.planUnifyObjectsRec(a, b, keys, index+1, iter) + }) +} + +func (p *Planner) planTerm(t *ast.Term, iter planiter) error { + + switch v := t.Value.(type) { + case ast.Null: + return p.planNull(v, iter) + case ast.Boolean: + return p.planBoolean(v, iter) + case ast.Number: + return p.planNumber(v, iter) + case ast.String: + return p.planString(v, iter) + case ast.Var: + return p.planVar(v, iter) + case ast.Ref: + return p.planRef(v, iter) + case *ast.Array: + return p.planArray(v, iter) + case ast.Object: + return p.planObject(v, iter) + case ast.Set: + return p.planSet(v, iter) + case *ast.SetComprehension: + p.loc = t.Loc() + return p.planSetComprehension(v, iter) + case *ast.ArrayComprehension: + p.loc = t.Loc() + return p.planArrayComprehension(v, iter) + case *ast.ObjectComprehension: + p.loc = t.Loc() + return p.planObjectComprehension(v, iter) + default: + return fmt.Errorf("%v term not implemented", ast.TypeName(v)) + } +} + +func (p *Planner) planNull(null ast.Null, iter planiter) error { + + target := p.newLocal() + + p.appendStmt(&ir.MakeNullStmt{ + Target: target, + }) + + p.ltarget = target + + return iter() +} + +func (p *Planner) planBoolean(b ast.Boolean, iter planiter) error { + + p.ltarget = ir.Bool(b) + return iter() +} + +func (p *Planner) planNumber(num ast.Number, iter planiter) error { + + index := p.getStringConst(string(num)) + target := p.newLocal() + + p.appendStmt(&ir.MakeNumberRefStmt{ + Index: index, + Target: target, + }) + + p.ltarget = target + return iter() +} + +func (p *Planner) planString(str ast.String, iter planiter) error { + + p.ltarget = ir.StringIndex(p.getStringConst(string(str))) + + return iter() +} + +func (p *Planner) planVar(v ast.Var, iter planiter) error { + p.ltarget = p.vars.GetOrElse(v, func() ir.Local { + return p.newLocal() + }) + return iter() +} + +func (p *Planner) planArray(arr *ast.Array, iter planiter) error { + + larr := p.newLocal() + + p.appendStmt(&ir.MakeArrayStmt{ + Capacity: int32(arr.Len()), + Target: larr, + }) + + return p.planArrayRec(arr, 0, larr, iter) +} + +func (p *Planner) planArrayRec(arr *ast.Array, index int, larr ir.Local, iter planiter) error { + if index == arr.Len() { + p.ltarget = larr + return iter() + } + + return p.planTerm(arr.Elem(index), func() error { + + p.appendStmt(&ir.ArrayAppendStmt{ + Value: p.ltarget, + Array: larr, + }) + + return p.planArrayRec(arr, index+1, larr, iter) + }) +} + +func (p *Planner) planObject(obj ast.Object, iter planiter) error { + + lobj := p.newLocal() + + p.appendStmt(&ir.MakeObjectStmt{ + Target: lobj, + }) + + return p.planObjectRec(obj, 0, obj.Keys(), lobj, iter) +} + +func (p *Planner) planObjectRec(obj ast.Object, index int, keys []*ast.Term, lobj ir.Local, iter planiter) error { + if index == len(keys) { + p.ltarget = lobj + return iter() + } + + return p.planTerm(keys[index], func() error { + lkey := p.ltarget + + return p.planTerm(obj.Get(keys[index]), func() error { + lval := p.ltarget + p.appendStmt(&ir.ObjectInsertStmt{ + Key: lkey, + Value: lval, + Object: lobj, + }) + + return p.planObjectRec(obj, index+1, keys, lobj, iter) + }) + }) +} + +func (p *Planner) planSet(set ast.Set, iter planiter) error { + lset := p.newLocal() + + p.appendStmt(&ir.MakeSetStmt{ + Target: lset, + }) + + return p.planSetRec(set, 0, set.Slice(), lset, iter) +} + +func (p *Planner) planSetRec(set ast.Set, index int, elems []*ast.Term, lset ir.Local, iter planiter) error { + if index == len(elems) { + p.ltarget = lset + return iter() + } + + return p.planTerm(elems[index], func() error { + p.appendStmt(&ir.SetAddStmt{ + Value: p.ltarget, + Set: lset, + }) + return p.planSetRec(set, index+1, elems, lset, iter) + }) +} + +func (p *Planner) planSetComprehension(sc *ast.SetComprehension, iter planiter) error { + + lset := p.newLocal() + + p.appendStmt(&ir.MakeSetStmt{ + Target: lset, + }) + + return p.planComprehension(sc.Body, func() error { + return p.planTerm(sc.Term, func() error { + p.appendStmt(&ir.SetAddStmt{ + Value: p.ltarget, + Set: lset, + }) + return nil + }) + }, lset, iter) +} + +func (p *Planner) planArrayComprehension(ac *ast.ArrayComprehension, iter planiter) error { + + larr := p.newLocal() + + p.appendStmt(&ir.MakeArrayStmt{ + Target: larr, + }) + + return p.planComprehension(ac.Body, func() error { + return p.planTerm(ac.Term, func() error { + p.appendStmt(&ir.ArrayAppendStmt{ + Value: p.ltarget, + Array: larr, + }) + return nil + }) + }, larr, iter) +} + +func (p *Planner) planObjectComprehension(oc *ast.ObjectComprehension, iter planiter) error { + + lobj := p.newLocal() + + p.appendStmt(&ir.MakeObjectStmt{ + Target: lobj, + }) + return p.planComprehension(oc.Body, func() error { + return p.planTerm(oc.Key, func() error { + lkey := p.ltarget + return p.planTerm(oc.Value, func() error { + p.appendStmt(&ir.ObjectInsertOnceStmt{ + Key: lkey, + Value: p.ltarget, + Object: lobj, + }) + return nil + }) + }) + }, lobj, iter) +} + +func (p *Planner) planComprehension(body ast.Body, closureIter planiter, target ir.Local, iter planiter) error { + + // Variables that have been introduced in this comprehension have + // no effect on other parts of the policy, so they'll be dropped + // below. + p.vars.Push(map[ast.Var]ir.Local{}) + prev := p.curr + p.curr = &ir.Block{} + ploc := p.loc + + if err := p.planQuery(body, 0, func() error { + return closureIter() + }); err != nil { + return err + } + + block := p.curr + p.curr = prev + p.loc = ploc + p.vars.Pop() + + p.appendStmt(&ir.BlockStmt{ + Blocks: []*ir.Block{ + block, + }, + }) + + p.ltarget = target + return iter() +} + +func (p *Planner) planRef(ref ast.Ref, iter planiter) error { + + head, ok := ref[0].Value.(ast.Var) + if !ok { + return fmt.Errorf("illegal ref: non-var head") + } + + if head.Compare(ast.DefaultRootDocument.Value) == 0 { + virtual := p.rules.Get(ref[0].Value) + base := &baseptr{local: p.vars.GetOrEmpty(ast.DefaultRootDocument.Value.(ast.Var))} + return p.planRefData(virtual, base, ref, 1, iter) + } + + if ref.Equal(ast.InputRootRef) { + p.appendStmt(&ir.IsDefinedStmt{ + Source: p.vars.GetOrEmpty(ast.InputRootDocument.Value.(ast.Var)), + }) + } + + p.ltarget, ok = p.vars.Get(head) + if !ok { + return fmt.Errorf("illegal ref: unsafe head") + } + + return p.planRefRec(ref, 1, iter) +} + +func (p *Planner) planRefRec(ref ast.Ref, index int, iter planiter) error { + + if len(ref) == index { + return iter() + } + + if !p.unseenVars(ref[index]) { + return p.planDot(ref[index], func() error { + return p.planRefRec(ref, index+1, iter) + }) + } + + return p.planScan(ref[index], func(ir.Local) error { + return p.planRefRec(ref, index+1, iter) + }) +} + +type baseptr struct { + local ir.Local + path ast.Ref +} + +// planRefData implements the virtual document model by generating the value of +// the ref parameter and invoking the iterator with the planner target set to +// the virtual document and all variables in the reference assigned. +func (p *Planner) planRefData(virtual *ruletrie, base *baseptr, ref ast.Ref, index int, iter planiter) error { + + // Early-exit if the end of the reference has been reached. In this case the + // plan has to materialize the full extent of the referenced value. + if index >= len(ref) { + return p.planRefDataExtent(virtual, base, iter) + } + + // On the first iteration, we check if this can be optimized using a + // CallDynamicStatement + // NOTE(sr): we do it on the first index because later on, the recursion + // on subtrees of virtual already lost parts of the path we've taken. + if index == 1 && virtual != nil { + rulesets, path, index, optimize := p.optimizeLookup(virtual, ref) + if optimize { + // If there are no rulesets in a situation that otherwise would + // allow for a call_indirect optimization, then there's nothing + // to do for this ref, except scanning the base document. + if len(rulesets) == 0 { + return p.planRefData(nil, base, ref, 1, iter) // ignore index returned by optimizeLookup + } + // plan rules + for _, rules := range rulesets { + if _, err := p.planRules(rules); err != nil { + return err + } + } + + // We're planning a structure like this: + // + // block a + // block b + // block c1 + // opa_mapping_lookup || br c1 + // call_indirect || br a + // br b + // end + // block c2 + // dot i || br c2 + // dot i+1 || br c2 + // br b + // end + // br a + // end + // dot i+2 || br a + // dot i+3 || br a + // end + // + // We have to do it like this because the dot IR stmts + // are compiled to `br 0`, the innermost block, if they + // fail. + // The "c2" block will construct the reference from `data` + // only, in case the mapping lookup doesn't yield a func + // to call_dynamic. + + ltarget := p.newLocal() + p.ltarget = ltarget + prev := p.curr + + callDynBlock := &ir.Block{} // "c1" in the sketch + p.curr = callDynBlock + p.appendStmt(&ir.CallDynamicStmt{ + Args: []ir.Local{ + p.vars.GetOrEmpty(ast.InputRootDocument.Value.(ast.Var)), + p.vars.GetOrEmpty(ast.DefaultRootDocument.Value.(ast.Var)), + }, + Path: path, + Result: ltarget, + }) + p.appendStmt(&ir.BreakStmt{Index: 1}) + + dotBlock := &ir.Block{} // "c2" in the sketch above + p.curr = dotBlock + p.ltarget = p.vars.GetOrEmpty(ast.DefaultRootDocument.Value.(ast.Var)) + + return p.planRefRec(ref[:index+1], 1, func() error { + p.appendStmt(&ir.AssignVarStmt{ + Source: p.ltarget.(ir.Local), + Target: ltarget, + }) + p.appendStmt(&ir.BreakStmt{Index: 1}) + p.ltarget = ltarget + + outerBlock := &ir.Block{Stmts: []ir.Stmt{ + &ir.BlockStmt{Blocks: []*ir.Block{ + { // block "b" in the sketch above + Stmts: []ir.Stmt{ + &ir.BlockStmt{Blocks: []*ir.Block{callDynBlock, dotBlock}}, + &ir.BreakStmt{Index: 1}}, + }, + }}, + }} + p.curr = outerBlock + return p.planRefRec(ref, index+1, func() error { // rest of the ref + p.curr = prev + p.appendStmt(&ir.BlockStmt{Blocks: []*ir.Block{outerBlock}}) + return iter() + }) + }) + } + } + + // If the reference operand is ground then either continue to the next + // operand or invoke the function for the rule referred to by this operand. + if ref[index].IsGround() { + + var vchild *ruletrie + + if virtual != nil { + vchild = virtual.Get(ref[index].Value) + } + + rules := vchild.Rules() + + if len(rules) > 0 { + p.ltarget = p.newLocal() + + funcName, err := p.planRules(rules) + if err != nil { + return err + } + + p.appendStmt(&ir.CallStmt{ + Func: funcName, + Args: []ir.LocalOrConst{ + p.vars.GetOrEmpty(ast.InputRootDocument.Value.(ast.Var)), + p.vars.GetOrEmpty(ast.DefaultRootDocument.Value.(ast.Var)), + }, + Result: p.ltarget.(ir.Local), + }) + + return p.planRefRec(ref, index+1, iter) + } + + bchild := *base + bchild.path = append(bchild.path, ref[index]) + + return p.planRefData(vchild, &bchild, ref, index+1, iter) + } + + exclude := ast.NewSet() + + // The planner does not support dynamic dispatch so generate blocks to + // evaluate each of the rulesets on the child nodes. + if virtual != nil { + + stmt := &ir.BlockStmt{} + + for _, child := range virtual.Children() { + + block := &ir.Block{} + prev := p.curr + p.curr = block + key := ast.NewTerm(child) + exclude.Add(key) + + // Assignments in each block due to local unification must be undone + // so create a new frame that will be popped after this key is + // processed. + p.vars.Push(map[ast.Var]ir.Local{}) + + if err := p.planTerm(key, func() error { + return p.planUnifyLocal(p.ltarget, ref[index], func() error { + // Create a copy of the reference with this operand plugged. + // This will result in evaluation of the rulesets on the + // child node. + cpy := ref.Copy() + cpy[index] = key + return p.planRefData(virtual, base, cpy, index, iter) + }) + }); err != nil { + return err + } + + p.vars.Pop() + p.curr = prev + stmt.Blocks = append(stmt.Blocks, block) + } + + p.appendStmt(stmt) + } + + // If the virtual tree was enumerated then we do not want to enumerate base + // trees that are rooted at the same key as any of the virtual sub trees. To + // prevent this we build a set of keys that are to be excluded and check + // below during the base scan. + var lexclude *ir.Local + + if exclude.Len() > 0 { + if err := p.planSet(exclude, func() error { + v := p.ltarget.(ir.Local) + lexclude = &v + return nil + }); err != nil { + return err + } + + // Perform a scan of the base documents starting from the location referred + // to by the 'path' data pointer. Use the `lexclude` set to avoid revisiting + // sub trees. + p.ltarget = base.local + return p.planRefRec(base.path, 0, func() error { + return p.planScan(ref[index], func(lkey ir.Local) error { + if lexclude != nil { + lignore := p.newLocal() + p.appendStmt(&ir.NotStmt{ + Block: p.blockWithStmt(&ir.DotStmt{ + Source: *lexclude, + Key: lkey, + Target: lignore, + })}) + } + + // Assume that virtual sub trees have been visited already so + // recurse without the virtual node. + return p.planRefData(nil, &baseptr{local: p.ltarget.(ir.Local)}, ref, index+1, iter) + }) + }) + } + + // There is nothing to exclude, so we do the same thing done above, but + // use planRefRec to avoid the scan if ref[index] is ground or seen. + p.ltarget = base.local + base.path = append(base.path, ref[index]) + return p.planRefRec(base.path, 0, func() error { + return p.planRefData(nil, &baseptr{local: p.ltarget.(ir.Local)}, ref, index+1, iter) + }) +} + +// planRefDataExtent generates the full extent (combined) of the base and +// virtual nodes and then invokes the iterator with the planner target set to +// the full extent. +func (p *Planner) planRefDataExtent(virtual *ruletrie, base *baseptr, iter planiter) error { + + vtarget := p.newLocal() + + // Generate the virtual document out of rules contained under the virtual + // node (recursively). This document will _ONLY_ contain values generated by + // rules. No base document values will be included. + if virtual != nil { + + p.appendStmt(&ir.MakeObjectStmt{ + Target: vtarget, + }) + + for _, key := range virtual.Children() { + child := virtual.Get(key) + + // Skip functions. + if child.Arity() > 0 { + continue + } + + lkey := ir.StringIndex(p.getStringConst(string(key.(ast.String)))) + + rules := child.Rules() + + // Build object hierarchy depth-first. + if len(rules) == 0 { + err := p.planRefDataExtent(child, nil, func() error { + p.appendStmt(&ir.ObjectInsertStmt{ + Object: vtarget, + Key: lkey, + Value: p.ltarget, + }) + return nil + }) + if err != nil { + return err + } + continue + } + + // Generate virtual document for leaf. + lvalue := p.newLocal() + + funcName, err := p.planRules(rules) + if err != nil { + return err + } + + // Add leaf to object if defined. + b := &ir.Block{} + p.appendStmtToBlock(&ir.CallStmt{ + Func: funcName, + Args: []ir.LocalOrConst{ + p.vars.GetOrEmpty(ast.InputRootDocument.Value.(ast.Var)), + p.vars.GetOrEmpty(ast.DefaultRootDocument.Value.(ast.Var)), + }, + Result: lvalue, + }, b) + p.appendStmtToBlock(&ir.ObjectInsertStmt{ + Object: vtarget, + Key: lkey, + Value: lvalue, + }, b) + p.appendStmt(&ir.BlockStmt{Blocks: []*ir.Block{b}}) + } + + // At this point vtarget refers to the full extent of the virtual + // document at ref. If the base pointer is unset, no further processing + // is required. + if base == nil { + p.ltarget = vtarget + return iter() + } + } + + // Obtain the base document value and merge (recursively) with the virtual + // document value above if needed. + prev := p.curr + p.curr = &ir.Block{} + p.ltarget = base.local + target := p.newLocal() + + err := p.planRefRec(base.path, 0, func() error { + + if virtual == nil { + target = p.ltarget.(ir.Local) + } else { + stmt := &ir.ObjectMergeStmt{ + A: p.ltarget.(ir.Local), + B: vtarget, + Target: target, + } + p.appendStmt(stmt) + } + + p.appendStmt(&ir.BreakStmt{Index: 1}) + return nil + }) + + if err != nil { + return err + } + + inner := p.curr + + // Fallback to virtual document value if base document is undefined. + // Otherwise, this block is undefined. + p.curr = &ir.Block{} + p.appendStmt(&ir.BlockStmt{Blocks: []*ir.Block{inner}}) + + if virtual != nil { + p.appendStmt(&ir.AssignVarStmt{ + Source: vtarget, + Target: target, + }) + } else { + p.appendStmt(&ir.BreakStmt{Index: 1}) + } + + outer := p.curr + p.curr = prev + p.appendStmt(&ir.BlockStmt{Blocks: []*ir.Block{outer}}) + + // At this point, target refers to either the full extent of the base and + // virtual documents at ref or just the base document at ref. + p.ltarget = target + + return iter() +} + +func (p *Planner) planDot(key *ast.Term, iter planiter) error { + + source := p.ltarget + + return p.planTerm(key, func() error { + + target := p.newLocal() + + p.appendStmt(&ir.DotStmt{ + Source: source, + Key: p.ltarget, + Target: target, + }) + + p.ltarget = target + + return iter() + }) +} + +type scaniter func(ir.Local) error + +func (p *Planner) planScan(key *ast.Term, iter scaniter) error { + + scan := &ir.ScanStmt{ + Source: p.ltarget.(ir.Local), + Key: p.newLocal(), + Value: p.newLocal(), + Block: &ir.Block{}, + } + + prev := p.curr + p.curr = scan.Block + + if err := p.planUnifyLocal(scan.Key, key, func() error { + p.ltarget = scan.Value + return iter(scan.Key) + }); err != nil { + return err + } + + p.curr = prev + p.appendStmt(scan) + + return nil + +} + +func (p *Planner) planScanValues(val *ast.Term, iter scaniter) error { + + scan := &ir.ScanStmt{ + Source: p.ltarget.(ir.Local), + Key: p.newLocal(), + Value: p.newLocal(), + Block: &ir.Block{}, + } + + prev := p.curr + p.curr = scan.Block + + if err := p.planUnifyLocal(scan.Value, val, func() error { + p.ltarget = scan.Value + return iter(scan.Value) + }); err != nil { + return err + } + + p.curr = prev + p.appendStmt(scan) + + return nil +} + +type termsliceiter func([]ir.LocalOrConst) error + +func (p *Planner) planTermSlice(terms []*ast.Term, iter termsliceiter) error { + return p.planTermSliceRec(terms, make([]ir.LocalOrConst, len(terms)), 0, iter) +} + +func (p *Planner) planTermSliceRec(terms []*ast.Term, locals []ir.LocalOrConst, index int, iter termsliceiter) error { + if index >= len(terms) { + return iter(locals) + } + + return p.planTerm(terms[index], func() error { + locals[index] = p.ltarget + return p.planTermSliceRec(terms, locals, index+1, iter) + }) +} + +func (p *Planner) planExterns() error { + + p.policy.Static.BuiltinFuncs = make([]*ir.BuiltinFunc, 0, len(p.externs)) + + for name, decl := range p.externs { + p.policy.Static.BuiltinFuncs = append(p.policy.Static.BuiltinFuncs, &ir.BuiltinFunc{Name: name, Decl: decl.Decl}) + } + + sort.Slice(p.policy.Static.BuiltinFuncs, func(i, j int) bool { + return p.policy.Static.BuiltinFuncs[i].Name < p.policy.Static.BuiltinFuncs[j].Name + }) + + return nil +} + +func (p *Planner) getStringConst(s string) int { + index, ok := p.strings[s] + if !ok { + index = len(p.policy.Static.Strings) + p.policy.Static.Strings = append(p.policy.Static.Strings, &ir.StringConst{ + Value: s, + }) + p.strings[s] = index + } + return index +} + +func (p *Planner) getFileConst(s string) int { + index, ok := p.files[s] + if !ok { + index = len(p.policy.Static.Files) + p.policy.Static.Files = append(p.policy.Static.Files, &ir.StringConst{ + Value: s, + }) + p.files[s] = index + } + return index +} + +func (p *Planner) appendStmt(s ir.Stmt) { + p.appendStmtToBlock(s, p.curr) +} + +func (p *Planner) appendStmtToBlock(s ir.Stmt, b *ir.Block) { + if p.loc != nil { + str := p.loc.File + if str == "" { + str = `` + } + s.SetLocation(p.getFileConst(str), p.loc.Row, p.loc.Col, str, string(p.loc.Text)) + } + b.Stmts = append(b.Stmts, s) +} + +func (p *Planner) blockWithStmt(s ir.Stmt) *ir.Block { + b := &ir.Block{} + p.appendStmtToBlock(s, b) + return b +} + +func (p *Planner) appendBlock(b *ir.Block) { + p.plan.Blocks = append(p.plan.Blocks, b) +} + +func (p *Planner) appendFunc(f *ir.Func) { + p.policy.Funcs.Funcs = append(p.policy.Funcs.Funcs, f) +} + +func (p *Planner) newLocal() ir.Local { + x := p.lnext + p.lnext++ + return x +} + +func rewrittenVar(vars map[ast.Var]ast.Var, k ast.Var) ast.Var { + rw, ok := vars[k] + if !ok { + return k + } + return rw +} + +// optimizeLookup returns a set of rulesets and required statements planning +// the locals (strings) needed with the used local variables, and the index +// into ref's parth that is still to be planned; if the passed ref's vars +// allow for optimization using CallDynamicStmt. +// +// It's possible if all of these conditions hold: +// - all vars in ref have been seen +// - all ground terms (strings) match some child key on their respective +// layer of the ruletrie +// - there are no child trees left (only rulesets) if we're done checking +// ref +// +// The last condition is necessary because we don't deal with _which key a +// var actually matched_ -- so we don't know which subtree to evaluate +// with the results. +func (p *Planner) optimizeLookup(t *ruletrie, ref ast.Ref) ([][]*ast.Rule, []ir.LocalOrConst, int, bool) { + dont := func() ([][]*ast.Rule, []ir.LocalOrConst, int, bool) { + return nil, nil, 0, false + } + if t == nil { + p.debugf("no optimization of %s: trie is nil", ref) + return dont() + } + + nodes := []*ruletrie{t} + opt := false + var index int + + // ref[0] is data, ignore + for i := 1; i < len(ref); i++ { + index = i + r := ref[i] + var nextNodes []*ruletrie + + switch r := r.Value.(type) { + case ast.Var: + // check if it's been "seen" before + _, ok := p.vars.Get(r) + if !ok { + p.debugf("no optimization of %s: ref[%d] is unseen var: %v", ref, i, r) + return dont() + } + opt = true + // take all children, they might match + for _, node := range nodes { + for _, child := range node.Children() { + if node := node.Get(child); node != nil { + nextNodes = append(nextNodes, node) + } + } + } + case ast.String: + // take matching children + for _, node := range nodes { + if node := node.Get(r); node != nil { + nextNodes = append(nextNodes, node) + } + } + default: + p.debugf("no optimization of %s: ref[%d] is type %T", ref, i, r) // TODO(sr): think more about this + return dont() + } + + nodes = nextNodes + + // if all nodes have 0 children, abort ref check and optimize + all := true + for _, node := range nodes { + all = all && len(node.Children()) == 0 + } + if all { + p.debugf("ref %s: all nodes have 0 children, break", ref[0:index+1]) + break + } + } + + var res [][]*ast.Rule + + // if there hasn't been any var, we're not making things better by + // introducing CallDynamicStmt + if !opt { + p.debugf("no optimization of %s: no vars seen before trie descend encountered no children", ref) + return dont() + } + + for _, node := range nodes { + // we're done with ref, check if there's only ruleset leaves; collect rules + if index == len(ref)-1 { + if len(node.Children()) > 0 { + p.debugf("no optimization of %s: unbalanced ruletrie", ref) + return dont() + } + } + if rules := node.Rules(); len(rules) > 0 { + res = append(res, rules) + } + } + if len(res) == 0 { + p.debugf("ref %s: nothing to plan, no rule leaves", ref[0:index+1]) + return nil, nil, index, true + } + + var path []ir.LocalOrConst + + // plan generation + path = append(path, ir.StringIndex(p.getStringConst(fmt.Sprintf("g%d", p.funcs.gen())))) + + for i := 1; i <= index; i++ { + switch r := ref[i].Value.(type) { + case ast.Var: + lv, ok := p.vars.Get(r) + if !ok { + p.debugf("no optimization of %s: ref[%d] not a seen var: %v", ref, i, ref[i]) + return dont() + } + path = append(path, lv) + case ast.String: + path = append(path, ir.StringIndex(p.getStringConst(string(r)))) + } + } + + return res, path, index, true +} + +func (p *Planner) unseenVars(t *ast.Term) bool { + unseen := false // any var unseen? + ast.WalkVars(t, func(v ast.Var) bool { + if !unseen { + _, exists := p.vars.Get(v) + if !exists { + unseen = true + } + } + return unseen + }) + return unseen +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/planner/rules.go b/vendor/github.com/open-policy-agent/opa/internal/planner/rules.go new file mode 100644 index 00000000..3abf3e39 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/planner/rules.go @@ -0,0 +1,168 @@ +package planner + +import ( + "sort" + + "github.com/open-policy-agent/opa/ast" +) + +// funcstack implements a simple map structure used to keep track of virtual +// document => planned function names. The structure supports Push and Pop +// operations so that the planner can shadow planned functions when 'with' +// statements are found. +// The "gen" numbers indicate the "generations"; whenever a 'with' statement +// is planned (a new map is `Push()`ed), it will jump to a previously unused +// number. +type funcstack struct { + stack []taggedPairs + next int +} + +type taggedPairs struct { + pairs map[string]string + gen int +} + +func newFuncstack() *funcstack { + return &funcstack{ + stack: []taggedPairs{{pairs: map[string]string{}, gen: 0}}, + next: 1} +} + +func (p funcstack) last() taggedPairs { + return p.stack[len(p.stack)-1] +} + +func (p funcstack) Add(key, value string) { + p.last().pairs[key] = value +} + +func (p funcstack) Get(key string) (string, bool) { + value, ok := p.last().pairs[key] + return value, ok +} + +func (p *funcstack) Push(funcs map[string]string) { + p.stack = append(p.stack, taggedPairs{pairs: funcs, gen: p.next}) + p.next++ +} + +func (p *funcstack) Pop() map[string]string { + last := p.last() + p.stack = p.stack[:len(p.stack)-1] + return last.pairs +} + +func (p funcstack) gen() int { + return p.last().gen +} + +// ruletrie implements a simple trie structure for organizing rules that may be +// planned. The trie nodes are keyed by the rule path. The ruletrie supports +// Push and Pop operations that allow the planner to shadow subtrees when 'with' +// statements are found. +type ruletrie struct { + children map[ast.Value][]*ruletrie + rules []*ast.Rule +} + +func newRuletrie() *ruletrie { + return &ruletrie{ + children: map[ast.Value][]*ruletrie{}, + } +} + +func (t *ruletrie) Arity() int { + rules := t.Rules() + if len(rules) > 0 { + return len(rules[0].Head.Args) + } + return 0 +} + +func (t *ruletrie) Rules() []*ast.Rule { + if t != nil { + return t.rules + } + return nil +} + +func (t *ruletrie) Push(key ast.Ref) { + node := t + for i := 0; i < len(key)-1; i++ { + node = node.Get(key[i].Value) + if node == nil { + return + } + } + elem := key[len(key)-1] + node.children[elem.Value] = append(node.children[elem.Value], nil) +} + +func (t *ruletrie) Pop(key ast.Ref) { + node := t + for i := 0; i < len(key)-1; i++ { + node = node.Get(key[i].Value) + if node == nil { + return + } + } + elem := key[len(key)-1] + sl := node.children[elem.Value] + node.children[elem.Value] = sl[:len(sl)-1] +} + +func (t *ruletrie) Insert(key ast.Ref) *ruletrie { + node := t + for _, elem := range key { + child := node.Get(elem.Value) + if child == nil { + child = newRuletrie() + node.children[elem.Value] = append(node.children[elem.Value], child) + } + node = child + } + return node +} + +func (t *ruletrie) Lookup(key ast.Ref) *ruletrie { + node := t + for _, elem := range key { + node = node.Get(elem.Value) + if node == nil { + return nil + } + } + return node +} + +func (t *ruletrie) LookupOrInsert(key ast.Ref) *ruletrie { + if val := t.Lookup(key); val != nil { + return val + } + return t.Insert(key) +} + +func (t *ruletrie) Children() []ast.Value { + sorted := make([]ast.Value, 0, len(t.children)) + for key := range t.children { + if t.Get(key) != nil { + sorted = append(sorted, key) + } + } + sort.Slice(sorted, func(i, j int) bool { + return sorted[i].Compare(sorted[j]) < 0 + }) + return sorted +} + +func (t *ruletrie) Get(k ast.Value) *ruletrie { + if t == nil { + return nil + } + nodes := t.children[k] + if len(nodes) == 0 { + return nil + } + return nodes[len(nodes)-1] +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/planner/varstack.go b/vendor/github.com/open-policy-agent/opa/internal/planner/varstack.go new file mode 100644 index 00000000..7dbf8f43 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/planner/varstack.go @@ -0,0 +1,58 @@ +// Copyright 2019 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package planner + +import ( + "github.com/open-policy-agent/opa/ast" + "github.com/open-policy-agent/opa/internal/ir" +) + +type varstack []map[ast.Var]ir.Local + +func newVarstack(frames ...map[ast.Var]ir.Local) *varstack { + vs := &varstack{} + for _, f := range frames { + vs.Push(f) + } + return vs +} + +func (vs varstack) GetOrElse(k ast.Var, orElse func() ir.Local) ir.Local { + l, ok := vs.Get(k) + if !ok { + l = orElse() + vs.Put(k, l) + } + return l +} + +func (vs varstack) GetOrEmpty(k ast.Var) ir.Local { + l, _ := vs.Get(k) + return l +} + +func (vs varstack) Get(k ast.Var) (ir.Local, bool) { + for i := len(vs) - 1; i >= 0; i-- { + if l, ok := vs[i][k]; ok { + return l, true + } + } + return 0, false +} + +func (vs varstack) Put(k ast.Var, v ir.Local) { + vs[len(vs)-1][k] = v +} + +func (vs *varstack) Push(frame map[ast.Var]ir.Local) { + *vs = append(*vs, frame) +} + +func (vs *varstack) Pop() map[ast.Var]ir.Local { + sl := *vs + last := sl[len(sl)-1] + *vs = sl[:len(sl)-1] + return last +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/rego/opa/engine.go b/vendor/github.com/open-policy-agent/opa/internal/rego/opa/engine.go new file mode 100644 index 00000000..36ee8445 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/rego/opa/engine.go @@ -0,0 +1,65 @@ +// Copyright 2021 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package opa + +import ( + "context" +) + +// ErrEngineNotFound is returned by LookupEngine if no wasm engine was +// registered by that name. +var ErrEngineNotFound error = &errEngineNotFound{} + +type errEngineNotFound struct{} + +func (*errEngineNotFound) Error() string { return "engine not found" } +func (*errEngineNotFound) Lines() []string { + return []string{ + `WebAssembly runtime not supported in this build.`, + `----------------------------------------------------------------------------------`, + `Please download an OPA binary with Wasm enabled from`, + `https://www.openpolicyagent.org/docs/latest/#running-opa`, + `or build it yourself (with Wasm enabled).`, + `----------------------------------------------------------------------------------`, + } +} + +// Engine repesents a factory for instances of EvalEngine implementations +type Engine interface { + New() EvalEngine +} + +// EvalEngine is the interface implemented by an engine used to eval a policy +type EvalEngine interface { + Init() (EvalEngine, error) + Entrypoints(context.Context) (map[string]int32, error) + WithPolicyBytes([]byte) EvalEngine + WithDataJSON(interface{}) EvalEngine + Eval(context.Context, EvalOpts) (*Result, error) + SetData(context.Context, interface{}) error + SetDataPath(context.Context, []string, interface{}) error + RemoveDataPath(context.Context, []string) error + Close() +} + +var engines = map[string]Engine{} + +// RegisterEngine registers an evaluation engine by its target name. +// Note that the "rego" target is always available. +func RegisterEngine(name string, e Engine) { + if engines[name] != nil { + panic("duplicate engine registration") + } + engines[name] = e +} + +// LookupEngine allows retrieving an engine registered by name +func LookupEngine(name string) (Engine, error) { + e, ok := engines[name] + if !ok { + return nil, ErrEngineNotFound + } + return e, nil +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/rego/opa/options.go b/vendor/github.com/open-policy-agent/opa/internal/rego/opa/options.go new file mode 100644 index 00000000..1e0477a0 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/rego/opa/options.go @@ -0,0 +1,28 @@ +package opa + +import ( + "io" + "time" + + "github.com/open-policy-agent/opa/ast" + "github.com/open-policy-agent/opa/metrics" + "github.com/open-policy-agent/opa/topdown/cache" + "github.com/open-policy-agent/opa/topdown/print" +) + +// Result holds the evaluation result. +type Result struct { + Result []byte +} + +// EvalOpts define options for performing an evaluation. +type EvalOpts struct { + Input *interface{} + Metrics metrics.Metrics + Entrypoint int32 + Time time.Time + Seed io.Reader + InterQueryBuiltinCache cache.InterQueryCache + PrintHook print.Hook + Capabilities *ast.Capabilities +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/semver/LICENSE b/vendor/github.com/open-policy-agent/opa/internal/semver/LICENSE new file mode 100644 index 00000000..d6456956 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/semver/LICENSE @@ -0,0 +1,202 @@ + + Apache License + Version 2.0, January 2004 + http://www.apache.org/licenses/ + + TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION + + 1. Definitions. + + "License" shall mean the terms and conditions for use, reproduction, + and distribution as defined by Sections 1 through 9 of this document. + + "Licensor" shall mean the copyright owner or entity authorized by + the copyright owner that is granting the License. + + "Legal Entity" shall mean the union of the acting entity and all + other entities that control, are controlled by, or are under common + control with that entity. For the purposes of this definition, + "control" means (i) the power, direct or indirect, to cause the + direction or management of such entity, whether by contract or + otherwise, or (ii) ownership of fifty percent (50%) or more of the + outstanding shares, or (iii) beneficial ownership of such entity. + + "You" (or "Your") shall mean an individual or Legal Entity + exercising permissions granted by this License. + + "Source" form shall mean the preferred form for making modifications, + including but not limited to software source code, documentation + source, and configuration files. + + "Object" form shall mean any form resulting from mechanical + transformation or translation of a Source form, including but + not limited to compiled object code, generated documentation, + and conversions to other media types. + + "Work" shall mean the work of authorship, whether in Source or + Object form, made available under the License, as indicated by a + copyright notice that is included in or attached to the work + (an example is provided in the Appendix below). + + "Derivative Works" shall mean any work, whether in Source or Object + form, that is based on (or derived from) the Work and for which the + editorial revisions, annotations, elaborations, or other modifications + represent, as a whole, an original work of authorship. For the purposes + of this License, Derivative Works shall not include works that remain + separable from, or merely link (or bind by name) to the interfaces of, + the Work and Derivative Works thereof. + + "Contribution" shall mean any work of authorship, including + the original version of the Work and any modifications or additions + to that Work or Derivative Works thereof, that is intentionally + submitted to Licensor for inclusion in the Work by the copyright owner + or by an individual or Legal Entity authorized to submit on behalf of + the copyright owner. For the purposes of this definition, "submitted" + means any form of electronic, verbal, or written communication sent + to the Licensor or its representatives, including but not limited to + communication on electronic mailing lists, source code control systems, + and issue tracking systems that are managed by, or on behalf of, the + Licensor for the purpose of discussing and improving the Work, but + excluding communication that is conspicuously marked or otherwise + designated in writing by the copyright owner as "Not a Contribution." + + "Contributor" shall mean Licensor and any individual or Legal Entity + on behalf of whom a Contribution has been received by Licensor and + subsequently incorporated within the Work. + + 2. Grant of Copyright License. Subject to the terms and conditions of + this License, each Contributor hereby grants to You a perpetual, + worldwide, non-exclusive, no-charge, royalty-free, irrevocable + copyright license to reproduce, prepare Derivative Works of, + publicly display, publicly perform, sublicense, and distribute the + Work and such Derivative Works in Source or Object form. + + 3. Grant of Patent License. Subject to the terms and conditions of + this License, each Contributor hereby grants to You a perpetual, + worldwide, non-exclusive, no-charge, royalty-free, irrevocable + (except as stated in this section) patent license to make, have made, + use, offer to sell, sell, import, and otherwise transfer the Work, + where such license applies only to those patent claims licensable + by such Contributor that are necessarily infringed by their + Contribution(s) alone or by combination of their Contribution(s) + with the Work to which such Contribution(s) was submitted. If You + institute patent litigation against any entity (including a + cross-claim or counterclaim in a lawsuit) alleging that the Work + or a Contribution incorporated within the Work constitutes direct + or contributory patent infringement, then any patent licenses + granted to You under this License for that Work shall terminate + as of the date such litigation is filed. + + 4. Redistribution. You may reproduce and distribute copies of the + Work or Derivative Works thereof in any medium, with or without + modifications, and in Source or Object form, provided that You + meet the following conditions: + + (a) You must give any other recipients of the Work or + Derivative Works a copy of this License; and + + (b) You must cause any modified files to carry prominent notices + stating that You changed the files; and + + (c) You must retain, in the Source form of any Derivative Works + that You distribute, all copyright, patent, trademark, and + attribution notices from the Source form of the Work, + excluding those notices that do not pertain to any part of + the Derivative Works; and + + (d) If the Work includes a "NOTICE" text file as part of its + distribution, then any Derivative Works that You distribute must + include a readable copy of the attribution notices contained + within such NOTICE file, excluding those notices that do not + pertain to any part of the Derivative Works, in at least one + of the following places: within a NOTICE text file distributed + as part of the Derivative Works; within the Source form or + documentation, if provided along with the Derivative Works; or, + within a display generated by the Derivative Works, if and + wherever such third-party notices normally appear. The contents + of the NOTICE file are for informational purposes only and + do not modify the License. You may add Your own attribution + notices within Derivative Works that You distribute, alongside + or as an addendum to the NOTICE text from the Work, provided + that such additional attribution notices cannot be construed + as modifying the License. + + You may add Your own copyright statement to Your modifications and + may provide additional or different license terms and conditions + for use, reproduction, or distribution of Your modifications, or + for any such Derivative Works as a whole, provided Your use, + reproduction, and distribution of the Work otherwise complies with + the conditions stated in this License. + + 5. Submission of Contributions. Unless You explicitly state otherwise, + any Contribution intentionally submitted for inclusion in the Work + by You to the Licensor shall be under the terms and conditions of + this License, without any additional terms or conditions. + Notwithstanding the above, nothing herein shall supersede or modify + the terms of any separate license agreement you may have executed + with Licensor regarding such Contributions. + + 6. Trademarks. This License does not grant permission to use the trade + names, trademarks, service marks, or product names of the Licensor, + except as required for reasonable and customary use in describing the + origin of the Work and reproducing the content of the NOTICE file. + + 7. Disclaimer of Warranty. Unless required by applicable law or + agreed to in writing, Licensor provides the Work (and each + Contributor provides its Contributions) on an "AS IS" BASIS, + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or + implied, including, without limitation, any warranties or conditions + of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A + PARTICULAR PURPOSE. You are solely responsible for determining the + appropriateness of using or redistributing the Work and assume any + risks associated with Your exercise of permissions under this License. + + 8. Limitation of Liability. In no event and under no legal theory, + whether in tort (including negligence), contract, or otherwise, + unless required by applicable law (such as deliberate and grossly + negligent acts) or agreed to in writing, shall any Contributor be + liable to You for damages, including any direct, indirect, special, + incidental, or consequential damages of any character arising as a + result of this License or out of the use or inability to use the + Work (including but not limited to damages for loss of goodwill, + work stoppage, computer failure or malfunction, or any and all + other commercial damages or losses), even if such Contributor + has been advised of the possibility of such damages. + + 9. Accepting Warranty or Additional Liability. While redistributing + the Work or Derivative Works thereof, You may choose to offer, + and charge a fee for, acceptance of support, warranty, indemnity, + or other liability obligations and/or rights consistent with this + License. However, in accepting such obligations, You may act only + on Your own behalf and on Your sole responsibility, not on behalf + of any other Contributor, and only if You agree to indemnify, + defend, and hold each Contributor harmless for any liability + incurred by, or claims asserted against, such Contributor by reason + of your accepting any such warranty or additional liability. + + END OF TERMS AND CONDITIONS + + APPENDIX: How to apply the Apache License to your work. + + To apply the Apache License to your work, attach the following + boilerplate notice, with the fields enclosed by brackets "[]" + replaced with your own identifying information. (Don't include + the brackets!) The text should be enclosed in the appropriate + comment syntax for the file format. We also recommend that a + file or class name and description of purpose be included on the + same "printed page" as the copyright notice for easier + identification within third-party archives. + + Copyright [yyyy] [name of copyright owner] + + Licensed under the Apache License, Version 2.0 (the "License"); + you may not use this file except in compliance with the License. + You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + + Unless required by applicable law or agreed to in writing, software + distributed under the License is distributed on an "AS IS" BASIS, + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + See the License for the specific language governing permissions and + limitations under the License. diff --git a/vendor/github.com/open-policy-agent/opa/internal/semver/semver.go b/vendor/github.com/open-policy-agent/opa/internal/semver/semver.go new file mode 100644 index 00000000..389eeccc --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/semver/semver.go @@ -0,0 +1,235 @@ +// Copyright 2013-2015 CoreOS, Inc. +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. + +// Semantic Versions http://semver.org + +// Package semver has been vendored from: +// https://github.com/coreos/go-semver/tree/e214231b295a8ea9479f11b70b35d5acf3556d9b/semver +// A number of the original functions of the package have been removed since +// they are not required for our built-ins. +package semver + +import ( + "bytes" + "fmt" + "regexp" + "strconv" + "strings" +) + +// Version represents a parsed SemVer +type Version struct { + Major int64 + Minor int64 + Patch int64 + PreRelease PreRelease + Metadata string +} + +// PreRelease represents a pre-release suffix string +type PreRelease string + +func splitOff(input *string, delim string) (val string) { + parts := strings.SplitN(*input, delim, 2) + + if len(parts) == 2 { + *input = parts[0] + val = parts[1] + } + + return val +} + +// NewVersion constructs new SemVers from strings +func NewVersion(version string) (*Version, error) { + v := Version{} + + if err := v.Set(version); err != nil { + return nil, err + } + + return &v, nil +} + +// Set parses and updates v from the given version string. Implements flag.Value +func (v *Version) Set(version string) error { + metadata := splitOff(&version, "+") + preRelease := PreRelease(splitOff(&version, "-")) + dotParts := strings.SplitN(version, ".", 3) + + if len(dotParts) != 3 { + return fmt.Errorf("%s is not in dotted-tri format", version) + } + + if err := validateIdentifier(string(preRelease)); err != nil { + return fmt.Errorf("failed to validate pre-release: %v", err) + } + + if err := validateIdentifier(metadata); err != nil { + return fmt.Errorf("failed to validate metadata: %v", err) + } + + parsed := make([]int64, 3) + + for i, v := range dotParts[:3] { + val, err := strconv.ParseInt(v, 10, 64) + parsed[i] = val + if err != nil { + return err + } + } + + v.Metadata = metadata + v.PreRelease = preRelease + v.Major = parsed[0] + v.Minor = parsed[1] + v.Patch = parsed[2] + return nil +} + +func (v Version) String() string { + var buffer bytes.Buffer + + fmt.Fprintf(&buffer, "%d.%d.%d", v.Major, v.Minor, v.Patch) + + if v.PreRelease != "" { + fmt.Fprintf(&buffer, "-%s", v.PreRelease) + } + + if v.Metadata != "" { + fmt.Fprintf(&buffer, "+%s", v.Metadata) + } + + return buffer.String() +} + +// Compare tests if v is less than, equal to, or greater than versionB, +// returning -1, 0, or +1 respectively. +func (v Version) Compare(versionB Version) int { + if cmp := recursiveCompare(v.Slice(), versionB.Slice()); cmp != 0 { + return cmp + } + return preReleaseCompare(v, versionB) +} + +// Slice converts the comparable parts of the semver into a slice of integers. +func (v Version) Slice() []int64 { + return []int64{v.Major, v.Minor, v.Patch} +} + +// Slice splits the pre-release suffix string +func (p PreRelease) Slice() []string { + preRelease := string(p) + return strings.Split(preRelease, ".") +} + +func preReleaseCompare(versionA Version, versionB Version) int { + a := versionA.PreRelease + b := versionB.PreRelease + + /* Handle the case where if two versions are otherwise equal it is the + * one without a PreRelease that is greater */ + if len(a) == 0 && (len(b) > 0) { + return 1 + } else if len(b) == 0 && (len(a) > 0) { + return -1 + } + + // If there is a prerelease, check and compare each part. + return recursivePreReleaseCompare(a.Slice(), b.Slice()) +} + +func recursiveCompare(versionA []int64, versionB []int64) int { + if len(versionA) == 0 { + return 0 + } + + a := versionA[0] + b := versionB[0] + + if a > b { + return 1 + } else if a < b { + return -1 + } + + return recursiveCompare(versionA[1:], versionB[1:]) +} + +func recursivePreReleaseCompare(versionA []string, versionB []string) int { + // A larger set of pre-release fields has a higher precedence than a smaller set, + // if all of the preceding identifiers are equal. + if len(versionA) == 0 { + if len(versionB) > 0 { + return -1 + } + return 0 + } else if len(versionB) == 0 { + // We're longer than versionB so return 1. + return 1 + } + + a := versionA[0] + b := versionB[0] + + aInt := false + bInt := false + + aI, err := strconv.Atoi(versionA[0]) + if err == nil { + aInt = true + } + + bI, err := strconv.Atoi(versionB[0]) + if err == nil { + bInt = true + } + + // Numeric identifiers always have lower precedence than non-numeric identifiers. + if aInt && !bInt { + return -1 + } else if !aInt && bInt { + return 1 + } + + // Handle Integer Comparison + if aInt && bInt { + if aI > bI { + return 1 + } else if aI < bI { + return -1 + } + } + + // Handle String Comparison + if a > b { + return 1 + } else if a < b { + return -1 + } + + return recursivePreReleaseCompare(versionA[1:], versionB[1:]) +} + +// validateIdentifier makes sure the provided identifier satisfies semver spec +func validateIdentifier(id string) error { + if id != "" && !reIdentifier.MatchString(id) { + return fmt.Errorf("%s is not a valid semver identifier", id) + } + return nil +} + +// reIdentifier is a regular expression used to check that pre-release and metadata +// identifiers satisfy the spec requirements +var reIdentifier = regexp.MustCompile(`^[0-9A-Za-z-]+(\.[0-9A-Za-z-]+)*$`) diff --git a/vendor/github.com/open-policy-agent/opa/internal/strings/strings.go b/vendor/github.com/open-policy-agent/opa/internal/strings/strings.go new file mode 100644 index 00000000..63a71be7 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/strings/strings.go @@ -0,0 +1,74 @@ +// Copyright 2021 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +// Package strings contains helpers to perform string manipulation +package strings + +import ( + "path/filepath" + "strings" + + "github.com/open-policy-agent/opa/internal/lcss" +) + +// TruncateFilePaths truncates the given file paths to conform to the given +// "ideal" width and returns the shortened paths by replacing the middle parts of paths +// with "...", ex: bundle1/.../a/b/policy.rego +func TruncateFilePaths(maxIdealWidth, maxWidth int, path ...string) (map[string]string, int) { + var canShorten [][]byte + + for _, p := range path { + canShorten = append(canShorten, []byte(getPathFromFirstSeparator(p))) + } + + // Find the longest common path segment + var lcs string + if len(canShorten) > 1 { + lcs = string(lcss.LongestCommonSubstring(canShorten...)) + } else { + lcs = string(canShorten[0]) + } + + // Don't just swap in the full LCSS, trim it down to be the least amount of + // characters to reach our "ideal" width boundary giving as much + // detail as possible without going too long. + diff := maxIdealWidth - (maxWidth - len(lcs) + 3) + if diff > 0 { + if diff > len(lcs) { + lcs = "" + } else { + // Favor data on the right hand side of the path + lcs = lcs[:len(lcs)-diff] + } + } + + result := map[string]string{} + for _, p := range path { + result[p] = p + } + + longestLocation := maxWidth + + // Swap in "..." for the longest common path, but if it makes things better + if len(lcs) > 3 { + for path := range result { + result[path] = strings.Replace(path, lcs, "...", 1) + } + + // Drop the overall length down to match our substitution + longestLocation = longestLocation - (len(lcs) - 3) + } + + return result, longestLocation +} + +func getPathFromFirstSeparator(path string) string { + s := filepath.Dir(path) + s = strings.TrimPrefix(s, string(filepath.Separator)) + firstSlash := strings.IndexRune(s, filepath.Separator) + if firstSlash > 0 { + return s[firstSlash+1:] + } + return s +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/uuid/uuid.go b/vendor/github.com/open-policy-agent/opa/internal/uuid/uuid.go new file mode 100644 index 00000000..e79e7337 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/uuid/uuid.go @@ -0,0 +1,22 @@ +// Copyright 2017 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package uuid + +import ( + "fmt" + "io" +) + +// New Create a version 4 random UUID +func New(r io.Reader) (string, error) { + bs := make([]byte, 16) + n, err := io.ReadFull(r, bs) + if n != len(bs) || err != nil { + return "", err + } + bs[8] = bs[8]&^0xc0 | 0x80 + bs[6] = bs[6]&^0xf0 | 0x40 + return fmt.Sprintf("%x-%x-%x-%x-%x", bs[0:4], bs[4:6], bs[6:8], bs[8:10], bs[10:]), nil +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/version/version.go b/vendor/github.com/open-policy-agent/opa/internal/version/version.go new file mode 100644 index 00000000..1c2e9ecd --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/version/version.go @@ -0,0 +1,36 @@ +// Copyright 2019 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +// Package version implements helper functions for the stored version. +package version + +import ( + "context" + "fmt" + "runtime" + + "github.com/open-policy-agent/opa/storage" + "github.com/open-policy-agent/opa/version" +) + +var versionPath = storage.MustParsePath("/system/version") + +// Write the build version information into storage. This makes the +// version information available to the REPL and the HTTP server. +func Write(ctx context.Context, store storage.Store, txn storage.Transaction) error { + + if err := storage.MakeDir(ctx, store, txn, versionPath); err != nil { + return err + } + + return store.Write(ctx, txn, storage.AddOp, versionPath, map[string]interface{}{ + "version": version.Version, + "build_commit": version.Vcs, + "build_timestamp": version.Timestamp, + "build_hostname": version.Hostname, + }) +} + +// UserAgent defines the current OPA instances User-Agent default header value. +var UserAgent = fmt.Sprintf("Open Policy Agent/%s (%s, %s)", version.Version, runtime.GOOS, runtime.GOARCH) diff --git a/vendor/github.com/open-policy-agent/opa/internal/wasm/constant/constant.go b/vendor/github.com/open-policy-agent/opa/internal/wasm/constant/constant.go new file mode 100644 index 00000000..878979fb --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/wasm/constant/constant.go @@ -0,0 +1,77 @@ +// Copyright 2018 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +// Package constant contains WASM constant definitions. +package constant + +// Magic bytes at the beginning of every WASM file ("\0asm"). +const Magic = uint32(0x6D736100) + +// Version defines the WASM version. +const Version = uint32(1) + +// WASM module section IDs. +const ( + CustomSectionID uint8 = iota + TypeSectionID + ImportSectionID + FunctionSectionID + TableSectionID + MemorySectionID + GlobalSectionID + ExportSectionID + StartSectionID + ElementSectionID + CodeSectionID + DataSectionID +) + +// FunctionTypeID indicates the start of a function type definition. +const FunctionTypeID = byte(0x60) + +// ValueType represents an intrinsic value type in WASM. +const ( + ValueTypeF64 byte = iota + 0x7C + ValueTypeF32 + ValueTypeI64 + ValueTypeI32 +) + +// WASM import descriptor types. +const ( + ImportDescType byte = iota + ImportDescTable + ImportDescMem + ImportDescGlobal +) + +// WASM export descriptor types. +const ( + ExportDescType byte = iota + ExportDescTable + ExportDescMem + ExportDescGlobal +) + +// ElementTypeAnyFunc indicates the type of a table import. +const ElementTypeAnyFunc byte = 0x70 + +// BlockTypeEmpty represents a block type. +const BlockTypeEmpty byte = 0x40 + +// WASM global varialbe mutability flag. +const ( + Const byte = iota + Mutable +) + +// NameSectionCustomID is the ID of the "Name" section Custom Section +const NameSectionCustomID = "name" + +// Subtypes of the 'name' custom section +const ( + NameSectionModuleType byte = iota + NameSectionFunctionsType + NameSectionLocalsType +) diff --git a/vendor/github.com/open-policy-agent/opa/internal/wasm/encoding/doc.go b/vendor/github.com/open-policy-agent/opa/internal/wasm/encoding/doc.go new file mode 100644 index 00000000..b2523696 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/wasm/encoding/doc.go @@ -0,0 +1,6 @@ +// Copyright 2018 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +// Package encoding implements WASM module reading and writing. +package encoding diff --git a/vendor/github.com/open-policy-agent/opa/internal/wasm/encoding/reader.go b/vendor/github.com/open-policy-agent/opa/internal/wasm/encoding/reader.go new file mode 100644 index 00000000..1ec26512 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/wasm/encoding/reader.go @@ -0,0 +1,967 @@ +// Copyright 2018 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package encoding + +import ( + "bytes" + "encoding/binary" + "fmt" + "io" + "io/ioutil" + + "github.com/pkg/errors" + + "github.com/open-policy-agent/opa/internal/leb128" + "github.com/open-policy-agent/opa/internal/wasm/constant" + "github.com/open-policy-agent/opa/internal/wasm/instruction" + "github.com/open-policy-agent/opa/internal/wasm/module" + "github.com/open-policy-agent/opa/internal/wasm/opcode" + "github.com/open-policy-agent/opa/internal/wasm/types" +) + +// ReadModule reads a binary-encoded WASM module from r. +func ReadModule(r io.Reader) (*module.Module, error) { + + wr := &reader{r: r, n: 0} + module, err := readModule(wr) + if err != nil { + return nil, errors.Wrapf(err, "offset 0x%x", wr.n) + } + + return module, nil +} + +// ReadCodeEntry reads a binary-encoded WASM code entry from r. +func ReadCodeEntry(r io.Reader) (*module.CodeEntry, error) { + + wr := &reader{r: r, n: 0} + entry, err := readCodeEntry(wr) + if err != nil { + return nil, errors.Wrapf(err, "offset 0x%x", wr.n) + } + + return entry, nil +} + +// CodeEntries returns the WASM code entries contained in r. +func CodeEntries(m *module.Module) ([]*module.CodeEntry, error) { + + entries := make([]*module.CodeEntry, len(m.Code.Segments)) + + for i, s := range m.Code.Segments { + buf := bytes.NewBuffer(s.Code) + entry, err := ReadCodeEntry(buf) + if err != nil { + return nil, err + } + entries[i] = entry + } + + return entries, nil +} + +type reader struct { + r io.Reader + n int +} + +func (r *reader) Read(bs []byte) (int, error) { + n, err := r.r.Read(bs) + r.n += n + return n, err +} + +func readModule(r io.Reader) (*module.Module, error) { + + if err := readMagic(r); err != nil { + return nil, err + } + + if err := readVersion(r); err != nil { + return nil, err + } + + var m module.Module + + if err := readSections(r, &m); err != nil && err != io.EOF { + return nil, err + } + + return &m, nil +} + +func readCodeEntry(r io.Reader) (*module.CodeEntry, error) { + + var entry module.CodeEntry + + if err := readLocals(r, &entry.Func.Locals); err != nil { + return nil, errors.Wrapf(err, "local declarations") + } + + return &entry, readExpr(r, &entry.Func.Expr) +} + +func readMagic(r io.Reader) error { + var v uint32 + if err := binary.Read(r, binary.LittleEndian, &v); err != nil { + return err + } else if v != constant.Magic { + return fmt.Errorf("illegal magic value") + } + return nil +} + +func readVersion(r io.Reader) error { + var v uint32 + if err := binary.Read(r, binary.LittleEndian, &v); err != nil { + return err + } else if v != constant.Version { + return fmt.Errorf("illegal wasm version") + } + return nil +} + +func readSections(r io.Reader, m *module.Module) error { + for { + id, err := readByte(r) + if err != nil { + return err + } + + size, err := leb128.ReadVarUint32(r) + if err != nil { + return err + } + + buf := make([]byte, size) + if _, err := io.ReadFull(r, buf); err != nil { + return err + } + + bufr := bytes.NewReader(buf) + + switch id { + case constant.StartSectionID: + if err := readStartSection(bufr, &m.Start); err != nil { + return errors.Wrap(err, "start section") + } + case constant.CustomSectionID: + var name string + if err := readByteVectorString(bufr, &name); err != nil { + return errors.Wrap(err, "read custom section type") + } + if name == "name" { + if err := readCustomNameSections(bufr, &m.Names); err != nil { + return errors.Wrap(err, "custom 'name' section") + } + } else { + if err := readCustomSection(bufr, name, &m.Customs); err != nil { + return errors.Wrap(err, "custom section") + } + } + case constant.TypeSectionID: + if err := readTypeSection(bufr, &m.Type); err != nil { + return errors.Wrap(err, "type section") + } + case constant.ImportSectionID: + if err := readImportSection(bufr, &m.Import); err != nil { + return errors.Wrap(err, "import section") + } + case constant.TableSectionID: + if err := readTableSection(bufr, &m.Table); err != nil { + return errors.Wrap(err, "table section") + } + case constant.MemorySectionID: + if err := readMemorySection(bufr, &m.Memory); err != nil { + return errors.Wrap(err, "memory section") + } + case constant.GlobalSectionID: + if err := readGlobalSection(bufr, &m.Global); err != nil { + return errors.Wrap(err, "global section") + } + case constant.FunctionSectionID: + if err := readFunctionSection(bufr, &m.Function); err != nil { + return errors.Wrap(err, "function section") + } + case constant.ExportSectionID: + if err := readExportSection(bufr, &m.Export); err != nil { + return errors.Wrap(err, "export section") + } + case constant.ElementSectionID: + if err := readElementSection(bufr, &m.Element); err != nil { + return errors.Wrap(err, "element section") + } + case constant.DataSectionID: + if err := readDataSection(bufr, &m.Data); err != nil { + return errors.Wrap(err, "data section") + } + case constant.CodeSectionID: + if err := readRawCodeSection(bufr, &m.Code); err != nil { + return errors.Wrap(err, "code section") + } + default: + return fmt.Errorf("illegal section id") + } + } +} + +func readCustomSection(r io.Reader, name string, s *[]module.CustomSection) error { + buf, err := ioutil.ReadAll(r) + if err != nil { + return err + } + + *s = append(*s, module.CustomSection{ + Name: name, + Data: buf, + }) + return nil +} + +func readCustomNameSections(r io.Reader, s *module.NameSection) error { + for { + id, err := readByte(r) + if err != nil { + if err == io.EOF { + break + } + return err + } + n, err := leb128.ReadVarUint32(r) + if err != nil { + return err + } + buf := make([]byte, n) + if _, err := io.ReadFull(r, buf); err != nil { + return err + } + bufr := bytes.NewReader(buf) + switch id { + case constant.NameSectionModuleType: + err = readNameSectionModule(bufr, s) + case constant.NameSectionFunctionsType: + err = readNameSectionFunctions(bufr, s) + case constant.NameSectionLocalsType: + err = readNameSectionLocals(bufr, s) + } + if err != nil { + return err + } + } + return nil +} + +func readNameSectionModule(r io.Reader, s *module.NameSection) error { + return readByteVectorString(r, &s.Module) +} + +func readNameSectionFunctions(r io.Reader, s *module.NameSection) error { + nm, err := readNameMap(r) + if err != nil { + return err + } + s.Functions = nm + return nil +} + +func readNameMap(r io.Reader) ([]module.NameMap, error) { + n, err := leb128.ReadVarUint32(r) + if err != nil { + return nil, err + } + nm := make([]module.NameMap, n) + for i := uint32(0); i < n; i++ { + var name string + id, err := leb128.ReadVarUint32(r) + if err != nil { + return nil, err + } + + if err := readByteVectorString(r, &name); err != nil { + return nil, err + } + nm[i] = module.NameMap{Index: id, Name: name} + } + return nm, nil +} + +func readNameSectionLocals(r io.Reader, s *module.NameSection) error { + n, err := leb128.ReadVarUint32(r) // length of vec(indirectnameassoc) + if err != nil { + return err + } + for i := uint32(0); i < n; i++ { + id, err := leb128.ReadVarUint32(r) // func index + if err != nil { + return err + } + nm, err := readNameMap(r) + if err != nil { + return err + } + for _, m := range nm { + s.Locals = append(s.Locals, module.LocalNameMap{ + FuncIndex: id, + NameMap: module.NameMap{ + Index: m.Index, + Name: m.Name, + }}) + } + } + return nil +} + +func readStartSection(r io.Reader, s *module.StartSection) error { + idx, err := leb128.ReadVarUint32(r) + if err != nil { + return err + } + s.FuncIndex = &idx + return nil +} + +func readTypeSection(r io.Reader, s *module.TypeSection) error { + + n, err := leb128.ReadVarUint32(r) + if err != nil { + return err + } + + for i := uint32(0); i < n; i++ { + + var ftype module.FunctionType + if err := readFunctionType(r, &ftype); err != nil { + return err + } + + s.Functions = append(s.Functions, ftype) + } + + return nil +} + +func readImportSection(r io.Reader, s *module.ImportSection) error { + + n, err := leb128.ReadVarUint32(r) + if err != nil { + return err + } + + for i := uint32(0); i < n; i++ { + + var imp module.Import + + if err := readImport(r, &imp); err != nil { + return err + } + + s.Imports = append(s.Imports, imp) + } + + return nil +} + +func readTableSection(r io.Reader, s *module.TableSection) error { + + n, err := leb128.ReadVarUint32(r) + if err != nil { + return err + } + + for i := uint32(0); i < n; i++ { + + var table module.Table + + if elem, err := readByte(r); err != nil { + return err + } else if elem != constant.ElementTypeAnyFunc { + return fmt.Errorf("illegal element type") + } else { + table.Type = types.Anyfunc + } + + if err := readLimits(r, &table.Lim); err != nil { + return err + } + + s.Tables = append(s.Tables, table) + } + + return nil +} + +func readMemorySection(r io.Reader, s *module.MemorySection) error { + + n, err := leb128.ReadVarUint32(r) + if err != nil { + return err + } + + for i := uint32(0); i < n; i++ { + + var mem module.Memory + + if err := readLimits(r, &mem.Lim); err != nil { + return err + } + + s.Memories = append(s.Memories, mem) + } + + return nil +} + +func readGlobalSection(r io.Reader, s *module.GlobalSection) error { + + n, err := leb128.ReadVarUint32(r) + if err != nil { + return err + } + + for i := uint32(0); i < n; i++ { + + var global module.Global + + if err := readGlobal(r, &global); err != nil { + return err + } + + s.Globals = append(s.Globals, global) + } + + return nil +} + +func readFunctionSection(r io.Reader, s *module.FunctionSection) error { + return readVarUint32Vector(r, &s.TypeIndices) +} + +func readExportSection(r io.Reader, s *module.ExportSection) error { + + n, err := leb128.ReadVarUint32(r) + if err != nil { + return err + } + + for i := uint32(0); i < n; i++ { + + var exp module.Export + + if err := readExport(r, &exp); err != nil { + return err + } + + s.Exports = append(s.Exports, exp) + } + + return nil +} + +func readElementSection(r io.Reader, s *module.ElementSection) error { + + n, err := leb128.ReadVarUint32(r) + if err != nil { + return err + } + + for i := uint32(0); i < n; i++ { + + var seg module.ElementSegment + + if err := readElementSegment(r, &seg); err != nil { + return err + } + + s.Segments = append(s.Segments, seg) + } + + return nil +} + +func readDataSection(r io.Reader, s *module.DataSection) error { + + n, err := leb128.ReadVarUint32(r) + if err != nil { + return err + } + + for i := uint32(0); i < n; i++ { + + var seg module.DataSegment + + if err := readDataSegment(r, &seg); err != nil { + return err + } + + s.Segments = append(s.Segments, seg) + } + + return nil +} + +func readRawCodeSection(r io.Reader, s *module.RawCodeSection) error { + + n, err := leb128.ReadVarUint32(r) + if err != nil { + return err + } + + for i := uint32(0); i < n; i++ { + var seg module.RawCodeSegment + + if err := readRawCodeSegment(r, &seg); err != nil { + return err + } + + s.Segments = append(s.Segments, seg) + } + + return nil +} + +func readFunctionType(r io.Reader, ftype *module.FunctionType) error { + + if b, err := readByte(r); err != nil { + return err + } else if b != constant.FunctionTypeID { + return fmt.Errorf("illegal function type id 0x%x", b) + } + + if err := readValueTypeVector(r, &ftype.Params); err != nil { + return err + } + + return readValueTypeVector(r, &ftype.Results) +} + +func readGlobal(r io.Reader, global *module.Global) error { + + if err := readValueType(r, &global.Type); err != nil { + return err + } + + b, err := readByte(r) + if err != nil { + return err + } + + if b == 1 { + global.Mutable = true + } else if b != 0 { + return fmt.Errorf("illegal mutability flag") + } + + return readConstantExpr(r, &global.Init) +} + +func readImport(r io.Reader, imp *module.Import) error { + + if err := readByteVectorString(r, &imp.Module); err != nil { + return err + } + + if err := readByteVectorString(r, &imp.Name); err != nil { + return err + } + + b, err := readByte(r) + if err != nil { + return err + + } + + if b == constant.ImportDescType { + index, err := leb128.ReadVarUint32(r) + if err != nil { + return err + } + imp.Descriptor = module.FunctionImport{ + Func: index, + } + return nil + } + + if b == constant.ImportDescTable { + if elem, err := readByte(r); err != nil { + return err + } else if elem != constant.ElementTypeAnyFunc { + return fmt.Errorf("illegal element type") + } + desc := module.TableImport{ + Type: types.Anyfunc, + } + if err := readLimits(r, &desc.Lim); err != nil { + return err + } + imp.Descriptor = desc + return nil + } + + if b == constant.ImportDescMem { + desc := module.MemoryImport{} + if err := readLimits(r, &desc.Mem.Lim); err != nil { + return err + } + imp.Descriptor = desc + return nil + } + + if b == constant.ImportDescGlobal { + desc := module.GlobalImport{} + if err := readValueType(r, &desc.Type); err != nil { + return err + } + b, err := readByte(r) + if err != nil { + return err + } + if b == 1 { + desc.Mutable = true + } else if b != 0 { + return fmt.Errorf("illegal mutability flag") + } + return nil + } + + return fmt.Errorf("illegal import descriptor type") +} + +func readExport(r io.Reader, exp *module.Export) error { + + if err := readByteVectorString(r, &exp.Name); err != nil { + return err + } + + b, err := readByte(r) + if err != nil { + return err + } + + switch b { + case constant.ExportDescType: + exp.Descriptor.Type = module.FunctionExportType + case constant.ExportDescTable: + exp.Descriptor.Type = module.TableExportType + case constant.ExportDescMem: + exp.Descriptor.Type = module.MemoryExportType + case constant.ExportDescGlobal: + exp.Descriptor.Type = module.GlobalExportType + default: + return fmt.Errorf("illegal export descriptor type") + } + + exp.Descriptor.Index, err = leb128.ReadVarUint32(r) + if err != nil { + return err + } + + return nil +} + +func readElementSegment(r io.Reader, seg *module.ElementSegment) error { + + if err := readVarUint32(r, &seg.Index); err != nil { + return err + } + + if err := readConstantExpr(r, &seg.Offset); err != nil { + return err + } + + return readVarUint32Vector(r, &seg.Indices) +} + +func readDataSegment(r io.Reader, seg *module.DataSegment) error { + + if err := readVarUint32(r, &seg.Index); err != nil { + return err + } + + if err := readConstantExpr(r, &seg.Offset); err != nil { + return err + } + + return readByteVector(r, &seg.Init) +} + +func readRawCodeSegment(r io.Reader, seg *module.RawCodeSegment) error { + return readByteVector(r, &seg.Code) +} + +func readConstantExpr(r io.Reader, expr *module.Expr) error { + + instrs := make([]instruction.Instruction, 0) + + for { + b, err := readByte(r) + if err != nil { + return err + } + + switch opcode.Opcode(b) { + case opcode.I32Const: + i32, err := leb128.ReadVarInt32(r) + if err != nil { + return err + } + instrs = append(instrs, instruction.I32Const{Value: i32}) + case opcode.I64Const: + i64, err := leb128.ReadVarInt64(r) + if err != nil { + return err + } + instrs = append(instrs, instruction.I64Const{Value: i64}) + case opcode.End: + expr.Instrs = instrs + return nil + default: + return fmt.Errorf("illegal constant expr opcode 0x%x", b) + } + } +} + +func readExpr(r io.Reader, expr *module.Expr) (err error) { + + defer func() { + if r := recover(); r != nil { + switch r := r.(type) { + case error: + err = r + default: + err = fmt.Errorf("unknown panic") + } + } + }() + + return readInstructions(r, &expr.Instrs) +} + +func readInstructions(r io.Reader, instrs *[]instruction.Instruction) error { + + ret := make([]instruction.Instruction, 0) + + for { + b, err := readByte(r) + if err != nil { + return err + } + + switch opcode.Opcode(b) { + case opcode.I32Const: + ret = append(ret, instruction.I32Const{Value: leb128.MustReadVarInt32(r)}) + case opcode.I64Const: + ret = append(ret, instruction.I64Const{Value: leb128.MustReadVarInt64(r)}) + case opcode.I32Eqz: + ret = append(ret, instruction.I32Eqz{}) + case opcode.GetLocal: + ret = append(ret, instruction.GetLocal{Index: leb128.MustReadVarUint32(r)}) + case opcode.SetLocal: + ret = append(ret, instruction.SetLocal{Index: leb128.MustReadVarUint32(r)}) + case opcode.Call: + ret = append(ret, instruction.Call{Index: leb128.MustReadVarUint32(r)}) + case opcode.CallIndirect: + ret = append(ret, instruction.CallIndirect{ + Index: leb128.MustReadVarUint32(r), + Reserved: mustReadByte(r), + }) + case opcode.BrIf: + ret = append(ret, instruction.BrIf{Index: leb128.MustReadVarUint32(r)}) + case opcode.Return: + ret = append(ret, instruction.Return{}) + case opcode.Block: + block := instruction.Block{} + if err := readBlockValueType(r, block.Type); err != nil { + return err + } + if err := readInstructions(r, &block.Instrs); err != nil { + return err + } + ret = append(ret, block) + case opcode.Loop: + loop := instruction.Loop{} + if err := readBlockValueType(r, loop.Type); err != nil { + return err + } + if err := readInstructions(r, &loop.Instrs); err != nil { + return err + } + ret = append(ret, loop) + case opcode.End: + *instrs = ret + return nil + default: + return fmt.Errorf("illegal opcode 0x%x", b) + } + } +} + +func mustReadByte(r io.Reader) byte { + b, err := readByte(r) + if err != nil { + panic(err) + } + return b +} + +func readLimits(r io.Reader, l *module.Limit) error { + + b, err := readByte(r) + if err != nil { + return err + } + + min, err := leb128.ReadVarUint32(r) + if err != nil { + return err + } + + l.Min = min + + if b == 1 { + max, err := leb128.ReadVarUint32(r) + if err != nil { + return err + } + l.Max = &max + } else if b != 0 { + return fmt.Errorf("illegal limit flag") + } + + return nil +} + +func readLocals(r io.Reader, locals *[]module.LocalDeclaration) error { + + n, err := leb128.ReadVarUint32(r) + if err != nil { + return err + } + + ret := make([]module.LocalDeclaration, n) + + for i := uint32(0); i < n; i++ { + if err := readVarUint32(r, &ret[i].Count); err != nil { + return err + } + if err := readValueType(r, &ret[i].Type); err != nil { + return err + } + } + + *locals = ret + return nil +} + +func readByteVector(r io.Reader, v *[]byte) error { + + n, err := leb128.ReadVarUint32(r) + if err != nil { + return err + } + + buf := make([]byte, n) + if _, err := io.ReadFull(r, buf); err != nil { + return err + } + + *v = buf + return nil +} + +func readByteVectorString(r io.Reader, v *string) error { + + var buf []byte + + if err := readByteVector(r, &buf); err != nil { + return err + } + + *v = string(buf) + return nil +} + +func readVarUint32Vector(r io.Reader, v *[]uint32) error { + + n, err := leb128.ReadVarUint32(r) + if err != nil { + return err + } + + ret := make([]uint32, n) + + for i := uint32(0); i < n; i++ { + if err := readVarUint32(r, &ret[i]); err != nil { + return err + } + } + + *v = ret + return nil +} + +func readValueTypeVector(r io.Reader, v *[]types.ValueType) error { + + n, err := leb128.ReadVarUint32(r) + if err != nil { + return err + } + + ret := make([]types.ValueType, n) + + for i := uint32(0); i < n; i++ { + if err := readValueType(r, &ret[i]); err != nil { + return err + } + } + + *v = ret + return nil +} + +func readVarUint32(r io.Reader, v *uint32) error { + var err error + *v, err = leb128.ReadVarUint32(r) + return err +} + +func readValueType(r io.Reader, v *types.ValueType) error { + if b, err := readByte(r); err != nil { + return err + } else if b == constant.ValueTypeI32 { + *v = types.I32 + } else if b == constant.ValueTypeI64 { + *v = types.I64 + } else if b == constant.ValueTypeF32 { + *v = types.F32 + } else if b == constant.ValueTypeF64 { + *v = types.F64 + } else { + return fmt.Errorf("illegal value type: 0x%x", b) + } + return nil +} + +func readBlockValueType(r io.Reader, v *types.ValueType) error { + if b, err := readByte(r); err != nil { + return err + } else if b == constant.ValueTypeI32 { + *v = types.I32 + } else if b == constant.ValueTypeI64 { + *v = types.I64 + } else if b == constant.ValueTypeF32 { + *v = types.F32 + } else if b == constant.ValueTypeF64 { + *v = types.F64 + } else if b != constant.BlockTypeEmpty { + return fmt.Errorf("illegal value type: 0x%x", b) + } + return nil +} + +func readByte(r io.Reader) (byte, error) { + buf := make([]byte, 1) + _, err := io.ReadFull(r, buf) + return buf[0], err +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/wasm/encoding/writer.go b/vendor/github.com/open-policy-agent/opa/internal/wasm/encoding/writer.go new file mode 100644 index 00000000..91e4389a --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/wasm/encoding/writer.go @@ -0,0 +1,777 @@ +// Copyright 2018 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package encoding + +import ( + "bytes" + "encoding/binary" + "fmt" + "io" + "math" + + "github.com/open-policy-agent/opa/internal/leb128" + "github.com/open-policy-agent/opa/internal/wasm/constant" + "github.com/open-policy-agent/opa/internal/wasm/instruction" + "github.com/open-policy-agent/opa/internal/wasm/module" + "github.com/open-policy-agent/opa/internal/wasm/opcode" + "github.com/open-policy-agent/opa/internal/wasm/types" +) + +// WriteModule writes a binary-encoded representation of module to w. +func WriteModule(w io.Writer, module *module.Module) error { + + if err := writeMagic(w); err != nil { + return err + } + + if err := writeVersion(w); err != nil { + return err + } + + if module == nil { + return nil + } + + if err := writeTypeSection(w, module.Type); err != nil { + return err + } + + if err := writeImportSection(w, module.Import); err != nil { + return err + } + + if err := writeFunctionSection(w, module.Function); err != nil { + return err + } + + if err := writeTableSection(w, module.Table); err != nil { + return err + } + + if err := writeMemorySection(w, module.Memory); err != nil { + return err + } + + if err := writeGlobalSection(w, module.Global); err != nil { + return err + } + + if err := writeExportSection(w, module.Export); err != nil { + return err + } + + if err := writeStartSection(w, module.Start); err != nil { + return err + } + + if err := writeElementSection(w, module.Element); err != nil { + return err + } + + if err := writeRawCodeSection(w, module.Code); err != nil { + return err + } + + if err := writeDataSection(w, module.Data); err != nil { + return err + } + + if err := writeNameSection(w, module.Names); err != nil { + return err + } + + for _, custom := range module.Customs { + if err := writeCustomSection(w, custom); err != nil { + return err + } + } + + return nil +} + +// WriteCodeEntry writes a binary encoded representation of entry to w. +func WriteCodeEntry(w io.Writer, entry *module.CodeEntry) error { + + if err := leb128.WriteVarUint32(w, uint32(len(entry.Func.Locals))); err != nil { + return err + } + + for _, local := range entry.Func.Locals { + + if err := leb128.WriteVarUint32(w, local.Count); err != nil { + return err + } + + if err := writeValueType(w, local.Type); err != nil { + return err + } + } + + return writeInstructions(w, entry.Func.Expr.Instrs) +} + +func writeMagic(w io.Writer) error { + return binary.Write(w, binary.LittleEndian, constant.Magic) +} + +func writeVersion(w io.Writer) error { + return binary.Write(w, binary.LittleEndian, constant.Version) +} + +func writeStartSection(w io.Writer, s module.StartSection) error { + + if s.FuncIndex == nil { + return nil + } + + if err := writeByte(w, constant.StartSectionID); err != nil { + return err + } + + var buf bytes.Buffer + if err := leb128.WriteVarUint32(&buf, *s.FuncIndex); err != nil { + return err + } + return writeRawSection(w, &buf) +} + +func writeTypeSection(w io.Writer, s module.TypeSection) error { + + if len(s.Functions) == 0 { + return nil + } + + if err := writeByte(w, constant.TypeSectionID); err != nil { + return err + } + + var buf bytes.Buffer + + if err := leb128.WriteVarUint32(&buf, uint32(len(s.Functions))); err != nil { + return err + } + + for _, fsig := range s.Functions { + if err := writeFunctionType(&buf, fsig); err != nil { + return err + } + } + + return writeRawSection(w, &buf) +} + +func writeImportSection(w io.Writer, s module.ImportSection) error { + + if len(s.Imports) == 0 { + return nil + } + + if err := writeByte(w, constant.ImportSectionID); err != nil { + return err + } + + var buf bytes.Buffer + + if err := leb128.WriteVarUint32(&buf, uint32(len(s.Imports))); err != nil { + return err + } + + for _, imp := range s.Imports { + if err := writeImport(&buf, imp); err != nil { + return err + } + } + + return writeRawSection(w, &buf) +} + +func writeGlobalSection(w io.Writer, s module.GlobalSection) error { + + if len(s.Globals) == 0 { + return nil + } + + if err := writeByte(w, constant.GlobalSectionID); err != nil { + return err + } + + var buf bytes.Buffer + + if err := leb128.WriteVarUint32(&buf, uint32(len(s.Globals))); err != nil { + return err + } + + for _, global := range s.Globals { + if err := writeGlobal(&buf, global); err != nil { + return err + } + } + + return writeRawSection(w, &buf) +} + +func writeFunctionSection(w io.Writer, s module.FunctionSection) error { + + if len(s.TypeIndices) == 0 { + return nil + } + + if err := writeByte(w, constant.FunctionSectionID); err != nil { + return err + } + + var buf bytes.Buffer + + if err := leb128.WriteVarUint32(&buf, uint32(len(s.TypeIndices))); err != nil { + return err + } + + for _, idx := range s.TypeIndices { + if err := leb128.WriteVarUint32(&buf, uint32(idx)); err != nil { + return err + } + } + + return writeRawSection(w, &buf) +} + +func writeTableSection(w io.Writer, s module.TableSection) error { + + if len(s.Tables) == 0 { + return nil + } + + if err := writeByte(w, constant.TableSectionID); err != nil { + return err + } + + var buf bytes.Buffer + + if err := leb128.WriteVarUint32(&buf, uint32(len(s.Tables))); err != nil { + return err + } + + for _, table := range s.Tables { + switch table.Type { + case types.Anyfunc: + if err := writeByte(&buf, constant.ElementTypeAnyFunc); err != nil { + return err + } + default: + return fmt.Errorf("illegal table element type") + } + if err := writeLimits(&buf, table.Lim); err != nil { + return err + } + } + + return writeRawSection(w, &buf) +} + +func writeMemorySection(w io.Writer, s module.MemorySection) error { + + if len(s.Memories) == 0 { + return nil + } + + if err := writeByte(w, constant.MemorySectionID); err != nil { + return err + } + + var buf bytes.Buffer + + if err := leb128.WriteVarUint32(&buf, uint32(len(s.Memories))); err != nil { + return err + } + + for _, mem := range s.Memories { + if err := writeLimits(&buf, mem.Lim); err != nil { + return err + } + } + + return writeRawSection(w, &buf) +} + +func writeExportSection(w io.Writer, s module.ExportSection) error { + + if len(s.Exports) == 0 { + return nil + } + + if err := writeByte(w, constant.ExportSectionID); err != nil { + return err + } + + var buf bytes.Buffer + + if err := leb128.WriteVarUint32(&buf, uint32(len(s.Exports))); err != nil { + return err + } + + for _, exp := range s.Exports { + if err := writeByteVector(&buf, []byte(exp.Name)); err != nil { + return err + } + var tpe byte + switch exp.Descriptor.Type { + case module.FunctionExportType: + tpe = constant.ExportDescType + case module.TableExportType: + tpe = constant.ExportDescTable + case module.MemoryExportType: + tpe = constant.ExportDescMem + case module.GlobalExportType: + tpe = constant.ExportDescGlobal + default: + return fmt.Errorf("illegal export descriptor type 0x%x", exp.Descriptor.Type) + } + if err := writeByte(&buf, tpe); err != nil { + return err + } + if err := leb128.WriteVarUint32(&buf, exp.Descriptor.Index); err != nil { + return err + } + } + + return writeRawSection(w, &buf) +} + +func writeElementSection(w io.Writer, s module.ElementSection) error { + + if len(s.Segments) == 0 { + return nil + } + + if err := writeByte(w, constant.ElementSectionID); err != nil { + return err + } + + var buf bytes.Buffer + + if err := leb128.WriteVarUint32(&buf, uint32(len(s.Segments))); err != nil { + return err + } + + for _, seg := range s.Segments { + if err := leb128.WriteVarUint32(&buf, seg.Index); err != nil { + return err + } + if err := writeInstructions(&buf, seg.Offset.Instrs); err != nil { + return err + } + if err := writeVarUint32Vector(&buf, seg.Indices); err != nil { + return err + } + } + + return writeRawSection(w, &buf) +} + +func writeRawCodeSection(w io.Writer, s module.RawCodeSection) error { + + if len(s.Segments) == 0 { + return nil + } + + if err := writeByte(w, constant.CodeSectionID); err != nil { + return err + } + + var buf bytes.Buffer + + if err := leb128.WriteVarUint32(&buf, uint32(len(s.Segments))); err != nil { + return err + } + + for _, seg := range s.Segments { + if err := leb128.WriteVarUint32(&buf, uint32(len(seg.Code))); err != nil { + return err + } + if _, err := buf.Write(seg.Code); err != nil { + return err + } + } + + return writeRawSection(w, &buf) +} + +func writeDataSection(w io.Writer, s module.DataSection) error { + + if len(s.Segments) == 0 { + return nil + } + + if err := writeByte(w, constant.DataSectionID); err != nil { + return err + } + + var buf bytes.Buffer + + if err := leb128.WriteVarUint32(&buf, uint32(len(s.Segments))); err != nil { + return err + } + + for _, seg := range s.Segments { + if err := leb128.WriteVarUint32(&buf, seg.Index); err != nil { + return err + } + if err := writeInstructions(&buf, seg.Offset.Instrs); err != nil { + return err + } + if err := writeByteVector(&buf, seg.Init); err != nil { + return err + } + } + + return writeRawSection(w, &buf) +} + +func writeNameSection(w io.Writer, s module.NameSection) error { + if s.Module == "" && len(s.Functions) == 0 && len(s.Locals) == 0 { + return nil + } + + if err := writeByte(w, constant.CustomSectionID); err != nil { + return err + } + + var buf bytes.Buffer + if err := writeByteVector(&buf, []byte(constant.NameSectionCustomID)); err != nil { + return err + } + + // "module" subsection + if s.Module != "" { + if err := writeByte(&buf, constant.NameSectionModuleType); err != nil { + return err + } + var mbuf bytes.Buffer + if err := writeByteVector(&mbuf, []byte(s.Module)); err != nil { + return err + } + if err := writeRawSection(&buf, &mbuf); err != nil { + return err + } + } + + // "functions" subsection + if len(s.Functions) != 0 { + if err := writeByte(&buf, constant.NameSectionFunctionsType); err != nil { + return err + } + + var fbuf bytes.Buffer + if err := writeNameMap(&fbuf, s.Functions); err != nil { + return err + } + if err := writeRawSection(&buf, &fbuf); err != nil { + return err + } + } + + // "locals" subsection + if len(s.Locals) != 0 { + if err := writeByte(&buf, constant.NameSectionLocalsType); err != nil { + return err + } + funs := map[uint32][]module.NameMap{} + for _, e := range s.Locals { + funs[e.FuncIndex] = append(funs[e.FuncIndex], module.NameMap{Index: e.Index, Name: e.Name}) + } + var lbuf bytes.Buffer + if err := leb128.WriteVarUint32(&lbuf, uint32(len(funs))); err != nil { + return err + } + for fidx, e := range funs { + if err := leb128.WriteVarUint32(&lbuf, fidx); err != nil { + return err + } + if err := writeNameMap(&lbuf, e); err != nil { + return err + } + } + if err := writeRawSection(&buf, &lbuf); err != nil { + return err + } + } + + return writeRawSection(w, &buf) +} + +func writeNameMap(buf io.Writer, nm []module.NameMap) error { + if err := leb128.WriteVarUint32(buf, uint32(len(nm))); err != nil { + return err + } + for _, m := range nm { + if err := leb128.WriteVarUint32(buf, m.Index); err != nil { + return err + } + if err := writeByteVector(buf, []byte(m.Name)); err != nil { + return err + } + } + return nil +} + +func writeCustomSection(w io.Writer, s module.CustomSection) error { + + if err := writeByte(w, constant.CustomSectionID); err != nil { + return err + } + + var buf bytes.Buffer + if err := writeByteVector(&buf, []byte(s.Name)); err != nil { + return err + } + + if _, err := io.Copy(&buf, bytes.NewReader(s.Data)); err != nil { + return err + } + + return writeRawSection(w, &buf) +} + +func writeFunctionType(w io.Writer, fsig module.FunctionType) error { + + if err := writeByte(w, constant.FunctionTypeID); err != nil { + return err + } + + if err := writeValueTypeVector(w, fsig.Params); err != nil { + return err + } + + return writeValueTypeVector(w, fsig.Results) +} + +func writeImport(w io.Writer, imp module.Import) error { + + if err := writeByteVector(w, []byte(imp.Module)); err != nil { + return err + } + + if err := writeByteVector(w, []byte(imp.Name)); err != nil { + return err + } + + switch desc := imp.Descriptor.(type) { + case module.FunctionImport: + if err := writeByte(w, constant.ImportDescType); err != nil { + return err + } + return leb128.WriteVarUint32(w, desc.Func) + case module.TableImport: + if err := writeByte(w, constant.ImportDescTable); err != nil { + return err + } + if err := writeByte(w, constant.ElementTypeAnyFunc); err != nil { + return err + } + return writeLimits(w, desc.Lim) + case module.MemoryImport: + if err := writeByte(w, constant.ImportDescMem); err != nil { + return err + } + return writeLimits(w, desc.Mem.Lim) + case module.GlobalImport: + if err := writeByte(w, constant.ImportDescGlobal); err != nil { + return err + } + if err := writeValueType(w, desc.Type); err != nil { + return err + } + if desc.Mutable { + return writeByte(w, constant.Mutable) + } + return writeByte(w, constant.Const) + default: + return fmt.Errorf("illegal import descriptor type") + } +} + +func writeGlobal(w io.Writer, global module.Global) error { + + if err := writeValueType(w, global.Type); err != nil { + return err + } + + var err error + + if global.Mutable { + err = writeByte(w, constant.Mutable) + } else { + err = writeByte(w, constant.Const) + } + + if err != nil { + return err + } + + return writeInstructions(w, global.Init.Instrs) +} + +func writeInstructions(w io.Writer, instrs []instruction.Instruction) error { + + for i, instr := range instrs { + + _, err := w.Write([]byte{byte(instr.Op())}) + if err != nil { + return err + } + + for _, arg := range instr.ImmediateArgs() { + var err error + switch arg := arg.(type) { + case int32: + err = leb128.WriteVarInt32(w, arg) + case int64: + err = leb128.WriteVarInt64(w, arg) + case uint32: + err = leb128.WriteVarUint32(w, arg) + case uint64: + err = leb128.WriteVarUint64(w, arg) + case float32: + u32 := math.Float32bits(arg) + err = binary.Write(w, binary.LittleEndian, u32) + case float64: + u64 := math.Float64bits(arg) + err = binary.Write(w, binary.LittleEndian, u64) + case byte: + _, err = w.Write([]byte{arg}) + default: + return fmt.Errorf("illegal immediate argument type on instruction %d", i) + } + if err != nil { + return err + } + } + + if si, ok := instr.(instruction.StructuredInstruction); ok { + if err := writeBlockValueType(w, si.BlockType()); err != nil { + return err + } + if err := writeInstructions(w, si.Instructions()); err != nil { + return err + } + } + + } + + _, err := w.Write([]byte{byte(opcode.End)}) + return err +} + +func writeLimits(w io.Writer, lim module.Limit) error { + if lim.Max == nil { + if err := writeByte(w, 0); err != nil { + return err + } + } else { + if err := writeByte(w, 1); err != nil { + return err + } + } + if err := leb128.WriteVarUint32(w, lim.Min); err != nil { + return err + } + if lim.Max != nil { + return leb128.WriteVarUint32(w, *lim.Max) + } + return nil +} + +func writeVarUint32Vector(w io.Writer, v []uint32) error { + + if err := leb128.WriteVarUint32(w, uint32(len(v))); err != nil { + return err + } + + for i := range v { + if err := leb128.WriteVarUint32(w, v[i]); err != nil { + return err + } + } + + return nil +} + +func writeValueTypeVector(w io.Writer, v []types.ValueType) error { + + if err := leb128.WriteVarUint32(w, uint32(len(v))); err != nil { + return err + } + + for i := range v { + if err := writeValueType(w, v[i]); err != nil { + return err + } + } + + return nil +} + +func writeBlockValueType(w io.Writer, v *types.ValueType) error { + var b byte + if v != nil { + switch *v { + case types.I32: + b = constant.ValueTypeI32 + case types.I64: + b = constant.ValueTypeI64 + case types.F32: + b = constant.ValueTypeF32 + case types.F64: + b = constant.ValueTypeF64 + } + } else { + b = constant.BlockTypeEmpty + } + return writeByte(w, b) +} + +func writeValueType(w io.Writer, v types.ValueType) error { + var b byte + switch v { + case types.I32: + b = constant.ValueTypeI32 + case types.I64: + b = constant.ValueTypeI64 + case types.F32: + b = constant.ValueTypeF32 + case types.F64: + b = constant.ValueTypeF64 + } + return writeByte(w, b) +} + +func writeRawSection(w io.Writer, buf *bytes.Buffer) error { + + size := buf.Len() + + if err := leb128.WriteVarUint32(w, uint32(size)); err != nil { + return err + } + + _, err := io.Copy(w, buf) + return err +} + +func writeByteVector(w io.Writer, bs []byte) error { + + if err := leb128.WriteVarUint32(w, uint32(len(bs))); err != nil { + return err + } + + _, err := w.Write(bs) + return err +} + +func writeByte(w io.Writer, b byte) error { + buf := make([]byte, 1) + buf[0] = b + _, err := w.Write(buf) + return err +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/wasm/instruction/control.go b/vendor/github.com/open-policy-agent/opa/internal/wasm/instruction/control.go new file mode 100644 index 00000000..38f03098 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/wasm/instruction/control.go @@ -0,0 +1,183 @@ +// Copyright 2018 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package instruction + +import ( + "github.com/open-policy-agent/opa/internal/wasm/opcode" + "github.com/open-policy-agent/opa/internal/wasm/types" +) + +// !!! If you find yourself adding support for more control +// instructions (br_table, if, ...), please adapt the +// `withControlInstr` functions of +// `compiler/wasm/optimizations.go` + +// Unreachable represents a WASM unreachable instruction. +type Unreachable struct { + NoImmediateArgs +} + +// Op returns the opcode of the instruction. +func (Unreachable) Op() opcode.Opcode { + return opcode.Unreachable +} + +// Nop represents a WASM no-op instruction. +type Nop struct { + NoImmediateArgs +} + +// Op returns the opcode of the instruction. +func (Nop) Op() opcode.Opcode { + return opcode.Nop +} + +// Block represents a WASM block instruction. +type Block struct { + NoImmediateArgs + Type *types.ValueType + Instrs []Instruction +} + +// Op returns the opcode of the instruction +func (Block) Op() opcode.Opcode { + return opcode.Block +} + +// BlockType returns the type of the block's return value. +func (i Block) BlockType() *types.ValueType { + return i.Type +} + +// Instructions returns the instructions contained in the block. +func (i Block) Instructions() []Instruction { + return i.Instrs +} + +// If represents a WASM if instruction. +// NOTE(sr): we only use if with one branch so far! +type If struct { + NoImmediateArgs + Type *types.ValueType + Instrs []Instruction +} + +// Op returns the opcode of the instruction. +func (If) Op() opcode.Opcode { + return opcode.If +} + +// BlockType returns the type of the if's THEN branch. +func (i If) BlockType() *types.ValueType { + return i.Type +} + +// Instructions represents the instructions contained in the if's THEN branch. +func (i If) Instructions() []Instruction { + return i.Instrs +} + +// Loop represents a WASM loop instruction. +type Loop struct { + NoImmediateArgs + Type *types.ValueType + Instrs []Instruction +} + +// Op returns the opcode of the instruction. +func (Loop) Op() opcode.Opcode { + return opcode.Loop +} + +// BlockType returns the type of the loop's return value. +func (i Loop) BlockType() *types.ValueType { + return i.Type +} + +// Instructions represents the instructions contained in the loop. +func (i Loop) Instructions() []Instruction { + return i.Instrs +} + +// Br represents a WASM br instruction. +type Br struct { + Index uint32 +} + +// Op returns the opcode of the instruction. +func (Br) Op() opcode.Opcode { + return opcode.Br +} + +// ImmediateArgs returns the block index to break to. +func (i Br) ImmediateArgs() []interface{} { + return []interface{}{i.Index} +} + +// BrIf represents a WASM br_if instruction. +type BrIf struct { + Index uint32 +} + +// Op returns the opcode of the instruction. +func (BrIf) Op() opcode.Opcode { + return opcode.BrIf +} + +// ImmediateArgs returns the block index to break to. +func (i BrIf) ImmediateArgs() []interface{} { + return []interface{}{i.Index} +} + +// Call represents a WASM call instruction. +type Call struct { + Index uint32 +} + +// Op returns the opcode of the instruction. +func (Call) Op() opcode.Opcode { + return opcode.Call +} + +// ImmediateArgs returns the function index. +func (i Call) ImmediateArgs() []interface{} { + return []interface{}{i.Index} +} + +// CallIndirect represents a WASM call_indirect instruction. +type CallIndirect struct { + Index uint32 // type index + Reserved byte // zero for now +} + +// Op returns the opcode of the instruction. +func (CallIndirect) Op() opcode.Opcode { + return opcode.CallIndirect +} + +// ImmediateArgs returns the function index. +func (i CallIndirect) ImmediateArgs() []interface{} { + return []interface{}{i.Index, i.Reserved} +} + +// Return represents a WASM return instruction. +type Return struct { + NoImmediateArgs +} + +// Op returns the opcode of the instruction. +func (Return) Op() opcode.Opcode { + return opcode.Return +} + +// End represents the special WASM end instruction. +type End struct { + NoImmediateArgs +} + +// Op returns the opcode of the instruction. +func (End) Op() opcode.Opcode { + return opcode.End +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/wasm/instruction/instruction.go b/vendor/github.com/open-policy-agent/opa/internal/wasm/instruction/instruction.go new file mode 100644 index 00000000..066be77c --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/wasm/instruction/instruction.go @@ -0,0 +1,33 @@ +// Copyright 2018 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +// Package instruction defines WASM instruction types. +package instruction + +import ( + "github.com/open-policy-agent/opa/internal/wasm/opcode" + "github.com/open-policy-agent/opa/internal/wasm/types" +) + +// NoImmediateArgs indicates the instruction has no immediate arguments. +type NoImmediateArgs struct { +} + +// ImmediateArgs returns the immedate arguments of an instruction. +func (NoImmediateArgs) ImmediateArgs() []interface{} { + return nil +} + +// Instruction represents a single WASM instruction. +type Instruction interface { + Op() opcode.Opcode + ImmediateArgs() []interface{} +} + +// StructuredInstruction represents a structured control instruction like br_if. +type StructuredInstruction interface { + Instruction + BlockType() *types.ValueType + Instructions() []Instruction +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/wasm/instruction/memory.go b/vendor/github.com/open-policy-agent/opa/internal/wasm/instruction/memory.go new file mode 100644 index 00000000..c449cb1b --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/wasm/instruction/memory.go @@ -0,0 +1,39 @@ +// Copyright 2019 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package instruction + +import "github.com/open-policy-agent/opa/internal/wasm/opcode" + +// I32Load represents the WASM i32.load instruction. +type I32Load struct { + Offset int32 + Align int32 // expressed as a power of two +} + +// Op returns the opcode of the instruction. +func (I32Load) Op() opcode.Opcode { + return opcode.I32Load +} + +// ImmediateArgs returns the static offset and alignment operands. +func (i I32Load) ImmediateArgs() []interface{} { + return []interface{}{i.Align, i.Offset} +} + +// I32Store represents the WASM i32.store instruction. +type I32Store struct { + Offset int32 + Align int32 // expressed as a power of two +} + +// Op returns the opcode of the instruction. +func (I32Store) Op() opcode.Opcode { + return opcode.I32Store +} + +// ImmediateArgs returns the static offset and alignment operands. +func (i I32Store) ImmediateArgs() []interface{} { + return []interface{}{i.Align, i.Offset} +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/wasm/instruction/numeric.go b/vendor/github.com/open-policy-agent/opa/internal/wasm/instruction/numeric.go new file mode 100644 index 00000000..03f33752 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/wasm/instruction/numeric.go @@ -0,0 +1,199 @@ +// Copyright 2018 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package instruction + +import ( + "github.com/open-policy-agent/opa/internal/wasm/opcode" +) + +// I32Const represents the WASM i32.const instruction. +type I32Const struct { + Value int32 +} + +// Op returns the opcode of the instruction. +func (I32Const) Op() opcode.Opcode { + return opcode.I32Const +} + +// ImmediateArgs returns the i32 value to push onto the stack. +func (i I32Const) ImmediateArgs() []interface{} { + return []interface{}{i.Value} +} + +// I64Const represents the WASM i64.const instruction. +type I64Const struct { + Value int64 +} + +// Op returns the opcode of the instruction. +func (I64Const) Op() opcode.Opcode { + return opcode.I64Const +} + +// ImmediateArgs returns the i64 value to push onto the stack. +func (i I64Const) ImmediateArgs() []interface{} { + return []interface{}{i.Value} +} + +// F32Const represents the WASM f32.const instruction. +type F32Const struct { + Value int32 +} + +// Op returns the opcode of the instruction. +func (F32Const) Op() opcode.Opcode { + return opcode.F32Const +} + +// ImmediateArgs returns the f32 value to push onto the stack. +func (i F32Const) ImmediateArgs() []interface{} { + return []interface{}{i.Value} +} + +// F64Const represents the WASM f64.const instruction. +type F64Const struct { + Value float64 +} + +// Op returns the opcode of the instruction. +func (F64Const) Op() opcode.Opcode { + return opcode.F64Const +} + +// ImmediateArgs returns the f64 value to push onto the stack. +func (i F64Const) ImmediateArgs() []interface{} { + return []interface{}{i.Value} +} + +// I32Eqz represents the WASM i32.eqz instruction. +type I32Eqz struct { + NoImmediateArgs +} + +// Op returns the opcode of the instruction. +func (I32Eqz) Op() opcode.Opcode { + return opcode.I32Eqz +} + +// I32Eq represents the WASM i32.eq instruction. +type I32Eq struct { + NoImmediateArgs +} + +// Op returns the opcode of the instruction. +func (I32Eq) Op() opcode.Opcode { + return opcode.I32Eq +} + +// I32Ne represents the WASM i32.ne instruction. +type I32Ne struct { + NoImmediateArgs +} + +// Op returns the opcode of the instruction. +func (I32Ne) Op() opcode.Opcode { + return opcode.I32Ne +} + +// I32GtS represents the WASM i32.gt_s instruction. +type I32GtS struct { + NoImmediateArgs +} + +// Op returns the opcode of the instruction. +func (I32GtS) Op() opcode.Opcode { + return opcode.I32GtS +} + +// I32GeS represents the WASM i32.ge_s instruction. +type I32GeS struct { + NoImmediateArgs +} + +// Op returns the opcode of the instruction. +func (I32GeS) Op() opcode.Opcode { + return opcode.I32GeS +} + +// I32LtS represents the WASM i32.lt_s instruction. +type I32LtS struct { + NoImmediateArgs +} + +// Op returns the opcode of the instruction. +func (I32LtS) Op() opcode.Opcode { + return opcode.I32LtS +} + +// I32LeS represents the WASM i32.le_s instruction. +type I32LeS struct { + NoImmediateArgs +} + +// Op returns the opcode of the instruction. +func (I32LeS) Op() opcode.Opcode { + return opcode.I32LeS +} + +// I32Add represents the WASM i32.add instruction. +type I32Add struct { + NoImmediateArgs +} + +// Op returns the opcode of the instruction. +func (I32Add) Op() opcode.Opcode { + return opcode.I32Add +} + +// I64Add represents the WASM i64.add instruction. +type I64Add struct { + NoImmediateArgs +} + +// Op returns the opcode of the instruction. +func (I64Add) Op() opcode.Opcode { + return opcode.I64Add +} + +// F32Add represents the WASM f32.add instruction. +type F32Add struct { + NoImmediateArgs +} + +// Op returns the opcode of the instruction. +func (F32Add) Op() opcode.Opcode { + return opcode.F32Add +} + +// F64Add represents the WASM f64.add instruction. +type F64Add struct { + NoImmediateArgs +} + +// Op returns the opcode of the instruction. +func (F64Add) Op() opcode.Opcode { + return opcode.F64Add +} + +// I32Mul represents the WASM i32.mul instruction. +type I32Mul struct { + NoImmediateArgs +} + +// Op returns the opcode of the instruction. +func (I32Mul) Op() opcode.Opcode { + return opcode.I32Mul +} + +// I32Sub represents the WASM i32.sub instruction. +type I32Sub struct { + NoImmediateArgs +} + +// Op returns the opcode of the instruction. +func (I32Sub) Op() opcode.Opcode { + return opcode.I32Sub +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/wasm/instruction/parametric.go b/vendor/github.com/open-policy-agent/opa/internal/wasm/instruction/parametric.go new file mode 100644 index 00000000..8b1abfa9 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/wasm/instruction/parametric.go @@ -0,0 +1,29 @@ +// Copyright 2021 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package instruction + +import ( + "github.com/open-policy-agent/opa/internal/wasm/opcode" +) + +// Drop reprsents a WASM drop instruction. +type Drop struct { + NoImmediateArgs +} + +// Op returns the opcode of the instruction. +func (Drop) Op() opcode.Opcode { + return opcode.Drop +} + +// Select reprsents a WASM select instruction. +type Select struct { + NoImmediateArgs +} + +// Op returns the opcode of the instruction. +func (Select) Op() opcode.Opcode { + return opcode.Select +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/wasm/instruction/variable.go b/vendor/github.com/open-policy-agent/opa/internal/wasm/instruction/variable.go new file mode 100644 index 00000000..063ffdb9 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/wasm/instruction/variable.go @@ -0,0 +1,54 @@ +// Copyright 2018 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package instruction + +import "github.com/open-policy-agent/opa/internal/wasm/opcode" + +// GetLocal represents the WASM get_local instruction. +type GetLocal struct { + Index uint32 +} + +// Op returns the opcode of the instruction. +func (GetLocal) Op() opcode.Opcode { + return opcode.GetLocal +} + +// ImmediateArgs returns the index of the local variable to push onto the stack. +func (i GetLocal) ImmediateArgs() []interface{} { + return []interface{}{i.Index} +} + +// SetLocal represents the WASM set_local instruction. +type SetLocal struct { + Index uint32 +} + +// Op returns the opcode of the instruction. +func (SetLocal) Op() opcode.Opcode { + return opcode.SetLocal +} + +// ImmediateArgs returns the index of the local variable to set with the top of +// the stack. +func (i SetLocal) ImmediateArgs() []interface{} { + return []interface{}{i.Index} +} + +// TeeLocal represents the WASM tee_local instruction. +type TeeLocal struct { + Index uint32 +} + +// Op returns the opcode of the instruction. +func (TeeLocal) Op() opcode.Opcode { + return opcode.TeeLocal +} + +// ImmediateArgs returns the index of the local variable to "tee" with the top of +// the stack (like set, but retaining the top of the stack). +func (i TeeLocal) ImmediateArgs() []interface{} { + return []interface{}{i.Index} +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/wasm/module/module.go b/vendor/github.com/open-policy-agent/opa/internal/wasm/module/module.go new file mode 100644 index 00000000..913863c1 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/wasm/module/module.go @@ -0,0 +1,385 @@ +// Copyright 2018 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package module + +import ( + "fmt" + "strings" + + "github.com/open-policy-agent/opa/internal/wasm/instruction" + "github.com/open-policy-agent/opa/internal/wasm/types" +) + +type ( + // Module represents a WASM module. + Module struct { + Version uint32 + Start StartSection + Type TypeSection + Import ImportSection + Function FunctionSection + Table TableSection + Memory MemorySection + Element ElementSection + Global GlobalSection + Export ExportSection + Code RawCodeSection + Data DataSection + Customs []CustomSection + Names NameSection + } + + // StartSection represents a WASM start section. + StartSection struct { + FuncIndex *uint32 + } + + // TypeSection represents a WASM type section. + TypeSection struct { + Functions []FunctionType + } + + // ImportSection represents a WASM import section. + ImportSection struct { + Imports []Import + } + + // FunctionSection represents a WASM function section. + FunctionSection struct { + TypeIndices []uint32 + } + + // TableSection represents a WASM table section. + TableSection struct { + Tables []Table + } + + // MemorySection represents a Wasm memory section. + MemorySection struct { + Memories []Memory + } + + // ElementSection represents a WASM element section. + ElementSection struct { + Segments []ElementSegment + } + + // GlobalSection represents a WASM global section. + GlobalSection struct { + Globals []Global + } + + // ExportSection represents a WASM export section. + ExportSection struct { + Exports []Export + } + + // RawCodeSection represents a WASM code section. The code section is left as a + // raw byte sequence. + RawCodeSection struct { + Segments []RawCodeSegment + } + + // DataSection represents a WASM data section. + DataSection struct { + Segments []DataSegment + } + + // CustomSection represents a WASM custom section. + CustomSection struct { + Name string + Data []byte + } + + // NameSection represents the WASM custom section "name". + NameSection struct { + Module string + Functions []NameMap + Locals []LocalNameMap + } + + // NameMap maps function or local arg indices to their names. + NameMap struct { + Index uint32 + Name string + } + + // LocalNameMap maps function indices, and argument indices for the args + // of the indexed function to their names. + LocalNameMap struct { + FuncIndex uint32 + NameMap + } + + // FunctionType represents a WASM function type definition. + FunctionType struct { + Params []types.ValueType + Results []types.ValueType + } + + // Import represents a WASM import statement. + Import struct { + Module string + Name string + Descriptor ImportDescriptor + } + + // ImportDescriptor represents a WASM import descriptor. + ImportDescriptor interface { + fmt.Stringer + Kind() ImportDescriptorType + } + + // ImportDescriptorType defines allowed kinds of import descriptors. + ImportDescriptorType int + + // FunctionImport represents a WASM function import statement. + FunctionImport struct { + Func uint32 + } + + // MemoryImport represents a WASM memory import statement. + MemoryImport struct { + Mem MemType + } + + // MemType defines the attributes of a memory import. + MemType struct { + Lim Limit + } + + // TableImport represents a WASM table import statement. + TableImport struct { + Type types.ElementType + Lim Limit + } + + // ElementSegment represents a WASM element segment. + ElementSegment struct { + Index uint32 + Offset Expr + Indices []uint32 + } + + // GlobalImport represents a WASM global variable import statement. + GlobalImport struct { + Type types.ValueType + Mutable bool + } + + // Limit represents a WASM limit. + Limit struct { + Min uint32 + Max *uint32 + } + + // Table represents a WASM table statement. + Table struct { + Type types.ElementType + Lim Limit + } + + // Memory represents a Wasm memory statement. + Memory struct { + Lim Limit + } + + // Global represents a WASM global statement. + Global struct { + Type types.ValueType + Mutable bool + Init Expr + } + + // Export represents a WASM export statement. + Export struct { + Name string + Descriptor ExportDescriptor + } + + // ExportDescriptor represents a WASM export descriptor. + ExportDescriptor struct { + Type ExportDescriptorType + Index uint32 + } + + // ExportDescriptorType defines the allowed kinds of export descriptors. + ExportDescriptorType int + + // RawCodeSegment represents a binary-encoded WASM code segment. + RawCodeSegment struct { + Code []byte + } + + // DataSegment represents a WASM data segment. + DataSegment struct { + Index uint32 + Offset Expr + Init []byte + } + + // Expr represents a WASM expression. + Expr struct { + Instrs []instruction.Instruction + } + + // CodeEntry represents a code segment entry. + CodeEntry struct { + Func Function + } + + // Function represents a function in a code segment. + Function struct { + Locals []LocalDeclaration + Expr Expr + } + + // LocalDeclaration represents a local variable declaration. + LocalDeclaration struct { + Count uint32 + Type types.ValueType + } +) + +// Defines the allowed kinds of imports. +const ( + FunctionImportType ImportDescriptorType = iota + TableImportType + MemoryImportType + GlobalImportType +) + +func (x ImportDescriptorType) String() string { + switch x { + case FunctionImportType: + return "func" + case TableImportType: + return "table" + case MemoryImportType: + return "memory" + case GlobalImportType: + return "global" + } + panic("illegal value") +} + +// Defines the allowed kinds of exports. +const ( + FunctionExportType ExportDescriptorType = iota + TableExportType + MemoryExportType + GlobalExportType +) + +func (x ExportDescriptorType) String() string { + switch x { + case FunctionExportType: + return "func" + case TableExportType: + return "table" + case MemoryExportType: + return "memory" + case GlobalExportType: + return "global" + } + panic("illegal value") +} + +// Kind returns the function import type kind. +func (i FunctionImport) Kind() ImportDescriptorType { + return FunctionImportType +} + +func (i FunctionImport) String() string { + return fmt.Sprintf("%v[type=%v]", i.Kind(), i.Func) +} + +// Kind returns the memory import type kind. +func (i MemoryImport) Kind() ImportDescriptorType { + return MemoryImportType +} + +func (i MemoryImport) String() string { + return fmt.Sprintf("%v[%v]", i.Kind(), i.Mem.Lim) +} + +// Kind returns the table import type kind. +func (i TableImport) Kind() ImportDescriptorType { + return TableImportType +} + +func (i TableImport) String() string { + return fmt.Sprintf("%v[%v, %v]", i.Kind(), i.Type, i.Lim) +} + +// Kind returns the global import type kind. +func (i GlobalImport) Kind() ImportDescriptorType { + return GlobalImportType +} + +func (i GlobalImport) String() string { + return fmt.Sprintf("%v[%v, mut=%v]", i.Kind(), i.Type, i.Mutable) +} + +func (tpe FunctionType) String() string { + params := make([]string, len(tpe.Params)) + results := make([]string, len(tpe.Results)) + for i := range tpe.Params { + params[i] = tpe.Params[i].String() + } + for i := range tpe.Results { + results[i] = tpe.Results[i].String() + } + return "(" + strings.Join(params, ", ") + ") -> (" + strings.Join(results, ", ") + ")" +} + +// Equal returns true if tpe equals other. +func (tpe FunctionType) Equal(other FunctionType) bool { + + if len(tpe.Params) != len(other.Params) || len(tpe.Results) != len(other.Results) { + return false + } + + for i := range tpe.Params { + if tpe.Params[i] != other.Params[i] { + return false + } + } + + for i := range tpe.Results { + if tpe.Results[i] != other.Results[i] { + return false + } + } + + return true +} + +func (imp Import) String() string { + return fmt.Sprintf("%v %v.%v", imp.Descriptor.String(), imp.Module, imp.Name) +} + +func (exp Export) String() string { + return fmt.Sprintf("%v[%v] %v", exp.Descriptor.Type, exp.Descriptor.Index, exp.Name) +} + +func (seg RawCodeSegment) String() string { + return fmt.Sprintf("", len(seg.Code)) +} + +func (seg DataSegment) String() string { + return fmt.Sprintf("", seg.Index, seg.Offset, len(seg.Init)) +} + +func (e Expr) String() string { + return fmt.Sprintf("%d instr(s)", len(e.Instrs)) +} + +func (lim Limit) String() string { + if lim.Max == nil { + return fmt.Sprintf("min=%v", lim.Min) + } + return fmt.Sprintf("min=%v max=%v", lim.Min, lim.Max) +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/wasm/module/pretty.go b/vendor/github.com/open-policy-agent/opa/internal/wasm/module/pretty.go new file mode 100644 index 00000000..2b28ad85 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/wasm/module/pretty.go @@ -0,0 +1,84 @@ +// Copyright 2018 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package module + +import ( + "encoding/hex" + "fmt" + "io" +) + +// PrettyOption defines options for controlling pretty printing. +type PrettyOption struct { + Contents bool // show raw byte content of data+code sections. +} + +// Pretty writes a human-readable representation of m to w. +func Pretty(w io.Writer, m *Module, opts ...PrettyOption) { + fmt.Fprintln(w, "version:", m.Version) + fmt.Fprintln(w, "types:") + for _, fn := range m.Type.Functions { + fmt.Fprintln(w, " -", fn) + } + fmt.Fprintln(w, "imports:") + for i, imp := range m.Import.Imports { + if imp.Descriptor.Kind() == FunctionImportType { + fmt.Printf(" - [%d] %v\n", i, imp) + } else { + fmt.Fprintln(w, " -", imp) + } + } + fmt.Fprintln(w, "functions:") + for _, fn := range m.Function.TypeIndices { + if fn >= uint32(len(m.Type.Functions)) { + fmt.Fprintln(w, " -", "???") + } else { + fmt.Fprintln(w, " -", m.Type.Functions[fn]) + } + } + fmt.Fprintln(w, "exports:") + for _, exp := range m.Export.Exports { + fmt.Fprintln(w, " -", exp) + } + fmt.Fprintln(w, "code:") + for _, seg := range m.Code.Segments { + fmt.Fprintln(w, " -", seg) + } + fmt.Fprintln(w, "data:") + for _, seg := range m.Data.Segments { + fmt.Fprintln(w, " -", seg) + } + if len(opts) == 0 { + return + } + fmt.Fprintln(w) + for _, opt := range opts { + if opt.Contents { + newline := false + if len(m.Data.Segments) > 0 { + fmt.Fprintln(w, "data section:") + for _, seg := range m.Data.Segments { + if newline { + fmt.Fprintln(w) + } + fmt.Fprintln(w, hex.Dump(seg.Init)) + newline = true + } + newline = false + } + if len(m.Code.Segments) > 0 { + fmt.Fprintln(w, "code section:") + for _, seg := range m.Code.Segments { + if newline { + fmt.Fprintln(w) + } + fmt.Fprintln(w, hex.Dump(seg.Code)) + newline = true + } + newline = false + } + } + } +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/wasm/opcode/opcode.go b/vendor/github.com/open-policy-agent/opa/internal/wasm/opcode/opcode.go new file mode 100644 index 00000000..7d35a301 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/wasm/opcode/opcode.go @@ -0,0 +1,218 @@ +// Copyright 2018 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +// Package opcode contains constants and utilities for working with WASM opcodes. +package opcode + +// Opcode represents a WASM instruction opcode. +type Opcode byte + +// Control instructions. +const ( + Unreachable Opcode = iota + Nop + Block + Loop + If + Else +) + +const ( + // End defines the special end WASM opcode. + End Opcode = 0x0B +) + +// Extended control instructions. +const ( + Br Opcode = iota + 0x0C + BrIf + BrTable + Return + Call + CallIndirect +) + +// Parameter instructions. +const ( + Drop Opcode = iota + 0x1A + Select +) + +// Variable instructions. +const ( + GetLocal Opcode = iota + 0x20 + SetLocal + TeeLocal + GetGlobal + SetGlobal +) + +// Memory instructions. +const ( + I32Load Opcode = iota + 0x28 + I64Load + F32Load + F64Load + I32Load8S + I32Load8U + I32Load16S + I32Load16U + I64Load8S + I64Load8U + I64Load16S + I64Load16U + I64Load32S + I64Load32U + I32Store + I64Store + F32Store + F64Store + I32Store8 + I32Store16 + I64Store8 + I64Store16 + I64Store32 + MemorySize + MemoryGrow +) + +// Numeric instructions. +const ( + I32Const Opcode = iota + 0x41 + I64Const + F32Const + F64Const + + I32Eqz + I32Eq + I32Ne + I32LtS + I32LtU + I32GtS + I32GtU + I32LeS + I32LeU + I32GeS + I32GeU + + I64Eqz + I64Eq + I64Ne + I64LtS + I64LtU + I64GtS + I64GtU + I64LeS + I64LeU + I64GeS + I64GeU + + F32Eq + F32Ne + F32Lt + F32Gt + F32Le + F32Ge + + F64Eq + F64Ne + F64Lt + F64Gt + F64Le + F64Ge + + I32Clz + I32Ctz + I32Popcnt + I32Add + I32Sub + I32Mul + I32DivS + I32DivU + I32RemS + I32RemU + I32And + I32Or + I32Xor + I32Shl + I32ShrS + I32ShrU + I32Rotl + I32Rotr + + I64Clz + I64Ctz + I64Popcnt + I64Add + I64Sub + I64Mul + I64DivS + I64DivU + I64RemS + I64RemU + I64And + I64Or + I64Xor + I64Shl + I64ShrS + I64ShrU + I64Rotl + I64Rotr + + F32Abs + F32Neg + F32Ceil + F32Floor + F32Trunc + F32Nearest + F32Sqrt + F32Add + F32Sub + F32Mul + F32Div + F32Min + F32Max + F32Copysign + + F64Abs + F64Neg + F64Ceil + F64Floor + F64Trunc + F64Nearest + F64Sqrt + F64Add + F64Sub + F64Mul + F64Div + F64Min + F64Max + F64Copysign + + I32WrapI64 + I32TruncSF32 + I32TruncUF32 + I32TruncSF64 + I32TruncUF64 + I64ExtendSI32 + I64ExtendUI32 + I64TruncSF32 + I64TruncUF32 + I64TruncSF64 + I64TruncUF64 + F32ConvertSI32 + F32ConvertUI32 + F32ConvertSI64 + F32ConvertUI64 + F32DemoteF64 + F64ConvertSI32 + F64ConvertUI32 + F64ConvertSI64 + F64ConvertUI64 + F64PromoteF32 + I32ReinterpretF32 + I64ReinterpretF64 + F32ReinterpretI32 + F64ReinterpretI64 +) diff --git a/vendor/github.com/open-policy-agent/opa/internal/wasm/sdk/opa/capabilities/capabilities.go b/vendor/github.com/open-policy-agent/opa/internal/wasm/sdk/opa/capabilities/capabilities.go new file mode 100644 index 00000000..9f939284 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/wasm/sdk/opa/capabilities/capabilities.go @@ -0,0 +1,12 @@ +// Copyright 2021 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +// +build opa_wasm generate + +package capabilities + +// ABIVersions returns the ABI versions that this SDK supports +func ABIVersions() [][2]int { + return [][2]int{{1, 1}, {1, 2}} +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/wasm/sdk/opa/capabilities/capabilities_nowasm.go b/vendor/github.com/open-policy-agent/opa/internal/wasm/sdk/opa/capabilities/capabilities_nowasm.go new file mode 100644 index 00000000..6b17984b --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/wasm/sdk/opa/capabilities/capabilities_nowasm.go @@ -0,0 +1,14 @@ +// Copyright 2021 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +//go:build !opa_wasm && !generate +// +build !opa_wasm,!generate + +package capabilities + +// ABIVersions returns the supported Wasm ABI versions for this +// build: none +func ABIVersions() [][2]int { + return nil +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/wasm/types/types.go b/vendor/github.com/open-policy-agent/opa/internal/wasm/types/types.go new file mode 100644 index 00000000..4e2b7762 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/wasm/types/types.go @@ -0,0 +1,36 @@ +// Copyright 2018 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +// Package types defines the WASM value type constants. +package types + +// ValueType represents an intrinsic value in WASM. +type ValueType int + +// Defines the intrinsic value types. +const ( + I32 ValueType = iota + I64 + F32 + F64 +) + +func (tpe ValueType) String() string { + if tpe == I32 { + return "i32" + } else if tpe == I64 { + return "i64" + } else if tpe == F32 { + return "f32" + } + return "f64" +} + +// ElementType defines the type of table elements. +type ElementType int + +const ( + // Anyfunc is the union of all table types. + Anyfunc ElementType = iota +) diff --git a/vendor/github.com/open-policy-agent/opa/internal/wasm/util/util.go b/vendor/github.com/open-policy-agent/opa/internal/wasm/util/util.go new file mode 100644 index 00000000..42b81256 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/wasm/util/util.go @@ -0,0 +1,18 @@ +// Copyright 2020 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package util + +// PageSize represents the WASM page size in bytes. +const PageSize = 65535 + +// Pages converts a byte size to Pages, rounding up as necessary. +func Pages(n uint32) uint32 { + pages := n / PageSize + if pages*PageSize == n { + return pages + } + + return pages + 1 +} diff --git a/vendor/github.com/open-policy-agent/opa/keys/keys.go b/vendor/github.com/open-policy-agent/opa/keys/keys.go new file mode 100644 index 00000000..8272168e --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/keys/keys.go @@ -0,0 +1,100 @@ +package keys + +import ( + "encoding/json" + "fmt" + "io/ioutil" + "os" + + "github.com/open-policy-agent/opa/util" +) + +const defaultSigningAlgorithm = "RS256" + +var supportedAlgos = map[string]struct{}{ + "ES256": {}, "ES384": {}, "ES512": {}, + "HS256": {}, "HS384": {}, "HS512": {}, + "PS256": {}, "PS384": {}, "PS512": {}, + "RS256": {}, "RS384": {}, "RS512": {}, +} + +// IsSupportedAlgorithm true if provided alg is supported +func IsSupportedAlgorithm(alg string) bool { + _, ok := supportedAlgos[alg] + return ok +} + +// Config holds the keys used to sign or verify bundles and tokens +type Config struct { + Key string `json:"key"` + PrivateKey string `json:"private_key"` + Algorithm string `json:"algorithm"` + Scope string `json:"scope"` +} + +// Equal returns true if this key config is equal to the other. +func (k *Config) Equal(other *Config) bool { + return other != nil && *k == *other +} + +func (k *Config) validateAndInjectDefaults(id string) error { + if k.Key == "" && k.PrivateKey == "" { + return fmt.Errorf("invalid keys configuration: no keys provided for key ID %v", id) + } + + if k.Algorithm == "" { + k.Algorithm = defaultSigningAlgorithm + } + + if !IsSupportedAlgorithm(k.Algorithm) { + return fmt.Errorf("unsupported algorithm '%v'", k.Algorithm) + } + + return nil +} + +// NewKeyConfig return a new Config +func NewKeyConfig(key, alg, scope string) (*Config, error) { + var pubKey string + if _, err := os.Stat(key); err == nil { + bs, err := ioutil.ReadFile(key) + if err != nil { + return nil, err + } + pubKey = string(bs) + } else if os.IsNotExist(err) { + pubKey = key + } else { + return nil, err + } + + return &Config{ + Key: pubKey, + Algorithm: alg, + Scope: scope, + }, nil +} + +// ParseKeysConfig returns a map containing the key and the signing algorithm +func ParseKeysConfig(raw json.RawMessage) (map[string]*Config, error) { + keys := map[string]*Config{} + var obj map[string]json.RawMessage + + if err := util.Unmarshal(raw, &obj); err == nil { + for k := range obj { + var keyConfig Config + if err = util.Unmarshal(obj[k], &keyConfig); err != nil { + return nil, err + } + + if err = keyConfig.validateAndInjectDefaults(k); err != nil { + return nil, err + } + + keys[k] = &keyConfig + } + } else { + return nil, err + } + return keys, nil +} diff --git a/vendor/github.com/open-policy-agent/opa/loader/errors.go b/vendor/github.com/open-policy-agent/opa/loader/errors.go new file mode 100644 index 00000000..d6da40a0 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/loader/errors.go @@ -0,0 +1,62 @@ +// Copyright 2017 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package loader + +import ( + "fmt" + "strings" + + "github.com/open-policy-agent/opa/ast" +) + +// Errors is a wrapper for multiple loader errors. +type Errors []error + +func (e Errors) Error() string { + if len(e) == 0 { + return "no error(s)" + } + if len(e) == 1 { + return "1 error occurred during loading: " + e[0].Error() + } + buf := make([]string, len(e)) + for i := range buf { + buf[i] = e[i].Error() + } + return fmt.Sprintf("%v errors occurred during loading:\n", len(e)) + strings.Join(buf, "\n") +} + +func (e *Errors) add(err error) { + if errs, ok := err.(ast.Errors); ok { + for i := range errs { + *e = append(*e, errs[i]) + } + } else { + *e = append(*e, err) + } +} + +type unsupportedDocumentType string + +func (path unsupportedDocumentType) Error() string { + return string(path) + ": bad document type" +} + +type unrecognizedFile string + +func (path unrecognizedFile) Error() string { + return string(path) + ": can't recognize file type" +} + +func isUnrecognizedFile(err error) bool { + _, ok := err.(unrecognizedFile) + return ok +} + +type mergeError string + +func (e mergeError) Error() string { + return string(e) + ": merge error" +} diff --git a/vendor/github.com/open-policy-agent/opa/loader/loader.go b/vendor/github.com/open-policy-agent/opa/loader/loader.go new file mode 100644 index 00000000..d2f7cee6 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/loader/loader.go @@ -0,0 +1,670 @@ +// Copyright 2017 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +// Package loader contains utilities for loading files into OPA. +package loader + +import ( + "bytes" + "fmt" + "io/ioutil" + "os" + "path/filepath" + "sort" + "strings" + + "github.com/ghodss/yaml" + + "github.com/open-policy-agent/opa/ast" + "github.com/open-policy-agent/opa/bundle" + fileurl "github.com/open-policy-agent/opa/internal/file/url" + "github.com/open-policy-agent/opa/internal/merge" + "github.com/open-policy-agent/opa/metrics" + "github.com/open-policy-agent/opa/storage" + "github.com/open-policy-agent/opa/storage/inmem" + "github.com/open-policy-agent/opa/util" +) + +// Result represents the result of successfully loading zero or more files. +type Result struct { + Documents map[string]interface{} + Modules map[string]*RegoFile + path []string +} + +// ParsedModules returns the parsed modules stored on the result. +func (l *Result) ParsedModules() map[string]*ast.Module { + modules := make(map[string]*ast.Module) + for _, module := range l.Modules { + modules[module.Name] = module.Parsed + } + return modules +} + +// Compiler returns a Compiler object with the compiled modules from this loader +// result. +func (l *Result) Compiler() (*ast.Compiler, error) { + compiler := ast.NewCompiler() + compiler.Compile(l.ParsedModules()) + if compiler.Failed() { + return nil, compiler.Errors + } + return compiler, nil +} + +// Store returns a Store object with the documents from this loader result. +func (l *Result) Store() (storage.Store, error) { + return inmem.NewFromObject(l.Documents), nil +} + +// RegoFile represents the result of loading a single Rego source file. +type RegoFile struct { + Name string + Parsed *ast.Module + Raw []byte +} + +// Filter defines the interface for filtering files during loading. If the +// filter returns true, the file should be excluded from the result. +type Filter func(abspath string, info os.FileInfo, depth int) bool + +// GlobExcludeName excludes files and directories whose names do not match the +// shell style pattern at minDepth or greater. +func GlobExcludeName(pattern string, minDepth int) Filter { + return func(abspath string, info os.FileInfo, depth int) bool { + match, _ := filepath.Match(pattern, info.Name()) + return match && depth >= minDepth + } +} + +// FileLoader defines an interface for loading OPA data files +// and Rego policies. +type FileLoader interface { + All(paths []string) (*Result, error) + Filtered(paths []string, filter Filter) (*Result, error) + AsBundle(path string) (*bundle.Bundle, error) + WithMetrics(m metrics.Metrics) FileLoader + WithBundleVerificationConfig(*bundle.VerificationConfig) FileLoader + WithSkipBundleVerification(skipVerify bool) FileLoader + WithProcessAnnotation(processAnnotation bool) FileLoader +} + +// NewFileLoader returns a new FileLoader instance. +func NewFileLoader() FileLoader { + return &fileLoader{ + metrics: metrics.New(), + files: make(map[string]bundle.FileInfo), + } +} + +type fileLoader struct { + metrics metrics.Metrics + bvc *bundle.VerificationConfig + skipVerify bool + files map[string]bundle.FileInfo + opts ast.ParserOptions +} + +// WithMetrics provides the metrics instance to use while loading +func (fl *fileLoader) WithMetrics(m metrics.Metrics) FileLoader { + fl.metrics = m + return fl +} + +// WithBundleVerificationConfig sets the key configuration used to verify a signed bundle +func (fl *fileLoader) WithBundleVerificationConfig(config *bundle.VerificationConfig) FileLoader { + fl.bvc = config + return fl +} + +// WithSkipBundleVerification skips verification of a signed bundle +func (fl *fileLoader) WithSkipBundleVerification(skipVerify bool) FileLoader { + fl.skipVerify = skipVerify + return fl +} + +// WithProcessAnnotation enables or disables processing of schema annotations on rules +func (fl *fileLoader) WithProcessAnnotation(processAnnotation bool) FileLoader { + fl.opts.ProcessAnnotation = processAnnotation + return fl +} + +// All returns a Result object loaded (recursively) from the specified paths. +func (fl fileLoader) All(paths []string) (*Result, error) { + return fl.Filtered(paths, nil) +} + +// Filtered returns a Result object loaded (recursively) from the specified +// paths while applying the given filters. If any filter returns true, the +// file/directory is excluded. +func (fl fileLoader) Filtered(paths []string, filter Filter) (*Result, error) { + return all(paths, filter, func(curr *Result, path string, depth int) error { + + bs, err := ioutil.ReadFile(path) + if err != nil { + return err + } + + result, err := loadKnownTypes(path, bs, fl.metrics, fl.opts) + if err != nil { + if !isUnrecognizedFile(err) { + return err + } + if depth > 0 { + return nil + } + result, err = loadFileForAnyType(path, bs, fl.metrics, fl.opts) + if err != nil { + return err + } + } + + return curr.merge(path, result) + }) +} + +// AsBundle loads a path as a bundle. If it is a single file +// it will be treated as a normal tarball bundle. If a directory +// is supplied it will be loaded as an unzipped bundle tree. +func (fl fileLoader) AsBundle(path string) (*bundle.Bundle, error) { + path, err := fileurl.Clean(path) + if err != nil { + return nil, err + } + bundleLoader, isDir, err := GetBundleDirectoryLoader(path) + if err != nil { + return nil, err + } + + br := bundle.NewCustomReader(bundleLoader). + WithMetrics(fl.metrics). + WithBundleVerificationConfig(fl.bvc). + WithSkipBundleVerification(fl.skipVerify). + WithProcessAnnotations(fl.opts.ProcessAnnotation) + + // For bundle directories add the full path in front of module file names + // to simplify debugging. + if isDir { + br.WithBaseDir(path) + } + + b, err := br.Read() + if err != nil { + err = fmt.Errorf("bundle %s: %w", path, err) + } + + return &b, err +} + +// GetBundleDirectoryLoader returns a bundle directory loader which can be used to load +// files in the directory. +func GetBundleDirectoryLoader(path string) (bundle.DirectoryLoader, bool, error) { + path, err := fileurl.Clean(path) + if err != nil { + return nil, false, err + } + + fi, err := os.Stat(path) + if err != nil { + return nil, false, fmt.Errorf("error reading %q: %s", path, err) + } + + var bundleLoader bundle.DirectoryLoader + + if fi.IsDir() { + bundleLoader = bundle.NewDirectoryLoader(path) + } else { + fh, err := os.Open(path) + if err != nil { + return nil, false, err + } + bundleLoader = bundle.NewTarballLoaderWithBaseURL(fh, path) + } + return bundleLoader, fi.IsDir(), nil +} + +// FilteredPaths return a list of files from the specified +// paths while applying the given filters. If any filter returns true, the +// file/directory is excluded. +func FilteredPaths(paths []string, filter Filter) ([]string, error) { + result := []string{} + + _, err := all(paths, filter, func(_ *Result, path string, _ int) error { + result = append(result, path) + return nil + }) + if err != nil { + return nil, err + } + return result, nil +} + +// Schemas loads a schema set from the specified file path. +func Schemas(schemaPath string) (*ast.SchemaSet, error) { + + var errs Errors + ss, err := loadSchemas(schemaPath) + if err != nil { + errs.add(err) + return nil, errs + } + + return ss, nil +} + +func loadSchemas(schemaPath string) (*ast.SchemaSet, error) { + + if schemaPath == "" { + return nil, nil + } + + ss := ast.NewSchemaSet() + path, err := fileurl.Clean(schemaPath) + if err != nil { + return nil, err + } + + info, err := os.Stat(path) + if err != nil { + return nil, err + } + + // Handle single file case. + if !info.IsDir() { + schema, err := loadOneSchema(path) + if err != nil { + return nil, err + } + ss.Put(ast.SchemaRootRef, schema) + return ss, nil + + } + + // Handle directory case. + rootDir := path + + err = filepath.Walk(path, + func(path string, info os.FileInfo, err error) error { + if err != nil { + return err + } else if info.IsDir() { + return nil + } + + schema, err := loadOneSchema(path) + if err != nil { + return err + } + + relPath, err := filepath.Rel(rootDir, path) + if err != nil { + return err + } + + key := getSchemaSetByPathKey(relPath) + ss.Put(key, schema) + return nil + }) + + if err != nil { + return nil, err + } + + return ss, nil +} + +func getSchemaSetByPathKey(path string) ast.Ref { + + front := filepath.Dir(path) + last := strings.TrimSuffix(filepath.Base(path), filepath.Ext(path)) + + var parts []string + + if front != "." { + parts = append(strings.Split(filepath.ToSlash(front), "/"), last) + } else { + parts = []string{last} + } + + key := make(ast.Ref, 1+len(parts)) + key[0] = ast.SchemaRootDocument + for i := range parts { + key[i+1] = ast.StringTerm(parts[i]) + } + + return key +} + +func loadOneSchema(path string) (interface{}, error) { + bs, err := ioutil.ReadFile(path) + if err != nil { + return nil, err + } + + var schema interface{} + if err := util.Unmarshal(bs, &schema); err != nil { + return nil, fmt.Errorf("%s: %w", path, err) + } + + return schema, nil +} + +// All returns a Result object loaded (recursively) from the specified paths. +// Deprecated: Use FileLoader.Filtered() instead. +func All(paths []string) (*Result, error) { + return NewFileLoader().Filtered(paths, nil) +} + +// Filtered returns a Result object loaded (recursively) from the specified +// paths while applying the given filters. If any filter returns true, the +// file/directory is excluded. +// Deprecated: Use FileLoader.Filtered() instead. +func Filtered(paths []string, filter Filter) (*Result, error) { + return NewFileLoader().Filtered(paths, filter) +} + +// AsBundle loads a path as a bundle. If it is a single file +// it will be treated as a normal tarball bundle. If a directory +// is supplied it will be loaded as an unzipped bundle tree. +// Deprecated: Use FileLoader.AsBundle() instead. +func AsBundle(path string) (*bundle.Bundle, error) { + return NewFileLoader().AsBundle(path) +} + +// AllRegos returns a Result object loaded (recursively) with all Rego source +// files from the specified paths. +func AllRegos(paths []string) (*Result, error) { + return NewFileLoader().Filtered(paths, func(_ string, info os.FileInfo, depth int) bool { + return !info.IsDir() && !strings.HasSuffix(info.Name(), bundle.RegoExt) + }) +} + +// Rego is deprecated. Use RegoWithOpts instead. +func Rego(path string) (*RegoFile, error) { + return RegoWithOpts(path, ast.ParserOptions{}) +} + +// RegoWithOpts returns a RegoFile object loaded from the given path. +func RegoWithOpts(path string, opts ast.ParserOptions) (*RegoFile, error) { + path, err := fileurl.Clean(path) + if err != nil { + return nil, err + } + bs, err := ioutil.ReadFile(path) + if err != nil { + return nil, err + } + return loadRego(path, bs, metrics.New(), opts) +} + +// CleanPath returns the normalized version of a path that can be used as an identifier. +func CleanPath(path string) string { + return strings.Trim(path, "/") +} + +// Paths returns a sorted list of files contained at path. If recurse is true +// and path is a directory, then Paths will walk the directory structure +// recursively and list files at each level. +func Paths(path string, recurse bool) (paths []string, err error) { + path, err = fileurl.Clean(path) + if err != nil { + return nil, err + } + err = filepath.Walk(path, func(f string, info os.FileInfo, err error) error { + if !recurse { + if path != f && path != filepath.Dir(f) { + return filepath.SkipDir + } + } + paths = append(paths, f) + return nil + }) + return paths, err +} + +// Dirs resolves filepaths to directories. It will return a list of unique +// directories. +func Dirs(paths []string) []string { + unique := map[string]struct{}{} + + for _, path := range paths { + // TODO: /dir/dir will register top level directory /dir + dir := filepath.Dir(path) + unique[dir] = struct{}{} + } + + var u []string + for k := range unique { + u = append(u, k) + } + sort.Strings(u) + return u +} + +// SplitPrefix returns a tuple specifying the document prefix and the file +// path. +func SplitPrefix(path string) ([]string, string) { + // Non-prefixed URLs can be returned without modification and their contents + // can be rooted directly under data. + if strings.Index(path, "://") == strings.Index(path, ":") { + return nil, path + } + parts := strings.SplitN(path, ":", 2) + if len(parts) == 2 && len(parts[0]) > 0 { + return strings.Split(parts[0], "."), parts[1] + } + return nil, path +} + +func (l *Result) merge(path string, result interface{}) error { + switch result := result.(type) { + case bundle.Bundle: + for _, module := range result.Modules { + l.Modules[module.Path] = &RegoFile{ + Name: module.Path, + Parsed: module.Parsed, + Raw: module.Raw, + } + } + return l.mergeDocument(path, result.Data) + case *RegoFile: + l.Modules[CleanPath(path)] = result + return nil + default: + return l.mergeDocument(path, result) + } +} + +func (l *Result) mergeDocument(path string, doc interface{}) error { + obj, ok := makeDir(l.path, doc) + if !ok { + return unsupportedDocumentType(path) + } + merged, ok := merge.InterfaceMaps(l.Documents, obj) + if !ok { + return mergeError(path) + } + for k := range merged { + l.Documents[k] = merged[k] + } + return nil +} + +func (l *Result) withParent(p string) *Result { + path := append(l.path, p) + return &Result{ + Documents: l.Documents, + Modules: l.Modules, + path: path, + } +} + +func newResult() *Result { + return &Result{ + Documents: map[string]interface{}{}, + Modules: map[string]*RegoFile{}, + } +} + +func all(paths []string, filter Filter, f func(*Result, string, int) error) (*Result, error) { + errs := Errors{} + root := newResult() + + for _, path := range paths { + + // Paths can be prefixed with a string that specifies where content should be + // loaded under data. E.g., foo.bar:/path/to/some.json will load the content + // of some.json under {"foo": {"bar": ...}}. + loaded := root + prefix, path := SplitPrefix(path) + if len(prefix) > 0 { + for _, part := range prefix { + loaded = loaded.withParent(part) + } + } + + allRec(path, filter, &errs, loaded, 0, f) + } + + if len(errs) > 0 { + return nil, errs + } + + return root, nil +} + +func allRec(path string, filter Filter, errors *Errors, loaded *Result, depth int, f func(*Result, string, int) error) { + + path, err := fileurl.Clean(path) + if err != nil { + errors.add(err) + return + } + + info, err := os.Stat(path) + if err != nil { + errors.add(err) + return + } + + if filter != nil && filter(path, info, depth) { + return + } + + if !info.IsDir() { + if err := f(loaded, path, depth); err != nil { + errors.add(err) + } + return + } + + // If we are recursing on directories then content must be loaded under path + // specified by directory hierarchy. + if depth > 0 { + loaded = loaded.withParent(info.Name()) + } + + files, err := ioutil.ReadDir(path) + if err != nil { + errors.add(err) + return + } + + for _, file := range files { + allRec(filepath.Join(path, file.Name()), filter, errors, loaded, depth+1, f) + } +} + +func loadKnownTypes(path string, bs []byte, m metrics.Metrics, opts ast.ParserOptions) (interface{}, error) { + switch filepath.Ext(path) { + case ".json": + return loadJSON(path, bs, m) + case ".rego": + return loadRego(path, bs, m, opts) + case ".yaml", ".yml": + return loadYAML(path, bs, m) + default: + if strings.HasSuffix(path, ".tar.gz") { + r, err := loadBundleFile(path, bs, m) + if err != nil { + err = fmt.Errorf("bundle %s: %w", path, err) + } + return r, err + } + } + return nil, unrecognizedFile(path) +} + +func loadFileForAnyType(path string, bs []byte, m metrics.Metrics, opts ast.ParserOptions) (interface{}, error) { + module, err := loadRego(path, bs, m, opts) + if err == nil { + return module, nil + } + doc, err := loadJSON(path, bs, m) + if err == nil { + return doc, nil + } + doc, err = loadYAML(path, bs, m) + if err == nil { + return doc, nil + } + return nil, unrecognizedFile(path) +} + +func loadBundleFile(path string, bs []byte, m metrics.Metrics) (bundle.Bundle, error) { + tl := bundle.NewTarballLoaderWithBaseURL(bytes.NewBuffer(bs), path) + br := bundle.NewCustomReader(tl).WithMetrics(m).WithSkipBundleVerification(true).IncludeManifestInData(true) + return br.Read() +} + +func loadRego(path string, bs []byte, m metrics.Metrics, opts ast.ParserOptions) (*RegoFile, error) { + m.Timer(metrics.RegoModuleParse).Start() + var module *ast.Module + var err error + module, err = ast.ParseModuleWithOpts(path, string(bs), opts) + m.Timer(metrics.RegoModuleParse).Stop() + if err != nil { + return nil, err + } + result := &RegoFile{ + Name: path, + Parsed: module, + Raw: bs, + } + return result, nil +} + +func loadJSON(path string, bs []byte, m metrics.Metrics) (interface{}, error) { + m.Timer(metrics.RegoDataParse).Start() + buf := bytes.NewBuffer(bs) + decoder := util.NewJSONDecoder(buf) + var x interface{} + err := decoder.Decode(&x) + m.Timer(metrics.RegoDataParse).Stop() + if err != nil { + return nil, fmt.Errorf("%s: %w", path, err) + } + return x, nil +} + +func loadYAML(path string, bs []byte, m metrics.Metrics) (interface{}, error) { + m.Timer(metrics.RegoDataParse).Start() + bs, err := yaml.YAMLToJSON(bs) + m.Timer(metrics.RegoDataParse).Stop() + if err != nil { + return nil, fmt.Errorf("%v: error converting YAML to JSON: %v", path, err) + } + return loadJSON(path, bs, m) +} + +func makeDir(path []string, x interface{}) (map[string]interface{}, bool) { + if len(path) == 0 { + obj, ok := x.(map[string]interface{}) + if !ok { + return nil, false + } + return obj, true + } + return makeDir(path[:len(path)-1], map[string]interface{}{path[len(path)-1]: x}) +} diff --git a/vendor/github.com/open-policy-agent/opa/metrics/metrics.go b/vendor/github.com/open-policy-agent/opa/metrics/metrics.go new file mode 100644 index 00000000..4f31a445 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/metrics/metrics.go @@ -0,0 +1,312 @@ +// Copyright 2017 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +// Package metrics contains helpers for performance metric management inside the policy engine. +package metrics + +import ( + "encoding/json" + "fmt" + "sort" + "strings" + "sync" + "sync/atomic" + "time" + + go_metrics "github.com/rcrowley/go-metrics" +) + +// Well-known metric names. +const ( + BundleRequest = "bundle_request" + ServerHandler = "server_handler" + ServerQueryCacheHit = "server_query_cache_hit" + SDKDecisionEval = "sdk_decision_eval" + RegoQueryCompile = "rego_query_compile" + RegoQueryEval = "rego_query_eval" + RegoQueryParse = "rego_query_parse" + RegoModuleParse = "rego_module_parse" + RegoDataParse = "rego_data_parse" + RegoModuleCompile = "rego_module_compile" + RegoPartialEval = "rego_partial_eval" + RegoInputParse = "rego_input_parse" + RegoLoadFiles = "rego_load_files" + RegoLoadBundles = "rego_load_bundles" + RegoExternalResolve = "rego_external_resolve" +) + +// Info contains attributes describing the underlying metrics provider. +type Info struct { + Name string `json:"name"` // name is a unique human-readable identifier for the provider. +} + +// Metrics defines the interface for a collection of performance metrics in the +// policy engine. +type Metrics interface { + Info() Info + Timer(name string) Timer + Histogram(name string) Histogram + Counter(name string) Counter + All() map[string]interface{} + Clear() + json.Marshaler +} + +type TimerMetrics interface { + Timers() map[string]interface{} +} + +type metrics struct { + mtx sync.Mutex + timers map[string]Timer + histograms map[string]Histogram + counters map[string]Counter +} + +// New returns a new Metrics object. +func New() Metrics { + m := &metrics{} + m.Clear() + return m +} + +type metric struct { + Key string + Value interface{} +} + +func (m *metrics) Info() Info { + return Info{ + Name: "", + } +} + +func (m *metrics) String() string { + + all := m.All() + sorted := make([]metric, 0, len(all)) + + for key, value := range all { + sorted = append(sorted, metric{ + Key: key, + Value: value, + }) + } + + sort.Slice(sorted, func(i, j int) bool { + return sorted[i].Key < sorted[j].Key + }) + + buf := make([]string, len(sorted)) + for i := range sorted { + buf[i] = fmt.Sprintf("%v:%v", sorted[i].Key, sorted[i].Value) + } + + return strings.Join(buf, " ") +} + +func (m *metrics) MarshalJSON() ([]byte, error) { + return json.Marshal(m.All()) +} + +func (m *metrics) Timer(name string) Timer { + m.mtx.Lock() + defer m.mtx.Unlock() + t, ok := m.timers[name] + if !ok { + t = &timer{} + m.timers[name] = t + } + return t +} + +func (m *metrics) Histogram(name string) Histogram { + m.mtx.Lock() + defer m.mtx.Unlock() + h, ok := m.histograms[name] + if !ok { + h = newHistogram() + m.histograms[name] = h + } + return h +} + +func (m *metrics) Counter(name string) Counter { + m.mtx.Lock() + defer m.mtx.Unlock() + c, ok := m.counters[name] + if !ok { + zero := counter{} + c = &zero + m.counters[name] = c + } + return c +} + +func (m *metrics) All() map[string]interface{} { + m.mtx.Lock() + defer m.mtx.Unlock() + result := map[string]interface{}{} + for name, timer := range m.timers { + result[m.formatKey(name, timer)] = timer.Value() + } + for name, hist := range m.histograms { + result[m.formatKey(name, hist)] = hist.Value() + } + for name, cntr := range m.counters { + result[m.formatKey(name, cntr)] = cntr.Value() + } + return result +} + +func (m *metrics) Timers() map[string]interface{} { + m.mtx.Lock() + defer m.mtx.Unlock() + ts := map[string]interface{}{} + for n, t := range m.timers { + ts[m.formatKey(n, t)] = t.Value() + } + return ts +} + +func (m *metrics) Clear() { + m.mtx.Lock() + defer m.mtx.Unlock() + m.timers = map[string]Timer{} + m.histograms = map[string]Histogram{} + m.counters = map[string]Counter{} +} + +func (m *metrics) formatKey(name string, metrics interface{}) string { + switch metrics.(type) { + case Timer: + return "timer_" + name + "_ns" + case Histogram: + return "histogram_" + name + case Counter: + return "counter_" + name + default: + return name + } +} + +// Timer defines the interface for a restartable timer that accumulates elapsed +// time. +type Timer interface { + Value() interface{} + Int64() int64 + Start() + Stop() int64 +} + +type timer struct { + mtx sync.Mutex + start time.Time + value int64 +} + +func (t *timer) Start() { + t.mtx.Lock() + defer t.mtx.Unlock() + t.start = time.Now() +} + +func (t *timer) Stop() int64 { + t.mtx.Lock() + defer t.mtx.Unlock() + delta := time.Since(t.start).Nanoseconds() + t.value += delta + return delta +} + +func (t *timer) Value() interface{} { + return t.Int64() +} + +func (t *timer) Int64() int64 { + t.mtx.Lock() + defer t.mtx.Unlock() + return t.value +} + +// Histogram defines the interface for a histogram with hardcoded percentiles. +type Histogram interface { + Value() interface{} + Update(int64) +} + +type histogram struct { + hist go_metrics.Histogram // is thread-safe because of the underlying ExpDecaySample +} + +func newHistogram() Histogram { + // NOTE(tsandall): the reservoir size and alpha factor are taken from + // https://github.com/rcrowley/go-metrics. They may need to be tweaked in + // the future. + sample := go_metrics.NewExpDecaySample(1028, 0.015) + hist := go_metrics.NewHistogram(sample) + return &histogram{hist} +} + +func (h *histogram) Update(v int64) { + h.hist.Update(v) +} + +func (h *histogram) Value() interface{} { + values := map[string]interface{}{} + snap := h.hist.Snapshot() + percentiles := snap.Percentiles([]float64{ + 0.5, + 0.75, + 0.9, + 0.95, + 0.99, + 0.999, + 0.9999, + }) + values["count"] = snap.Count() + values["min"] = snap.Min() + values["max"] = snap.Max() + values["mean"] = snap.Mean() + values["stddev"] = snap.StdDev() + values["median"] = percentiles[0] + values["75%"] = percentiles[1] + values["90%"] = percentiles[2] + values["95%"] = percentiles[3] + values["99%"] = percentiles[4] + values["99.9%"] = percentiles[5] + values["99.99%"] = percentiles[6] + return values +} + +// Counter defines the interface for a monotonic increasing counter. +type Counter interface { + Value() interface{} + Incr() + Add(n uint64) +} + +type counter struct { + c uint64 +} + +func (c *counter) Incr() { + atomic.AddUint64(&c.c, 1) +} + +func (c *counter) Add(n uint64) { + atomic.AddUint64(&c.c, n) +} + +func (c *counter) Value() interface{} { + return atomic.LoadUint64(&c.c) +} + +func Statistics(num ...int64) interface{} { + t := newHistogram() + for _, n := range num { + t.Update(n) + } + return t.Value() +} diff --git a/vendor/github.com/open-policy-agent/opa/rego/errors.go b/vendor/github.com/open-policy-agent/opa/rego/errors.go new file mode 100644 index 00000000..dcc5e267 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/rego/errors.go @@ -0,0 +1,24 @@ +package rego + +// HaltError is an error type to return from a custom function implementation +// that will abort the evaluation process (analogous to topdown.Halt). +type HaltError struct { + err error +} + +// Error delegates to the wrapped error +func (h *HaltError) Error() string { + return h.err.Error() +} + +// NewHaltError wraps an error such that the evaluation process will stop +// when it occurs. +func NewHaltError(err error) error { + return &HaltError{err: err} +} + +// ErrorDetails interface is satisfied by an error that provides further +// details. +type ErrorDetails interface { + Lines() []string +} diff --git a/vendor/github.com/open-policy-agent/opa/rego/rego.go b/vendor/github.com/open-policy-agent/opa/rego/rego.go new file mode 100644 index 00000000..4d82bab3 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/rego/rego.go @@ -0,0 +1,2534 @@ +// Copyright 2017 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +// Package rego exposes high level APIs for evaluating Rego policies. +package rego + +import ( + "bytes" + "context" + "errors" + "fmt" + "io" + "strings" + "time" + + "github.com/open-policy-agent/opa/ast" + "github.com/open-policy-agent/opa/bundle" + bundleUtils "github.com/open-policy-agent/opa/internal/bundle" + "github.com/open-policy-agent/opa/internal/compiler/wasm" + "github.com/open-policy-agent/opa/internal/future" + "github.com/open-policy-agent/opa/internal/ir" + "github.com/open-policy-agent/opa/internal/planner" + "github.com/open-policy-agent/opa/internal/rego/opa" + "github.com/open-policy-agent/opa/internal/wasm/encoding" + "github.com/open-policy-agent/opa/loader" + "github.com/open-policy-agent/opa/metrics" + "github.com/open-policy-agent/opa/resolver" + "github.com/open-policy-agent/opa/storage" + "github.com/open-policy-agent/opa/storage/inmem" + "github.com/open-policy-agent/opa/topdown" + "github.com/open-policy-agent/opa/topdown/cache" + "github.com/open-policy-agent/opa/topdown/print" + "github.com/open-policy-agent/opa/tracing" + "github.com/open-policy-agent/opa/types" + "github.com/open-policy-agent/opa/util" +) + +const ( + defaultPartialNamespace = "partial" + wasmVarPrefix = "^" +) + +// nolint: deadcode,varcheck +const ( + targetWasm = "wasm" + targetRego = "rego" +) + +// CompileResult represents the result of compiling a Rego query, zero or more +// Rego modules, and arbitrary contextual data into an executable. +type CompileResult struct { + Bytes []byte `json:"bytes"` +} + +// PartialQueries contains the queries and support modules produced by partial +// evaluation. +type PartialQueries struct { + Queries []ast.Body `json:"queries,omitempty"` + Support []*ast.Module `json:"modules,omitempty"` +} + +// PartialResult represents the result of partial evaluation. The result can be +// used to generate a new query that can be run when inputs are known. +type PartialResult struct { + compiler *ast.Compiler + store storage.Store + body ast.Body + builtinDecls map[string]*ast.Builtin + builtinFuncs map[string]*topdown.Builtin +} + +// Rego returns an object that can be evaluated to produce a query result. +func (pr PartialResult) Rego(options ...func(*Rego)) *Rego { + options = append(options, Compiler(pr.compiler), Store(pr.store), ParsedQuery(pr.body)) + r := New(options...) + + // Propagate any custom builtins. + for k, v := range pr.builtinDecls { + r.builtinDecls[k] = v + } + for k, v := range pr.builtinFuncs { + r.builtinFuncs[k] = v + } + return r +} + +// preparedQuery is a wrapper around a Rego object which has pre-processed +// state stored on it. Once prepared there are a more limited number of actions +// that can be taken with it. It will, however, be able to evaluate faster since +// it will not have to re-parse or compile as much. +type preparedQuery struct { + r *Rego + cfg *PrepareConfig +} + +// EvalContext defines the set of options allowed to be set at evaluation +// time. Any other options will need to be set on a new Rego object. +type EvalContext struct { + hasInput bool + time time.Time + seed io.Reader + rawInput *interface{} + parsedInput ast.Value + metrics metrics.Metrics + txn storage.Transaction + instrument bool + instrumentation *topdown.Instrumentation + partialNamespace string + queryTracers []topdown.QueryTracer + compiledQuery compiledQuery + unknowns []string + disableInlining []ast.Ref + parsedUnknowns []*ast.Term + indexing bool + earlyExit bool + interQueryBuiltinCache cache.InterQueryCache + resolvers []refResolver + sortSets bool + printHook print.Hook + capabilities *ast.Capabilities +} + +// EvalOption defines a function to set an option on an EvalConfig +type EvalOption func(*EvalContext) + +// EvalInput configures the input for a Prepared Query's evaluation +func EvalInput(input interface{}) EvalOption { + return func(e *EvalContext) { + e.rawInput = &input + e.hasInput = true + } +} + +// EvalParsedInput configures the input for a Prepared Query's evaluation +func EvalParsedInput(input ast.Value) EvalOption { + return func(e *EvalContext) { + e.parsedInput = input + e.hasInput = true + } +} + +// EvalMetrics configures the metrics for a Prepared Query's evaluation +func EvalMetrics(metric metrics.Metrics) EvalOption { + return func(e *EvalContext) { + e.metrics = metric + } +} + +// EvalTransaction configures the Transaction for a Prepared Query's evaluation +func EvalTransaction(txn storage.Transaction) EvalOption { + return func(e *EvalContext) { + e.txn = txn + } +} + +// EvalInstrument enables or disables instrumenting for a Prepared Query's evaluation +func EvalInstrument(instrument bool) EvalOption { + return func(e *EvalContext) { + e.instrument = instrument + } +} + +// EvalTracer configures a tracer for a Prepared Query's evaluation +// Deprecated: Use EvalQueryTracer instead. +func EvalTracer(tracer topdown.Tracer) EvalOption { + return func(e *EvalContext) { + if tracer != nil { + e.queryTracers = append(e.queryTracers, topdown.WrapLegacyTracer(tracer)) + } + } +} + +// EvalQueryTracer configures a tracer for a Prepared Query's evaluation +func EvalQueryTracer(tracer topdown.QueryTracer) EvalOption { + return func(e *EvalContext) { + if tracer != nil { + e.queryTracers = append(e.queryTracers, tracer) + } + } +} + +// EvalPartialNamespace returns an argument that sets the namespace to use for +// partial evaluation results. The namespace must be a valid package path +// component. +func EvalPartialNamespace(ns string) EvalOption { + return func(e *EvalContext) { + e.partialNamespace = ns + } +} + +// EvalUnknowns returns an argument that sets the values to treat as +// unknown during partial evaluation. +func EvalUnknowns(unknowns []string) EvalOption { + return func(e *EvalContext) { + e.unknowns = unknowns + } +} + +// EvalDisableInlining returns an argument that adds a set of paths to exclude from +// partial evaluation inlining. +func EvalDisableInlining(paths []ast.Ref) EvalOption { + return func(e *EvalContext) { + e.disableInlining = paths + } +} + +// EvalParsedUnknowns returns an argument that sets the values to treat +// as unknown during partial evaluation. +func EvalParsedUnknowns(unknowns []*ast.Term) EvalOption { + return func(e *EvalContext) { + e.parsedUnknowns = unknowns + } +} + +// EvalRuleIndexing will disable indexing optimizations for the +// evaluation. This should only be used when tracing in debug mode. +func EvalRuleIndexing(enabled bool) EvalOption { + return func(e *EvalContext) { + e.indexing = enabled + } +} + +// EvalEarlyExit will disable 'early exit' optimizations for the +// evaluation. This should only be used when tracing in debug mode. +func EvalEarlyExit(enabled bool) EvalOption { + return func(e *EvalContext) { + e.earlyExit = enabled + } +} + +// EvalTime sets the wall clock time to use during policy evaluation. +// time.now_ns() calls will return this value. +func EvalTime(x time.Time) EvalOption { + return func(e *EvalContext) { + e.time = x + } +} + +// EvalSeed sets a reader that will seed randomization required by built-in functions. +// If a seed is not provided crypto/rand.Reader is used. +func EvalSeed(r io.Reader) EvalOption { + return func(e *EvalContext) { + e.seed = r + } +} + +// EvalInterQueryBuiltinCache sets the inter-query cache that built-in functions can utilize +// during evaluation. +func EvalInterQueryBuiltinCache(c cache.InterQueryCache) EvalOption { + return func(e *EvalContext) { + e.interQueryBuiltinCache = c + } +} + +// EvalResolver sets a Resolver for a specified ref path for this evaluation. +func EvalResolver(ref ast.Ref, r resolver.Resolver) EvalOption { + return func(e *EvalContext) { + e.resolvers = append(e.resolvers, refResolver{ref, r}) + } +} + +// EvalSortSets causes the evaluator to sort sets before returning them as JSON arrays. +func EvalSortSets(yes bool) EvalOption { + return func(e *EvalContext) { + e.sortSets = yes + } +} + +// EvalPrintHook sets the object to use for handling print statement outputs. +func EvalPrintHook(ph print.Hook) EvalOption { + return func(e *EvalContext) { + e.printHook = ph + } +} + +func (pq preparedQuery) Modules() map[string]*ast.Module { + mods := make(map[string]*ast.Module) + + for name, mod := range pq.r.parsedModules { + mods[name] = mod + } + + for _, b := range pq.r.bundles { + for _, mod := range b.Modules { + mods[mod.Path] = mod.Parsed + } + } + + return mods +} + +// newEvalContext creates a new EvalContext overlaying any EvalOptions over top +// the Rego object on the preparedQuery. The returned function should be called +// once the evaluation is complete to close any transactions that might have +// been opened. +func (pq preparedQuery) newEvalContext(ctx context.Context, options []EvalOption) (*EvalContext, func(context.Context), error) { + ectx := &EvalContext{ + hasInput: false, + rawInput: nil, + parsedInput: nil, + metrics: nil, + txn: nil, + instrument: false, + instrumentation: nil, + partialNamespace: pq.r.partialNamespace, + queryTracers: nil, + unknowns: pq.r.unknowns, + parsedUnknowns: pq.r.parsedUnknowns, + compiledQuery: compiledQuery{}, + indexing: true, + earlyExit: true, + resolvers: pq.r.resolvers, + printHook: pq.r.printHook, + capabilities: pq.r.capabilities, + } + + for _, o := range options { + o(ectx) + } + + if ectx.metrics == nil { + ectx.metrics = metrics.New() + } + + if ectx.instrument { + ectx.instrumentation = topdown.NewInstrumentation(ectx.metrics) + } + + // Default to an empty "finish" function + finishFunc := func(context.Context) {} + + var err error + ectx.disableInlining, err = parseStringsToRefs(pq.r.disableInlining) + if err != nil { + return nil, finishFunc, err + } + + if ectx.txn == nil { + ectx.txn, err = pq.r.store.NewTransaction(ctx) + if err != nil { + return nil, finishFunc, err + } + finishFunc = func(ctx context.Context) { + pq.r.store.Abort(ctx, ectx.txn) + } + } + + // If we didn't get an input specified in the Eval options + // then fall back to the Rego object's input fields. + if !ectx.hasInput { + ectx.rawInput = pq.r.rawInput + ectx.parsedInput = pq.r.parsedInput + } + + if ectx.parsedInput == nil { + if ectx.rawInput == nil { + // Fall back to the original Rego objects input if none was specified + // Note that it could still be nil + ectx.rawInput = pq.r.rawInput + } + if pq.r.target != targetWasm { + ectx.parsedInput, err = pq.r.parseRawInput(ectx.rawInput, ectx.metrics) + if err != nil { + return nil, finishFunc, err + } + } + } + + return ectx, finishFunc, nil +} + +// PreparedEvalQuery holds the prepared Rego state that has been pre-processed +// for subsequent evaluations. +type PreparedEvalQuery struct { + preparedQuery +} + +// Eval evaluates this PartialResult's Rego object with additional eval options +// and returns a ResultSet. +// If options are provided they will override the original Rego options respective value. +// The original Rego object transaction will *not* be re-used. A new transaction will be opened +// if one is not provided with an EvalOption. +func (pq PreparedEvalQuery) Eval(ctx context.Context, options ...EvalOption) (ResultSet, error) { + ectx, finish, err := pq.newEvalContext(ctx, options) + if err != nil { + return nil, err + } + defer finish(ctx) + + ectx.compiledQuery = pq.r.compiledQueries[evalQueryType] + + return pq.r.eval(ctx, ectx) +} + +// PreparedPartialQuery holds the prepared Rego state that has been pre-processed +// for partial evaluations. +type PreparedPartialQuery struct { + preparedQuery +} + +// Partial runs partial evaluation on the prepared query and returns the result. +// The original Rego object transaction will *not* be re-used. A new transaction will be opened +// if one is not provided with an EvalOption. +func (pq PreparedPartialQuery) Partial(ctx context.Context, options ...EvalOption) (*PartialQueries, error) { + ectx, finish, err := pq.newEvalContext(ctx, options) + if err != nil { + return nil, err + } + defer finish(ctx) + + ectx.compiledQuery = pq.r.compiledQueries[partialQueryType] + + return pq.r.partial(ctx, ectx) +} + +// Errors represents a collection of errors returned when evaluating Rego. +type Errors []error + +func (errs Errors) Error() string { + if len(errs) == 0 { + return "no error" + } + if len(errs) == 1 { + return fmt.Sprintf("1 error occurred: %v", errs[0].Error()) + } + buf := []string{fmt.Sprintf("%v errors occurred", len(errs))} + for _, err := range errs { + buf = append(buf, err.Error()) + } + return strings.Join(buf, "\n") +} + +var errPartialEvaluationNotEffective = errors.New("partial evaluation not effective") + +// IsPartialEvaluationNotEffectiveErr returns true if err is an error returned by +// this package to indicate that partial evaluation was ineffective. +func IsPartialEvaluationNotEffectiveErr(err error) bool { + errs, ok := err.(Errors) + if !ok { + return false + } + return len(errs) == 1 && errs[0] == errPartialEvaluationNotEffective +} + +type compiledQuery struct { + query ast.Body + compiler ast.QueryCompiler +} + +type queryType int + +// Define a query type for each of the top level Rego +// API's that compile queries differently. +const ( + evalQueryType queryType = iota + partialResultQueryType queryType = iota + partialQueryType queryType = iota + compileQueryType queryType = iota +) + +type loadPaths struct { + paths []string + filter loader.Filter +} + +// Rego constructs a query and can be evaluated to obtain results. +type Rego struct { + query string + parsedQuery ast.Body + compiledQueries map[queryType]compiledQuery + pkg string + parsedPackage *ast.Package + imports []string + parsedImports []*ast.Import + rawInput *interface{} + parsedInput ast.Value + unknowns []string + parsedUnknowns []*ast.Term + disableInlining []string + shallowInlining bool + skipPartialNamespace bool + partialNamespace string + modules []rawModule + parsedModules map[string]*ast.Module + compiler *ast.Compiler + store storage.Store + ownStore bool + txn storage.Transaction + metrics metrics.Metrics + queryTracers []topdown.QueryTracer + tracebuf *topdown.BufferTracer + trace bool + instrumentation *topdown.Instrumentation + instrument bool + capture map[*ast.Expr]ast.Var // map exprs to generated capture vars + termVarID int + dump io.Writer + runtime *ast.Term + time time.Time + seed io.Reader + capabilities *ast.Capabilities + builtinDecls map[string]*ast.Builtin + builtinFuncs map[string]*topdown.Builtin + unsafeBuiltins map[string]struct{} + loadPaths loadPaths + bundlePaths []string + bundles map[string]*bundle.Bundle + skipBundleVerification bool + interQueryBuiltinCache cache.InterQueryCache + strictBuiltinErrors bool + resolvers []refResolver + schemaSet *ast.SchemaSet + target string // target type (wasm, rego, etc.) + opa opa.EvalEngine + generateJSON func(*ast.Term, *EvalContext) (interface{}, error) + printHook print.Hook + enablePrintStatements bool + distributedTacingOpts tracing.Options +} + +// Function represents a built-in function that is callable in Rego. +type Function struct { + Name string + Decl *types.Function + Memoize bool +} + +// BuiltinContext contains additional attributes from the evaluator that +// built-in functions can use, e.g., the request context.Context, caches, etc. +type BuiltinContext = topdown.BuiltinContext + +type ( + // Builtin1 defines a built-in function that accepts 1 argument. + Builtin1 func(bctx BuiltinContext, op1 *ast.Term) (*ast.Term, error) + + // Builtin2 defines a built-in function that accepts 2 arguments. + Builtin2 func(bctx BuiltinContext, op1, op2 *ast.Term) (*ast.Term, error) + + // Builtin3 defines a built-in function that accepts 3 argument. + Builtin3 func(bctx BuiltinContext, op1, op2, op3 *ast.Term) (*ast.Term, error) + + // Builtin4 defines a built-in function that accepts 4 argument. + Builtin4 func(bctx BuiltinContext, op1, op2, op3, op4 *ast.Term) (*ast.Term, error) + + // BuiltinDyn defines a built-in function that accepts a list of arguments. + BuiltinDyn func(bctx BuiltinContext, terms []*ast.Term) (*ast.Term, error) +) + +// RegisterBuiltin1 adds a built-in function globally inside the OPA runtime. +func RegisterBuiltin1(decl *Function, impl Builtin1) { + ast.RegisterBuiltin(&ast.Builtin{ + Name: decl.Name, + Decl: decl.Decl, + }) + topdown.RegisterBuiltinFunc(decl.Name, func(bctx BuiltinContext, terms []*ast.Term, iter func(*ast.Term) error) error { + result, err := memoize(decl, bctx, terms, func() (*ast.Term, error) { return impl(bctx, terms[0]) }) + return finishFunction(decl.Name, bctx, result, err, iter) + }) +} + +// RegisterBuiltin2 adds a built-in function globally inside the OPA runtime. +func RegisterBuiltin2(decl *Function, impl Builtin2) { + ast.RegisterBuiltin(&ast.Builtin{ + Name: decl.Name, + Decl: decl.Decl, + }) + topdown.RegisterBuiltinFunc(decl.Name, func(bctx BuiltinContext, terms []*ast.Term, iter func(*ast.Term) error) error { + result, err := memoize(decl, bctx, terms, func() (*ast.Term, error) { return impl(bctx, terms[0], terms[1]) }) + return finishFunction(decl.Name, bctx, result, err, iter) + }) +} + +// RegisterBuiltin3 adds a built-in function globally inside the OPA runtime. +func RegisterBuiltin3(decl *Function, impl Builtin3) { + ast.RegisterBuiltin(&ast.Builtin{ + Name: decl.Name, + Decl: decl.Decl, + }) + topdown.RegisterBuiltinFunc(decl.Name, func(bctx BuiltinContext, terms []*ast.Term, iter func(*ast.Term) error) error { + result, err := memoize(decl, bctx, terms, func() (*ast.Term, error) { return impl(bctx, terms[0], terms[1], terms[2]) }) + return finishFunction(decl.Name, bctx, result, err, iter) + }) +} + +// RegisterBuiltin4 adds a built-in function globally inside the OPA runtime. +func RegisterBuiltin4(decl *Function, impl Builtin4) { + ast.RegisterBuiltin(&ast.Builtin{ + Name: decl.Name, + Decl: decl.Decl, + }) + topdown.RegisterBuiltinFunc(decl.Name, func(bctx BuiltinContext, terms []*ast.Term, iter func(*ast.Term) error) error { + result, err := memoize(decl, bctx, terms, func() (*ast.Term, error) { return impl(bctx, terms[0], terms[1], terms[2], terms[3]) }) + return finishFunction(decl.Name, bctx, result, err, iter) + }) +} + +// RegisterBuiltinDyn adds a built-in function globally inside the OPA runtime. +func RegisterBuiltinDyn(decl *Function, impl BuiltinDyn) { + ast.RegisterBuiltin(&ast.Builtin{ + Name: decl.Name, + Decl: decl.Decl, + }) + topdown.RegisterBuiltinFunc(decl.Name, func(bctx BuiltinContext, terms []*ast.Term, iter func(*ast.Term) error) error { + result, err := memoize(decl, bctx, terms, func() (*ast.Term, error) { return impl(bctx, terms) }) + return finishFunction(decl.Name, bctx, result, err, iter) + }) +} + +// Function1 returns an option that adds a built-in function to the Rego object. +func Function1(decl *Function, f Builtin1) func(*Rego) { + return newFunction(decl, func(bctx BuiltinContext, terms []*ast.Term, iter func(*ast.Term) error) error { + result, err := memoize(decl, bctx, terms, func() (*ast.Term, error) { return f(bctx, terms[0]) }) + return finishFunction(decl.Name, bctx, result, err, iter) + }) +} + +// Function2 returns an option that adds a built-in function to the Rego object. +func Function2(decl *Function, f Builtin2) func(*Rego) { + return newFunction(decl, func(bctx BuiltinContext, terms []*ast.Term, iter func(*ast.Term) error) error { + result, err := memoize(decl, bctx, terms, func() (*ast.Term, error) { return f(bctx, terms[0], terms[1]) }) + return finishFunction(decl.Name, bctx, result, err, iter) + }) +} + +// Function3 returns an option that adds a built-in function to the Rego object. +func Function3(decl *Function, f Builtin3) func(*Rego) { + return newFunction(decl, func(bctx BuiltinContext, terms []*ast.Term, iter func(*ast.Term) error) error { + result, err := memoize(decl, bctx, terms, func() (*ast.Term, error) { return f(bctx, terms[0], terms[1], terms[2]) }) + return finishFunction(decl.Name, bctx, result, err, iter) + }) +} + +// Function4 returns an option that adds a built-in function to the Rego object. +func Function4(decl *Function, f Builtin4) func(*Rego) { + return newFunction(decl, func(bctx BuiltinContext, terms []*ast.Term, iter func(*ast.Term) error) error { + result, err := memoize(decl, bctx, terms, func() (*ast.Term, error) { return f(bctx, terms[0], terms[1], terms[2], terms[3]) }) + return finishFunction(decl.Name, bctx, result, err, iter) + }) +} + +// FunctionDyn returns an option that adds a built-in function to the Rego object. +func FunctionDyn(decl *Function, f BuiltinDyn) func(*Rego) { + return newFunction(decl, func(bctx BuiltinContext, terms []*ast.Term, iter func(*ast.Term) error) error { + result, err := memoize(decl, bctx, terms, func() (*ast.Term, error) { return f(bctx, terms) }) + return finishFunction(decl.Name, bctx, result, err, iter) + }) +} + +// FunctionDecl returns an option that adds a custom-built-in function +// __declaration__. NO implementation is provided. This is used for +// non-interpreter execution envs (e.g., Wasm). +func FunctionDecl(decl *Function) func(*Rego) { + return newDecl(decl) +} + +func newDecl(decl *Function) func(*Rego) { + return func(r *Rego) { + r.builtinDecls[decl.Name] = &ast.Builtin{ + Name: decl.Name, + Decl: decl.Decl, + } + } +} + +type memo struct { + term *ast.Term + err error +} + +type memokey string + +func memoize(decl *Function, bctx BuiltinContext, terms []*ast.Term, ifEmpty func() (*ast.Term, error)) (*ast.Term, error) { + + if !decl.Memoize { + return ifEmpty() + } + + // NOTE(tsandall): we assume memoization is applied to infrequent built-in + // calls that do things like fetch data from remote locations. As such, + // converting the terms to strings is acceptable for now. + var b strings.Builder + if _, err := b.WriteString(decl.Name); err != nil { + return nil, err + } + + // The term slice _may_ include an output term depending on how the caller + // referred to the built-in function. Only use the arguments as the cache + // key. Unification ensures we don't get false positive matches. + for i := 0; i < len(decl.Decl.Args()); i++ { + if _, err := b.WriteString(terms[i].String()); err != nil { + return nil, err + } + } + + key := memokey(b.String()) + hit, ok := bctx.Cache.Get(key) + var m memo + if ok { + m = hit.(memo) + } else { + m.term, m.err = ifEmpty() + bctx.Cache.Put(key, m) + } + + return m.term, m.err +} + +// Dump returns an argument that sets the writer to dump debugging information to. +func Dump(w io.Writer) func(r *Rego) { + return func(r *Rego) { + r.dump = w + } +} + +// Query returns an argument that sets the Rego query. +func Query(q string) func(r *Rego) { + return func(r *Rego) { + r.query = q + } +} + +// ParsedQuery returns an argument that sets the Rego query. +func ParsedQuery(q ast.Body) func(r *Rego) { + return func(r *Rego) { + r.parsedQuery = q + } +} + +// Package returns an argument that sets the Rego package on the query's +// context. +func Package(p string) func(r *Rego) { + return func(r *Rego) { + r.pkg = p + } +} + +// ParsedPackage returns an argument that sets the Rego package on the query's +// context. +func ParsedPackage(pkg *ast.Package) func(r *Rego) { + return func(r *Rego) { + r.parsedPackage = pkg + } +} + +// Imports returns an argument that adds a Rego import to the query's context. +func Imports(p []string) func(r *Rego) { + return func(r *Rego) { + r.imports = append(r.imports, p...) + } +} + +// ParsedImports returns an argument that adds Rego imports to the query's +// context. +func ParsedImports(imp []*ast.Import) func(r *Rego) { + return func(r *Rego) { + r.parsedImports = append(r.parsedImports, imp...) + } +} + +// Input returns an argument that sets the Rego input document. Input should be +// a native Go value representing the input document. +func Input(x interface{}) func(r *Rego) { + return func(r *Rego) { + r.rawInput = &x + } +} + +// ParsedInput returns an argument that sets the Rego input document. +func ParsedInput(x ast.Value) func(r *Rego) { + return func(r *Rego) { + r.parsedInput = x + } +} + +// Unknowns returns an argument that sets the values to treat as unknown during +// partial evaluation. +func Unknowns(unknowns []string) func(r *Rego) { + return func(r *Rego) { + r.unknowns = unknowns + } +} + +// ParsedUnknowns returns an argument that sets the values to treat as unknown +// during partial evaluation. +func ParsedUnknowns(unknowns []*ast.Term) func(r *Rego) { + return func(r *Rego) { + r.parsedUnknowns = unknowns + } +} + +// DisableInlining adds a set of paths to exclude from partial evaluation inlining. +func DisableInlining(paths []string) func(r *Rego) { + return func(r *Rego) { + r.disableInlining = paths + } +} + +// ShallowInlining prevents rules that depend on unknown values from being inlined. +// Rules that only depend on known values are inlined. +func ShallowInlining(yes bool) func(r *Rego) { + return func(r *Rego) { + r.shallowInlining = yes + } +} + +// SkipPartialNamespace disables namespacing of partial evalution results for support +// rules generated from policy. Synthetic support rules are still namespaced. +func SkipPartialNamespace(yes bool) func(r *Rego) { + return func(r *Rego) { + r.skipPartialNamespace = yes + } +} + +// PartialNamespace returns an argument that sets the namespace to use for +// partial evaluation results. The namespace must be a valid package path +// component. +func PartialNamespace(ns string) func(r *Rego) { + return func(r *Rego) { + r.partialNamespace = ns + } +} + +// Module returns an argument that adds a Rego module. +func Module(filename, input string) func(r *Rego) { + return func(r *Rego) { + r.modules = append(r.modules, rawModule{ + filename: filename, + module: input, + }) + } +} + +// ParsedModule returns an argument that adds a parsed Rego module. If a string +// module with the same filename name is added, it will override the parsed +// module. +func ParsedModule(module *ast.Module) func(*Rego) { + return func(r *Rego) { + var filename string + if module.Package.Location != nil { + filename = module.Package.Location.File + } else { + filename = fmt.Sprintf("module_%p.rego", module) + } + r.parsedModules[filename] = module + } +} + +// Load returns an argument that adds a filesystem path to load data +// and Rego modules from. Any file with a *.rego, *.yaml, or *.json +// extension will be loaded. The path can be either a directory or file, +// directories are loaded recursively. The optional ignore string patterns +// can be used to filter which files are used. +// The Load option can only be used once. +// Note: Loading files will require a write transaction on the store. +func Load(paths []string, filter loader.Filter) func(r *Rego) { + return func(r *Rego) { + r.loadPaths = loadPaths{paths, filter} + } +} + +// LoadBundle returns an argument that adds a filesystem path to load +// a bundle from. The path can be a compressed bundle file or a directory +// to be loaded as a bundle. +// Note: Loading bundles will require a write transaction on the store. +func LoadBundle(path string) func(r *Rego) { + return func(r *Rego) { + r.bundlePaths = append(r.bundlePaths, path) + } +} + +// ParsedBundle returns an argument that adds a bundle to be loaded. +func ParsedBundle(name string, b *bundle.Bundle) func(r *Rego) { + return func(r *Rego) { + r.bundles[name] = b + } +} + +// Compiler returns an argument that sets the Rego compiler. +func Compiler(c *ast.Compiler) func(r *Rego) { + return func(r *Rego) { + r.compiler = c + } +} + +// Store returns an argument that sets the policy engine's data storage layer. +// +// If using the Load, LoadBundle, or ParsedBundle options then a transaction +// must also be provided via the Transaction() option. After loading files +// or bundles the transaction should be aborted or committed. +func Store(s storage.Store) func(r *Rego) { + return func(r *Rego) { + r.store = s + } +} + +// Transaction returns an argument that sets the transaction to use for storage +// layer operations. +// +// Requires the store associated with the transaction to be provided via the +// Store() option. If using Load(), LoadBundle(), or ParsedBundle() options +// the transaction will likely require write params. +func Transaction(txn storage.Transaction) func(r *Rego) { + return func(r *Rego) { + r.txn = txn + } +} + +// Metrics returns an argument that sets the metrics collection. +func Metrics(m metrics.Metrics) func(r *Rego) { + return func(r *Rego) { + r.metrics = m + } +} + +// Instrument returns an argument that enables instrumentation for diagnosing +// performance issues. +func Instrument(yes bool) func(r *Rego) { + return func(r *Rego) { + r.instrument = yes + } +} + +// Trace returns an argument that enables tracing on r. +func Trace(yes bool) func(r *Rego) { + return func(r *Rego) { + r.trace = yes + } +} + +// Tracer returns an argument that adds a query tracer to r. +// Deprecated: Use QueryTracer instead. +func Tracer(t topdown.Tracer) func(r *Rego) { + return func(r *Rego) { + if t != nil { + r.queryTracers = append(r.queryTracers, topdown.WrapLegacyTracer(t)) + } + } +} + +// QueryTracer returns an argument that adds a query tracer to r. +func QueryTracer(t topdown.QueryTracer) func(r *Rego) { + return func(r *Rego) { + if t != nil { + r.queryTracers = append(r.queryTracers, t) + } + } +} + +// Runtime returns an argument that sets the runtime data to provide to the +// evaluation engine. +func Runtime(term *ast.Term) func(r *Rego) { + return func(r *Rego) { + r.runtime = term + } +} + +// Time sets the wall clock time to use during policy evaluation. Prepared queries +// do not inherit this parameter. Use EvalTime to set the wall clock time when +// executing a prepared query. +func Time(x time.Time) func(r *Rego) { + return func(r *Rego) { + r.time = x + } +} + +// Seed sets a reader that will seed randomization required by built-in functions. +// If a seed is not provided crypto/rand.Reader is used. +func Seed(r io.Reader) func(*Rego) { + return func(e *Rego) { + e.seed = r + } +} + +// PrintTrace is a helper function to write a human-readable version of the +// trace to the writer w. +func PrintTrace(w io.Writer, r *Rego) { + if r == nil || r.tracebuf == nil { + return + } + topdown.PrettyTrace(w, *r.tracebuf) +} + +// PrintTraceWithLocation is a helper function to write a human-readable version of the +// trace to the writer w. +func PrintTraceWithLocation(w io.Writer, r *Rego) { + if r == nil || r.tracebuf == nil { + return + } + topdown.PrettyTraceWithLocation(w, *r.tracebuf) +} + +// UnsafeBuiltins sets the built-in functions to treat as unsafe and not allow. +// This option is ignored for module compilation if the caller supplies the +// compiler. This option is always honored for query compilation. Provide an +// empty (non-nil) map to disable checks on queries. +func UnsafeBuiltins(unsafeBuiltins map[string]struct{}) func(r *Rego) { + return func(r *Rego) { + r.unsafeBuiltins = unsafeBuiltins + } +} + +// SkipBundleVerification skips verification of a signed bundle. +func SkipBundleVerification(yes bool) func(r *Rego) { + return func(r *Rego) { + r.skipBundleVerification = yes + } +} + +// InterQueryBuiltinCache sets the inter-query cache that built-in functions can utilize +// during evaluation. +func InterQueryBuiltinCache(c cache.InterQueryCache) func(r *Rego) { + return func(r *Rego) { + r.interQueryBuiltinCache = c + } +} + +// StrictBuiltinErrors tells the evaluator to treat all built-in function errors as fatal errors. +func StrictBuiltinErrors(yes bool) func(r *Rego) { + return func(r *Rego) { + r.strictBuiltinErrors = yes + } +} + +// Resolver sets a Resolver for a specified ref path. +func Resolver(ref ast.Ref, r resolver.Resolver) func(r *Rego) { + return func(rego *Rego) { + rego.resolvers = append(rego.resolvers, refResolver{ref, r}) + } +} + +// Schemas sets the schemaSet +func Schemas(x *ast.SchemaSet) func(r *Rego) { + return func(r *Rego) { + r.schemaSet = x + } +} + +// Capabilities configures the underlying compiler's capabilities. +// This option is ignored for module compilation if the caller supplies the +// compiler. +func Capabilities(c *ast.Capabilities) func(r *Rego) { + return func(r *Rego) { + r.capabilities = c + } +} + +// Target sets the runtime to exercise. +func Target(t string) func(r *Rego) { + return func(r *Rego) { + r.target = t + } +} + +// GenerateJSON sets the AST to JSON converter for the results. +func GenerateJSON(f func(*ast.Term, *EvalContext) (interface{}, error)) func(r *Rego) { + return func(r *Rego) { + r.generateJSON = f + } +} + +// PrintHook sets the object to use for handling print statement outputs. +func PrintHook(h print.Hook) func(r *Rego) { + return func(r *Rego) { + r.printHook = h + } +} + +// DistributedTracingOpts sets the options to be used by distributed tracing. +func DistributedTracingOpts(tr tracing.Options) func(r *Rego) { + return func(r *Rego) { + r.distributedTacingOpts = tr + } +} + +// EnablePrintStatements enables print() calls. If this option is not provided, +// print() calls will be erased from the policy. This option only applies to +// queries and policies that passed as raw strings, i.e., this function will not +// have any affect if the caller supplies the ast.Compiler instance. +func EnablePrintStatements(yes bool) func(r *Rego) { + return func(r *Rego) { + r.enablePrintStatements = yes + } +} + +// New returns a new Rego object. +func New(options ...func(r *Rego)) *Rego { + + r := &Rego{ + parsedModules: map[string]*ast.Module{}, + capture: map[*ast.Expr]ast.Var{}, + compiledQueries: map[queryType]compiledQuery{}, + builtinDecls: map[string]*ast.Builtin{}, + builtinFuncs: map[string]*topdown.Builtin{}, + bundles: map[string]*bundle.Bundle{}, + } + + for _, option := range options { + option(r) + } + + if r.compiler == nil { + r.compiler = ast.NewCompiler(). + WithUnsafeBuiltins(r.unsafeBuiltins). + WithBuiltins(r.builtinDecls). + WithDebug(r.dump). + WithSchemas(r.schemaSet). + WithCapabilities(r.capabilities). + WithEnablePrintStatements(r.enablePrintStatements) + } + + if r.store == nil { + r.store = inmem.New() + r.ownStore = true + } else { + r.ownStore = false + } + + if r.metrics == nil { + r.metrics = metrics.New() + } + + if r.instrument { + r.instrumentation = topdown.NewInstrumentation(r.metrics) + r.compiler.WithMetrics(r.metrics) + } + + if r.trace { + r.tracebuf = topdown.NewBufferTracer() + r.queryTracers = append(r.queryTracers, r.tracebuf) + } + + if r.partialNamespace == "" { + r.partialNamespace = defaultPartialNamespace + } + + if r.generateJSON == nil { + r.generateJSON = generateJSON + } + + return r +} + +// Eval evaluates this Rego object and returns a ResultSet. +func (r *Rego) Eval(ctx context.Context) (ResultSet, error) { + var err error + var txnClose transactionCloser + r.txn, txnClose, err = r.getTxn(ctx) + if err != nil { + return nil, err + } + + pq, err := r.PrepareForEval(ctx) + if err != nil { + _ = txnClose(ctx, err) // Ignore error + return nil, err + } + + evalArgs := []EvalOption{ + EvalTransaction(r.txn), + EvalMetrics(r.metrics), + EvalInstrument(r.instrument), + EvalTime(r.time), + EvalInterQueryBuiltinCache(r.interQueryBuiltinCache), + EvalSeed(r.seed), + } + + for _, qt := range r.queryTracers { + evalArgs = append(evalArgs, EvalQueryTracer(qt)) + } + + for i := range r.resolvers { + evalArgs = append(evalArgs, EvalResolver(r.resolvers[i].ref, r.resolvers[i].r)) + } + + rs, err := pq.Eval(ctx, evalArgs...) + txnErr := txnClose(ctx, err) // Always call closer + if err == nil { + err = txnErr + } + return rs, err +} + +// PartialEval has been deprecated and renamed to PartialResult. +func (r *Rego) PartialEval(ctx context.Context) (PartialResult, error) { + return r.PartialResult(ctx) +} + +// PartialResult partially evaluates this Rego object and returns a PartialResult. +func (r *Rego) PartialResult(ctx context.Context) (PartialResult, error) { + var err error + var txnClose transactionCloser + r.txn, txnClose, err = r.getTxn(ctx) + if err != nil { + return PartialResult{}, err + } + + pq, err := r.PrepareForEval(ctx, WithPartialEval()) + txnErr := txnClose(ctx, err) // Always call closer + if err != nil { + return PartialResult{}, err + } + if txnErr != nil { + return PartialResult{}, txnErr + } + + pr := PartialResult{ + compiler: pq.r.compiler, + store: pq.r.store, + body: pq.r.parsedQuery, + builtinDecls: pq.r.builtinDecls, + builtinFuncs: pq.r.builtinFuncs, + } + + return pr, nil +} + +// Partial runs partial evaluation on r and returns the result. +func (r *Rego) Partial(ctx context.Context) (*PartialQueries, error) { + var err error + var txnClose transactionCloser + r.txn, txnClose, err = r.getTxn(ctx) + if err != nil { + return nil, err + } + + pq, err := r.PrepareForPartial(ctx) + if err != nil { + _ = txnClose(ctx, err) // Ignore error + return nil, err + } + + evalArgs := []EvalOption{ + EvalTransaction(r.txn), + EvalMetrics(r.metrics), + EvalInstrument(r.instrument), + EvalInterQueryBuiltinCache(r.interQueryBuiltinCache), + } + + for _, t := range r.queryTracers { + evalArgs = append(evalArgs, EvalQueryTracer(t)) + } + + for i := range r.resolvers { + evalArgs = append(evalArgs, EvalResolver(r.resolvers[i].ref, r.resolvers[i].r)) + } + + pqs, err := pq.Partial(ctx, evalArgs...) + txnErr := txnClose(ctx, err) // Always call closer + if err == nil { + err = txnErr + } + return pqs, err +} + +// CompileOption defines a function to set options on Compile calls. +type CompileOption func(*CompileContext) + +// CompileContext contains options for Compile calls. +type CompileContext struct { + partial bool +} + +// CompilePartial defines an option to control whether partial evaluation is run +// before the query is planned and compiled. +func CompilePartial(yes bool) CompileOption { + return func(cfg *CompileContext) { + cfg.partial = yes + } +} + +// Compile returns a compiled policy query. +func (r *Rego) Compile(ctx context.Context, opts ...CompileOption) (*CompileResult, error) { + + var cfg CompileContext + + for _, opt := range opts { + opt(&cfg) + } + + var queries []ast.Body + var modules []*ast.Module + + if cfg.partial { + + pq, err := r.Partial(ctx) + if err != nil { + return nil, err + } + if r.dump != nil { + if len(pq.Queries) != 0 { + msg := fmt.Sprintf("QUERIES (%d total):", len(pq.Queries)) + fmt.Fprintln(r.dump, msg) + fmt.Fprintln(r.dump, strings.Repeat("-", len(msg))) + for i := range pq.Queries { + fmt.Println(pq.Queries[i]) + } + fmt.Fprintln(r.dump) + } + if len(pq.Support) != 0 { + msg := fmt.Sprintf("SUPPORT (%d total):", len(pq.Support)) + fmt.Fprintln(r.dump, msg) + fmt.Fprintln(r.dump, strings.Repeat("-", len(msg))) + for i := range pq.Support { + fmt.Println(pq.Support[i]) + } + fmt.Fprintln(r.dump) + } + } + + queries = pq.Queries + modules = pq.Support + + for _, module := range r.compiler.Modules { + modules = append(modules, module) + } + } else { + var err error + // If creating a new transaction it should be closed before calling the + // planner to avoid holding open the transaction longer than needed. + // + // TODO(tsandall): in future, planner could make use of store, in which + // case this will need to change. + var txnClose transactionCloser + r.txn, txnClose, err = r.getTxn(ctx) + if err != nil { + return nil, err + } + + err = r.prepare(ctx, compileQueryType, nil) + txnErr := txnClose(ctx, err) // Always call closer + if err != nil { + return nil, err + } + if txnErr != nil { + return nil, err + } + + for _, module := range r.compiler.Modules { + modules = append(modules, module) + } + + queries = []ast.Body{r.compiledQueries[compileQueryType].query} + } + + return r.compileWasm(modules, queries, compileQueryType) +} + +func (r *Rego) compileWasm(modules []*ast.Module, queries []ast.Body, qType queryType) (*CompileResult, error) { + decls := make(map[string]*ast.Builtin, len(r.builtinDecls)+len(ast.BuiltinMap)) + + for k, v := range ast.BuiltinMap { + decls[k] = v + } + + for k, v := range r.builtinDecls { + decls[k] = v + } + + const queryName = "eval" // NOTE(tsandall): the query name is arbitrary + + p := planner.New(). + WithQueries([]planner.QuerySet{ + { + Name: queryName, + Queries: queries, + RewrittenVars: r.compiledQueries[qType].compiler.RewrittenVars(), + }, + }). + WithModules(modules). + WithBuiltinDecls(decls). + WithDebug(r.dump) + policy, err := p.Plan() + if err != nil { + return nil, err + } + + if r.dump != nil { + fmt.Fprintln(r.dump, "PLAN:") + fmt.Fprintln(r.dump, "-----") + err = ir.Pretty(r.dump, policy) + if err != nil { + return nil, err + } + fmt.Fprintln(r.dump) + } + + m, err := wasm.New().WithPolicy(policy).Compile() + if err != nil { + return nil, err + } + + var out bytes.Buffer + + if err := encoding.WriteModule(&out, m); err != nil { + return nil, err + } + + result := &CompileResult{ + Bytes: out.Bytes(), + } + + return result, nil +} + +// PrepareOption defines a function to set an option to control +// the behavior of the Prepare call. +type PrepareOption func(*PrepareConfig) + +// PrepareConfig holds settings to control the behavior of the +// Prepare call. +type PrepareConfig struct { + doPartialEval bool + disableInlining *[]string +} + +// WithPartialEval configures an option for PrepareForEval +// which will have it perform partial evaluation while preparing +// the query (similar to rego.Rego#PartialResult) +func WithPartialEval() PrepareOption { + return func(p *PrepareConfig) { + p.doPartialEval = true + } +} + +// WithNoInline adds a set of paths to exclude from partial evaluation inlining. +func WithNoInline(paths []string) PrepareOption { + return func(p *PrepareConfig) { + p.disableInlining = &paths + } +} + +// PrepareForEval will parse inputs, modules, and query arguments in preparation +// of evaluating them. +func (r *Rego) PrepareForEval(ctx context.Context, opts ...PrepareOption) (PreparedEvalQuery, error) { + if !r.hasQuery() { + return PreparedEvalQuery{}, fmt.Errorf("cannot evaluate empty query") + } + + pCfg := &PrepareConfig{} + for _, o := range opts { + o(pCfg) + } + + var err error + var txnClose transactionCloser + r.txn, txnClose, err = r.getTxn(ctx) + if err != nil { + return PreparedEvalQuery{}, err + } + + // If the caller wanted to do partial evaluation as part of preparation + // do it now and use the new Rego object. + if pCfg.doPartialEval { + + pr, err := r.partialResult(ctx, pCfg) + if err != nil { + _ = txnClose(ctx, err) // Ignore error + return PreparedEvalQuery{}, err + } + + // Prepare the new query using the result of partial evaluation + pq, err := pr.Rego(Transaction(r.txn)).PrepareForEval(ctx) + txnErr := txnClose(ctx, err) + if err != nil { + return pq, err + } + return pq, txnErr + } + + err = r.prepare(ctx, evalQueryType, []extraStage{ + { + after: "ResolveRefs", + stage: ast.QueryCompilerStageDefinition{ + Name: "RewriteToCaptureValue", + MetricName: "query_compile_stage_rewrite_to_capture_value", + Stage: r.rewriteQueryToCaptureValue, + }, + }, + }) + if err != nil { + _ = txnClose(ctx, err) // Ignore error + return PreparedEvalQuery{}, err + } + + if r.target == targetWasm { + + if r.hasWasmModule() { + _ = txnClose(ctx, err) // Ignore error + return PreparedEvalQuery{}, fmt.Errorf("wasm target not supported") + } + + var modules []*ast.Module + for _, module := range r.compiler.Modules { + modules = append(modules, module) + } + + queries := []ast.Body{r.compiledQueries[evalQueryType].query} + + e, err := opa.LookupEngine(targetWasm) + if err != nil { + return PreparedEvalQuery{}, err + } + + // nolint: staticcheck // SA4006 false positive + cr, err := r.compileWasm(modules, queries, evalQueryType) + if err != nil { + _ = txnClose(ctx, err) // Ignore error + return PreparedEvalQuery{}, err + } + + // nolint: staticcheck // SA4006 false positive + data, err := r.store.Read(ctx, r.txn, storage.Path{}) + if err != nil { + _ = txnClose(ctx, err) // Ignore error + return PreparedEvalQuery{}, err + } + + o, err := e.New().WithPolicyBytes(cr.Bytes).WithDataJSON(data).Init() + if err != nil { + _ = txnClose(ctx, err) // Ignore error + return PreparedEvalQuery{}, err + } + r.opa = o + } + + txnErr := txnClose(ctx, err) // Always call closer + if err != nil { + return PreparedEvalQuery{}, err + } + if txnErr != nil { + return PreparedEvalQuery{}, txnErr + } + + return PreparedEvalQuery{preparedQuery{r, pCfg}}, err +} + +// PrepareForPartial will parse inputs, modules, and query arguments in preparation +// of partially evaluating them. +func (r *Rego) PrepareForPartial(ctx context.Context, opts ...PrepareOption) (PreparedPartialQuery, error) { + if !r.hasQuery() { + return PreparedPartialQuery{}, fmt.Errorf("cannot evaluate empty query") + } + + pCfg := &PrepareConfig{} + for _, o := range opts { + o(pCfg) + } + + var err error + var txnClose transactionCloser + r.txn, txnClose, err = r.getTxn(ctx) + if err != nil { + return PreparedPartialQuery{}, err + } + + err = r.prepare(ctx, partialQueryType, []extraStage{ + { + after: "CheckSafety", + stage: ast.QueryCompilerStageDefinition{ + Name: "RewriteEquals", + MetricName: "query_compile_stage_rewrite_equals", + Stage: r.rewriteEqualsForPartialQueryCompile, + }, + }, + }) + txnErr := txnClose(ctx, err) // Always call closer + if err != nil { + return PreparedPartialQuery{}, err + } + if txnErr != nil { + return PreparedPartialQuery{}, txnErr + } + + return PreparedPartialQuery{preparedQuery{r, pCfg}}, err +} + +func (r *Rego) prepare(ctx context.Context, qType queryType, extras []extraStage) error { + var err error + + r.parsedInput, err = r.parseInput() + if err != nil { + return err + } + + err = r.loadFiles(ctx, r.txn, r.metrics) + if err != nil { + return err + } + + err = r.loadBundles(ctx, r.txn, r.metrics) + if err != nil { + return err + } + + err = r.parseModules(ctx, r.txn, r.metrics) + if err != nil { + return err + } + + // Compile the modules *before* the query, else functions + // defined in the module won't be found... + err = r.compileModules(ctx, r.txn, r.metrics) + if err != nil { + return err + } + + imports, err := r.prepareImports() + if err != nil { + return err + } + + futureImports := []*ast.Import{} + for _, imp := range imports { + if imp.Path.Value.(ast.Ref).HasPrefix(ast.Ref([]*ast.Term{ast.FutureRootDocument})) { + futureImports = append(futureImports, imp) + } + } + + r.parsedQuery, err = r.parseQuery(futureImports, r.metrics) + if err != nil { + return err + } + + err = r.compileAndCacheQuery(qType, r.parsedQuery, imports, r.metrics, extras) + if err != nil { + return err + } + + return nil +} + +func (r *Rego) parseModules(ctx context.Context, txn storage.Transaction, m metrics.Metrics) error { + if len(r.modules) == 0 { + return nil + } + + m.Timer(metrics.RegoModuleParse).Start() + defer m.Timer(metrics.RegoModuleParse).Stop() + var errs Errors + + // Parse any modules in the are saved to the store, but only if + // another compile step is going to occur (ie. we have parsed modules + // that need to be compiled). + ids, err := r.store.ListPolicies(ctx, txn) + if err != nil { + return err + } + + for _, id := range ids { + // if it is already on the compiler we're using + // then don't bother to re-parse it from source + if _, haveMod := r.compiler.Modules[id]; haveMod { + continue + } + + bs, err := r.store.GetPolicy(ctx, txn, id) + if err != nil { + return err + } + + parsed, err := ast.ParseModule(id, string(bs)) + if err != nil { + errs = append(errs, err) + } + + r.parsedModules[id] = parsed + } + + // Parse any passed in as arguments to the Rego object + for _, module := range r.modules { + p, err := module.Parse() + if err != nil { + errs = append(errs, err) + } + r.parsedModules[module.filename] = p + } + + if len(errs) > 0 { + return errs + } + + return nil +} + +func (r *Rego) loadFiles(ctx context.Context, txn storage.Transaction, m metrics.Metrics) error { + if len(r.loadPaths.paths) == 0 { + return nil + } + + m.Timer(metrics.RegoLoadFiles).Start() + defer m.Timer(metrics.RegoLoadFiles).Stop() + + result, err := loader.NewFileLoader(). + WithMetrics(m). + WithProcessAnnotation(r.schemaSet != nil). + Filtered(r.loadPaths.paths, r.loadPaths.filter) + if err != nil { + return err + } + for name, mod := range result.Modules { + r.parsedModules[name] = mod.Parsed + } + + if len(result.Documents) > 0 { + err = r.store.Write(ctx, txn, storage.AddOp, storage.Path{}, result.Documents) + if err != nil { + return err + } + } + return nil +} + +func (r *Rego) loadBundles(ctx context.Context, txn storage.Transaction, m metrics.Metrics) error { + if len(r.bundlePaths) == 0 { + return nil + } + + m.Timer(metrics.RegoLoadBundles).Start() + defer m.Timer(metrics.RegoLoadBundles).Stop() + + for _, path := range r.bundlePaths { + bndl, err := loader.NewFileLoader(). + WithMetrics(m). + WithProcessAnnotation(r.schemaSet != nil). + WithSkipBundleVerification(r.skipBundleVerification). + AsBundle(path) + if err != nil { + return fmt.Errorf("loading error: %s", err) + } + r.bundles[path] = bndl + } + return nil +} + +func (r *Rego) parseInput() (ast.Value, error) { + if r.parsedInput != nil { + return r.parsedInput, nil + } + return r.parseRawInput(r.rawInput, r.metrics) +} + +func (r *Rego) parseRawInput(rawInput *interface{}, m metrics.Metrics) (ast.Value, error) { + var input ast.Value + + if rawInput == nil { + return input, nil + } + + m.Timer(metrics.RegoInputParse).Start() + defer m.Timer(metrics.RegoInputParse).Stop() + + rawPtr := util.Reference(rawInput) + + // roundtrip through json: this turns slices (e.g. []string, []bool) into + // []interface{}, the only array type ast.InterfaceToValue can work with + if err := util.RoundTrip(rawPtr); err != nil { + return nil, err + } + + return ast.InterfaceToValue(*rawPtr) +} + +func (r *Rego) parseQuery(futureImports []*ast.Import, m metrics.Metrics) (ast.Body, error) { + if r.parsedQuery != nil { + return r.parsedQuery, nil + } + + m.Timer(metrics.RegoQueryParse).Start() + defer m.Timer(metrics.RegoQueryParse).Stop() + + popts, err := future.ParserOptionsFromFutureImports(futureImports) + if err != nil { + return nil, err + } + return ast.ParseBodyWithOpts(r.query, popts) +} + +func (r *Rego) compileModules(ctx context.Context, txn storage.Transaction, m metrics.Metrics) error { + + // Only compile again if there are new modules. + if len(r.bundles) > 0 || len(r.parsedModules) > 0 { + + // The bundle.Activate call will activate any bundles passed in + // (ie compile + handle data store changes), and include any of + // the additional modules passed in. If no bundles are provided + // it will only compile the passed in modules. + // Use this as the single-point of compiling everything only a + // single time. + opts := &bundle.ActivateOpts{ + Ctx: ctx, + Store: r.store, + Txn: txn, + Compiler: r.compilerForTxn(ctx, r.store, txn), + Metrics: m, + Bundles: r.bundles, + ExtraModules: r.parsedModules, + } + err := bundle.Activate(opts) + if err != nil { + return err + } + } + + // Ensure all configured resolvers from the store are loaded. Skip if any were explicitly provided. + if len(r.resolvers) == 0 { + resolvers, err := bundleUtils.LoadWasmResolversFromStore(ctx, r.store, txn, r.bundles) + if err != nil { + return err + } + + for _, rslvr := range resolvers { + for _, ep := range rslvr.Entrypoints() { + r.resolvers = append(r.resolvers, refResolver{ep, rslvr}) + } + } + } + return nil +} + +func (r *Rego) compileAndCacheQuery(qType queryType, query ast.Body, imports []*ast.Import, m metrics.Metrics, extras []extraStage) error { + m.Timer(metrics.RegoQueryCompile).Start() + defer m.Timer(metrics.RegoQueryCompile).Stop() + + cachedQuery, ok := r.compiledQueries[qType] + if ok && cachedQuery.query != nil && cachedQuery.compiler != nil { + return nil + } + + qc, compiled, err := r.compileQuery(query, imports, m, extras) + if err != nil { + return err + } + + // cache the query for future use + r.compiledQueries[qType] = compiledQuery{ + query: compiled, + compiler: qc, + } + return nil +} + +func (r *Rego) prepareImports() ([]*ast.Import, error) { + imports := r.parsedImports + + if len(r.imports) > 0 { + s := make([]string, len(r.imports)) + for i := range r.imports { + s[i] = fmt.Sprintf("import %v", r.imports[i]) + } + parsed, err := ast.ParseImports(strings.Join(s, "\n")) + if err != nil { + return nil, err + } + imports = append(imports, parsed...) + } + return imports, nil +} + +func (r *Rego) compileQuery(query ast.Body, imports []*ast.Import, m metrics.Metrics, extras []extraStage) (ast.QueryCompiler, ast.Body, error) { + var pkg *ast.Package + + if r.pkg != "" { + var err error + pkg, err = ast.ParsePackage(fmt.Sprintf("package %v", r.pkg)) + if err != nil { + return nil, nil, err + } + } else { + pkg = r.parsedPackage + } + + qctx := ast.NewQueryContext(). + WithPackage(pkg). + WithImports(imports) + + qc := r.compiler.QueryCompiler(). + WithContext(qctx). + WithUnsafeBuiltins(r.unsafeBuiltins). + WithEnablePrintStatements(r.enablePrintStatements) + + for _, extra := range extras { + qc = qc.WithStageAfter(extra.after, extra.stage) + } + + compiled, err := qc.Compile(query) + + return qc, compiled, err + +} + +func (r *Rego) eval(ctx context.Context, ectx *EvalContext) (ResultSet, error) { + if r.opa != nil { + return r.evalWasm(ctx, ectx) + } + + q := topdown.NewQuery(ectx.compiledQuery.query). + WithQueryCompiler(ectx.compiledQuery.compiler). + WithCompiler(r.compiler). + WithStore(r.store). + WithTransaction(ectx.txn). + WithBuiltins(r.builtinFuncs). + WithMetrics(ectx.metrics). + WithInstrumentation(ectx.instrumentation). + WithRuntime(r.runtime). + WithIndexing(ectx.indexing). + WithEarlyExit(ectx.earlyExit). + WithInterQueryBuiltinCache(ectx.interQueryBuiltinCache). + WithStrictBuiltinErrors(r.strictBuiltinErrors). + WithSeed(ectx.seed). + WithPrintHook(ectx.printHook). + WithDistributedTracingOpts(r.distributedTacingOpts) + + if !ectx.time.IsZero() { + q = q.WithTime(ectx.time) + } + + for i := range ectx.queryTracers { + q = q.WithQueryTracer(ectx.queryTracers[i]) + } + + if ectx.parsedInput != nil { + q = q.WithInput(ast.NewTerm(ectx.parsedInput)) + } + + for i := range ectx.resolvers { + q = q.WithResolver(ectx.resolvers[i].ref, ectx.resolvers[i].r) + } + + // Cancel query if context is cancelled or deadline is reached. + c := topdown.NewCancel() + q = q.WithCancel(c) + exit := make(chan struct{}) + defer close(exit) + go waitForDone(ctx, exit, func() { + c.Cancel() + }) + + var rs ResultSet + err := q.Iter(ctx, func(qr topdown.QueryResult) error { + result, err := r.generateResult(qr, ectx) + if err != nil { + return err + } + rs = append(rs, result) + return nil + }) + + if err != nil { + return nil, err + } + + if len(rs) == 0 { + return nil, nil + } + + return rs, nil +} + +func (r *Rego) evalWasm(ctx context.Context, ectx *EvalContext) (ResultSet, error) { + + input := ectx.rawInput + if ectx.parsedInput != nil { + i := interface{}(ectx.parsedInput) + input = &i + } + result, err := r.opa.Eval(ctx, opa.EvalOpts{ + Metrics: r.metrics, + Input: input, + Time: ectx.time, + Seed: ectx.seed, + InterQueryBuiltinCache: ectx.interQueryBuiltinCache, + PrintHook: ectx.printHook, + Capabilities: ectx.capabilities, + }) + if err != nil { + return nil, err + } + + parsed, err := ast.ParseTerm(string(result.Result)) + if err != nil { + return nil, err + } + + resultSet, ok := parsed.Value.(ast.Set) + if !ok { + return nil, fmt.Errorf("illegal result type") + } + + if resultSet.Len() == 0 { + return nil, nil + } + + var rs ResultSet + err = resultSet.Iter(func(term *ast.Term) error { + obj, ok := term.Value.(ast.Object) + if !ok { + return fmt.Errorf("illegal result type") + } + qr := topdown.QueryResult{} + obj.Foreach(func(k, v *ast.Term) { + kvt := ast.VarTerm(string(k.Value.(ast.String))) + qr[kvt.Value.(ast.Var)] = v + }) + result, err := r.generateResult(qr, ectx) + if err != nil { + return err + } + rs = append(rs, result) + return nil + }) + + return rs, err +} + +func (r *Rego) generateResult(qr topdown.QueryResult, ectx *EvalContext) (Result, error) { + + rewritten := ectx.compiledQuery.compiler.RewrittenVars() + + result := newResult() + for k, term := range qr { + v, err := r.generateJSON(term, ectx) + if err != nil { + return result, err + } + + if rw, ok := rewritten[k]; ok { + k = rw + } + if isTermVar(k) || isTermWasmVar(k) || k.IsGenerated() || k.IsWildcard() { + continue + } + result.Bindings[string(k)] = v + } + + for _, expr := range ectx.compiledQuery.query { + if expr.Generated { + continue + } + + if k, ok := r.capture[expr]; ok { + v, err := r.generateJSON(qr[k], ectx) + if err != nil { + return result, err + } + result.Expressions = append(result.Expressions, newExpressionValue(expr, v)) + } else { + result.Expressions = append(result.Expressions, newExpressionValue(expr, true)) + } + + } + return result, nil +} + +func (r *Rego) partialResult(ctx context.Context, pCfg *PrepareConfig) (PartialResult, error) { + + err := r.prepare(ctx, partialResultQueryType, []extraStage{ + { + after: "ResolveRefs", + stage: ast.QueryCompilerStageDefinition{ + Name: "RewriteForPartialEval", + MetricName: "query_compile_stage_rewrite_for_partial_eval", + Stage: r.rewriteQueryForPartialEval, + }, + }, + }) + if err != nil { + return PartialResult{}, err + } + + ectx := &EvalContext{ + parsedInput: r.parsedInput, + metrics: r.metrics, + txn: r.txn, + partialNamespace: r.partialNamespace, + queryTracers: r.queryTracers, + compiledQuery: r.compiledQueries[partialResultQueryType], + instrumentation: r.instrumentation, + indexing: true, + resolvers: r.resolvers, + capabilities: r.capabilities, + } + + disableInlining := r.disableInlining + + if pCfg.disableInlining != nil { + disableInlining = *pCfg.disableInlining + } + + ectx.disableInlining, err = parseStringsToRefs(disableInlining) + if err != nil { + return PartialResult{}, err + } + + pq, err := r.partial(ctx, ectx) + if err != nil { + return PartialResult{}, err + } + + // Construct module for queries. + id := fmt.Sprintf("__partialresult__%s__", ectx.partialNamespace) + + module, err := ast.ParseModule(id, "package "+ectx.partialNamespace) + if err != nil { + return PartialResult{}, fmt.Errorf("bad partial namespace") + } + + module.Rules = make([]*ast.Rule, len(pq.Queries)) + for i, body := range pq.Queries { + rule := &ast.Rule{ + Head: ast.NewHead(ast.Var("__result__"), nil, ast.Wildcard), + Body: body, + Module: module, + } + module.Rules[i] = rule + if checkPartialResultForRecursiveRefs(body, rule.Path()) { + return PartialResult{}, Errors{errPartialEvaluationNotEffective} + } + } + + // Update compiler with partial evaluation output. + r.compiler.Modules[id] = module + for i, module := range pq.Support { + r.compiler.Modules[fmt.Sprintf("__partialsupport__%s__%d__", ectx.partialNamespace, i)] = module + } + + r.metrics.Timer(metrics.RegoModuleCompile).Start() + r.compilerForTxn(ctx, r.store, r.txn).Compile(r.compiler.Modules) + r.metrics.Timer(metrics.RegoModuleCompile).Stop() + + if r.compiler.Failed() { + return PartialResult{}, r.compiler.Errors + } + + result := PartialResult{ + compiler: r.compiler, + store: r.store, + body: ast.MustParseBody(fmt.Sprintf("data.%v.__result__", ectx.partialNamespace)), + builtinDecls: r.builtinDecls, + builtinFuncs: r.builtinFuncs, + } + + return result, nil +} + +func (r *Rego) partial(ctx context.Context, ectx *EvalContext) (*PartialQueries, error) { + + var unknowns []*ast.Term + + if ectx.parsedUnknowns != nil { + unknowns = ectx.parsedUnknowns + } else if ectx.unknowns != nil { + unknowns = make([]*ast.Term, len(ectx.unknowns)) + for i := range ectx.unknowns { + var err error + unknowns[i], err = ast.ParseTerm(ectx.unknowns[i]) + if err != nil { + return nil, err + } + } + } else { + // Use input document as unknown if caller has not specified any. + unknowns = []*ast.Term{ast.NewTerm(ast.InputRootRef)} + } + + q := topdown.NewQuery(ectx.compiledQuery.query). + WithQueryCompiler(ectx.compiledQuery.compiler). + WithCompiler(r.compiler). + WithStore(r.store). + WithTransaction(ectx.txn). + WithBuiltins(r.builtinFuncs). + WithMetrics(ectx.metrics). + WithInstrumentation(ectx.instrumentation). + WithUnknowns(unknowns). + WithDisableInlining(ectx.disableInlining). + WithRuntime(r.runtime). + WithIndexing(ectx.indexing). + WithEarlyExit(ectx.earlyExit). + WithPartialNamespace(ectx.partialNamespace). + WithSkipPartialNamespace(r.skipPartialNamespace). + WithShallowInlining(r.shallowInlining). + WithInterQueryBuiltinCache(ectx.interQueryBuiltinCache). + WithStrictBuiltinErrors(r.strictBuiltinErrors). + WithSeed(ectx.seed). + WithPrintHook(ectx.printHook) + + if !ectx.time.IsZero() { + q = q.WithTime(ectx.time) + } + + for i := range ectx.queryTracers { + q = q.WithQueryTracer(ectx.queryTracers[i]) + } + + if ectx.parsedInput != nil { + q = q.WithInput(ast.NewTerm(ectx.parsedInput)) + } + + for i := range ectx.resolvers { + q = q.WithResolver(ectx.resolvers[i].ref, ectx.resolvers[i].r) + } + + // Cancel query if context is cancelled or deadline is reached. + c := topdown.NewCancel() + q = q.WithCancel(c) + exit := make(chan struct{}) + defer close(exit) + go waitForDone(ctx, exit, func() { + c.Cancel() + }) + + queries, support, err := q.PartialRun(ctx) + if err != nil { + return nil, err + } + + pq := &PartialQueries{ + Queries: queries, + Support: support, + } + + return pq, nil +} + +func (r *Rego) rewriteQueryToCaptureValue(qc ast.QueryCompiler, query ast.Body) (ast.Body, error) { + + checkCapture := iteration(query) || len(query) > 1 + + for _, expr := range query { + + if expr.Negated { + continue + } + + if expr.IsAssignment() || expr.IsEquality() { + continue + } + + var capture *ast.Term + + // If the expression can be evaluated as a function, rewrite it to + // capture the return value. E.g., neq(1,2) becomes neq(1,2,x) but + // plus(1,2,x) does not get rewritten. + switch terms := expr.Terms.(type) { + case *ast.Term: + capture = r.generateTermVar() + expr.Terms = ast.Equality.Expr(terms, capture).Terms + r.capture[expr] = capture.Value.(ast.Var) + case []*ast.Term: + tpe := r.compiler.TypeEnv.Get(terms[0]) + if !types.Void(tpe) && types.Arity(tpe) == len(terms)-1 { + capture = r.generateTermVar() + expr.Terms = append(terms, capture) + r.capture[expr] = capture.Value.(ast.Var) + } + } + + if capture != nil && checkCapture { + cpy := expr.Copy() + cpy.Terms = capture + cpy.Generated = true + cpy.With = nil + query.Append(cpy) + } + } + + return query, nil +} + +func (r *Rego) rewriteQueryForPartialEval(_ ast.QueryCompiler, query ast.Body) (ast.Body, error) { + if len(query) != 1 { + return nil, fmt.Errorf("partial evaluation requires single ref (not multiple expressions)") + } + + term, ok := query[0].Terms.(*ast.Term) + if !ok { + return nil, fmt.Errorf("partial evaluation requires ref (not expression)") + } + + ref, ok := term.Value.(ast.Ref) + if !ok { + return nil, fmt.Errorf("partial evaluation requires ref (not %v)", ast.TypeName(term.Value)) + } + + if !ref.IsGround() { + return nil, fmt.Errorf("partial evaluation requires ground ref") + } + + return ast.NewBody(ast.Equality.Expr(ast.Wildcard, term)), nil +} + +// rewriteEqualsForPartialQueryCompile will rewrite == to = in queries. Normally +// this wouldn't be done, except for handling queries with the `Partial` API +// where rewriting them can substantially simplify the result, and it is unlikely +// that the caller would need expression values. +func (r *Rego) rewriteEqualsForPartialQueryCompile(_ ast.QueryCompiler, query ast.Body) (ast.Body, error) { + doubleEq := ast.Equal.Ref() + unifyOp := ast.Equality.Ref() + ast.WalkExprs(query, func(x *ast.Expr) bool { + if x.IsCall() { + operator := x.Operator() + if operator.Equal(doubleEq) && len(x.Operands()) == 2 { + x.SetOperator(ast.NewTerm(unifyOp)) + } + } + return false + }) + return query, nil +} + +func (r *Rego) generateTermVar() *ast.Term { + r.termVarID++ + + if r.target == targetWasm { + return ast.VarTerm(wasmVarPrefix + fmt.Sprintf("term%v", r.termVarID)) + } + return ast.VarTerm(ast.WildcardPrefix + fmt.Sprintf("term%v", r.termVarID)) +} + +func (r Rego) hasQuery() bool { + return len(r.query) != 0 || len(r.parsedQuery) != 0 +} + +func (r Rego) hasWasmModule() bool { + for _, b := range r.bundles { + if len(b.WasmModules) > 0 { + return true + } + } + return false +} + +type transactionCloser func(ctx context.Context, err error) error + +// getTxn will conditionally create a read or write transaction suitable for +// the configured Rego object. The returned function should be used to close the txn +// regardless of status. +func (r *Rego) getTxn(ctx context.Context) (storage.Transaction, transactionCloser, error) { + + noopCloser := func(ctx context.Context, err error) error { + return nil // no-op default + } + + if r.txn != nil { + // Externally provided txn + return r.txn, noopCloser, nil + } + + // Create a new transaction.. + params := storage.TransactionParams{} + + // Bundles and data paths may require writing data files or manifests to storage + if len(r.bundles) > 0 || len(r.bundlePaths) > 0 || len(r.loadPaths.paths) > 0 { + + // If we were given a store we will *not* write to it, only do that on one + // which was created automatically on behalf of the user. + if !r.ownStore { + return nil, noopCloser, errors.New("unable to start write transaction when store was provided") + } + + params.Write = true + } + + txn, err := r.store.NewTransaction(ctx, params) + if err != nil { + return nil, noopCloser, err + } + + // Setup a closer function that will abort or commit as needed. + closer := func(ctx context.Context, txnErr error) error { + var err error + + if txnErr == nil && params.Write { + err = r.store.Commit(ctx, txn) + } else { + r.store.Abort(ctx, txn) + } + + // Clear the auto created transaction now that it is closed. + r.txn = nil + + return err + } + + return txn, closer, nil +} + +func (r *Rego) compilerForTxn(ctx context.Context, store storage.Store, txn storage.Transaction) *ast.Compiler { + // Update the compiler to have a valid path conflict check + // for the current context and transaction. + return r.compiler.WithPathConflictsCheck(storage.NonEmpty(ctx, store, txn)) +} + +func checkPartialResultForRecursiveRefs(body ast.Body, path ast.Ref) bool { + var stop bool + ast.WalkRefs(body, func(x ast.Ref) bool { + if !stop { + if path.HasPrefix(x) { + stop = true + } + } + return stop + }) + return stop +} + +func isTermVar(v ast.Var) bool { + return strings.HasPrefix(string(v), ast.WildcardPrefix+"term") +} + +func isTermWasmVar(v ast.Var) bool { + return strings.HasPrefix(string(v), wasmVarPrefix+"term") +} + +func waitForDone(ctx context.Context, exit chan struct{}, f func()) { + select { + case <-exit: + return + case <-ctx.Done(): + f() + return + } +} + +type rawModule struct { + filename string + module string +} + +func (m rawModule) Parse() (*ast.Module, error) { + return ast.ParseModule(m.filename, m.module) +} + +type extraStage struct { + after string + stage ast.QueryCompilerStageDefinition +} + +type refResolver struct { + ref ast.Ref + r resolver.Resolver +} + +func iteration(x interface{}) bool { + + var stopped bool + + vis := ast.NewGenericVisitor(func(x interface{}) bool { + switch x := x.(type) { + case *ast.Term: + if ast.IsComprehension(x.Value) { + return true + } + case ast.Ref: + if !stopped { + if bi := ast.BuiltinMap[x.String()]; bi != nil { + if bi.Relation { + stopped = true + return stopped + } + } + for i := 1; i < len(x); i++ { + if _, ok := x[i].Value.(ast.Var); ok { + stopped = true + return stopped + } + } + } + return stopped + } + return stopped + }) + + vis.Walk(x) + + return stopped +} + +func parseStringsToRefs(s []string) ([]ast.Ref, error) { + + refs := make([]ast.Ref, len(s)) + for i := range refs { + var err error + refs[i], err = ast.ParseRef(s[i]) + if err != nil { + return nil, err + } + } + + return refs, nil +} + +// helper function to finish a built-in function call. If an error occurred, +// wrap the error and return it. Otherwise, invoke the iterator if the result +// was defined. +func finishFunction(name string, bctx topdown.BuiltinContext, result *ast.Term, err error, iter func(*ast.Term) error) error { + if err != nil { + var e *HaltError + if errors.As(err, &e) { + return topdown.Halt{Err: &topdown.Error{ + Code: topdown.BuiltinErr, + Message: fmt.Sprintf("%v: %v", name, e.Error()), + Location: bctx.Location, + }} + } + return &topdown.Error{ + Code: topdown.BuiltinErr, + Message: fmt.Sprintf("%v: %v", name, err.Error()), + Location: bctx.Location, + } + } + if result == nil { + return nil + } + return iter(result) +} + +// helper function to return an option that sets a custom built-in function. +func newFunction(decl *Function, f topdown.BuiltinFunc) func(*Rego) { + return func(r *Rego) { + r.builtinDecls[decl.Name] = &ast.Builtin{ + Name: decl.Name, + Decl: decl.Decl, + } + r.builtinFuncs[decl.Name] = &topdown.Builtin{ + Decl: r.builtinDecls[decl.Name], + Func: f, + } + } +} + +func generateJSON(term *ast.Term, ectx *EvalContext) (interface{}, error) { + return ast.JSONWithOpt(term.Value, ast.JSONOpt{SortSets: ectx.sortSets}) +} diff --git a/vendor/github.com/open-policy-agent/opa/rego/resultset.go b/vendor/github.com/open-policy-agent/opa/rego/resultset.go new file mode 100644 index 00000000..e60fa6fb --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/rego/resultset.go @@ -0,0 +1,90 @@ +package rego + +import ( + "fmt" + + "github.com/open-policy-agent/opa/ast" +) + +// ResultSet represents a collection of output from Rego evaluation. An empty +// result set represents an undefined query. +type ResultSet []Result + +// Vars represents a collection of variable bindings. The keys are the variable +// names and the values are the binding values. +type Vars map[string]interface{} + +// WithoutWildcards returns a copy of v with wildcard variables removed. +func (v Vars) WithoutWildcards() Vars { + n := Vars{} + for k, v := range v { + if ast.Var(k).IsWildcard() || ast.Var(k).IsGenerated() { + continue + } + n[k] = v + } + return n +} + +// Result defines the output of Rego evaluation. +type Result struct { + Expressions []*ExpressionValue `json:"expressions"` + Bindings Vars `json:"bindings,omitempty"` +} + +func newResult() Result { + return Result{ + Bindings: Vars{}, + } +} + +// Location defines a position in a Rego query or module. +type Location struct { + Row int `json:"row"` + Col int `json:"col"` +} + +// ExpressionValue defines the value of an expression in a Rego query. +type ExpressionValue struct { + Value interface{} `json:"value"` + Text string `json:"text"` + Location *Location `json:"location"` +} + +func newExpressionValue(expr *ast.Expr, value interface{}) *ExpressionValue { + result := &ExpressionValue{ + Value: value, + } + if expr.Location != nil { + result.Text = string(expr.Location.Text) + result.Location = &Location{ + Row: expr.Location.Row, + Col: expr.Location.Col, + } + } + return result +} + +func (ev *ExpressionValue) String() string { + return fmt.Sprint(ev.Value) +} + +// Allowed is a helper method that'll return true if all of these conditions hold: +// - the result set only has one element +// - there is only one expression in the result set's only element +// - that expression has the value `true` +// - there are no bindings. +// +// If bindings are present, this will yield `false`: it would be a pitfall to +// return `true` for a query like `data.authz.allow = x`, which always has result +// set element with value true, but could also have a binding `x: false`. +func (rs ResultSet) Allowed() bool { + if len(rs) == 1 && len(rs[0].Bindings) == 0 { + if exprs := rs[0].Expressions; len(exprs) == 1 { + if b, ok := exprs[0].Value.(bool); ok { + return b + } + } + } + return false +} diff --git a/vendor/github.com/open-policy-agent/opa/resolver/interface.go b/vendor/github.com/open-policy-agent/opa/resolver/interface.go new file mode 100644 index 00000000..fc02329f --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/resolver/interface.go @@ -0,0 +1,29 @@ +// Copyright 2020 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package resolver + +import ( + "context" + + "github.com/open-policy-agent/opa/ast" + "github.com/open-policy-agent/opa/metrics" +) + +// Resolver defines an external value resolver for OPA evaluations. +type Resolver interface { + Eval(context.Context, Input) (Result, error) +} + +// Input as provided to a Resolver instance when evaluating. +type Input struct { + Ref ast.Ref + Input *ast.Term + Metrics metrics.Metrics +} + +// Result of resolving a ref. +type Result struct { + Value ast.Value +} diff --git a/vendor/github.com/open-policy-agent/opa/resolver/wasm/wasm.go b/vendor/github.com/open-policy-agent/opa/resolver/wasm/wasm.go new file mode 100644 index 00000000..9c13879d --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/resolver/wasm/wasm.go @@ -0,0 +1,173 @@ +// Copyright 2020 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package wasm + +import ( + "context" + "fmt" + "strconv" + + "github.com/open-policy-agent/opa/ast" + "github.com/open-policy-agent/opa/internal/rego/opa" + "github.com/open-policy-agent/opa/resolver" +) + +// New creates a new Resolver instance which is using the Wasm module +// policy for the given entrypoint ref. +func New(entrypoints []ast.Ref, policy []byte, data interface{}) (*Resolver, error) { + e, err := opa.LookupEngine("wasm") + if err != nil { + return nil, err + } + o, err := e.New(). + WithPolicyBytes(policy). + WithDataJSON(data). + Init() + if err != nil { + return nil, err + } + + // Construct a quick lookup table of ref -> entrypoint ID + // for handling evaluations. Only the entrypoints provided + // by the caller will be constructed, this may be a subset + // of entrypoints available in the Wasm module, however + // only the configured ones will be used when Eval() is + // called. + entrypointRefToID := ast.NewValueMap() + epIDs, err := o.Entrypoints(context.Background()) + if err != nil { + return nil, err + } + for path, id := range epIDs { + for _, ref := range entrypoints { + refPtr, err := ref.Ptr() + if err != nil { + return nil, err + } + if refPtr == path { + entrypointRefToID.Put(ref, ast.Number(strconv.Itoa(int(id)))) + } + } + } + + return &Resolver{ + entrypoints: entrypoints, + entrypointIDs: entrypointRefToID, + o: o, + }, nil +} + +// Resolver implements the resolver.Resolver interface +// using Wasm modules to perform an evaluation. +type Resolver struct { + entrypoints []ast.Ref + entrypointIDs *ast.ValueMap + o opa.EvalEngine +} + +// Entrypoints returns a list of entrypoints this resolver is configured to +// perform evaluations on. +func (r *Resolver) Entrypoints() []ast.Ref { + return r.entrypoints +} + +// Close shuts down the resolver. +func (r *Resolver) Close() { + r.o.Close() +} + +// Eval performs an evaluation using the provided input and the Wasm module +// associated with this Resolver instance. +func (r *Resolver) Eval(ctx context.Context, input resolver.Input) (resolver.Result, error) { + v := r.entrypointIDs.Get(input.Ref) + if v == nil { + return resolver.Result{}, fmt.Errorf("unknown entrypoint %s", input.Ref) + } + + numValue, ok := v.(ast.Number) + if !ok { + return resolver.Result{}, fmt.Errorf("internal error: invalid entrypoint id %s", numValue) + } + + epID, ok := numValue.Int() + if !ok { + return resolver.Result{}, fmt.Errorf("internal error: invalid entrypoint id %s", numValue) + } + + var in *interface{} + if input.Input != nil { + var str interface{} = []byte(input.Input.String()) + in = &str + } + + opts := opa.EvalOpts{ + Input: in, + Entrypoint: int32(epID), + Metrics: input.Metrics, + } + out, err := r.o.Eval(ctx, opts) + if err != nil { + return resolver.Result{}, err + } + + result, err := getResult(out) + if err != nil { + return resolver.Result{}, err + } + + return resolver.Result{Value: result}, nil +} + +// SetData will update the external data for the Wasm instance. +func (r *Resolver) SetData(ctx context.Context, data interface{}) error { + return r.o.SetData(ctx, data) +} + +// SetDataPath will set the provided data on the wasm instance at the specified path. +func (r *Resolver) SetDataPath(ctx context.Context, path []string, data interface{}) error { + return r.o.SetDataPath(ctx, path, data) +} + +// RemoveDataPath will remove any data at the specified path. +func (r *Resolver) RemoveDataPath(ctx context.Context, path []string) error { + return r.o.RemoveDataPath(ctx, path) +} + +func getResult(evalResult *opa.Result) (ast.Value, error) { + + parsed, err := ast.ParseTerm(string(evalResult.Result)) + if err != nil { + return nil, fmt.Errorf("failed to parse wasm result: %s", err) + } + + resultSet, ok := parsed.Value.(ast.Set) + if !ok { + return nil, fmt.Errorf("illegal result type") + } + + if resultSet.Len() == 0 { + return nil, nil + } + + if resultSet.Len() > 1 { + return nil, fmt.Errorf("illegal result type") + } + + var obj ast.Object + err = resultSet.Iter(func(term *ast.Term) error { + obj, ok = term.Value.(ast.Object) + if !ok || obj.Len() != 1 { + return fmt.Errorf("illegal result type") + } + return nil + }) + if err != nil { + return nil, err + } + + result := obj.Get(ast.StringTerm("result")) + + return result.Value, nil +} diff --git a/vendor/github.com/open-policy-agent/opa/storage/doc.go b/vendor/github.com/open-policy-agent/opa/storage/doc.go new file mode 100644 index 00000000..6fa2f86d --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/storage/doc.go @@ -0,0 +1,6 @@ +// Copyright 2016 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +// Package storage exposes the policy engine's storage layer. +package storage diff --git a/vendor/github.com/open-policy-agent/opa/storage/errors.go b/vendor/github.com/open-policy-agent/opa/storage/errors.go new file mode 100644 index 00000000..d83b275a --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/storage/errors.go @@ -0,0 +1,122 @@ +// Copyright 2016 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package storage + +import ( + "fmt" +) + +const ( + // InternalErr indicates an unknown, internal error has occurred. + InternalErr = "storage_internal_error" + + // NotFoundErr indicates the path used in the storage operation does not + // locate a document. + NotFoundErr = "storage_not_found_error" + + // WriteConflictErr indicates a write on the path enocuntered a conflicting + // value inside the transaction. + WriteConflictErr = "storage_write_conflict_error" + + // InvalidPatchErr indicates an invalid patch/write was issued. The patch + // was rejected. + InvalidPatchErr = "storage_invalid_patch_error" + + // InvalidTransactionErr indicates an invalid operation was performed + // inside of the transaction. + InvalidTransactionErr = "storage_invalid_txn_error" + + // TriggersNotSupportedErr indicates the caller attempted to register a + // trigger against a store that does not support them. + TriggersNotSupportedErr = "storage_triggers_not_supported_error" + + // WritesNotSupportedErr indicate the caller attempted to perform a write + // against a store that does not support them. + WritesNotSupportedErr = "storage_writes_not_supported_error" + + // PolicyNotSupportedErr indicate the caller attempted to perform a policy + // management operation against a store that does not support them. + PolicyNotSupportedErr = "storage_policy_not_supported_error" +) + +// Error is the error type returned by the storage layer. +type Error struct { + Code string `json:"code"` + Message string `json:"message"` +} + +func (err *Error) Error() string { + if err.Message != "" { + return fmt.Sprintf("%v: %v", err.Code, err.Message) + } + return string(err.Code) +} + +// IsNotFound returns true if this error is a NotFoundErr. +func IsNotFound(err error) bool { + switch err := err.(type) { + case *Error: + return err.Code == NotFoundErr + } + return false +} + +// IsWriteConflictError returns true if this error a WriteConflictErr. +func IsWriteConflictError(err error) bool { + switch err := err.(type) { + case *Error: + return err.Code == WriteConflictErr + } + return false +} + +// IsInvalidPatch returns true if this error is a InvalidPatchErr. +func IsInvalidPatch(err error) bool { + switch err := err.(type) { + case *Error: + return err.Code == InvalidPatchErr + } + return false +} + +// IsInvalidTransaction returns true if this error is a InvalidTransactionErr. +func IsInvalidTransaction(err error) bool { + switch err := err.(type) { + case *Error: + return err.Code == InvalidTransactionErr + } + return false +} + +// IsIndexingNotSupported is a stub for backwards-compatibility. +// +// Deprecated: We no longer return IndexingNotSupported errors, so it is +// unnecessary to check for them. +func IsIndexingNotSupported(error) bool { return false } + +func writeConflictError(path Path) *Error { + return &Error{ + Code: WriteConflictErr, + Message: fmt.Sprint(path), + } +} + +func triggersNotSupportedError() *Error { + return &Error{ + Code: TriggersNotSupportedErr, + } +} + +func writesNotSupportedError() *Error { + return &Error{ + Code: WritesNotSupportedErr, + } +} + +func policyNotSupportedError() *Error { + return &Error{ + Code: PolicyNotSupportedErr, + } +} diff --git a/vendor/github.com/open-policy-agent/opa/storage/inmem/inmem.go b/vendor/github.com/open-policy-agent/opa/storage/inmem/inmem.go new file mode 100644 index 00000000..b6918004 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/storage/inmem/inmem.go @@ -0,0 +1,250 @@ +// Copyright 2016 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +// Package inmem implements an in-memory version of the policy engine's storage +// layer. +// +// The in-memory store is used as the default storage layer implementation. The +// in-memory store supports multi-reader/single-writer concurrency with +// rollback. +// +// Callers should assume the in-memory store does not make copies of written +// data. Once data is written to the in-memory store, it should not be modified +// (outside of calling Store.Write). Furthermore, data read from the in-memory +// store should be treated as read-only. +package inmem + +import ( + "context" + "fmt" + "io" + "sync" + "sync/atomic" + + "github.com/open-policy-agent/opa/storage" + "github.com/open-policy-agent/opa/util" +) + +// New returns an empty in-memory store. +func New() storage.Store { + return &store{ + data: map[string]interface{}{}, + triggers: map[*handle]storage.TriggerConfig{}, + policies: map[string][]byte{}, + } +} + +// NewFromObject returns a new in-memory store from the supplied data object. +func NewFromObject(data map[string]interface{}) storage.Store { + db := New() + ctx := context.Background() + txn, err := db.NewTransaction(ctx, storage.WriteParams) + if err != nil { + panic(err) + } + if err := db.Write(ctx, txn, storage.AddOp, storage.Path{}, data); err != nil { + panic(err) + } + if err := db.Commit(ctx, txn); err != nil { + panic(err) + } + return db +} + +// NewFromReader returns a new in-memory store from a reader that produces a +// JSON serialized object. This function is for test purposes. +func NewFromReader(r io.Reader) storage.Store { + d := util.NewJSONDecoder(r) + var data map[string]interface{} + if err := d.Decode(&data); err != nil { + panic(err) + } + return NewFromObject(data) +} + +type store struct { + rmu sync.RWMutex // reader-writer lock + wmu sync.Mutex // writer lock + xid uint64 // last generated transaction id + data map[string]interface{} // raw data + policies map[string][]byte // raw policies + triggers map[*handle]storage.TriggerConfig // registered triggers +} + +type handle struct { + db *store +} + +func (db *store) NewTransaction(ctx context.Context, params ...storage.TransactionParams) (storage.Transaction, error) { + var write bool + var context *storage.Context + if len(params) > 0 { + write = params[0].Write + context = params[0].Context + } + xid := atomic.AddUint64(&db.xid, uint64(1)) + if write { + db.wmu.Lock() + } else { + db.rmu.RLock() + } + return newTransaction(xid, write, context, db), nil +} + +func (db *store) Commit(ctx context.Context, txn storage.Transaction) error { + underlying, err := db.underlying(txn) + if err != nil { + return err + } + if underlying.write { + db.rmu.Lock() + event := underlying.Commit() + db.runOnCommitTriggers(ctx, txn, event) + // Mark the transaction stale after executing triggers so they can + // perform store operations if needed. + underlying.stale = true + db.rmu.Unlock() + db.wmu.Unlock() + } else { + db.rmu.RUnlock() + } + return nil +} + +func (db *store) Abort(ctx context.Context, txn storage.Transaction) { + underlying, err := db.underlying(txn) + if err != nil { + panic(err) + } + underlying.stale = true + if underlying.write { + db.wmu.Unlock() + } else { + db.rmu.RUnlock() + } +} + +func (db *store) ListPolicies(_ context.Context, txn storage.Transaction) ([]string, error) { + underlying, err := db.underlying(txn) + if err != nil { + return nil, err + } + return underlying.ListPolicies(), nil +} + +func (db *store) GetPolicy(_ context.Context, txn storage.Transaction, id string) ([]byte, error) { + underlying, err := db.underlying(txn) + if err != nil { + return nil, err + } + return underlying.GetPolicy(id) +} + +func (db *store) UpsertPolicy(_ context.Context, txn storage.Transaction, id string, bs []byte) error { + underlying, err := db.underlying(txn) + if err != nil { + return err + } + return underlying.UpsertPolicy(id, bs) +} + +func (db *store) DeletePolicy(_ context.Context, txn storage.Transaction, id string) error { + underlying, err := db.underlying(txn) + if err != nil { + return err + } + if _, err := underlying.GetPolicy(id); err != nil { + return err + } + return underlying.DeletePolicy(id) +} + +func (db *store) Register(ctx context.Context, txn storage.Transaction, config storage.TriggerConfig) (storage.TriggerHandle, error) { + underlying, err := db.underlying(txn) + if err != nil { + return nil, err + } + if !underlying.write { + return nil, &storage.Error{ + Code: storage.InvalidTransactionErr, + Message: "triggers must be registered with a write transaction", + } + } + h := &handle{db} + db.triggers[h] = config + return h, nil +} + +func (db *store) Read(ctx context.Context, txn storage.Transaction, path storage.Path) (interface{}, error) { + underlying, err := db.underlying(txn) + if err != nil { + return nil, err + } + return underlying.Read(path) +} + +func (db *store) Write(ctx context.Context, txn storage.Transaction, op storage.PatchOp, path storage.Path, value interface{}) error { + underlying, err := db.underlying(txn) + if err != nil { + return err + } + val := util.Reference(value) + if err := util.RoundTrip(val); err != nil { + return err + } + return underlying.Write(op, path, *val) +} + +func (h *handle) Unregister(ctx context.Context, txn storage.Transaction) { + underlying, err := h.db.underlying(txn) + if err != nil { + panic(err) + } + if !underlying.write { + panic(&storage.Error{ + Code: storage.InvalidTransactionErr, + Message: "triggers must be unregistered with a write transaction", + }) + } + delete(h.db.triggers, h) +} + +func (db *store) runOnCommitTriggers(ctx context.Context, txn storage.Transaction, event storage.TriggerEvent) { + for _, t := range db.triggers { + t.OnCommit(ctx, txn, event) + } +} + +func (db *store) underlying(txn storage.Transaction) (*transaction, error) { + underlying, ok := txn.(*transaction) + if !ok { + return nil, &storage.Error{ + Code: storage.InvalidTransactionErr, + Message: fmt.Sprintf("unexpected transaction type %T", txn), + } + } + if underlying.db != db { + return nil, &storage.Error{ + Code: storage.InvalidTransactionErr, + Message: "unknown transaction", + } + } + if underlying.stale { + return nil, &storage.Error{ + Code: storage.InvalidTransactionErr, + Message: "stale transaction", + } + } + return underlying, nil +} + +var rootMustBeObjectMsg = "root must be object" +var rootCannotBeRemovedMsg = "root cannot be removed" + +func invalidPatchError(f string, a ...interface{}) *storage.Error { + return &storage.Error{ + Code: storage.InvalidPatchErr, + Message: fmt.Sprintf(f, a...), + } +} diff --git a/vendor/github.com/open-policy-agent/opa/storage/inmem/txn.go b/vendor/github.com/open-policy-agent/opa/storage/inmem/txn.go new file mode 100644 index 00000000..2f52972b --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/storage/inmem/txn.go @@ -0,0 +1,392 @@ +// Copyright 2017 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package inmem + +import ( + "container/list" + "encoding/json" + "strconv" + + "github.com/open-policy-agent/opa/internal/deepcopy" + "github.com/open-policy-agent/opa/storage" + "github.com/open-policy-agent/opa/storage/internal/errors" + "github.com/open-policy-agent/opa/storage/internal/ptr" +) + +// transaction implements the low-level read/write operations on the in-memory +// store and contains the state required for pending transactions. +// +// For write transactions, the struct contains a logical set of updates +// performed by write operations in the transaction. Each write operation +// compacts the set such that two updates never overlap: +// +// - If new update path is a prefix of existing update path, existing update is +// removed, new update is added. +// +// - If existing update path is a prefix of new update path, existing update is +// modified. +// +// - Otherwise, new update is added. +// +// Read transactions do not require any special handling and simply passthrough +// to the underlying store. Read transactions do not support upgrade. +type transaction struct { + xid uint64 + write bool + stale bool + db *store + updates *list.List + policies map[string]policyUpdate + context *storage.Context +} + +type policyUpdate struct { + value []byte + remove bool +} + +func newTransaction(xid uint64, write bool, context *storage.Context, db *store) *transaction { + return &transaction{ + xid: xid, + write: write, + db: db, + policies: map[string]policyUpdate{}, + updates: list.New(), + context: context, + } +} + +func (txn *transaction) ID() uint64 { + return txn.xid +} + +func (txn *transaction) Write(op storage.PatchOp, path storage.Path, value interface{}) error { + + if !txn.write { + return &storage.Error{ + Code: storage.InvalidTransactionErr, + Message: "data write during read transaction", + } + } + + if len(path) == 0 { + return txn.updateRoot(op, value) + } + + for curr := txn.updates.Front(); curr != nil; { + update := curr.Value.(*update) + + // Check if new update masks existing update exactly. In this case, the + // existing update can be removed and no other updates have to be + // visited (because no two updates overlap.) + if update.path.Equal(path) { + if update.remove { + if op != storage.AddOp { + return errors.NewNotFoundError(path) + } + } + txn.updates.Remove(curr) + break + } + + // Check if new update masks existing update. In this case, the + // existing update has to be removed but other updates may overlap, so + // we must continue. + if update.path.HasPrefix(path) { + remove := curr + curr = curr.Next() + txn.updates.Remove(remove) + continue + } + + // Check if new update modifies existing update. In this case, the + // existing update is mutated. + if path.HasPrefix(update.path) { + if update.remove { + return errors.NewNotFoundError(path) + } + suffix := path[len(update.path):] + newUpdate, err := newUpdate(update.value, op, suffix, 0, value) + if err != nil { + return err + } + update.value = newUpdate.Apply(update.value) + return nil + } + + curr = curr.Next() + } + + update, err := newUpdate(txn.db.data, op, path, 0, value) + if err != nil { + return err + } + + txn.updates.PushFront(update) + return nil +} + +func (txn *transaction) updateRoot(op storage.PatchOp, value interface{}) error { + if op == storage.RemoveOp { + return invalidPatchError(rootCannotBeRemovedMsg) + } + if _, ok := value.(map[string]interface{}); !ok { + return invalidPatchError(rootMustBeObjectMsg) + } + txn.updates.Init() + txn.updates.PushFront(&update{ + path: storage.Path{}, + remove: false, + value: value, + }) + return nil +} + +func (txn *transaction) Commit() (result storage.TriggerEvent) { + result.Context = txn.context + for curr := txn.updates.Front(); curr != nil; curr = curr.Next() { + action := curr.Value.(*update) + updated := action.Apply(txn.db.data) + txn.db.data = updated.(map[string]interface{}) + + result.Data = append(result.Data, storage.DataEvent{ + Path: action.path, + Data: action.value, + Removed: action.remove, + }) + } + for id, update := range txn.policies { + if update.remove { + delete(txn.db.policies, id) + } else { + txn.db.policies[id] = update.value + } + + result.Policy = append(result.Policy, storage.PolicyEvent{ + ID: id, + Data: update.value, + Removed: update.remove, + }) + } + return result +} + +func (txn *transaction) Read(path storage.Path) (interface{}, error) { + + if !txn.write { + return ptr.Ptr(txn.db.data, path) + } + + merge := []*update{} + + for curr := txn.updates.Front(); curr != nil; curr = curr.Next() { + + update := curr.Value.(*update) + + if path.HasPrefix(update.path) { + if update.remove { + return nil, errors.NewNotFoundError(path) + } + return ptr.Ptr(update.value, path[len(update.path):]) + } + + if update.path.HasPrefix(path) { + merge = append(merge, update) + } + } + + data, err := ptr.Ptr(txn.db.data, path) + + if err != nil { + return nil, err + } + + if len(merge) == 0 { + return data, nil + } + + cpy := deepcopy.DeepCopy(data) + + for _, update := range merge { + cpy = update.Relative(path).Apply(cpy) + } + + return cpy, nil +} + +func (txn *transaction) ListPolicies() []string { + var ids []string + for id := range txn.db.policies { + if _, ok := txn.policies[id]; !ok { + ids = append(ids, id) + } + } + for id, update := range txn.policies { + if !update.remove { + ids = append(ids, id) + } + } + return ids +} + +func (txn *transaction) GetPolicy(id string) ([]byte, error) { + if update, ok := txn.policies[id]; ok { + if !update.remove { + return update.value, nil + } + return nil, errors.NewNotFoundErrorf("policy id %q", id) + } + if exist, ok := txn.db.policies[id]; ok { + return exist, nil + } + return nil, errors.NewNotFoundErrorf("policy id %q", id) +} + +func (txn *transaction) UpsertPolicy(id string, bs []byte) error { + if !txn.write { + return &storage.Error{ + Code: storage.InvalidTransactionErr, + Message: "policy write during read transaction", + } + } + txn.policies[id] = policyUpdate{bs, false} + return nil +} + +func (txn *transaction) DeletePolicy(id string) error { + if !txn.write { + return &storage.Error{ + Code: storage.InvalidTransactionErr, + Message: "policy write during read transaction", + } + } + txn.policies[id] = policyUpdate{nil, true} + return nil +} + +// update contains state associated with an update to be applied to the +// in-memory data store. +type update struct { + path storage.Path // data path modified by update + remove bool // indicates whether update removes the value at path + value interface{} // value to add/replace at path (ignored if remove is true) +} + +func newUpdate(data interface{}, op storage.PatchOp, path storage.Path, idx int, value interface{}) (*update, error) { + + switch data := data.(type) { + case map[string]interface{}: + return newUpdateObject(data, op, path, idx, value) + + case []interface{}: + return newUpdateArray(data, op, path, idx, value) + + case nil, bool, json.Number, string: + return nil, errors.NewNotFoundError(path) + } + + return nil, &storage.Error{ + Code: storage.InternalErr, + Message: "invalid data value encountered", + } +} + +func newUpdateArray(data []interface{}, op storage.PatchOp, path storage.Path, idx int, value interface{}) (*update, error) { + + if idx == len(path)-1 { + if path[idx] == "-" { + if op != storage.AddOp { + return nil, invalidPatchError("%v: invalid patch path", path) + } + cpy := make([]interface{}, len(data)+1) + copy(cpy, data) + cpy[len(data)] = value + return &update{path[:len(path)-1], false, cpy}, nil + } + + pos, err := ptr.ValidateArrayIndex(data, path[idx], path) + if err != nil { + return nil, err + } + + if op == storage.AddOp { + cpy := make([]interface{}, len(data)+1) + copy(cpy[:pos], data[:pos]) + copy(cpy[pos+1:], data[pos:]) + cpy[pos] = value + return &update{path[:len(path)-1], false, cpy}, nil + + } else if op == storage.RemoveOp { + cpy := make([]interface{}, len(data)-1) + copy(cpy[:pos], data[:pos]) + copy(cpy[pos:], data[pos+1:]) + return &update{path[:len(path)-1], false, cpy}, nil + + } else { + cpy := make([]interface{}, len(data)) + copy(cpy, data) + cpy[pos] = value + return &update{path[:len(path)-1], false, cpy}, nil + } + } + + pos, err := ptr.ValidateArrayIndex(data, path[idx], path) + if err != nil { + return nil, err + } + + return newUpdate(data[pos], op, path, idx+1, value) +} + +func newUpdateObject(data map[string]interface{}, op storage.PatchOp, path storage.Path, idx int, value interface{}) (*update, error) { + + if idx == len(path)-1 { + switch op { + case storage.ReplaceOp, storage.RemoveOp: + if _, ok := data[path[idx]]; !ok { + return nil, errors.NewNotFoundError(path) + } + } + return &update{path, op == storage.RemoveOp, value}, nil + } + + if data, ok := data[path[idx]]; ok { + return newUpdate(data, op, path, idx+1, value) + } + + return nil, errors.NewNotFoundError(path) +} +func (u *update) Apply(data interface{}) interface{} { + if len(u.path) == 0 { + return u.value + } + parent, err := ptr.Ptr(data, u.path[:len(u.path)-1]) + if err != nil { + panic(err) + } + key := u.path[len(u.path)-1] + if u.remove { + obj := parent.(map[string]interface{}) + delete(obj, key) + return data + } + switch parent := parent.(type) { + case map[string]interface{}: + parent[key] = u.value + case []interface{}: + idx, err := strconv.Atoi(key) + if err != nil { + panic(err) + } + parent[idx] = u.value + } + return data +} + +func (u *update) Relative(path storage.Path) *update { + cpy := *u + cpy.path = cpy.path[len(path):] + return &cpy +} diff --git a/vendor/github.com/open-policy-agent/opa/storage/interface.go b/vendor/github.com/open-policy-agent/opa/storage/interface.go new file mode 100644 index 00000000..f1507161 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/storage/interface.go @@ -0,0 +1,194 @@ +// Copyright 2016 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package storage + +import ( + "context" +) + +// Transaction defines the interface that identifies a consistent snapshot over +// the policy engine's storage layer. +type Transaction interface { + ID() uint64 +} + +// Store defines the interface for the storage layer's backend. +type Store interface { + Trigger + Policy + + // NewTransaction is called create a new transaction in the store. + NewTransaction(ctx context.Context, params ...TransactionParams) (Transaction, error) + + // Read is called to fetch a document referred to by path. + Read(ctx context.Context, txn Transaction, path Path) (interface{}, error) + + // Write is called to modify a document referred to by path. + Write(ctx context.Context, txn Transaction, op PatchOp, path Path, value interface{}) error + + // Commit is called to finish the transaction. If Commit returns an error, the + // transaction must be automatically aborted by the Store implementation. + Commit(ctx context.Context, txn Transaction) error + + // Abort is called to cancel the transaction. + Abort(ctx context.Context, txn Transaction) +} + +// TransactionParams describes a new transaction. +type TransactionParams struct { + + // Write indicates if this transaction will perform any write operations. + Write bool + + // Context contains key/value pairs passed to triggers. + Context *Context +} + +// Context is a simple container for key/value pairs. +type Context struct { + values map[interface{}]interface{} +} + +// NewContext returns a new context object. +func NewContext() *Context { + return &Context{ + values: map[interface{}]interface{}{}, + } +} + +// Get returns the key value in the context. +func (ctx *Context) Get(key interface{}) interface{} { + if ctx == nil { + return nil + } + return ctx.values[key] +} + +// Put adds a key/value pair to the context. +func (ctx *Context) Put(key, value interface{}) { + ctx.values[key] = value +} + +// WriteParams specifies the TransactionParams for a write transaction. +var WriteParams = TransactionParams{ + Write: true, +} + +// PatchOp is the enumeration of supposed modifications. +type PatchOp int + +// Patch supports add, remove, and replace operations. +const ( + AddOp PatchOp = iota + RemoveOp = iota + ReplaceOp = iota +) + +// WritesNotSupported provides a default implementation of the write +// interface which may be used if the backend does not support writes. +type WritesNotSupported struct{} + +func (WritesNotSupported) Write(ctx context.Context, txn Transaction, op PatchOp, path Path, value interface{}) error { + return writesNotSupportedError() +} + +// Policy defines the interface for policy module storage. +type Policy interface { + ListPolicies(context.Context, Transaction) ([]string, error) + GetPolicy(context.Context, Transaction, string) ([]byte, error) + UpsertPolicy(context.Context, Transaction, string, []byte) error + DeletePolicy(context.Context, Transaction, string) error +} + +// PolicyNotSupported provides a default implementation of the policy interface +// which may be used if the backend does not support policy storage. +type PolicyNotSupported struct{} + +// ListPolicies always returns a PolicyNotSupportedErr. +func (PolicyNotSupported) ListPolicies(context.Context, Transaction) ([]string, error) { + return nil, policyNotSupportedError() +} + +// GetPolicy always returns a PolicyNotSupportedErr. +func (PolicyNotSupported) GetPolicy(context.Context, Transaction, string) ([]byte, error) { + return nil, policyNotSupportedError() +} + +// UpsertPolicy always returns a PolicyNotSupportedErr. +func (PolicyNotSupported) UpsertPolicy(context.Context, Transaction, string, []byte) error { + return policyNotSupportedError() +} + +// DeletePolicy always returns a PolicyNotSupportedErr. +func (PolicyNotSupported) DeletePolicy(context.Context, Transaction, string) error { + return policyNotSupportedError() +} + +// PolicyEvent describes a change to a policy. +type PolicyEvent struct { + ID string + Data []byte + Removed bool +} + +// DataEvent describes a change to a base data document. +type DataEvent struct { + Path Path + Data interface{} + Removed bool +} + +// TriggerEvent describes the changes that caused the trigger to be invoked. +type TriggerEvent struct { + Policy []PolicyEvent + Data []DataEvent + Context *Context +} + +// IsZero returns true if the TriggerEvent indicates no changes occurred. This +// function is primarily for test purposes. +func (e TriggerEvent) IsZero() bool { + return !e.PolicyChanged() && !e.DataChanged() +} + +// PolicyChanged returns true if the trigger was caused by a policy change. +func (e TriggerEvent) PolicyChanged() bool { + return len(e.Policy) > 0 +} + +// DataChanged returns true if the trigger was caused by a data change. +func (e TriggerEvent) DataChanged() bool { + return len(e.Data) > 0 +} + +// TriggerConfig contains the trigger registration configuration. +type TriggerConfig struct { + + // OnCommit is invoked when a transaction is successfully committed. The + // callback is invoked with a handle to the write transaction that + // successfully committed before other clients see the changes. + OnCommit func(ctx context.Context, txn Transaction, event TriggerEvent) +} + +// Trigger defines the interface that stores implement to register for change +// notifications when the store is changed. +type Trigger interface { + Register(ctx context.Context, txn Transaction, config TriggerConfig) (TriggerHandle, error) +} + +// TriggersNotSupported provides default implementations of the Trigger +// interface which may be used if the backend does not support triggers. +type TriggersNotSupported struct{} + +// Register always returns an error indicating triggers are not supported. +func (TriggersNotSupported) Register(context.Context, Transaction, TriggerConfig) (TriggerHandle, error) { + return nil, triggersNotSupportedError() +} + +// TriggerHandle defines the interface that can be used to unregister triggers that have +// been registered on a Store. +type TriggerHandle interface { + Unregister(ctx context.Context, txn Transaction) +} diff --git a/vendor/github.com/open-policy-agent/opa/storage/internal/errors/errors.go b/vendor/github.com/open-policy-agent/opa/storage/internal/errors/errors.go new file mode 100644 index 00000000..74792f63 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/storage/internal/errors/errors.go @@ -0,0 +1,32 @@ +// Copyright 2021 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +// Package errors contains reusable error-related code for the storage layer. +package errors + +import ( + "fmt" + + "github.com/open-policy-agent/opa/storage" +) + +const ArrayIndexTypeMsg = "array index must be integer" +const DoesNotExistMsg = "document does not exist" +const OutOfRangeMsg = "array index out of range" + +func NewNotFoundError(path storage.Path) *storage.Error { + return NewNotFoundErrorWithHint(path, DoesNotExistMsg) +} + +func NewNotFoundErrorWithHint(path storage.Path, hint string) *storage.Error { + return NewNotFoundErrorf("%v: %v", path.String(), hint) +} + +func NewNotFoundErrorf(f string, a ...interface{}) *storage.Error { + msg := fmt.Sprintf(f, a...) + return &storage.Error{ + Code: storage.NotFoundErr, + Message: msg, + } +} diff --git a/vendor/github.com/open-policy-agent/opa/storage/internal/ptr/ptr.go b/vendor/github.com/open-policy-agent/opa/storage/internal/ptr/ptr.go new file mode 100644 index 00000000..6778805f --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/storage/internal/ptr/ptr.go @@ -0,0 +1,48 @@ +// Copyright 2021 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +// Package ptr provides utilities for pointer operations using storage layer paths. +package ptr + +import ( + "strconv" + + "github.com/open-policy-agent/opa/storage" + "github.com/open-policy-agent/opa/storage/internal/errors" +) + +func Ptr(data interface{}, path storage.Path) (interface{}, error) { + node := data + for i := range path { + key := path[i] + switch curr := node.(type) { + case map[string]interface{}: + var ok bool + if node, ok = curr[key]; !ok { + return nil, errors.NewNotFoundError(path) + } + case []interface{}: + pos, err := ValidateArrayIndex(curr, key, path) + if err != nil { + return nil, err + } + node = curr[pos] + default: + return nil, errors.NewNotFoundError(path) + } + } + + return node, nil +} + +func ValidateArrayIndex(arr []interface{}, s string, path storage.Path) (int, error) { + idx, err := strconv.Atoi(s) + if err != nil { + return 0, errors.NewNotFoundErrorWithHint(path, errors.ArrayIndexTypeMsg) + } + if idx < 0 || idx >= len(arr) { + return 0, errors.NewNotFoundErrorWithHint(path, errors.OutOfRangeMsg) + } + return idx, nil +} diff --git a/vendor/github.com/open-policy-agent/opa/storage/path.go b/vendor/github.com/open-policy-agent/opa/storage/path.go new file mode 100644 index 00000000..02ef4cab --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/storage/path.go @@ -0,0 +1,154 @@ +// Copyright 2016 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package storage + +import ( + "fmt" + "net/url" + "strconv" + "strings" + + "github.com/open-policy-agent/opa/ast" +) + +// Path refers to a document in storage. +type Path []string + +// ParsePath returns a new path for the given str. +func ParsePath(str string) (path Path, ok bool) { + if len(str) == 0 { + return nil, false + } + if str[0] != '/' { + return nil, false + } + if len(str) == 1 { + return Path{}, true + } + parts := strings.Split(str[1:], "/") + return parts, true +} + +// ParsePathEscaped returns a new path for the given escaped str. +func ParsePathEscaped(str string) (path Path, ok bool) { + path, ok = ParsePath(str) + if !ok { + return + } + for i := range path { + segment, err := url.PathUnescape(path[i]) + if err == nil { + path[i] = segment + } + } + return +} + +// NewPathForRef returns a new path for the given ref. +func NewPathForRef(ref ast.Ref) (path Path, err error) { + + if len(ref) == 0 { + return nil, fmt.Errorf("empty reference (indicates error in caller)") + } + + if len(ref) == 1 { + return Path{}, nil + } + + path = make(Path, 0, len(ref)-1) + + for _, term := range ref[1:] { + switch v := term.Value.(type) { + case ast.String: + path = append(path, string(v)) + case ast.Number: + path = append(path, v.String()) + case ast.Boolean, ast.Null: + return nil, &Error{ + Code: NotFoundErr, + Message: fmt.Sprintf("%v: does not exist", ref), + } + case *ast.Array, ast.Object, ast.Set: + return nil, fmt.Errorf("composites cannot be base document keys: %v", ref) + default: + return nil, fmt.Errorf("unresolved reference (indicates error in caller): %v", ref) + } + } + + return path, nil +} + +// Compare performs lexigraphical comparison on p and other and returns -1 if p +// is less than other, 0 if p is equal to other, or 1 if p is greater than +// other. +func (p Path) Compare(other Path) (cmp int) { + min := len(p) + if len(other) < min { + min = len(other) + } + for i := 0; i < min; i++ { + if cmp := strings.Compare(p[i], other[i]); cmp != 0 { + return cmp + } + } + if len(p) < len(other) { + return -1 + } + if len(p) == len(other) { + return 0 + } + return 1 +} + +// Equal returns true if p is the same as other. +func (p Path) Equal(other Path) bool { + return p.Compare(other) == 0 +} + +// HasPrefix returns true if p starts with other. +func (p Path) HasPrefix(other Path) bool { + if len(other) > len(p) { + return false + } + for i := range other { + if p[i] != other[i] { + return false + } + } + return true +} + +// Ref returns a ref that represents p rooted at head. +func (p Path) Ref(head *ast.Term) (ref ast.Ref) { + ref = make(ast.Ref, len(p)+1) + ref[0] = head + for i := range p { + idx, err := strconv.ParseInt(p[i], 10, 64) + if err == nil { + ref[i+1] = ast.UIntNumberTerm(uint64(idx)) + } else { + ref[i+1] = ast.StringTerm(p[i]) + } + } + return ref +} + +func (p Path) String() string { + buf := make([]string, len(p)) + for i := range buf { + buf[i] = url.PathEscape(p[i]) + } + return "/" + strings.Join(buf, "/") +} + +// MustParsePath returns a new Path for s. If s cannot be parsed, this function +// will panic. This is mostly for test purposes. +func MustParsePath(s string) Path { + path, ok := ParsePath(s) + if !ok { + panic(s) + } + return path +} diff --git a/vendor/github.com/open-policy-agent/opa/storage/storage.go b/vendor/github.com/open-policy-agent/opa/storage/storage.go new file mode 100644 index 00000000..323a0dba --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/storage/storage.go @@ -0,0 +1,126 @@ +// Copyright 2016 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package storage + +import ( + "context" +) + +// NewTransactionOrDie is a helper function to create a new transaction. If the +// storage layer cannot create a new transaction, this function will panic. This +// function should only be used for tests. +func NewTransactionOrDie(ctx context.Context, store Store, params ...TransactionParams) Transaction { + txn, err := store.NewTransaction(ctx, params...) + if err != nil { + panic(err) + } + return txn +} + +// ReadOne is a convenience function to read a single value from the provided Store. It +// will create a new Transaction to perform the read with, and clean up after itself +// should an error occur. +func ReadOne(ctx context.Context, store Store, path Path) (interface{}, error) { + txn, err := store.NewTransaction(ctx) + if err != nil { + return nil, err + } + defer store.Abort(ctx, txn) + + return store.Read(ctx, txn, path) +} + +// WriteOne is a convenience function to write a single value to the provided Store. It +// will create a new Transaction to perform the write with, and clean up after itself +// should an error occur. +func WriteOne(ctx context.Context, store Store, op PatchOp, path Path, value interface{}) error { + txn, err := store.NewTransaction(ctx, WriteParams) + if err != nil { + return err + } + + if err := store.Write(ctx, txn, op, path, value); err != nil { + store.Abort(ctx, txn) + return err + } + + return store.Commit(ctx, txn) +} + +// MakeDir inserts an empty object at path. If the parent path does not exist, +// MakeDir will create it recursively. +func MakeDir(ctx context.Context, store Store, txn Transaction, path Path) (err error) { + + if len(path) == 0 { + return nil + } + + node, err := store.Read(ctx, txn, path) + + if err != nil { + if !IsNotFound(err) { + return err + } + + if err := MakeDir(ctx, store, txn, path[:len(path)-1]); err != nil { + return err + } else if err := store.Write(ctx, txn, AddOp, path, map[string]interface{}{}); err != nil { + return err + } + + return nil + } + + if _, ok := node.(map[string]interface{}); ok { + return nil + } + + return writeConflictError(path) + +} + +// Txn is a convenience function that executes f inside a new transaction +// opened on the store. If the function returns an error, the transaction is +// aborted and the error is returned. Otherwise, the transaction is committed +// and the result of the commit is returned. +func Txn(ctx context.Context, store Store, params TransactionParams, f func(Transaction) error) error { + + txn, err := store.NewTransaction(ctx, params) + if err != nil { + return err + } + + if err := f(txn); err != nil { + store.Abort(ctx, txn) + return err + } + + return store.Commit(ctx, txn) +} + +// NonEmpty returns a function that tests if a path is non-empty. A +// path is non-empty if a Read on the path returns a value or a Read +// on any of the path prefixes returns a non-object value. +func NonEmpty(ctx context.Context, store Store, txn Transaction) func([]string) (bool, error) { + return func(path []string) (bool, error) { + if _, err := store.Read(ctx, txn, Path(path)); err == nil { + return true, nil + } else if !IsNotFound(err) { + return false, err + } + for i := len(path) - 1; i > 0; i-- { + val, err := store.Read(ctx, txn, Path(path[:i])) + if err != nil && !IsNotFound(err) { + return false, err + } else if err == nil { + if _, ok := val.(map[string]interface{}); ok { + return false, nil + } + return true, nil + } + } + return false, nil + } +} diff --git a/vendor/github.com/open-policy-agent/opa/topdown/aggregates.go b/vendor/github.com/open-policy-agent/opa/topdown/aggregates.go new file mode 100644 index 00000000..63d60f56 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/topdown/aggregates.go @@ -0,0 +1,260 @@ +// Copyright 2016 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package topdown + +import ( + "math/big" + + "github.com/open-policy-agent/opa/ast" + "github.com/open-policy-agent/opa/topdown/builtins" +) + +func builtinCount(a ast.Value) (ast.Value, error) { + switch a := a.(type) { + case *ast.Array: + return ast.IntNumberTerm(a.Len()).Value, nil + case ast.Object: + return ast.IntNumberTerm(a.Len()).Value, nil + case ast.Set: + return ast.IntNumberTerm(a.Len()).Value, nil + case ast.String: + return ast.IntNumberTerm(len([]rune(a))).Value, nil + } + return nil, builtins.NewOperandTypeErr(1, a, "array", "object", "set") +} + +func builtinSum(a ast.Value) (ast.Value, error) { + switch a := a.(type) { + case *ast.Array: + sum := big.NewFloat(0) + err := a.Iter(func(x *ast.Term) error { + n, ok := x.Value.(ast.Number) + if !ok { + return builtins.NewOperandElementErr(1, a, x.Value, "number") + } + sum = new(big.Float).Add(sum, builtins.NumberToFloat(n)) + return nil + }) + return builtins.FloatToNumber(sum), err + case ast.Set: + sum := big.NewFloat(0) + err := a.Iter(func(x *ast.Term) error { + n, ok := x.Value.(ast.Number) + if !ok { + return builtins.NewOperandElementErr(1, a, x.Value, "number") + } + sum = new(big.Float).Add(sum, builtins.NumberToFloat(n)) + return nil + }) + return builtins.FloatToNumber(sum), err + } + return nil, builtins.NewOperandTypeErr(1, a, "set", "array") +} + +func builtinProduct(a ast.Value) (ast.Value, error) { + switch a := a.(type) { + case *ast.Array: + product := big.NewFloat(1) + err := a.Iter(func(x *ast.Term) error { + n, ok := x.Value.(ast.Number) + if !ok { + return builtins.NewOperandElementErr(1, a, x.Value, "number") + } + product = new(big.Float).Mul(product, builtins.NumberToFloat(n)) + return nil + }) + return builtins.FloatToNumber(product), err + case ast.Set: + product := big.NewFloat(1) + err := a.Iter(func(x *ast.Term) error { + n, ok := x.Value.(ast.Number) + if !ok { + return builtins.NewOperandElementErr(1, a, x.Value, "number") + } + product = new(big.Float).Mul(product, builtins.NumberToFloat(n)) + return nil + }) + return builtins.FloatToNumber(product), err + } + return nil, builtins.NewOperandTypeErr(1, a, "set", "array") +} + +func builtinMax(a ast.Value) (ast.Value, error) { + switch a := a.(type) { + case *ast.Array: + if a.Len() == 0 { + return nil, BuiltinEmpty{} + } + var max = ast.Value(ast.Null{}) + a.Foreach(func(x *ast.Term) { + if ast.Compare(max, x.Value) <= 0 { + max = x.Value + } + }) + return max, nil + case ast.Set: + if a.Len() == 0 { + return nil, BuiltinEmpty{} + } + max, err := a.Reduce(ast.NullTerm(), func(max *ast.Term, elem *ast.Term) (*ast.Term, error) { + if ast.Compare(max, elem) <= 0 { + return elem, nil + } + return max, nil + }) + return max.Value, err + } + + return nil, builtins.NewOperandTypeErr(1, a, "set", "array") +} + +func builtinMin(a ast.Value) (ast.Value, error) { + switch a := a.(type) { + case *ast.Array: + if a.Len() == 0 { + return nil, BuiltinEmpty{} + } + min := a.Elem(0).Value + a.Foreach(func(x *ast.Term) { + if ast.Compare(min, x.Value) >= 0 { + min = x.Value + } + }) + return min, nil + case ast.Set: + if a.Len() == 0 { + return nil, BuiltinEmpty{} + } + min, err := a.Reduce(ast.NullTerm(), func(min *ast.Term, elem *ast.Term) (*ast.Term, error) { + // The null term is considered to be less than any other term, + // so in order for min of a set to make sense, we need to check + // for it. + if min.Value.Compare(ast.Null{}) == 0 { + return elem, nil + } + + if ast.Compare(min, elem) >= 0 { + return elem, nil + } + return min, nil + }) + return min.Value, err + } + + return nil, builtins.NewOperandTypeErr(1, a, "set", "array") +} + +func builtinSort(a ast.Value) (ast.Value, error) { + switch a := a.(type) { + case *ast.Array: + return a.Sorted(), nil + case ast.Set: + return a.Sorted(), nil + } + return nil, builtins.NewOperandTypeErr(1, a, "set", "array") +} + +func builtinAll(a ast.Value) (ast.Value, error) { + switch val := a.(type) { + case ast.Set: + res := true + match := ast.BooleanTerm(true) + val.Until(func(term *ast.Term) bool { + if !match.Equal(term) { + res = false + return true + } + return false + }) + return ast.Boolean(res), nil + case *ast.Array: + res := true + match := ast.BooleanTerm(true) + val.Until(func(term *ast.Term) bool { + if !match.Equal(term) { + res = false + return true + } + return false + }) + return ast.Boolean(res), nil + default: + return nil, builtins.NewOperandTypeErr(1, a, "array", "set") + } +} + +func builtinAny(a ast.Value) (ast.Value, error) { + switch val := a.(type) { + case ast.Set: + res := val.Len() > 0 && val.Contains(ast.BooleanTerm(true)) + return ast.Boolean(res), nil + case *ast.Array: + res := false + match := ast.BooleanTerm(true) + val.Until(func(term *ast.Term) bool { + if match.Equal(term) { + res = true + return true + } + return false + }) + return ast.Boolean(res), nil + default: + return nil, builtins.NewOperandTypeErr(1, a, "array", "set") + } +} + +func builtinMember(_ BuiltinContext, args []*ast.Term, iter func(*ast.Term) error) error { + containee := args[0] + switch c := args[1].Value.(type) { + case ast.Set: + return iter(ast.BooleanTerm(c.Contains(containee))) + case *ast.Array: + ret := false + c.Until(func(v *ast.Term) bool { + if v.Value.Compare(containee.Value) == 0 { + ret = true + } + return ret + }) + return iter(ast.BooleanTerm(ret)) + case ast.Object: + ret := false + c.Until(func(_, v *ast.Term) bool { + if v.Value.Compare(containee.Value) == 0 { + ret = true + } + return ret + }) + return iter(ast.BooleanTerm(ret)) + } + return iter(ast.BooleanTerm(false)) +} + +func builtinMemberWithKey(_ BuiltinContext, args []*ast.Term, iter func(*ast.Term) error) error { + key, val := args[0], args[1] + switch c := args[2].Value.(type) { + case interface{ Get(*ast.Term) *ast.Term }: + ret := false + if act := c.Get(key); act != nil { + ret = act.Value.Compare(val.Value) == 0 + } + return iter(ast.BooleanTerm(ret)) + } + return iter(ast.BooleanTerm(false)) +} + +func init() { + RegisterFunctionalBuiltin1(ast.Count.Name, builtinCount) + RegisterFunctionalBuiltin1(ast.Sum.Name, builtinSum) + RegisterFunctionalBuiltin1(ast.Product.Name, builtinProduct) + RegisterFunctionalBuiltin1(ast.Max.Name, builtinMax) + RegisterFunctionalBuiltin1(ast.Min.Name, builtinMin) + RegisterFunctionalBuiltin1(ast.Sort.Name, builtinSort) + RegisterFunctionalBuiltin1(ast.Any.Name, builtinAny) + RegisterFunctionalBuiltin1(ast.All.Name, builtinAll) + RegisterBuiltinFunc(ast.Member.Name, builtinMember) + RegisterBuiltinFunc(ast.MemberWithKey.Name, builtinMemberWithKey) +} diff --git a/vendor/github.com/open-policy-agent/opa/topdown/arithmetic.go b/vendor/github.com/open-policy-agent/opa/topdown/arithmetic.go new file mode 100644 index 00000000..7e9252ff --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/topdown/arithmetic.go @@ -0,0 +1,184 @@ +// Copyright 2016 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package topdown + +import ( + "math/big" + + "fmt" + + "github.com/open-policy-agent/opa/ast" + "github.com/open-policy-agent/opa/topdown/builtins" +) + +type arithArity1 func(a *big.Float) (*big.Float, error) +type arithArity2 func(a, b *big.Float) (*big.Float, error) + +func arithAbs(a *big.Float) (*big.Float, error) { + return a.Abs(a), nil +} + +var halfAwayFromZero = big.NewFloat(0.5) + +func arithRound(a *big.Float) (*big.Float, error) { + var i *big.Int + if a.Signbit() { + i, _ = new(big.Float).Sub(a, halfAwayFromZero).Int(nil) + } else { + i, _ = new(big.Float).Add(a, halfAwayFromZero).Int(nil) + } + return new(big.Float).SetInt(i), nil +} + +func arithCeil(a *big.Float) (*big.Float, error) { + i, _ := a.Int(nil) + f := new(big.Float).SetInt(i) + + if f.Signbit() || a.Cmp(f) == 0 { + return f, nil + } + + return new(big.Float).Add(f, big.NewFloat(1.0)), nil +} + +func arithFloor(a *big.Float) (*big.Float, error) { + i, _ := a.Int(nil) + f := new(big.Float).SetInt(i) + + if !f.Signbit() || a.Cmp(f) == 0 { + return f, nil + } + + return new(big.Float).Sub(f, big.NewFloat(1.0)), nil +} + +func arithPlus(a, b *big.Float) (*big.Float, error) { + return new(big.Float).Add(a, b), nil +} + +func arithMinus(a, b *big.Float) (*big.Float, error) { + return new(big.Float).Sub(a, b), nil +} + +func arithMultiply(a, b *big.Float) (*big.Float, error) { + return new(big.Float).Mul(a, b), nil +} + +func arithDivide(a, b *big.Float) (*big.Float, error) { + i, acc := b.Int64() + if acc == big.Exact && i == 0 { + return nil, fmt.Errorf("divide by zero") + } + return new(big.Float).Quo(a, b), nil +} + +func arithRem(a, b *big.Int) (*big.Int, error) { + if b.Int64() == 0 { + return nil, fmt.Errorf("modulo by zero") + } + return new(big.Int).Rem(a, b), nil +} + +func builtinArithArity1(fn arithArity1) FunctionalBuiltin1 { + return func(a ast.Value) (ast.Value, error) { + n, err := builtins.NumberOperand(a, 1) + if err != nil { + return nil, err + } + f, err := fn(builtins.NumberToFloat(n)) + if err != nil { + return nil, err + } + return builtins.FloatToNumber(f), nil + } +} + +func builtinArithArity2(fn arithArity2) FunctionalBuiltin2 { + return func(a, b ast.Value) (ast.Value, error) { + n1, err := builtins.NumberOperand(a, 1) + if err != nil { + return nil, err + } + n2, err := builtins.NumberOperand(b, 2) + if err != nil { + return nil, err + } + f, err := fn(builtins.NumberToFloat(n1), builtins.NumberToFloat(n2)) + if err != nil { + return nil, err + } + return builtins.FloatToNumber(f), nil + } +} + +func builtinMinus(a, b ast.Value) (ast.Value, error) { + + n1, ok1 := a.(ast.Number) + n2, ok2 := b.(ast.Number) + + if ok1 && ok2 { + f, err := arithMinus(builtins.NumberToFloat(n1), builtins.NumberToFloat(n2)) + if err != nil { + return nil, err + } + return builtins.FloatToNumber(f), nil + } + + s1, ok3 := a.(ast.Set) + s2, ok4 := b.(ast.Set) + + if ok3 && ok4 { + return s1.Diff(s2), nil + } + + if !ok1 && !ok3 { + return nil, builtins.NewOperandTypeErr(1, a, "number", "set") + } + + if ok2 { + return nil, builtins.NewOperandTypeErr(2, b, "set") + } + + return nil, builtins.NewOperandTypeErr(2, b, "number") +} + +func builtinRem(a, b ast.Value) (ast.Value, error) { + n1, ok1 := a.(ast.Number) + n2, ok2 := b.(ast.Number) + + if ok1 && ok2 { + + op1, err1 := builtins.NumberToInt(n1) + op2, err2 := builtins.NumberToInt(n2) + + if err1 != nil || err2 != nil { + return nil, fmt.Errorf("modulo on floating-point number") + } + + i, err := arithRem(op1, op2) + if err != nil { + return nil, err + } + return builtins.IntToNumber(i), nil + } + + if !ok1 { + return nil, builtins.NewOperandTypeErr(1, a, "number") + } + + return nil, builtins.NewOperandTypeErr(2, b, "number") +} + +func init() { + RegisterFunctionalBuiltin1(ast.Abs.Name, builtinArithArity1(arithAbs)) + RegisterFunctionalBuiltin1(ast.Round.Name, builtinArithArity1(arithRound)) + RegisterFunctionalBuiltin1(ast.Ceil.Name, builtinArithArity1(arithCeil)) + RegisterFunctionalBuiltin1(ast.Floor.Name, builtinArithArity1(arithFloor)) + RegisterFunctionalBuiltin2(ast.Plus.Name, builtinArithArity2(arithPlus)) + RegisterFunctionalBuiltin2(ast.Minus.Name, builtinMinus) + RegisterFunctionalBuiltin2(ast.Multiply.Name, builtinArithArity2(arithMultiply)) + RegisterFunctionalBuiltin2(ast.Divide.Name, builtinArithArity2(arithDivide)) + RegisterFunctionalBuiltin2(ast.Rem.Name, builtinRem) +} diff --git a/vendor/github.com/open-policy-agent/opa/topdown/array.go b/vendor/github.com/open-policy-agent/opa/topdown/array.go new file mode 100644 index 00000000..939de499 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/topdown/array.go @@ -0,0 +1,94 @@ +// Copyright 2018 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package topdown + +import ( + "github.com/open-policy-agent/opa/ast" + "github.com/open-policy-agent/opa/topdown/builtins" +) + +func builtinArrayConcat(a, b ast.Value) (ast.Value, error) { + arrA, err := builtins.ArrayOperand(a, 1) + if err != nil { + return nil, err + } + + arrB, err := builtins.ArrayOperand(b, 2) + if err != nil { + return nil, err + } + + arrC := make([]*ast.Term, arrA.Len()+arrB.Len()) + + i := 0 + arrA.Foreach(func(elemA *ast.Term) { + arrC[i] = elemA + i++ + }) + + arrB.Foreach(func(elemB *ast.Term) { + arrC[i] = elemB + i++ + }) + + return ast.NewArray(arrC...), nil +} + +func builtinArraySlice(a, i, j ast.Value) (ast.Value, error) { + arr, err := builtins.ArrayOperand(a, 1) + if err != nil { + return nil, err + } + + startIndex, err := builtins.IntOperand(i, 2) + if err != nil { + return nil, err + } + + stopIndex, err := builtins.IntOperand(j, 3) + if err != nil { + return nil, err + } + + // Clamp stopIndex to avoid out-of-range errors. If negative, clamp to zero. + // Otherwise, clamp to length of array. + if stopIndex < 0 { + stopIndex = 0 + } else if stopIndex > arr.Len() { + stopIndex = arr.Len() + } + + // Clamp startIndex to avoid out-of-range errors. If negative, clamp to zero. + // Otherwise, clamp to stopIndex to avoid to avoid cases like arr[1:0]. + if startIndex < 0 { + startIndex = 0 + } else if startIndex > stopIndex { + startIndex = stopIndex + } + + return arr.Slice(startIndex, stopIndex), nil +} + +func builtinArrayReverse(bctx BuiltinContext, operands []*ast.Term, iter func(*ast.Term) error) error { + arr, err := builtins.ArrayOperand(operands[0].Value, 1) + if err != nil { + return err + } + + length := arr.Len() + reversedArr := make([]*ast.Term, length) + + for index := 0; index < length; index++ { + reversedArr[index] = arr.Elem(length - index - 1) + } + + return iter(ast.ArrayTerm(reversedArr...)) +} + +func init() { + RegisterFunctionalBuiltin2(ast.ArrayConcat.Name, builtinArrayConcat) + RegisterFunctionalBuiltin3(ast.ArraySlice.Name, builtinArraySlice) + RegisterBuiltinFunc(ast.ArrayReverse.Name, builtinArrayReverse) +} diff --git a/vendor/github.com/open-policy-agent/opa/topdown/binary.go b/vendor/github.com/open-policy-agent/opa/topdown/binary.go new file mode 100644 index 00000000..3cab5def --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/topdown/binary.go @@ -0,0 +1,45 @@ +// Copyright 2017 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package topdown + +import ( + "github.com/open-policy-agent/opa/ast" + "github.com/open-policy-agent/opa/topdown/builtins" +) + +func builtinBinaryAnd(a ast.Value, b ast.Value) (ast.Value, error) { + + s1, err := builtins.SetOperand(a, 1) + if err != nil { + return nil, err + } + + s2, err := builtins.SetOperand(b, 2) + if err != nil { + return nil, err + } + + return s1.Intersect(s2), nil +} + +func builtinBinaryOr(a ast.Value, b ast.Value) (ast.Value, error) { + + s1, err := builtins.SetOperand(a, 1) + if err != nil { + return nil, err + } + + s2, err := builtins.SetOperand(b, 2) + if err != nil { + return nil, err + } + + return s1.Union(s2), nil +} + +func init() { + RegisterFunctionalBuiltin2(ast.And.Name, builtinBinaryAnd) + RegisterFunctionalBuiltin2(ast.Or.Name, builtinBinaryOr) +} diff --git a/vendor/github.com/open-policy-agent/opa/topdown/bindings.go b/vendor/github.com/open-policy-agent/opa/topdown/bindings.go new file mode 100644 index 00000000..7621dac5 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/topdown/bindings.go @@ -0,0 +1,400 @@ +// Copyright 2017 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package topdown + +import ( + "fmt" + "strings" + + "github.com/open-policy-agent/opa/ast" +) + +type undo struct { + k *ast.Term + u *bindings +} + +func (u *undo) Undo() { + if u == nil { + // Allow call on zero value of Undo for ease-of-use. + return + } + if u.u == nil { + // Call on empty unifier undos a no-op unify operation. + return + } + u.u.delete(u.k) +} + +type bindings struct { + id uint64 + values bindingsArrayHashmap + instr *Instrumentation +} + +func newBindings(id uint64, instr *Instrumentation) *bindings { + values := newBindingsArrayHashmap() + return &bindings{id, values, instr} +} + +func (u *bindings) Iter(caller *bindings, iter func(*ast.Term, *ast.Term) error) error { + + var err error + + u.values.Iter(func(k *ast.Term, v value) bool { + if err != nil { + return true + } + err = iter(k, u.PlugNamespaced(k, caller)) + + return false + }) + + return err +} + +func (u *bindings) Namespace(x ast.Node, caller *bindings) { + vis := namespacingVisitor{ + b: u, + caller: caller, + } + ast.NewGenericVisitor(vis.Visit).Walk(x) +} + +func (u *bindings) Plug(a *ast.Term) *ast.Term { + return u.PlugNamespaced(a, nil) +} + +func (u *bindings) PlugNamespaced(a *ast.Term, caller *bindings) *ast.Term { + if u != nil { + u.instr.startTimer(evalOpPlug) + t := u.plugNamespaced(a, caller) + u.instr.stopTimer(evalOpPlug) + return t + } + + return u.plugNamespaced(a, caller) +} + +func (u *bindings) plugNamespaced(a *ast.Term, caller *bindings) *ast.Term { + switch v := a.Value.(type) { + case ast.Var: + b, next := u.apply(a) + if a != b || u != next { + return next.plugNamespaced(b, caller) + } + return u.namespaceVar(b, caller) + case *ast.Array: + if a.IsGround() { + return a + } + cpy := *a + arr := make([]*ast.Term, v.Len()) + for i := 0; i < len(arr); i++ { + arr[i] = u.plugNamespaced(v.Elem(i), caller) + } + cpy.Value = ast.NewArray(arr...) + return &cpy + case ast.Object: + if a.IsGround() { + return a + } + cpy := *a + cpy.Value, _ = v.Map(func(k, v *ast.Term) (*ast.Term, *ast.Term, error) { + return u.plugNamespaced(k, caller), u.plugNamespaced(v, caller), nil + }) + return &cpy + case ast.Set: + if a.IsGround() { + return a + } + cpy := *a + cpy.Value, _ = v.Map(func(x *ast.Term) (*ast.Term, error) { + return u.plugNamespaced(x, caller), nil + }) + return &cpy + case ast.Ref: + cpy := *a + ref := make(ast.Ref, len(v)) + for i := 0; i < len(ref); i++ { + ref[i] = u.plugNamespaced(v[i], caller) + } + cpy.Value = ref + return &cpy + } + return a +} + +func (u *bindings) bind(a *ast.Term, b *ast.Term, other *bindings, und *undo) { + u.values.Put(a, value{ + u: other, + v: b, + }) + und.k = a + und.u = u +} + +func (u *bindings) apply(a *ast.Term) (*ast.Term, *bindings) { + // Early exit for non-var terms. Only vars are bound in the binding list, + // so the lookup below will always fail for non-var terms. In some cases, + // the lookup may be expensive as it has to hash the term (which for large + // inputs can be costly). + _, ok := a.Value.(ast.Var) + if !ok { + return a, u + } + val, ok := u.get(a) + if !ok { + return a, u + } + return val.u.apply(val.v) +} + +func (u *bindings) delete(v *ast.Term) { + u.values.Delete(v) +} + +func (u *bindings) get(v *ast.Term) (value, bool) { + if u == nil { + return value{}, false + } + return u.values.Get(v) +} + +func (u *bindings) String() string { + if u == nil { + return "()" + } + var buf []string + u.values.Iter(func(a *ast.Term, b value) bool { + buf = append(buf, fmt.Sprintf("%v: %v", a, b)) + return false + }) + return fmt.Sprintf("({%v}, %v)", strings.Join(buf, ", "), u.id) +} + +func (u *bindings) namespaceVar(v *ast.Term, caller *bindings) *ast.Term { + name, ok := v.Value.(ast.Var) + if !ok { + panic("illegal value") + } + if caller != nil && caller != u { + // Root documents (i.e., data, input) should never be namespaced because they + // are globally unique. + if !ast.RootDocumentNames.Contains(v) { + return ast.NewTerm(ast.Var(string(name) + fmt.Sprint(u.id))) + } + } + return v +} + +type value struct { + u *bindings + v *ast.Term +} + +func (v value) String() string { + return fmt.Sprintf("(%v, %d)", v.v, v.u.id) +} + +func (v value) equal(other *value) bool { + if v.u == other.u { + return v.v.Equal(other.v) + } + return false +} + +type namespacingVisitor struct { + b *bindings + caller *bindings +} + +func (vis namespacingVisitor) Visit(x interface{}) bool { + switch x := x.(type) { + case *ast.ArrayComprehension: + x.Term = vis.namespaceTerm(x.Term) + ast.NewGenericVisitor(vis.Visit).Walk(x.Body) + return true + case *ast.SetComprehension: + x.Term = vis.namespaceTerm(x.Term) + ast.NewGenericVisitor(vis.Visit).Walk(x.Body) + return true + case *ast.ObjectComprehension: + x.Key = vis.namespaceTerm(x.Key) + x.Value = vis.namespaceTerm(x.Value) + ast.NewGenericVisitor(vis.Visit).Walk(x.Body) + return true + case *ast.Expr: + switch terms := x.Terms.(type) { + case []*ast.Term: + for i := 1; i < len(terms); i++ { + terms[i] = vis.namespaceTerm(terms[i]) + } + case *ast.Term: + x.Terms = vis.namespaceTerm(terms) + } + for _, w := range x.With { + w.Target = vis.namespaceTerm(w.Target) + w.Value = vis.namespaceTerm(w.Value) + } + } + return false +} + +func (vis namespacingVisitor) namespaceTerm(a *ast.Term) *ast.Term { + switch v := a.Value.(type) { + case ast.Var: + return vis.b.namespaceVar(a, vis.caller) + case *ast.Array: + if a.IsGround() { + return a + } + cpy := *a + arr := make([]*ast.Term, v.Len()) + for i := 0; i < len(arr); i++ { + arr[i] = vis.namespaceTerm(v.Elem(i)) + } + cpy.Value = ast.NewArray(arr...) + return &cpy + case ast.Object: + if a.IsGround() { + return a + } + cpy := *a + cpy.Value, _ = v.Map(func(k, v *ast.Term) (*ast.Term, *ast.Term, error) { + return vis.namespaceTerm(k), vis.namespaceTerm(v), nil + }) + return &cpy + case ast.Set: + if a.IsGround() { + return a + } + cpy := *a + cpy.Value, _ = v.Map(func(x *ast.Term) (*ast.Term, error) { + return vis.namespaceTerm(x), nil + }) + return &cpy + case ast.Ref: + cpy := *a + ref := make(ast.Ref, len(v)) + for i := 0; i < len(ref); i++ { + ref[i] = vis.namespaceTerm(v[i]) + } + cpy.Value = ref + return &cpy + } + return a +} + +const maxLinearScan = 16 + +// bindingsArrayHashMap uses an array with linear scan instead +// of a hash map for smaller # of entries. Hash maps start to +// show off their performance advantage only after 16 keys. +type bindingsArrayHashmap struct { + n int // Entries in the array. + a *[maxLinearScan]bindingArrayKeyValue + m map[ast.Var]bindingArrayKeyValue +} + +type bindingArrayKeyValue struct { + key *ast.Term + value value +} + +func newBindingsArrayHashmap() bindingsArrayHashmap { + return bindingsArrayHashmap{} +} + +func (b *bindingsArrayHashmap) Put(key *ast.Term, value value) { + if b.m == nil { + if b.a == nil { + b.a = new([maxLinearScan]bindingArrayKeyValue) + } else if i := b.find(key); i >= 0 { + (*b.a)[i].value = value + return + } + + if b.n < maxLinearScan { + (*b.a)[b.n] = bindingArrayKeyValue{key, value} + b.n++ + return + } + + // Array is full, revert to using the hash map instead. + + b.m = make(map[ast.Var]bindingArrayKeyValue, maxLinearScan+1) + for _, kv := range *b.a { + b.m[kv.key.Value.(ast.Var)] = bindingArrayKeyValue{kv.key, kv.value} + } + b.m[key.Value.(ast.Var)] = bindingArrayKeyValue{key, value} + + b.n = 0 + return + } + + b.m[key.Value.(ast.Var)] = bindingArrayKeyValue{key, value} +} + +func (b *bindingsArrayHashmap) Get(key *ast.Term) (value, bool) { + if b.m == nil { + if i := b.find(key); i >= 0 { + return (*b.a)[i].value, true + } + + return value{}, false + } + + v, ok := b.m[key.Value.(ast.Var)] + if ok { + return v.value, true + } + + return value{}, false +} + +func (b *bindingsArrayHashmap) Delete(key *ast.Term) { + if b.m == nil { + if i := b.find(key); i >= 0 { + n := b.n - 1 + if i < n { + (*b.a)[i] = (*b.a)[n] + } + + b.n = n + } + return + } + + delete(b.m, key.Value.(ast.Var)) +} + +func (b *bindingsArrayHashmap) Iter(f func(k *ast.Term, v value) bool) { + if b.m == nil { + for i := 0; i < b.n; i++ { + if f((*b.a)[i].key, (*b.a)[i].value) { + return + } + } + return + } + + for _, v := range b.m { + if f(v.key, v.value) { + return + } + } +} + +func (b *bindingsArrayHashmap) find(key *ast.Term) int { + v := key.Value.(ast.Var) + for i := 0; i < b.n; i++ { + if (*b.a)[i].key.Value.(ast.Var) == v { + return i + } + } + + return -1 +} diff --git a/vendor/github.com/open-policy-agent/opa/topdown/bits.go b/vendor/github.com/open-policy-agent/opa/topdown/bits.go new file mode 100644 index 00000000..7a63c0df --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/topdown/bits.go @@ -0,0 +1,88 @@ +// Copyright 2020 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package topdown + +import ( + "math/big" + + "github.com/open-policy-agent/opa/ast" + "github.com/open-policy-agent/opa/topdown/builtins" +) + +type bitsArity1 func(a *big.Int) (*big.Int, error) +type bitsArity2 func(a, b *big.Int) (*big.Int, error) + +func bitsOr(a, b *big.Int) (*big.Int, error) { + return new(big.Int).Or(a, b), nil +} + +func bitsAnd(a, b *big.Int) (*big.Int, error) { + return new(big.Int).And(a, b), nil +} + +func bitsNegate(a *big.Int) (*big.Int, error) { + return new(big.Int).Not(a), nil +} + +func bitsXOr(a, b *big.Int) (*big.Int, error) { + return new(big.Int).Xor(a, b), nil +} + +func bitsShiftLeft(a, b *big.Int) (*big.Int, error) { + if b.Sign() == -1 { + return nil, builtins.NewOperandErr(2, "must be an unsigned integer number but got a negative integer") + } + shift := uint(b.Uint64()) + return new(big.Int).Lsh(a, shift), nil +} + +func bitsShiftRight(a, b *big.Int) (*big.Int, error) { + if b.Sign() == -1 { + return nil, builtins.NewOperandErr(2, "must be an unsigned integer number but got a negative integer") + } + shift := uint(b.Uint64()) + return new(big.Int).Rsh(a, shift), nil +} + +func builtinBitsArity1(fn bitsArity1) BuiltinFunc { + return func(_ BuiltinContext, operands []*ast.Term, iter func(*ast.Term) error) error { + i, err := builtins.BigIntOperand(operands[0].Value, 1) + if err != nil { + return err + } + iOut, err := fn(i) + if err != nil { + return err + } + return iter(ast.NewTerm(builtins.IntToNumber(iOut))) + } +} + +func builtinBitsArity2(fn bitsArity2) BuiltinFunc { + return func(_ BuiltinContext, operands []*ast.Term, iter func(*ast.Term) error) error { + i1, err := builtins.BigIntOperand(operands[0].Value, 1) + if err != nil { + return err + } + i2, err := builtins.BigIntOperand(operands[1].Value, 2) + if err != nil { + return err + } + iOut, err := fn(i1, i2) + if err != nil { + return err + } + return iter(ast.NewTerm(builtins.IntToNumber(iOut))) + } +} + +func init() { + RegisterBuiltinFunc(ast.BitsOr.Name, builtinBitsArity2(bitsOr)) + RegisterBuiltinFunc(ast.BitsAnd.Name, builtinBitsArity2(bitsAnd)) + RegisterBuiltinFunc(ast.BitsNegate.Name, builtinBitsArity1(bitsNegate)) + RegisterBuiltinFunc(ast.BitsXOr.Name, builtinBitsArity2(bitsXOr)) + RegisterBuiltinFunc(ast.BitsShiftLeft.Name, builtinBitsArity2(bitsShiftLeft)) + RegisterBuiltinFunc(ast.BitsShiftRight.Name, builtinBitsArity2(bitsShiftRight)) +} diff --git a/vendor/github.com/open-policy-agent/opa/topdown/builtins.go b/vendor/github.com/open-policy-agent/opa/topdown/builtins.go new file mode 100644 index 00000000..0df8bb84 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/topdown/builtins.go @@ -0,0 +1,205 @@ +// Copyright 2016 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package topdown + +import ( + "context" + "encoding/binary" + "fmt" + "io" + "math/rand" + + "github.com/open-policy-agent/opa/ast" + "github.com/open-policy-agent/opa/metrics" + "github.com/open-policy-agent/opa/topdown/builtins" + "github.com/open-policy-agent/opa/topdown/cache" + "github.com/open-policy-agent/opa/topdown/print" + "github.com/open-policy-agent/opa/tracing" +) + +type ( + // FunctionalBuiltin1 is deprecated. Use BuiltinFunc instead. + FunctionalBuiltin1 func(op1 ast.Value) (output ast.Value, err error) + + // FunctionalBuiltin2 is deprecated. Use BuiltinFunc instead. + FunctionalBuiltin2 func(op1, op2 ast.Value) (output ast.Value, err error) + + // FunctionalBuiltin3 is deprecated. Use BuiltinFunc instead. + FunctionalBuiltin3 func(op1, op2, op3 ast.Value) (output ast.Value, err error) + + // FunctionalBuiltin4 is deprecated. Use BuiltinFunc instead. + FunctionalBuiltin4 func(op1, op2, op3, op4 ast.Value) (output ast.Value, err error) + + // BuiltinContext contains context from the evaluator that may be used by + // built-in functions. + BuiltinContext struct { + Context context.Context // request context that was passed when query started + Metrics metrics.Metrics // metrics registry for recording built-in specific metrics + Seed io.Reader // randomization source + Time *ast.Term // wall clock time + Cancel Cancel // atomic value that signals evaluation to halt + Runtime *ast.Term // runtime information on the OPA instance + Cache builtins.Cache // built-in function state cache + InterQueryBuiltinCache cache.InterQueryCache // cross-query built-in function state cache + Location *ast.Location // location of built-in call + Tracers []Tracer // Deprecated: Use QueryTracers instead + QueryTracers []QueryTracer // tracer objects for trace() built-in function + TraceEnabled bool // indicates whether tracing is enabled for the evaluation + QueryID uint64 // identifies query being evaluated + ParentID uint64 // identifies parent of query being evaluated + PrintHook print.Hook // provides callback function to use for printing + DistributedTracingOpts tracing.Options // options to be used by distributed tracing. + rand *rand.Rand // randomization source for non-security-sensitive operations + Capabilities *ast.Capabilities + } + + // BuiltinFunc defines an interface for implementing built-in functions. + // The built-in function is called with the plugged operands from the call + // (including the output operands.) The implementation should evaluate the + // operands and invoke the iterator for each successful/defined output + // value. + BuiltinFunc func(bctx BuiltinContext, operands []*ast.Term, iter func(*ast.Term) error) error +) + +// Rand returns a random number generator based on the Seed for this built-in +// context. The random number will be re-used across multiple calls to this +// function. If a random number generator cannot be created, an error is +// returned. +func (bctx *BuiltinContext) Rand() (*rand.Rand, error) { + + if bctx.rand != nil { + return bctx.rand, nil + } + + seed, err := readInt64(bctx.Seed) + if err != nil { + return nil, err + } + + bctx.rand = rand.New(rand.NewSource(seed)) + return bctx.rand, nil +} + +// RegisterBuiltinFunc adds a new built-in function to the evaluation engine. +func RegisterBuiltinFunc(name string, f BuiltinFunc) { + builtinFunctions[name] = builtinErrorWrapper(name, f) +} + +// RegisterFunctionalBuiltin1 is deprecated use RegisterBuiltinFunc instead. +func RegisterFunctionalBuiltin1(name string, fun FunctionalBuiltin1) { + builtinFunctions[name] = functionalWrapper1(name, fun) +} + +// RegisterFunctionalBuiltin2 is deprecated use RegisterBuiltinFunc instead. +func RegisterFunctionalBuiltin2(name string, fun FunctionalBuiltin2) { + builtinFunctions[name] = functionalWrapper2(name, fun) +} + +// RegisterFunctionalBuiltin3 is deprecated use RegisterBuiltinFunc instead. +func RegisterFunctionalBuiltin3(name string, fun FunctionalBuiltin3) { + builtinFunctions[name] = functionalWrapper3(name, fun) +} + +// RegisterFunctionalBuiltin4 is deprecated use RegisterBuiltinFunc instead. +func RegisterFunctionalBuiltin4(name string, fun FunctionalBuiltin4) { + builtinFunctions[name] = functionalWrapper4(name, fun) +} + +// GetBuiltin returns a built-in function implementation, nil if no built-in found. +func GetBuiltin(name string) BuiltinFunc { + return builtinFunctions[name] +} + +// BuiltinEmpty is deprecated. +type BuiltinEmpty struct{} + +func (BuiltinEmpty) Error() string { + return "" +} + +var builtinFunctions = map[string]BuiltinFunc{} + +func builtinErrorWrapper(name string, fn BuiltinFunc) BuiltinFunc { + return func(bctx BuiltinContext, args []*ast.Term, iter func(*ast.Term) error) error { + err := fn(bctx, args, iter) + if err == nil { + return nil + } + return handleBuiltinErr(name, bctx.Location, err) + } +} + +func functionalWrapper1(name string, fn FunctionalBuiltin1) BuiltinFunc { + return func(bctx BuiltinContext, args []*ast.Term, iter func(*ast.Term) error) error { + result, err := fn(args[0].Value) + if err == nil { + return iter(ast.NewTerm(result)) + } + return handleBuiltinErr(name, bctx.Location, err) + } +} + +func functionalWrapper2(name string, fn FunctionalBuiltin2) BuiltinFunc { + return func(bctx BuiltinContext, args []*ast.Term, iter func(*ast.Term) error) error { + result, err := fn(args[0].Value, args[1].Value) + if err == nil { + return iter(ast.NewTerm(result)) + } + return handleBuiltinErr(name, bctx.Location, err) + } +} + +func functionalWrapper3(name string, fn FunctionalBuiltin3) BuiltinFunc { + return func(bctx BuiltinContext, args []*ast.Term, iter func(*ast.Term) error) error { + result, err := fn(args[0].Value, args[1].Value, args[2].Value) + if err == nil { + return iter(ast.NewTerm(result)) + } + return handleBuiltinErr(name, bctx.Location, err) + } +} + +func functionalWrapper4(name string, fn FunctionalBuiltin4) BuiltinFunc { + return func(bctx BuiltinContext, args []*ast.Term, iter func(*ast.Term) error) error { + result, err := fn(args[0].Value, args[1].Value, args[2].Value, args[3].Value) + if err == nil { + return iter(ast.NewTerm(result)) + } + if _, empty := err.(BuiltinEmpty); empty { + return nil + } + return handleBuiltinErr(name, bctx.Location, err) + } +} + +func handleBuiltinErr(name string, loc *ast.Location, err error) error { + switch err := err.(type) { + case BuiltinEmpty: + return nil + case *Error, Halt: + return err + case builtins.ErrOperand: + return &Error{ + Code: TypeErr, + Message: fmt.Sprintf("%v: %v", string(name), err.Error()), + Location: loc, + } + default: + return &Error{ + Code: BuiltinErr, + Message: fmt.Sprintf("%v: %v", string(name), err.Error()), + Location: loc, + } + } +} + +func readInt64(r io.Reader) (int64, error) { + bs := make([]byte, 8) + n, err := io.ReadFull(r, bs) + if n != len(bs) || err != nil { + return 0, err + } + return int64(binary.BigEndian.Uint64(bs)), nil +} diff --git a/vendor/github.com/open-policy-agent/opa/topdown/builtins/builtins.go b/vendor/github.com/open-policy-agent/opa/topdown/builtins/builtins.go new file mode 100644 index 00000000..fa0e0a28 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/topdown/builtins/builtins.go @@ -0,0 +1,237 @@ +// Copyright 2016 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +// Package builtins contains utilities for implementing built-in functions. +package builtins + +import ( + "fmt" + "math/big" + "strings" + + "github.com/open-policy-agent/opa/ast" +) + +// Cache defines the built-in cache used by the top-down evaluation. The keys +// must be comparable and should not be of type string. +type Cache map[interface{}]interface{} + +// Put updates the cache for the named built-in. +func (c Cache) Put(k, v interface{}) { + c[k] = v +} + +// Get returns the cached value for k. +func (c Cache) Get(k interface{}) (interface{}, bool) { + v, ok := c[k] + return v, ok +} + +// ErrOperand represents an invalid operand has been passed to a built-in +// function. Built-ins should return ErrOperand to indicate a type error has +// occurred. +type ErrOperand string + +func (err ErrOperand) Error() string { + return string(err) +} + +// NewOperandErr returns a generic operand error. +func NewOperandErr(pos int, f string, a ...interface{}) error { + f = fmt.Sprintf("operand %v ", pos) + f + return ErrOperand(fmt.Sprintf(f, a...)) +} + +// NewOperandTypeErr returns an operand error indicating the operand's type was wrong. +func NewOperandTypeErr(pos int, got ast.Value, expected ...string) error { + + if len(expected) == 1 { + return NewOperandErr(pos, "must be %v but got %v", expected[0], ast.TypeName(got)) + } + + return NewOperandErr(pos, "must be one of {%v} but got %v", strings.Join(expected, ", "), ast.TypeName(got)) +} + +// NewOperandElementErr returns an operand error indicating an element in the +// composite operand was wrong. +func NewOperandElementErr(pos int, composite ast.Value, got ast.Value, expected ...string) error { + + tpe := ast.TypeName(composite) + + if len(expected) == 1 { + return NewOperandErr(pos, "must be %v of %vs but got %v containing %v", tpe, expected[0], tpe, ast.TypeName(got)) + } + + return NewOperandErr(pos, "must be %v of (any of) {%v} but got %v containing %v", tpe, strings.Join(expected, ", "), tpe, ast.TypeName(got)) +} + +// NewOperandEnumErr returns an operand error indicating a value was wrong. +func NewOperandEnumErr(pos int, expected ...string) error { + + if len(expected) == 1 { + return NewOperandErr(pos, "must be %v", expected[0]) + } + + return NewOperandErr(pos, "must be one of {%v}", strings.Join(expected, ", ")) +} + +// IntOperand converts x to an int. If the cast fails, a descriptive error is +// returned. +func IntOperand(x ast.Value, pos int) (int, error) { + n, ok := x.(ast.Number) + if !ok { + return 0, NewOperandTypeErr(pos, x, "number") + } + + i, ok := n.Int() + if !ok { + return 0, NewOperandErr(pos, "must be integer number but got floating-point number") + } + + return i, nil +} + +// BigIntOperand converts x to a big int. If the cast fails, a descriptive error +// is returned. +func BigIntOperand(x ast.Value, pos int) (*big.Int, error) { + n, err := NumberOperand(x, 1) + if err != nil { + return nil, NewOperandTypeErr(pos, x, "integer") + } + bi, err := NumberToInt(n) + if err != nil { + return nil, NewOperandErr(pos, "must be integer number but got floating-point number") + } + + return bi, nil +} + +// NumberOperand converts x to a number. If the cast fails, a descriptive error is +// returned. +func NumberOperand(x ast.Value, pos int) (ast.Number, error) { + n, ok := x.(ast.Number) + if !ok { + return ast.Number(""), NewOperandTypeErr(pos, x, "number") + } + return n, nil +} + +// SetOperand converts x to a set. If the cast fails, a descriptive error is +// returned. +func SetOperand(x ast.Value, pos int) (ast.Set, error) { + s, ok := x.(ast.Set) + if !ok { + return nil, NewOperandTypeErr(pos, x, "set") + } + return s, nil +} + +// StringOperand converts x to a string. If the cast fails, a descriptive error is +// returned. +func StringOperand(x ast.Value, pos int) (ast.String, error) { + s, ok := x.(ast.String) + if !ok { + return ast.String(""), NewOperandTypeErr(pos, x, "string") + } + return s, nil +} + +// ObjectOperand converts x to an object. If the cast fails, a descriptive +// error is returned. +func ObjectOperand(x ast.Value, pos int) (ast.Object, error) { + o, ok := x.(ast.Object) + if !ok { + return nil, NewOperandTypeErr(pos, x, "object") + } + return o, nil +} + +// ArrayOperand converts x to an array. If the cast fails, a descriptive +// error is returned. +func ArrayOperand(x ast.Value, pos int) (*ast.Array, error) { + a, ok := x.(*ast.Array) + if !ok { + return ast.NewArray(), NewOperandTypeErr(pos, x, "array") + } + return a, nil +} + +// NumberToFloat converts n to a big float. +func NumberToFloat(n ast.Number) *big.Float { + r, ok := new(big.Float).SetString(string(n)) + if !ok { + panic("illegal value") + } + return r +} + +// FloatToNumber converts f to a number. +func FloatToNumber(f *big.Float) ast.Number { + return ast.Number(f.Text('g', -1)) +} + +// NumberToInt converts n to a big int. +// If n cannot be converted to an big int, an error is returned. +func NumberToInt(n ast.Number) (*big.Int, error) { + f := NumberToFloat(n) + r, accuracy := f.Int(nil) + if accuracy != big.Exact { + return nil, fmt.Errorf("illegal value") + } + return r, nil +} + +// IntToNumber converts i to a number. +func IntToNumber(i *big.Int) ast.Number { + return ast.Number(i.String()) +} + +// StringSliceOperand converts x to a []string. If the cast fails, a descriptive error is +// returned. +func StringSliceOperand(x ast.Value, pos int) ([]string, error) { + a, err := ArrayOperand(x, pos) + if err != nil { + return nil, err + } + + var f = make([]string, a.Len()) + for k := 0; k < a.Len(); k++ { + b := a.Elem(k) + c, ok := b.Value.(ast.String) + if !ok { + return nil, NewOperandElementErr(pos, x, b.Value, "[]string") + } + + f[k] = string(c) + } + + return f, nil +} + +// RuneSliceOperand converts x to a []rune. If the cast fails, a descriptive error is +// returned. +func RuneSliceOperand(x ast.Value, pos int) ([]rune, error) { + a, err := ArrayOperand(x, pos) + if err != nil { + return nil, err + } + + var f = make([]rune, a.Len()) + for k := 0; k < a.Len(); k++ { + b := a.Elem(k) + c, ok := b.Value.(ast.String) + if !ok { + return nil, NewOperandElementErr(pos, x, b.Value, "string") + } + + d := []rune(string(c)) + if len(d) != 1 { + return nil, NewOperandElementErr(pos, x, b.Value, "rune") + } + + f[k] = d[0] + } + + return f, nil +} diff --git a/vendor/github.com/open-policy-agent/opa/topdown/cache.go b/vendor/github.com/open-policy-agent/opa/topdown/cache.go new file mode 100644 index 00000000..96c250e1 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/topdown/cache.go @@ -0,0 +1,237 @@ +// Copyright 2017 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package topdown + +import ( + "github.com/open-policy-agent/opa/ast" + "github.com/open-policy-agent/opa/util" +) + +type virtualCache struct { + stack []*virtualCacheElem +} + +type virtualCacheElem struct { + value *ast.Term + children *util.HashMap +} + +func newVirtualCache() *virtualCache { + cache := &virtualCache{} + cache.Push() + return cache +} + +func (c *virtualCache) Push() { + c.stack = append(c.stack, newVirtualCacheElem()) +} + +func (c *virtualCache) Pop() { + c.stack = c.stack[:len(c.stack)-1] +} + +func (c *virtualCache) Get(ref ast.Ref) *ast.Term { + node := c.stack[len(c.stack)-1] + for i := 0; i < len(ref); i++ { + x, ok := node.children.Get(ref[i]) + if !ok { + return nil + } + node = x.(*virtualCacheElem) + } + return node.value +} + +func (c *virtualCache) Put(ref ast.Ref, value *ast.Term) { + node := c.stack[len(c.stack)-1] + for i := 0; i < len(ref); i++ { + x, ok := node.children.Get(ref[i]) + if ok { + node = x.(*virtualCacheElem) + } else { + next := newVirtualCacheElem() + node.children.Put(ref[i], next) + node = next + } + } + node.value = value +} + +func newVirtualCacheElem() *virtualCacheElem { + return &virtualCacheElem{children: newVirtualCacheHashMap()} +} + +func newVirtualCacheHashMap() *util.HashMap { + return util.NewHashMap(func(a, b util.T) bool { + return a.(*ast.Term).Equal(b.(*ast.Term)) + }, func(x util.T) int { + return x.(*ast.Term).Hash() + }) +} + +// baseCache implements a trie structure to cache base documents read out of +// storage. Values inserted into the cache may contain other values that were +// previously inserted. In this case, the previous values are erased from the +// structure. +type baseCache struct { + root *baseCacheElem +} + +func newBaseCache() *baseCache { + return &baseCache{ + root: newBaseCacheElem(), + } +} + +func (c *baseCache) Get(ref ast.Ref) ast.Value { + node := c.root + for i := 0; i < len(ref); i++ { + node = node.children[ref[i].Value] + if node == nil { + return nil + } else if node.value != nil { + result, err := node.value.Find(ref[i+1:]) + if err != nil { + return nil + } + return result + } + } + return nil +} + +func (c *baseCache) Put(ref ast.Ref, value ast.Value) { + node := c.root + for i := 0; i < len(ref); i++ { + if child, ok := node.children[ref[i].Value]; ok { + node = child + } else { + child := newBaseCacheElem() + node.children[ref[i].Value] = child + node = child + } + } + node.set(value) +} + +type baseCacheElem struct { + value ast.Value + children map[ast.Value]*baseCacheElem +} + +func newBaseCacheElem() *baseCacheElem { + return &baseCacheElem{ + children: map[ast.Value]*baseCacheElem{}, + } +} + +func (e *baseCacheElem) set(value ast.Value) { + e.value = value + e.children = map[ast.Value]*baseCacheElem{} +} + +type refStack struct { + sl []refStackElem +} + +type refStackElem struct { + refs []ast.Ref +} + +func newRefStack() *refStack { + return &refStack{} +} + +func (s *refStack) Push(refs []ast.Ref) { + s.sl = append(s.sl, refStackElem{refs: refs}) +} + +func (s *refStack) Pop() { + s.sl = s.sl[:len(s.sl)-1] +} + +func (s *refStack) Prefixed(ref ast.Ref) bool { + if s != nil { + for i := len(s.sl) - 1; i >= 0; i-- { + for j := range s.sl[i].refs { + if ref.HasPrefix(s.sl[i].refs[j]) { + return true + } + } + } + } + return false +} + +type comprehensionCache struct { + stack []map[*ast.Term]*comprehensionCacheElem +} + +type comprehensionCacheElem struct { + value *ast.Term + children *util.HashMap +} + +func newComprehensionCache() *comprehensionCache { + cache := &comprehensionCache{} + cache.Push() + return cache +} + +func (c *comprehensionCache) Push() { + c.stack = append(c.stack, map[*ast.Term]*comprehensionCacheElem{}) +} + +func (c *comprehensionCache) Pop() { + c.stack = c.stack[:len(c.stack)-1] +} + +func (c *comprehensionCache) Elem(t *ast.Term) (*comprehensionCacheElem, bool) { + elem, ok := c.stack[len(c.stack)-1][t] + return elem, ok +} + +func (c *comprehensionCache) Set(t *ast.Term, elem *comprehensionCacheElem) { + c.stack[len(c.stack)-1][t] = elem +} + +func newComprehensionCacheElem() *comprehensionCacheElem { + return &comprehensionCacheElem{children: newComprehensionCacheHashMap()} +} + +func (c *comprehensionCacheElem) Get(key []*ast.Term) *ast.Term { + node := c + for i := 0; i < len(key); i++ { + x, ok := node.children.Get(key[i]) + if !ok { + return nil + } + node = x.(*comprehensionCacheElem) + } + return node.value +} + +func (c *comprehensionCacheElem) Put(key []*ast.Term, value *ast.Term) { + node := c + for i := 0; i < len(key); i++ { + x, ok := node.children.Get(key[i]) + if ok { + node = x.(*comprehensionCacheElem) + } else { + next := newComprehensionCacheElem() + node.children.Put(key[i], next) + node = next + } + } + node.value = value +} + +func newComprehensionCacheHashMap() *util.HashMap { + return util.NewHashMap(func(a, b util.T) bool { + return a.(*ast.Term).Equal(b.(*ast.Term)) + }, func(x util.T) int { + return x.(*ast.Term).Hash() + }) +} diff --git a/vendor/github.com/open-policy-agent/opa/topdown/cache/cache.go b/vendor/github.com/open-policy-agent/opa/topdown/cache/cache.go new file mode 100644 index 00000000..5e32b39c --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/topdown/cache/cache.go @@ -0,0 +1,167 @@ +// Copyright 2020 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +// Package cache defines the inter-query cache interface that can cache data across queries +package cache + +import ( + "container/list" + + "github.com/open-policy-agent/opa/ast" + + "sync" + + "github.com/open-policy-agent/opa/util" +) + +const ( + defaultMaxSizeBytes = int64(0) // unlimited +) + +// Config represents the configuration of the inter-query cache. +type Config struct { + InterQueryBuiltinCache InterQueryBuiltinCacheConfig `json:"inter_query_builtin_cache"` +} + +// InterQueryBuiltinCacheConfig represents the configuration of the inter-query cache that built-in functions can utilize. +type InterQueryBuiltinCacheConfig struct { + MaxSizeBytes *int64 `json:"max_size_bytes,omitempty"` +} + +// ParseCachingConfig returns the config for the inter-query cache. +func ParseCachingConfig(raw []byte) (*Config, error) { + if raw == nil { + maxSize := new(int64) + *maxSize = defaultMaxSizeBytes + return &Config{InterQueryBuiltinCache: InterQueryBuiltinCacheConfig{MaxSizeBytes: maxSize}}, nil + } + + var config Config + + if err := util.Unmarshal(raw, &config); err == nil { + if err = config.validateAndInjectDefaults(); err != nil { + return nil, err + } + } else { + return nil, err + } + + return &config, nil +} + +func (c *Config) validateAndInjectDefaults() error { + if c.InterQueryBuiltinCache.MaxSizeBytes == nil { + maxSize := new(int64) + *maxSize = defaultMaxSizeBytes + c.InterQueryBuiltinCache.MaxSizeBytes = maxSize + } + return nil +} + +// InterQueryCacheValue defines the interface for the data that the inter-query cache holds. +type InterQueryCacheValue interface { + SizeInBytes() int64 +} + +// InterQueryCache defines the interface for the inter-query cache. +type InterQueryCache interface { + Get(key ast.Value) (value InterQueryCacheValue, found bool) + Insert(key ast.Value, value InterQueryCacheValue) int + Delete(key ast.Value) + UpdateConfig(config *Config) +} + +// NewInterQueryCache returns a new inter-query cache. +func NewInterQueryCache(config *Config) InterQueryCache { + return &cache{ + items: map[string]InterQueryCacheValue{}, + usage: 0, + config: config, + l: list.New(), + } +} + +type cache struct { + items map[string]InterQueryCacheValue + usage int64 + config *Config + l *list.List + mtx sync.Mutex +} + +// Insert inserts a key k into the cache with value v. +func (c *cache) Insert(k ast.Value, v InterQueryCacheValue) (dropped int) { + c.mtx.Lock() + defer c.mtx.Unlock() + return c.unsafeInsert(k, v) +} + +// Get returns the value in the cache for k. +func (c *cache) Get(k ast.Value) (InterQueryCacheValue, bool) { + c.mtx.Lock() + defer c.mtx.Unlock() + return c.unsafeGet(k) +} + +// Delete deletes the value in the cache for k. +func (c *cache) Delete(k ast.Value) { + c.mtx.Lock() + defer c.mtx.Unlock() + c.unsafeDelete(k) +} + +func (c *cache) UpdateConfig(config *Config) { + if config == nil { + return + } + c.mtx.Lock() + defer c.mtx.Unlock() + c.config = config +} + +func (c *cache) unsafeInsert(k ast.Value, v InterQueryCacheValue) (dropped int) { + size := v.SizeInBytes() + limit := c.maxSizeBytes() + + if limit > 0 { + if size > limit { + dropped++ + return dropped + } + + for key := c.l.Front(); key != nil && (c.usage+size > limit); key = key.Next() { + dropKey := key.Value.(ast.Value) + c.unsafeDelete(dropKey) + c.l.Remove(key) + dropped++ + } + } + + c.items[k.String()] = v + c.l.PushBack(k) + c.usage += size + return dropped +} + +func (c *cache) unsafeGet(k ast.Value) (InterQueryCacheValue, bool) { + value, ok := c.items[k.String()] + return value, ok +} + +func (c *cache) unsafeDelete(k ast.Value) { + value, ok := c.unsafeGet(k) + if !ok { + return + } + + c.usage -= int64(value.SizeInBytes()) + delete(c.items, k.String()) +} + +func (c *cache) maxSizeBytes() int64 { + if c.config == nil { + return defaultMaxSizeBytes + } + return *c.config.InterQueryBuiltinCache.MaxSizeBytes +} diff --git a/vendor/github.com/open-policy-agent/opa/topdown/cancel.go b/vendor/github.com/open-policy-agent/opa/topdown/cancel.go new file mode 100644 index 00000000..534e0799 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/topdown/cancel.go @@ -0,0 +1,33 @@ +// Copyright 2017 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package topdown + +import ( + "sync/atomic" +) + +// Cancel defines the interface for cancelling topdown queries. Cancel +// operations are thread-safe and idempotent. +type Cancel interface { + Cancel() + Cancelled() bool +} + +type cancel struct { + flag int32 +} + +// NewCancel returns a new Cancel object. +func NewCancel() Cancel { + return &cancel{} +} + +func (c *cancel) Cancel() { + atomic.StoreInt32(&c.flag, 1) +} + +func (c *cancel) Cancelled() bool { + return atomic.LoadInt32(&c.flag) != 0 +} diff --git a/vendor/github.com/open-policy-agent/opa/topdown/casts.go b/vendor/github.com/open-policy-agent/opa/topdown/casts.go new file mode 100644 index 00000000..c207bfd1 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/topdown/casts.go @@ -0,0 +1,117 @@ +// Copyright 2018 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package topdown + +import ( + "strconv" + + "github.com/open-policy-agent/opa/ast" + "github.com/open-policy-agent/opa/topdown/builtins" +) + +func builtinToNumber(a ast.Value) (ast.Value, error) { + switch a := a.(type) { + case ast.Null: + return ast.Number("0"), nil + case ast.Boolean: + if a { + return ast.Number("1"), nil + } + return ast.Number("0"), nil + case ast.Number: + return a, nil + case ast.String: + _, err := strconv.ParseFloat(string(a), 64) + if err != nil { + return nil, err + } + return ast.Number(a), nil + } + return nil, builtins.NewOperandTypeErr(1, a, "null", "boolean", "number", "string") +} + +// Deprecated in v0.13.0. +func builtinToArray(a ast.Value) (ast.Value, error) { + switch val := a.(type) { + case *ast.Array: + return val, nil + case ast.Set: + arr := make([]*ast.Term, val.Len()) + i := 0 + val.Foreach(func(term *ast.Term) { + arr[i] = term + i++ + }) + return ast.NewArray(arr...), nil + default: + return nil, builtins.NewOperandTypeErr(1, a, "array", "set") + } +} + +// Deprecated in v0.13.0. +func builtinToSet(a ast.Value) (ast.Value, error) { + switch val := a.(type) { + case *ast.Array: + s := ast.NewSet() + val.Foreach(func(v *ast.Term) { + s.Add(v) + }) + return s, nil + case ast.Set: + return val, nil + default: + return nil, builtins.NewOperandTypeErr(1, a, "array", "set") + } +} + +// Deprecated in v0.13.0. +func builtinToString(a ast.Value) (ast.Value, error) { + switch val := a.(type) { + case ast.String: + return val, nil + default: + return nil, builtins.NewOperandTypeErr(1, a, "string") + } +} + +// Deprecated in v0.13.0. +func builtinToBoolean(a ast.Value) (ast.Value, error) { + switch val := a.(type) { + case ast.Boolean: + return val, nil + default: + return nil, builtins.NewOperandTypeErr(1, a, "boolean") + } +} + +// Deprecated in v0.13.0. +func builtinToNull(a ast.Value) (ast.Value, error) { + switch val := a.(type) { + case ast.Null: + return val, nil + default: + return nil, builtins.NewOperandTypeErr(1, a, "null") + } +} + +// Deprecated in v0.13.0. +func builtinToObject(a ast.Value) (ast.Value, error) { + switch val := a.(type) { + case ast.Object: + return val, nil + default: + return nil, builtins.NewOperandTypeErr(1, a, "object") + } +} + +func init() { + RegisterFunctionalBuiltin1(ast.ToNumber.Name, builtinToNumber) + RegisterFunctionalBuiltin1(ast.CastArray.Name, builtinToArray) + RegisterFunctionalBuiltin1(ast.CastSet.Name, builtinToSet) + RegisterFunctionalBuiltin1(ast.CastString.Name, builtinToString) + RegisterFunctionalBuiltin1(ast.CastBoolean.Name, builtinToBoolean) + RegisterFunctionalBuiltin1(ast.CastNull.Name, builtinToNull) + RegisterFunctionalBuiltin1(ast.CastObject.Name, builtinToObject) +} diff --git a/vendor/github.com/open-policy-agent/opa/topdown/cidr.go b/vendor/github.com/open-policy-agent/opa/topdown/cidr.go new file mode 100644 index 00000000..2123698c --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/topdown/cidr.go @@ -0,0 +1,405 @@ +package topdown + +import ( + "bytes" + "errors" + "fmt" + "math/big" + "net" + "sort" + + "github.com/open-policy-agent/opa/ast" + cidrMerge "github.com/open-policy-agent/opa/internal/cidr/merge" + "github.com/open-policy-agent/opa/topdown/builtins" +) + +func getNetFromOperand(v ast.Value) (*net.IPNet, error) { + subnetStringA, err := builtins.StringOperand(v, 1) + if err != nil { + return nil, err + } + + _, cidrnet, err := net.ParseCIDR(string(subnetStringA)) + if err != nil { + return nil, err + } + + return cidrnet, nil +} + +func getLastIP(cidr *net.IPNet) (net.IP, error) { + prefixLen, bits := cidr.Mask.Size() + if prefixLen == 0 && bits == 0 { + // non-standard mask, see https://golang.org/pkg/net/#IPMask.Size + return nil, fmt.Errorf("CIDR mask is in non-standard format") + } + var lastIP []byte + if prefixLen == bits { + // Special case for single ip address ranges ex: 192.168.1.1/32 + // We can just use the starting IP as the last IP + lastIP = cidr.IP + } else { + // Use big.Int's so we can handle ipv6 addresses + firstIPInt := new(big.Int) + firstIPInt.SetBytes(cidr.IP) + hostLen := uint(bits) - uint(prefixLen) + lastIPInt := big.NewInt(1) + lastIPInt.Lsh(lastIPInt, hostLen) + lastIPInt.Sub(lastIPInt, big.NewInt(1)) + lastIPInt.Or(lastIPInt, firstIPInt) + + ipBytes := lastIPInt.Bytes() + lastIP = make([]byte, bits/8) + + // Pack our IP bytes into the end of the return array, + // since big.Int.Bytes() removes front zero padding. + for i := 1; i <= len(lastIPInt.Bytes()); i++ { + lastIP[len(lastIP)-i] = ipBytes[len(ipBytes)-i] + } + } + + return lastIP, nil +} + +func builtinNetCIDRIntersects(a, b ast.Value) (ast.Value, error) { + cidrnetA, err := getNetFromOperand(a) + if err != nil { + return nil, err + } + + cidrnetB, err := getNetFromOperand(b) + if err != nil { + return nil, err + } + + // If either net contains the others starting IP they are overlapping + cidrsOverlap := (cidrnetA.Contains(cidrnetB.IP) || cidrnetB.Contains(cidrnetA.IP)) + + return ast.Boolean(cidrsOverlap), nil +} + +func builtinNetCIDRContains(a, b ast.Value) (ast.Value, error) { + cidrnetA, err := getNetFromOperand(a) + if err != nil { + return nil, err + } + + // b could be either an IP addressor CIDR string, try to parse it as an IP first, fall back to CIDR + bStr, err := builtins.StringOperand(b, 1) + if err != nil { + return nil, err + } + + ip := net.ParseIP(string(bStr)) + if ip != nil { + return ast.Boolean(cidrnetA.Contains(ip)), nil + } + + // It wasn't an IP, try and parse it as a CIDR + cidrnetB, err := getNetFromOperand(b) + if err != nil { + return nil, fmt.Errorf("not a valid textual representation of an IP address or CIDR: %s", string(bStr)) + } + + // We can determine if cidr A contains cidr B if and only if A contains + // the starting address of B and the last address in B. + cidrContained := false + if cidrnetA.Contains(cidrnetB.IP) { + // Only spend time calculating the last IP if the starting IP is already verified to be in cidr A + lastIP, err := getLastIP(cidrnetB) + if err != nil { + return nil, err + } + cidrContained = cidrnetA.Contains(lastIP) + } + + return ast.Boolean(cidrContained), nil +} + +var errNetCIDRContainsMatchElementType = errors.New("element must be string or non-empty array") + +func getCIDRMatchTerm(a *ast.Term) (*ast.Term, error) { + switch v := a.Value.(type) { + case ast.String: + return a, nil + case *ast.Array: + if v.Len() == 0 { + return nil, errNetCIDRContainsMatchElementType + } + return v.Elem(0), nil + default: + return nil, errNetCIDRContainsMatchElementType + } +} + +func evalNetCIDRContainsMatchesOperand(operand int, a *ast.Term, iter func(cidr, index *ast.Term) error) error { + switch v := a.Value.(type) { + case ast.String: + return iter(a, a) + case *ast.Array: + for i := 0; i < v.Len(); i++ { + cidr, err := getCIDRMatchTerm(v.Elem(i)) + if err != nil { + return fmt.Errorf("operand %v: %v", operand, err) + } + if err := iter(cidr, ast.IntNumberTerm(i)); err != nil { + return err + } + } + return nil + case ast.Set: + return v.Iter(func(x *ast.Term) error { + cidr, err := getCIDRMatchTerm(x) + if err != nil { + return fmt.Errorf("operand %v: %v", operand, err) + } + return iter(cidr, x) + }) + case ast.Object: + return v.Iter(func(k, v *ast.Term) error { + cidr, err := getCIDRMatchTerm(v) + if err != nil { + return fmt.Errorf("operand %v: %v", operand, err) + } + return iter(cidr, k) + }) + } + return nil +} + +func builtinNetCIDRContainsMatches(bctx BuiltinContext, args []*ast.Term, iter func(*ast.Term) error) error { + result := ast.NewSet() + err := evalNetCIDRContainsMatchesOperand(1, args[0], func(cidr1 *ast.Term, index1 *ast.Term) error { + return evalNetCIDRContainsMatchesOperand(2, args[1], func(cidr2 *ast.Term, index2 *ast.Term) error { + if v, err := builtinNetCIDRContains(cidr1.Value, cidr2.Value); err != nil { + return err + } else if vb, ok := v.(ast.Boolean); ok && bool(vb) { + result.Add(ast.ArrayTerm(index1, index2)) + } + return nil + }) + }) + if err == nil { + return iter(ast.NewTerm(result)) + } + return err +} + +func builtinNetCIDRExpand(bctx BuiltinContext, operands []*ast.Term, iter func(*ast.Term) error) error { + + s, err := builtins.StringOperand(operands[0].Value, 1) + if err != nil { + return err + } + + ip, ipNet, err := net.ParseCIDR(string(s)) + if err != nil { + return err + } + + result := ast.NewSet() + + for ip := ip.Mask(ipNet.Mask); ipNet.Contains(ip); incIP(ip) { + + if bctx.Cancel != nil && bctx.Cancel.Cancelled() { + return Halt{ + Err: &Error{ + Code: CancelErr, + Message: "net.cidr_expand: timed out before generating all IP addresses", + }, + } + } + + result.Add(ast.StringTerm(ip.String())) + } + + return iter(ast.NewTerm(result)) +} + +type cidrBlockRange struct { + First *net.IP + Last *net.IP + Network *net.IPNet +} + +type cidrBlockRanges []*cidrBlockRange + +// Implement Sort interface +func (c cidrBlockRanges) Len() int { + return len(c) +} + +func (c cidrBlockRanges) Swap(i, j int) { + c[i], c[j] = c[j], c[i] +} + +func (c cidrBlockRanges) Less(i, j int) bool { + // Compare last IP. + cmp := bytes.Compare(*c[i].Last, *c[j].Last) + if cmp < 0 { + return true + } else if cmp > 0 { + return false + } + + // Then compare first IP. + cmp = bytes.Compare(*c[i].First, *c[i].First) + if cmp < 0 { + return true + } else if cmp > 0 { + return false + } + + // Ranges are Equal. + return false +} + +// builtinNetCIDRMerge merges the provided list of IP addresses and subnets into the smallest possible list of CIDRs. +// It merges adjacent subnets where possible, those contained within others and also removes any duplicates. +// Original Algorithm: https://github.com/netaddr/netaddr. +func builtinNetCIDRMerge(bctx BuiltinContext, operands []*ast.Term, iter func(*ast.Term) error) error { + networks := []*net.IPNet{} + + switch v := operands[0].Value.(type) { + case *ast.Array: + for i := 0; i < v.Len(); i++ { + network, err := generateIPNet(v.Elem(i)) + if err != nil { + return err + } + networks = append(networks, network) + } + case ast.Set: + err := v.Iter(func(x *ast.Term) error { + network, err := generateIPNet(x) + if err != nil { + return err + } + networks = append(networks, network) + return nil + }) + if err != nil { + return err + } + default: + return errors.New("operand must be an array") + } + + merged := evalNetCIDRMerge(networks) + + result := ast.NewSet() + for _, network := range merged { + result.Add(ast.StringTerm(network.String())) + } + + return iter(ast.NewTerm(result)) +} + +func evalNetCIDRMerge(networks []*net.IPNet) []*net.IPNet { + if len(networks) == 0 { + return nil + } + + var ranges cidrBlockRanges + + // For each CIDR, create an IP range. Sort them and merge when possible. + for _, network := range networks { + firstIP, lastIP := cidrMerge.GetAddressRange(*network) + ranges = append(ranges, &cidrBlockRange{ + First: &firstIP, + Last: &lastIP, + Network: network, + }) + } + + // merge CIDRs. + merged := mergeCIDRs(ranges) + + // convert ranges into an equivalent list of net.IPNet. + result := []*net.IPNet{} + + for _, r := range merged { + // Not merged with any other CIDR. + if r.Network != nil { + result = append(result, r.Network) + } else { + // Find new network that represents the merged range. + rangeCIDRs := cidrMerge.RangeToCIDRs(*r.First, *r.Last) + result = append(result, rangeCIDRs...) + } + } + return result +} + +func generateIPNet(term *ast.Term) (*net.IPNet, error) { + switch e := term.Value.(type) { + case ast.String: + network := &net.IPNet{} + // try to parse element as an IP first, fall back to CIDR + ip := net.ParseIP(string(e)) + if ip != nil { + network.IP = ip + network.Mask = ip.DefaultMask() + } else { + var err error + _, network, err = net.ParseCIDR(string(e)) + if err != nil { + return nil, err + } + } + return network, nil + default: + return nil, errors.New("element must be string") + } +} + +func mergeCIDRs(ranges cidrBlockRanges) cidrBlockRanges { + sort.Sort(ranges) + + // Merge adjacent CIDRs if possible. + for i := len(ranges) - 1; i > 0; i-- { + previousIP := cidrMerge.GetPreviousIP(*ranges[i].First) + + // If the previous IP of the current network overlaps + // with the last IP of the previous network in the + // list, then merge the two ranges together. + if bytes.Compare(previousIP, *ranges[i-1].Last) <= 0 { + var firstIP *net.IP + if bytes.Compare(*ranges[i-1].First, *ranges[i].First) < 0 { + firstIP = ranges[i-1].First + } else { + firstIP = ranges[i].First + } + + lastIPRange := make(net.IP, len(*ranges[i].Last)) + copy(lastIPRange, *ranges[i].Last) + + firstIPRange := make(net.IP, len(*firstIP)) + copy(firstIPRange, *firstIP) + + ranges[i-1] = &cidrBlockRange{First: &firstIPRange, Last: &lastIPRange, Network: nil} + + // Delete ranges[i] since merged with the previous. + ranges = append(ranges[:i], ranges[i+1:]...) + } + } + return ranges +} + +func incIP(ip net.IP) { + for j := len(ip) - 1; j >= 0; j-- { + ip[j]++ + if ip[j] > 0 { + break + } + } +} + +func init() { + RegisterFunctionalBuiltin2(ast.NetCIDROverlap.Name, builtinNetCIDRContains) + RegisterFunctionalBuiltin2(ast.NetCIDRIntersects.Name, builtinNetCIDRIntersects) + RegisterFunctionalBuiltin2(ast.NetCIDRContains.Name, builtinNetCIDRContains) + RegisterBuiltinFunc(ast.NetCIDRContainsMatches.Name, builtinNetCIDRContainsMatches) + RegisterBuiltinFunc(ast.NetCIDRExpand.Name, builtinNetCIDRExpand) + RegisterBuiltinFunc(ast.NetCIDRMerge.Name, builtinNetCIDRMerge) +} diff --git a/vendor/github.com/open-policy-agent/opa/topdown/comparison.go b/vendor/github.com/open-policy-agent/opa/topdown/comparison.go new file mode 100644 index 00000000..96be984a --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/topdown/comparison.go @@ -0,0 +1,48 @@ +// Copyright 2016 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package topdown + +import "github.com/open-policy-agent/opa/ast" + +type compareFunc func(a, b ast.Value) bool + +func compareGreaterThan(a, b ast.Value) bool { + return ast.Compare(a, b) > 0 +} + +func compareGreaterThanEq(a, b ast.Value) bool { + return ast.Compare(a, b) >= 0 +} + +func compareLessThan(a, b ast.Value) bool { + return ast.Compare(a, b) < 0 +} + +func compareLessThanEq(a, b ast.Value) bool { + return ast.Compare(a, b) <= 0 +} + +func compareNotEq(a, b ast.Value) bool { + return ast.Compare(a, b) != 0 +} + +func compareEq(a, b ast.Value) bool { + return ast.Compare(a, b) == 0 +} + +func builtinCompare(cmp compareFunc) FunctionalBuiltin2 { + return func(a, b ast.Value) (ast.Value, error) { + return ast.Boolean(cmp(a, b)), nil + } +} + +func init() { + RegisterFunctionalBuiltin2(ast.GreaterThan.Name, builtinCompare(compareGreaterThan)) + RegisterFunctionalBuiltin2(ast.GreaterThanEq.Name, builtinCompare(compareGreaterThanEq)) + RegisterFunctionalBuiltin2(ast.LessThan.Name, builtinCompare(compareLessThan)) + RegisterFunctionalBuiltin2(ast.LessThanEq.Name, builtinCompare(compareLessThanEq)) + RegisterFunctionalBuiltin2(ast.NotEqual.Name, builtinCompare(compareNotEq)) + RegisterFunctionalBuiltin2(ast.Equal.Name, builtinCompare(compareEq)) +} diff --git a/vendor/github.com/open-policy-agent/opa/topdown/copypropagation/copypropagation.go b/vendor/github.com/open-policy-agent/opa/topdown/copypropagation/copypropagation.go new file mode 100644 index 00000000..2dee16f0 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/topdown/copypropagation/copypropagation.go @@ -0,0 +1,439 @@ +// Copyright 2018 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package copypropagation + +import ( + "sort" + + "github.com/open-policy-agent/opa/ast" +) + +// CopyPropagator implements a simple copy propagation optimization to remove +// intermediate variables in partial evaluation results. +// +// For example, given the query: input.x > 1 where 'input' is unknown, the +// compiled query would become input.x = a; a > 1 which would remain in the +// partial evaluation result. The CopyPropagator will remove the variable +// assignment so that partial evaluation simply outputs input.x > 1. +// +// In many cases, copy propagation can remove all variables from the result of +// partial evaluation which simplifies evaluation for non-OPA consumers. +// +// In some cases, copy propagation cannot remove all variables. If the output of +// a built-in call is subsequently used as a ref head, the output variable must +// be kept. For example. sort(input, x); x[0] == 1. In this case, copy +// propagation cannot replace x[0] == 1 with sort(input, x)[0] == 1 as this is +// not legal. +type CopyPropagator struct { + livevars ast.VarSet // vars that must be preserved in the resulting query + sorted []ast.Var // sorted copy of vars to ensure deterministic result + ensureNonEmptyBody bool + compiler *ast.Compiler +} + +// New returns a new CopyPropagator that optimizes queries while preserving vars +// in the livevars set. +func New(livevars ast.VarSet) *CopyPropagator { + + sorted := make([]ast.Var, 0, len(livevars)) + for v := range livevars { + sorted = append(sorted, v) + } + + sort.Slice(sorted, func(i, j int) bool { + return sorted[i].Compare(sorted[j]) < 0 + }) + + return &CopyPropagator{livevars: livevars, sorted: sorted} +} + +// WithEnsureNonEmptyBody configures p to ensure that results are always non-empty. +func (p *CopyPropagator) WithEnsureNonEmptyBody(yes bool) *CopyPropagator { + p.ensureNonEmptyBody = yes + return p +} + +// WithCompiler configures the compiler to read from while processing the query. This +// should be the same compiler used to compile the original policy. +func (p *CopyPropagator) WithCompiler(c *ast.Compiler) *CopyPropagator { + p.compiler = c + return p +} + +// Apply executes the copy propagation optimization and returns a new query. +func (p *CopyPropagator) Apply(query ast.Body) ast.Body { + + result := ast.NewBody() + + uf, ok := makeDisjointSets(p.livevars, query) + if !ok { + return query + } + + // Compute set of vars that appear in the head of refs in the query. If a var + // is dereferenced, we can plug it with a constant value, but it is not always + // optimal to do so. + // TODO: Improve the algorithm for when we should plug constants/calls/etc + headvars := ast.NewVarSet() + ast.WalkRefs(query, func(x ast.Ref) bool { + if v, ok := x[0].Value.(ast.Var); ok { + if root, ok := uf.Find(v); ok { + root.constant = nil + headvars.Add(root.key.(ast.Var)) + } else { + headvars.Add(v) + } + } + return false + }) + + removedEqs := ast.NewValueMap() + + for _, expr := range query { + + pctx := &plugContext{ + removedEqs: removedEqs, + uf: uf, + negated: expr.Negated, + headvars: headvars, + } + + expr = p.plugBindings(pctx, expr) + + if p.updateBindings(pctx, expr) { + result.Append(expr) + } + } + + // Run post-processing step on the query to ensure that all live vars are bound + // in the result. The plugging that happens above substitutes all vars in the + // same set with the root. + // + // This step should run before the next step to prevent unnecessary bindings + // from being added to the result. For example: + // + // - Given the following result: + // - Given the following removed equalities: "x = input.x" and "y = input" + // - Given the following liveset: {x} + // + // If this step were to run AFTER the following step, the output would be: + // + // x = input.x; y = input + // + // Even though y = input is not required. + for _, v := range p.sorted { + if root, ok := uf.Find(v); ok { + if root.constant != nil { + result.Append(ast.Equality.Expr(ast.NewTerm(v), root.constant)) + } else if b := removedEqs.Get(root.key); b != nil { + result.Append(ast.Equality.Expr(ast.NewTerm(v), ast.NewTerm(b))) + } else if root.key != v { + result.Append(ast.Equality.Expr(ast.NewTerm(v), ast.NewTerm(root.key))) + } + } + } + + // Run post-processing step on query to ensure that all killed exprs are + // accounted for. There are several cases we look for: + // + // * If an expr is killed but the binding is never used, the query + // must still include the expr. For example, given the query 'input.x = a' and + // an empty livevar set, the result must include the ref input.x otherwise the + // query could be satisfied without input.x being defined. + // + // * If an expr is killed that provided safety to vars which are not + // otherwise being made safe by the current result. + // + // For any of these cases we re-add the removed equality expression + // to the current result. + + // Invariant: Live vars are bound (above) and reserved vars are implicitly ground. + safe := ast.ReservedVars.Copy() + safe.Update(p.livevars) + safe.Update(ast.OutputVarsFromBody(p.compiler, result, safe)) + unsafe := result.Vars(ast.SafetyCheckVisitorParams).Diff(safe) + + for _, b := range sortbindings(removedEqs) { + removedEq := ast.Equality.Expr(ast.NewTerm(b.k), ast.NewTerm(b.v)) + + providesSafety := false + outputVars := ast.OutputVarsFromExpr(p.compiler, removedEq, safe) + diff := unsafe.Diff(outputVars) + if len(diff) < len(unsafe) { + unsafe = diff + providesSafety = true + } + + if providesSafety || !containedIn(b.v, result) { + result.Append(removedEq) + safe.Update(outputVars) + } + } + + if len(unsafe) > 0 { + // NOTE(tsandall): This should be impossible but if it does occur, throw + // away the result rather than generating unsafe output. + return query + } + + if p.ensureNonEmptyBody && len(result) == 0 { + result = append(result, ast.NewExpr(ast.BooleanTerm(true))) + } + + return result +} + +// plugBindings applies the binding list and union-find to x. This process +// removes as many variables as possible. +func (p *CopyPropagator) plugBindings(pctx *plugContext, expr *ast.Expr) *ast.Expr { + + xform := bindingPlugTransform{ + pctx: pctx, + } + + // Deep copy the expression as it may be mutated during the transform and + // the caller running copy propagation may have references to the + // expression. Note, the transform does not contain any error paths and + // should never return a non-expression value for the root so consider + // errors unreachable. + x, err := ast.Transform(xform, expr.Copy()) + + if expr, ok := x.(*ast.Expr); !ok || err != nil { + panic("unreachable") + } else { + return expr + } +} + +type bindingPlugTransform struct { + pctx *plugContext +} + +func (t bindingPlugTransform) Transform(x interface{}) (interface{}, error) { + switch x := x.(type) { + case ast.Var: + return t.plugBindingsVar(t.pctx, x), nil + case ast.Ref: + return t.plugBindingsRef(t.pctx, x), nil + default: + return x, nil + } +} + +func (t bindingPlugTransform) plugBindingsVar(pctx *plugContext, v ast.Var) ast.Value { + + var result ast.Value = v + + // Apply union-find to remove redundant variables from input. + root, ok := pctx.uf.Find(v) + if ok { + result = root.Value() + } + + // Apply binding list to substitute remaining vars. + v, ok = result.(ast.Var) + if !ok { + return result + } + b := pctx.removedEqs.Get(v) + if b == nil { + return result + } + if pctx.negated && !b.IsGround() { + return result + } + + if r, ok := b.(ast.Ref); ok && r.OutputVars().Contains(v) { + return result + } + + return b +} + +func (t bindingPlugTransform) plugBindingsRef(pctx *plugContext, v ast.Ref) ast.Ref { + + // Apply union-find to remove redundant variables from input. + if root, ok := pctx.uf.Find(v[0].Value); ok { + v[0].Value = root.Value() + } + + result := v + + // Refs require special handling. If the head of the ref was killed, then + // the rest of the ref must be concatenated with the new base. + if b := pctx.removedEqs.Get(v[0].Value); b != nil { + if !pctx.negated || b.IsGround() { + var base ast.Ref + switch x := b.(type) { + case ast.Ref: + base = x + default: + base = ast.Ref{ast.NewTerm(x)} + } + result = base.Concat(v[1:]) + } + } + + return result +} + +// updateBindings returns false if the expression can be killed. If the +// expression is killed, the binding list is updated to map a var to value. +func (p *CopyPropagator) updateBindings(pctx *plugContext, expr *ast.Expr) bool { + if pctx.negated || len(expr.With) > 0 { + return true + } + if expr.IsEquality() { + a, b := expr.Operand(0), expr.Operand(1) + if a.Equal(b) { + return false + } + k, v, keep := p.updateBindingsEq(a, b) + if !keep { + if v != nil { + pctx.removedEqs.Put(k, v) + } + return false + } + } else if expr.IsCall() { + terms := expr.Terms.([]*ast.Term) + if p.compiler.GetArity(expr.Operator()) == len(terms)-2 { // with captured output + output := terms[len(terms)-1] + if k, ok := output.Value.(ast.Var); ok && !p.livevars.Contains(k) && !pctx.headvars.Contains(k) { + pctx.removedEqs.Put(k, ast.CallTerm(terms[:len(terms)-1]...).Value) + return false + } + } + } + return !isNoop(expr) +} + +func (p *CopyPropagator) updateBindingsEq(a, b *ast.Term) (ast.Var, ast.Value, bool) { + k, v, keep := p.updateBindingsEqAsymmetric(a, b) + if !keep { + return k, v, keep + } + return p.updateBindingsEqAsymmetric(b, a) +} + +func (p *CopyPropagator) updateBindingsEqAsymmetric(a, b *ast.Term) (ast.Var, ast.Value, bool) { + k, ok := a.Value.(ast.Var) + if !ok || p.livevars.Contains(k) { + return "", nil, true + } + + switch b.Value.(type) { + case ast.Ref, ast.Call: + return k, b.Value, false + } + + return "", nil, true +} + +type plugContext struct { + removedEqs *ast.ValueMap + uf *unionFind + headvars ast.VarSet + negated bool +} + +type binding struct { + k ast.Value + v ast.Value +} + +func containedIn(value ast.Value, x interface{}) bool { + var stop bool + switch v := value.(type) { + case ast.Ref: + ast.WalkRefs(x, func(other ast.Ref) bool { + if stop || other.HasPrefix(v) { + stop = true + return stop + } + return false + }) + default: + ast.WalkTerms(x, func(other *ast.Term) bool { + if stop || other.Value.Compare(v) == 0 { + stop = true + return stop + } + return false + }) + } + return stop +} + +func sortbindings(bindings *ast.ValueMap) []*binding { + sorted := make([]*binding, 0, bindings.Len()) + bindings.Iter(func(k ast.Value, v ast.Value) bool { + sorted = append(sorted, &binding{k, v}) + return false + }) + sort.Slice(sorted, func(i, j int) bool { + return sorted[i].k.Compare(sorted[j].k) < 0 + }) + return sorted +} + +// makeDisjointSets builds the union-find structure for the query. The structure +// is built by processing all of the equality exprs in the query. Sets represent +// vars that must be equal to each other. In addition to vars, each set can have +// at most one constant. If the query contains expressions that cannot be +// satisfied (e.g., because a set has multiple constants) this function returns +// false. +func makeDisjointSets(livevars ast.VarSet, query ast.Body) (*unionFind, bool) { + uf := newUnionFind(func(r1, r2 *unionFindRoot) (*unionFindRoot, *unionFindRoot) { + if v, ok := r1.key.(ast.Var); ok && livevars.Contains(v) { + return r1, r2 + } + return r2, r1 + }) + for _, expr := range query { + if expr.IsEquality() && !expr.Negated && len(expr.With) == 0 { + a, b := expr.Operand(0), expr.Operand(1) + varA, ok1 := a.Value.(ast.Var) + varB, ok2 := b.Value.(ast.Var) + if ok1 && ok2 { + if _, ok := uf.Merge(varA, varB); !ok { + return nil, false + } + } else if ok1 && ast.IsConstant(b.Value) { + root := uf.MakeSet(varA) + if root.constant != nil && !root.constant.Equal(b) { + return nil, false + } + root.constant = b + } else if ok2 && ast.IsConstant(a.Value) { + root := uf.MakeSet(varB) + if root.constant != nil && !root.constant.Equal(a) { + return nil, false + } + root.constant = a + } + } + } + + return uf, true +} + +func isNoop(expr *ast.Expr) bool { + + if !expr.IsCall() { + term := expr.Terms.(*ast.Term) + if !ast.IsConstant(term.Value) { + return false + } + return !ast.Boolean(false).Equal(term.Value) + } + + // A==A can be ignored + if expr.Operator().Equal(ast.Equal.Ref()) { + return expr.Operand(0).Equal(expr.Operand(1)) + } + + return false +} diff --git a/vendor/github.com/open-policy-agent/opa/topdown/copypropagation/unionfind.go b/vendor/github.com/open-policy-agent/opa/topdown/copypropagation/unionfind.go new file mode 100644 index 00000000..38ec56f3 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/topdown/copypropagation/unionfind.go @@ -0,0 +1,135 @@ +// Copyright 2020 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package copypropagation + +import ( + "fmt" + + "github.com/open-policy-agent/opa/ast" + "github.com/open-policy-agent/opa/util" +) + +type rankFunc func(*unionFindRoot, *unionFindRoot) (*unionFindRoot, *unionFindRoot) + +type unionFind struct { + roots *util.HashMap + parents *ast.ValueMap + rank rankFunc +} + +func newUnionFind(rank rankFunc) *unionFind { + return &unionFind{ + roots: util.NewHashMap(func(a util.T, b util.T) bool { + return a.(ast.Value).Compare(b.(ast.Value)) == 0 + }, func(v util.T) int { + return v.(ast.Value).Hash() + }), + parents: ast.NewValueMap(), + rank: rank, + } +} + +func (uf *unionFind) MakeSet(v ast.Value) *unionFindRoot { + + root, ok := uf.Find(v) + if ok { + return root + } + + root = newUnionFindRoot(v) + uf.parents.Put(v, v) + uf.roots.Put(v, root) + return root +} + +func (uf *unionFind) Find(v ast.Value) (*unionFindRoot, bool) { + + parent := uf.parents.Get(v) + if parent == nil { + return nil, false + } + + if parent.Compare(v) == 0 { + r, ok := uf.roots.Get(v) + return r.(*unionFindRoot), ok + } + + return uf.Find(parent) +} + +func (uf *unionFind) Merge(a, b ast.Value) (*unionFindRoot, bool) { + + r1 := uf.MakeSet(a) + r2 := uf.MakeSet(b) + + if r1 != r2 { + + r1, r2 = uf.rank(r1, r2) + + uf.parents.Put(r2.key, r1.key) + uf.roots.Delete(r2.key) + + // Sets can have at most one constant value associated with them. When + // unioning, we must preserve this invariant. If a set has two constants, + // there will be no way to prove the query. + if r1.constant != nil && r2.constant != nil && !r1.constant.Equal(r2.constant) { + return nil, false + } else if r1.constant == nil { + r1.constant = r2.constant + } + } + + return r1, true +} + +func (uf *unionFind) String() string { + o := struct { + Roots map[string]interface{} + Parents map[string]ast.Value + }{ + map[string]interface{}{}, + map[string]ast.Value{}, + } + + uf.roots.Iter(func(k util.T, v util.T) bool { + o.Roots[k.(ast.Value).String()] = struct { + Constant *ast.Term + Key ast.Value + }{ + v.(*unionFindRoot).constant, + v.(*unionFindRoot).key, + } + return true + }) + + uf.parents.Iter(func(k ast.Value, v ast.Value) bool { + o.Parents[k.String()] = v + return true + }) + + return string(util.MustMarshalJSON(o)) +} + +type unionFindRoot struct { + key ast.Value + constant *ast.Term +} + +func newUnionFindRoot(key ast.Value) *unionFindRoot { + return &unionFindRoot{ + key: key, + } +} + +func (r *unionFindRoot) Value() ast.Value { + if r.constant != nil { + return r.constant.Value + } + return r.key +} + +func (r *unionFindRoot) String() string { + return fmt.Sprintf("{key: %s, constant: %s", r.key, r.constant) +} diff --git a/vendor/github.com/open-policy-agent/opa/topdown/crypto.go b/vendor/github.com/open-policy-agent/opa/topdown/crypto.go new file mode 100644 index 00000000..2b5a96ea --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/topdown/crypto.go @@ -0,0 +1,419 @@ +// Copyright 2018 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package topdown + +import ( + "bytes" + "crypto/hmac" + "crypto/md5" + "crypto/sha1" + "crypto/sha256" + "crypto/sha512" + "crypto/x509" + "encoding/base64" + "encoding/json" + "encoding/pem" + "fmt" + "hash" + "io/ioutil" + "os" + "strings" + + "github.com/open-policy-agent/opa/ast" + "github.com/open-policy-agent/opa/internal/jwx/jwk" + "github.com/open-policy-agent/opa/topdown/builtins" + "github.com/open-policy-agent/opa/util" +) + +const ( + // blockTypeCertificate indicates this PEM block contains the signed certificate. + // Exported for tests. + blockTypeCertificate = "CERTIFICATE" + // blockTypeCertificateRequest indicates this PEM block contains a certificate + // request. Exported for tests. + blockTypeCertificateRequest = "CERTIFICATE REQUEST" + // blockTypeRSAPrivateKey indicates this PEM block contains a RSA private key. + // Exported for tests. + blockTypeRSAPrivateKey = "RSA PRIVATE KEY" + // blockTypeRSAPrivateKey indicates this PEM block contains a RSA private key. + // Exported for tests. + blockTypePrivateKey = "PRIVATE KEY" +) + +func builtinCryptoX509ParseCertificates(a ast.Value) (ast.Value, error) { + input, err := builtins.StringOperand(a, 1) + if err != nil { + return nil, err + } + + certs, err := getX509CertsFromString(string(input)) + if err != nil { + return nil, err + } + + return ast.InterfaceToValue(certs) +} + +func builtinCryptoX509ParseAndVerifyCertificates( + _ BuiltinContext, args []*ast.Term, iter func(*ast.Term) error) error { + + a := args[0].Value + input, err := builtins.StringOperand(a, 1) + if err != nil { + return err + } + + invalid := ast.ArrayTerm( + ast.BooleanTerm(false), + ast.NewTerm(ast.NewArray()), + ) + + certs, err := getX509CertsFromString(string(input)) + if err != nil { + return iter(invalid) + } + + verified, err := verifyX509CertificateChain(certs) + if err != nil { + return iter(invalid) + } + + value, err := ast.InterfaceToValue(verified) + if err != nil { + return err + } + + valid := ast.ArrayTerm( + ast.BooleanTerm(true), + ast.NewTerm(value), + ) + + return iter(valid) +} + +func builtinCryptoX509ParseCertificateRequest(a ast.Value) (ast.Value, error) { + + input, err := builtins.StringOperand(a, 1) + if err != nil { + return nil, err + } + + // data to be passed to x509.ParseCertificateRequest + bytes := []byte(input) + + // if the input is not a PEM string, attempt to decode b64 + if str := string(input); !strings.HasPrefix(str, "-----BEGIN CERTIFICATE REQUEST-----") { + bytes, err = base64.StdEncoding.DecodeString(str) + if err != nil { + return nil, err + } + } + + p, _ := pem.Decode(bytes) + if p != nil && p.Type != blockTypeCertificateRequest { + return nil, fmt.Errorf("invalid PEM-encoded certificate signing request") + } + if p != nil { + bytes = p.Bytes + } + + csr, err := x509.ParseCertificateRequest(bytes) + if err != nil { + return nil, err + } + + bs, err := json.Marshal(csr) + if err != nil { + return nil, err + } + + var x interface{} + if err := util.UnmarshalJSON(bs, &x); err != nil { + return nil, err + } + return ast.InterfaceToValue(x) +} + +func builtinCryptoX509ParseRSAPrivateKey(_ BuiltinContext, args []*ast.Term, iter func(*ast.Term) error) error { + + a := args[0].Value + input, err := builtins.StringOperand(a, 1) + if err != nil { + return err + } + + // get the raw private key + rawKey, err := getRSAPrivateKeyFromString(string(input)) + if err != nil { + return err + } + + rsaPrivateKey, err := jwk.New(rawKey) + if err != nil { + return err + } + + jsonKey, err := json.Marshal(rsaPrivateKey) + if err != nil { + return err + } + + var x interface{} + if err := util.UnmarshalJSON(jsonKey, &x); err != nil { + return err + } + + value, err := ast.InterfaceToValue(x) + if err != nil { + return err + } + + return iter(ast.NewTerm(value)) +} + +func hashHelper(a ast.Value, h func(ast.String) string) (ast.Value, error) { + s, err := builtins.StringOperand(a, 1) + if err != nil { + return nil, err + } + return ast.String(h(s)), nil +} + +func builtinCryptoMd5(a ast.Value) (ast.Value, error) { + return hashHelper(a, func(s ast.String) string { return fmt.Sprintf("%x", md5.Sum([]byte(s))) }) +} + +func builtinCryptoSha1(a ast.Value) (ast.Value, error) { + return hashHelper(a, func(s ast.String) string { return fmt.Sprintf("%x", sha1.Sum([]byte(s))) }) +} + +func builtinCryptoSha256(a ast.Value) (ast.Value, error) { + return hashHelper(a, func(s ast.String) string { return fmt.Sprintf("%x", sha256.Sum256([]byte(s))) }) +} + +func hmacHelper(args []*ast.Term, iter func(*ast.Term) error, h func() hash.Hash) error { + a1 := args[0].Value + message, err := builtins.StringOperand(a1, 1) + if err != nil { + return err + } + + a2 := args[1].Value + key, err := builtins.StringOperand(a2, 2) + if err != nil { + return err + } + + mac := hmac.New(h, []byte(key)) + mac.Write([]byte(message)) + messageDigest := mac.Sum(nil) + + return iter(ast.StringTerm(fmt.Sprintf("%x", messageDigest))) +} + +func builtinCryptoHmacMd5(_ BuiltinContext, args []*ast.Term, iter func(*ast.Term) error) error { + return hmacHelper(args, iter, md5.New) +} + +func builtinCryptoHmacSha1(_ BuiltinContext, args []*ast.Term, iter func(*ast.Term) error) error { + return hmacHelper(args, iter, sha1.New) +} + +func builtinCryptoHmacSha256(_ BuiltinContext, args []*ast.Term, iter func(*ast.Term) error) error { + return hmacHelper(args, iter, sha256.New) +} + +func builtinCryptoHmacSha512(_ BuiltinContext, args []*ast.Term, iter func(*ast.Term) error) error { + return hmacHelper(args, iter, sha512.New) +} + +func init() { + RegisterFunctionalBuiltin1(ast.CryptoX509ParseCertificates.Name, builtinCryptoX509ParseCertificates) + RegisterBuiltinFunc(ast.CryptoX509ParseAndVerifyCertificates.Name, builtinCryptoX509ParseAndVerifyCertificates) + RegisterFunctionalBuiltin1(ast.CryptoMd5.Name, builtinCryptoMd5) + RegisterFunctionalBuiltin1(ast.CryptoSha1.Name, builtinCryptoSha1) + RegisterFunctionalBuiltin1(ast.CryptoSha256.Name, builtinCryptoSha256) + RegisterFunctionalBuiltin1(ast.CryptoX509ParseCertificateRequest.Name, builtinCryptoX509ParseCertificateRequest) + RegisterBuiltinFunc(ast.CryptoX509ParseRSAPrivateKey.Name, builtinCryptoX509ParseRSAPrivateKey) + RegisterBuiltinFunc(ast.CryptoHmacMd5.Name, builtinCryptoHmacMd5) + RegisterBuiltinFunc(ast.CryptoHmacSha1.Name, builtinCryptoHmacSha1) + RegisterBuiltinFunc(ast.CryptoHmacSha256.Name, builtinCryptoHmacSha256) + RegisterBuiltinFunc(ast.CryptoHmacSha512.Name, builtinCryptoHmacSha512) +} + +func verifyX509CertificateChain(certs []*x509.Certificate) ([]*x509.Certificate, error) { + if len(certs) < 2 { + return nil, builtins.NewOperandErr(1, "must supply at least two certificates to be able to verify") + } + + // first cert is the root + roots := x509.NewCertPool() + roots.AddCert(certs[0]) + + // all other certs except the last are intermediates + intermediates := x509.NewCertPool() + for i := 1; i < len(certs)-1; i++ { + intermediates.AddCert(certs[i]) + } + + // last cert is the leaf + leaf := certs[len(certs)-1] + + // verify the cert chain back to the root + verifyOpts := x509.VerifyOptions{ + Roots: roots, + Intermediates: intermediates, + } + chains, err := leaf.Verify(verifyOpts) + if err != nil { + return nil, err + } + + return chains[0], nil +} + +func getX509CertsFromString(certs string) ([]*x509.Certificate, error) { + // if the input is PEM handle that + if strings.HasPrefix(certs, "-----BEGIN") { + return getX509CertsFromPem([]byte(certs)) + } + + // assume input is base64 if not PEM + b64, err := base64.StdEncoding.DecodeString(certs) + if err != nil { + return nil, err + } + + // handle if the decoded base64 contains PEM rather than the expected DER + if bytes.HasPrefix(b64, []byte("-----BEGIN")) { + return getX509CertsFromPem(b64) + } + + // otherwise assume the contents are DER + return x509.ParseCertificates(b64) +} + +func getX509CertsFromPem(pemBlocks []byte) ([]*x509.Certificate, error) { + var decodedCerts []byte + for len(pemBlocks) > 0 { + p, r := pem.Decode(pemBlocks) + if p != nil && p.Type != blockTypeCertificate { + return nil, fmt.Errorf("PEM block type is '%s', expected %s", p.Type, blockTypeCertificate) + } + + if p == nil { + break + } + + pemBlocks = r + decodedCerts = append(decodedCerts, p.Bytes...) + } + + return x509.ParseCertificates(decodedCerts) +} + +func getRSAPrivateKeyFromString(key string) (interface{}, error) { + // if the input is PEM handle that + if strings.HasPrefix(key, "-----BEGIN") { + return getRSAPrivateKeyFromPEM([]byte(key)) + } + + // assume input is base64 if not PEM + b64, err := base64.StdEncoding.DecodeString(key) + if err != nil { + return nil, err + } + + return getRSAPrivateKeyFromPEM(b64) +} + +func getRSAPrivateKeyFromPEM(pemBlocks []byte) (interface{}, error) { + + // decode the pem into the Block struct + p, _ := pem.Decode(pemBlocks) + if p == nil { + return nil, fmt.Errorf("failed to parse PEM block containing the key") + } + + // if the key is in PKCS1 format + if p.Type == blockTypeRSAPrivateKey { + return x509.ParsePKCS1PrivateKey(p.Bytes) + } + + // if the key is in PKCS8 format + if p.Type == blockTypePrivateKey { + return x509.ParsePKCS8PrivateKey(p.Bytes) + } + + // unsupported key format + return nil, fmt.Errorf("PEM block type is '%s', expected %s or %s", p.Type, blockTypeRSAPrivateKey, + blockTypePrivateKey) + +} + +// addCACertsFromFile adds CA certificates from filePath into the given pool. +// If pool is nil, it creates a new x509.CertPool. pool is returned. +func addCACertsFromFile(pool *x509.CertPool, filePath string) (*x509.CertPool, error) { + if pool == nil { + pool = x509.NewCertPool() + } + + caCert, err := readCertFromFile(filePath) + if err != nil { + return nil, err + } + + if ok := pool.AppendCertsFromPEM(caCert); !ok { + return nil, fmt.Errorf("could not append CA certificates from %q", filePath) + } + + return pool, nil +} + +// addCACertsFromBytes adds CA certificates from pemBytes into the given pool. +// If pool is nil, it creates a new x509.CertPool. pool is returned. +func addCACertsFromBytes(pool *x509.CertPool, pemBytes []byte) (*x509.CertPool, error) { + if pool == nil { + pool = x509.NewCertPool() + } + + if ok := pool.AppendCertsFromPEM(pemBytes); !ok { + return nil, fmt.Errorf("could not append certificates") + } + + return pool, nil +} + +// addCACertsFromBytes adds CA certificates from the environment variable named +// by envName into the given pool. If pool is nil, it creates a new x509.CertPool. +// pool is returned. +func addCACertsFromEnv(pool *x509.CertPool, envName string) (*x509.CertPool, error) { + pool, err := addCACertsFromBytes(pool, []byte(os.Getenv(envName))) + if err != nil { + return nil, fmt.Errorf("could not add CA certificates from envvar %q: %w", envName, err) + } + + return pool, err +} + +// ReadCertFromFile reads a cert from file +func readCertFromFile(localCertFile string) ([]byte, error) { + // Read in the cert file + certPEM, err := ioutil.ReadFile(localCertFile) + if err != nil { + return nil, err + } + return certPEM, nil +} + +// ReadKeyFromFile reads a key from file +func readKeyFromFile(localKeyFile string) ([]byte, error) { + // Read in the cert file + key, err := ioutil.ReadFile(localKeyFile) + if err != nil { + return nil, err + } + return key, nil +} diff --git a/vendor/github.com/open-policy-agent/opa/topdown/doc.go b/vendor/github.com/open-policy-agent/opa/topdown/doc.go new file mode 100644 index 00000000..9aa7aa45 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/topdown/doc.go @@ -0,0 +1,10 @@ +// Copyright 2016 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +// Package topdown provides low-level query evaluation support. +// +// The topdown implementation is a modified version of the standard top-down +// evaluation algorithm used in Datalog. References and comprehensions are +// evaluated eagerly while all other terms are evaluated lazily. +package topdown diff --git a/vendor/github.com/open-policy-agent/opa/topdown/encoding.go b/vendor/github.com/open-policy-agent/opa/topdown/encoding.go new file mode 100644 index 00000000..fa1cfc88 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/topdown/encoding.go @@ -0,0 +1,309 @@ +// Copyright 2017 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package topdown + +import ( + "bytes" + "encoding/base64" + "encoding/hex" + "encoding/json" + "fmt" + "net/url" + "strings" + + ghodss "github.com/ghodss/yaml" + + "github.com/open-policy-agent/opa/ast" + "github.com/open-policy-agent/opa/topdown/builtins" + "github.com/open-policy-agent/opa/util" +) + +func builtinJSONMarshal(a ast.Value) (ast.Value, error) { + + asJSON, err := ast.JSON(a) + if err != nil { + return nil, err + } + + bs, err := json.Marshal(asJSON) + if err != nil { + return nil, err + } + + return ast.String(string(bs)), nil +} + +func builtinJSONUnmarshal(a ast.Value) (ast.Value, error) { + + str, err := builtins.StringOperand(a, 1) + if err != nil { + return nil, err + } + + var x interface{} + + if err := util.UnmarshalJSON([]byte(str), &x); err != nil { + return nil, err + } + + return ast.InterfaceToValue(x) +} + +func builtinJSONIsValid(a ast.Value) (ast.Value, error) { + + str, err := builtins.StringOperand(a, 1) + if err != nil { + return nil, err + } + + var x interface{} + err = util.UnmarshalJSON([]byte(str), &x) + return ast.Boolean(err == nil), nil +} + +func builtinBase64Encode(a ast.Value) (ast.Value, error) { + str, err := builtins.StringOperand(a, 1) + if err != nil { + return nil, err + } + + return ast.String(base64.StdEncoding.EncodeToString([]byte(str))), nil +} + +func builtinBase64Decode(a ast.Value) (ast.Value, error) { + str, err := builtins.StringOperand(a, 1) + if err != nil { + return nil, err + } + + result, err := base64.StdEncoding.DecodeString(string(str)) + return ast.String(result), err +} + +func builtinBase64IsValid(a ast.Value) (ast.Value, error) { + str, err := builtins.StringOperand(a, 1) + if err != nil { + return nil, err + } + + _, err = base64.StdEncoding.DecodeString(string(str)) + return ast.Boolean(err == nil), nil +} + +func builtinBase64UrlEncode(a ast.Value) (ast.Value, error) { + str, err := builtins.StringOperand(a, 1) + if err != nil { + return nil, err + } + + return ast.String(base64.URLEncoding.EncodeToString([]byte(str))), nil +} + +func builtinBase64UrlEncodeNoPad(a ast.Value) (ast.Value, error) { + str, err := builtins.StringOperand(a, 1) + if err != nil { + return nil, err + } + return ast.String(base64.RawURLEncoding.EncodeToString([]byte(str))), nil +} + +func builtinBase64UrlDecode(a ast.Value) (ast.Value, error) { + str, err := builtins.StringOperand(a, 1) + if err != nil { + return nil, err + } + s := string(str) + + // Some base64url encoders omit the padding at the end, so this case + // corrects such representations using the method given in RFC 7515 + // Appendix C: https://tools.ietf.org/html/rfc7515#appendix-C + if !strings.HasSuffix(s, "=") { + switch len(s) % 4 { + case 0: + case 2: + s += "==" + case 3: + s += "=" + default: + return nil, fmt.Errorf("illegal base64url string: %s", s) + } + } + result, err := base64.URLEncoding.DecodeString(s) + return ast.String(result), err +} + +func builtinURLQueryEncode(a ast.Value) (ast.Value, error) { + str, err := builtins.StringOperand(a, 1) + if err != nil { + return nil, err + } + return ast.String(url.QueryEscape(string(str))), nil +} + +func builtinURLQueryDecode(a ast.Value) (ast.Value, error) { + str, err := builtins.StringOperand(a, 1) + if err != nil { + return nil, err + } + s, err := url.QueryUnescape(string(str)) + if err != nil { + return nil, err + } + return ast.String(s), nil +} + +var encodeObjectErr = builtins.NewOperandErr(1, "values must be string, array[string], or set[string]") + +func builtinURLQueryEncodeObject(a ast.Value) (ast.Value, error) { + asJSON, err := ast.JSON(a) + if err != nil { + return nil, err + } + + inputs, ok := asJSON.(map[string]interface{}) + if !ok { + return nil, builtins.NewOperandTypeErr(1, a, "object") + } + + query := url.Values{} + + for k, v := range inputs { + switch vv := v.(type) { + case string: + query.Set(k, vv) + case []interface{}: + for _, val := range vv { + strVal, ok := val.(string) + if !ok { + return nil, encodeObjectErr + } + query.Add(k, strVal) + } + default: + return nil, encodeObjectErr + } + } + + return ast.String(query.Encode()), nil +} + +func builtinURLQueryDecodeObject(bctx BuiltinContext, operands []*ast.Term, iter func(*ast.Term) error) error { + query, err := builtins.StringOperand(operands[0].Value, 1) + if err != nil { + return err + } + + queryParams, err := url.ParseQuery(string(query)) + if err != nil { + return err + } + + queryObject := ast.NewObject() + for k, v := range queryParams { + paramsArray := make([]*ast.Term, len(v)) + for i, param := range v { + paramsArray[i] = ast.StringTerm(param) + } + queryObject.Insert(ast.StringTerm(k), ast.ArrayTerm(paramsArray...)) + } + + return iter(ast.NewTerm(queryObject)) +} + +func builtinYAMLMarshal(a ast.Value) (ast.Value, error) { + + asJSON, err := ast.JSON(a) + if err != nil { + return nil, err + } + + var buf bytes.Buffer + encoder := json.NewEncoder(&buf) + if err := encoder.Encode(asJSON); err != nil { + return nil, err + } + + bs, err := ghodss.JSONToYAML(buf.Bytes()) + if err != nil { + return nil, err + } + + return ast.String(string(bs)), nil +} + +func builtinYAMLUnmarshal(a ast.Value) (ast.Value, error) { + + str, err := builtins.StringOperand(a, 1) + if err != nil { + return nil, err + } + + bs, err := ghodss.YAMLToJSON([]byte(str)) + if err != nil { + return nil, err + } + + buf := bytes.NewBuffer(bs) + decoder := util.NewJSONDecoder(buf) + var val interface{} + err = decoder.Decode(&val) + if err != nil { + return nil, err + } + + return ast.InterfaceToValue(val) +} + +func builtinYAMLIsValid(a ast.Value) (ast.Value, error) { + str, err := builtins.StringOperand(a, 1) + if err != nil { + return nil, err + } + + var x interface{} + err = ghodss.Unmarshal([]byte(str), &x) + return ast.Boolean(err == nil), nil +} + +func builtinHexEncode(a ast.Value) (ast.Value, error) { + str, err := builtins.StringOperand(a, 1) + if err != nil { + return nil, err + } + return ast.String(hex.EncodeToString([]byte(str))), nil +} + +func builtinHexDecode(a ast.Value) (ast.Value, error) { + str, err := builtins.StringOperand(a, 1) + if err != nil { + return nil, err + } + val, err := hex.DecodeString(string(str)) + if err != nil { + return nil, err + } + return ast.String(val), nil +} + +func init() { + RegisterFunctionalBuiltin1(ast.JSONMarshal.Name, builtinJSONMarshal) + RegisterFunctionalBuiltin1(ast.JSONUnmarshal.Name, builtinJSONUnmarshal) + RegisterFunctionalBuiltin1(ast.JSONIsValid.Name, builtinJSONIsValid) + RegisterFunctionalBuiltin1(ast.Base64Encode.Name, builtinBase64Encode) + RegisterFunctionalBuiltin1(ast.Base64Decode.Name, builtinBase64Decode) + RegisterFunctionalBuiltin1(ast.Base64IsValid.Name, builtinBase64IsValid) + RegisterFunctionalBuiltin1(ast.Base64UrlEncode.Name, builtinBase64UrlEncode) + RegisterFunctionalBuiltin1(ast.Base64UrlEncodeNoPad.Name, builtinBase64UrlEncodeNoPad) + RegisterFunctionalBuiltin1(ast.Base64UrlDecode.Name, builtinBase64UrlDecode) + RegisterFunctionalBuiltin1(ast.URLQueryDecode.Name, builtinURLQueryDecode) + RegisterFunctionalBuiltin1(ast.URLQueryEncode.Name, builtinURLQueryEncode) + RegisterFunctionalBuiltin1(ast.URLQueryEncodeObject.Name, builtinURLQueryEncodeObject) + RegisterBuiltinFunc(ast.URLQueryDecodeObject.Name, builtinURLQueryDecodeObject) + RegisterFunctionalBuiltin1(ast.YAMLMarshal.Name, builtinYAMLMarshal) + RegisterFunctionalBuiltin1(ast.YAMLUnmarshal.Name, builtinYAMLUnmarshal) + RegisterFunctionalBuiltin1(ast.YAMLIsValid.Name, builtinYAMLIsValid) + RegisterFunctionalBuiltin1(ast.HexEncode.Name, builtinHexEncode) + RegisterFunctionalBuiltin1(ast.HexDecode.Name, builtinHexDecode) +} diff --git a/vendor/github.com/open-policy-agent/opa/topdown/errors.go b/vendor/github.com/open-policy-agent/opa/topdown/errors.go new file mode 100644 index 00000000..f2459d2a --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/topdown/errors.go @@ -0,0 +1,139 @@ +// Copyright 2017 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package topdown + +import ( + "errors" + "fmt" + + "github.com/open-policy-agent/opa/ast" +) + +// Halt is a special error type that built-in function implementations return to indicate +// that policy evaluation should stop immediately. +type Halt struct { + Err error +} + +func (h Halt) Error() string { + return h.Err.Error() +} + +func (h Halt) Unwrap() error { return h.Err } + +// Error is the error type returned by the Eval and Query functions when +// an evaluation error occurs. +type Error struct { + Code string `json:"code"` + Message string `json:"message"` + Location *ast.Location `json:"location,omitempty"` +} + +const ( + + // InternalErr represents an unknown evaluation error. + InternalErr string = "eval_internal_error" + + // CancelErr indicates the evaluation process was cancelled. + CancelErr string = "eval_cancel_error" + + // ConflictErr indicates a conflict was encountered during evaluation. For + // instance, a conflict occurs if a rule produces multiple, differing values + // for the same key in an object. Conflict errors indicate the policy does + // not account for the data loaded into the policy engine. + ConflictErr string = "eval_conflict_error" + + // TypeErr indicates evaluation stopped because an expression was applied to + // a value of an inappropriate type. + TypeErr string = "eval_type_error" + + // BuiltinErr indicates a built-in function received a semantically invalid + // input or encountered some kind of runtime error, e.g., connection + // timeout, connection refused, etc. + BuiltinErr string = "eval_builtin_error" + + // WithMergeErr indicates that the real and replacement data could not be merged. + WithMergeErr string = "eval_with_merge_error" +) + +// IsError returns true if the err is an Error. +func IsError(err error) bool { + var e *Error + return errors.As(err, &e) +} + +// IsCancel returns true if err was caused by cancellation. +func IsCancel(err error) bool { + return errors.Is(err, &Error{Code: CancelErr}) +} + +// Is allows matching topdown errors using errors.Is (see IsCancel). +func (e *Error) Is(target error) bool { + var t *Error + if errors.As(target, &t) { + return (t.Code == "" || e.Code == t.Code) && + (t.Message == "" || e.Message == t.Message) && + (t.Location == nil || t.Location.Compare(e.Location) == 0) + } + return false +} + +func (e *Error) Error() string { + msg := fmt.Sprintf("%v: %v", e.Code, e.Message) + + if e.Location != nil { + msg = e.Location.String() + ": " + msg + } + + return msg +} + +func functionConflictErr(loc *ast.Location) error { + return &Error{ + Code: ConflictErr, + Location: loc, + Message: "functions must not produce multiple outputs for same inputs", + } +} + +func completeDocConflictErr(loc *ast.Location) error { + return &Error{ + Code: ConflictErr, + Location: loc, + Message: "complete rules must not produce multiple outputs", + } +} + +func objectDocKeyConflictErr(loc *ast.Location) error { + return &Error{ + Code: ConflictErr, + Location: loc, + Message: "object keys must be unique", + } +} + +func unsupportedBuiltinErr(loc *ast.Location) error { + return &Error{ + Code: InternalErr, + Location: loc, + Message: "unsupported built-in", + } +} + +func mergeConflictErr(loc *ast.Location) error { + return &Error{ + Code: WithMergeErr, + Location: loc, + Message: "real and replacement data could not be merged", + } +} + +func internalErr(loc *ast.Location, msg string) error { + return &Error{ + Code: InternalErr, + Location: loc, + Message: msg, + } +} diff --git a/vendor/github.com/open-policy-agent/opa/topdown/eval.go b/vendor/github.com/open-policy-agent/opa/topdown/eval.go new file mode 100644 index 00000000..aacff970 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/topdown/eval.go @@ -0,0 +1,3163 @@ +package topdown + +import ( + "context" + "fmt" + "io" + "sort" + "strconv" + "strings" + + "github.com/open-policy-agent/opa/ast" + "github.com/open-policy-agent/opa/metrics" + "github.com/open-policy-agent/opa/storage" + "github.com/open-policy-agent/opa/topdown/builtins" + "github.com/open-policy-agent/opa/topdown/cache" + "github.com/open-policy-agent/opa/topdown/copypropagation" + "github.com/open-policy-agent/opa/topdown/print" + "github.com/open-policy-agent/opa/tracing" +) + +type evalIterator func(*eval) error + +type unifyIterator func() error + +type queryIDFactory struct { + curr uint64 +} + +// Note: The first call to Next() returns 0. +func (f *queryIDFactory) Next() uint64 { + curr := f.curr + f.curr++ + return curr +} + +type builtinErrors struct { + errs []error +} + +// earlyExitError is used to abort iteration where early exit is possible +type earlyExitError struct { + prev error + e *eval +} + +func (ee *earlyExitError) Error() string { + return fmt.Sprintf("%v: early exit", ee.e.query) +} + +type eval struct { + ctx context.Context + metrics metrics.Metrics + seed io.Reader + time *ast.Term + queryID uint64 + queryIDFact *queryIDFactory + parent *eval + caller *eval + cancel Cancel + query ast.Body + queryCompiler ast.QueryCompiler + index int + indexing bool + earlyExit bool + bindings *bindings + store storage.Store + baseCache *baseCache + txn storage.Transaction + compiler *ast.Compiler + input *ast.Term + data *ast.Term + external *resolverTrie + targetStack *refStack + tracers []QueryTracer + traceEnabled bool + plugTraceVars bool + instr *Instrumentation + builtins map[string]*Builtin + builtinCache builtins.Cache + virtualCache *virtualCache + comprehensionCache *comprehensionCache + interQueryBuiltinCache cache.InterQueryCache + saveSet *saveSet + saveStack *saveStack + saveSupport *saveSupport + saveNamespace *ast.Term + skipSaveNamespace bool + inliningControl *inliningControl + genvarprefix string + genvarid int + runtime *ast.Term + builtinErrors *builtinErrors + printHook print.Hook + tracingOpts tracing.Options + findOne bool +} + +func (e *eval) Run(iter evalIterator) error { + e.traceEnter(e.query) + return e.eval(func(e *eval) error { + e.traceExit(e.query) + err := iter(e) + e.traceRedo(e.query) + return err + }) +} + +func (e *eval) String() string { + s := strings.Builder{} + e.string(&s) + return s.String() +} + +func (e *eval) string(s *strings.Builder) { + fmt.Fprintf(s, "') +} + +func (e *eval) builtinFunc(name string) (*ast.Builtin, BuiltinFunc, bool) { + decl, ok := ast.BuiltinMap[name] + if ok { + f, ok := builtinFunctions[name] + if ok { + return decl, f, true + } + } else { + bi, ok := e.builtins[name] + if ok { + return bi.Decl, bi.Func, true + } + } + return nil, nil, false +} + +func (e *eval) closure(query ast.Body) *eval { + cpy := *e + cpy.index = 0 + cpy.query = query + cpy.queryID = cpy.queryIDFact.Next() + cpy.parent = e + cpy.findOne = false + return &cpy +} + +func (e *eval) child(query ast.Body) *eval { + cpy := *e + cpy.index = 0 + cpy.query = query + cpy.queryID = cpy.queryIDFact.Next() + cpy.bindings = newBindings(cpy.queryID, e.instr) + cpy.parent = e + cpy.findOne = false + return &cpy +} + +func (e *eval) next(iter evalIterator) error { + e.index++ + err := e.evalExpr(iter) + e.index-- + return err +} + +func (e *eval) partial() bool { + return e.saveSet != nil +} + +func (e *eval) unknown(x interface{}, b *bindings) bool { + if !e.partial() { + return false + } + + // If the caller provided an ast.Value directly (e.g., an ast.Ref) wrap + // it as an ast.Term because the saveSet Contains() function expects + // ast.Term. + if v, ok := x.(ast.Value); ok { + x = ast.NewTerm(v) + } + + return saveRequired(e.compiler, e.inliningControl, true, e.saveSet, b, x, false) +} + +func (e *eval) traceEnter(x ast.Node) { + e.traceEvent(EnterOp, x, "", nil) +} + +func (e *eval) traceExit(x ast.Node) { + var msg string + if e.findOne { + msg = "early" + } + e.traceEvent(ExitOp, x, msg, nil) +} + +func (e *eval) traceEval(x ast.Node) { + e.traceEvent(EvalOp, x, "", nil) +} + +func (e *eval) traceDuplicate(x ast.Node) { + e.traceEvent(DuplicateOp, x, "", nil) +} + +func (e *eval) traceFail(x ast.Node) { + e.traceEvent(FailOp, x, "", nil) +} + +func (e *eval) traceRedo(x ast.Node) { + e.traceEvent(RedoOp, x, "", nil) +} + +func (e *eval) traceSave(x ast.Node) { + e.traceEvent(SaveOp, x, "", nil) +} + +func (e *eval) traceIndex(x ast.Node, msg string, target *ast.Ref) { + e.traceEvent(IndexOp, x, msg, target) +} + +func (e *eval) traceWasm(x ast.Node, target *ast.Ref) { + e.traceEvent(WasmOp, x, "", target) +} + +func (e *eval) traceEvent(op Op, x ast.Node, msg string, target *ast.Ref) { + + if !e.traceEnabled { + return + } + + var parentID uint64 + if e.parent != nil { + parentID = e.parent.queryID + } + + evt := Event{ + QueryID: e.queryID, + ParentID: parentID, + Op: op, + Node: x, + Location: x.Loc(), + Message: msg, + Ref: target, + } + + // Skip plugging the local variables, unless any of the tracers + // had required it via their configuration. If any required the + // variable bindings then we will plug and give values for all + // tracers. + if e.plugTraceVars { + + evt.Locals = ast.NewValueMap() + evt.LocalMetadata = map[ast.Var]VarMetadata{} + + _ = e.bindings.Iter(nil, func(k, v *ast.Term) error { + original := k.Value.(ast.Var) + rewritten, _ := e.rewrittenVar(original) + evt.LocalMetadata[original] = VarMetadata{ + Name: rewritten, + Location: k.Loc(), + } + + // For backwards compatibility save a copy of the values too.. + evt.Locals.Put(k.Value, v.Value) + return nil + }) // cannot return error + + ast.WalkTerms(x, func(term *ast.Term) bool { + if v, ok := term.Value.(ast.Var); ok { + if _, ok := evt.LocalMetadata[v]; !ok { + if rewritten, ok := e.rewrittenVar(v); ok { + evt.LocalMetadata[v] = VarMetadata{ + Name: rewritten, + Location: term.Loc(), + } + } + } + } + return false + }) + } + + for i := range e.tracers { + e.tracers[i].TraceEvent(evt) + } +} + +func (e *eval) eval(iter evalIterator) error { + return e.evalExpr(iter) +} + +func (e *eval) evalExpr(iter evalIterator) error { + + if e.cancel != nil && e.cancel.Cancelled() { + return &Error{ + Code: CancelErr, + Message: "caller cancelled query execution", + } + } + + if e.index >= len(e.query) { + err := iter(e) + if err != nil { + ee, ok := err.(*earlyExitError) + if !ok { + return err + } + if !e.findOne { + return nil + } + return &earlyExitError{prev: ee, e: e} + } + if e.findOne && !e.partial() { // we've found one! + return &earlyExitError{e: e} + } + return nil + } + + expr := e.query[e.index] + + e.traceEval(expr) + + if len(expr.With) > 0 { + return e.evalWith(iter) + } + + return e.evalStep(func(e *eval) error { + return e.next(iter) + }) +} + +func (e *eval) evalStep(iter evalIterator) error { + + expr := e.query[e.index] + + if expr.Negated { + return e.evalNot(iter) + } + + var defined bool + var err error + + switch terms := expr.Terms.(type) { + case []*ast.Term: + if expr.IsEquality() { + err = e.unify(terms[1], terms[2], func() error { + defined = true + err := iter(e) + e.traceRedo(expr) + return err + }) + } else { + err = e.evalCall(terms, func() error { + defined = true + err := iter(e) + e.traceRedo(expr) + return err + }) + } + case *ast.Term: + rterm := e.generateVar(fmt.Sprintf("term_%d_%d", e.queryID, e.index)) + err = e.unify(terms, rterm, func() error { + if e.saveSet.Contains(rterm, e.bindings) { + return e.saveExpr(ast.NewExpr(rterm), e.bindings, func() error { + return iter(e) + }) + } + if !e.bindings.Plug(rterm).Equal(ast.BooleanTerm(false)) { + defined = true + err := iter(e) + e.traceRedo(expr) + return err + } + return nil + }) + } + + if err != nil { + return err + } + + if !defined { + e.traceFail(expr) + } + + return nil +} + +func (e *eval) evalNot(iter evalIterator) error { + + expr := e.query[e.index] + + if e.unknown(expr, e.bindings) { + return e.evalNotPartial(iter) + } + + negation := ast.NewBody(expr.Complement().NoWith()) + child := e.closure(negation) + + var defined bool + child.traceEnter(negation) + + err := child.eval(func(*eval) error { + child.traceExit(negation) + defined = true + child.traceRedo(negation) + return nil + }) + + if err != nil { + return err + } + + if !defined { + return iter(e) + } + + e.traceFail(expr) + return nil +} + +func (e *eval) evalWith(iter evalIterator) error { + + expr := e.query[e.index] + var disable []ast.Ref + + if e.partial() { + + // If the value is unknown the with statement cannot be evaluated and so + // the entire expression should be saved to be safe. In the future this + // could be relaxed in certain cases (e.g., if the with statement would + // have no affect.) + for _, with := range expr.With { + if e.saveSet.ContainsRecursive(with.Value, e.bindings) { + return e.saveExprMarkUnknowns(expr, e.bindings, func() error { + return e.next(iter) + }) + } + } + + // Disable inlining on all references in the expression so the result of + // partial evaluation has the same semantics w/ the with statements + // preserved. + ast.WalkRefs(expr, func(x ast.Ref) bool { + disable = append(disable, x.GroundPrefix()) + return false + }) + } + + pairsInput := [][2]*ast.Term{} + pairsData := [][2]*ast.Term{} + targets := []ast.Ref{} + + for i := range expr.With { + plugged := e.bindings.Plug(expr.With[i].Value) + if isInputRef(expr.With[i].Target) { + pairsInput = append(pairsInput, [...]*ast.Term{expr.With[i].Target, plugged}) + } else if isDataRef(expr.With[i].Target) { + pairsData = append(pairsData, [...]*ast.Term{expr.With[i].Target, plugged}) + } + targets = append(targets, expr.With[i].Target.Value.(ast.Ref)) + } + + input, err := mergeTermWithValues(e.input, pairsInput) + if err != nil { + return &Error{ + Code: ConflictErr, + Location: expr.Location, + Message: err.Error(), + } + } + + data, err := mergeTermWithValues(e.data, pairsData) + if err != nil { + return &Error{ + Code: ConflictErr, + Location: expr.Location, + Message: err.Error(), + } + } + + oldInput, oldData := e.evalWithPush(input, data, targets, disable) + + err = e.evalStep(func(e *eval) error { + e.evalWithPop(oldInput, oldData) + err := e.next(iter) + oldInput, oldData = e.evalWithPush(input, data, targets, disable) + return err + }) + + e.evalWithPop(oldInput, oldData) + + return err +} + +func (e *eval) evalWithPush(input, data *ast.Term, targets, disable []ast.Ref) (*ast.Term, *ast.Term) { + var oldInput *ast.Term + + if input != nil { + oldInput = e.input + e.input = input + } + + var oldData *ast.Term + + if data != nil { + oldData = e.data + e.data = data + } + + e.comprehensionCache.Push() + e.virtualCache.Push() + e.targetStack.Push(targets) + e.inliningControl.PushDisable(disable, true) + + return oldInput, oldData +} + +func (e *eval) evalWithPop(input, data *ast.Term) { + e.inliningControl.PopDisable() + e.targetStack.Pop() + e.virtualCache.Pop() + e.comprehensionCache.Pop() + e.data = data + e.input = input +} + +func (e *eval) evalNotPartial(iter evalIterator) error { + + // Prepare query normally. + expr := e.query[e.index] + negation := expr.Complement().NoWith() + child := e.closure(ast.NewBody(negation)) + + // Unknowns is the set of variables that are marked as unknown. The variables + // are namespaced with the query ID that they originate in. This ensures that + // variables across two or more queries are identified uniquely. + // + // NOTE(tsandall): this is greedy in the sense that we only need variable + // dependencies of the negation. + unknowns := e.saveSet.Vars(e.caller.bindings) + + // Run partial evaluation. Since the result may require support, push a new + // query onto the save stack to avoid mutating the current save query. If + // shallow inlining is not enabled, run copy propagation to further simplify + // the result. + var cp *copypropagation.CopyPropagator + + if !e.inliningControl.shallow { + cp = copypropagation.New(unknowns).WithEnsureNonEmptyBody(true).WithCompiler(e.compiler) + } + + var savedQueries []ast.Body + e.saveStack.PushQuery(nil) + + _ = child.eval(func(*eval) error { + query := e.saveStack.Peek() + plugged := query.Plug(e.caller.bindings) + // Skip this rule body if it fails to type-check. + // Type-checking failure means the rule body will never succeed. + if !e.compiler.PassesTypeCheck(plugged) { + return nil + } + if cp != nil { + plugged = applyCopyPropagation(cp, e.instr, plugged) + } + savedQueries = append(savedQueries, plugged) + return nil + }) // cannot return error + + e.saveStack.PopQuery() + + // If partial evaluation produced no results, the expression is always undefined + // so it does not have to be saved. + if len(savedQueries) == 0 { + return iter(e) + } + + // Check if the partial evaluation result can be inlined in this query. If not, + // generate support rules for the result. Depending on the size of the partial + // evaluation result and the contents, it may or may not be inlinable. We treat + // the unknowns as safe because vars in the save set will either be known to + // the caller or made safe by an expression on the save stack. + if !canInlineNegation(unknowns, savedQueries) { + return e.evalNotPartialSupport(child.queryID, expr, unknowns, savedQueries, iter) + } + + // If we can inline the result, we have to generate the cross product of the + // queries. For example: + // + // (A && B) || (C && D) + // + // Becomes: + // + // (!A && !C) || (!A && !D) || (!B && !C) || (!B && !D) + return complementedCartesianProduct(savedQueries, 0, nil, func(q ast.Body) error { + return e.saveInlinedNegatedExprs(q, func() error { + return iter(e) + }) + }) +} + +func (e *eval) evalNotPartialSupport(negationID uint64, expr *ast.Expr, unknowns ast.VarSet, queries []ast.Body, iter evalIterator) error { + + // Prepare support rule head. + supportName := fmt.Sprintf("__not%d_%d_%d__", e.queryID, e.index, negationID) + term := ast.RefTerm(ast.DefaultRootDocument, e.saveNamespace, ast.StringTerm(supportName)) + path := term.Value.(ast.Ref) + head := ast.NewHead(ast.Var(supportName), nil, ast.BooleanTerm(true)) + + bodyVars := ast.NewVarSet() + + for _, q := range queries { + bodyVars.Update(q.Vars(ast.VarVisitorParams{})) + } + + unknowns = unknowns.Intersect(bodyVars) + + // Make rule args. Sort them to ensure order is deterministic. + args := make([]*ast.Term, 0, len(unknowns)) + + for v := range unknowns { + args = append(args, ast.NewTerm(v)) + } + + sort.Slice(args, func(i, j int) bool { + return args[i].Value.Compare(args[j].Value) < 0 + }) + + if len(args) > 0 { + head.Args = args + } + + // Save support rules. + for _, query := range queries { + e.saveSupport.Insert(path, &ast.Rule{ + Head: head, + Body: query, + }) + } + + // Save expression that refers to support rule set. + + terms := expr.Terms + expr.Terms = nil // Prevent unnecessary copying the terms. + cpy := expr.Copy() + expr.Terms = terms + + if len(args) > 0 { + terms := make([]*ast.Term, len(args)+1) + terms[0] = term + for i := 0; i < len(args); i++ { + terms[i+1] = args[i] + } + cpy.Terms = terms + } else { + cpy.Terms = term + } + + return e.saveInlinedNegatedExprs([]*ast.Expr{cpy}, func() error { + return e.next(iter) + }) +} + +func (e *eval) evalCall(terms []*ast.Term, iter unifyIterator) error { + + ref := terms[0].Value.(ast.Ref) + + if ref[0].Equal(ast.DefaultRootDocument) { + var ir *ast.IndexResult + var err error + if e.partial() { + ir, err = e.getRules(ref, nil) + } else { + ir, err = e.getRules(ref, terms[1:]) + } + if err != nil { + return err + } + eval := evalFunc{ + e: e, + ref: ref, + terms: terms, + ir: ir, + } + return eval.eval(iter) + } + + bi, f, ok := e.builtinFunc(ref.String()) + if !ok { + return unsupportedBuiltinErr(e.query[e.index].Location) + } + + if e.unknown(e.query[e.index], e.bindings) { + return e.saveCall(len(bi.Decl.Args()), terms, iter) + } + + var parentID uint64 + if e.parent != nil { + parentID = e.parent.queryID + } + + var capabilities *ast.Capabilities + if e.compiler != nil { + capabilities = e.compiler.Capabilities() + } + + bctx := BuiltinContext{ + Context: e.ctx, + Metrics: e.metrics, + Seed: e.seed, + Time: e.time, + Cancel: e.cancel, + Runtime: e.runtime, + Cache: e.builtinCache, + InterQueryBuiltinCache: e.interQueryBuiltinCache, + Location: e.query[e.index].Location, + QueryTracers: e.tracers, + TraceEnabled: e.traceEnabled, + QueryID: e.queryID, + ParentID: parentID, + PrintHook: e.printHook, + DistributedTracingOpts: e.tracingOpts, + Capabilities: capabilities, + } + + eval := evalBuiltin{ + e: e, + bi: bi, + bctx: bctx, + f: f, + terms: terms[1:], + } + return eval.eval(iter) +} + +func (e *eval) unify(a, b *ast.Term, iter unifyIterator) error { + return e.biunify(a, b, e.bindings, e.bindings, iter) +} + +func (e *eval) biunify(a, b *ast.Term, b1, b2 *bindings, iter unifyIterator) error { + a, b1 = b1.apply(a) + b, b2 = b2.apply(b) + switch vA := a.Value.(type) { + case ast.Var, ast.Ref, *ast.ArrayComprehension, *ast.SetComprehension, *ast.ObjectComprehension: + return e.biunifyValues(a, b, b1, b2, iter) + case ast.Null: + switch b.Value.(type) { + case ast.Var, ast.Null, ast.Ref: + return e.biunifyValues(a, b, b1, b2, iter) + } + case ast.Boolean: + switch b.Value.(type) { + case ast.Var, ast.Boolean, ast.Ref: + return e.biunifyValues(a, b, b1, b2, iter) + } + case ast.Number: + switch b.Value.(type) { + case ast.Var, ast.Number, ast.Ref: + return e.biunifyValues(a, b, b1, b2, iter) + } + case ast.String: + switch b.Value.(type) { + case ast.Var, ast.String, ast.Ref: + return e.biunifyValues(a, b, b1, b2, iter) + } + case *ast.Array: + switch vB := b.Value.(type) { + case ast.Var, ast.Ref, *ast.ArrayComprehension: + return e.biunifyValues(a, b, b1, b2, iter) + case *ast.Array: + return e.biunifyArrays(vA, vB, b1, b2, iter) + } + case ast.Object: + switch vB := b.Value.(type) { + case ast.Var, ast.Ref, *ast.ObjectComprehension: + return e.biunifyValues(a, b, b1, b2, iter) + case ast.Object: + return e.biunifyObjects(vA, vB, b1, b2, iter) + } + case ast.Set: + return e.biunifyValues(a, b, b1, b2, iter) + } + return nil +} + +func (e *eval) biunifyArrays(a, b *ast.Array, b1, b2 *bindings, iter unifyIterator) error { + if a.Len() != b.Len() { + return nil + } + return e.biunifyArraysRec(a, b, b1, b2, iter, 0) +} + +func (e *eval) biunifyArraysRec(a, b *ast.Array, b1, b2 *bindings, iter unifyIterator, idx int) error { + if idx == a.Len() { + return iter() + } + return e.biunify(a.Elem(idx), b.Elem(idx), b1, b2, func() error { + return e.biunifyArraysRec(a, b, b1, b2, iter, idx+1) + }) +} + +func (e *eval) biunifyObjects(a, b ast.Object, b1, b2 *bindings, iter unifyIterator) error { + if a.Len() != b.Len() { + return nil + } + + // Objects must not contain unbound variables as keys at this point as we + // cannot unify them. Similar to sets, plug both sides before comparing the + // keys and unifying the values. + if nonGroundKeys(a) { + a = plugKeys(a, b1) + } + + if nonGroundKeys(b) { + b = plugKeys(b, b2) + } + + return e.biunifyObjectsRec(a, b, b1, b2, iter, a, 0) +} + +func (e *eval) biunifyObjectsRec(a, b ast.Object, b1, b2 *bindings, iter unifyIterator, keys ast.Object, idx int) error { + if idx == keys.Len() { + return iter() + } + key, _ := keys.Elem(idx) + v2 := b.Get(key) + if v2 == nil { + return nil + } + return e.biunify(a.Get(key), v2, b1, b2, func() error { + return e.biunifyObjectsRec(a, b, b1, b2, iter, keys, idx+1) + }) +} + +func (e *eval) biunifyValues(a, b *ast.Term, b1, b2 *bindings, iter unifyIterator) error { + // Try to evaluate refs and comprehensions. If partial evaluation is + // enabled, then skip evaluation (and save the expression) if the term is + // in the save set. Currently, comprehensions are not evaluated during + // partial eval. This could be improved in the future. + + var saveA, saveB bool + + if _, ok := a.Value.(ast.Set); ok { + saveA = e.saveSet.ContainsRecursive(a, b1) + } else { + saveA = e.saveSet.Contains(a, b1) + if !saveA { + if _, refA := a.Value.(ast.Ref); refA { + return e.biunifyRef(a, b, b1, b2, iter) + } + } + } + + if _, ok := b.Value.(ast.Set); ok { + saveB = e.saveSet.ContainsRecursive(b, b2) + } else { + saveB = e.saveSet.Contains(b, b2) + if !saveB { + if _, refB := b.Value.(ast.Ref); refB { + return e.biunifyRef(b, a, b2, b1, iter) + } + } + } + + if saveA || saveB { + return e.saveUnify(a, b, b1, b2, iter) + } + + if ast.IsComprehension(a.Value) { + return e.biunifyComprehension(a, b, b1, b2, false, iter) + } else if ast.IsComprehension(b.Value) { + return e.biunifyComprehension(b, a, b2, b1, true, iter) + } + + // Perform standard unification. + _, varA := a.Value.(ast.Var) + _, varB := b.Value.(ast.Var) + + var undo undo + + if varA && varB { + if b1 == b2 && a.Equal(b) { + return iter() + } + b1.bind(a, b, b2, &undo) + err := iter() + undo.Undo() + return err + } else if varA && !varB { + b1.bind(a, b, b2, &undo) + err := iter() + undo.Undo() + return err + } else if varB && !varA { + b2.bind(b, a, b1, &undo) + err := iter() + undo.Undo() + return err + } + + // Sets must not contain unbound variables at this point as we cannot unify + // them. So simply plug both sides (to substitute any bound variables with + // values) and then check for equality. + switch a.Value.(type) { + case ast.Set: + a = b1.Plug(a) + b = b2.Plug(b) + } + + if a.Equal(b) { + return iter() + } + + return nil +} + +func (e *eval) biunifyRef(a, b *ast.Term, b1, b2 *bindings, iter unifyIterator) error { + + ref := a.Value.(ast.Ref) + + if ref[0].Equal(ast.DefaultRootDocument) { + node := e.compiler.RuleTree.Child(ref[0].Value) + eval := evalTree{ + e: e, + ref: ref, + pos: 1, + plugged: ref.Copy(), + bindings: b1, + rterm: b, + rbindings: b2, + node: node, + } + return eval.eval(iter) + } + + var term *ast.Term + var termbindings *bindings + + if ref[0].Equal(ast.InputRootDocument) { + term = e.input + termbindings = b1 + } else { + term, termbindings = b1.apply(ref[0]) + if term == ref[0] { + term = nil + } + } + + if term == nil { + return nil + } + + eval := evalTerm{ + e: e, + ref: ref, + pos: 1, + bindings: b1, + term: term, + termbindings: termbindings, + rterm: b, + rbindings: b2, + } + + return eval.eval(iter) +} + +func (e *eval) biunifyComprehension(a, b *ast.Term, b1, b2 *bindings, swap bool, iter unifyIterator) error { + + if e.unknown(a, b1) { + return e.biunifyComprehensionPartial(a, b, b1, b2, swap, iter) + } + + value, err := e.buildComprehensionCache(a) + + if err != nil { + return err + } else if value != nil { + return e.biunify(value, b, b1, b2, iter) + } else { + e.instr.counterIncr(evalOpComprehensionCacheMiss) + } + + switch a := a.Value.(type) { + case *ast.ArrayComprehension: + return e.biunifyComprehensionArray(a, b, b1, b2, iter) + case *ast.SetComprehension: + return e.biunifyComprehensionSet(a, b, b1, b2, iter) + case *ast.ObjectComprehension: + return e.biunifyComprehensionObject(a, b, b1, b2, iter) + } + + return internalErr(e.query[e.index].Location, "illegal comprehension type") +} + +func (e *eval) buildComprehensionCache(a *ast.Term) (*ast.Term, error) { + + index := e.comprehensionIndex(a) + if index == nil { + e.instr.counterIncr(evalOpComprehensionCacheSkip) + return nil, nil + } + + cache, ok := e.comprehensionCache.Elem(a) + if !ok { + var err error + switch x := a.Value.(type) { + case *ast.ArrayComprehension: + cache, err = e.buildComprehensionCacheArray(x, index.Keys) + case *ast.SetComprehension: + cache, err = e.buildComprehensionCacheSet(x, index.Keys) + case *ast.ObjectComprehension: + cache, err = e.buildComprehensionCacheObject(x, index.Keys) + default: + err = internalErr(e.query[e.index].Location, "illegal comprehension type") + } + if err != nil { + return nil, err + } + e.comprehensionCache.Set(a, cache) + e.instr.counterIncr(evalOpComprehensionCacheBuild) + } else { + e.instr.counterIncr(evalOpComprehensionCacheHit) + } + + values := make([]*ast.Term, len(index.Keys)) + + for i := range index.Keys { + values[i] = e.bindings.Plug(index.Keys[i]) + } + + return cache.Get(values), nil +} + +func (e *eval) buildComprehensionCacheArray(x *ast.ArrayComprehension, keys []*ast.Term) (*comprehensionCacheElem, error) { + child := e.child(x.Body) + node := newComprehensionCacheElem() + return node, child.Run(func(child *eval) error { + values := make([]*ast.Term, len(keys)) + for i := range keys { + values[i] = child.bindings.Plug(keys[i]) + } + head := child.bindings.Plug(x.Term) + cached := node.Get(values) + if cached != nil { + cached.Value = cached.Value.(*ast.Array).Append(head) + } else { + node.Put(values, ast.ArrayTerm(head)) + } + return nil + }) +} + +func (e *eval) buildComprehensionCacheSet(x *ast.SetComprehension, keys []*ast.Term) (*comprehensionCacheElem, error) { + child := e.child(x.Body) + node := newComprehensionCacheElem() + return node, child.Run(func(child *eval) error { + values := make([]*ast.Term, len(keys)) + for i := range keys { + values[i] = child.bindings.Plug(keys[i]) + } + head := child.bindings.Plug(x.Term) + cached := node.Get(values) + if cached != nil { + set := cached.Value.(ast.Set) + set.Add(head) + } else { + node.Put(values, ast.SetTerm(head)) + } + return nil + }) +} + +func (e *eval) buildComprehensionCacheObject(x *ast.ObjectComprehension, keys []*ast.Term) (*comprehensionCacheElem, error) { + child := e.child(x.Body) + node := newComprehensionCacheElem() + return node, child.Run(func(child *eval) error { + values := make([]*ast.Term, len(keys)) + for i := range keys { + values[i] = child.bindings.Plug(keys[i]) + } + headKey := child.bindings.Plug(x.Key) + headValue := child.bindings.Plug(x.Value) + cached := node.Get(values) + if cached != nil { + obj := cached.Value.(ast.Object) + obj.Insert(headKey, headValue) + } else { + node.Put(values, ast.ObjectTerm(ast.Item(headKey, headValue))) + } + return nil + }) +} + +func (e *eval) biunifyComprehensionPartial(a, b *ast.Term, b1, b2 *bindings, swap bool, iter unifyIterator) error { + var err error + cpyA, err := e.amendComprehension(a, b1) + if err != nil { + return err + } + + if ast.IsComprehension(b.Value) { + b, err = e.amendComprehension(b, b2) + if err != nil { + return err + } + } + + // The other term might need to be plugged so include the bindings. The + // bindings for the comprehension term are saved (for compatibility) but + // the eventual plug operation on the comprehension will be a no-op. + if !swap { + return e.saveUnify(cpyA, b, b1, b2, iter) + } + + return e.saveUnify(b, cpyA, b2, b1, iter) +} + +// amendComprehension captures bindings available to the comprehension, +// and used within its term or body. +func (e *eval) amendComprehension(a *ast.Term, b1 *bindings) (*ast.Term, error) { + cpyA := a.Copy() + + // Namespace the variables in the body to avoid collision when the final + // queries returned by partial evaluation. + var body *ast.Body + + switch a := cpyA.Value.(type) { + case *ast.ArrayComprehension: + body = &a.Body + case *ast.SetComprehension: + body = &a.Body + case *ast.ObjectComprehension: + body = &a.Body + default: + return nil, fmt.Errorf("illegal comprehension %T", a) + } + + vars := a.Vars() + err := b1.Iter(e.caller.bindings, func(k, v *ast.Term) error { + if vars.Contains(k.Value.(ast.Var)) { + body.Append(ast.Equality.Expr(k, v)) + } + return nil + }) + if err != nil { + return nil, err + } + + b1.Namespace(cpyA, e.caller.bindings) + return cpyA, nil +} + +func (e *eval) biunifyComprehensionArray(x *ast.ArrayComprehension, b *ast.Term, b1, b2 *bindings, iter unifyIterator) error { + result := ast.NewArray() + child := e.closure(x.Body) + err := child.Run(func(child *eval) error { + result = result.Append(child.bindings.Plug(x.Term)) + return nil + }) + if err != nil { + return err + } + return e.biunify(ast.NewTerm(result), b, b1, b2, iter) +} + +func (e *eval) biunifyComprehensionSet(x *ast.SetComprehension, b *ast.Term, b1, b2 *bindings, iter unifyIterator) error { + result := ast.NewSet() + child := e.closure(x.Body) + err := child.Run(func(child *eval) error { + result.Add(child.bindings.Plug(x.Term)) + return nil + }) + if err != nil { + return err + } + return e.biunify(ast.NewTerm(result), b, b1, b2, iter) +} + +func (e *eval) biunifyComprehensionObject(x *ast.ObjectComprehension, b *ast.Term, b1, b2 *bindings, iter unifyIterator) error { + result := ast.NewObject() + child := e.closure(x.Body) + err := child.Run(func(child *eval) error { + key := child.bindings.Plug(x.Key) + value := child.bindings.Plug(x.Value) + exist := result.Get(key) + if exist != nil && !exist.Equal(value) { + return objectDocKeyConflictErr(x.Key.Location) + } + result.Insert(key, value) + return nil + }) + if err != nil { + return err + } + return e.biunify(ast.NewTerm(result), b, b1, b2, iter) +} + +func (e *eval) saveExpr(expr *ast.Expr, b *bindings, iter unifyIterator) error { + expr.With = e.query[e.index].With + expr.Location = e.query[e.index].Location + e.saveStack.Push(expr, b, b) + e.traceSave(expr) + err := iter() + e.saveStack.Pop() + return err +} + +func (e *eval) saveExprMarkUnknowns(expr *ast.Expr, b *bindings, iter unifyIterator) error { + expr.With = e.query[e.index].With + expr.Location = e.query[e.index].Location + declArgsLen, err := e.getDeclArgsLen(expr) + if err != nil { + return err + } + var pops int + if pairs := getSavePairsFromExpr(declArgsLen, expr, b, nil); len(pairs) > 0 { + pops += len(pairs) + for _, p := range pairs { + e.saveSet.Push([]*ast.Term{p.term}, p.b) + } + } + e.saveStack.Push(expr, b, b) + e.traceSave(expr) + err = iter() + e.saveStack.Pop() + for i := 0; i < pops; i++ { + e.saveSet.Pop() + } + return err +} + +func (e *eval) saveUnify(a, b *ast.Term, b1, b2 *bindings, iter unifyIterator) error { + e.instr.startTimer(partialOpSaveUnify) + expr := ast.Equality.Expr(a, b) + expr.With = e.query[e.index].With + expr.Location = e.query[e.index].Location + pops := 0 + if pairs := getSavePairsFromTerm(a, b1, nil); len(pairs) > 0 { + pops += len(pairs) + for _, p := range pairs { + e.saveSet.Push([]*ast.Term{p.term}, p.b) + } + + } + if pairs := getSavePairsFromTerm(b, b2, nil); len(pairs) > 0 { + pops += len(pairs) + for _, p := range pairs { + e.saveSet.Push([]*ast.Term{p.term}, p.b) + } + } + e.saveStack.Push(expr, b1, b2) + e.traceSave(expr) + e.instr.stopTimer(partialOpSaveUnify) + err := iter() + + e.saveStack.Pop() + for i := 0; i < pops; i++ { + e.saveSet.Pop() + } + + return err +} + +func (e *eval) saveCall(declArgsLen int, terms []*ast.Term, iter unifyIterator) error { + expr := ast.NewExpr(terms) + expr.With = e.query[e.index].With + expr.Location = e.query[e.index].Location + + // If call-site includes output value then partial eval must add vars in output + // position to the save set. + pops := 0 + if declArgsLen == len(terms)-2 { + if pairs := getSavePairsFromTerm(terms[len(terms)-1], e.bindings, nil); len(pairs) > 0 { + pops += len(pairs) + for _, p := range pairs { + e.saveSet.Push([]*ast.Term{p.term}, p.b) + } + } + } + e.saveStack.Push(expr, e.bindings, nil) + e.traceSave(expr) + err := iter() + + e.saveStack.Pop() + for i := 0; i < pops; i++ { + e.saveSet.Pop() + } + return err +} + +func (e *eval) saveInlinedNegatedExprs(exprs []*ast.Expr, iter unifyIterator) error { + + with := make([]*ast.With, len(e.query[e.index].With)) + + for i := range e.query[e.index].With { + cpy := e.query[e.index].With[i].Copy() + cpy.Value = e.bindings.PlugNamespaced(cpy.Value, e.caller.bindings) + with[i] = cpy + } + + for _, expr := range exprs { + expr.With = with + e.saveStack.Push(expr, nil, nil) + e.traceSave(expr) + } + err := iter() + for i := 0; i < len(exprs); i++ { + e.saveStack.Pop() + } + return err +} + +func (e *eval) getRules(ref ast.Ref, args []*ast.Term) (*ast.IndexResult, error) { + e.instr.startTimer(evalOpRuleIndex) + defer e.instr.stopTimer(evalOpRuleIndex) + + index := e.compiler.RuleIndex(ref) + if index == nil { + return nil, nil + } + + var result *ast.IndexResult + var err error + if e.indexing { + result, err = index.Lookup(&evalResolver{e: e, args: args}) + } else { + result, err = index.AllRules(&evalResolver{e: e}) + } + if err != nil { + return nil, err + } + + result.EarlyExit = result.EarlyExit && e.earlyExit + + var msg strings.Builder + if len(result.Rules) == 1 { + msg.WriteString("(matched 1 rule") + } else { + msg.Grow(len("(matched NNNN rules)")) + msg.WriteString("(matched ") + msg.WriteString(strconv.Itoa(len(result.Rules))) + msg.WriteString(" rules") + } + if result.EarlyExit { + msg.WriteString(", early exit") + } + msg.WriteRune(')') + e.traceIndex(e.query[e.index], msg.String(), &ref) + return result, err +} + +func (e *eval) Resolve(ref ast.Ref) (ast.Value, error) { + return (&evalResolver{e: e}).Resolve(ref) +} + +type evalResolver struct { + e *eval + args []*ast.Term +} + +func (e *evalResolver) Resolve(ref ast.Ref) (ast.Value, error) { + e.e.instr.startTimer(evalOpResolve) + + if e.e.inliningControl.Disabled(ref, true) || e.e.saveSet.Contains(ast.NewTerm(ref), nil) { + e.e.instr.stopTimer(evalOpResolve) + return nil, ast.UnknownValueErr{} + } + + // Lookup of function argument values works by using the `args` ref[0], + // where the ast.Number in ref[1] references the function argument of + // that number. The callsite-local arguments are passed in e.args, + // indexed by argument index. + if ref[0].Equal(ast.FunctionArgRootDocument) { + v, ok := ref[1].Value.(ast.Number) + if ok { + i, ok := v.Int() + if ok && i >= 0 && i < len(e.args) { + e.e.instr.stopTimer(evalOpResolve) + plugged := e.e.bindings.PlugNamespaced(e.args[i], e.e.caller.bindings) + return plugged.Value, nil + } + } + e.e.instr.stopTimer(evalOpResolve) + return nil, ast.UnknownValueErr{} + } + + if ref[0].Equal(ast.InputRootDocument) { + if e.e.input != nil { + v, err := e.e.input.Value.Find(ref[1:]) + if err != nil { + v = nil + } + e.e.instr.stopTimer(evalOpResolve) + return v, nil + } + e.e.instr.stopTimer(evalOpResolve) + return nil, nil + } + + if ref[0].Equal(ast.DefaultRootDocument) { + + var repValue ast.Value + + if e.e.data != nil { + if v, err := e.e.data.Value.Find(ref[1:]); err == nil { + repValue = v + } else { + repValue = nil + } + } + + if e.e.targetStack.Prefixed(ref) { + e.e.instr.stopTimer(evalOpResolve) + return repValue, nil + } + + var merged ast.Value + var err error + + // Converting large JSON values into AST values can be fairly expensive. For + // example, a 2MB JSON value can take upwards of 30 millisceonds to convert. + // We cache the result of conversion here in case the same base document is + // being read multiple times during evaluation. + realValue := e.e.baseCache.Get(ref) + if realValue != nil { + e.e.instr.counterIncr(evalOpBaseCacheHit) + if repValue == nil { + e.e.instr.stopTimer(evalOpResolve) + return realValue, nil + } + var ok bool + merged, ok = merge(repValue, realValue) + if !ok { + err = mergeConflictErr(ref[0].Location) + } + } else { + e.e.instr.counterIncr(evalOpBaseCacheMiss) + merged, err = e.e.resolveReadFromStorage(ref, repValue) + } + e.e.instr.stopTimer(evalOpResolve) + return merged, err + } + e.e.instr.stopTimer(evalOpResolve) + return nil, fmt.Errorf("illegal ref") +} + +func (e *eval) resolveReadFromStorage(ref ast.Ref, a ast.Value) (ast.Value, error) { + if refContainsNonScalar(ref) { + return a, nil + } + + v, err := e.external.Resolve(e, ref) + + if err != nil { + return nil, err + } else if v == nil { + + path, err := storage.NewPathForRef(ref) + if err != nil { + if !storage.IsNotFound(err) { + return nil, err + } + return a, nil + } + + blob, err := e.store.Read(e.ctx, e.txn, path) + if err != nil { + if !storage.IsNotFound(err) { + return nil, err + } + return a, nil + } + + if len(path) == 0 { + switch obj := blob.(type) { + case map[string]interface{}: + if len(obj) > 0 { + cpy := make(map[string]interface{}, len(obj)-1) + for k, v := range obj { + if string(ast.SystemDocumentKey) != k { + cpy[k] = v + } + } + blob = cpy + } + case ast.Object: + if obj.Len() > 0 { + cpy := ast.NewObject() + if err := obj.Iter(func(k *ast.Term, v *ast.Term) error { + if !ast.SystemDocumentKey.Equal(k.Value) { + cpy.Insert(k, v) + } + return nil + }); err != nil { + return nil, err + } + blob = cpy + + } + } + } + + var ok bool + v, ok = blob.(ast.Value) + if !ok { + v, err = ast.InterfaceToValue(blob) + if err != nil { + return nil, err + } + } + } + + e.baseCache.Put(ref, v) + if a == nil { + return v, nil + } + + merged, ok := merge(a, v) + if !ok { + return nil, mergeConflictErr(ref[0].Location) + } + + return merged, nil +} + +func (e *eval) generateVar(suffix string) *ast.Term { + return ast.VarTerm(fmt.Sprintf("%v_%v", e.genvarprefix, suffix)) +} + +func (e *eval) rewrittenVar(v ast.Var) (ast.Var, bool) { + if e.compiler != nil { + if rw, ok := e.compiler.RewrittenVars[v]; ok { + return rw, true + } + } + if e.queryCompiler != nil { + if rw, ok := e.queryCompiler.RewrittenVars()[v]; ok { + return rw, true + } + } + return v, false +} + +func (e *eval) getDeclArgsLen(x *ast.Expr) (int, error) { + + if !x.IsCall() { + return -1, nil + } + + operator := x.Operator() + bi, _, ok := e.builtinFunc(operator.String()) + + if ok { + return len(bi.Decl.Args()), nil + } + + ir, err := e.getRules(operator, nil) + if err != nil { + return -1, err + } else if ir == nil || ir.Empty() { + return -1, nil + } + + return len(ir.Rules[0].Head.Args), nil +} + +type evalBuiltin struct { + e *eval + bi *ast.Builtin + bctx BuiltinContext + f BuiltinFunc + terms []*ast.Term +} + +func (e evalBuiltin) eval(iter unifyIterator) error { + + operands := make([]*ast.Term, len(e.terms)) + + for i := 0; i < len(e.terms); i++ { + operands[i] = e.e.bindings.Plug(e.terms[i]) + } + + numDeclArgs := len(e.bi.Decl.FuncArgs().Args) + + e.e.instr.startTimer(evalOpBuiltinCall) + + err := e.f(e.bctx, operands, func(output *ast.Term) error { + + e.e.instr.stopTimer(evalOpBuiltinCall) + + var err error + + if e.bi.Decl.Result() == nil { + err = iter() + } else if len(operands) == numDeclArgs { + if output.Value.Compare(ast.Boolean(false)) != 0 { + err = iter() + } + } else { + err = e.e.unify(e.terms[len(e.terms)-1], output, iter) + } + + if err != nil { + err = Halt{Err: err} + } + + e.e.instr.startTimer(evalOpBuiltinCall) + return err + }) + + if err != nil { + if t, ok := err.(Halt); ok { + err = t.Err + } else { + e.e.builtinErrors.errs = append(e.e.builtinErrors.errs, err) + err = nil + } + } + + e.e.instr.stopTimer(evalOpBuiltinCall) + return err +} + +type evalFunc struct { + e *eval + ref ast.Ref + terms []*ast.Term + ir *ast.IndexResult +} + +func (e evalFunc) eval(iter unifyIterator) error { + + // default functions aren't supported: + // https://github.com/open-policy-agent/opa/issues/2445 + if len(e.ir.Rules) == 0 { + return nil + } + + argCount := len(e.ir.Rules[0].Head.Args) + + if len(e.ir.Else) > 0 && e.e.unknown(e.e.query[e.e.index], e.e.bindings) { + // Partial evaluation of ordered rules is not supported currently. Save the + // expression and continue. This could be revisited in the future. + return e.e.saveCall(argCount, e.terms, iter) + } + + if e.e.partial() && (e.e.inliningControl.shallow || e.e.inliningControl.Disabled(e.ref, false)) { + // check if the function definitions, or any of the arguments + // contain something unknown + unknown := e.e.unknown(e.ref, e.e.bindings) + for i := 1; !unknown && i <= argCount; i++ { + unknown = e.e.unknown(e.terms[i], e.e.bindings) + } + if unknown { + return e.partialEvalSupport(argCount, iter) + } + } + return suppressEarlyExit(e.evalValue(iter, argCount, e.ir.EarlyExit)) +} + +func (e evalFunc) evalValue(iter unifyIterator, argCount int, findOne bool) error { + var cacheKey ast.Ref + var hit bool + var err error + if !e.e.partial() { + cacheKey, hit, err = e.evalCache(argCount, iter) + if err != nil { + return err + } else if hit { + return nil + } + } + + var prev *ast.Term + + for _, rule := range e.ir.Rules { + next, err := e.evalOneRule(iter, rule, cacheKey, prev, findOne) + if err != nil { + return err + } + if next == nil { + for _, erule := range e.ir.Else[rule] { + next, err = e.evalOneRule(iter, erule, cacheKey, prev, findOne) + if err != nil { + return err + } + if next != nil { + break + } + } + } + if next != nil { + prev = next + } + } + + return nil +} + +func (e evalFunc) evalCache(argCount int, iter unifyIterator) (ast.Ref, bool, error) { + var plen int + if len(e.terms) == argCount+2 { // func name + output = 2 + plen = len(e.terms) - 1 + } else { + plen = len(e.terms) + } + cacheKey := make([]*ast.Term, plen) + for i := 0; i < plen; i++ { + cacheKey[i] = e.e.bindings.Plug(e.terms[i]) + } + + cached := e.e.virtualCache.Get(cacheKey) + if cached != nil { + e.e.instr.counterIncr(evalOpVirtualCacheHit) + if argCount == len(e.terms)-1 { // f(x) + if ast.Boolean(false).Equal(cached.Value) { + return nil, true, nil + } + return nil, true, iter() + } + // f(x, y), y captured output value + return nil, true, e.e.unify(e.terms[len(e.terms)-1] /* y */, cached, iter) + } + e.e.instr.counterIncr(evalOpVirtualCacheMiss) + return cacheKey, false, nil +} + +func (e evalFunc) evalOneRule(iter unifyIterator, rule *ast.Rule, cacheKey ast.Ref, prev *ast.Term, findOne bool) (*ast.Term, error) { + + child := e.e.child(rule.Body) + child.findOne = findOne + + args := make([]*ast.Term, len(e.terms)-1) + copy(args, rule.Head.Args) + + if len(args) == len(rule.Head.Args)+1 { + args[len(args)-1] = rule.Head.Value + } + + var result *ast.Term + + child.traceEnter(rule) + + err := child.biunifyArrays(ast.NewArray(e.terms[1:]...), ast.NewArray(args...), e.e.bindings, child.bindings, func() error { + return child.eval(func(child *eval) error { + child.traceExit(rule) + + // Partial evaluation must save an expression that tests the output value if the output value + // was not captured to handle the case where the output value may be `false`. + if len(rule.Head.Args) == len(e.terms)-1 && e.e.saveSet.Contains(rule.Head.Value, child.bindings) { + err := e.e.saveExpr(ast.NewExpr(rule.Head.Value), child.bindings, iter) + child.traceRedo(rule) + return err + } + + result = child.bindings.Plug(rule.Head.Value) + if cacheKey != nil { + e.e.virtualCache.Put(cacheKey, result) // the redos confirm this, or the evaluation is aborted + } + + if len(rule.Head.Args) == len(e.terms)-1 { + if result.Value.Compare(ast.Boolean(false)) == 0 { + if prev != nil && ast.Compare(prev, result) != 0 { + return functionConflictErr(rule.Location) + } + return nil + } + } + + // Partial evaluation should explore all rules and may not produce + // a ground result so we do not perform conflict detection or + // deduplication. See "ignore conflicts: functions" test case for + // an example. + if !e.e.partial() { + if prev != nil { + if ast.Compare(prev, result) != 0 { + return functionConflictErr(rule.Location) + } + child.traceRedo(rule) + return nil + } + } + + prev = result + + if err := iter(); err != nil { + return err + } + + child.traceRedo(rule) + return nil + }) + }) + + return result, err +} + +func (e evalFunc) partialEvalSupport(declArgsLen int, iter unifyIterator) error { + + path := e.e.namespaceRef(e.ref) + term := ast.NewTerm(path) + + if !e.e.saveSupport.Exists(path) { + for _, rule := range e.ir.Rules { + err := e.partialEvalSupportRule(rule, path) + if err != nil { + return err + } + } + } + + if !e.e.saveSupport.Exists(path) { // we haven't saved anything, nothing to call + return nil + } + + return e.e.saveCall(declArgsLen, append([]*ast.Term{term}, e.terms[1:]...), iter) +} + +func (e evalFunc) partialEvalSupportRule(rule *ast.Rule, path ast.Ref) error { + + child := e.e.child(rule.Body) + child.traceEnter(rule) + + e.e.saveStack.PushQuery(nil) + + // treat the function arguments as unknown during rule body evaluation + var args []*ast.Term + ast.WalkVars(rule.Head.Args, func(v ast.Var) bool { + args = append(args, ast.VarTerm(string(v))) + return false + }) + e.e.saveSet.Push(args, child.bindings) + + err := child.eval(func(child *eval) error { + child.traceExit(rule) + + current := e.e.saveStack.PopQuery() + plugged := current.Plug(e.e.caller.bindings) + + // Skip this rule body if it fails to type-check. + // Type-checking failure means the rule body will never succeed. + if e.e.compiler.PassesTypeCheck(plugged) { + head := &ast.Head{ + Name: rule.Head.Name, + Value: child.bindings.PlugNamespaced(rule.Head.Value, e.e.caller.bindings), + Args: make([]*ast.Term, len(rule.Head.Args)), + } + for i, a := range rule.Head.Args { + head.Args[i] = child.bindings.PlugNamespaced(a, e.e.caller.bindings) + } + + e.e.saveSupport.Insert(path, &ast.Rule{ + Head: head, + Body: plugged, + }) + } + child.traceRedo(rule) + e.e.saveStack.PushQuery(current) + return nil + }) + + e.e.saveSet.Pop() + e.e.saveStack.PopQuery() + return err +} + +type evalTree struct { + e *eval + ref ast.Ref + plugged ast.Ref + pos int + bindings *bindings + rterm *ast.Term + rbindings *bindings + node *ast.TreeNode +} + +func (e evalTree) eval(iter unifyIterator) error { + + if len(e.ref) == e.pos { + return e.finish(iter) + } + + plugged := e.bindings.Plug(e.ref[e.pos]) + + if plugged.IsGround() { + return e.next(iter, plugged) + } + + return e.enumerate(iter) +} + +func (e evalTree) finish(iter unifyIterator) error { + + // In some cases, it may not be possible to PE the ref. If the path refers + // to virtual docs that PE does not support or base documents where inlining + // has been disabled, then we have to save. + save := e.e.unknown(e.plugged, e.e.bindings) + + if save { + return e.e.saveUnify(ast.NewTerm(e.plugged), e.rterm, e.bindings, e.rbindings, iter) + } + + v, err := e.extent() + if err != nil || v == nil { + return err + } + + return e.e.biunify(e.rterm, v, e.rbindings, e.bindings, iter) +} + +func (e evalTree) next(iter unifyIterator, plugged *ast.Term) error { + + var node *ast.TreeNode + + cpy := e + cpy.plugged[e.pos] = plugged + cpy.pos++ + + if !e.e.targetStack.Prefixed(cpy.plugged[:cpy.pos]) { + if e.node != nil { + node = e.node.Child(plugged.Value) + if node != nil && len(node.Values) > 0 { + r := evalVirtual{ + e: e.e, + ref: e.ref, + plugged: e.plugged, + pos: e.pos, + bindings: e.bindings, + rterm: e.rterm, + rbindings: e.rbindings, + } + r.plugged[e.pos] = plugged + return r.eval(iter) + } + } + } + + cpy.node = node + return cpy.eval(iter) +} + +func (e evalTree) enumerate(iter unifyIterator) error { + + if e.e.inliningControl.Disabled(e.plugged[:e.pos], true) { + return e.e.saveUnify(ast.NewTerm(e.plugged), e.rterm, e.bindings, e.rbindings, iter) + } + + doc, err := e.e.Resolve(e.plugged[:e.pos]) + if err != nil { + return err + } + + if doc != nil { + switch doc := doc.(type) { + case *ast.Array: + for i := 0; i < doc.Len(); i++ { + k := ast.IntNumberTerm(i) + err := e.e.biunify(k, e.ref[e.pos], e.bindings, e.bindings, func() error { + return e.next(iter, k) + }) + if err != nil { + return err + } + } + case ast.Object: + err := doc.Iter(func(k, _ *ast.Term) error { + return e.e.biunify(k, e.ref[e.pos], e.bindings, e.bindings, func() error { + return e.next(iter, k) + }) + }) + if err != nil { + return err + } + case ast.Set: + err := doc.Iter(func(elem *ast.Term) error { + return e.e.biunify(elem, e.ref[e.pos], e.bindings, e.bindings, func() error { + return e.next(iter, elem) + }) + }) + if err != nil { + return err + } + } + } + + if e.node == nil { + return nil + } + + for _, k := range e.node.Sorted { + key := ast.NewTerm(k) + if err := e.e.biunify(key, e.ref[e.pos], e.bindings, e.bindings, func() error { + return e.next(iter, key) + }); err != nil { + return err + } + } + + return nil +} + +func (e evalTree) extent() (*ast.Term, error) { + base, err := e.e.Resolve(e.plugged) + if err != nil { + return nil, err + } + + virtual, err := e.leaves(e.plugged, e.node) + if err != nil { + return nil, err + } + + if virtual == nil { + if base == nil { + return nil, nil + } + return ast.NewTerm(base), nil + } + + if base != nil { + merged, ok := merge(base, virtual) + if !ok { + return nil, mergeConflictErr(e.plugged[0].Location) + } + return ast.NewTerm(merged), nil + } + + return ast.NewTerm(virtual), nil +} + +func (e evalTree) leaves(plugged ast.Ref, node *ast.TreeNode) (ast.Object, error) { + + if e.node == nil { + return nil, nil + } + + result := ast.NewObject() + + for _, k := range node.Sorted { + + child := node.Children[k] + + if child.Hide { + continue + } + + plugged = append(plugged, ast.NewTerm(child.Key)) + + var save ast.Value + var err error + + if len(child.Values) > 0 { + rterm := e.e.generateVar("leaf") + err = e.e.unify(ast.NewTerm(plugged), rterm, func() error { + save = e.e.bindings.Plug(rterm).Value + return nil + }) + } else { + save, err = e.leaves(plugged, child) + } + + if err != nil { + return nil, err + } + + if save != nil { + v := ast.NewObject([2]*ast.Term{plugged[len(plugged)-1], ast.NewTerm(save)}) + result, _ = result.Merge(v) + } + + plugged = plugged[:len(plugged)-1] + } + + return result, nil +} + +type evalVirtual struct { + e *eval + ref ast.Ref + plugged ast.Ref + pos int + bindings *bindings + rterm *ast.Term + rbindings *bindings +} + +func (e evalVirtual) eval(iter unifyIterator) error { + + ir, err := e.e.getRules(e.plugged[:e.pos+1], nil) + if err != nil { + return err + } + + // Partial evaluation of ordered rules is not supported currently. Save the + // expression and continue. This could be revisited in the future. + if len(ir.Else) > 0 && e.e.unknown(e.ref, e.bindings) { + return e.e.saveUnify(ast.NewTerm(e.ref), e.rterm, e.bindings, e.rbindings, iter) + } + + switch ir.Kind { + case ast.PartialSetDoc: + eval := evalVirtualPartial{ + e: e.e, + ref: e.ref, + plugged: e.plugged, + pos: e.pos, + ir: ir, + bindings: e.bindings, + rterm: e.rterm, + rbindings: e.rbindings, + empty: ast.SetTerm(), + } + return eval.eval(iter) + case ast.PartialObjectDoc: + eval := evalVirtualPartial{ + e: e.e, + ref: e.ref, + plugged: e.plugged, + pos: e.pos, + ir: ir, + bindings: e.bindings, + rterm: e.rterm, + rbindings: e.rbindings, + empty: ast.ObjectTerm(), + } + return eval.eval(iter) + default: + eval := evalVirtualComplete{ + e: e.e, + ref: e.ref, + plugged: e.plugged, + pos: e.pos, + ir: ir, + bindings: e.bindings, + rterm: e.rterm, + rbindings: e.rbindings, + } + return eval.eval(iter) + } +} + +type evalVirtualPartial struct { + e *eval + ref ast.Ref + plugged ast.Ref + pos int + ir *ast.IndexResult + bindings *bindings + rterm *ast.Term + rbindings *bindings + empty *ast.Term +} + +type evalVirtualPartialCacheHint struct { + key ast.Ref + hit bool + full bool +} + +func (e evalVirtualPartial) eval(iter unifyIterator) error { + + unknown := e.e.unknown(e.ref[:e.pos+1], e.bindings) + + if len(e.ref) == e.pos+1 { + if unknown { + return e.partialEvalSupport(iter) + } + return e.evalAllRules(iter, e.ir.Rules) + } + + if (unknown && e.e.inliningControl.shallow) || e.e.inliningControl.Disabled(e.ref[:e.pos+1], false) { + return e.partialEvalSupport(iter) + } + + return e.evalEachRule(iter, unknown) +} + +func (e evalVirtualPartial) evalEachRule(iter unifyIterator, unknown bool) error { + + if e.e.unknown(e.ref[e.pos+1], e.bindings) { + for _, rule := range e.ir.Rules { + if err := e.evalOneRulePostUnify(iter, rule); err != nil { + return err + } + } + return nil + } + + hint, err := e.evalCache(iter) + if err != nil { + return err + } else if hint.hit { + return nil + } + + if hint.full { + result, err := e.evalAllRulesNoCache(e.ir.Rules) + if err != nil { + return err + } + e.e.virtualCache.Put(hint.key, result) + return e.evalTerm(iter, e.pos+1, result, e.bindings) + } + + result := e.empty + for _, rule := range e.ir.Rules { + if err := e.evalOneRulePreUnify(iter, rule, hint, result, unknown); err != nil { + return err + } + } + + return nil +} + +func (e evalVirtualPartial) evalAllRules(iter unifyIterator, rules []*ast.Rule) error { + + cacheKey := e.plugged[:e.pos+1] + result := e.e.virtualCache.Get(cacheKey) + if result != nil { + e.e.instr.counterIncr(evalOpVirtualCacheHit) + return e.e.biunify(result, e.rterm, e.bindings, e.rbindings, iter) + } + + e.e.instr.counterIncr(evalOpVirtualCacheMiss) + + result, err := e.evalAllRulesNoCache(rules) + if err != nil { + return err + } + + if cacheKey != nil { + e.e.virtualCache.Put(cacheKey, result) + } + + return e.e.biunify(result, e.rterm, e.bindings, e.rbindings, iter) +} + +func (e evalVirtualPartial) evalAllRulesNoCache(rules []*ast.Rule) (*ast.Term, error) { + result := e.empty + + for _, rule := range rules { + child := e.e.child(rule.Body) + child.traceEnter(rule) + err := child.eval(func(*eval) error { + child.traceExit(rule) + var err error + result, _, err = e.reduce(rule.Head, child.bindings, result) + if err != nil { + return err + } + + child.traceRedo(rule) + return nil + }) + + if err != nil { + return nil, err + } + } + + return result, nil +} + +func (e evalVirtualPartial) evalOneRulePreUnify(iter unifyIterator, rule *ast.Rule, hint evalVirtualPartialCacheHint, result *ast.Term, unknown bool) error { + + key := e.ref[e.pos+1] + child := e.e.child(rule.Body) + + child.traceEnter(rule) + var defined bool + + err := child.biunify(rule.Head.Key, key, child.bindings, e.bindings, func() error { + defined = true + return child.eval(func(child *eval) error { + + term := rule.Head.Value + if term == nil { + term = rule.Head.Key + } + + if hint.key != nil { + result := child.bindings.Plug(term) + e.e.virtualCache.Put(hint.key, result) + } + + // NOTE(tsandall): if the rule set depends on any unknowns then do + // not perform the duplicate check because evaluation of the ruleset + // may not produce a definitive result. This is a bit strict--we + // could improve by skipping only when saves occur. + if !unknown { + var dup bool + var err error + result, dup, err = e.reduce(rule.Head, child.bindings, result) + if err != nil { + return err + } else if dup { + child.traceDuplicate(rule) + return nil + } + } + + child.traceExit(rule) + term, termbindings := child.bindings.apply(term) + err := e.evalTerm(iter, e.pos+2, term, termbindings) + if err != nil { + return err + } + + child.traceRedo(rule) + return nil + }) + }) + + if err != nil { + return err + } + + // TODO(tsandall): why are we tracing here? this looks wrong. + if !defined { + child.traceFail(rule) + } + + return nil +} + +func (e evalVirtualPartial) evalOneRulePostUnify(iter unifyIterator, rule *ast.Rule) error { + + key := e.ref[e.pos+1] + child := e.e.child(rule.Body) + + child.traceEnter(rule) + var defined bool + + err := child.eval(func(child *eval) error { + defined = true + return e.e.biunify(rule.Head.Key, key, child.bindings, e.bindings, func() error { + return e.evalOneRuleContinue(iter, rule, child) + }) + }) + + if err != nil { + return err + } + + if !defined { + child.traceFail(rule) + } + + return nil +} + +func (e evalVirtualPartial) evalOneRuleContinue(iter unifyIterator, rule *ast.Rule, child *eval) error { + + child.traceExit(rule) + + term := rule.Head.Value + if term == nil { + term = rule.Head.Key + } + + term, termbindings := child.bindings.apply(term) + err := e.evalTerm(iter, e.pos+2, term, termbindings) + if err != nil { + return err + } + + child.traceRedo(rule) + return nil +} + +func (e evalVirtualPartial) partialEvalSupport(iter unifyIterator) error { + + path := e.e.namespaceRef(e.plugged[:e.pos+1]) + term := ast.NewTerm(e.e.namespaceRef(e.ref)) + + var defined bool + + if e.e.saveSupport.Exists(path) { + defined = true + } else { + for i := range e.ir.Rules { + ok, err := e.partialEvalSupportRule(e.ir.Rules[i], path) + if err != nil { + return err + } else if ok { + defined = true + } + } + } + + if !defined { + if len(e.ref) != e.pos+1 { + return nil + } + + // the entire partial set/obj was queried, e.g. data.a.q (not data.a.q[x]) + term = e.empty + } + + return e.e.saveUnify(term, e.rterm, e.bindings, e.rbindings, iter) +} + +func (e evalVirtualPartial) partialEvalSupportRule(rule *ast.Rule, path ast.Ref) (bool, error) { + + child := e.e.child(rule.Body) + child.traceEnter(rule) + + e.e.saveStack.PushQuery(nil) + var defined bool + + err := child.eval(func(child *eval) error { + child.traceExit(rule) + defined = true + + current := e.e.saveStack.PopQuery() + plugged := current.Plug(e.e.caller.bindings) + // Skip this rule body if it fails to type-check. + // Type-checking failure means the rule body will never succeed. + if e.e.compiler.PassesTypeCheck(plugged) { + var key, value *ast.Term + + if rule.Head.Key != nil { + key = child.bindings.PlugNamespaced(rule.Head.Key, e.e.caller.bindings) + } + + if rule.Head.Value != nil { + value = child.bindings.PlugNamespaced(rule.Head.Value, e.e.caller.bindings) + } + + head := ast.NewHead(rule.Head.Name, key, value) + + if !e.e.inliningControl.shallow { + cp := copypropagation.New(head.Vars()). + WithEnsureNonEmptyBody(true). + WithCompiler(e.e.compiler) + plugged = applyCopyPropagation(cp, e.e.instr, plugged) + } + + e.e.saveSupport.Insert(path, &ast.Rule{ + Head: head, + Body: plugged, + Default: rule.Default, + }) + } + child.traceRedo(rule) + e.e.saveStack.PushQuery(current) + return nil + }) + e.e.saveStack.PopQuery() + return defined, err +} + +func (e evalVirtualPartial) evalTerm(iter unifyIterator, pos int, term *ast.Term, termbindings *bindings) error { + eval := evalTerm{ + e: e.e, + ref: e.ref, + pos: pos, + bindings: e.bindings, + term: term, + termbindings: termbindings, + rterm: e.rterm, + rbindings: e.rbindings, + } + return eval.eval(iter) +} + +func (e evalVirtualPartial) evalCache(iter unifyIterator) (evalVirtualPartialCacheHint, error) { + + var hint evalVirtualPartialCacheHint + + if e.e.unknown(e.ref[:e.pos+1], e.bindings) { + return hint, nil + } + + if cached := e.e.virtualCache.Get(e.plugged[:e.pos+1]); cached != nil { // have full extent cached + e.e.instr.counterIncr(evalOpVirtualCacheHit) + hint.hit = true + return hint, e.evalTerm(iter, e.pos+1, cached, e.bindings) + } + + plugged := e.bindings.Plug(e.ref[e.pos+1]) + + if plugged.IsGround() { + hint.key = append(e.plugged[:e.pos+1], plugged) + + if cached := e.e.virtualCache.Get(hint.key); cached != nil { + e.e.instr.counterIncr(evalOpVirtualCacheHit) + hint.hit = true + return hint, e.evalTerm(iter, e.pos+2, cached, e.bindings) + } + } else if _, ok := plugged.Value.(ast.Var); ok { + hint.full = true + hint.key = e.plugged[:e.pos+1] + } + + e.e.instr.counterIncr(evalOpVirtualCacheMiss) + + return hint, nil +} + +func (e evalVirtualPartial) reduce(head *ast.Head, b *bindings, result *ast.Term) (*ast.Term, bool, error) { + + var exists bool + key := b.Plug(head.Key) + + switch v := result.Value.(type) { + case ast.Set: + exists = v.Contains(key) + v.Add(key) + case ast.Object: + value := b.Plug(head.Value) + if curr := v.Get(key); curr != nil { + if !curr.Equal(value) { + return nil, false, objectDocKeyConflictErr(head.Location) + } + exists = true + } else { + v.Insert(key, value) + } + } + + return result, exists, nil +} + +type evalVirtualComplete struct { + e *eval + ref ast.Ref + plugged ast.Ref + pos int + ir *ast.IndexResult + bindings *bindings + rterm *ast.Term + rbindings *bindings +} + +func (e evalVirtualComplete) eval(iter unifyIterator) error { + + if e.ir.Empty() { + return nil + } + + if len(e.ir.Rules) > 0 && len(e.ir.Rules[0].Head.Args) > 0 { + return nil + } + + if !e.e.unknown(e.ref, e.bindings) { + return suppressEarlyExit(e.evalValue(iter, e.ir.EarlyExit)) + } + + var generateSupport bool + + if e.ir.Default != nil { + // If the other term is not constant OR it's equal to the default value, then + // a support rule must be produced as the default value _may_ be required. On + // the other hand, if the other term is constant (i.e., it does not require + // evaluation) and it differs from the default value then the default value is + // _not_ required, so partially evaluate the rule normally. + rterm := e.rbindings.Plug(e.rterm) + generateSupport = !ast.IsConstant(rterm.Value) || e.ir.Default.Head.Value.Equal(rterm) + } + + if generateSupport || e.e.inliningControl.shallow || e.e.inliningControl.Disabled(e.plugged[:e.pos+1], false) { + return e.partialEvalSupport(iter) + } + + return e.partialEval(iter) +} + +func (e evalVirtualComplete) evalValue(iter unifyIterator, findOne bool) error { + cached := e.e.virtualCache.Get(e.plugged[:e.pos+1]) + if cached != nil { + e.e.instr.counterIncr(evalOpVirtualCacheHit) + return e.evalTerm(iter, cached, e.bindings) + } + + e.e.instr.counterIncr(evalOpVirtualCacheMiss) + + var prev *ast.Term + + for _, rule := range e.ir.Rules { + next, err := e.evalValueRule(iter, rule, prev, findOne) + if err != nil { + return err + } + if next == nil { + for _, erule := range e.ir.Else[rule] { + next, err = e.evalValueRule(iter, erule, prev, findOne) + if err != nil { + return err + } + if next != nil { + break + } + } + } + if next != nil { + prev = next + } + } + + if e.ir.Default != nil && prev == nil { + _, err := e.evalValueRule(iter, e.ir.Default, prev, findOne) + return err + } + + return nil +} + +func (e evalVirtualComplete) evalValueRule(iter unifyIterator, rule *ast.Rule, prev *ast.Term, findOne bool) (*ast.Term, error) { + + child := e.e.child(rule.Body) + child.findOne = findOne + child.traceEnter(rule) + var result *ast.Term + err := child.eval(func(child *eval) error { + child.traceExit(rule) + result = child.bindings.Plug(rule.Head.Value) + + if prev != nil { + if ast.Compare(result, prev) != 0 { + return completeDocConflictErr(rule.Location) + } + child.traceRedo(rule) + return nil + } + + prev = result + e.e.virtualCache.Put(e.plugged[:e.pos+1], result) + term, termbindings := child.bindings.apply(rule.Head.Value) + + err := e.evalTerm(iter, term, termbindings) + if err != nil { + return err + } + + child.traceRedo(rule) + return nil + }) + + return result, err +} + +func (e evalVirtualComplete) partialEval(iter unifyIterator) error { + + for _, rule := range e.ir.Rules { + child := e.e.child(rule.Body) + child.traceEnter(rule) + + err := child.eval(func(child *eval) error { + child.traceExit(rule) + term, termbindings := child.bindings.apply(rule.Head.Value) + + err := e.evalTerm(iter, term, termbindings) + if err != nil { + return err + } + + child.traceRedo(rule) + return nil + }) + + if err != nil { + return err + } + } + + return nil +} + +func (e evalVirtualComplete) partialEvalSupport(iter unifyIterator) error { + + path := e.e.namespaceRef(e.plugged[:e.pos+1]) + term := ast.NewTerm(e.e.namespaceRef(e.ref)) + + if !e.e.saveSupport.Exists(path) { + + for i := range e.ir.Rules { + err := e.partialEvalSupportRule(e.ir.Rules[i], path) + if err != nil { + return err + } + } + + if e.ir.Default != nil { + err := e.partialEvalSupportRule(e.ir.Default, path) + if err != nil { + return err + } + } + } + + return e.e.saveUnify(term, e.rterm, e.bindings, e.rbindings, iter) +} + +func (e evalVirtualComplete) partialEvalSupportRule(rule *ast.Rule, path ast.Ref) error { + + child := e.e.child(rule.Body) + child.traceEnter(rule) + + e.e.saveStack.PushQuery(nil) + + err := child.eval(func(child *eval) error { + child.traceExit(rule) + + current := e.e.saveStack.PopQuery() + plugged := current.Plug(e.e.caller.bindings) + // Skip this rule body if it fails to type-check. + // Type-checking failure means the rule body will never succeed. + if e.e.compiler.PassesTypeCheck(plugged) { + head := ast.NewHead(rule.Head.Name, nil, child.bindings.PlugNamespaced(rule.Head.Value, e.e.caller.bindings)) + + if !e.e.inliningControl.shallow { + cp := copypropagation.New(head.Vars()). + WithEnsureNonEmptyBody(true). + WithCompiler(e.e.compiler) + plugged = applyCopyPropagation(cp, e.e.instr, plugged) + } + + e.e.saveSupport.Insert(path, &ast.Rule{ + Head: head, + Body: plugged, + Default: rule.Default, + }) + } + child.traceRedo(rule) + e.e.saveStack.PushQuery(current) + return nil + }) + e.e.saveStack.PopQuery() + return err +} + +func (e evalVirtualComplete) evalTerm(iter unifyIterator, term *ast.Term, termbindings *bindings) error { + eval := evalTerm{ + e: e.e, + ref: e.ref, + pos: e.pos + 1, + bindings: e.bindings, + term: term, + termbindings: termbindings, + rterm: e.rterm, + rbindings: e.rbindings, + } + return eval.eval(iter) +} + +type evalTerm struct { + e *eval + ref ast.Ref + pos int + bindings *bindings + term *ast.Term + termbindings *bindings + rterm *ast.Term + rbindings *bindings +} + +func (e evalTerm) eval(iter unifyIterator) error { + + if len(e.ref) == e.pos { + return e.e.biunify(e.term, e.rterm, e.termbindings, e.rbindings, iter) + } + + if e.e.saveSet.Contains(e.term, e.termbindings) { + return e.save(iter) + } + + plugged := e.bindings.Plug(e.ref[e.pos]) + + if plugged.IsGround() { + return e.next(iter, plugged) + } + + return e.enumerate(iter) +} + +func (e evalTerm) next(iter unifyIterator, plugged *ast.Term) error { + + term, bindings := e.get(plugged) + if term == nil { + return nil + } + + cpy := e + cpy.term = term + cpy.termbindings = bindings + cpy.pos++ + return cpy.eval(iter) +} + +func (e evalTerm) enumerate(iter unifyIterator) error { + + switch v := e.term.Value.(type) { + case *ast.Array: + for i := 0; i < v.Len(); i++ { + k := ast.IntNumberTerm(i) + err := e.e.biunify(k, e.ref[e.pos], e.bindings, e.bindings, func() error { + return e.next(iter, k) + }) + if err != nil { + return err + } + } + case ast.Object: + return v.Iter(func(k, _ *ast.Term) error { + return e.e.biunify(k, e.ref[e.pos], e.termbindings, e.bindings, func() error { + return e.next(iter, e.termbindings.Plug(k)) + }) + }) + case ast.Set: + return v.Iter(func(elem *ast.Term) error { + return e.e.biunify(elem, e.ref[e.pos], e.termbindings, e.bindings, func() error { + return e.next(iter, e.termbindings.Plug(elem)) + }) + }) + } + + return nil +} + +func (e evalTerm) get(plugged *ast.Term) (*ast.Term, *bindings) { + switch v := e.term.Value.(type) { + case ast.Set: + if v.IsGround() { + if v.Contains(plugged) { + return e.termbindings.apply(plugged) + } + } else { + var t *ast.Term + var b *bindings + stop := v.Until(func(elem *ast.Term) bool { + if e.termbindings.Plug(elem).Equal(plugged) { + t, b = e.termbindings.apply(plugged) + return true + } + return false + }) + if stop { + return t, b + } + } + case ast.Object: + if v.IsGround() { + term := v.Get(plugged) + if term != nil { + return e.termbindings.apply(term) + } + } else { + var t *ast.Term + var b *bindings + stop := v.Until(func(k, v *ast.Term) bool { + if e.termbindings.Plug(k).Equal(plugged) { + t, b = e.termbindings.apply(v) + return true + } + return false + }) + if stop { + return t, b + } + } + case *ast.Array: + term := v.Get(plugged) + if term != nil { + return e.termbindings.apply(term) + } + } + return nil, nil +} + +func (e evalTerm) save(iter unifyIterator) error { + + v := e.e.generateVar(fmt.Sprintf("ref_%d", e.e.genvarid)) + e.e.genvarid++ + + return e.e.biunify(e.term, v, e.termbindings, e.bindings, func() error { + + suffix := e.ref[e.pos:] + ref := make(ast.Ref, len(suffix)+1) + ref[0] = v + for i := 0; i < len(suffix); i++ { + ref[i+1] = suffix[i] + } + + return e.e.biunify(ast.NewTerm(ref), e.rterm, e.bindings, e.rbindings, iter) + }) + +} + +func (e *eval) comprehensionIndex(term *ast.Term) *ast.ComprehensionIndex { + if e.queryCompiler != nil { + return e.queryCompiler.ComprehensionIndex(term) + } + return e.compiler.ComprehensionIndex(term) +} + +func (e *eval) namespaceRef(ref ast.Ref) ast.Ref { + if e.skipSaveNamespace { + return ref.Copy() + } + return ref.Insert(e.saveNamespace, 1) +} + +type savePair struct { + term *ast.Term + b *bindings +} + +func getSavePairsFromExpr(declArgsLen int, x *ast.Expr, b *bindings, result []savePair) []savePair { + switch terms := x.Terms.(type) { + case *ast.Term: + return getSavePairsFromTerm(terms, b, result) + case []*ast.Term: + if x.IsEquality() { + return getSavePairsFromTerm(terms[2], b, getSavePairsFromTerm(terms[1], b, result)) + } + if declArgsLen == len(terms)-2 { + return getSavePairsFromTerm(terms[len(terms)-1], b, result) + } + } + return result +} + +func getSavePairsFromTerm(x *ast.Term, b *bindings, result []savePair) []savePair { + if _, ok := x.Value.(ast.Var); ok { + result = append(result, savePair{x, b}) + return result + } + vis := ast.NewVarVisitor().WithParams(ast.VarVisitorParams{ + SkipClosures: true, + SkipRefHead: true, + }) + vis.Walk(x) + for v := range vis.Vars() { + y, next := b.apply(ast.NewTerm(v)) + result = getSavePairsFromTerm(y, next, result) + } + return result +} + +func applyCopyPropagation(p *copypropagation.CopyPropagator, instr *Instrumentation, body ast.Body) ast.Body { + instr.startTimer(partialOpCopyPropagation) + result := p.Apply(body) + instr.stopTimer(partialOpCopyPropagation) + return result +} + +func nonGroundKeys(a ast.Object) bool { + return a.Until(func(k, _ *ast.Term) bool { + return !k.IsGround() + }) +} + +func plugKeys(a ast.Object, b *bindings) ast.Object { + plugged, _ := a.Map(func(k, v *ast.Term) (*ast.Term, *ast.Term, error) { + return b.Plug(k), v, nil + }) + return plugged +} + +func canInlineNegation(safe ast.VarSet, queries []ast.Body) bool { + + size := 1 + vis := newNestedCheckVisitor() + + for _, query := range queries { + size *= len(query) + for _, expr := range query { + if containsNestedRefOrCall(vis, expr) { + // Expressions containing nested refs or calls cannot be trivially negated + // because the semantics would change. For example, the complement of `not f(input.x)` + // is _not_ `f(input.x)`--it is `not input.x` OR `f(input.x)`. + // + // NOTE(tsandall): Since this would require the complement function to undo the + // copy propagation optimization, just bail out here. If this becomes a problem + // in the future, we can handle more cases. + return false + } + if !expr.Negated { + // Positive expressions containing variables cannot be trivially negated + // because they become unsafe (e.g., "x = 1" negated is "not x = 1" making x + // unsafe.) We check if the vars in the expr are already safe. + vis := ast.NewVarVisitor().WithParams(ast.VarVisitorParams{ + SkipRefCallHead: true, + SkipClosures: true, + }) + vis.Walk(expr) + unsafe := vis.Vars().Diff(safe).Diff(ast.ReservedVars) + if len(unsafe) > 0 { + return false + } + } + } + } + + // NOTE(tsandall): this limit is arbitrary–it's only in place to prevent the + // partial evaluation result from blowing up. In the future, we could make this + // configurable or do something more clever. + return size <= 16 +} + +type nestedCheckVisitor struct { + vis *ast.GenericVisitor + found bool +} + +func newNestedCheckVisitor() *nestedCheckVisitor { + v := &nestedCheckVisitor{} + v.vis = ast.NewGenericVisitor(v.visit) + return v +} + +func (v *nestedCheckVisitor) visit(x interface{}) bool { + switch x.(type) { + case ast.Ref, ast.Call: + v.found = true + } + return v.found +} + +func containsNestedRefOrCall(vis *nestedCheckVisitor, expr *ast.Expr) bool { + + if expr.IsEquality() { + for _, term := range expr.Operands() { + if containsNestedRefOrCallInTerm(vis, term) { + return true + } + } + return false + } + + if expr.IsCall() { + for _, term := range expr.Operands() { + vis.vis.Walk(term) + if vis.found { + return true + } + } + return false + } + + return containsNestedRefOrCallInTerm(vis, expr.Terms.(*ast.Term)) +} + +func containsNestedRefOrCallInTerm(vis *nestedCheckVisitor, term *ast.Term) bool { + switch v := term.Value.(type) { + case ast.Ref: + for i := 1; i < len(v); i++ { + vis.vis.Walk(v[i]) + if vis.found { + return true + } + } + return false + default: + vis.vis.Walk(v) + if vis.found { + return true + } + return false + } +} + +func complementedCartesianProduct(queries []ast.Body, idx int, curr ast.Body, iter func(ast.Body) error) error { + if idx == len(queries) { + return iter(curr) + } + for _, expr := range queries[idx] { + curr = append(curr, expr.Complement()) + if err := complementedCartesianProduct(queries, idx+1, curr, iter); err != nil { + return err + } + curr = curr[:len(curr)-1] + } + return nil +} + +func isInputRef(term *ast.Term) bool { + if ref, ok := term.Value.(ast.Ref); ok { + if ref.HasPrefix(ast.InputRootRef) { + return true + } + } + return false +} + +func isDataRef(term *ast.Term) bool { + if ref, ok := term.Value.(ast.Ref); ok { + if ref.HasPrefix(ast.DefaultRootRef) { + return true + } + } + return false +} + +func merge(a, b ast.Value) (ast.Value, bool) { + aObj, ok1 := a.(ast.Object) + bObj, ok2 := b.(ast.Object) + + if ok1 && ok2 { + return mergeObjects(aObj, bObj) + } + + // nothing to merge, a wins + return a, true +} + +// mergeObjects returns a new Object containing the non-overlapping keys of +// the objA and objB. If there are overlapping keys between objA and objB, +// the values of associated with the keys are merged. Only +// objects can be merged with other objects. If the values cannot be merged, +// objB value will be overwritten by objA value. +func mergeObjects(objA, objB ast.Object) (result ast.Object, ok bool) { + result = ast.NewObject() + stop := objA.Until(func(k, v *ast.Term) bool { + if v2 := objB.Get(k); v2 == nil { + result.Insert(k, v) + } else { + obj1, ok1 := v.Value.(ast.Object) + obj2, ok2 := v2.Value.(ast.Object) + + if !ok1 || !ok2 { + result.Insert(k, v) + return false + } + obj3, ok := mergeObjects(obj1, obj2) + if !ok { + return true + } + result.Insert(k, ast.NewTerm(obj3)) + } + return false + }) + if stop { + return nil, false + } + objB.Foreach(func(k, v *ast.Term) { + if v2 := objA.Get(k); v2 == nil { + result.Insert(k, v) + } + }) + return result, true +} + +func refContainsNonScalar(ref ast.Ref) bool { + for _, term := range ref[1:] { + if !ast.IsScalar(term.Value) { + return true + } + } + return false +} + +func suppressEarlyExit(err error) error { + ee, ok := err.(*earlyExitError) + if !ok { + return err + } + return ee.prev // nil if we're done +} diff --git a/vendor/github.com/open-policy-agent/opa/topdown/glob.go b/vendor/github.com/open-policy-agent/opa/topdown/glob.go new file mode 100644 index 00000000..98052a0c --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/topdown/glob.go @@ -0,0 +1,65 @@ +package topdown + +import ( + "fmt" + "sync" + + "github.com/gobwas/glob" + + "github.com/open-policy-agent/opa/ast" + "github.com/open-policy-agent/opa/topdown/builtins" +) + +var globCacheLock = sync.Mutex{} +var globCache map[string]glob.Glob + +func builtinGlobMatch(a, b, c ast.Value) (ast.Value, error) { + pattern, err := builtins.StringOperand(a, 1) + if err != nil { + return nil, err + } + + delimiters, err := builtins.RuneSliceOperand(b, 2) + if err != nil { + return nil, err + } + + if len(delimiters) == 0 { + delimiters = []rune{'.'} + } + + match, err := builtins.StringOperand(c, 3) + if err != nil { + return nil, err + } + + id := fmt.Sprintf("%s-%v", pattern, delimiters) + + globCacheLock.Lock() + defer globCacheLock.Unlock() + p, ok := globCache[id] + if !ok { + var err error + if p, err = glob.Compile(string(pattern), delimiters...); err != nil { + return nil, err + } + globCache[id] = p + } + + return ast.Boolean(p.Match(string(match))), nil +} + +func builtinGlobQuoteMeta(a ast.Value) (ast.Value, error) { + pattern, err := builtins.StringOperand(a, 1) + if err != nil { + return nil, err + } + + return ast.String(glob.QuoteMeta(string(pattern))), nil +} + +func init() { + globCache = map[string]glob.Glob{} + RegisterFunctionalBuiltin3(ast.GlobMatch.Name, builtinGlobMatch) + RegisterFunctionalBuiltin1(ast.GlobQuoteMeta.Name, builtinGlobQuoteMeta) +} diff --git a/vendor/github.com/open-policy-agent/opa/topdown/http.go b/vendor/github.com/open-policy-agent/opa/topdown/http.go new file mode 100644 index 00000000..c8205f18 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/topdown/http.go @@ -0,0 +1,1367 @@ +// Copyright 2018 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package topdown + +import ( + "bytes" + "context" + "crypto/tls" + "crypto/x509" + "encoding/json" + "fmt" + "io/ioutil" + "math" + "net" + "net/http" + "net/url" + "os" + "runtime" + "strconv" + "strings" + "time" + + "github.com/open-policy-agent/opa/ast" + "github.com/open-policy-agent/opa/internal/version" + "github.com/open-policy-agent/opa/topdown/builtins" + "github.com/open-policy-agent/opa/topdown/cache" + "github.com/open-policy-agent/opa/tracing" + "github.com/open-policy-agent/opa/util" +) + +type cachingMode string + +const ( + defaultHTTPRequestTimeoutEnv = "HTTP_SEND_TIMEOUT" + defaultCachingMode cachingMode = "serialized" + cachingModeDeserialized cachingMode = "deserialized" +) + +var defaultHTTPRequestTimeout = time.Second * 5 + +var allowedKeyNames = [...]string{ + "method", + "url", + "body", + "enable_redirect", + "force_json_decode", + "headers", + "raw_body", + "tls_use_system_certs", + "tls_ca_cert", + "tls_ca_cert_file", + "tls_ca_cert_env_variable", + "tls_client_cert", + "tls_client_cert_file", + "tls_client_cert_env_variable", + "tls_client_key", + "tls_client_key_file", + "tls_client_key_env_variable", + "tls_insecure_skip_verify", + "tls_server_name", + "timeout", + "cache", + "force_cache", + "force_cache_duration_seconds", + "raise_error", + "caching_mode", +} + +var ( + allowedKeys = ast.NewSet() + requiredKeys = ast.NewSet(ast.StringTerm("method"), ast.StringTerm("url")) + httpSendLatencyMetricKey = "rego_builtin_" + strings.ReplaceAll(ast.HTTPSend.Name, ".", "_") + httpSendInterQueryCacheHits = httpSendLatencyMetricKey + "_interquery_cache_hits" +) + +type httpSendKey string + +const ( + // httpSendBuiltinCacheKey is the key in the builtin context cache that + // points to the http.send() specific cache resides at. + httpSendBuiltinCacheKey httpSendKey = "HTTP_SEND_CACHE_KEY" + + // HTTPSendInternalErr represents a runtime evaluation error. + HTTPSendInternalErr string = "eval_http_send_internal_error" + + // HTTPSendNetworkErr represents a network error. + HTTPSendNetworkErr string = "eval_http_send_network_error" +) + +func builtinHTTPSend(bctx BuiltinContext, args []*ast.Term, iter func(*ast.Term) error) error { + req, err := validateHTTPRequestOperand(args[0], 1) + if err != nil { + return handleBuiltinErr(ast.HTTPSend.Name, bctx.Location, err) + } + + raiseError, err := getRaiseErrorValue(req) + if err != nil { + return handleBuiltinErr(ast.HTTPSend.Name, bctx.Location, err) + } + + result, err := getHTTPResponse(bctx, req) + if err != nil { + if raiseError { + return handleHTTPSendErr(bctx, err) + } + + obj := ast.NewObject() + obj.Insert(ast.StringTerm("status_code"), ast.IntNumberTerm(0)) + + errObj := ast.NewObject() + + switch err.(type) { + case *url.Error: + errObj.Insert(ast.StringTerm("code"), ast.StringTerm(HTTPSendNetworkErr)) + default: + errObj.Insert(ast.StringTerm("code"), ast.StringTerm(HTTPSendInternalErr)) + } + + errObj.Insert(ast.StringTerm("message"), ast.StringTerm(err.Error())) + obj.Insert(ast.StringTerm("error"), ast.NewTerm(errObj)) + + result = ast.NewTerm(obj) + } + return iter(result) +} + +func getHTTPResponse(bctx BuiltinContext, req ast.Object) (*ast.Term, error) { + + bctx.Metrics.Timer(httpSendLatencyMetricKey).Start() + + reqExecutor, err := newHTTPRequestExecutor(bctx, req) + if err != nil { + return nil, err + } + + // check if cache already has a response for this query + resp, err := reqExecutor.CheckCache() + if err != nil { + return nil, err + } + + if resp == nil { + httpResp, err := reqExecutor.ExecuteHTTPRequest() + if err != nil { + return nil, err + } + defer util.Close(httpResp) + // add result to cache + resp, err = reqExecutor.InsertIntoCache(httpResp) + if err != nil { + return nil, err + } + } + + bctx.Metrics.Timer(httpSendLatencyMetricKey).Stop() + + return ast.NewTerm(resp), nil +} + +func init() { + createAllowedKeys() + initDefaults() + RegisterBuiltinFunc(ast.HTTPSend.Name, builtinHTTPSend) +} + +func handleHTTPSendErr(bctx BuiltinContext, err error) error { + // Return HTTP client timeout errors in a generic error message to avoid confusion about what happened. + // Do not do this if the builtin context was cancelled and is what caused the request to stop. + if urlErr, ok := err.(*url.Error); ok && urlErr.Timeout() && bctx.Context.Err() == nil { + err = fmt.Errorf("%s %s: request timed out", urlErr.Op, urlErr.URL) + } + if err := bctx.Context.Err(); err != nil { + return Halt{ + Err: &Error{ + Code: CancelErr, + Message: fmt.Sprintf("http.send: timed out (%s)", err.Error()), + }, + } + } + return handleBuiltinErr(ast.HTTPSend.Name, bctx.Location, err) +} + +func initDefaults() { + timeoutDuration := os.Getenv(defaultHTTPRequestTimeoutEnv) + if timeoutDuration != "" { + var err error + defaultHTTPRequestTimeout, err = time.ParseDuration(timeoutDuration) + if err != nil { + // If it is set to something not valid don't let the process continue in a state + // that will almost definitely give unexpected results by having it set at 0 + // which means no timeout.. + // This environment variable isn't considered part of the public API. + // TODO(patrick-east): Remove the environment variable + panic(fmt.Sprintf("invalid value for HTTP_SEND_TIMEOUT: %s", err)) + } + } +} + +func validateHTTPRequestOperand(term *ast.Term, pos int) (ast.Object, error) { + + obj, err := builtins.ObjectOperand(term.Value, pos) + if err != nil { + return nil, err + } + + requestKeys := ast.NewSet(obj.Keys()...) + + invalidKeys := requestKeys.Diff(allowedKeys) + if invalidKeys.Len() != 0 { + return nil, builtins.NewOperandErr(pos, "invalid request parameters(s): %v", invalidKeys) + } + + missingKeys := requiredKeys.Diff(requestKeys) + if missingKeys.Len() != 0 { + return nil, builtins.NewOperandErr(pos, "missing required request parameters(s): %v", missingKeys) + } + + return obj, nil + +} + +// canonicalizeHeaders returns a copy of the headers where the keys are in +// canonical HTTP form. +func canonicalizeHeaders(headers map[string]interface{}) map[string]interface{} { + canonicalized := map[string]interface{}{} + + for k, v := range headers { + canonicalized[http.CanonicalHeaderKey(k)] = v + } + + return canonicalized +} + +// useSocket examines the url for "unix://" and returns a *http.Transport with +// a DialContext that opens a socket (specified in the http call). +// The url is expected to contain socket=/path/to/socket (url encoded) +// Ex. "unix:localhost/end/point?socket=%2Ftmp%2Fhttp.sock" +func useSocket(rawURL string, tlsConfig *tls.Config) (bool, string, *http.Transport) { + u, err := url.Parse(rawURL) + if err != nil { + return false, "", nil + } + + if u.Scheme != "unix" || u.RawQuery == "" { + return false, rawURL, nil + } + + // Get the path to the socket + v, err := url.ParseQuery(u.RawQuery) + if err != nil { + return false, rawURL, nil + } + + tr := http.DefaultTransport.(*http.Transport).Clone() + tr.DialContext = func(ctx context.Context, network, addr string) (net.Conn, error) { + return http.DefaultTransport.(*http.Transport).DialContext(ctx, u.Scheme, v.Get("socket")) + } + tr.TLSClientConfig = tlsConfig + tr.DisableKeepAlives = true + + rawURL = strings.Replace(rawURL, "unix:", "http:", 1) + return true, rawURL, tr +} + +func verifyHost(bctx BuiltinContext, host string) error { + if bctx.Capabilities == nil || bctx.Capabilities.AllowNet == nil { + return nil + } + + for _, allowed := range bctx.Capabilities.AllowNet { + if allowed == host { + return nil + } + } + + return fmt.Errorf("unallowed host: %s", host) +} + +func verifyURLHost(bctx BuiltinContext, unverifiedURL string) error { + // Eager return to avoid unnecessary URL parsing + if bctx.Capabilities == nil || bctx.Capabilities.AllowNet == nil { + return nil + } + + parsedURL, err := url.Parse(unverifiedURL) + if err != nil { + return err + } + + host := strings.Split(parsedURL.Host, ":")[0] + + return verifyHost(bctx, host) +} + +func createHTTPRequest(bctx BuiltinContext, obj ast.Object) (*http.Request, *http.Client, error) { + var url string + var method string + + // Additional CA certificates loading options. + var tlsCaCert []byte + var tlsCaCertEnvVar string + var tlsCaCertFile string + + // Client TLS certificate and key options. Each input source + // comes in a matched pair. + var tlsClientCert []byte + var tlsClientKey []byte + + var tlsClientCertEnvVar string + var tlsClientKeyEnvVar string + + var tlsClientCertFile string + var tlsClientKeyFile string + + var tlsServerName string + var body *bytes.Buffer + var rawBody *bytes.Buffer + var enableRedirect bool + var tlsUseSystemCerts *bool + var tlsConfig tls.Config + var customHeaders map[string]interface{} + var tlsInsecureSkipVerify bool + var timeout = defaultHTTPRequestTimeout + + for _, val := range obj.Keys() { + key, err := ast.JSON(val.Value) + if err != nil { + return nil, nil, err + } + + key = key.(string) + + var strVal string + + if s, ok := obj.Get(val).Value.(ast.String); ok { + strVal = strings.Trim(string(s), "\"") + } else { + // Most parameters are strings, so consolidate the type checking. + switch key { + case "method", + "url", + "raw_body", + "tls_ca_cert", + "tls_ca_cert_file", + "tls_ca_cert_env_variable", + "tls_client_cert", + "tls_client_cert_file", + "tls_client_cert_env_variable", + "tls_client_key", + "tls_client_key_file", + "tls_client_key_env_variable", + "tls_server_name": + return nil, nil, fmt.Errorf("%q must be a string", key) + } + } + + switch key { + case "method": + method = strings.ToUpper(strVal) + case "url": + err := verifyURLHost(bctx, strVal) + if err != nil { + return nil, nil, err + } + url = strVal + case "enable_redirect": + enableRedirect, err = strconv.ParseBool(obj.Get(val).String()) + if err != nil { + return nil, nil, err + } + case "body": + bodyVal := obj.Get(val).Value + bodyValInterface, err := ast.JSON(bodyVal) + if err != nil { + return nil, nil, err + } + + bodyValBytes, err := json.Marshal(bodyValInterface) + if err != nil { + return nil, nil, err + } + body = bytes.NewBuffer(bodyValBytes) + case "raw_body": + rawBody = bytes.NewBuffer([]byte(strVal)) + case "tls_use_system_certs": + tempTLSUseSystemCerts, err := strconv.ParseBool(obj.Get(val).String()) + if err != nil { + return nil, nil, err + } + tlsUseSystemCerts = &tempTLSUseSystemCerts + case "tls_ca_cert": + tlsCaCert = []byte(strVal) + case "tls_ca_cert_file": + tlsCaCertFile = strVal + case "tls_ca_cert_env_variable": + tlsCaCertEnvVar = strVal + case "tls_client_cert": + tlsClientCert = []byte(strVal) + case "tls_client_cert_file": + tlsClientCertFile = strVal + case "tls_client_cert_env_variable": + tlsClientCertEnvVar = strVal + case "tls_client_key": + tlsClientKey = []byte(strVal) + case "tls_client_key_file": + tlsClientKeyFile = strVal + case "tls_client_key_env_variable": + tlsClientKeyEnvVar = strVal + case "tls_server_name": + tlsServerName = strVal + case "headers": + headersVal := obj.Get(val).Value + headersValInterface, err := ast.JSON(headersVal) + if err != nil { + return nil, nil, err + } + var ok bool + customHeaders, ok = headersValInterface.(map[string]interface{}) + if !ok { + return nil, nil, fmt.Errorf("invalid type for headers key") + } + case "tls_insecure_skip_verify": + tlsInsecureSkipVerify, err = strconv.ParseBool(obj.Get(val).String()) + if err != nil { + return nil, nil, err + } + case "timeout": + timeout, err = parseTimeout(obj.Get(val).Value) + if err != nil { + return nil, nil, err + } + case "cache", "force_cache", "force_cache_duration_seconds", "force_json_decode", "raise_error", "caching_mode": // no-op + default: + return nil, nil, fmt.Errorf("invalid parameter %q", key) + } + } + + isTLS := false + client := &http.Client{ + Timeout: timeout, + } + + if tlsInsecureSkipVerify { + isTLS = true + tlsConfig.InsecureSkipVerify = tlsInsecureSkipVerify + } + + if len(tlsClientCert) > 0 && len(tlsClientKey) > 0 { + cert, err := tls.X509KeyPair(tlsClientCert, tlsClientKey) + if err != nil { + return nil, nil, err + } + + isTLS = true + tlsConfig.Certificates = append(tlsConfig.Certificates, cert) + } + + if tlsClientCertFile != "" && tlsClientKeyFile != "" { + cert, err := tls.LoadX509KeyPair(tlsClientCertFile, tlsClientKeyFile) + if err != nil { + return nil, nil, err + } + + isTLS = true + tlsConfig.Certificates = append(tlsConfig.Certificates, cert) + } + + if tlsClientCertEnvVar != "" && tlsClientKeyEnvVar != "" { + cert, err := tls.X509KeyPair( + []byte(os.Getenv(tlsClientCertEnvVar)), + []byte(os.Getenv(tlsClientKeyEnvVar))) + if err != nil { + return nil, nil, fmt.Errorf("cannot extract public/private key pair from envvars %q, %q: %w", + tlsClientCertEnvVar, tlsClientKeyEnvVar, err) + } + + isTLS = true + tlsConfig.Certificates = append(tlsConfig.Certificates, cert) + } + + // Use system certs if no CA cert is provided + // or system certs flag is not set + if len(tlsCaCert) == 0 && tlsCaCertFile == "" && tlsCaCertEnvVar == "" && tlsUseSystemCerts == nil { + trueValue := true + tlsUseSystemCerts = &trueValue + } + + // Check the system certificates config first so that we + // load additional certificated into the correct pool. + if tlsUseSystemCerts != nil && *tlsUseSystemCerts && runtime.GOOS != "windows" { + pool, err := x509.SystemCertPool() + if err != nil { + return nil, nil, err + } + + isTLS = true + tlsConfig.RootCAs = pool + } + + if len(tlsCaCert) != 0 { + tlsCaCert = bytes.Replace(tlsCaCert, []byte("\\n"), []byte("\n"), -1) + pool, err := addCACertsFromBytes(tlsConfig.RootCAs, []byte(tlsCaCert)) + if err != nil { + return nil, nil, err + } + + isTLS = true + tlsConfig.RootCAs = pool + } + + if tlsCaCertFile != "" { + pool, err := addCACertsFromFile(tlsConfig.RootCAs, tlsCaCertFile) + if err != nil { + return nil, nil, err + } + + isTLS = true + tlsConfig.RootCAs = pool + } + + if tlsCaCertEnvVar != "" { + pool, err := addCACertsFromEnv(tlsConfig.RootCAs, tlsCaCertEnvVar) + if err != nil { + return nil, nil, err + } + + isTLS = true + tlsConfig.RootCAs = pool + } + + if isTLS { + if ok, parsedURL, tr := useSocket(url, &tlsConfig); ok { + client.Transport = tr + url = parsedURL + } else { + tr := http.DefaultTransport.(*http.Transport).Clone() + tr.TLSClientConfig = &tlsConfig + tr.DisableKeepAlives = true + client.Transport = tr + } + } else { + if ok, parsedURL, tr := useSocket(url, nil); ok { + client.Transport = tr + url = parsedURL + } + } + + // check if redirects are enabled + if !enableRedirect { + client.CheckRedirect = func(*http.Request, []*http.Request) error { + return http.ErrUseLastResponse + } + } + + if rawBody != nil { + body = rawBody + } else if body == nil { + body = bytes.NewBufferString("") + } + + // create the http request, use the builtin context's context to ensure + // the request is cancelled if evaluation is cancelled. + req, err := http.NewRequest(method, url, body) + if err != nil { + return nil, nil, err + } + + req = req.WithContext(bctx.Context) + + // Add custom headers + if len(customHeaders) != 0 { + customHeaders = canonicalizeHeaders(customHeaders) + + for k, v := range customHeaders { + header, ok := v.(string) + if !ok { + return nil, nil, fmt.Errorf("invalid type for headers value %q", v) + } + + req.Header.Add(k, header) + } + + // Don't overwrite or append to one that was set in the custom headers + if _, hasUA := customHeaders["User-Agent"]; !hasUA { + req.Header.Add("User-Agent", version.UserAgent) + } + + // If the caller specifies the Host header, use it for the HTTP + // request host and the TLS server name. + if host, hasHost := customHeaders["Host"]; hasHost { + host := host.(string) // We already checked that it's a string. + req.Host = host + + // Only default the ServerName if the caller has + // specified the host. If we don't specify anything, + // Go will default to the target hostname. This name + // is not the same as the default that Go populates + // `req.Host` with, which is why we don't just set + // this unconditionally. + tlsConfig.ServerName = host + } + } + + if tlsServerName != "" { + tlsConfig.ServerName = tlsServerName + } + + if len(bctx.DistributedTracingOpts) > 0 { + client.Transport = tracing.NewTransport(client.Transport, bctx.DistributedTracingOpts) + } + + return req, client, nil +} + +func executeHTTPRequest(req *http.Request, client *http.Client) (*http.Response, error) { + return client.Do(req) +} + +func isContentTypeJSON(header http.Header) bool { + return strings.Contains(header.Get("Content-Type"), "application/json") +} + +// In the BuiltinContext cache we only store a single entry that points to +// our ValueMap which is the "real" http.send() cache. +func getHTTPSendCache(bctx BuiltinContext) *ast.ValueMap { + raw, ok := bctx.Cache.Get(httpSendBuiltinCacheKey) + if !ok { + // Initialize if it isn't there + cache := ast.NewValueMap() + bctx.Cache.Put(httpSendBuiltinCacheKey, cache) + return cache + } + + cache, ok := raw.(*ast.ValueMap) + if !ok { + return nil + } + return cache +} + +// checkHTTPSendCache checks for the given key's value in the cache +func checkHTTPSendCache(bctx BuiltinContext, key ast.Object) ast.Value { + requestCache := getHTTPSendCache(bctx) + if requestCache == nil { + return nil + } + + return requestCache.Get(key) +} + +func insertIntoHTTPSendCache(bctx BuiltinContext, key ast.Object, value ast.Value) { + requestCache := getHTTPSendCache(bctx) + if requestCache == nil { + // Should never happen.. if it does just skip caching the value + return + } + requestCache.Put(key, value) +} + +// checkHTTPSendInterQueryCache checks for the given key's value in the inter-query cache +func (c *interQueryCache) checkHTTPSendInterQueryCache() (ast.Value, error) { + requestCache := c.bctx.InterQueryBuiltinCache + + value, found := requestCache.Get(c.key) + if !found { + return nil, nil + } + c.bctx.Metrics.Counter(httpSendInterQueryCacheHits).Incr() + var cachedRespData *interQueryCacheData + + switch v := value.(type) { + case *interQueryCacheValue: + var err error + cachedRespData, err = v.copyCacheData() + if err != nil { + return nil, err + } + case *interQueryCacheData: + cachedRespData = v + default: + return nil, nil + } + + headers, err := parseResponseHeaders(cachedRespData.Headers) + if err != nil { + return nil, err + } + + // check the freshness of the cached response + if isCachedResponseFresh(c.bctx, headers, c.forceCacheParams) { + return cachedRespData.formatToAST(c.forceJSONDecode) + } + + c.httpReq, c.httpClient, err = createHTTPRequest(c.bctx, c.key) + if err != nil { + return nil, handleHTTPSendErr(c.bctx, err) + } + + // check with the server if the stale response is still up-to-date. + // If server returns a new response (ie. status_code=200), update the cache with the new response + // If server returns an unmodified response (ie. status_code=304), update the headers for the existing response + result, modified, err := revalidateCachedResponse(c.httpReq, c.httpClient, headers) + requestCache.Delete(c.key) + if err != nil || result == nil { + return nil, err + } + + defer result.Body.Close() + + if !modified { + // update the headers in the cached response with their corresponding values from the 304 (Not Modified) response + for headerName, values := range result.Header { + cachedRespData.Headers.Del(headerName) + for _, v := range values { + cachedRespData.Headers.Add(headerName, v) + } + } + + cachingMode, err := getCachingMode(c.key) + if err != nil { + return nil, err + } + + var pcv cache.InterQueryCacheValue + + if cachingMode == defaultCachingMode { + pcv, err = cachedRespData.toCacheValue() + if err != nil { + return nil, err + } + } else { + pcv = cachedRespData + } + + c.bctx.InterQueryBuiltinCache.Insert(c.key, pcv) + + return cachedRespData.formatToAST(c.forceJSONDecode) + } + + newValue, respBody, err := formatHTTPResponseToAST(result, c.forceJSONDecode) + if err != nil { + return nil, err + } + + err = insertIntoHTTPSendInterQueryCache(c.bctx, c.key, result, respBody, c.forceCacheParams != nil) + if err != nil { + return nil, err + } + + return newValue, nil +} + +// insertIntoHTTPSendInterQueryCache inserts given key and value in the inter-query cache +func insertIntoHTTPSendInterQueryCache(bctx BuiltinContext, key ast.Value, resp *http.Response, respBody []byte, force bool) error { + if resp == nil || (!force && !canStore(resp.Header)) { + return nil + } + + requestCache := bctx.InterQueryBuiltinCache + + obj, ok := key.(ast.Object) + if !ok { + return fmt.Errorf("interface conversion error") + } + + cachingMode, err := getCachingMode(obj) + if err != nil { + return err + } + + var pcv cache.InterQueryCacheValue + + if cachingMode == defaultCachingMode { + pcv, err = newInterQueryCacheValue(resp, respBody) + } else { + pcv, err = newInterQueryCacheData(resp, respBody) + } + + if err != nil { + return err + } + + requestCache.Insert(key, pcv) + return nil +} + +func createAllowedKeys() { + for _, element := range allowedKeyNames { + allowedKeys.Add(ast.StringTerm(element)) + } +} + +func parseTimeout(timeoutVal ast.Value) (time.Duration, error) { + var timeout time.Duration + switch t := timeoutVal.(type) { + case ast.Number: + timeoutInt, ok := t.Int64() + if !ok { + return timeout, fmt.Errorf("invalid timeout number value %v, must be int64", timeoutVal) + } + return time.Duration(timeoutInt), nil + case ast.String: + // Support strings without a unit, treat them the same as just a number value (ns) + var err error + timeoutInt, err := strconv.ParseInt(string(t), 10, 64) + if err == nil { + return time.Duration(timeoutInt), nil + } + + // Try parsing it as a duration (requires a supported units suffix) + timeout, err = time.ParseDuration(string(t)) + if err != nil { + return timeout, fmt.Errorf("invalid timeout value %v: %s", timeoutVal, err) + } + return timeout, nil + default: + return timeout, builtins.NewOperandErr(1, "'timeout' must be one of {string, number} but got %s", ast.TypeName(t)) + } +} + +func getBoolValFromReqObj(req ast.Object, key *ast.Term) (bool, error) { + var b ast.Boolean + var ok bool + if v := req.Get(key); v != nil { + if b, ok = v.Value.(ast.Boolean); !ok { + return false, fmt.Errorf("invalid value for %v field", key.String()) + } + } + return bool(b), nil +} + +func getCachingMode(req ast.Object) (cachingMode, error) { + key := ast.StringTerm("caching_mode") + var s ast.String + var ok bool + if v := req.Get(key); v != nil { + if s, ok = v.Value.(ast.String); !ok { + return "", fmt.Errorf("invalid value for %v field", key.String()) + } + + switch cachingMode(s) { + case defaultCachingMode, cachingModeDeserialized: + return cachingMode(s), nil + default: + return "", fmt.Errorf("invalid value specified for %v field: %v", key.String(), string(s)) + } + } + return cachingMode(defaultCachingMode), nil +} + +type interQueryCacheValue struct { + Data []byte +} + +func newInterQueryCacheValue(resp *http.Response, respBody []byte) (*interQueryCacheValue, error) { + data, err := newInterQueryCacheData(resp, respBody) + if err != nil { + return nil, err + } + + b, err := json.Marshal(data) + if err != nil { + return nil, err + } + return &interQueryCacheValue{Data: b}, nil +} + +func (cb interQueryCacheValue) SizeInBytes() int64 { + return int64(len(cb.Data)) +} + +func (cb *interQueryCacheValue) copyCacheData() (*interQueryCacheData, error) { + var res interQueryCacheData + err := util.UnmarshalJSON(cb.Data, &res) + if err != nil { + return nil, err + } + return &res, nil +} + +type interQueryCacheData struct { + RespBody []byte + Status string + StatusCode int + Headers http.Header +} + +func newInterQueryCacheData(resp *http.Response, respBody []byte) (*interQueryCacheData, error) { + _, err := parseResponseHeaders(resp.Header) + if err != nil { + return nil, err + } + + cv := interQueryCacheData{RespBody: respBody, + Status: resp.Status, + StatusCode: resp.StatusCode, + Headers: resp.Header} + + return &cv, nil +} + +func (c *interQueryCacheData) formatToAST(forceJSONDecode bool) (ast.Value, error) { + return prepareASTResult(c.Headers, forceJSONDecode, c.RespBody, c.Status, c.StatusCode) +} + +func (c *interQueryCacheData) toCacheValue() (*interQueryCacheValue, error) { + b, err := json.Marshal(c) + if err != nil { + return nil, err + } + return &interQueryCacheValue{Data: b}, nil +} + +func (c *interQueryCacheData) SizeInBytes() int64 { + return 0 +} + +type responseHeaders struct { + date time.Time // origination date and time of response + cacheControl map[string]string // response cache-control header + maxAge deltaSeconds // max-age cache control directive + expires time.Time // date/time after which the response is considered stale + etag string // identifier for a specific version of the response + lastModified string // date and time response was last modified as per origin server +} + +// deltaSeconds specifies a non-negative integer, representing +// time in seconds: http://tools.ietf.org/html/rfc7234#section-1.2.1 +type deltaSeconds int32 + +func parseResponseHeaders(headers http.Header) (*responseHeaders, error) { + var err error + result := responseHeaders{} + + result.date, err = getResponseHeaderDate(headers) + if err != nil { + return nil, err + } + + result.cacheControl = parseCacheControlHeader(headers) + result.maxAge, err = parseMaxAgeCacheDirective(result.cacheControl) + if err != nil { + return nil, err + } + + result.expires = getResponseHeaderExpires(headers) + + result.etag = headers.Get("etag") + + result.lastModified = headers.Get("last-modified") + + return &result, nil +} + +func revalidateCachedResponse(req *http.Request, client *http.Client, headers *responseHeaders) (*http.Response, bool, error) { + etag := headers.etag + lastModified := headers.lastModified + + if etag == "" && lastModified == "" { + return nil, false, nil + } + + cloneReq := req.Clone(req.Context()) + + if etag != "" { + cloneReq.Header.Set("if-none-match", etag) + } + + if lastModified != "" { + cloneReq.Header.Set("if-modified-since", lastModified) + } + + response, err := client.Do(cloneReq) + if err != nil { + return nil, false, err + } + + switch response.StatusCode { + case http.StatusOK: + return response, true, nil + + case http.StatusNotModified: + return response, false, nil + } + util.Close(response) + return nil, false, nil +} + +func canStore(headers http.Header) bool { + ccHeaders := parseCacheControlHeader(headers) + + // Check "no-store" cache directive + // The "no-store" response directive indicates that a cache MUST NOT + // store any part of either the immediate request or response. + if _, ok := ccHeaders["no-store"]; ok { + return false + } + return true +} + +func isCachedResponseFresh(bctx BuiltinContext, headers *responseHeaders, cacheParams *forceCacheParams) bool { + if headers.date.IsZero() { + return false + } + + currentTime := getCurrentTime(bctx) + if currentTime.IsZero() { + return false + } + + currentAge := currentTime.Sub(headers.date) + + // The time.Sub operation uses wall clock readings and + // not monotonic clock readings as the parsed version of the response time + // does not contain monotonic clock readings. This can result in negative durations. + // Another scenario where a negative duration can occur, is when a server sets the Date + // response header. As per https://tools.ietf.org/html/rfc7231#section-7.1.1.2, + // an origin server MUST NOT send a Date header field if it does not + // have a clock capable of providing a reasonable approximation of the + // current instance in Coordinated Universal Time. + // Hence, consider the cached response as stale if a negative duration is encountered. + if currentAge < 0 { + return false + } + + if cacheParams != nil { + // override the cache directives set by the server + maxAgeDur := time.Second * time.Duration(cacheParams.forceCacheDurationSeconds) + if maxAgeDur > currentAge { + return true + } + } else { + // Check "max-age" cache directive. + // The "max-age" response directive indicates that the response is to be + // considered stale after its age is greater than the specified number + // of seconds. + if headers.maxAge != -1 { + maxAgeDur := time.Second * time.Duration(headers.maxAge) + if maxAgeDur > currentAge { + return true + } + } else { + // Check "Expires" header. + // Note: "max-age" if set, takes precedence over "Expires" + if headers.expires.Sub(headers.date) > currentAge { + return true + } + } + } + return false +} + +func getCurrentTime(bctx BuiltinContext) time.Time { + var current time.Time + + value, err := ast.JSON(bctx.Time.Value) + if err != nil { + return current + } + + valueNum, ok := value.(json.Number) + if !ok { + return current + } + + valueNumInt, err := valueNum.Int64() + if err != nil { + return current + } + + current = time.Unix(0, valueNumInt).UTC() + return current +} + +func parseCacheControlHeader(headers http.Header) map[string]string { + ccDirectives := map[string]string{} + ccHeader := headers.Get("cache-control") + + for _, part := range strings.Split(ccHeader, ",") { + part = strings.Trim(part, " ") + if part == "" { + continue + } + if strings.ContainsRune(part, '=') { + items := strings.Split(part, "=") + if len(items) != 2 { + continue + } + ccDirectives[strings.Trim(items[0], " ")] = strings.Trim(items[1], ",") + } else { + ccDirectives[part] = "" + } + } + + return ccDirectives +} + +func getResponseHeaderDate(headers http.Header) (date time.Time, err error) { + dateHeader := headers.Get("date") + if dateHeader == "" { + err = fmt.Errorf("no date header") + return + } + return http.ParseTime(dateHeader) +} + +func getResponseHeaderExpires(headers http.Header) time.Time { + expiresHeader := headers.Get("expires") + if expiresHeader == "" { + return time.Time{} + } + + date, err := http.ParseTime(expiresHeader) + if err != nil { + // servers can set `Expires: 0` which is an invalid date to indicate expired content + return time.Time{} + } + + return date +} + +// parseMaxAgeCacheDirective parses the max-age directive expressed in delta-seconds as per +// https://tools.ietf.org/html/rfc7234#section-1.2.1 +func parseMaxAgeCacheDirective(cc map[string]string) (deltaSeconds, error) { + maxAge, ok := cc["max-age"] + if !ok { + return deltaSeconds(-1), nil + } + + val, err := strconv.ParseUint(maxAge, 10, 32) + if err != nil { + if numError, ok := err.(*strconv.NumError); ok { + if numError.Err == strconv.ErrRange { + return deltaSeconds(math.MaxInt32), nil + } + } + return deltaSeconds(-1), err + } + + if val > math.MaxInt32 { + return deltaSeconds(math.MaxInt32), nil + } + return deltaSeconds(val), nil +} + +func formatHTTPResponseToAST(resp *http.Response, forceJSONDecode bool) (ast.Value, []byte, error) { + + resultRawBody, err := ioutil.ReadAll(resp.Body) + if err != nil { + return nil, nil, err + } + + resultObj, err := prepareASTResult(resp.Header, forceJSONDecode, resultRawBody, resp.Status, resp.StatusCode) + if err != nil { + return nil, nil, err + } + + return resultObj, resultRawBody, nil +} + +func prepareASTResult(headers http.Header, forceJSONDecode bool, body []byte, status string, statusCode int) (ast.Value, error) { + var resultBody interface{} + + // If the response body cannot be JSON decoded, + // an error will not be returned. Instead the "body" field + // in the result will be null. + if isContentTypeJSON(headers) || forceJSONDecode { + _ = util.UnmarshalJSON(body, &resultBody) + } + + result := make(map[string]interface{}) + result["status"] = status + result["status_code"] = statusCode + result["body"] = resultBody + result["raw_body"] = string(body) + result["headers"] = getResponseHeaders(headers) + + resultObj, err := ast.InterfaceToValue(result) + if err != nil { + return nil, err + } + + return resultObj, nil +} + +func getResponseHeaders(headers http.Header) map[string]interface{} { + respHeaders := map[string]interface{}{} + for headerName, values := range headers { + var respValues []interface{} + for _, v := range values { + respValues = append(respValues, v) + } + respHeaders[strings.ToLower(headerName)] = respValues + } + return respHeaders +} + +// httpRequestExecutor defines an interface for the http send cache +type httpRequestExecutor interface { + CheckCache() (ast.Value, error) + InsertIntoCache(value *http.Response) (ast.Value, error) + ExecuteHTTPRequest() (*http.Response, error) +} + +// newHTTPRequestExecutor returns a new HTTP request executor that wraps either an inter-query or +// intra-query cache implementation +func newHTTPRequestExecutor(bctx BuiltinContext, key ast.Object) (httpRequestExecutor, error) { + useInterQueryCache, forceCacheParams, err := useInterQueryCache(key) + if err != nil { + return nil, handleHTTPSendErr(bctx, err) + } + + if useInterQueryCache && bctx.InterQueryBuiltinCache != nil { + return newInterQueryCache(bctx, key, forceCacheParams) + } + return newIntraQueryCache(bctx, key) +} + +type interQueryCache struct { + bctx BuiltinContext + key ast.Object + httpReq *http.Request + httpClient *http.Client + forceJSONDecode bool + forceCacheParams *forceCacheParams +} + +func newInterQueryCache(bctx BuiltinContext, key ast.Object, forceCacheParams *forceCacheParams) (*interQueryCache, error) { + return &interQueryCache{bctx: bctx, key: key, forceCacheParams: forceCacheParams}, nil +} + +// CheckCache checks the cache for the value of the key set on this object +func (c *interQueryCache) CheckCache() (ast.Value, error) { + var err error + + c.forceJSONDecode, err = getBoolValFromReqObj(c.key, ast.StringTerm("force_json_decode")) + if err != nil { + return nil, handleHTTPSendErr(c.bctx, err) + } + + resp, err := c.checkHTTPSendInterQueryCache() + + // fallback to the http send cache if response not found in the inter-query cache or inter-query cache look-up results + // in an error + if resp == nil || err != nil { + return checkHTTPSendCache(c.bctx, c.key), nil + } + + return resp, err +} + +// InsertIntoCache inserts the key set on this object into the cache with the given value +func (c *interQueryCache) InsertIntoCache(value *http.Response) (ast.Value, error) { + result, respBody, err := formatHTTPResponseToAST(value, c.forceJSONDecode) + if err != nil { + return nil, handleHTTPSendErr(c.bctx, err) + } + + // fallback to the http send cache if error encountered while inserting response in inter-query cache + err = insertIntoHTTPSendInterQueryCache(c.bctx, c.key, value, respBody, c.forceCacheParams != nil) + if err != nil { + insertIntoHTTPSendCache(c.bctx, c.key, result) + } + return result, nil +} + +// ExecuteHTTPRequest executes a HTTP request +func (c *interQueryCache) ExecuteHTTPRequest() (*http.Response, error) { + var err error + c.httpReq, c.httpClient, err = createHTTPRequest(c.bctx, c.key) + if err != nil { + return nil, handleHTTPSendErr(c.bctx, err) + } + + return executeHTTPRequest(c.httpReq, c.httpClient) +} + +type intraQueryCache struct { + bctx BuiltinContext + key ast.Object +} + +func newIntraQueryCache(bctx BuiltinContext, key ast.Object) (*intraQueryCache, error) { + return &intraQueryCache{bctx: bctx, key: key}, nil +} + +// CheckCache checks the cache for the value of the key set on this object +func (c *intraQueryCache) CheckCache() (ast.Value, error) { + return checkHTTPSendCache(c.bctx, c.key), nil +} + +// InsertIntoCache inserts the key set on this object into the cache with the given value +func (c *intraQueryCache) InsertIntoCache(value *http.Response) (ast.Value, error) { + forceJSONDecode, err := getBoolValFromReqObj(c.key, ast.StringTerm("force_json_decode")) + if err != nil { + return nil, handleHTTPSendErr(c.bctx, err) + } + + result, _, err := formatHTTPResponseToAST(value, forceJSONDecode) + if err != nil { + return nil, handleHTTPSendErr(c.bctx, err) + } + + insertIntoHTTPSendCache(c.bctx, c.key, result) + return result, nil +} + +// ExecuteHTTPRequest executes a HTTP request +func (c *intraQueryCache) ExecuteHTTPRequest() (*http.Response, error) { + httpReq, httpClient, err := createHTTPRequest(c.bctx, c.key) + if err != nil { + return nil, handleHTTPSendErr(c.bctx, err) + } + return executeHTTPRequest(httpReq, httpClient) +} + +func useInterQueryCache(req ast.Object) (bool, *forceCacheParams, error) { + value, err := getBoolValFromReqObj(req, ast.StringTerm("cache")) + if err != nil { + return false, nil, err + } + + valueForceCache, err := getBoolValFromReqObj(req, ast.StringTerm("force_cache")) + if err != nil { + return false, nil, err + } + + if valueForceCache { + forceCacheParams, err := newForceCacheParams(req) + return true, forceCacheParams, err + } + + return value, nil, nil +} + +type forceCacheParams struct { + forceCacheDurationSeconds int32 +} + +func newForceCacheParams(req ast.Object) (*forceCacheParams, error) { + term := req.Get(ast.StringTerm("force_cache_duration_seconds")) + if term == nil { + return nil, fmt.Errorf("'force_cache' set but 'force_cache_duration_seconds' parameter is missing") + } + + forceCacheDurationSeconds := term.String() + + value, err := strconv.ParseInt(forceCacheDurationSeconds, 10, 32) + if err != nil { + return nil, err + } + + return &forceCacheParams{forceCacheDurationSeconds: int32(value)}, nil +} + +func getRaiseErrorValue(req ast.Object) (bool, error) { + result := ast.Boolean(true) + var ok bool + if v := req.Get(ast.StringTerm("raise_error")); v != nil { + if result, ok = v.Value.(ast.Boolean); !ok { + return false, fmt.Errorf("invalid value for raise_error field") + } + } + return bool(result), nil +} diff --git a/vendor/github.com/open-policy-agent/opa/topdown/input.go b/vendor/github.com/open-policy-agent/opa/topdown/input.go new file mode 100644 index 00000000..cb70aeb7 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/topdown/input.go @@ -0,0 +1,100 @@ +// Copyright 2016 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package topdown + +import ( + "fmt" + + "github.com/open-policy-agent/opa/ast" +) + +var errBadPath = fmt.Errorf("bad document path") + +func mergeTermWithValues(exist *ast.Term, pairs [][2]*ast.Term) (*ast.Term, error) { + + var result *ast.Term + var init bool + + for i, pair := range pairs { + + if err := ast.IsValidImportPath(pair[0].Value); err != nil { + return nil, errBadPath + } + + target := pair[0].Value.(ast.Ref) + + // Copy the value if subsequent pairs in the slice would modify it. + for j := i + 1; j < len(pairs); j++ { + other := pairs[j][0].Value.(ast.Ref) + if len(other) > len(target) && other.HasPrefix(target) { + pair[1] = pair[1].Copy() + break + } + } + + if len(target) == 1 { + result = pair[1] + init = true + } else { + if !init { + result = exist.Copy() + init = true + } + if result == nil { + result = ast.NewTerm(makeTree(target[1:], pair[1])) + } else { + node := result + done := false + for i := 1; i < len(target)-1 && !done; i++ { + obj, ok := node.Value.(ast.Object) + if !ok { + result = ast.NewTerm(makeTree(target[i:], pair[1])) + done = true + continue + } + if child := obj.Get(target[i]); !isObject(child) { + obj.Insert(target[i], ast.NewTerm(makeTree(target[i+1:], pair[1]))) + done = true + } else { // child is object + node = child + } + } + if !done { + if obj, ok := node.Value.(ast.Object); ok { + obj.Insert(target[len(target)-1], pair[1]) + } else { + result = ast.NewTerm(makeTree(target[len(target)-1:], pair[1])) + } + } + } + } + } + + if !init { + result = exist + } + + return result, nil +} + +// makeTree returns an object that represents a document where the value v is +// the leaf and elements in k represent intermediate objects. +func makeTree(k ast.Ref, v *ast.Term) ast.Object { + var obj ast.Object + for i := len(k) - 1; i >= 1; i-- { + obj = ast.NewObject(ast.Item(k[i], v)) + v = &ast.Term{Value: obj} + } + obj = ast.NewObject(ast.Item(k[0], v)) + return obj +} + +func isObject(x *ast.Term) bool { + if x == nil { + return false + } + _, ok := x.Value.(ast.Object) + return ok +} diff --git a/vendor/github.com/open-policy-agent/opa/topdown/instrumentation.go b/vendor/github.com/open-policy-agent/opa/topdown/instrumentation.go new file mode 100644 index 00000000..6eacc338 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/topdown/instrumentation.go @@ -0,0 +1,63 @@ +// Copyright 2018 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package topdown + +import "github.com/open-policy-agent/opa/metrics" + +const ( + evalOpPlug = "eval_op_plug" + evalOpResolve = "eval_op_resolve" + evalOpRuleIndex = "eval_op_rule_index" + evalOpBuiltinCall = "eval_op_builtin_call" + evalOpVirtualCacheHit = "eval_op_virtual_cache_hit" + evalOpVirtualCacheMiss = "eval_op_virtual_cache_miss" + evalOpBaseCacheHit = "eval_op_base_cache_hit" + evalOpBaseCacheMiss = "eval_op_base_cache_miss" + evalOpComprehensionCacheSkip = "eval_op_comprehension_cache_skip" + evalOpComprehensionCacheBuild = "eval_op_comprehension_cache_build" + evalOpComprehensionCacheHit = "eval_op_comprehension_cache_hit" + evalOpComprehensionCacheMiss = "eval_op_comprehension_cache_miss" + partialOpSaveUnify = "partial_op_save_unify" + partialOpSaveSetContains = "partial_op_save_set_contains" + partialOpSaveSetContainsRec = "partial_op_save_set_contains_rec" + partialOpCopyPropagation = "partial_op_copy_propagation" +) + +// Instrumentation implements helper functions to instrument query evaluation +// to diagnose performance issues. Instrumentation may be expensive in some +// cases, so it is disabled by default. +type Instrumentation struct { + m metrics.Metrics +} + +// NewInstrumentation returns a new Instrumentation object. Performance +// diagnostics recorded on this Instrumentation object will stored in m. +func NewInstrumentation(m metrics.Metrics) *Instrumentation { + return &Instrumentation{ + m: m, + } +} + +func (instr *Instrumentation) startTimer(name string) { + if instr == nil { + return + } + instr.m.Timer(name).Start() +} + +func (instr *Instrumentation) stopTimer(name string) { + if instr == nil { + return + } + delta := instr.m.Timer(name).Stop() + instr.m.Histogram(name).Update(delta) +} + +func (instr *Instrumentation) counterIncr(name string) { + if instr == nil { + return + } + instr.m.Counter(name).Incr() +} diff --git a/vendor/github.com/open-policy-agent/opa/topdown/json.go b/vendor/github.com/open-policy-agent/opa/topdown/json.go new file mode 100644 index 00000000..09a0ea6a --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/topdown/json.go @@ -0,0 +1,623 @@ +// Copyright 2019 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package topdown + +import ( + "fmt" + "strconv" + "strings" + + "github.com/open-policy-agent/opa/ast" + "github.com/open-policy-agent/opa/topdown/builtins" +) + +func builtinJSONRemove(_ BuiltinContext, operands []*ast.Term, iter func(*ast.Term) error) error { + + // Expect an object and a string or array/set of strings + _, err := builtins.ObjectOperand(operands[0].Value, 1) + if err != nil { + return err + } + + // Build a list of json pointers to remove + paths, err := getJSONPaths(operands[1].Value) + if err != nil { + return err + } + + newObj, err := jsonRemove(operands[0], ast.NewTerm(pathsToObject(paths))) + if err != nil { + return err + } + + if newObj == nil { + return nil + } + + return iter(newObj) +} + +// jsonRemove returns a new term that is the result of walking +// through a and omitting removing any values that are in b but +// have ast.Null values (ie leaf nodes for b). +func jsonRemove(a *ast.Term, b *ast.Term) (*ast.Term, error) { + if b == nil { + // The paths diverged, return a + return a, nil + } + + var bObj ast.Object + switch bValue := b.Value.(type) { + case ast.Object: + bObj = bValue + case ast.Null: + // Means we hit a leaf node on "b", dont add the value for a + return nil, nil + default: + // The paths diverged, return a + return a, nil + } + + switch aValue := a.Value.(type) { + case ast.String, ast.Number, ast.Boolean, ast.Null: + return a, nil + case ast.Object: + newObj := ast.NewObject() + err := aValue.Iter(func(k *ast.Term, v *ast.Term) error { + // recurse and add the diff of sub objects as needed + diffValue, err := jsonRemove(v, bObj.Get(k)) + if err != nil || diffValue == nil { + return err + } + newObj.Insert(k, diffValue) + return nil + }) + if err != nil { + return nil, err + } + return ast.NewTerm(newObj), nil + case ast.Set: + newSet := ast.NewSet() + err := aValue.Iter(func(v *ast.Term) error { + // recurse and add the diff of sub objects as needed + diffValue, err := jsonRemove(v, bObj.Get(v)) + if err != nil || diffValue == nil { + return err + } + newSet.Add(diffValue) + return nil + }) + if err != nil { + return nil, err + } + return ast.NewTerm(newSet), nil + case *ast.Array: + // When indexes are removed we shift left to close empty spots in the array + // as per the JSON patch spec. + newArray := ast.NewArray() + for i := 0; i < aValue.Len(); i++ { + v := aValue.Elem(i) + // recurse and add the diff of sub objects as needed + // Note: Keys in b will be strings for the index, eg path /a/1/b => {"a": {"1": {"b": null}}} + diffValue, err := jsonRemove(v, bObj.Get(ast.StringTerm(strconv.Itoa(i)))) + if err != nil { + return nil, err + } + if diffValue != nil { + newArray = newArray.Append(diffValue) + } + } + return ast.NewTerm(newArray), nil + default: + return nil, fmt.Errorf("invalid value type %T", a) + } +} + +func builtinJSONFilter(_ BuiltinContext, operands []*ast.Term, iter func(*ast.Term) error) error { + + // Ensure we have the right parameters, expect an object and a string or array/set of strings + obj, err := builtins.ObjectOperand(operands[0].Value, 1) + if err != nil { + return err + } + + // Build a list of filter strings + filters, err := getJSONPaths(operands[1].Value) + if err != nil { + return err + } + + // Actually do the filtering + filterObj := pathsToObject(filters) + r, err := obj.Filter(filterObj) + if err != nil { + return err + } + + return iter(ast.NewTerm(r)) +} + +func getJSONPaths(operand ast.Value) ([]ast.Ref, error) { + var paths []ast.Ref + + switch v := operand.(type) { + case *ast.Array: + for i := 0; i < v.Len(); i++ { + filter, err := parsePath(v.Elem(i)) + if err != nil { + return nil, err + } + paths = append(paths, filter) + } + case ast.Set: + err := v.Iter(func(f *ast.Term) error { + filter, err := parsePath(f) + if err != nil { + return err + } + paths = append(paths, filter) + return nil + }) + if err != nil { + return nil, err + } + default: + return nil, builtins.NewOperandTypeErr(2, v, "set", "array") + } + + return paths, nil +} + +func parsePath(path *ast.Term) (ast.Ref, error) { + // paths can either be a `/` separated json path or + // an array or set of values + var pathSegments ast.Ref + switch p := path.Value.(type) { + case ast.String: + if p == "" { + return ast.Ref{}, nil + } + parts := strings.Split(strings.TrimLeft(string(p), "/"), "/") + for _, part := range parts { + part = strings.ReplaceAll(strings.ReplaceAll(part, "~1", "/"), "~0", "~") + pathSegments = append(pathSegments, ast.StringTerm(part)) + } + case *ast.Array: + p.Foreach(func(term *ast.Term) { + pathSegments = append(pathSegments, term) + }) + default: + return nil, builtins.NewOperandErr(2, "must be one of {set, array} containing string paths or array of path segments but got %v", ast.TypeName(p)) + } + + return pathSegments, nil +} + +func pathsToObject(paths []ast.Ref) ast.Object { + + root := ast.NewObject() + + for _, path := range paths { + node := root + var done bool + + for i := 0; i < len(path)-1 && !done; i++ { + + k := path[i] + child := node.Get(k) + + if child == nil { + obj := ast.NewObject() + node.Insert(k, ast.NewTerm(obj)) + node = obj + continue + } + + switch v := child.Value.(type) { + case ast.Null: + done = true + case ast.Object: + node = v + default: + panic("unreachable") + } + } + + if !done { + node.Insert(path[len(path)-1], ast.NullTerm()) + } + } + + return root +} + +// toIndex tries to convert path elements (that may be strings) into indices into +// an array. +func toIndex(arr *ast.Array, term *ast.Term) (int, error) { + i := 0 + var ok bool + switch v := term.Value.(type) { + case ast.Number: + if i, ok = v.Int(); !ok { + return 0, fmt.Errorf("Invalid number type for indexing") + } + case ast.String: + if v == "-" { + return arr.Len(), nil + } + num := ast.Number(v) + if i, ok = num.Int(); !ok { + return 0, fmt.Errorf("Invalid string for indexing") + } + if v != "0" && strings.HasPrefix(string(v), "0") { + return 0, fmt.Errorf("Leading zeros are not allowed in JSON paths") + } + default: + return 0, fmt.Errorf("Invalid type for indexing") + } + + return i, nil +} + +// patchWorkerris a worker that modifies a direct child of a term located +// at the given key. It returns the new term, and optionally a result that +// is passed back to the caller. +type patchWorker = func(parent, key *ast.Term) (updated, result *ast.Term) + +func jsonPatchTraverse( + target *ast.Term, + path ast.Ref, + worker patchWorker, +) (*ast.Term, *ast.Term) { + if len(path) < 1 { + return nil, nil + } + + key := path[0] + if len(path) == 1 { + return worker(target, key) + } + + success := false + var updated, result *ast.Term + switch parent := target.Value.(type) { + case ast.Object: + obj := ast.NewObject() + parent.Foreach(func(k, v *ast.Term) { + if k.Equal(key) { + if v, result = jsonPatchTraverse(v, path[1:], worker); v != nil { + obj.Insert(k, v) + success = true + } + } else { + obj.Insert(k, v) + } + }) + updated = ast.NewTerm(obj) + + case *ast.Array: + idx, err := toIndex(parent, key) + if err != nil { + return nil, nil + } + arr := ast.NewArray() + for i := 0; i < parent.Len(); i++ { + v := parent.Elem(i) + if idx == i { + if v, result = jsonPatchTraverse(v, path[1:], worker); v != nil { + arr = arr.Append(v) + success = true + } + } else { + arr = arr.Append(v) + } + } + updated = ast.NewTerm(arr) + + case ast.Set: + set := ast.NewSet() + parent.Foreach(func(k *ast.Term) { + if k.Equal(key) { + if k, result = jsonPatchTraverse(k, path[1:], worker); k != nil { + set.Add(k) + success = true + } + } else { + set.Add(k) + } + }) + updated = ast.NewTerm(set) + } + + if success { + return updated, result + } + + return nil, nil +} + +// jsonPatchGet goes one step further than jsonPatchTraverse and returns the +// term at the location specified by the path. It is used in functions +// where we want to read a value but not manipulate its parent: for example +// jsonPatchTest and jsonPatchCopy. +// +// Because it uses jsonPatchTraverse, it makes shallow copies of the objects +// along the path. We could possibly add a signaling mechanism that we didn't +// make any changes to avoid this. +func jsonPatchGet(target *ast.Term, path ast.Ref) *ast.Term { + // Special case: get entire document. + if len(path) == 0 { + return target + } + + _, result := jsonPatchTraverse(target, path, func(parent, key *ast.Term) (*ast.Term, *ast.Term) { + switch v := parent.Value.(type) { + case ast.Object: + return parent, v.Get(key) + case *ast.Array: + i, err := toIndex(v, key) + if err == nil { + return parent, v.Elem(i) + } + case ast.Set: + if v.Contains(key) { + return parent, key + } + } + return nil, nil + }) + return result +} + +func jsonPatchAdd(target *ast.Term, path ast.Ref, value *ast.Term) *ast.Term { + // Special case: replacing root document. + if len(path) == 0 { + return value + } + + target, _ = jsonPatchTraverse(target, path, func(parent *ast.Term, key *ast.Term) (*ast.Term, *ast.Term) { + switch original := parent.Value.(type) { + case ast.Object: + obj := ast.NewObject() + original.Foreach(func(k, v *ast.Term) { + obj.Insert(k, v) + }) + obj.Insert(key, value) + return ast.NewTerm(obj), nil + case *ast.Array: + idx, err := toIndex(original, key) + if err != nil || idx < 0 || idx > original.Len() { + return nil, nil + } + arr := ast.NewArray() + for i := 0; i < idx; i++ { + arr = arr.Append(original.Elem(i)) + } + arr = arr.Append(value) + for i := idx; i < original.Len(); i++ { + arr = arr.Append(original.Elem(i)) + } + return ast.NewTerm(arr), nil + case ast.Set: + if !key.Equal(value) { + return nil, nil + } + set := ast.NewSet() + original.Foreach(func(k *ast.Term) { + set.Add(k) + }) + set.Add(key) + return ast.NewTerm(set), nil + } + return nil, nil + }) + + return target +} + +func jsonPatchRemove(target *ast.Term, path ast.Ref) (*ast.Term, *ast.Term) { + // Special case: replacing root document. + if len(path) == 0 { + return nil, nil + } + + target, removed := jsonPatchTraverse(target, path, func(parent *ast.Term, key *ast.Term) (*ast.Term, *ast.Term) { + var removed *ast.Term + switch original := parent.Value.(type) { + case ast.Object: + obj := ast.NewObject() + original.Foreach(func(k, v *ast.Term) { + if k.Equal(key) { + removed = v + } else { + obj.Insert(k, v) + } + }) + return ast.NewTerm(obj), removed + case *ast.Array: + idx, err := toIndex(original, key) + if err != nil || idx < 0 || idx >= original.Len() { + return nil, nil + } + arr := ast.NewArray() + for i := 0; i < idx; i++ { + arr = arr.Append(original.Elem(i)) + } + removed = original.Elem(idx) + for i := idx + 1; i < original.Len(); i++ { + arr = arr.Append(original.Elem(i)) + } + return ast.NewTerm(arr), removed + case ast.Set: + set := ast.NewSet() + original.Foreach(func(k *ast.Term) { + if k.Equal(key) { + removed = k + } else { + set.Add(k) + } + }) + return ast.NewTerm(set), removed + } + return nil, nil + }) + + if target != nil && removed != nil { + return target, removed + } + + return nil, nil +} + +func jsonPatchReplace(target *ast.Term, path ast.Ref, value *ast.Term) *ast.Term { + // Special case: replacing the whole document. + if len(path) == 0 { + return value + } + + // Replace is specified as `remove` followed by `add`. + if target, _ = jsonPatchRemove(target, path); target == nil { + return nil + } + + return jsonPatchAdd(target, path, value) +} + +func jsonPatchMove(target *ast.Term, path ast.Ref, from ast.Ref) *ast.Term { + // Move is specified as `remove` followed by `add`. + target, removed := jsonPatchRemove(target, from) + if target == nil || removed == nil { + return nil + } + + return jsonPatchAdd(target, path, removed) +} + +func jsonPatchCopy(target *ast.Term, path ast.Ref, from ast.Ref) *ast.Term { + value := jsonPatchGet(target, from) + if value == nil { + return nil + } + + return jsonPatchAdd(target, path, value) +} + +func jsonPatchTest(target *ast.Term, path ast.Ref, value *ast.Term) *ast.Term { + actual := jsonPatchGet(target, path) + if actual == nil { + return nil + } + + if actual.Equal(value) { + return target + } + + return nil +} + +func builtinJSONPatch(_ BuiltinContext, operands []*ast.Term, iter func(*ast.Term) error) error { + // JSON patch supports arrays, objects as well as values as the target. + target := ast.NewTerm(operands[0].Value) + + // Expect an array of operations. + operations, err := builtins.ArrayOperand(operands[1].Value, 2) + if err != nil { + return err + } + + // Apply operations one by one. + for i := 0; i < operations.Len(); i++ { + if object, ok := operations.Elem(i).Value.(ast.Object); ok { + getAttribute := func(attr string) (*ast.Term, error) { + if term := object.Get(ast.StringTerm(attr)); term != nil { + return term, nil + } + + return nil, builtins.NewOperandErr(2, fmt.Sprintf("patch is missing '%s' attribute", attr)) + } + + getPathAttribute := func(attr string) (ast.Ref, error) { + term, err := getAttribute(attr) + if err != nil { + return ast.Ref{}, err + } + path, err := parsePath(term) + if err != nil { + return ast.Ref{}, err + } + return path, nil + } + + // Parse operation. + opTerm, err := getAttribute("op") + if err != nil { + return err + } + op, ok := opTerm.Value.(ast.String) + if !ok { + return builtins.NewOperandErr(2, "patch attribute 'op' must be a string") + } + + // Parse path. + path, err := getPathAttribute("path") + if err != nil { + return err + } + + switch op { + case "add": + value, err := getAttribute("value") + if err != nil { + return err + } + target = jsonPatchAdd(target, path, value) + case "remove": + target, _ = jsonPatchRemove(target, path) + case "replace": + value, err := getAttribute("value") + if err != nil { + return err + } + target = jsonPatchReplace(target, path, value) + case "move": + from, err := getPathAttribute("from") + if err != nil { + return err + } + target = jsonPatchMove(target, path, from) + case "copy": + from, err := getPathAttribute("from") + if err != nil { + return err + } + target = jsonPatchCopy(target, path, from) + case "test": + value, err := getAttribute("value") + if err != nil { + return err + } + target = jsonPatchTest(target, path, value) + default: + return builtins.NewOperandErr(2, "must be an array of JSON-Patch objects") + } + } else { + return builtins.NewOperandErr(2, "must be an array of JSON-Patch objects") + } + + // JSON patches should work atomically; and if one of them fails, + // we should not try to continue. + if target == nil { + return nil + } + } + + return iter(target) +} + +func init() { + RegisterBuiltinFunc(ast.JSONFilter.Name, builtinJSONFilter) + RegisterBuiltinFunc(ast.JSONRemove.Name, builtinJSONRemove) + RegisterBuiltinFunc(ast.JSONPatch.Name, builtinJSONPatch) +} diff --git a/vendor/github.com/open-policy-agent/opa/topdown/net.go b/vendor/github.com/open-policy-agent/opa/topdown/net.go new file mode 100644 index 00000000..a167cf13 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/topdown/net.go @@ -0,0 +1,63 @@ +// Copyright 2021 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package topdown + +import ( + "net" + + "github.com/open-policy-agent/opa/ast" + "github.com/open-policy-agent/opa/topdown/builtins" +) + +type lookupIPAddrCacheKey string + +// resolv is the same as net.DefaultResolver -- this is for mocking it out in tests +var resolv = &net.Resolver{} + +func builtinLookupIPAddr(bctx BuiltinContext, operands []*ast.Term, iter func(*ast.Term) error) error { + a, err := builtins.StringOperand(operands[0].Value, 1) + if err != nil { + return err + } + name := string(a) + + err = verifyHost(bctx, name) + if err != nil { + return err + } + + key := lookupIPAddrCacheKey(name) + if val, ok := bctx.Cache.Get(key); ok { + return iter(val.(*ast.Term)) + } + + addrs, err := resolv.LookupIPAddr(bctx.Context, name) + if err != nil { + // NOTE(sr): We can't do better than this right now, see https://github.com/golang/go/issues/36208 + if err.Error() == "operation was canceled" || err.Error() == "i/o timeout" { + return Halt{ + Err: &Error{ + Code: CancelErr, + Message: ast.NetLookupIPAddr.Name + ": " + err.Error(), + Location: bctx.Location, + }, + } + } + return err + } + + ret := ast.NewSet() + for _, a := range addrs { + ret.Add(ast.StringTerm(a.String())) + + } + t := ast.NewTerm(ret) + bctx.Cache.Put(key, t) + return iter(t) +} + +func init() { + RegisterBuiltinFunc(ast.NetLookupIPAddr.Name, builtinLookupIPAddr) +} diff --git a/vendor/github.com/open-policy-agent/opa/topdown/numbers.go b/vendor/github.com/open-policy-agent/opa/topdown/numbers.go new file mode 100644 index 00000000..9d57c879 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/topdown/numbers.go @@ -0,0 +1,99 @@ +// Copyright 2020 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package topdown + +import ( + "fmt" + "math/big" + + "github.com/open-policy-agent/opa/ast" + "github.com/open-policy-agent/opa/topdown/builtins" +) + +type randIntCachingKey string + +var one = big.NewInt(1) + +func builtinNumbersRange(bctx BuiltinContext, operands []*ast.Term, iter func(*ast.Term) error) error { + + x, err := builtins.BigIntOperand(operands[0].Value, 1) + if err != nil { + return err + } + + y, err := builtins.BigIntOperand(operands[1].Value, 2) + if err != nil { + return err + } + + result := ast.NewArray() + cmp := x.Cmp(y) + haltErr := Halt{ + Err: &Error{ + Code: CancelErr, + Message: "numbers.range: timed out before generating all numbers in range", + }, + } + + if cmp <= 0 { + for i := new(big.Int).Set(x); i.Cmp(y) <= 0; i = i.Add(i, one) { + if bctx.Cancel != nil && bctx.Cancel.Cancelled() { + return haltErr + } + result = result.Append(ast.NewTerm(builtins.IntToNumber(i))) + } + } else { + for i := new(big.Int).Set(x); i.Cmp(y) >= 0; i = i.Sub(i, one) { + if bctx.Cancel != nil && bctx.Cancel.Cancelled() { + return haltErr + } + result = result.Append(ast.NewTerm(builtins.IntToNumber(i))) + } + } + + return iter(ast.NewTerm(result)) +} + +func builtinRandIntn(bctx BuiltinContext, args []*ast.Term, iter func(*ast.Term) error) error { + + strOp, err := builtins.StringOperand(args[0].Value, 1) + if err != nil { + return err + + } + + n, err := builtins.IntOperand(args[1].Value, 2) + if err != nil { + return err + } + + if n == 0 { + return iter(ast.IntNumberTerm(0)) + } + + if n < 0 { + n = -n + } + + var key = randIntCachingKey(fmt.Sprintf("%s-%d", strOp, n)) + + if val, ok := bctx.Cache.Get(key); ok { + return iter(val.(*ast.Term)) + } + + r, err := bctx.Rand() + if err != nil { + return err + } + result := ast.IntNumberTerm(r.Intn(n)) + bctx.Cache.Put(key, result) + + return iter(result) +} + +func init() { + RegisterBuiltinFunc(ast.NumbersRange.Name, builtinNumbersRange) + RegisterBuiltinFunc(ast.RandIntn.Name, builtinRandIntn) +} diff --git a/vendor/github.com/open-policy-agent/opa/topdown/object.go b/vendor/github.com/open-policy-agent/opa/topdown/object.go new file mode 100644 index 00000000..ff01b4f9 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/topdown/object.go @@ -0,0 +1,140 @@ +// Copyright 2020 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package topdown + +import ( + "github.com/open-policy-agent/opa/ast" + "github.com/open-policy-agent/opa/topdown/builtins" + "github.com/open-policy-agent/opa/types" +) + +func builtinObjectUnion(_ BuiltinContext, operands []*ast.Term, iter func(*ast.Term) error) error { + objA, err := builtins.ObjectOperand(operands[0].Value, 1) + if err != nil { + return err + } + + objB, err := builtins.ObjectOperand(operands[1].Value, 2) + if err != nil { + return err + } + + r := mergeWithOverwrite(objA, objB) + + return iter(ast.NewTerm(r)) +} + +func builtinObjectRemove(_ BuiltinContext, operands []*ast.Term, iter func(*ast.Term) error) error { + // Expect an object and an array/set/object of keys + obj, err := builtins.ObjectOperand(operands[0].Value, 1) + if err != nil { + return err + } + + // Build a set of keys to remove + keysToRemove, err := getObjectKeysParam(operands[1].Value) + if err != nil { + return err + } + r := ast.NewObject() + obj.Foreach(func(key *ast.Term, value *ast.Term) { + if !keysToRemove.Contains(key) { + r.Insert(key, value) + } + }) + + return iter(ast.NewTerm(r)) +} + +func builtinObjectFilter(_ BuiltinContext, operands []*ast.Term, iter func(*ast.Term) error) error { + // Expect an object and an array/set/object of keys + obj, err := builtins.ObjectOperand(operands[0].Value, 1) + if err != nil { + return err + } + + // Build a new object from the supplied filter keys + keys, err := getObjectKeysParam(operands[1].Value) + if err != nil { + return err + } + + filterObj := ast.NewObject() + keys.Foreach(func(key *ast.Term) { + filterObj.Insert(key, ast.NullTerm()) + }) + + // Actually do the filtering + r, err := obj.Filter(filterObj) + if err != nil { + return err + } + + return iter(ast.NewTerm(r)) +} + +func builtinObjectGet(_ BuiltinContext, operands []*ast.Term, iter func(*ast.Term) error) error { + object, err := builtins.ObjectOperand(operands[0].Value, 1) + if err != nil { + return err + } + + if ret := object.Get(operands[1]); ret != nil { + return iter(ret) + } + + return iter(operands[2]) +} + +// getObjectKeysParam returns a set of key values +// from a supplied ast array, object, set value +func getObjectKeysParam(arrayOrSet ast.Value) (ast.Set, error) { + keys := ast.NewSet() + + switch v := arrayOrSet.(type) { + case *ast.Array: + _ = v.Iter(func(f *ast.Term) error { + keys.Add(f) + return nil + }) + case ast.Set: + _ = v.Iter(func(f *ast.Term) error { + keys.Add(f) + return nil + }) + case ast.Object: + _ = v.Iter(func(k *ast.Term, _ *ast.Term) error { + keys.Add(k) + return nil + }) + default: + return nil, builtins.NewOperandTypeErr(2, arrayOrSet, ast.TypeName(types.Object{}), ast.TypeName(types.S), ast.TypeName(types.Array{})) + } + + return keys, nil +} + +func mergeWithOverwrite(objA, objB ast.Object) ast.Object { + merged, _ := objA.MergeWith(objB, func(v1, v2 *ast.Term) (*ast.Term, bool) { + originalValueObj, ok2 := v1.Value.(ast.Object) + updateValueObj, ok1 := v2.Value.(ast.Object) + if !ok1 || !ok2 { + // If we can't merge, stick with the right-hand value + return v2, false + } + + // Recursively update the existing value + merged := mergeWithOverwrite(originalValueObj, updateValueObj) + return ast.NewTerm(merged), false + }) + return merged +} + +func init() { + RegisterBuiltinFunc(ast.ObjectUnion.Name, builtinObjectUnion) + RegisterBuiltinFunc(ast.ObjectRemove.Name, builtinObjectRemove) + RegisterBuiltinFunc(ast.ObjectFilter.Name, builtinObjectFilter) + RegisterBuiltinFunc(ast.ObjectGet.Name, builtinObjectGet) +} diff --git a/vendor/github.com/open-policy-agent/opa/topdown/parse.go b/vendor/github.com/open-policy-agent/opa/topdown/parse.go new file mode 100644 index 00000000..dc6e6fc5 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/topdown/parse.go @@ -0,0 +1,47 @@ +// Copyright 2018 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package topdown + +import ( + "bytes" + "encoding/json" + + "github.com/open-policy-agent/opa/ast" + "github.com/open-policy-agent/opa/topdown/builtins" +) + +func builtinRegoParseModule(a, b ast.Value) (ast.Value, error) { + + filename, err := builtins.StringOperand(a, 1) + if err != nil { + return nil, err + } + + input, err := builtins.StringOperand(b, 1) + if err != nil { + return nil, err + } + + module, err := ast.ParseModule(string(filename), string(input)) + if err != nil { + return nil, err + } + + var buf bytes.Buffer + if err := json.NewEncoder(&buf).Encode(module); err != nil { + return nil, err + } + + term, err := ast.ParseTerm(buf.String()) + if err != nil { + return nil, err + } + + return term.Value, nil +} + +func init() { + RegisterFunctionalBuiltin2(ast.RegoParseModule.Name, builtinRegoParseModule) +} diff --git a/vendor/github.com/open-policy-agent/opa/topdown/parse_bytes.go b/vendor/github.com/open-policy-agent/opa/topdown/parse_bytes.go new file mode 100644 index 00000000..6e58d232 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/topdown/parse_bytes.go @@ -0,0 +1,143 @@ +// Copyright 2016 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package topdown + +import ( + "fmt" + "math/big" + "strings" + "unicode" + + "github.com/open-policy-agent/opa/ast" + "github.com/open-policy-agent/opa/topdown/builtins" +) + +const ( + none uint64 = 1 << (10 * iota) + ki + mi + gi + ti + pi + ei + + kb uint64 = 1000 + mb = kb * 1000 + gb = mb * 1000 + tb = gb * 1000 + pb = tb * 1000 + eb = pb * 1000 +) + +func parseNumBytesError(msg string) error { + return fmt.Errorf("%s error: %s", ast.UnitsParseBytes.Name, msg) +} + +func errUnitNotRecognized(unit string) error { + return parseNumBytesError(fmt.Sprintf("byte unit %s not recognized", unit)) +} + +var ( + errNoAmount = parseNumBytesError("no byte amount provided") + errNumConv = parseNumBytesError("could not parse byte amount to a number") + errIncludesSpaces = parseNumBytesError("spaces not allowed in resource strings") +) + +func builtinNumBytes(bctx BuiltinContext, operands []*ast.Term, iter func(*ast.Term) error) error { + var m big.Float + + raw, err := builtins.StringOperand(operands[0].Value, 1) + if err != nil { + return err + } + + s := formatString(raw) + + if strings.Contains(s, " ") { + return errIncludesSpaces + } + + num, unit := extractNumAndUnit(s) + if num == "" { + return errNoAmount + } + + switch unit { + case "": + m.SetUint64(none) + case "kb", "k": + m.SetUint64(kb) + case "kib", "ki": + m.SetUint64(ki) + case "mb", "m": + m.SetUint64(mb) + case "mib", "mi": + m.SetUint64(mi) + case "gb", "g": + m.SetUint64(gb) + case "gib", "gi": + m.SetUint64(gi) + case "tb", "t": + m.SetUint64(tb) + case "tib", "ti": + m.SetUint64(ti) + case "pb", "p": + m.SetUint64(pb) + case "pib", "pi": + m.SetUint64(pi) + case "eb", "e": + m.SetUint64(eb) + case "eib", "ei": + m.SetUint64(ei) + default: + return errUnitNotRecognized(unit) + } + + numFloat, ok := new(big.Float).SetString(num) + if !ok { + return errNumConv + } + + var total big.Int + numFloat.Mul(numFloat, &m).Int(&total) + return iter(ast.NewTerm(builtins.IntToNumber(&total))) +} + +// Makes the string lower case and removes quotation marks +func formatString(s ast.String) string { + str := string(s) + lower := strings.ToLower(str) + return strings.Replace(lower, "\"", "", -1) +} + +// Splits the string into a number string à la "10" or "10.2" and a unit +// string à la "gb" or "MiB" or "foo". Either can be an empty string +// (error handling is provided elsewhere). +func extractNumAndUnit(s string) (string, string) { + isNum := func(r rune) bool { + return unicode.IsDigit(r) || r == '.' + } + + firstNonNumIdx := -1 + for idx, r := range s { + if !isNum(r) { + firstNonNumIdx = idx + break + } + } + + if firstNonNumIdx == -1 { // only digits and '.' + return s, "" + } + if firstNonNumIdx == 0 { // only units (starts with non-digit) + return "", s + } + + return s[0:firstNonNumIdx], s[firstNonNumIdx:] +} + +func init() { + RegisterBuiltinFunc(ast.UnitsParseBytes.Name, builtinNumBytes) +} diff --git a/vendor/github.com/open-policy-agent/opa/topdown/print.go b/vendor/github.com/open-policy-agent/opa/topdown/print.go new file mode 100644 index 00000000..765b344b --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/topdown/print.go @@ -0,0 +1,86 @@ +// Copyright 2021 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package topdown + +import ( + "fmt" + "io" + "strings" + + "github.com/open-policy-agent/opa/ast" + "github.com/open-policy-agent/opa/topdown/builtins" + "github.com/open-policy-agent/opa/topdown/print" +) + +func NewPrintHook(w io.Writer) print.Hook { + return printHook{w: w} +} + +type printHook struct { + w io.Writer +} + +func (h printHook) Print(_ print.Context, msg string) error { + _, err := fmt.Fprintln(h.w, msg) + return err +} + +func builtinPrint(bctx BuiltinContext, operands []*ast.Term, iter func(*ast.Term) error) error { + + if bctx.PrintHook == nil { + return iter(nil) + } + + arr, err := builtins.ArrayOperand(operands[0].Value, 1) + if err != nil { + return err + } + + buf := make([]string, arr.Len()) + + err = builtinPrintCrossProductOperands(bctx, buf, arr, 0, func(buf []string) error { + pctx := print.Context{ + Context: bctx.Context, + Location: bctx.Location, + } + return bctx.PrintHook.Print(pctx, strings.Join(buf, " ")) + }) + if err != nil { + return err + } + + return iter(nil) +} + +func builtinPrintCrossProductOperands(bctx BuiltinContext, buf []string, operands *ast.Array, i int, f func([]string) error) error { + + if i >= operands.Len() { + return f(buf) + } + + xs, ok := operands.Elem(i).Value.(ast.Set) + if !ok { + return Halt{Err: internalErr(bctx.Location, fmt.Sprintf("illegal argument type: %v", ast.TypeName(operands.Elem(i).Value)))} + } + + if xs.Len() == 0 { + buf[i] = "" + return builtinPrintCrossProductOperands(bctx, buf, operands, i+1, f) + } + + return xs.Iter(func(x *ast.Term) error { + switch v := x.Value.(type) { + case ast.String: + buf[i] = string(v) + default: + buf[i] = v.String() + } + return builtinPrintCrossProductOperands(bctx, buf, operands, i+1, f) + }) +} + +func init() { + RegisterBuiltinFunc(ast.InternalPrint.Name, builtinPrint) +} diff --git a/vendor/github.com/open-policy-agent/opa/topdown/print/print.go b/vendor/github.com/open-policy-agent/opa/topdown/print/print.go new file mode 100644 index 00000000..0fb6abdc --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/topdown/print/print.go @@ -0,0 +1,21 @@ +package print + +import ( + "context" + + "github.com/open-policy-agent/opa/ast" +) + +// Context provides the Hook implementation context about the print() call. +type Context struct { + Context context.Context // request context passed when query executed + Location *ast.Location // location of print call +} + +// Hook defines the interface that callers can implement to receive print +// statement outputs. If the hook returns an error, it will be surfaced if +// strict builtin error checking is enabled (otherwise, it will not halt +// execution.) +type Hook interface { + Print(Context, string) error +} diff --git a/vendor/github.com/open-policy-agent/opa/topdown/query.go b/vendor/github.com/open-policy-agent/opa/topdown/query.go new file mode 100644 index 00000000..9a9a3c67 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/topdown/query.go @@ -0,0 +1,482 @@ +package topdown + +import ( + "context" + "crypto/rand" + "io" + "sort" + "time" + + "github.com/open-policy-agent/opa/ast" + "github.com/open-policy-agent/opa/metrics" + "github.com/open-policy-agent/opa/resolver" + "github.com/open-policy-agent/opa/storage" + "github.com/open-policy-agent/opa/topdown/builtins" + "github.com/open-policy-agent/opa/topdown/cache" + "github.com/open-policy-agent/opa/topdown/copypropagation" + "github.com/open-policy-agent/opa/topdown/print" + "github.com/open-policy-agent/opa/tracing" +) + +// QueryResultSet represents a collection of results returned by a query. +type QueryResultSet []QueryResult + +// QueryResult represents a single result returned by a query. The result +// contains bindings for all variables that appear in the query. +type QueryResult map[ast.Var]*ast.Term + +// Query provides a configurable interface for performing query evaluation. +type Query struct { + seed io.Reader + time time.Time + cancel Cancel + query ast.Body + queryCompiler ast.QueryCompiler + compiler *ast.Compiler + store storage.Store + txn storage.Transaction + input *ast.Term + external *resolverTrie + tracers []QueryTracer + plugTraceVars bool + unknowns []*ast.Term + partialNamespace string + skipSaveNamespace bool + metrics metrics.Metrics + instr *Instrumentation + disableInlining []ast.Ref + shallowInlining bool + genvarprefix string + runtime *ast.Term + builtins map[string]*Builtin + indexing bool + earlyExit bool + interQueryBuiltinCache cache.InterQueryCache + strictBuiltinErrors bool + printHook print.Hook + tracingOpts tracing.Options +} + +// Builtin represents a built-in function that queries can call. +type Builtin struct { + Decl *ast.Builtin + Func BuiltinFunc +} + +// NewQuery returns a new Query object that can be run. +func NewQuery(query ast.Body) *Query { + return &Query{ + query: query, + genvarprefix: ast.WildcardPrefix, + indexing: true, + earlyExit: true, + external: newResolverTrie(), + } +} + +// WithQueryCompiler sets the queryCompiler used for the query. +func (q *Query) WithQueryCompiler(queryCompiler ast.QueryCompiler) *Query { + q.queryCompiler = queryCompiler + return q +} + +// WithCompiler sets the compiler to use for the query. +func (q *Query) WithCompiler(compiler *ast.Compiler) *Query { + q.compiler = compiler + return q +} + +// WithStore sets the store to use for the query. +func (q *Query) WithStore(store storage.Store) *Query { + q.store = store + return q +} + +// WithTransaction sets the transaction to use for the query. All queries +// should be performed over a consistent snapshot of the storage layer. +func (q *Query) WithTransaction(txn storage.Transaction) *Query { + q.txn = txn + return q +} + +// WithCancel sets the cancellation object to use for the query. Set this if +// you need to abort queries based on a deadline. This is optional. +func (q *Query) WithCancel(cancel Cancel) *Query { + q.cancel = cancel + return q +} + +// WithInput sets the input object to use for the query. References rooted at +// input will be evaluated against this value. This is optional. +func (q *Query) WithInput(input *ast.Term) *Query { + q.input = input + return q +} + +// WithTracer adds a query tracer to use during evaluation. This is optional. +// Deprecated: Use WithQueryTracer instead. +func (q *Query) WithTracer(tracer Tracer) *Query { + qt, ok := tracer.(QueryTracer) + if !ok { + qt = WrapLegacyTracer(tracer) + } + return q.WithQueryTracer(qt) +} + +// WithQueryTracer adds a query tracer to use during evaluation. This is optional. +// Disabled QueryTracers will be ignored. +func (q *Query) WithQueryTracer(tracer QueryTracer) *Query { + if !tracer.Enabled() { + return q + } + + q.tracers = append(q.tracers, tracer) + + // If *any* of the tracers require local variable metadata we need to + // enabled plugging local trace variables. + conf := tracer.Config() + if conf.PlugLocalVars { + q.plugTraceVars = true + } + + return q +} + +// WithMetrics sets the metrics collection to add evaluation metrics to. This +// is optional. +func (q *Query) WithMetrics(m metrics.Metrics) *Query { + q.metrics = m + return q +} + +// WithInstrumentation sets the instrumentation configuration to enable on the +// evaluation process. By default, instrumentation is turned off. +func (q *Query) WithInstrumentation(instr *Instrumentation) *Query { + q.instr = instr + return q +} + +// WithUnknowns sets the initial set of variables or references to treat as +// unknown during query evaluation. This is required for partial evaluation. +func (q *Query) WithUnknowns(terms []*ast.Term) *Query { + q.unknowns = terms + return q +} + +// WithPartialNamespace sets the namespace to use for supporting rules +// generated as part of the partial evaluation process. The ns value must be a +// valid package path component. +func (q *Query) WithPartialNamespace(ns string) *Query { + q.partialNamespace = ns + return q +} + +// WithSkipPartialNamespace disables namespacing of saved support rules that are generated +// from the original policy (rules which are completely synthetic are still namespaced.) +func (q *Query) WithSkipPartialNamespace(yes bool) *Query { + q.skipSaveNamespace = yes + return q +} + +// WithDisableInlining adds a set of paths to the query that should be excluded from +// inlining. Inlining during partial evaluation can be expensive in some cases +// (e.g., when a cross-product is computed.) Disabling inlining avoids expensive +// computation at the cost of generating support rules. +func (q *Query) WithDisableInlining(paths []ast.Ref) *Query { + q.disableInlining = paths + return q +} + +// WithShallowInlining disables aggressive inlining performed during partial evaluation. +// When shallow inlining is enabled rules that depend (transitively) on unknowns are not inlined. +// Only rules/values that are completely known will be inlined. +func (q *Query) WithShallowInlining(yes bool) *Query { + q.shallowInlining = yes + return q +} + +// WithRuntime sets the runtime data to execute the query with. The runtime data +// can be returned by the `opa.runtime` built-in function. +func (q *Query) WithRuntime(runtime *ast.Term) *Query { + q.runtime = runtime + return q +} + +// WithBuiltins adds a set of built-in functions that can be called by the +// query. +func (q *Query) WithBuiltins(builtins map[string]*Builtin) *Query { + q.builtins = builtins + return q +} + +// WithIndexing will enable or disable using rule indexing for the evaluation +// of the query. The default is enabled. +func (q *Query) WithIndexing(enabled bool) *Query { + q.indexing = enabled + return q +} + +// WithEarlyExit will enable or disable using 'early exit' for the evaluation +// of the query. The default is enabled. +func (q *Query) WithEarlyExit(enabled bool) *Query { + q.earlyExit = enabled + return q +} + +// WithSeed sets a reader that will seed randomization required by built-in functions. +// If a seed is not provided crypto/rand.Reader is used. +func (q *Query) WithSeed(r io.Reader) *Query { + q.seed = r + return q +} + +// WithTime sets the time that will be returned by the time.now_ns() built-in function. +func (q *Query) WithTime(x time.Time) *Query { + q.time = x + return q +} + +// WithInterQueryBuiltinCache sets the inter-query cache that built-in functions can utilize. +func (q *Query) WithInterQueryBuiltinCache(c cache.InterQueryCache) *Query { + q.interQueryBuiltinCache = c + return q +} + +// WithStrictBuiltinErrors tells the evaluator to treat all built-in function errors as fatal errors. +func (q *Query) WithStrictBuiltinErrors(yes bool) *Query { + q.strictBuiltinErrors = yes + return q +} + +// WithResolver configures an external resolver to use for the given ref. +func (q *Query) WithResolver(ref ast.Ref, r resolver.Resolver) *Query { + q.external.Put(ref, r) + return q +} + +func (q *Query) WithPrintHook(h print.Hook) *Query { + q.printHook = h + return q +} + +// WithDistributedTracingOpts sets the options to be used by distributed tracing. +func (q *Query) WithDistributedTracingOpts(tr tracing.Options) *Query { + q.tracingOpts = tr + return q +} + +// PartialRun executes partial evaluation on the query with respect to unknown +// values. Partial evaluation attempts to evaluate as much of the query as +// possible without requiring values for the unknowns set on the query. The +// result of partial evaluation is a new set of queries that can be evaluated +// once the unknown value is known. In addition to new queries, partial +// evaluation may produce additional support modules that should be used in +// conjunction with the partially evaluated queries. +func (q *Query) PartialRun(ctx context.Context) (partials []ast.Body, support []*ast.Module, err error) { + if q.partialNamespace == "" { + q.partialNamespace = "partial" // lazily initialize partial namespace + } + if q.seed == nil { + q.seed = rand.Reader + } + if !q.time.IsZero() { + q.time = time.Now() + } + if q.metrics == nil { + q.metrics = metrics.New() + } + f := &queryIDFactory{} + b := newBindings(0, q.instr) + e := &eval{ + ctx: ctx, + metrics: q.metrics, + seed: q.seed, + time: ast.NumberTerm(int64ToJSONNumber(q.time.UnixNano())), + cancel: q.cancel, + query: q.query, + queryCompiler: q.queryCompiler, + queryIDFact: f, + queryID: f.Next(), + bindings: b, + compiler: q.compiler, + store: q.store, + baseCache: newBaseCache(), + targetStack: newRefStack(), + txn: q.txn, + input: q.input, + external: q.external, + tracers: q.tracers, + traceEnabled: len(q.tracers) > 0, + plugTraceVars: q.plugTraceVars, + instr: q.instr, + builtins: q.builtins, + builtinCache: builtins.Cache{}, + interQueryBuiltinCache: q.interQueryBuiltinCache, + virtualCache: newVirtualCache(), + comprehensionCache: newComprehensionCache(), + saveSet: newSaveSet(q.unknowns, b, q.instr), + saveStack: newSaveStack(), + saveSupport: newSaveSupport(), + saveNamespace: ast.StringTerm(q.partialNamespace), + skipSaveNamespace: q.skipSaveNamespace, + inliningControl: &inliningControl{ + shallow: q.shallowInlining, + }, + genvarprefix: q.genvarprefix, + runtime: q.runtime, + indexing: q.indexing, + earlyExit: q.earlyExit, + builtinErrors: &builtinErrors{}, + printHook: q.printHook, + } + + if len(q.disableInlining) > 0 { + e.inliningControl.PushDisable(q.disableInlining, false) + } + + e.caller = e + q.metrics.Timer(metrics.RegoPartialEval).Start() + defer q.metrics.Timer(metrics.RegoPartialEval).Stop() + + livevars := ast.NewVarSet() + + ast.WalkVars(q.query, func(x ast.Var) bool { + if !x.IsGenerated() { + livevars.Add(x) + } + return false + }) + + p := copypropagation.New(livevars).WithCompiler(q.compiler) + + err = e.Run(func(e *eval) error { + + // Build output from saved expressions. + body := ast.NewBody() + + for _, elem := range e.saveStack.Stack[len(e.saveStack.Stack)-1] { + body.Append(elem.Plug(e.bindings)) + } + + // Include bindings as exprs so that when caller evals the result, they + // can obtain values for the vars in their query. + bindingExprs := []*ast.Expr{} + _ = e.bindings.Iter(e.bindings, func(a, b *ast.Term) error { + bindingExprs = append(bindingExprs, ast.Equality.Expr(a, b)) + return nil + }) // cannot return error + + // Sort binding expressions so that results are deterministic. + sort.Slice(bindingExprs, func(i, j int) bool { + return bindingExprs[i].Compare(bindingExprs[j]) < 0 + }) + + for i := range bindingExprs { + body.Append(bindingExprs[i]) + } + + // Skip this rule body if it fails to type-check. + // Type-checking failure means the rule body will never succeed. + if !e.compiler.PassesTypeCheck(body) { + return nil + } + + if !q.shallowInlining { + body = applyCopyPropagation(p, e.instr, body) + } + + partials = append(partials, body) + return nil + }) + + support = e.saveSupport.List() + + if q.strictBuiltinErrors && len(e.builtinErrors.errs) > 0 { + err = e.builtinErrors.errs[0] + } + + for i := range support { + sort.Slice(support[i].Rules, func(j, k int) bool { + return support[i].Rules[j].Compare(support[i].Rules[k]) < 0 + }) + } + + return partials, support, err +} + +// Run is a wrapper around Iter that accumulates query results and returns them +// in one shot. +func (q *Query) Run(ctx context.Context) (QueryResultSet, error) { + qrs := QueryResultSet{} + return qrs, q.Iter(ctx, func(qr QueryResult) error { + qrs = append(qrs, qr) + return nil + }) +} + +// Iter executes the query and invokes the iter function with query results +// produced by evaluating the query. +func (q *Query) Iter(ctx context.Context, iter func(QueryResult) error) error { + if q.seed == nil { + q.seed = rand.Reader + } + if q.time.IsZero() { + q.time = time.Now() + } + if q.metrics == nil { + q.metrics = metrics.New() + } + f := &queryIDFactory{} + e := &eval{ + ctx: ctx, + metrics: q.metrics, + seed: q.seed, + time: ast.NumberTerm(int64ToJSONNumber(q.time.UnixNano())), + cancel: q.cancel, + query: q.query, + queryCompiler: q.queryCompiler, + queryIDFact: f, + queryID: f.Next(), + bindings: newBindings(0, q.instr), + compiler: q.compiler, + store: q.store, + baseCache: newBaseCache(), + targetStack: newRefStack(), + txn: q.txn, + input: q.input, + external: q.external, + tracers: q.tracers, + traceEnabled: len(q.tracers) > 0, + plugTraceVars: q.plugTraceVars, + instr: q.instr, + builtins: q.builtins, + builtinCache: builtins.Cache{}, + interQueryBuiltinCache: q.interQueryBuiltinCache, + virtualCache: newVirtualCache(), + comprehensionCache: newComprehensionCache(), + genvarprefix: q.genvarprefix, + runtime: q.runtime, + indexing: q.indexing, + earlyExit: q.earlyExit, + builtinErrors: &builtinErrors{}, + printHook: q.printHook, + tracingOpts: q.tracingOpts, + } + e.caller = e + q.metrics.Timer(metrics.RegoQueryEval).Start() + err := e.Run(func(e *eval) error { + qr := QueryResult{} + _ = e.bindings.Iter(nil, func(k, v *ast.Term) error { + qr[k.Value.(ast.Var)] = v + return nil + }) // cannot return error + return iter(qr) + }) + + if q.strictBuiltinErrors && err == nil && len(e.builtinErrors.errs) > 0 { + err = e.builtinErrors.errs[0] + } + + q.metrics.Timer(metrics.RegoQueryEval).Stop() + return err +} diff --git a/vendor/github.com/open-policy-agent/opa/topdown/reachable.go b/vendor/github.com/open-policy-agent/opa/topdown/reachable.go new file mode 100644 index 00000000..6b73fa2c --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/topdown/reachable.go @@ -0,0 +1,61 @@ +// Copyright 2020 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package topdown + +import ( + "github.com/open-policy-agent/opa/ast" +) + +// Helper: sets of vertices can be represented as Arrays or Sets. +func foreachVertex(collection *ast.Term, f func(*ast.Term)) { + switch v := collection.Value.(type) { + case ast.Set: + v.Foreach(f) + case *ast.Array: + v.Foreach(f) + } +} + +func builtinReachable(bctx BuiltinContext, args []*ast.Term, iter func(*ast.Term) error) error { + // Return the empty set if the first argument is not an object. + graph, ok := args[0].Value.(ast.Object) + if !ok { + return iter(ast.NewTerm(ast.NewSet())) + } + + // This is a queue that holds all nodes we still need to visit. It is + // initialised to the initial set of nodes we start out with. + queue := []*ast.Term{} + foreachVertex(args[1], func(t *ast.Term) { + queue = append(queue, t) + }) + + // This is the set of nodes we have reached. + reached := ast.NewSet() + + // Keep going as long as we have nodes in the queue. + for len(queue) > 0 { + // Get the edges for this node. If the node was not in the graph, + // `edges` will be `nil` and we can ignore it. + node := queue[0] + if edges := graph.Get(node); edges != nil { + // Add all the newly discovered neighbors. + foreachVertex(edges, func(neighbor *ast.Term) { + if !reached.Contains(neighbor) { + queue = append(queue, neighbor) + } + }) + // Mark the node as reached. + reached.Add(node) + } + queue = queue[1:] + } + + return iter(ast.NewTerm(reached)) +} + +func init() { + RegisterBuiltinFunc(ast.ReachableBuiltin.Name, builtinReachable) +} diff --git a/vendor/github.com/open-policy-agent/opa/topdown/regex.go b/vendor/github.com/open-policy-agent/opa/topdown/regex.go new file mode 100644 index 00000000..7e29db79 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/topdown/regex.go @@ -0,0 +1,218 @@ +// Copyright 2016 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package topdown + +import ( + "fmt" + "regexp" + "sync" + + gintersect "github.com/yashtewari/glob-intersection" + + "github.com/open-policy-agent/opa/ast" + "github.com/open-policy-agent/opa/topdown/builtins" +) + +var regexpCacheLock = sync.Mutex{} +var regexpCache map[string]*regexp.Regexp + +func builtinRegexIsValid(_ BuiltinContext, operands []*ast.Term, iter func(*ast.Term) error) error { + + s, err := builtins.StringOperand(operands[0].Value, 1) + if err != nil { + return iter(ast.BooleanTerm(false)) + } + + _, err = regexp.Compile(string(s)) + if err != nil { + return iter(ast.BooleanTerm(false)) + } + + return iter(ast.BooleanTerm(true)) +} + +func builtinRegexMatch(a, b ast.Value) (ast.Value, error) { + s1, err := builtins.StringOperand(a, 1) + if err != nil { + return nil, err + } + s2, err := builtins.StringOperand(b, 2) + if err != nil { + return nil, err + } + re, err := getRegexp(string(s1)) + if err != nil { + return nil, err + } + return ast.Boolean(re.Match([]byte(s2))), nil +} + +func builtinRegexMatchTemplate(a, b, c, d ast.Value) (ast.Value, error) { + pattern, err := builtins.StringOperand(a, 1) + if err != nil { + return nil, err + } + match, err := builtins.StringOperand(b, 2) + if err != nil { + return nil, err + } + start, err := builtins.StringOperand(c, 3) + if err != nil { + return nil, err + } + end, err := builtins.StringOperand(d, 4) + if err != nil { + return nil, err + } + if len(start) != 1 { + return nil, fmt.Errorf("start delimiter has to be exactly one character long but is %d long", len(start)) + } + if len(end) != 1 { + return nil, fmt.Errorf("end delimiter has to be exactly one character long but is %d long", len(start)) + } + re, err := getRegexpTemplate(string(pattern), string(start)[0], string(end)[0]) + if err != nil { + return nil, err + } + return ast.Boolean(re.MatchString(string(match))), nil +} + +func builtinRegexSplit(a, b ast.Value) (ast.Value, error) { + s1, err := builtins.StringOperand(a, 1) + if err != nil { + return nil, err + } + s2, err := builtins.StringOperand(b, 2) + if err != nil { + return nil, err + } + re, err := getRegexp(string(s1)) + if err != nil { + return nil, err + } + + elems := re.Split(string(s2), -1) + arr := make([]*ast.Term, len(elems)) + for i := range elems { + arr[i] = ast.StringTerm(elems[i]) + } + return ast.NewArray(arr...), nil +} + +func getRegexp(pat string) (*regexp.Regexp, error) { + regexpCacheLock.Lock() + defer regexpCacheLock.Unlock() + re, ok := regexpCache[pat] + if !ok { + var err error + re, err = regexp.Compile(string(pat)) + if err != nil { + return nil, err + } + regexpCache[pat] = re + } + return re, nil +} + +func getRegexpTemplate(pat string, delimStart, delimEnd byte) (*regexp.Regexp, error) { + regexpCacheLock.Lock() + defer regexpCacheLock.Unlock() + re, ok := regexpCache[pat] + if !ok { + var err error + re, err = compileRegexTemplate(string(pat), delimStart, delimEnd) + if err != nil { + return nil, err + } + regexpCache[pat] = re + } + return re, nil +} + +func builtinGlobsMatch(a, b ast.Value) (ast.Value, error) { + s1, err := builtins.StringOperand(a, 1) + if err != nil { + return nil, err + } + s2, err := builtins.StringOperand(b, 2) + if err != nil { + return nil, err + } + ne, err := gintersect.NonEmpty(string(s1), string(s2)) + if err != nil { + return nil, err + } + return ast.Boolean(ne), nil +} + +func builtinRegexFind(a, b, c ast.Value) (ast.Value, error) { + s1, err := builtins.StringOperand(a, 1) + if err != nil { + return nil, err + } + s2, err := builtins.StringOperand(b, 2) + if err != nil { + return nil, err + } + n, err := builtins.IntOperand(c, 3) + if err != nil { + return nil, err + } + re, err := getRegexp(string(s1)) + if err != nil { + return nil, err + } + + elems := re.FindAllString(string(s2), n) + arr := make([]*ast.Term, len(elems)) + for i := range elems { + arr[i] = ast.StringTerm(elems[i]) + } + return ast.NewArray(arr...), nil +} + +func builtinRegexFindAllStringSubmatch(a, b, c ast.Value) (ast.Value, error) { + s1, err := builtins.StringOperand(a, 1) + if err != nil { + return nil, err + } + s2, err := builtins.StringOperand(b, 2) + if err != nil { + return nil, err + } + n, err := builtins.IntOperand(c, 3) + if err != nil { + return nil, err + } + + re, err := getRegexp(string(s1)) + if err != nil { + return nil, err + } + matches := re.FindAllStringSubmatch(string(s2), n) + + outer := make([]*ast.Term, len(matches)) + for i := range matches { + inner := make([]*ast.Term, len(matches[i])) + for j := range matches[i] { + inner[j] = ast.StringTerm(matches[i][j]) + } + outer[i] = ast.NewTerm(ast.NewArray(inner...)) + } + + return ast.NewArray(outer...), nil +} + +func init() { + regexpCache = map[string]*regexp.Regexp{} + RegisterBuiltinFunc(ast.RegexIsValid.Name, builtinRegexIsValid) + RegisterFunctionalBuiltin2(ast.RegexMatch.Name, builtinRegexMatch) + RegisterFunctionalBuiltin2(ast.RegexMatchDeprecated.Name, builtinRegexMatch) + RegisterFunctionalBuiltin2(ast.RegexSplit.Name, builtinRegexSplit) + RegisterFunctionalBuiltin2(ast.GlobsMatch.Name, builtinGlobsMatch) + RegisterFunctionalBuiltin4(ast.RegexTemplateMatch.Name, builtinRegexMatchTemplate) + RegisterFunctionalBuiltin3(ast.RegexFind.Name, builtinRegexFind) + RegisterFunctionalBuiltin3(ast.RegexFindAllStringSubmatch.Name, builtinRegexFindAllStringSubmatch) +} diff --git a/vendor/github.com/open-policy-agent/opa/topdown/regex_template.go b/vendor/github.com/open-policy-agent/opa/topdown/regex_template.go new file mode 100644 index 00000000..39f92346 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/topdown/regex_template.go @@ -0,0 +1,122 @@ +package topdown + +// Copyright 2012 The Gorilla Authors. All rights reserved. +// Use of this source code is governed by a BSD-style +// license as follows: + +// Copyright (c) 2012 Rodrigo Moraes. All rights reserved. +// +// Redistribution and use in source and binary forms, with or without +// modification, are permitted provided that the following conditions are +// met: +// +// * Redistributions of source code must retain the above copyright +// notice, this list of conditions and the following disclaimer. +// * Redistributions in binary form must reproduce the above +// copyright notice, this list of conditions and the following disclaimer +// in the documentation and/or other materials provided with the +// distribution. +// * Neither the name of Google Inc. nor the names of its +// contributors may be used to endorse or promote products derived from +// this software without specific prior written permission. +// +// THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +// "AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +// LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +// A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +// OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +// SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +// LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +// DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +// THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +// (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +// OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. + +// This file was forked from https://github.com/gorilla/mux/commit/eac83ba2c004bb75 + +import ( + "bytes" + "fmt" + "regexp" +) + +// delimiterIndices returns the first level delimiter indices from a string. +// It returns an error in case of unbalanced delimiters. +func delimiterIndices(s string, delimiterStart, delimiterEnd byte) ([]int, error) { + var level, idx int + idxs := make([]int, 0) + for i := 0; i < len(s); i++ { + switch s[i] { + case delimiterStart: + if level++; level == 1 { + idx = i + } + case delimiterEnd: + if level--; level == 0 { + idxs = append(idxs, idx, i+1) + } else if level < 0 { + return nil, fmt.Errorf(`unbalanced braces in %q`, s) + } + } + } + + if level != 0 { + return nil, fmt.Errorf(`unbalanced braces in %q`, s) + } + + return idxs, nil +} + +// compileRegexTemplate parses a template and returns a Regexp. +// +// You can define your own delimiters. It is e.g. common to use curly braces {} but I recommend using characters +// which have no special meaning in Regex, e.g.: <, > +// +// reg, err := compiler.CompileRegex("foo:bar.baz:<[0-9]{2,10}>", '<', '>') +// // if err != nil ... +// reg.MatchString("foo:bar.baz:123") +func compileRegexTemplate(tpl string, delimiterStart, delimiterEnd byte) (*regexp.Regexp, error) { + // Check if it is well-formed. + idxs, errBraces := delimiterIndices(tpl, delimiterStart, delimiterEnd) + if errBraces != nil { + return nil, errBraces + } + varsR := make([]*regexp.Regexp, len(idxs)/2) + pattern := bytes.NewBufferString("") + + // WriteByte's error value is always nil for bytes.Buffer, no need to check it. + pattern.WriteByte('^') + + var end int + var err error + for i := 0; i < len(idxs); i += 2 { + // Set all values we are interested in. + raw := tpl[end:idxs[i]] + end = idxs[i+1] + patt := tpl[idxs[i]+1 : end-1] + // Build the regexp pattern. + varIdx := i / 2 + fmt.Fprintf(pattern, "%s(%s)", regexp.QuoteMeta(raw), patt) + varsR[varIdx], err = regexp.Compile(fmt.Sprintf("^%s$", patt)) + if err != nil { + return nil, err + } + } + + // Add the remaining. + raw := tpl[end:] + + // WriteString's error value is always nil for bytes.Buffer, no need to check it. + pattern.WriteString(regexp.QuoteMeta(raw)) + + // WriteByte's error value is always nil for bytes.Buffer, no need to check it. + pattern.WriteByte('$') + + // Compile full regexp. + reg, errCompile := regexp.Compile(pattern.String()) + if errCompile != nil { + return nil, errCompile + } + + return reg, nil +} diff --git a/vendor/github.com/open-policy-agent/opa/topdown/resolver.go b/vendor/github.com/open-policy-agent/opa/topdown/resolver.go new file mode 100644 index 00000000..5ed6c1e4 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/topdown/resolver.go @@ -0,0 +1,107 @@ +// Copyright 2020 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package topdown + +import ( + "github.com/open-policy-agent/opa/ast" + "github.com/open-policy-agent/opa/metrics" + "github.com/open-policy-agent/opa/resolver" +) + +type resolverTrie struct { + r resolver.Resolver + children map[ast.Value]*resolverTrie +} + +func newResolverTrie() *resolverTrie { + return &resolverTrie{children: map[ast.Value]*resolverTrie{}} +} + +func (t *resolverTrie) Put(ref ast.Ref, r resolver.Resolver) { + node := t + for _, t := range ref { + child, ok := node.children[t.Value] + if !ok { + child = &resolverTrie{children: map[ast.Value]*resolverTrie{}} + node.children[t.Value] = child + } + node = child + } + node.r = r +} + +func (t *resolverTrie) Resolve(e *eval, ref ast.Ref) (ast.Value, error) { + e.metrics.Timer(metrics.RegoExternalResolve).Start() + defer e.metrics.Timer(metrics.RegoExternalResolve).Stop() + node := t + for i, t := range ref { + child, ok := node.children[t.Value] + if !ok { + return nil, nil + } + node = child + if node.r != nil { + in := resolver.Input{ + Ref: ref[:i+1], + Input: e.input, + Metrics: e.metrics, + } + e.traceWasm(e.query[e.index], &in.Ref) + if e.data != nil { + return nil, errInScopeWithStmt + } + result, err := node.r.Eval(e.ctx, in) + if err != nil { + return nil, err + } + if result.Value == nil { + return nil, nil + } + val, err := result.Value.Find(ref[i+1:]) + if err != nil { + return nil, nil + } + return val, nil + } + } + return node.mktree(e, resolver.Input{ + Ref: ref, + Input: e.input, + Metrics: e.metrics, + }) +} + +func (t *resolverTrie) mktree(e *eval, in resolver.Input) (ast.Value, error) { + if t.r != nil { + e.traceWasm(e.query[e.index], &in.Ref) + if e.data != nil { + return nil, errInScopeWithStmt + } + result, err := t.r.Eval(e.ctx, in) + if err != nil { + return nil, err + } + if result.Value == nil { + return nil, nil + } + return result.Value, nil + } + obj := ast.NewObject() + for k, child := range t.children { + v, err := child.mktree(e, resolver.Input{Ref: append(in.Ref, ast.NewTerm(k)), Input: in.Input, Metrics: in.Metrics}) + if err != nil { + return nil, err + } + if v != nil { + obj.Insert(ast.NewTerm(k), ast.NewTerm(v)) + } + } + return obj, nil +} + +var errInScopeWithStmt = &Error{ + Code: InternalErr, + Message: "wasm cannot be executed when 'with' statements are in-scope", +} diff --git a/vendor/github.com/open-policy-agent/opa/topdown/runtime.go b/vendor/github.com/open-policy-agent/opa/topdown/runtime.go new file mode 100644 index 00000000..67e183d0 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/topdown/runtime.go @@ -0,0 +1,20 @@ +// Copyright 2018 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package topdown + +import "github.com/open-policy-agent/opa/ast" + +func builtinOPARuntime(bctx BuiltinContext, _ []*ast.Term, iter func(*ast.Term) error) error { + + if bctx.Runtime == nil { + return iter(ast.ObjectTerm()) + } + + return iter(bctx.Runtime) +} + +func init() { + RegisterBuiltinFunc(ast.OPARuntime.Name, builtinOPARuntime) +} diff --git a/vendor/github.com/open-policy-agent/opa/topdown/save.go b/vendor/github.com/open-policy-agent/opa/topdown/save.go new file mode 100644 index 00000000..4ea644f0 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/topdown/save.go @@ -0,0 +1,427 @@ +package topdown + +import ( + "container/list" + "fmt" + "strings" + + "github.com/open-policy-agent/opa/ast" +) + +// saveSet contains a stack of terms that are considered 'unknown' during +// partial evaluation. Only var and ref terms (rooted at one of the root +// documents) can be added to the save set. Vars added to the save set are +// namespaced by the binding list they are added with. This means the save set +// can be shared across queries. +type saveSet struct { + instr *Instrumentation + l *list.List +} + +func newSaveSet(ts []*ast.Term, b *bindings, instr *Instrumentation) *saveSet { + ss := &saveSet{ + l: list.New(), + instr: instr, + } + ss.Push(ts, b) + return ss +} + +func (ss *saveSet) Push(ts []*ast.Term, b *bindings) { + ss.l.PushBack(newSaveSetElem(ts, b)) +} + +func (ss *saveSet) Pop() { + ss.l.Remove(ss.l.Back()) +} + +// Contains returns true if the term t is contained in the save set. Non-var and +// non-ref terms are never contained. Ref terms are contained if they share a +// prefix with a ref that was added (in either direction). +func (ss *saveSet) Contains(t *ast.Term, b *bindings) bool { + if ss != nil { + ss.instr.startTimer(partialOpSaveSetContains) + ret := ss.contains(t, b) + ss.instr.stopTimer(partialOpSaveSetContains) + return ret + } + return false +} + +func (ss *saveSet) contains(t *ast.Term, b *bindings) bool { + for el := ss.l.Back(); el != nil; el = el.Prev() { + if el.Value.(*saveSetElem).Contains(t, b) { + return true + } + } + return false +} + +// ContainsRecursive returns true if the term t is or contains a term that is +// contained in the save set. This function will close over the binding list +// when it encounters vars. +func (ss *saveSet) ContainsRecursive(t *ast.Term, b *bindings) bool { + if ss != nil { + ss.instr.startTimer(partialOpSaveSetContainsRec) + ret := ss.containsrec(t, b) + ss.instr.stopTimer(partialOpSaveSetContainsRec) + return ret + } + return false +} + +func (ss *saveSet) containsrec(t *ast.Term, b *bindings) bool { + var found bool + ast.WalkTerms(t, func(x *ast.Term) bool { + if _, ok := x.Value.(ast.Var); ok { + x1, b1 := b.apply(x) + if x1 != x || b1 != b { + if ss.containsrec(x1, b1) { + found = true + } + } else if ss.contains(x1, b1) { + found = true + } + } + return found + }) + return found +} + +func (ss *saveSet) Vars(caller *bindings) ast.VarSet { + result := ast.NewVarSet() + for x := ss.l.Front(); x != nil; x = x.Next() { + elem := x.Value.(*saveSetElem) + for _, v := range elem.vars { + if v, ok := elem.b.PlugNamespaced(v, caller).Value.(ast.Var); ok { + result.Add(v) + } + } + } + return result +} + +func (ss *saveSet) String() string { + var buf []string + + for x := ss.l.Front(); x != nil; x = x.Next() { + buf = append(buf, x.Value.(*saveSetElem).String()) + } + + return "(" + strings.Join(buf, " ") + ")" +} + +type saveSetElem struct { + refs []ast.Ref + vars []*ast.Term + b *bindings +} + +func newSaveSetElem(ts []*ast.Term, b *bindings) *saveSetElem { + + var refs []ast.Ref + var vars []*ast.Term + + for _, t := range ts { + switch v := t.Value.(type) { + case ast.Var: + vars = append(vars, t) + case ast.Ref: + refs = append(refs, v) + default: + panic("illegal value") + } + } + + return &saveSetElem{ + b: b, + vars: vars, + refs: refs, + } +} + +func (sse *saveSetElem) Contains(t *ast.Term, b *bindings) bool { + switch other := t.Value.(type) { + case ast.Var: + return sse.containsVar(t, b) + case ast.Ref: + for _, ref := range sse.refs { + if ref.HasPrefix(other) || other.HasPrefix(ref) { + return true + } + } + return sse.containsVar(other[0], b) + } + return false +} + +func (sse *saveSetElem) String() string { + return fmt.Sprintf("(refs: %v, vars: %v, b: %v)", sse.refs, sse.vars, sse.b) +} + +func (sse *saveSetElem) containsVar(t *ast.Term, b *bindings) bool { + if b == sse.b { + for _, v := range sse.vars { + if v.Equal(t) { + return true + } + } + } + return false +} + +// saveStack contains a stack of queries that represent the result of partial +// evaluation. When partial evaluation completes, the top of the stack +// represents a complete, partially evaluated query that can be saved and +// evaluated later. +// +// The result is stored in a stack so that partial evaluation of a query can be +// paused and then resumed in cases where different queries make up the result +// of partial evaluation, such as when a rule with a default clause is +// partially evaluated. In this case, the partially evaluated rule will be +// output in the support module. +type saveStack struct { + Stack []saveStackQuery +} + +func newSaveStack() *saveStack { + return &saveStack{ + Stack: []saveStackQuery{ + {}, + }, + } +} + +func (s *saveStack) PushQuery(query saveStackQuery) { + s.Stack = append(s.Stack, query) +} + +func (s *saveStack) PopQuery() saveStackQuery { + last := s.Stack[len(s.Stack)-1] + s.Stack = s.Stack[:len(s.Stack)-1] + return last +} + +func (s *saveStack) Peek() saveStackQuery { + return s.Stack[len(s.Stack)-1] +} + +func (s *saveStack) Push(expr *ast.Expr, b1 *bindings, b2 *bindings) { + idx := len(s.Stack) - 1 + s.Stack[idx] = append(s.Stack[idx], saveStackElem{expr, b1, b2}) +} + +func (s *saveStack) Pop() { + idx := len(s.Stack) - 1 + query := s.Stack[idx] + s.Stack[idx] = query[:len(query)-1] +} + +type saveStackQuery []saveStackElem + +func (s saveStackQuery) Plug(b *bindings) ast.Body { + if len(s) == 0 { + return ast.NewBody(ast.NewExpr(ast.BooleanTerm(true))) + } + result := make(ast.Body, len(s)) + for i := range s { + expr := s[i].Plug(b) + result.Set(expr, i) + } + return result +} + +type saveStackElem struct { + Expr *ast.Expr + B1 *bindings + B2 *bindings +} + +func (e saveStackElem) Plug(caller *bindings) *ast.Expr { + if e.B1 == nil && e.B2 == nil { + return e.Expr + } + expr := e.Expr.Copy() + switch terms := expr.Terms.(type) { + case []*ast.Term: + if expr.IsEquality() { + terms[1] = e.B1.PlugNamespaced(terms[1], caller) + terms[2] = e.B2.PlugNamespaced(terms[2], caller) + } else { + for i := 1; i < len(terms); i++ { + terms[i] = e.B1.PlugNamespaced(terms[i], caller) + } + } + case *ast.Term: + expr.Terms = e.B1.PlugNamespaced(terms, caller) + } + for i := range expr.With { + expr.With[i].Value = e.B1.PlugNamespaced(expr.With[i].Value, caller) + } + return expr +} + +// saveSupport contains additional partially evaluated policies that are part +// of the output of partial evaluation. +// +// The support structure is accumulated as partial evaluation runs and then +// considered complete once partial evaluation finishes (but not before). This +// differs from partially evaluated queries which are considered complete as +// soon as each one finishes. +type saveSupport struct { + modules map[string]*ast.Module +} + +func newSaveSupport() *saveSupport { + return &saveSupport{ + modules: map[string]*ast.Module{}, + } +} + +func (s *saveSupport) List() []*ast.Module { + result := []*ast.Module{} + for _, module := range s.modules { + result = append(result, module) + } + return result +} + +func (s *saveSupport) Exists(path ast.Ref) bool { + k := path[:len(path)-1].String() + module, ok := s.modules[k] + if !ok { + return false + } + name := ast.Var(path[len(path)-1].Value.(ast.String)) + for _, rule := range module.Rules { + if rule.Head.Name.Equal(name) { + return true + } + } + return false +} + +func (s *saveSupport) Insert(path ast.Ref, rule *ast.Rule) { + pkg := path[:len(path)-1] + k := pkg.String() + module, ok := s.modules[k] + if !ok { + module = &ast.Module{ + Package: &ast.Package{ + Path: pkg, + }, + } + s.modules[k] = module + } + rule.Module = module + module.Rules = append(module.Rules, rule) +} + +// saveRequired returns true if the statement x will result in some expressions +// being saved. This check allows the evaluator to evaluate statements +// completely during partial evaluation as long as they do not depend on any +// kind of unknown value or statements that would generate saves. +func saveRequired(c *ast.Compiler, ic *inliningControl, icIgnoreInternal bool, ss *saveSet, b *bindings, x interface{}, rec bool) bool { + + var found bool + + vis := ast.NewGenericVisitor(func(node interface{}) bool { + if found { + return found + } + switch node := node.(type) { + case *ast.Expr: + found = len(node.With) > 0 || ignoreExprDuringPartial(node) + case *ast.Term: + switch v := node.Value.(type) { + case ast.Var: + // Variables only need to be tested in the node from call site + // because once traversal recurses into a rule existing unknown + // variables are out-of-scope. + if !rec && ss.ContainsRecursive(node, b) { + found = true + } + case ast.Ref: + if ss.Contains(node, b) { + found = true + } else if ic.Disabled(v.ConstantPrefix(), icIgnoreInternal) { + found = true + } else { + for _, rule := range c.GetRulesDynamicWithOpts(v, ast.RulesOptions{IncludeHiddenModules: false}) { + if saveRequired(c, ic, icIgnoreInternal, ss, b, rule, true) { + found = true + break + } + } + } + } + } + return found + }) + + vis.Walk(x) + + return found +} + +func ignoreExprDuringPartial(expr *ast.Expr) bool { + if !expr.IsCall() { + return false + } + + bi, ok := ast.BuiltinMap[expr.Operator().String()] + + return ok && ignoreDuringPartial(bi) +} + +func ignoreDuringPartial(bi *ast.Builtin) bool { + for _, ignore := range ast.IgnoreDuringPartialEval { + if bi == ignore { + return true + } + } + return false +} + +type inliningControl struct { + shallow bool + disable []disableInliningFrame +} + +type disableInliningFrame struct { + internal bool + refs []ast.Ref +} + +func (i *inliningControl) PushDisable(refs []ast.Ref, internal bool) { + if i == nil { + return + } + i.disable = append(i.disable, disableInliningFrame{ + internal: internal, + refs: refs, + }) +} + +func (i *inliningControl) PopDisable() { + if i == nil { + return + } + i.disable = i.disable[:len(i.disable)-1] +} + +func (i *inliningControl) Disabled(ref ast.Ref, ignoreInternal bool) bool { + if i == nil { + return false + } + for _, frame := range i.disable { + if !frame.internal || !ignoreInternal { + for _, other := range frame.refs { + if other.HasPrefix(ref) || ref.HasPrefix(other) { + return true + } + } + } + } + return false +} diff --git a/vendor/github.com/open-policy-agent/opa/topdown/semver.go b/vendor/github.com/open-policy-agent/opa/topdown/semver.go new file mode 100644 index 00000000..b91a5a99 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/topdown/semver.go @@ -0,0 +1,60 @@ +// Copyright 2020 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package topdown + +import ( + "fmt" + + "github.com/open-policy-agent/opa/ast" + "github.com/open-policy-agent/opa/internal/semver" + "github.com/open-policy-agent/opa/topdown/builtins" +) + +func builtinSemVerCompare(bctx BuiltinContext, args []*ast.Term, iter func(*ast.Term) error) error { + versionStringA, err := builtins.StringOperand(args[0].Value, 1) + if err != nil { + return err + } + + versionStringB, err := builtins.StringOperand(args[1].Value, 2) + if err != nil { + return err + } + + versionA, err := semver.NewVersion(string(versionStringA)) + if err != nil { + return fmt.Errorf("operand 1: string %s is not a valid SemVer", versionStringA) + } + versionB, err := semver.NewVersion(string(versionStringB)) + if err != nil { + return fmt.Errorf("operand 2: string %s is not a valid SemVer", versionStringB) + } + + result := versionA.Compare(*versionB) + + return iter(ast.IntNumberTerm(result)) +} + +func builtinSemVerIsValid(bctx BuiltinContext, args []*ast.Term, iter func(*ast.Term) error) error { + versionString, err := builtins.StringOperand(args[0].Value, 1) + if err != nil { + result := ast.BooleanTerm(false) + return iter(result) + } + + result := true + + _, err = semver.NewVersion(string(versionString)) + if err != nil { + result = false + } + + return iter(ast.BooleanTerm(result)) +} + +func init() { + RegisterBuiltinFunc(ast.SemVerCompare.Name, builtinSemVerCompare) + RegisterBuiltinFunc(ast.SemVerIsValid.Name, builtinSemVerIsValid) +} diff --git a/vendor/github.com/open-policy-agent/opa/topdown/sets.go b/vendor/github.com/open-policy-agent/opa/topdown/sets.go new file mode 100644 index 00000000..a9c5ad86 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/topdown/sets.go @@ -0,0 +1,84 @@ +// Copyright 2016 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package topdown + +import ( + "github.com/open-policy-agent/opa/ast" + "github.com/open-policy-agent/opa/topdown/builtins" +) + +// Deprecated in v0.4.2 in favour of minus/infix "-" operation. +func builtinSetDiff(a, b ast.Value) (ast.Value, error) { + + s1, err := builtins.SetOperand(a, 1) + if err != nil { + return nil, err + } + + s2, err := builtins.SetOperand(b, 2) + if err != nil { + return nil, err + } + + return s1.Diff(s2), nil +} + +// builtinSetIntersection returns the intersection of the given input sets +func builtinSetIntersection(a ast.Value) (ast.Value, error) { + + inputSet, err := builtins.SetOperand(a, 1) + if err != nil { + return nil, err + } + + // empty input set + if inputSet.Len() == 0 { + return ast.NewSet(), nil + } + + var result ast.Set + + err = inputSet.Iter(func(x *ast.Term) error { + n, err := builtins.SetOperand(x.Value, 1) + if err != nil { + return err + } + + if result == nil { + result = n + } else { + result = result.Intersect(n) + } + return nil + }) + return result, err +} + +// builtinSetUnion returns the union of the given input sets +func builtinSetUnion(a ast.Value) (ast.Value, error) { + + inputSet, err := builtins.SetOperand(a, 1) + if err != nil { + return nil, err + } + + result := ast.NewSet() + + err = inputSet.Iter(func(x *ast.Term) error { + n, err := builtins.SetOperand(x.Value, 1) + if err != nil { + return err + } + result = result.Union(n) + return nil + }) + return result, err +} + +func init() { + RegisterFunctionalBuiltin2(ast.SetDiff.Name, builtinSetDiff) + RegisterFunctionalBuiltin1(ast.Intersection.Name, builtinSetIntersection) + RegisterFunctionalBuiltin1(ast.Union.Name, builtinSetUnion) +} diff --git a/vendor/github.com/open-policy-agent/opa/topdown/strings.go b/vendor/github.com/open-policy-agent/opa/topdown/strings.go new file mode 100644 index 00000000..e5118b77 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/topdown/strings.go @@ -0,0 +1,455 @@ +// Copyright 2016 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package topdown + +import ( + "fmt" + "math/big" + "sort" + "strings" + + "github.com/open-policy-agent/opa/ast" + "github.com/open-policy-agent/opa/topdown/builtins" +) + +func builtinFormatInt(a, b ast.Value) (ast.Value, error) { + + input, err := builtins.NumberOperand(a, 1) + if err != nil { + return nil, err + } + + base, err := builtins.NumberOperand(b, 2) + if err != nil { + return nil, err + } + + var format string + switch base { + case ast.Number("2"): + format = "%b" + case ast.Number("8"): + format = "%o" + case ast.Number("10"): + format = "%d" + case ast.Number("16"): + format = "%x" + default: + return nil, builtins.NewOperandEnumErr(2, "2", "8", "10", "16") + } + + f := builtins.NumberToFloat(input) + i, _ := f.Int(nil) + + return ast.String(fmt.Sprintf(format, i)), nil +} + +func builtinConcat(a, b ast.Value) (ast.Value, error) { + + join, err := builtins.StringOperand(a, 1) + if err != nil { + return nil, err + } + + strs := []string{} + + switch b := b.(type) { + case *ast.Array: + err := b.Iter(func(x *ast.Term) error { + s, ok := x.Value.(ast.String) + if !ok { + return builtins.NewOperandElementErr(2, b, x.Value, "string") + } + strs = append(strs, string(s)) + return nil + }) + if err != nil { + return nil, err + } + case ast.Set: + err := b.Iter(func(x *ast.Term) error { + s, ok := x.Value.(ast.String) + if !ok { + return builtins.NewOperandElementErr(2, b, x.Value, "string") + } + strs = append(strs, string(s)) + return nil + }) + if err != nil { + return nil, err + } + default: + return nil, builtins.NewOperandTypeErr(2, b, "set", "array") + } + + return ast.String(strings.Join(strs, string(join))), nil +} + +func builtinIndexOf(a, b ast.Value) (ast.Value, error) { + runesEqual := func(a, b []rune) bool { + if len(a) != len(b) { + return false + } + for i, v := range a { + if v != b[i] { + return false + } + } + return true + } + + base, err := builtins.StringOperand(a, 1) + if err != nil { + return nil, err + } + + search, err := builtins.StringOperand(b, 2) + if err != nil { + return nil, err + } + if len(string(search)) == 0 { + return nil, fmt.Errorf("empty search character") + } + + baseRunes := []rune(string(base)) + searchRunes := []rune(string(search)) + searchLen := len(searchRunes) + + for i, r := range baseRunes { + if r == searchRunes[0] { + if len(baseRunes) >= i+searchLen { + if runesEqual(baseRunes[i:i+searchLen], searchRunes) { + return ast.IntNumberTerm(i).Value, nil + } + } + } + } + + return ast.IntNumberTerm(-1).Value, nil +} + +func builtinSubstring(a, b, c ast.Value) (ast.Value, error) { + + base, err := builtins.StringOperand(a, 1) + if err != nil { + return nil, err + } + runes := []rune(base) + + startIndex, err := builtins.IntOperand(b, 2) + if err != nil { + return nil, err + } else if startIndex >= len(runes) { + return ast.String(""), nil + } else if startIndex < 0 { + return nil, fmt.Errorf("negative offset") + } + + length, err := builtins.IntOperand(c, 3) + if err != nil { + return nil, err + } + + var s ast.String + if length < 0 { + s = ast.String(runes[startIndex:]) + } else { + upto := startIndex + length + if len(runes) < upto { + upto = len(runes) + } + s = ast.String(runes[startIndex:upto]) + } + + return s, nil +} + +func builtinContains(a, b ast.Value) (ast.Value, error) { + s, err := builtins.StringOperand(a, 1) + if err != nil { + return nil, err + } + + substr, err := builtins.StringOperand(b, 2) + if err != nil { + return nil, err + } + + return ast.Boolean(strings.Contains(string(s), string(substr))), nil +} + +func builtinStartsWith(a, b ast.Value) (ast.Value, error) { + s, err := builtins.StringOperand(a, 1) + if err != nil { + return nil, err + } + + prefix, err := builtins.StringOperand(b, 2) + if err != nil { + return nil, err + } + + return ast.Boolean(strings.HasPrefix(string(s), string(prefix))), nil +} + +func builtinEndsWith(a, b ast.Value) (ast.Value, error) { + s, err := builtins.StringOperand(a, 1) + if err != nil { + return nil, err + } + + suffix, err := builtins.StringOperand(b, 2) + if err != nil { + return nil, err + } + + return ast.Boolean(strings.HasSuffix(string(s), string(suffix))), nil +} + +func builtinLower(a ast.Value) (ast.Value, error) { + s, err := builtins.StringOperand(a, 1) + if err != nil { + return nil, err + } + + return ast.String(strings.ToLower(string(s))), nil +} + +func builtinUpper(a ast.Value) (ast.Value, error) { + s, err := builtins.StringOperand(a, 1) + if err != nil { + return nil, err + } + + return ast.String(strings.ToUpper(string(s))), nil +} + +func builtinSplit(a, b ast.Value) (ast.Value, error) { + s, err := builtins.StringOperand(a, 1) + if err != nil { + return nil, err + } + d, err := builtins.StringOperand(b, 2) + if err != nil { + return nil, err + } + elems := strings.Split(string(s), string(d)) + arr := make([]*ast.Term, len(elems)) + for i := range elems { + arr[i] = ast.StringTerm(elems[i]) + } + return ast.NewArray(arr...), nil +} + +func builtinReplace(a, b, c ast.Value) (ast.Value, error) { + s, err := builtins.StringOperand(a, 1) + if err != nil { + return nil, err + } + + old, err := builtins.StringOperand(b, 2) + if err != nil { + return nil, err + } + + new, err := builtins.StringOperand(c, 3) + if err != nil { + return nil, err + } + + return ast.String(strings.Replace(string(s), string(old), string(new), -1)), nil +} + +func builtinReplaceN(a, b ast.Value) (ast.Value, error) { + patterns, err := builtins.ObjectOperand(a, 1) + if err != nil { + return nil, err + } + keys := patterns.Keys() + sort.Slice(keys, func(i, j int) bool { return ast.Compare(keys[i].Value, keys[j].Value) < 0 }) + + s, err := builtins.StringOperand(b, 2) + if err != nil { + return nil, err + } + + var oldnewArr []string + for _, k := range keys { + keyVal, ok := k.Value.(ast.String) + if !ok { + return nil, builtins.NewOperandErr(1, "non-string key found in pattern object") + } + val := patterns.Get(k) // cannot be nil + strVal, ok := val.Value.(ast.String) + if !ok { + return nil, builtins.NewOperandErr(1, "non-string value found in pattern object") + } + oldnewArr = append(oldnewArr, string(keyVal), string(strVal)) + } + if err != nil { + return nil, err + } + + r := strings.NewReplacer(oldnewArr...) + replaced := r.Replace(string(s)) + + return ast.String(replaced), nil +} + +func builtinTrim(a, b ast.Value) (ast.Value, error) { + s, err := builtins.StringOperand(a, 1) + if err != nil { + return nil, err + } + + c, err := builtins.StringOperand(b, 2) + if err != nil { + return nil, err + } + + return ast.String(strings.Trim(string(s), string(c))), nil +} + +func builtinTrimLeft(a, b ast.Value) (ast.Value, error) { + s, err := builtins.StringOperand(a, 1) + if err != nil { + return nil, err + } + + c, err := builtins.StringOperand(b, 2) + if err != nil { + return nil, err + } + + return ast.String(strings.TrimLeft(string(s), string(c))), nil +} + +func builtinTrimPrefix(a, b ast.Value) (ast.Value, error) { + s, err := builtins.StringOperand(a, 1) + if err != nil { + return nil, err + } + + pre, err := builtins.StringOperand(b, 2) + if err != nil { + return nil, err + } + + return ast.String(strings.TrimPrefix(string(s), string(pre))), nil +} + +func builtinTrimRight(a, b ast.Value) (ast.Value, error) { + s, err := builtins.StringOperand(a, 1) + if err != nil { + return nil, err + } + + c, err := builtins.StringOperand(b, 2) + if err != nil { + return nil, err + } + + return ast.String(strings.TrimRight(string(s), string(c))), nil +} + +func builtinTrimSuffix(a, b ast.Value) (ast.Value, error) { + s, err := builtins.StringOperand(a, 1) + if err != nil { + return nil, err + } + + suf, err := builtins.StringOperand(b, 2) + if err != nil { + return nil, err + } + + return ast.String(strings.TrimSuffix(string(s), string(suf))), nil +} + +func builtinTrimSpace(a ast.Value) (ast.Value, error) { + s, err := builtins.StringOperand(a, 1) + if err != nil { + return nil, err + } + + return ast.String(strings.TrimSpace(string(s))), nil +} + +func builtinSprintf(a, b ast.Value) (ast.Value, error) { + s, err := builtins.StringOperand(a, 1) + if err != nil { + return nil, err + } + + astArr, ok := b.(*ast.Array) + if !ok { + return nil, builtins.NewOperandTypeErr(2, b, "array") + } + + args := make([]interface{}, astArr.Len()) + + for i := range args { + switch v := astArr.Elem(i).Value.(type) { + case ast.Number: + if n, ok := v.Int(); ok { + args[i] = n + } else if b, ok := new(big.Int).SetString(v.String(), 10); ok { + args[i] = b + } else if f, ok := v.Float64(); ok { + args[i] = f + } else { + args[i] = v.String() + } + case ast.String: + args[i] = string(v) + default: + args[i] = astArr.Elem(i).String() + } + } + + return ast.String(fmt.Sprintf(string(s), args...)), nil +} + +func builtinReverse(bctx BuiltinContext, operands []*ast.Term, iter func(*ast.Term) error) error { + s, err := builtins.StringOperand(operands[0].Value, 1) + if err != nil { + return err + } + + sRunes := []rune(string(s)) + length := len(sRunes) + reversedRunes := make([]rune, length) + + for index, r := range sRunes { + reversedRunes[length-index-1] = r + } + + reversedString := string(reversedRunes) + + return iter(ast.StringTerm(reversedString)) +} + +func init() { + RegisterFunctionalBuiltin2(ast.FormatInt.Name, builtinFormatInt) + RegisterFunctionalBuiltin2(ast.Concat.Name, builtinConcat) + RegisterFunctionalBuiltin2(ast.IndexOf.Name, builtinIndexOf) + RegisterFunctionalBuiltin3(ast.Substring.Name, builtinSubstring) + RegisterFunctionalBuiltin2(ast.Contains.Name, builtinContains) + RegisterFunctionalBuiltin2(ast.StartsWith.Name, builtinStartsWith) + RegisterFunctionalBuiltin2(ast.EndsWith.Name, builtinEndsWith) + RegisterFunctionalBuiltin1(ast.Upper.Name, builtinUpper) + RegisterFunctionalBuiltin1(ast.Lower.Name, builtinLower) + RegisterFunctionalBuiltin2(ast.Split.Name, builtinSplit) + RegisterFunctionalBuiltin3(ast.Replace.Name, builtinReplace) + RegisterFunctionalBuiltin2(ast.ReplaceN.Name, builtinReplaceN) + RegisterFunctionalBuiltin2(ast.Trim.Name, builtinTrim) + RegisterFunctionalBuiltin2(ast.TrimLeft.Name, builtinTrimLeft) + RegisterFunctionalBuiltin2(ast.TrimPrefix.Name, builtinTrimPrefix) + RegisterFunctionalBuiltin2(ast.TrimRight.Name, builtinTrimRight) + RegisterFunctionalBuiltin2(ast.TrimSuffix.Name, builtinTrimSuffix) + RegisterFunctionalBuiltin1(ast.TrimSpace.Name, builtinTrimSpace) + RegisterFunctionalBuiltin2(ast.Sprintf.Name, builtinSprintf) + RegisterBuiltinFunc(ast.StringReverse.Name, builtinReverse) +} diff --git a/vendor/github.com/open-policy-agent/opa/topdown/time.go b/vendor/github.com/open-policy-agent/opa/topdown/time.go new file mode 100644 index 00000000..481326da --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/topdown/time.go @@ -0,0 +1,297 @@ +// Copyright 2017 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package topdown + +import ( + "encoding/json" + "fmt" + "math" + "math/big" + "strconv" + "sync" + "time" + + "github.com/open-policy-agent/opa/ast" + "github.com/open-policy-agent/opa/topdown/builtins" +) + +var tzCache map[string]*time.Location +var tzCacheMutex *sync.Mutex + +// 1677-09-21T00:12:43.145224192-00:00 +var minDateAllowedForNsConversion = time.Unix(0, math.MinInt64) + +// 2262-04-11T23:47:16.854775807-00:00 +var maxDateAllowedForNsConversion = time.Unix(0, math.MaxInt64) + +func toSafeUnixNano(t time.Time, iter func(*ast.Term) error) error { + if t.Before(minDateAllowedForNsConversion) || t.After(maxDateAllowedForNsConversion) { + return fmt.Errorf("time outside of valid range") + } + + return iter(ast.NewTerm(ast.Number(int64ToJSONNumber(t.UnixNano())))) +} + +func builtinTimeNowNanos(bctx BuiltinContext, _ []*ast.Term, iter func(*ast.Term) error) error { + return iter(bctx.Time) +} + +func builtinTimeParseNanos(_ BuiltinContext, operands []*ast.Term, iter func(*ast.Term) error) error { + a := operands[0].Value + format, err := builtins.StringOperand(a, 1) + if err != nil { + return err + } + + b := operands[1].Value + value, err := builtins.StringOperand(b, 2) + if err != nil { + return err + } + + result, err := time.Parse(string(format), string(value)) + if err != nil { + return err + } + + return toSafeUnixNano(result, iter) +} + +func builtinTimeParseRFC3339Nanos(_ BuiltinContext, operands []*ast.Term, iter func(*ast.Term) error) error { + a := operands[0].Value + value, err := builtins.StringOperand(a, 1) + if err != nil { + return err + } + + result, err := time.Parse(time.RFC3339, string(value)) + if err != nil { + return err + } + + return toSafeUnixNano(result, iter) +} +func builtinParseDurationNanos(a ast.Value) (ast.Value, error) { + + duration, err := builtins.StringOperand(a, 1) + if err != nil { + return nil, err + } + value, err := time.ParseDuration(string(duration)) + if err != nil { + return nil, err + } + return ast.Number(int64ToJSONNumber(int64(value))), nil +} + +func builtinDate(a ast.Value) (ast.Value, error) { + t, err := tzTime(a) + if err != nil { + return nil, err + } + year, month, day := t.Date() + result := ast.NewArray(ast.IntNumberTerm(year), ast.IntNumberTerm(int(month)), ast.IntNumberTerm(day)) + return result, nil +} + +func builtinClock(a ast.Value) (ast.Value, error) { + t, err := tzTime(a) + if err != nil { + return nil, err + } + hour, minute, second := t.Clock() + result := ast.NewArray(ast.IntNumberTerm(hour), ast.IntNumberTerm(minute), ast.IntNumberTerm(second)) + return result, nil +} + +func builtinWeekday(a ast.Value) (ast.Value, error) { + t, err := tzTime(a) + if err != nil { + return nil, err + } + weekday := t.Weekday().String() + return ast.String(weekday), nil +} + +func builtinAddDate(_ BuiltinContext, operands []*ast.Term, iter func(*ast.Term) error) error { + t, err := tzTime(operands[0].Value) + if err != nil { + return err + } + + years, err := builtins.IntOperand(operands[1].Value, 2) + if err != nil { + return err + } + + months, err := builtins.IntOperand(operands[2].Value, 3) + if err != nil { + return err + } + + days, err := builtins.IntOperand(operands[3].Value, 4) + if err != nil { + return err + } + + result := t.AddDate(years, months, days) + + return toSafeUnixNano(result, iter) +} + +func builtinDiff(_ BuiltinContext, operands []*ast.Term, iter func(*ast.Term) error) error { + t1, err := tzTime(operands[0].Value) + if err != nil { + return err + } + t2, err := tzTime(operands[1].Value) + if err != nil { + return err + } + + // The following implementation of this function is taken + // from https://github.com/icza/gox licensed under Apache 2.0. + // The only modification made is to variable names. + // + // For details, see https://stackoverflow.com/a/36531443/1705598 + // + // Copyright 2021 icza + // BEGIN REDISTRIBUTION FROM APACHE 2.0 LICENSED PROJECT + if t1.Location() != t2.Location() { + t2 = t2.In(t1.Location()) + } + if t1.After(t2) { + t1, t2 = t2, t1 + } + y1, M1, d1 := t1.Date() + y2, M2, d2 := t2.Date() + + h1, m1, s1 := t1.Clock() + h2, m2, s2 := t2.Clock() + + year := y2 - y1 + month := int(M2 - M1) + day := d2 - d1 + hour := h2 - h1 + min := m2 - m1 + sec := s2 - s1 + + // Normalize negative values + if sec < 0 { + sec += 60 + min-- + } + if min < 0 { + min += 60 + hour-- + } + if hour < 0 { + hour += 24 + day-- + } + if day < 0 { + // Days in month: + t := time.Date(y1, M1, 32, 0, 0, 0, 0, time.UTC) + day += 32 - t.Day() + month-- + } + if month < 0 { + month += 12 + year-- + } + // END REDISTRIBUTION FROM APACHE 2.0 LICENSED PROJECT + + return iter(ast.ArrayTerm(ast.IntNumberTerm(year), ast.IntNumberTerm(month), ast.IntNumberTerm(day), + ast.IntNumberTerm(hour), ast.IntNumberTerm(min), ast.IntNumberTerm(sec))) +} + +func tzTime(a ast.Value) (t time.Time, err error) { + var nVal ast.Value + loc := time.UTC + + switch va := a.(type) { + case *ast.Array: + if va.Len() == 0 { + return time.Time{}, builtins.NewOperandTypeErr(1, a, "either number (ns) or [number (ns), string (tz)]") + } + + nVal, err = builtins.NumberOperand(va.Elem(0).Value, 1) + if err != nil { + return time.Time{}, err + } + + if va.Len() > 1 { + tzVal, err := builtins.StringOperand(va.Elem(1).Value, 1) + if err != nil { + return time.Time{}, err + } + + tzName := string(tzVal) + + switch tzName { + case "", "UTC": + // loc is already UTC + + case "Local": + loc = time.Local + + default: + var ok bool + + tzCacheMutex.Lock() + loc, ok = tzCache[tzName] + + if !ok { + loc, err = time.LoadLocation(tzName) + if err != nil { + tzCacheMutex.Unlock() + return time.Time{}, err + } + tzCache[tzName] = loc + } + tzCacheMutex.Unlock() + } + } + + case ast.Number: + nVal = a + + default: + return time.Time{}, builtins.NewOperandTypeErr(1, a, "either number (ns) or [number (ns), string (tz)]") + } + + value, err := builtins.NumberOperand(nVal, 1) + if err != nil { + return time.Time{}, err + } + + f := builtins.NumberToFloat(value) + i64, acc := f.Int64() + if acc != big.Exact { + return time.Time{}, fmt.Errorf("timestamp too big") + } + + t = time.Unix(0, i64).In(loc) + + return t, nil +} + +func int64ToJSONNumber(i int64) json.Number { + return json.Number(strconv.FormatInt(i, 10)) +} + +func init() { + RegisterBuiltinFunc(ast.NowNanos.Name, builtinTimeNowNanos) + RegisterBuiltinFunc(ast.ParseRFC3339Nanos.Name, builtinTimeParseRFC3339Nanos) + RegisterBuiltinFunc(ast.ParseNanos.Name, builtinTimeParseNanos) + RegisterFunctionalBuiltin1(ast.ParseDurationNanos.Name, builtinParseDurationNanos) + RegisterFunctionalBuiltin1(ast.Date.Name, builtinDate) + RegisterFunctionalBuiltin1(ast.Clock.Name, builtinClock) + RegisterFunctionalBuiltin1(ast.Weekday.Name, builtinWeekday) + RegisterBuiltinFunc(ast.AddDate.Name, builtinAddDate) + RegisterBuiltinFunc(ast.Diff.Name, builtinDiff) + tzCacheMutex = &sync.Mutex{} + tzCache = make(map[string]*time.Location) +} diff --git a/vendor/github.com/open-policy-agent/opa/topdown/tokens.go b/vendor/github.com/open-policy-agent/opa/topdown/tokens.go new file mode 100644 index 00000000..64828831 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/topdown/tokens.go @@ -0,0 +1,1113 @@ +// Copyright 2017 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package topdown + +import ( + "crypto" + "crypto/ecdsa" + "crypto/hmac" + "crypto/rsa" + "crypto/sha256" + "crypto/sha512" + "crypto/x509" + "encoding/hex" + "encoding/json" + "encoding/pem" + "fmt" + "hash" + "math/big" + "strings" + + "github.com/pkg/errors" + + "github.com/open-policy-agent/opa/ast" + "github.com/open-policy-agent/opa/internal/jwx/jwk" + "github.com/open-policy-agent/opa/internal/jwx/jws" + "github.com/open-policy-agent/opa/topdown/builtins" +) + +var ( + jwtEncKey = ast.StringTerm("enc") + jwtCtyKey = ast.StringTerm("cty") + jwtIssKey = ast.StringTerm("iss") + jwtExpKey = ast.StringTerm("exp") + jwtNbfKey = ast.StringTerm("nbf") + jwtAudKey = ast.StringTerm("aud") +) + +const ( + headerJwt = "JWT" +) + +// JSONWebToken represent the 3 parts (header, payload & signature) of +// a JWT in Base64. +type JSONWebToken struct { + header string + payload string + signature string + decodedHeader ast.Object +} + +// decodeHeader populates the decodedHeader field. +func (token *JSONWebToken) decodeHeader() error { + h, err := builtinBase64UrlDecode(ast.String(token.header)) + if err != nil { + return fmt.Errorf("JWT header had invalid encoding: %w", err) + } + decodedHeader, err := validateJWTHeader(string(h.(ast.String))) + if err != nil { + return err + } + token.decodedHeader = decodedHeader + return nil +} + +// Implements JWT decoding/validation based on RFC 7519 Section 7.2: +// https://tools.ietf.org/html/rfc7519#section-7.2 +// It does no data validation, it merely checks that the given string +// represents a structurally valid JWT. It supports JWTs using JWS compact +// serialization. +func builtinJWTDecode(a ast.Value) (ast.Value, error) { + token, err := decodeJWT(a) + if err != nil { + return nil, err + } + + if err = token.decodeHeader(); err != nil { + return nil, err + } + + p, err := builtinBase64UrlDecode(ast.String(token.payload)) + if err != nil { + return nil, fmt.Errorf("JWT payload had invalid encoding: %v", err) + } + + if cty := token.decodedHeader.Get(jwtCtyKey); cty != nil { + ctyVal := string(cty.Value.(ast.String)) + // It is possible for the contents of a token to be another + // token as a result of nested signing or encryption. To handle + // the case where we are given a token such as this, we check + // the content type and recurse on the payload if the content + // is "JWT". + // When the payload is itself another encoded JWT, then its + // contents are quoted (behavior of https://jwt.io/). To fix + // this, remove leading and trailing quotes. + if ctyVal == headerJwt { + p, err = builtinTrim(p, ast.String(`"'`)) + if err != nil { + panic("not reached") + } + return builtinJWTDecode(p) + } + } + + payload, err := extractJSONObject(string(p.(ast.String))) + if err != nil { + return nil, err + } + + s, err := builtinBase64UrlDecode(ast.String(token.signature)) + if err != nil { + return nil, fmt.Errorf("JWT signature had invalid encoding: %v", err) + } + sign := hex.EncodeToString([]byte(s.(ast.String))) + + arr := []*ast.Term{ + ast.NewTerm(token.decodedHeader), + ast.NewTerm(payload), + ast.StringTerm(sign), + } + + return ast.NewArray(arr...), nil +} + +// Implements RS256 JWT signature verification +func builtinJWTVerifyRS256(bctx BuiltinContext, args []*ast.Term, iter func(*ast.Term) error) error { + result, err := builtinJWTVerifyRSA(args[0].Value, args[1].Value, sha256.New, func(publicKey *rsa.PublicKey, digest []byte, signature []byte) error { + return rsa.VerifyPKCS1v15( + publicKey, + crypto.SHA256, + digest, + signature) + }) + if err == nil { + return iter(ast.NewTerm(result)) + } + return err +} + +// Implements RS384 JWT signature verification +func builtinJWTVerifyRS384(bctx BuiltinContext, args []*ast.Term, iter func(*ast.Term) error) error { + result, err := builtinJWTVerifyRSA(args[0].Value, args[1].Value, sha512.New384, func(publicKey *rsa.PublicKey, digest []byte, signature []byte) error { + return rsa.VerifyPKCS1v15( + publicKey, + crypto.SHA384, + digest, + signature) + }) + if err == nil { + return iter(ast.NewTerm(result)) + } + return err +} + +// Implements RS512 JWT signature verification +func builtinJWTVerifyRS512(bctx BuiltinContext, args []*ast.Term, iter func(*ast.Term) error) error { + result, err := builtinJWTVerifyRSA(args[0].Value, args[1].Value, sha512.New, func(publicKey *rsa.PublicKey, digest []byte, signature []byte) error { + return rsa.VerifyPKCS1v15( + publicKey, + crypto.SHA512, + digest, + signature) + }) + if err == nil { + return iter(ast.NewTerm(result)) + } + return err +} + +// Implements PS256 JWT signature verification +func builtinJWTVerifyPS256(bctx BuiltinContext, args []*ast.Term, iter func(*ast.Term) error) error { + result, err := builtinJWTVerifyRSA(args[0].Value, args[1].Value, sha256.New, func(publicKey *rsa.PublicKey, digest []byte, signature []byte) error { + return rsa.VerifyPSS( + publicKey, + crypto.SHA256, + digest, + signature, + nil) + }) + if err == nil { + return iter(ast.NewTerm(result)) + } + return err +} + +// Implements PS384 JWT signature verification +func builtinJWTVerifyPS384(bctx BuiltinContext, args []*ast.Term, iter func(*ast.Term) error) error { + result, err := builtinJWTVerifyRSA(args[0].Value, args[1].Value, sha512.New384, func(publicKey *rsa.PublicKey, digest []byte, signature []byte) error { + return rsa.VerifyPSS( + publicKey, + crypto.SHA384, + digest, + signature, + nil) + }) + if err == nil { + return iter(ast.NewTerm(result)) + } + return err +} + +// Implements PS512 JWT signature verification +func builtinJWTVerifyPS512(bctx BuiltinContext, args []*ast.Term, iter func(*ast.Term) error) error { + result, err := builtinJWTVerifyRSA(args[0].Value, args[1].Value, sha512.New, func(publicKey *rsa.PublicKey, digest []byte, signature []byte) error { + return rsa.VerifyPSS( + publicKey, + crypto.SHA512, + digest, + signature, + nil) + }) + if err == nil { + return iter(ast.NewTerm(result)) + } + return err +} + +// Implements RSA JWT signature verification. +func builtinJWTVerifyRSA(a ast.Value, b ast.Value, hasher func() hash.Hash, verify func(publicKey *rsa.PublicKey, digest []byte, signature []byte) error) (ast.Value, error) { + return builtinJWTVerify(a, b, hasher, func(publicKey interface{}, digest []byte, signature []byte) error { + publicKeyRsa, ok := publicKey.(*rsa.PublicKey) + if !ok { + return fmt.Errorf("incorrect public key type") + } + return verify(publicKeyRsa, digest, signature) + }) +} + +// Implements ES256 JWT signature verification. +func builtinJWTVerifyES256(bctx BuiltinContext, args []*ast.Term, iter func(*ast.Term) error) error { + result, err := builtinJWTVerify(args[0].Value, args[1].Value, sha256.New, verifyES) + if err == nil { + return iter(ast.NewTerm(result)) + } + return err +} + +// Implements ES384 JWT signature verification +func builtinJWTVerifyES384(bctx BuiltinContext, args []*ast.Term, iter func(*ast.Term) error) error { + result, err := builtinJWTVerify(args[0].Value, args[1].Value, sha512.New384, verifyES) + if err == nil { + return iter(ast.NewTerm(result)) + } + return err +} + +// Implements ES512 JWT signature verification +func builtinJWTVerifyES512(bctx BuiltinContext, args []*ast.Term, iter func(*ast.Term) error) error { + result, err := builtinJWTVerify(args[0].Value, args[1].Value, sha512.New, verifyES) + if err == nil { + return iter(ast.NewTerm(result)) + } + return err +} + +func verifyES(publicKey interface{}, digest []byte, signature []byte) error { + publicKeyEcdsa, ok := publicKey.(*ecdsa.PublicKey) + if !ok { + return fmt.Errorf("incorrect public key type") + } + r, s := &big.Int{}, &big.Int{} + n := len(signature) / 2 + r.SetBytes(signature[:n]) + s.SetBytes(signature[n:]) + if ecdsa.Verify(publicKeyEcdsa, digest, r, s) { + return nil + } + return fmt.Errorf("ECDSA signature verification error") +} + +// getKeyFromCertOrJWK returns the public key found in a X.509 certificate or JWK key(s). +// A valid PEM block is never valid JSON (and vice versa), hence can try parsing both. +func getKeyFromCertOrJWK(certificate string) ([]interface{}, error) { + if block, rest := pem.Decode([]byte(certificate)); block != nil { + if len(rest) > 0 { + return nil, fmt.Errorf("extra data after a PEM certificate block") + } + + if block.Type == blockTypeCertificate { + cert, err := x509.ParseCertificate(block.Bytes) + if err != nil { + return nil, errors.Wrap(err, "failed to parse a PEM certificate") + } + + return []interface{}{cert.PublicKey}, nil + } + + if block.Type == "PUBLIC KEY" { + key, err := x509.ParsePKIXPublicKey(block.Bytes) + if err != nil { + return nil, errors.Wrap(err, "failed to parse a PEM public key") + } + + return []interface{}{key}, nil + } + + return nil, fmt.Errorf("failed to extract a Key from the PEM certificate") + } + + jwks, err := jwk.ParseString(certificate) + if err != nil { + return nil, errors.Wrap(err, "failed to parse a JWK key (set)") + } + + var keys []interface{} + for _, k := range jwks.Keys { + key, err := k.Materialize() + if err != nil { + return nil, err + } + keys = append(keys, key) + } + + return keys, nil +} + +// Implements JWT signature verification. +func builtinJWTVerify(a ast.Value, b ast.Value, hasher func() hash.Hash, verify func(publicKey interface{}, digest []byte, signature []byte) error) (ast.Value, error) { + token, err := decodeJWT(a) + if err != nil { + return nil, err + } + + s, err := builtins.StringOperand(b, 2) + if err != nil { + return nil, err + } + + keys, err := getKeyFromCertOrJWK(string(s)) + if err != nil { + return nil, err + } + + signature, err := token.decodeSignature() + if err != nil { + return nil, err + } + + // Validate the JWT signature + for _, key := range keys { + err = verify(key, + getInputSHA([]byte(token.header+"."+token.payload), hasher), + []byte(signature)) + + if err == nil { + return ast.Boolean(true), nil + } + } + + // None of the keys worked, return false + return ast.Boolean(false), nil +} + +// Implements HS256 (secret) JWT signature verification +func builtinJWTVerifyHS256(bctx BuiltinContext, args []*ast.Term, iter func(*ast.Term) error) error { + // Decode the JSON Web Token + token, err := decodeJWT(args[0].Value) + if err != nil { + return err + } + + // Process Secret input + astSecret, err := builtins.StringOperand(args[1].Value, 2) + if err != nil { + return err + } + secret := string(astSecret) + + mac := hmac.New(sha256.New, []byte(secret)) + _, err = mac.Write([]byte(token.header + "." + token.payload)) + if err != nil { + return err + } + + signature, err := token.decodeSignature() + if err != nil { + return err + } + + return iter(ast.NewTerm(ast.Boolean(hmac.Equal([]byte(signature), mac.Sum(nil))))) +} + +// Implements HS384 JWT signature verification +func builtinJWTVerifyHS384(bctx BuiltinContext, args []*ast.Term, iter func(*ast.Term) error) error { + // Decode the JSON Web Token + token, err := decodeJWT(args[0].Value) + if err != nil { + return err + } + + // Process Secret input + astSecret, err := builtins.StringOperand(args[1].Value, 2) + if err != nil { + return err + } + secret := string(astSecret) + + mac := hmac.New(sha512.New384, []byte(secret)) + _, err = mac.Write([]byte(token.header + "." + token.payload)) + if err != nil { + return err + } + + signature, err := token.decodeSignature() + if err != nil { + return err + } + + return iter(ast.NewTerm(ast.Boolean(hmac.Equal([]byte(signature), mac.Sum(nil))))) +} + +// Implements HS512 JWT signature verification +func builtinJWTVerifyHS512(bctx BuiltinContext, args []*ast.Term, iter func(*ast.Term) error) error { + // Decode the JSON Web Token + token, err := decodeJWT(args[0].Value) + if err != nil { + return err + } + + // Process Secret input + astSecret, err := builtins.StringOperand(args[1].Value, 2) + if err != nil { + return err + } + secret := string(astSecret) + + mac := hmac.New(sha512.New, []byte(secret)) + _, err = mac.Write([]byte(token.header + "." + token.payload)) + if err != nil { + return err + } + + signature, err := token.decodeSignature() + if err != nil { + return err + } + + return iter(ast.NewTerm(ast.Boolean(hmac.Equal([]byte(signature), mac.Sum(nil))))) +} + +// -- Full JWT verification and decoding -- + +// Verification constraints. See tokens_test.go for unit tests. + +// tokenConstraints holds decoded JWT verification constraints. +type tokenConstraints struct { + // The set of asymmetric keys we can verify with. + keys []interface{} + + // The single symmetric key we will verify with. + secret string + + // The algorithm that must be used to verify. + // If "", any algorithm is acceptable. + alg string + + // The required issuer. + // If "", any issuer is acceptable. + iss string + + // The required audience. + // If "", no audience is acceptable. + aud string + + // The time to validate against, or -1 if no constraint set. + // (If unset, the current time will be used.) + time float64 +} + +// tokenConstraintHandler is the handler type for JWT verification constraints. +type tokenConstraintHandler func(value ast.Value, parameters *tokenConstraints) error + +// tokenConstraintTypes maps known JWT verification constraints to handlers. +var tokenConstraintTypes = map[string]tokenConstraintHandler{ + "cert": tokenConstraintCert, + "secret": func(value ast.Value, constraints *tokenConstraints) error { + return tokenConstraintString("secret", value, &constraints.secret) + }, + "alg": func(value ast.Value, constraints *tokenConstraints) error { + return tokenConstraintString("alg", value, &constraints.alg) + }, + "iss": func(value ast.Value, constraints *tokenConstraints) error { + return tokenConstraintString("iss", value, &constraints.iss) + }, + "aud": func(value ast.Value, constraints *tokenConstraints) error { + return tokenConstraintString("aud", value, &constraints.aud) + }, + "time": tokenConstraintTime, +} + +// tokenConstraintCert handles the `cert` constraint. +func tokenConstraintCert(value ast.Value, constraints *tokenConstraints) error { + s, ok := value.(ast.String) + if !ok { + return fmt.Errorf("cert constraint: must be a string") + } + + keys, err := getKeyFromCertOrJWK(string(s)) + if err != nil { + return err + } + constraints.keys = keys + return nil +} + +// tokenConstraintTime handles the `time` constraint. +func tokenConstraintTime(value ast.Value, constraints *tokenConstraints) error { + t, err := timeFromValue(value) + if err != nil { + return err + } + constraints.time = t + return nil +} + +func timeFromValue(value ast.Value) (float64, error) { + time, ok := value.(ast.Number) + if !ok { + return 0, fmt.Errorf("token time constraint: must be a number") + } + timeFloat, ok := time.Float64() + if !ok { + return 0, fmt.Errorf("token time constraint: unvalid float64") + } + if timeFloat < 0 { + return 0, fmt.Errorf("token time constraint: must not be negative") + } + return timeFloat, nil +} + +// tokenConstraintString handles string constraints. +func tokenConstraintString(name string, value ast.Value, where *string) error { + av, ok := value.(ast.String) + if !ok { + return fmt.Errorf("%s constraint: must be a string", name) + } + *where = string(av) + return nil +} + +// parseTokenConstraints parses the constraints argument. +func parseTokenConstraints(o ast.Object, wallclock *ast.Term) (*tokenConstraints, error) { + constraints := tokenConstraints{ + time: -1, + } + if err := o.Iter(func(k *ast.Term, v *ast.Term) error { + name := string(k.Value.(ast.String)) + handler, ok := tokenConstraintTypes[name] + if ok { + return handler(v.Value, &constraints) + } + // Anything unknown is rejected. + return fmt.Errorf("unknown token validation constraint: %s", name) + }); err != nil { + return nil, err + } + if constraints.time == -1 { // no time provided in constraint object + t, err := timeFromValue(wallclock.Value) + if err != nil { + return nil, err + } + constraints.time = t + } + return &constraints, nil +} + +// validate validates the constraints argument. +func (constraints *tokenConstraints) validate() error { + keys := 0 + if constraints.keys != nil { + keys++ + } + if constraints.secret != "" { + keys++ + } + if keys > 1 { + return fmt.Errorf("duplicate key constraints") + } + if keys < 1 { + return fmt.Errorf("no key constraint") + } + return nil +} + +// verify verifies a JWT using the constraints and the algorithm from the header +func (constraints *tokenConstraints) verify(kid, alg, header, payload, signature string) error { + // Construct the payload + plaintext := []byte(header) + plaintext = append(plaintext, []byte(".")...) + plaintext = append(plaintext, payload...) + // Look up the algorithm + a, ok := tokenAlgorithms[alg] + if !ok { + return fmt.Errorf("unknown JWS algorithm: %s", alg) + } + // If we're configured with asymmetric key(s) then only trust that + if constraints.keys != nil { + verified := false + for _, key := range constraints.keys { + err := a.verify(key, a.hash, plaintext, []byte(signature)) + if err == nil { + verified = true + break + } + } + if !verified { + return errSignatureNotVerified + } + return nil + } + if constraints.secret != "" { + return a.verify([]byte(constraints.secret), a.hash, plaintext, []byte(signature)) + } + // (*tokenConstraints)validate() should prevent this happening + return errors.New("unexpectedly found no keys to trust") +} + +// validAudience checks the audience of the JWT. +// It returns true if it meets the constraints and false otherwise. +func (constraints *tokenConstraints) validAudience(aud ast.Value) bool { + s, ok := aud.(ast.String) + if ok { + return string(s) == constraints.aud + } + a, ok := aud.(*ast.Array) + if !ok { + return false + } + return a.Until(func(elem *ast.Term) bool { + if s, ok := elem.Value.(ast.String); ok { + return string(s) == constraints.aud + } + return false + }) +} + +// JWT algorithms + +type tokenVerifyFunction func(key interface{}, hash crypto.Hash, payload []byte, signature []byte) error +type tokenVerifyAsymmetricFunction func(key interface{}, hash crypto.Hash, digest []byte, signature []byte) error + +// jwtAlgorithm describes a JWS 'alg' value +type tokenAlgorithm struct { + hash crypto.Hash + verify tokenVerifyFunction +} + +// tokenAlgorithms is the known JWT algorithms +var tokenAlgorithms = map[string]tokenAlgorithm{ + "RS256": {crypto.SHA256, verifyAsymmetric(verifyRSAPKCS)}, + "RS384": {crypto.SHA384, verifyAsymmetric(verifyRSAPKCS)}, + "RS512": {crypto.SHA512, verifyAsymmetric(verifyRSAPKCS)}, + "PS256": {crypto.SHA256, verifyAsymmetric(verifyRSAPSS)}, + "PS384": {crypto.SHA384, verifyAsymmetric(verifyRSAPSS)}, + "PS512": {crypto.SHA512, verifyAsymmetric(verifyRSAPSS)}, + "ES256": {crypto.SHA256, verifyAsymmetric(verifyECDSA)}, + "ES384": {crypto.SHA384, verifyAsymmetric(verifyECDSA)}, + "ES512": {crypto.SHA512, verifyAsymmetric(verifyECDSA)}, + "HS256": {crypto.SHA256, verifyHMAC}, + "HS384": {crypto.SHA384, verifyHMAC}, + "HS512": {crypto.SHA512, verifyHMAC}, +} + +// errSignatureNotVerified is returned when a signature cannot be verified. +var errSignatureNotVerified = errors.New("signature not verified") + +func verifyHMAC(key interface{}, hash crypto.Hash, payload []byte, signature []byte) error { + macKey, ok := key.([]byte) + if !ok { + return fmt.Errorf("incorrect symmetric key type") + } + mac := hmac.New(hash.New, macKey) + if _, err := mac.Write([]byte(payload)); err != nil { + return err + } + if !hmac.Equal(signature, mac.Sum([]byte{})) { + return errSignatureNotVerified + } + return nil +} + +func verifyAsymmetric(verify tokenVerifyAsymmetricFunction) tokenVerifyFunction { + return func(key interface{}, hash crypto.Hash, payload []byte, signature []byte) error { + h := hash.New() + h.Write(payload) + return verify(key, hash, h.Sum([]byte{}), signature) + } +} + +func verifyRSAPKCS(key interface{}, hash crypto.Hash, digest []byte, signature []byte) error { + publicKeyRsa, ok := key.(*rsa.PublicKey) + if !ok { + return fmt.Errorf("incorrect public key type") + } + if err := rsa.VerifyPKCS1v15(publicKeyRsa, hash, digest, signature); err != nil { + return errSignatureNotVerified + } + return nil +} + +func verifyRSAPSS(key interface{}, hash crypto.Hash, digest []byte, signature []byte) error { + publicKeyRsa, ok := key.(*rsa.PublicKey) + if !ok { + return fmt.Errorf("incorrect public key type") + } + if err := rsa.VerifyPSS(publicKeyRsa, hash, digest, signature, nil); err != nil { + return errSignatureNotVerified + } + return nil +} + +func verifyECDSA(key interface{}, hash crypto.Hash, digest []byte, signature []byte) error { + publicKeyEcdsa, ok := key.(*ecdsa.PublicKey) + if !ok { + return fmt.Errorf("incorrect public key type") + } + r, s := &big.Int{}, &big.Int{} + n := len(signature) / 2 + r.SetBytes(signature[:n]) + s.SetBytes(signature[n:]) + if ecdsa.Verify(publicKeyEcdsa, digest, r, s) { + return nil + } + return errSignatureNotVerified +} + +// JWT header parsing and parameters. See tokens_test.go for unit tests. + +// tokenHeaderType represents a recognized JWT header field +// tokenHeader is a parsed JWT header +type tokenHeader struct { + alg string + kid string + typ string + cty string + crit map[string]bool + unknown []string +} + +// tokenHeaderHandler handles a JWT header parameters +type tokenHeaderHandler func(header *tokenHeader, value ast.Value) error + +// tokenHeaderTypes maps known JWT header parameters to handlers +var tokenHeaderTypes = map[string]tokenHeaderHandler{ + "alg": func(header *tokenHeader, value ast.Value) error { + return tokenHeaderString("alg", &header.alg, value) + }, + "kid": func(header *tokenHeader, value ast.Value) error { + return tokenHeaderString("kid", &header.kid, value) + }, + "typ": func(header *tokenHeader, value ast.Value) error { + return tokenHeaderString("typ", &header.typ, value) + }, + "cty": func(header *tokenHeader, value ast.Value) error { + return tokenHeaderString("cty", &header.cty, value) + }, + "crit": tokenHeaderCrit, +} + +// tokenHeaderCrit handles the 'crit' header parameter +func tokenHeaderCrit(header *tokenHeader, value ast.Value) error { + v, ok := value.(*ast.Array) + if !ok { + return fmt.Errorf("crit: must be a list") + } + header.crit = map[string]bool{} + _ = v.Iter(func(elem *ast.Term) error { + tv, ok := elem.Value.(ast.String) + if !ok { + return fmt.Errorf("crit: must be a list of strings") + } + header.crit[string(tv)] = true + return nil + }) + if len(header.crit) == 0 { + return fmt.Errorf("crit: must be a nonempty list") // 'MUST NOT' use the empty list + } + return nil +} + +// tokenHeaderString handles string-format JWT header parameters +func tokenHeaderString(name string, where *string, value ast.Value) error { + v, ok := value.(ast.String) + if !ok { + return fmt.Errorf("%s: must be a string", name) + } + *where = string(v) + return nil +} + +// parseTokenHeader parses the JWT header. +func parseTokenHeader(token *JSONWebToken) (*tokenHeader, error) { + header := tokenHeader{ + unknown: []string{}, + } + if err := token.decodedHeader.Iter(func(k *ast.Term, v *ast.Term) error { + ks := string(k.Value.(ast.String)) + handler, ok := tokenHeaderTypes[ks] + if !ok { + header.unknown = append(header.unknown, ks) + return nil + } + return handler(&header, v.Value) + }); err != nil { + return nil, err + } + return &header, nil +} + +// validTokenHeader returns true if the JOSE header is valid, otherwise false. +func (header *tokenHeader) valid() bool { + // RFC7515 s4.1.1 alg MUST be present + if header.alg == "" { + return false + } + // RFC7515 4.1.11 JWS is invalid if there is a critical parameter that we did not recognize + for _, u := range header.unknown { + if header.crit[u] { + return false + } + } + return true +} + +func commonBuiltinJWTEncodeSign(bctx BuiltinContext, inputHeaders, jwsPayload, jwkSrc string, iter func(*ast.Term) error) error { + + keys, err := jwk.ParseString(jwkSrc) + if err != nil { + return err + } + key, err := keys.Keys[0].Materialize() + if err != nil { + return err + } + if jwk.GetKeyTypeFromKey(key) != keys.Keys[0].GetKeyType() { + return fmt.Errorf("JWK derived key type and keyType parameter do not match") + } + + standardHeaders := &jws.StandardHeaders{} + jwsHeaders := []byte(inputHeaders) + err = json.Unmarshal(jwsHeaders, standardHeaders) + if err != nil { + return err + } + alg := standardHeaders.GetAlgorithm() + + if (standardHeaders.Type == "" || standardHeaders.Type == headerJwt) && !json.Valid([]byte(jwsPayload)) { + return fmt.Errorf("type is JWT but payload is not JSON") + } + + // process payload and sign + var jwsCompact []byte + jwsCompact, err = jws.SignLiteral([]byte(jwsPayload), alg, key, jwsHeaders, bctx.Seed) + if err != nil { + return err + } + return iter(ast.StringTerm(string(jwsCompact))) + +} + +func builtinJWTEncodeSign(bctx BuiltinContext, args []*ast.Term, iter func(*ast.Term) error) error { + + inputHeaders := args[0].String() + jwsPayload := args[1].String() + jwkSrc := args[2].String() + return commonBuiltinJWTEncodeSign(bctx, inputHeaders, jwsPayload, jwkSrc, iter) + +} + +func builtinJWTEncodeSignRaw(bctx BuiltinContext, args []*ast.Term, iter func(*ast.Term) error) error { + + jwkSrc, err := builtins.StringOperand(args[2].Value, 3) + if err != nil { + return err + } + inputHeaders, err := builtins.StringOperand(args[0].Value, 1) + if err != nil { + return err + } + jwsPayload, err := builtins.StringOperand(args[1].Value, 2) + if err != nil { + return err + } + return commonBuiltinJWTEncodeSign(bctx, string(inputHeaders), string(jwsPayload), string(jwkSrc), iter) +} + +// Implements full JWT decoding, validation and verification. +func builtinJWTDecodeVerify(bctx BuiltinContext, args []*ast.Term, iter func(*ast.Term) error) error { + // io.jwt.decode_verify(string, constraints, [valid, header, payload]) + // + // If valid is true then the signature verifies and all constraints are met. + // If valid is false then either the signature did not verify or some constrain + // was not met. + // + // Decoding errors etc are returned as errors. + a := args[0].Value + + b, err := builtins.ObjectOperand(args[1].Value, 2) + if err != nil { + return err + } + + unverified := ast.ArrayTerm( + ast.BooleanTerm(false), + ast.NewTerm(ast.NewObject()), + ast.NewTerm(ast.NewObject()), + ) + constraints, err := parseTokenConstraints(b, bctx.Time) + if err != nil { + return err + } + if err := constraints.validate(); err != nil { + return err + } + var token *JSONWebToken + var p ast.Value + for { + // RFC7519 7.2 #1-2 split into parts + if token, err = decodeJWT(a); err != nil { + return err + } + // RFC7519 7.2 #3, #4, #6 + if err := token.decodeHeader(); err != nil { + return err + } + // RFC7159 7.2 #5 (and RFC7159 5.2 #5) validate header fields + header, err := parseTokenHeader(token) + if err != nil { + return err + } + if !header.valid() { + return iter(unverified) + } + // Check constraints that impact signature verification. + if constraints.alg != "" && constraints.alg != header.alg { + return iter(unverified) + } + // RFC7159 7.2 #7 verify the signature + signature, err := token.decodeSignature() + if err != nil { + return err + } + if err := constraints.verify(header.kid, header.alg, token.header, token.payload, signature); err != nil { + if err == errSignatureNotVerified { + return iter(unverified) + } + return err + } + // RFC7159 7.2 #9-10 decode the payload + p, err = builtinBase64UrlDecode(ast.String(token.payload)) + if err != nil { + return fmt.Errorf("JWT payload had invalid encoding: %v", err) + } + // RFC7159 7.2 #8 and 5.2 cty + if strings.ToUpper(header.cty) == headerJwt { + // Nested JWT, go round again with payload as first argument + a = p + continue + } else { + // Non-nested JWT (or we've reached the bottom of the nesting). + break + } + } + payload, err := extractJSONObject(string(p.(ast.String))) + if err != nil { + return err + } + // Check registered claim names against constraints or environment + // RFC7159 4.1.1 iss + if constraints.iss != "" { + if iss := payload.Get(jwtIssKey); iss != nil { + issVal := string(iss.Value.(ast.String)) + if constraints.iss != issVal { + return iter(unverified) + } + } + } + // RFC7159 4.1.3 aud + if aud := payload.Get(jwtAudKey); aud != nil { + if !constraints.validAudience(aud.Value) { + return iter(unverified) + } + } else { + if constraints.aud != "" { + return iter(unverified) + } + } + // RFC7159 4.1.4 exp + if exp := payload.Get(jwtExpKey); exp != nil { + // constraints.time is in nanoseconds but exp Value is in seconds + compareTime := ast.FloatNumberTerm(float64(constraints.time) / 1000000000) + if ast.Compare(compareTime, exp.Value.(ast.Number)) != -1 { + return iter(unverified) + } + } + // RFC7159 4.1.5 nbf + if nbf := payload.Get(jwtNbfKey); nbf != nil { + // constraints.time is in nanoseconds but nbf Value is in seconds + compareTime := ast.FloatNumberTerm(float64(constraints.time) / 1000000000) + if ast.Compare(compareTime, nbf.Value.(ast.Number)) == -1 { + return iter(unverified) + } + } + + verified := ast.ArrayTerm( + ast.BooleanTerm(true), + ast.NewTerm(token.decodedHeader), + ast.NewTerm(payload), + ) + return iter(verified) +} + +// -- Utilities -- + +func decodeJWT(a ast.Value) (*JSONWebToken, error) { + // Parse the JSON Web Token + astEncode, err := builtins.StringOperand(a, 1) + if err != nil { + return nil, err + } + + encoding := string(astEncode) + if !strings.Contains(encoding, ".") { + return nil, errors.New("encoded JWT had no period separators") + } + + parts := strings.Split(encoding, ".") + if len(parts) != 3 { + return nil, fmt.Errorf("encoded JWT must have 3 sections, found %d", len(parts)) + } + + return &JSONWebToken{header: parts[0], payload: parts[1], signature: parts[2]}, nil +} + +func (token *JSONWebToken) decodeSignature() (string, error) { + decodedSignature, err := builtinBase64UrlDecode(ast.String(token.signature)) + if err != nil { + return "", err + } + + signatureAst, err := builtins.StringOperand(decodedSignature, 1) + if err != nil { + return "", err + } + return string(signatureAst), err +} + +// Extract, validate and return the JWT header as an ast.Object. +func validateJWTHeader(h string) (ast.Object, error) { + header, err := extractJSONObject(h) + if err != nil { + return nil, fmt.Errorf("bad JWT header: %v", err) + } + + // There are two kinds of JWT tokens, a JSON Web Signature (JWS) and + // a JSON Web Encryption (JWE). The latter is very involved, and we + // won't support it for now. + // This code checks which kind of JWT we are dealing with according to + // RFC 7516 Section 9: https://tools.ietf.org/html/rfc7516#section-9 + if header.Get(jwtEncKey) != nil { + return nil, errors.New("JWT is a JWE object, which is not supported") + } + + return header, nil +} + +func extractJSONObject(s string) (ast.Object, error) { + // XXX: This code relies on undocumented behavior of Go's + // json.Unmarshal using the last occurrence of duplicate keys in a JSON + // Object. If duplicate keys are present in a JWT, the last must be + // used or the token rejected. Since detecting duplicates is tantamount + // to parsing it ourselves, we're relying on the Go implementation + // using the last occurring instance of the key, which is the behavior + // as of Go 1.8.1. + v, err := builtinJSONUnmarshal(ast.String(s)) + if err != nil { + return nil, fmt.Errorf("invalid JSON: %v", err) + } + + o, ok := v.(ast.Object) + if !ok { + return nil, errors.New("decoded JSON type was not an Object") + } + + return o, nil +} + +// getInputSha returns the SHA checksum of the input +func getInputSHA(input []byte, h func() hash.Hash) []byte { + hasher := h() + hasher.Write(input) + return hasher.Sum(nil) +} + +func init() { + RegisterFunctionalBuiltin1(ast.JWTDecode.Name, builtinJWTDecode) + RegisterBuiltinFunc(ast.JWTVerifyRS256.Name, builtinJWTVerifyRS256) + RegisterBuiltinFunc(ast.JWTVerifyRS384.Name, builtinJWTVerifyRS384) + RegisterBuiltinFunc(ast.JWTVerifyRS512.Name, builtinJWTVerifyRS512) + RegisterBuiltinFunc(ast.JWTVerifyPS256.Name, builtinJWTVerifyPS256) + RegisterBuiltinFunc(ast.JWTVerifyPS384.Name, builtinJWTVerifyPS384) + RegisterBuiltinFunc(ast.JWTVerifyPS512.Name, builtinJWTVerifyPS512) + RegisterBuiltinFunc(ast.JWTVerifyES256.Name, builtinJWTVerifyES256) + RegisterBuiltinFunc(ast.JWTVerifyES384.Name, builtinJWTVerifyES384) + RegisterBuiltinFunc(ast.JWTVerifyES512.Name, builtinJWTVerifyES512) + RegisterBuiltinFunc(ast.JWTVerifyHS256.Name, builtinJWTVerifyHS256) + RegisterBuiltinFunc(ast.JWTVerifyHS384.Name, builtinJWTVerifyHS384) + RegisterBuiltinFunc(ast.JWTVerifyHS512.Name, builtinJWTVerifyHS512) + RegisterBuiltinFunc(ast.JWTDecodeVerify.Name, builtinJWTDecodeVerify) + RegisterBuiltinFunc(ast.JWTEncodeSignRaw.Name, builtinJWTEncodeSignRaw) + RegisterBuiltinFunc(ast.JWTEncodeSign.Name, builtinJWTEncodeSign) +} diff --git a/vendor/github.com/open-policy-agent/opa/topdown/trace.go b/vendor/github.com/open-policy-agent/opa/topdown/trace.go new file mode 100644 index 00000000..49aa5aff --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/topdown/trace.go @@ -0,0 +1,423 @@ +// Copyright 2016 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package topdown + +import ( + "fmt" + "io" + "strings" + + iStrs "github.com/open-policy-agent/opa/internal/strings" + + "github.com/open-policy-agent/opa/ast" + "github.com/open-policy-agent/opa/topdown/builtins" +) + +const ( + minLocationWidth = 5 // len("query") + maxIdealLocationWidth = 64 + locationPadding = 4 +) + +// Op defines the types of tracing events. +type Op string + +const ( + // EnterOp is emitted when a new query is about to be evaluated. + EnterOp Op = "Enter" + + // ExitOp is emitted when a query has evaluated to true. + ExitOp Op = "Exit" + + // EvalOp is emitted when an expression is about to be evaluated. + EvalOp Op = "Eval" + + // RedoOp is emitted when an expression, rule, or query is being re-evaluated. + RedoOp Op = "Redo" + + // SaveOp is emitted when an expression is saved instead of evaluated + // during partial evaluation. + SaveOp Op = "Save" + + // FailOp is emitted when an expression evaluates to false. + FailOp Op = "Fail" + + // DuplicateOp is emitted when a query has produced a duplicate value. The search + // will stop at the point where the duplicate was emitted and backtrack. + DuplicateOp Op = "Duplicate" + + // NoteOp is emitted when an expression invokes a tracing built-in function. + NoteOp Op = "Note" + + // IndexOp is emitted during an expression evaluation to represent lookup + // matches. + IndexOp Op = "Index" + + // WasmOp is emitted when resolving a ref using an external + // Resolver. + WasmOp Op = "Wasm" +) + +// VarMetadata provides some user facing information about +// a variable in some policy. +type VarMetadata struct { + Name ast.Var `json:"name"` + Location *ast.Location `json:"location"` +} + +// Event contains state associated with a tracing event. +type Event struct { + Op Op // Identifies type of event. + Node ast.Node // Contains AST node relevant to the event. + Location *ast.Location // The location of the Node this event relates to. + QueryID uint64 // Identifies the query this event belongs to. + ParentID uint64 // Identifies the parent query this event belongs to. + Locals *ast.ValueMap // Contains local variable bindings from the query context. Nil if variables were not included in the trace event. + LocalMetadata map[ast.Var]VarMetadata // Contains metadata for the local variable bindings. Nil if variables were not included in the trace event. + Message string // Contains message for Note events. + Ref *ast.Ref // Identifies the subject ref for the event. Only applies to Index and Wasm operations. +} + +// HasRule returns true if the Event contains an ast.Rule. +func (evt *Event) HasRule() bool { + _, ok := evt.Node.(*ast.Rule) + return ok +} + +// HasBody returns true if the Event contains an ast.Body. +func (evt *Event) HasBody() bool { + _, ok := evt.Node.(ast.Body) + return ok +} + +// HasExpr returns true if the Event contains an ast.Expr. +func (evt *Event) HasExpr() bool { + _, ok := evt.Node.(*ast.Expr) + return ok +} + +// Equal returns true if this event is equal to the other event. +func (evt *Event) Equal(other *Event) bool { + if evt.Op != other.Op { + return false + } + if evt.QueryID != other.QueryID { + return false + } + if evt.ParentID != other.ParentID { + return false + } + if !evt.equalNodes(other) { + return false + } + return evt.Locals.Equal(other.Locals) +} + +func (evt *Event) String() string { + return fmt.Sprintf("%v %v %v (qid=%v, pqid=%v)", evt.Op, evt.Node, evt.Locals, evt.QueryID, evt.ParentID) +} + +func (evt *Event) equalNodes(other *Event) bool { + switch a := evt.Node.(type) { + case ast.Body: + if b, ok := other.Node.(ast.Body); ok { + return a.Equal(b) + } + case *ast.Rule: + if b, ok := other.Node.(*ast.Rule); ok { + return a.Equal(b) + } + case *ast.Expr: + if b, ok := other.Node.(*ast.Expr); ok { + return a.Equal(b) + } + case nil: + return other.Node == nil + } + return false +} + +// Tracer defines the interface for tracing in the top-down evaluation engine. +// Deprecated: Use QueryTracer instead. +type Tracer interface { + Enabled() bool + Trace(*Event) +} + +// QueryTracer defines the interface for tracing in the top-down evaluation engine. +// The implementation can provide additional configuration to modify the tracing +// behavior for query evaluations. +type QueryTracer interface { + Enabled() bool + TraceEvent(Event) + Config() TraceConfig +} + +// TraceConfig defines some common configuration for Tracer implementations +type TraceConfig struct { + PlugLocalVars bool // Indicate whether to plug local variable bindings before calling into the tracer. +} + +// legacyTracer Implements the QueryTracer interface by wrapping an older Tracer instance. +type legacyTracer struct { + t Tracer +} + +func (l *legacyTracer) Enabled() bool { + return l.t.Enabled() +} + +func (l *legacyTracer) Config() TraceConfig { + return TraceConfig{ + PlugLocalVars: true, // For backwards compatibility old tracers will plug local variables + } +} + +func (l *legacyTracer) TraceEvent(evt Event) { + l.t.Trace(&evt) +} + +// WrapLegacyTracer will create a new QueryTracer which wraps an +// older Tracer instance. +func WrapLegacyTracer(tracer Tracer) QueryTracer { + return &legacyTracer{t: tracer} +} + +// BufferTracer implements the Tracer and QueryTracer interface by +// simply buffering all events received. +type BufferTracer []*Event + +// NewBufferTracer returns a new BufferTracer. +func NewBufferTracer() *BufferTracer { + return &BufferTracer{} +} + +// Enabled always returns true if the BufferTracer is instantiated. +func (b *BufferTracer) Enabled() bool { + return b != nil +} + +// Trace adds the event to the buffer. +// Deprecated: Use TraceEvent instead. +func (b *BufferTracer) Trace(evt *Event) { + *b = append(*b, evt) +} + +// TraceEvent adds the event to the buffer. +func (b *BufferTracer) TraceEvent(evt Event) { + *b = append(*b, &evt) +} + +// Config returns the Tracers standard configuration +func (b *BufferTracer) Config() TraceConfig { + return TraceConfig{PlugLocalVars: true} +} + +// PrettyTrace pretty prints the trace to the writer. +func PrettyTrace(w io.Writer, trace []*Event) { + depths := depths{} + for _, event := range trace { + depth := depths.GetOrSet(event.QueryID, event.ParentID) + fmt.Fprintln(w, formatEvent(event, depth)) + } +} + +// PrettyTraceWithLocation prints the trace to the writer and includes location information +func PrettyTraceWithLocation(w io.Writer, trace []*Event) { + depths := depths{} + + filePathAliases, longest := getShortenedFileNames(trace) + + // Always include some padding between the trace and location + locationWidth := longest + locationPadding + + for _, event := range trace { + depth := depths.GetOrSet(event.QueryID, event.ParentID) + location := formatLocation(event, filePathAliases) + fmt.Fprintf(w, "%-*s %s\n", locationWidth, location, formatEvent(event, depth)) + } +} + +func formatEvent(event *Event, depth int) string { + padding := formatEventPadding(event, depth) + if event.Op == NoteOp { + return fmt.Sprintf("%v%v %q", padding, event.Op, event.Message) + } + + var details interface{} + if node, ok := event.Node.(*ast.Rule); ok { + details = node.Path() + } else if event.Ref != nil { + details = event.Ref + } else { + details = rewrite(event).Node + } + + template := "%v%v %v" + opts := []interface{}{padding, event.Op, details} + + if event.Message != "" { + template += " %v" + opts = append(opts, event.Message) + } + + return fmt.Sprintf(template, opts...) +} + +func formatEventPadding(event *Event, depth int) string { + spaces := formatEventSpaces(event, depth) + padding := "" + if spaces > 1 { + padding += strings.Repeat("| ", spaces-1) + } + return padding +} + +func formatEventSpaces(event *Event, depth int) int { + switch event.Op { + case EnterOp: + return depth + case RedoOp: + if _, ok := event.Node.(*ast.Expr); !ok { + return depth + } + } + return depth + 1 +} + +// getShortenedFileNames will return a map of file paths to shortened aliases +// that were found in the trace. It also returns the longest location expected +func getShortenedFileNames(trace []*Event) (map[string]string, int) { + // Get a deduplicated list of all file paths + // and the longest file path size + fpAliases := map[string]string{} + var canShorten []string + longestLocation := 0 + for _, event := range trace { + if event.Location != nil { + if event.Location.File != "" { + // length of ":" + curLen := len(event.Location.File) + numDigits10(event.Location.Row) + 1 + if curLen > longestLocation { + longestLocation = curLen + } + + if _, ok := fpAliases[event.Location.File]; ok { + continue + } + + canShorten = append(canShorten, event.Location.File) + + // Default to just alias their full path + fpAliases[event.Location.File] = event.Location.File + } else { + // length of ":" + curLen := minLocationWidth + numDigits10(event.Location.Row) + 1 + if curLen > longestLocation { + longestLocation = curLen + } + } + } + } + + if len(canShorten) > 0 && longestLocation > maxIdealLocationWidth { + fpAliases, longestLocation = iStrs.TruncateFilePaths(maxIdealLocationWidth, longestLocation, canShorten...) + } + + return fpAliases, longestLocation +} + +func numDigits10(n int) int { + if n < 10 { + return 1 + } + return numDigits10(n/10) + 1 +} + +func formatLocation(event *Event, fileAliases map[string]string) string { + + location := event.Location + if location == nil { + return "" + } + + if location.File == "" { + return fmt.Sprintf("query:%v", location.Row) + } + + return fmt.Sprintf("%v:%v", fileAliases[location.File], location.Row) +} + +// depths is a helper for computing the depth of an event. Events within the +// same query all have the same depth. The depth of query is +// depth(parent(query))+1. +type depths map[uint64]int + +func (ds depths) GetOrSet(qid uint64, pqid uint64) int { + depth := ds[qid] + if depth == 0 { + depth = ds[pqid] + depth++ + ds[qid] = depth + } + return depth +} + +func builtinTrace(bctx BuiltinContext, args []*ast.Term, iter func(*ast.Term) error) error { + + str, err := builtins.StringOperand(args[0].Value, 1) + if err != nil { + return handleBuiltinErr(ast.Trace.Name, bctx.Location, err) + } + + if !bctx.TraceEnabled { + return iter(ast.BooleanTerm(true)) + } + + evt := Event{ + Op: NoteOp, + Location: bctx.Location, + QueryID: bctx.QueryID, + ParentID: bctx.ParentID, + Message: string(str), + } + + for i := range bctx.QueryTracers { + bctx.QueryTracers[i].TraceEvent(evt) + } + + return iter(ast.BooleanTerm(true)) +} + +func rewrite(event *Event) *Event { + + cpy := *event + + var node ast.Node + + switch v := event.Node.(type) { + case *ast.Expr: + node = v.Copy() + case ast.Body: + node = v.Copy() + case *ast.Rule: + node = v.Copy() + } + + _, _ = ast.TransformVars(node, func(v ast.Var) (ast.Value, error) { + if meta, ok := cpy.LocalMetadata[v]; ok { + return meta.Name, nil + } + return v, nil + }) + + cpy.Node = node + + return &cpy +} + +func init() { + RegisterBuiltinFunc(ast.Trace.Name, builtinTrace) +} diff --git a/vendor/github.com/open-policy-agent/opa/topdown/type.go b/vendor/github.com/open-policy-agent/opa/topdown/type.go new file mode 100644 index 00000000..2fe7a39a --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/topdown/type.go @@ -0,0 +1,82 @@ +// Copyright 2018 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package topdown + +import ( + "github.com/open-policy-agent/opa/ast" +) + +func builtinIsNumber(a ast.Value) (ast.Value, error) { + switch a.(type) { + case ast.Number: + return ast.Boolean(true), nil + default: + return nil, BuiltinEmpty{} + } +} + +func builtinIsString(a ast.Value) (ast.Value, error) { + switch a.(type) { + case ast.String: + return ast.Boolean(true), nil + default: + return nil, BuiltinEmpty{} + } +} + +func builtinIsBoolean(a ast.Value) (ast.Value, error) { + switch a.(type) { + case ast.Boolean: + return ast.Boolean(true), nil + default: + return nil, BuiltinEmpty{} + } +} + +func builtinIsArray(a ast.Value) (ast.Value, error) { + switch a.(type) { + case *ast.Array: + return ast.Boolean(true), nil + default: + return nil, BuiltinEmpty{} + } +} + +func builtinIsSet(a ast.Value) (ast.Value, error) { + switch a.(type) { + case ast.Set: + return ast.Boolean(true), nil + default: + return nil, BuiltinEmpty{} + } +} + +func builtinIsObject(a ast.Value) (ast.Value, error) { + switch a.(type) { + case ast.Object: + return ast.Boolean(true), nil + default: + return nil, BuiltinEmpty{} + } +} + +func builtinIsNull(a ast.Value) (ast.Value, error) { + switch a.(type) { + case ast.Null: + return ast.Boolean(true), nil + default: + return nil, BuiltinEmpty{} + } +} + +func init() { + RegisterFunctionalBuiltin1(ast.IsNumber.Name, builtinIsNumber) + RegisterFunctionalBuiltin1(ast.IsString.Name, builtinIsString) + RegisterFunctionalBuiltin1(ast.IsBoolean.Name, builtinIsBoolean) + RegisterFunctionalBuiltin1(ast.IsArray.Name, builtinIsArray) + RegisterFunctionalBuiltin1(ast.IsSet.Name, builtinIsSet) + RegisterFunctionalBuiltin1(ast.IsObject.Name, builtinIsObject) + RegisterFunctionalBuiltin1(ast.IsNull.Name, builtinIsNull) +} diff --git a/vendor/github.com/open-policy-agent/opa/topdown/type_name.go b/vendor/github.com/open-policy-agent/opa/topdown/type_name.go new file mode 100644 index 00000000..2154abb5 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/topdown/type_name.go @@ -0,0 +1,36 @@ +// Copyright 2018 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package topdown + +import ( + "fmt" + + "github.com/open-policy-agent/opa/ast" +) + +func builtinTypeName(a ast.Value) (ast.Value, error) { + switch a.(type) { + case ast.Null: + return ast.String("null"), nil + case ast.Boolean: + return ast.String("boolean"), nil + case ast.Number: + return ast.String("number"), nil + case ast.String: + return ast.String("string"), nil + case *ast.Array: + return ast.String("array"), nil + case ast.Object: + return ast.String("object"), nil + case ast.Set: + return ast.String("set"), nil + } + + return nil, fmt.Errorf("illegal value") +} + +func init() { + RegisterFunctionalBuiltin1(ast.TypeNameBuiltin.Name, builtinTypeName) +} diff --git a/vendor/github.com/open-policy-agent/opa/topdown/uuid.go b/vendor/github.com/open-policy-agent/opa/topdown/uuid.go new file mode 100644 index 00000000..d08adb0a --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/topdown/uuid.go @@ -0,0 +1,36 @@ +// Copyright 2020 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package topdown + +import ( + "github.com/open-policy-agent/opa/ast" + "github.com/open-policy-agent/opa/internal/uuid" +) + +type uuidCachingKey string + +func builtinUUIDRFC4122(bctx BuiltinContext, args []*ast.Term, iter func(*ast.Term) error) error { + + var key = uuidCachingKey(args[0].Value.String()) + + val, ok := bctx.Cache.Get(key) + if ok { + return iter(val.(*ast.Term)) + } + + s, err := uuid.New(bctx.Seed) + if err != nil { + return err + } + + result := ast.NewTerm(ast.String(s)) + bctx.Cache.Put(key, result) + + return iter(result) +} + +func init() { + RegisterBuiltinFunc(ast.UUIDRFC4122.Name, builtinUUIDRFC4122) +} diff --git a/vendor/github.com/open-policy-agent/opa/topdown/walk.go b/vendor/github.com/open-policy-agent/opa/topdown/walk.go new file mode 100644 index 00000000..092014f1 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/topdown/walk.go @@ -0,0 +1,96 @@ +// Copyright 2017 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package topdown + +import ( + "github.com/open-policy-agent/opa/ast" +) + +func evalWalk(_ BuiltinContext, args []*ast.Term, iter func(*ast.Term) error) error { + input := args[0] + filter := getOutputPath(args) + return walk(filter, nil, input, iter) +} + +func walk(filter, path *ast.Array, input *ast.Term, iter func(*ast.Term) error) error { + + if filter == nil || filter.Len() == 0 { + if path == nil { + path = ast.NewArray() + } + + if err := iter(ast.ArrayTerm(ast.NewTerm(path.Copy()), input)); err != nil { + return err + } + } + + if filter != nil && filter.Len() > 0 { + key := filter.Elem(0) + filter = filter.Slice(1, -1) + if key.IsGround() { + if term := input.Get(key); term != nil { + path = pathAppend(path, key) + return walk(filter, path, term, iter) + } + return nil + } + } + + switch v := input.Value.(type) { + case *ast.Array: + for i := 0; i < v.Len(); i++ { + path = pathAppend(path, ast.IntNumberTerm(i)) + if err := walk(filter, path, v.Elem(i), iter); err != nil { + return err + } + path = path.Slice(0, path.Len()-1) + } + case ast.Object: + return v.Iter(func(k, v *ast.Term) error { + path = pathAppend(path, k) + if err := walk(filter, path, v, iter); err != nil { + return err + } + path = path.Slice(0, path.Len()-1) + return nil + }) + case ast.Set: + return v.Iter(func(elem *ast.Term) error { + path = pathAppend(path, elem) + if err := walk(filter, path, elem, iter); err != nil { + return err + } + path = path.Slice(0, path.Len()-1) + return nil + }) + } + + return nil +} + +func pathAppend(path *ast.Array, key *ast.Term) *ast.Array { + if path == nil { + return ast.NewArray(key) + } + + return path.Append(key) +} + +func getOutputPath(args []*ast.Term) *ast.Array { + if len(args) == 2 { + if arr, ok := args[1].Value.(*ast.Array); ok { + if arr.Len() == 2 { + if path, ok := arr.Elem(0).Value.(*ast.Array); ok { + return path + } + } + } + } + return nil +} + +func init() { + RegisterBuiltinFunc(ast.WalkBuiltin.Name, evalWalk) +} diff --git a/vendor/github.com/open-policy-agent/opa/tracing/tracing.go b/vendor/github.com/open-policy-agent/opa/tracing/tracing.go new file mode 100644 index 00000000..2708b78e --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/tracing/tracing.go @@ -0,0 +1,55 @@ +// Copyright 2021 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +// Package tracing enables dependency-injection at runtime. When used +// together with an underscore-import of `github.com/open-policy-agent/opa/features/tracing`, +// the server and its runtime will emit OpenTelemetry spans to the +// configured sink. +package tracing + +import "net/http" + +// Options are options for the HTTPTracingService, passed along as-is. +type Options []interface{} + +// NewOptions is a helper method for constructing `tracing.Options` +func NewOptions(opts ...interface{}) Options { + return opts +} + +// HTTPTracingService defines how distributed tracing comes in, server- and client-side +type HTTPTracingService interface { + // NewTransport is used when setting up an HTTP client + NewTransport(http.RoundTripper, Options) http.RoundTripper + + // NewHandler is used to wrap an http.Handler in the server + NewHandler(http.Handler, string, Options) http.Handler +} + +var tracing HTTPTracingService + +// RegisterHTTPTracing enables a HTTPTracingService for further use. +func RegisterHTTPTracing(ht HTTPTracingService) { + tracing = ht +} + +// NewTransport returns another http.RoundTripper, instrumented to emit tracing +// spans according to Options. Provided by the HTTPTracingService registered with +// this package via RegisterHTTPTracing. +func NewTransport(tr http.RoundTripper, opts Options) http.RoundTripper { + if tracing == nil { + return tr + } + return tracing.NewTransport(tr, opts) +} + +// NewHandler returns another http.Handler, instrumented to emit tracing spans +// according to Options. Provided by the HTTPTracingService registered with +// this package via RegisterHTTPTracing. +func NewHandler(f http.Handler, label string, opts Options) http.Handler { + if tracing == nil { + return f + } + return tracing.NewHandler(f, label, opts) +} diff --git a/vendor/github.com/open-policy-agent/opa/types/decode.go b/vendor/github.com/open-policy-agent/opa/types/decode.go new file mode 100644 index 00000000..4e123384 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/types/decode.go @@ -0,0 +1,191 @@ +// Copyright 2020 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package types + +import ( + "encoding/json" + "fmt" + + "github.com/open-policy-agent/opa/util" +) + +const ( + typeNull = "null" + typeBoolean = "boolean" + typeNumber = "number" + typeString = "string" + typeArray = "array" + typeSet = "set" + typeObject = "object" + typeAny = "any" + typeFunction = "function" +) + +// Unmarshal deserializes bs and returns the resulting type. +func Unmarshal(bs []byte) (result Type, err error) { + + var hint rawtype + + if err = util.UnmarshalJSON(bs, &hint); err == nil { + switch hint.Type { + case typeNull: + result = NewNull() + case typeBoolean: + result = NewBoolean() + case typeNumber: + result = NewNumber() + case typeString: + result = NewString() + case typeArray: + var arr rawarray + if err = util.UnmarshalJSON(bs, &arr); err == nil { + var err error + var static []Type + var dynamic Type + if static, err = unmarshalSlice(arr.Static); err != nil { + return nil, err + } + if len(arr.Dynamic) != 0 { + if dynamic, err = Unmarshal(arr.Dynamic); err != nil { + return nil, err + } + } + result = NewArray(static, dynamic) + } + case typeObject: + var obj rawobject + if err = util.UnmarshalJSON(bs, &obj); err == nil { + var err error + var static []*StaticProperty + var dynamic *DynamicProperty + if static, err = unmarshalStaticPropertySlice(obj.Static); err != nil { + return nil, err + } + if dynamic, err = unmarshalDynamicProperty(obj.Dynamic); err != nil { + return nil, err + } + result = NewObject(static, dynamic) + } + case typeSet: + var set rawset + if err = util.UnmarshalJSON(bs, &set); err == nil { + var of Type + if of, err = Unmarshal(set.Of); err == nil { + result = NewSet(of) + } + } + case typeAny: + var any rawunion + if err = util.UnmarshalJSON(bs, &any); err == nil { + var of []Type + if of, err = unmarshalSlice(any.Of); err == nil { + result = NewAny(of...) + } + } + case typeFunction: + var decl rawdecl + if err = util.UnmarshalJSON(bs, &decl); err == nil { + args, err := unmarshalSlice(decl.Args) + if err != nil { + return nil, err + } + var ret Type + if len(decl.Result) > 0 { + ret, err = Unmarshal(decl.Result) + if err != nil { + return nil, err + } + } + if len(decl.Variadic) > 0 { + varargs, err := Unmarshal(decl.Variadic) + if err != nil { + return nil, err + } + result = NewVariadicFunction(args, varargs, ret) + } else { + result = NewFunction(args, ret) + } + } + default: + err = fmt.Errorf("unsupported type '%v'", hint.Type) + } + } + + return result, err +} + +type rawtype struct { + Type string `json:"type"` +} + +type rawarray struct { + Static []json.RawMessage `json:"static"` + Dynamic json.RawMessage `json:"dynamic"` +} + +type rawobject struct { + Static []rawstaticproperty `json:"static"` + Dynamic rawdynamicproperty `json:"dynamic"` +} + +type rawstaticproperty struct { + Key interface{} `json:"key"` + Value json.RawMessage `json:"value"` +} + +type rawdynamicproperty struct { + Key json.RawMessage `json:"key"` + Value json.RawMessage `json:"value"` +} + +type rawset struct { + Of json.RawMessage `json:"of"` +} + +type rawunion struct { + Of []json.RawMessage `json:"of"` +} + +type rawdecl struct { + Args []json.RawMessage `json:"args"` + Result json.RawMessage `json:"result"` + Variadic json.RawMessage `json:"variadic"` +} + +func unmarshalSlice(elems []json.RawMessage) (result []Type, err error) { + result = make([]Type, len(elems)) + for i := range elems { + if result[i], err = Unmarshal(elems[i]); err != nil { + return nil, err + } + } + return result, err +} + +func unmarshalStaticPropertySlice(elems []rawstaticproperty) (result []*StaticProperty, err error) { + result = make([]*StaticProperty, len(elems)) + for i := range elems { + value, err := Unmarshal(elems[i].Value) + if err != nil { + return nil, err + } + result[i] = NewStaticProperty(elems[i].Key, value) + } + return result, err +} + +func unmarshalDynamicProperty(x rawdynamicproperty) (result *DynamicProperty, err error) { + if len(x.Key) == 0 { + return nil, nil + } + var key Type + if key, err = Unmarshal(x.Key); err == nil { + var value Type + if value, err = Unmarshal(x.Value); err == nil { + return NewDynamicProperty(key, value), nil + } + } + return nil, err +} diff --git a/vendor/github.com/open-policy-agent/opa/types/types.go b/vendor/github.com/open-policy-agent/opa/types/types.go new file mode 100644 index 00000000..9b5cc66c --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/types/types.go @@ -0,0 +1,995 @@ +// Copyright 2017 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +// Package types declares data types for Rego values and helper functions to +// operate on these types. +package types + +import ( + "encoding/json" + "fmt" + "sort" + "strings" + + "github.com/open-policy-agent/opa/util" +) + +// Sprint returns the string representation of the type. +func Sprint(x Type) string { + if x == nil { + return "???" + } + return x.String() +} + +// Type represents a type of a term in the language. +type Type interface { + String() string + typeMarker() string + json.Marshaler +} + +func (Null) typeMarker() string { return typeNull } +func (Boolean) typeMarker() string { return typeBoolean } +func (Number) typeMarker() string { return typeNumber } +func (String) typeMarker() string { return typeString } +func (*Array) typeMarker() string { return typeArray } +func (*Object) typeMarker() string { return typeObject } +func (*Set) typeMarker() string { return typeSet } +func (Any) typeMarker() string { return typeAny } +func (Function) typeMarker() string { return typeFunction } + +// Null represents the null type. +type Null struct{} + +// NewNull returns a new Null type. +func NewNull() Null { + return Null{} +} + +// MarshalJSON returns the JSON encoding of t. +func (t Null) MarshalJSON() ([]byte, error) { + return json.Marshal(map[string]interface{}{ + "type": t.typeMarker(), + }) +} + +func (t Null) String() string { + return typeNull +} + +// Boolean represents the boolean type. +type Boolean struct{} + +// B represents an instance of the boolean type. +var B = NewBoolean() + +// NewBoolean returns a new Boolean type. +func NewBoolean() Boolean { + return Boolean{} +} + +// MarshalJSON returns the JSON encoding of t. +func (t Boolean) MarshalJSON() ([]byte, error) { + repr := map[string]interface{}{ + "type": t.typeMarker(), + } + return json.Marshal(repr) +} + +func (t Boolean) String() string { + return t.typeMarker() +} + +// String represents the string type. +type String struct{} + +// S represents an instance of the string type. +var S = NewString() + +// NewString returns a new String type. +func NewString() String { + return String{} +} + +// MarshalJSON returns the JSON encoding of t. +func (t String) MarshalJSON() ([]byte, error) { + return json.Marshal(map[string]interface{}{ + "type": t.typeMarker(), + }) +} + +func (t String) String() string { + return typeString +} + +// Number represents the number type. +type Number struct{} + +// N represents an instance of the number type. +var N = NewNumber() + +// NewNumber returns a new Number type. +func NewNumber() Number { + return Number{} +} + +// MarshalJSON returns the JSON encoding of t. +func (t Number) MarshalJSON() ([]byte, error) { + return json.Marshal(map[string]interface{}{ + "type": t.typeMarker(), + }) +} + +func (Number) String() string { + return typeNumber +} + +// Array represents the array type. +type Array struct { + static []Type // static items + dynamic Type // dynamic items +} + +// NewArray returns a new Array type. +func NewArray(static []Type, dynamic Type) *Array { + return &Array{ + static: static, + dynamic: dynamic, + } +} + +// MarshalJSON returns the JSON encoding of t. +func (t *Array) MarshalJSON() ([]byte, error) { + repr := map[string]interface{}{ + "type": t.typeMarker(), + } + if len(t.static) != 0 { + repr["static"] = t.static + } + if t.dynamic != nil { + repr["dynamic"] = t.dynamic + } + return json.Marshal(repr) +} + +func (t *Array) String() string { + prefix := "array" + buf := []string{} + for _, tpe := range t.static { + buf = append(buf, Sprint(tpe)) + } + var repr = prefix + if len(buf) > 0 { + repr += "<" + strings.Join(buf, ", ") + ">" + } + if t.dynamic != nil { + repr += "[" + t.dynamic.String() + "]" + } + return repr +} + +// Dynamic returns the type of the array's dynamic elements. +func (t *Array) Dynamic() Type { + return t.dynamic +} + +// Len returns the number of static array elements. +func (t *Array) Len() int { + return len(t.static) +} + +// Select returns the type of element at the zero-based pos. +func (t *Array) Select(pos int) Type { + if pos >= 0 { + if len(t.static) > pos { + return t.static[pos] + } + if t.dynamic != nil { + return t.dynamic + } + } + return nil +} + +// Set represents the set type. +type Set struct { + of Type +} + +// NewSet returns a new Set type. +func NewSet(of Type) *Set { + return &Set{ + of: of, + } +} + +// MarshalJSON returns the JSON encoding of t. +func (t *Set) MarshalJSON() ([]byte, error) { + repr := map[string]interface{}{ + "type": t.typeMarker(), + } + if t.of != nil { + repr["of"] = t.of + } + return json.Marshal(repr) +} + +func (t *Set) String() string { + prefix := typeSet + return prefix + "[" + Sprint(t.of) + "]" +} + +// StaticProperty represents a static object property. +type StaticProperty struct { + Key interface{} + Value Type +} + +// NewStaticProperty returns a new StaticProperty object. +func NewStaticProperty(key interface{}, value Type) *StaticProperty { + return &StaticProperty{ + Key: key, + Value: value, + } +} + +// MarshalJSON returns the JSON encoding of p. +func (p *StaticProperty) MarshalJSON() ([]byte, error) { + return json.Marshal(map[string]interface{}{ + "key": p.Key, + "value": p.Value, + }) +} + +// DynamicProperty represents a dynamic object property. +type DynamicProperty struct { + Key Type + Value Type +} + +// NewDynamicProperty returns a new DynamicProperty object. +func NewDynamicProperty(key, value Type) *DynamicProperty { + return &DynamicProperty{ + Key: key, + Value: value, + } +} + +// MarshalJSON returns the JSON encoding of p. +func (p *DynamicProperty) MarshalJSON() ([]byte, error) { + return json.Marshal(map[string]interface{}{ + "key": p.Key, + "value": p.Value, + }) +} + +func (p *DynamicProperty) String() string { + return fmt.Sprintf("%s: %s", Sprint(p.Key), Sprint(p.Value)) +} + +// Object represents the object type. +type Object struct { + static []*StaticProperty // constant properties + dynamic *DynamicProperty // dynamic properties +} + +// NewObject returns a new Object type. +func NewObject(static []*StaticProperty, dynamic *DynamicProperty) *Object { + sort.Slice(static, func(i, j int) bool { + cmp := util.Compare(static[i].Key, static[j].Key) + return cmp == -1 + }) + return &Object{ + static: static, + dynamic: dynamic, + } +} + +func (t *Object) String() string { + prefix := "object" + buf := make([]string, 0, len(t.static)) + for _, p := range t.static { + buf = append(buf, fmt.Sprintf("%v: %v", p.Key, Sprint(p.Value))) + } + var repr = prefix + if len(buf) > 0 { + repr += "<" + strings.Join(buf, ", ") + ">" + } + if t.dynamic != nil { + repr += "[" + t.dynamic.String() + "]" + } + return repr +} + +// DynamicValue returns the type of the object's dynamic elements. +func (t *Object) DynamicValue() Type { + if t.dynamic == nil { + return nil + } + return t.dynamic.Value +} + +// DynamicProperties returns the type of the object's dynamic elements. +func (t *Object) DynamicProperties() *DynamicProperty { + return t.dynamic +} + +// StaticProperties returns the type of the object's static elements. +func (t *Object) StaticProperties() []*StaticProperty { + return t.static +} + +// Keys returns the keys of the object's static elements. +func (t *Object) Keys() []interface{} { + sl := make([]interface{}, 0, len(t.static)) + for _, p := range t.static { + sl = append(sl, p.Key) + } + return sl +} + +// MarshalJSON returns the JSON encoding of t. +func (t *Object) MarshalJSON() ([]byte, error) { + repr := map[string]interface{}{ + "type": t.typeMarker(), + } + if len(t.static) != 0 { + repr["static"] = t.static + } + if t.dynamic != nil { + repr["dynamic"] = t.dynamic + } + return json.Marshal(repr) +} + +// Select returns the type of the named property. +func (t *Object) Select(name interface{}) Type { + + pos := sort.Search(len(t.static), func(x int) bool { + return util.Compare(t.static[x].Key, name) >= 0 + }) + + if pos < len(t.static) && util.Compare(t.static[pos].Key, name) == 0 { + return t.static[pos].Value + } + + if t.dynamic != nil { + if Contains(t.dynamic.Key, TypeOf(name)) { + return t.dynamic.Value + } + } + + return nil +} + +// Any represents a dynamic type. +type Any []Type + +// A represents the superset of all types. +var A = NewAny() + +// NewAny returns a new Any type. +func NewAny(of ...Type) Any { + sl := make(Any, len(of)) + for i := range sl { + sl[i] = of[i] + } + sort.Sort(typeSlice(sl)) + return sl +} + +// Contains returns true if t is a superset of other. +func (t Any) Contains(other Type) bool { + if _, ok := other.(*Function); ok { + return false + } + for i := range t { + if Compare(t[i], other) == 0 { + return true + } + } + return len(t) == 0 +} + +// MarshalJSON returns the JSON encoding of t. +func (t Any) MarshalJSON() ([]byte, error) { + data := map[string]interface{}{ + "type": t.typeMarker(), + } + if len(t) != 0 { + data["of"] = []Type(t) + } + return json.Marshal(data) +} + +// Merge return a new Any type that is the superset of t and other. +func (t Any) Merge(other Type) Any { + if otherAny, ok := other.(Any); ok { + return t.Union(otherAny) + } + if t.Contains(other) { + return t + } + cpy := make(Any, len(t)+1) + idx := sort.Search(len(t), func(i int) bool { + return Compare(t[i], other) >= 0 + }) + copy(cpy, t[:idx]) + cpy[idx] = other + copy(cpy[idx+1:], t[idx:]) + return cpy +} + +// Union returns a new Any type that is the union of the two Any types. +func (t Any) Union(other Any) Any { + if len(t) == 0 { + return t + } + if len(other) == 0 { + return other + } + cpy := make(Any, len(t)) + for i := range cpy { + cpy[i] = t[i] + } + for i := range other { + if !cpy.Contains(other[i]) { + cpy = append(cpy, other[i]) + } + } + sort.Sort(typeSlice(cpy)) + return cpy +} + +func (t Any) String() string { + prefix := "any" + if len(t) == 0 { + return prefix + } + buf := make([]string, len(t)) + for i := range t { + buf[i] = Sprint(t[i]) + } + return prefix + "<" + strings.Join(buf, ", ") + ">" +} + +// Function represents a function type. +type Function struct { + args []Type + result Type + variadic Type +} + +// Args returns an argument list. +func Args(x ...Type) []Type { + return x +} + +// Void returns true if the function has no return value. This function returns +// false if x is not a function. +func Void(x Type) bool { + f, ok := x.(*Function) + return ok && f.Result() == nil +} + +// Arity returns the number of arguments in the function signature or zero if x +// is not a function. If the type is unknown, this function returns -1. +func Arity(x Type) int { + if x == nil { + return -1 + } + f, ok := x.(*Function) + if !ok { + return 0 + } + return len(f.FuncArgs().Args) +} + +// NewFunction returns a new Function object where xs[:len(xs)-1] are arguments +// and xs[len(xs)-1] is the result type. +func NewFunction(args []Type, result Type) *Function { + return &Function{ + args: args, + result: result, + } +} + +// NewVariadicFunction returns a new Function object. This function sets the +// variadic bit on the signature. Non-void variadic functions are not currently +// supported. +func NewVariadicFunction(args []Type, varargs Type, result Type) *Function { + if result != nil { + panic("illegal value: non-void variadic functions not supported") + } + return &Function{ + args: args, + variadic: varargs, + result: nil, + } +} + +// FuncArgs returns the function's arguments. +func (t *Function) FuncArgs() FuncArgs { + return FuncArgs{Args: t.Args(), Variadic: t.variadic} +} + +// Args returns the function's arguments as a slice, ignoring variadic arguments. +// Deprecated: Use FuncArgs instead. +func (t *Function) Args() []Type { + cpy := make([]Type, len(t.args)) + copy(cpy, t.args) + return cpy +} + +// Result returns the function's result type. +func (t *Function) Result() Type { + return t.result +} + +func (t *Function) String() string { + return fmt.Sprintf("%v => %v", t.FuncArgs(), Sprint(t.Result())) +} + +// MarshalJSON returns the JSON encoding of t. +func (t *Function) MarshalJSON() ([]byte, error) { + repr := map[string]interface{}{ + "type": t.typeMarker(), + } + if len(t.args) > 0 { + repr["args"] = t.args + } + if t.result != nil { + repr["result"] = t.result + } + if t.variadic != nil { + repr["variadic"] = t.variadic + } + return json.Marshal(repr) +} + +// UnmarshalJSON decodes the JSON serialized function declaration. +func (t *Function) UnmarshalJSON(bs []byte) error { + + tpe, err := Unmarshal(bs) + if err != nil { + return err + } + + f, ok := tpe.(*Function) + if !ok { + return fmt.Errorf("invalid type") + } + + *t = *f + return nil +} + +// Union returns a new function represnting the union of t and other. Functions +// must have the same arity to be unioned. +func (t *Function) Union(other *Function) *Function { + if other == nil { + return t + } else if t == nil { + return other + } + + a := t.Args() + b := other.Args() + if len(a) != len(b) { + return nil + } + + aIsVariadic := t.FuncArgs().Variadic != nil + bIsVariadic := other.FuncArgs().Variadic != nil + + if aIsVariadic && !bIsVariadic { + return nil + } else if bIsVariadic && !aIsVariadic { + return nil + } + + args := make([]Type, len(a)) + for i := range a { + args[i] = Or(a[i], b[i]) + } + + result := NewFunction(args, Or(t.Result(), other.Result())) + result.variadic = Or(t.FuncArgs().Variadic, other.FuncArgs().Variadic) + + return result +} + +// FuncArgs represents the arguments that can be passed to a function. +type FuncArgs struct { + Args []Type `json:"args,omitempty"` + Variadic Type `json:"variadic,omitempty"` +} + +func (a FuncArgs) String() string { + var buf []string + for i := range a.Args { + buf = append(buf, Sprint(a.Args[i])) + } + if a.Variadic != nil { + buf = append(buf, Sprint(a.Variadic)+"...") + } + return "(" + strings.Join(buf, ", ") + ")" +} + +func (a FuncArgs) Arg(x int) Type { + if x < len(a.Args) { + return a.Args[x] + } + return a.Variadic +} + +// Compare returns -1, 0, 1 based on comparison between a and b. +func Compare(a, b Type) int { + x := typeOrder(a) + y := typeOrder(b) + if x > y { + return 1 + } else if x < y { + return -1 + } + switch a.(type) { + case nil, Null, Boolean, Number, String: + return 0 + case *Array: + arrA := a.(*Array) + arrB := b.(*Array) + if arrA.dynamic != nil && arrB.dynamic == nil { + return 1 + } else if arrB.dynamic != nil && arrA.dynamic == nil { + return -1 + } + if arrB.dynamic != nil && arrA.dynamic != nil { + if cmp := Compare(arrA.dynamic, arrB.dynamic); cmp != 0 { + return cmp + } + } + return typeSliceCompare(arrA.static, arrB.static) + case *Object: + objA := a.(*Object) + objB := b.(*Object) + if objA.dynamic != nil && objB.dynamic == nil { + return 1 + } else if objB.dynamic != nil && objA.dynamic == nil { + return -1 + } + if objA.dynamic != nil && objB.dynamic != nil { + if cmp := Compare(objA.dynamic.Key, objB.dynamic.Key); cmp != 0 { + return cmp + } + if cmp := Compare(objA.dynamic.Value, objB.dynamic.Value); cmp != 0 { + return cmp + } + } + + lenStaticA := len(objA.static) + lenStaticB := len(objB.static) + + minLen := lenStaticA + if lenStaticB < minLen { + minLen = lenStaticB + } + + for i := 0; i < minLen; i++ { + if cmp := util.Compare(objA.static[i].Key, objB.static[i].Key); cmp != 0 { + return cmp + } + if cmp := Compare(objA.static[i].Value, objB.static[i].Value); cmp != 0 { + return cmp + } + } + + if lenStaticA < lenStaticB { + return -1 + } else if lenStaticB < lenStaticA { + return 1 + } + + return 0 + case *Set: + setA := a.(*Set) + setB := b.(*Set) + if setA.of == nil && setB.of == nil { + return 0 + } else if setA.of == nil { + return -1 + } else if setB.of == nil { + return 1 + } + return Compare(setA.of, setB.of) + case Any: + sl1 := typeSlice(a.(Any)) + sl2 := typeSlice(b.(Any)) + return typeSliceCompare(sl1, sl2) + case *Function: + fA := a.(*Function) + fB := b.(*Function) + if len(fA.args) < len(fB.args) { + return -1 + } else if len(fA.args) > len(fB.args) { + return 1 + } + for i := 0; i < len(fA.args); i++ { + if cmp := Compare(fA.args[i], fB.args[i]); cmp != 0 { + return cmp + } + } + if cmp := Compare(fA.result, fB.result); cmp != 0 { + return cmp + } + return Compare(fA.variadic, fB.variadic) + default: + panic("unreachable") + } +} + +// Contains returns true if a is a superset or equal to b. +func Contains(a, b Type) bool { + if any, ok := a.(Any); ok { + return any.Contains(b) + } + return Compare(a, b) == 0 +} + +// Or returns a type that represents the union of a and b. If one type is a +// superset of the other, the superset is returned unchanged. +func Or(a, b Type) Type { + if a == nil { + return b + } else if b == nil { + return a + } + fA, ok1 := a.(*Function) + fB, ok2 := b.(*Function) + if ok1 && ok2 { + return fA.Union(fB) + } else if ok1 || ok2 { + return nil + } + anyA, ok1 := a.(Any) + anyB, ok2 := b.(Any) + if ok1 { + return anyA.Merge(b) + } + if ok2 { + return anyB.Merge(a) + } + if Compare(a, b) == 0 { + return a + } + return NewAny(a, b) +} + +// Select returns a property or item of a. +func Select(a Type, x interface{}) Type { + switch a := a.(type) { + case *Array: + n, ok := x.(json.Number) + if !ok { + return nil + } + pos, err := n.Int64() + if err != nil { + return nil + } + return a.Select(int(pos)) + case *Object: + return a.Select(x) + case *Set: + tpe := TypeOf(x) + if Compare(a.of, tpe) == 0 { + return a.of + } + if any, ok := a.of.(Any); ok { + if any.Contains(tpe) { + return tpe + } + } + return nil + case Any: + if Compare(a, A) == 0 { + return A + } + var tpe Type + for i := range a { + // TODO(tsandall): test nil/nil + tpe = Or(Select(a[i], x), tpe) + } + return tpe + default: + return nil + } +} + +// Keys returns the type of keys that can be enumerated for a. For arrays, the +// keys are always number types, for objects the keys are always string types, +// and for sets the keys are always the type of the set element. +func Keys(a Type) Type { + switch a := a.(type) { + case *Array: + return N + case *Object: + var tpe Type + for _, k := range a.Keys() { + tpe = Or(tpe, TypeOf(k)) + } + if a.dynamic != nil { + tpe = Or(tpe, a.dynamic.Key) + } + return tpe + case *Set: + return a.of + case Any: + // TODO(tsandall): ditto test + if Compare(a, A) == 0 { + return A + } + var tpe Type + for i := range a { + tpe = Or(Keys(a[i]), tpe) + } + return tpe + } + return nil +} + +// Values returns the type of values that can be enumerated for a. +func Values(a Type) Type { + switch a := a.(type) { + case *Array: + var tpe Type + for i := range a.static { + tpe = Or(tpe, a.static[i]) + } + return Or(tpe, a.dynamic) + case *Object: + var tpe Type + for _, v := range a.static { + tpe = Or(tpe, v.Value) + } + if a.dynamic != nil { + tpe = Or(tpe, a.dynamic.Value) + } + return tpe + case *Set: + return a.of + case Any: + if Compare(a, A) == 0 { + return A + } + var tpe Type + for i := range a { + tpe = Or(Values(a[i]), tpe) + } + return tpe + } + return nil +} + +// Nil returns true if a's type is unknown. +func Nil(a Type) bool { + switch a := a.(type) { + case nil: + return true + case *Function: + for i := range a.args { + if Nil(a.args[i]) { + return true + } + } + return Nil(a.result) + case *Array: + for i := range a.static { + if Nil(a.static[i]) { + return true + } + } + if a.dynamic != nil { + return Nil(a.dynamic) + } + case *Object: + for i := range a.static { + if Nil(a.static[i].Value) { + return true + } + } + if a.dynamic != nil { + return Nil(a.dynamic.Key) || Nil(a.dynamic.Value) + } + case *Set: + return Nil(a.of) + } + return false +} + +// TypeOf returns the type of the Golang native value. +func TypeOf(x interface{}) Type { + switch x := x.(type) { + case nil: + return NewNull() + case bool: + return B + case string: + return S + case json.Number: + return N + case map[string]interface{}: + // The ast.ValueToInterface() function returns ast.Object values as map[string]interface{} + // so map[string]interface{} must be handled here because the type checker uses the value + // to interface conversion when inferring object types. + static := make([]*StaticProperty, 0, len(x)) + for k, v := range x { + static = append(static, NewStaticProperty(k, TypeOf(v))) + } + return NewObject(static, nil) + case map[interface{}]interface{}: + static := make([]*StaticProperty, 0, len(x)) + for k, v := range x { + static = append(static, NewStaticProperty(k, TypeOf(v))) + } + return NewObject(static, nil) + case []interface{}: + static := make([]Type, len(x)) + for i := range x { + static[i] = TypeOf(x[i]) + } + return NewArray(static, nil) + } + panic("unreachable") +} + +type typeSlice []Type + +func (s typeSlice) Less(i, j int) bool { return Compare(s[i], s[j]) < 0 } +func (s typeSlice) Swap(i, j int) { x := s[i]; s[i] = s[j]; s[j] = x } +func (s typeSlice) Len() int { return len(s) } + +func typeSliceCompare(a, b []Type) int { + minLen := len(a) + if len(b) < minLen { + minLen = len(b) + } + for i := 0; i < minLen; i++ { + if cmp := Compare(a[i], b[i]); cmp != 0 { + return cmp + } + } + if len(a) < len(b) { + return -1 + } else if len(b) < len(a) { + return 1 + } + return 0 +} + +func typeOrder(x Type) int { + switch x.(type) { + case Null: + return 0 + case Boolean: + return 1 + case Number: + return 2 + case String: + return 3 + case *Array: + return 4 + case *Object: + return 5 + case *Set: + return 6 + case Any: + return 7 + case *Function: + return 8 + case nil: + return -1 + } + panic("unreachable") +} diff --git a/vendor/github.com/open-policy-agent/opa/util/backoff.go b/vendor/github.com/open-policy-agent/opa/util/backoff.go new file mode 100644 index 00000000..1b126fbe --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/util/backoff.go @@ -0,0 +1,42 @@ +// Copyright 2018 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package util + +import ( + "math/rand" + "time" +) + +// DefaultBackoff returns a delay with an exponential backoff based on the +// number of retries. +func DefaultBackoff(base, max float64, retries int) time.Duration { + return Backoff(base, max, .2, 1.6, retries) +} + +// Backoff returns a delay with an exponential backoff based on the number of +// retries. Same algorithm used in gRPC. +func Backoff(base, max, jitter, factor float64, retries int) time.Duration { + if retries == 0 { + return 0 + } + + backoff, max := base, max + for backoff < max && retries > 0 { + backoff *= factor + retries-- + } + if backoff > max { + backoff = max + } + + // Randomize backoff delays so that if a cluster of requests start at + // the same time, they won't operate in lockstep. + backoff *= 1 + jitter*(rand.Float64()*2-1) + if backoff < 0 { + return 0 + } + + return time.Duration(backoff) +} diff --git a/vendor/github.com/open-policy-agent/opa/util/close.go b/vendor/github.com/open-policy-agent/opa/util/close.go new file mode 100644 index 00000000..97c91773 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/util/close.go @@ -0,0 +1,23 @@ +// Copyright 2018 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package util + +import ( + "io" + "io/ioutil" + "net/http" +) + +// Close reads the remaining bytes from the response and then closes it to +// ensure that the connection is freed. If the body is not read and closed, a +// leak can occur. +func Close(resp *http.Response) { + if resp != nil && resp.Body != nil { + if _, err := io.Copy(ioutil.Discard, resp.Body); err != nil { + return + } + resp.Body.Close() + } +} diff --git a/vendor/github.com/open-policy-agent/opa/util/compare.go b/vendor/github.com/open-policy-agent/opa/util/compare.go new file mode 100644 index 00000000..dd187590 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/util/compare.go @@ -0,0 +1,161 @@ +// Copyright 2016 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package util + +import ( + "encoding/json" + "fmt" + "math/big" + "sort" +) + +// Compare returns 0 if a equals b, -1 if a is less than b, and 1 if b is than a. +// +// For comparison between values of different types, the following ordering is used: +// nil < bool < float64 < string < []interface{} < map[string]interface{}. Slices and maps +// are compared recursively. If one slice or map is a subset of the other slice or map +// it is considered "less than". Nil is always equal to nil. +// +func Compare(a, b interface{}) int { + aSortOrder := sortOrder(a) + bSortOrder := sortOrder(b) + if aSortOrder < bSortOrder { + return -1 + } else if bSortOrder < aSortOrder { + return 1 + } + switch a := a.(type) { + case nil: + return 0 + case bool: + switch b := b.(type) { + case bool: + if a == b { + return 0 + } + if !a { + return -1 + } + return 1 + } + case json.Number: + switch b := b.(type) { + case json.Number: + return compareJSONNumber(a, b) + } + case string: + switch b := b.(type) { + case string: + if a == b { + return 0 + } else if a < b { + return -1 + } + return 1 + } + case []interface{}: + switch b := b.(type) { + case []interface{}: + bLen := len(b) + aLen := len(a) + minLen := aLen + if bLen < minLen { + minLen = bLen + } + for i := 0; i < minLen; i++ { + cmp := Compare(a[i], b[i]) + if cmp != 0 { + return cmp + } + } + if aLen == bLen { + return 0 + } else if aLen < bLen { + return -1 + } + return 1 + } + case map[string]interface{}: + switch b := b.(type) { + case map[string]interface{}: + var aKeys []string + for k := range a { + aKeys = append(aKeys, k) + } + var bKeys []string + for k := range b { + bKeys = append(bKeys, k) + } + sort.Strings(aKeys) + sort.Strings(bKeys) + aLen := len(aKeys) + bLen := len(bKeys) + minLen := aLen + if bLen < minLen { + minLen = bLen + } + for i := 0; i < minLen; i++ { + if aKeys[i] < bKeys[i] { + return -1 + } else if bKeys[i] < aKeys[i] { + return 1 + } + aVal := a[aKeys[i]] + bVal := b[bKeys[i]] + cmp := Compare(aVal, bVal) + if cmp != 0 { + return cmp + } + } + if aLen == bLen { + return 0 + } else if aLen < bLen { + return -1 + } + return 1 + } + } + + panic(fmt.Sprintf("illegal arguments of type %T and type %T", a, b)) +} + +const ( + nilSort = iota + boolSort = iota + numberSort = iota + stringSort = iota + arraySort = iota + objectSort = iota +) + +func compareJSONNumber(a, b json.Number) int { + bigA, ok := new(big.Float).SetString(string(a)) + if !ok { + panic("illegal value") + } + bigB, ok := new(big.Float).SetString(string(b)) + if !ok { + panic("illegal value") + } + return bigA.Cmp(bigB) +} + +func sortOrder(v interface{}) int { + switch v.(type) { + case nil: + return nilSort + case bool: + return boolSort + case json.Number: + return numberSort + case string: + return stringSort + case []interface{}: + return arraySort + case map[string]interface{}: + return objectSort + } + panic(fmt.Sprintf("illegal argument of type %T", v)) +} diff --git a/vendor/github.com/open-policy-agent/opa/util/doc.go b/vendor/github.com/open-policy-agent/opa/util/doc.go new file mode 100644 index 00000000..900dff8c --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/util/doc.go @@ -0,0 +1,6 @@ +// Copyright 2016 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +// Package util provides generic utilities used throughout the policy engine. +package util diff --git a/vendor/github.com/open-policy-agent/opa/util/enumflag.go b/vendor/github.com/open-policy-agent/opa/util/enumflag.go new file mode 100644 index 00000000..4796f026 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/util/enumflag.go @@ -0,0 +1,59 @@ +// Copyright 2017 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package util + +import ( + "fmt" + "strings" +) + +// EnumFlag implements the pflag.Value interface to provide enumerated command +// line parameter values. +type EnumFlag struct { + defaultValue string + vs []string + i int +} + +// NewEnumFlag returns a new EnumFlag that has a defaultValue and vs enumerated +// values. +func NewEnumFlag(defaultValue string, vs []string) *EnumFlag { + f := &EnumFlag{ + i: -1, + vs: vs, + defaultValue: defaultValue, + } + return f +} + +// Type returns the valid enumeration values. +func (f *EnumFlag) Type() string { + return "{" + strings.Join(f.vs, ",") + "}" +} + +// String returns the EnumValue's value as string. +func (f *EnumFlag) String() string { + if f.i == -1 { + return f.defaultValue + } + return f.vs[f.i] +} + +// IsSet will return true if the EnumFlag has been set. +func (f *EnumFlag) IsSet() bool { + return f.i != -1 +} + +// Set sets the enum value. If s is not a valid enum value, an error is +// returned. +func (f *EnumFlag) Set(s string) error { + for i := range f.vs { + if f.vs[i] == s { + f.i = i + return nil + } + } + return fmt.Errorf("must be one of %v", f.Type()) +} diff --git a/vendor/github.com/open-policy-agent/opa/util/graph.go b/vendor/github.com/open-policy-agent/opa/util/graph.go new file mode 100644 index 00000000..f0e82424 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/util/graph.go @@ -0,0 +1,90 @@ +// Copyright 2016 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package util + +// Traversal defines a basic interface to perform traversals. +type Traversal interface { + + // Edges should return the neighbours of node "u". + Edges(u T) []T + + // Visited should return true if node "u" has already been visited in this + // traversal. If the same traversal is used multiple times, the state that + // tracks visited nodes should be reset. + Visited(u T) bool +} + +// Equals should return true if node "u" equals node "v". +type Equals func(u T, v T) bool + +// Iter should return true to indicate stop. +type Iter func(u T) bool + +// DFS performs a depth first traversal calling f for each node starting from u. +// If f returns true, traversal stops and DFS returns true. +func DFS(t Traversal, f Iter, u T) bool { + lifo := NewLIFO(u) + for lifo.Size() > 0 { + next, _ := lifo.Pop() + if t.Visited(next) { + continue + } + if f(next) { + return true + } + for _, v := range t.Edges(next) { + lifo.Push(v) + } + } + return false +} + +// BFS performs a breadth first traversal calling f for each node starting from +// u. If f returns true, traversal stops and BFS returns true. +func BFS(t Traversal, f Iter, u T) bool { + fifo := NewFIFO(u) + for fifo.Size() > 0 { + next, _ := fifo.Pop() + if t.Visited(next) { + continue + } + if f(next) { + return true + } + for _, v := range t.Edges(next) { + fifo.Push(v) + } + } + return false +} + +// DFSPath returns a path from node a to node z found by performing +// a depth first traversal. If no path is found, an empty slice is returned. +func DFSPath(t Traversal, eq Equals, a, z T) []T { + p := dfsRecursive(t, eq, a, z, []T{}) + for i := len(p)/2 - 1; i >= 0; i-- { + o := len(p) - i - 1 + p[i], p[o] = p[o], p[i] + } + return p +} + +func dfsRecursive(t Traversal, eq Equals, u, z T, path []T) []T { + if t.Visited(u) { + return path + } + for _, v := range t.Edges(u) { + if eq(v, z) { + path = append(path, z) + path = append(path, u) + return path + } + if p := dfsRecursive(t, eq, v, z, path); len(p) > 0 { + path = append(p, u) + return path + } + } + return path +} diff --git a/vendor/github.com/open-policy-agent/opa/util/hashmap.go b/vendor/github.com/open-policy-agent/opa/util/hashmap.go new file mode 100644 index 00000000..11e7dca4 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/util/hashmap.go @@ -0,0 +1,157 @@ +// Copyright 2016 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package util + +import ( + "fmt" + "strings" +) + +// T is a concise way to refer to T. +type T interface{} + +type hashEntry struct { + k T + v T + next *hashEntry +} + +// HashMap represents a key/value map. +type HashMap struct { + eq func(T, T) bool + hash func(T) int + table map[int]*hashEntry + size int +} + +// NewHashMap returns a new empty HashMap. +func NewHashMap(eq func(T, T) bool, hash func(T) int) *HashMap { + return &HashMap{ + eq: eq, + hash: hash, + table: make(map[int]*hashEntry), + size: 0, + } +} + +// Copy returns a shallow copy of this HashMap. +func (h *HashMap) Copy() *HashMap { + cpy := NewHashMap(h.eq, h.hash) + h.Iter(func(k, v T) bool { + cpy.Put(k, v) + return false + }) + return cpy +} + +// Equal returns true if this HashMap equals the other HashMap. +// Two hash maps are equal if they contain the same key/value pairs. +func (h *HashMap) Equal(other *HashMap) bool { + if h.Len() != other.Len() { + return false + } + return !h.Iter(func(k, v T) bool { + ov, ok := other.Get(k) + if !ok { + return true + } + return !h.eq(v, ov) + }) +} + +// Get returns the value for k. +func (h *HashMap) Get(k T) (T, bool) { + hash := h.hash(k) + for entry := h.table[hash]; entry != nil; entry = entry.next { + if h.eq(entry.k, k) { + return entry.v, true + } + } + return nil, false +} + +// Delete removes the the key k. +func (h *HashMap) Delete(k T) { + hash := h.hash(k) + var prev *hashEntry + for entry := h.table[hash]; entry != nil; entry = entry.next { + if h.eq(entry.k, k) { + if prev != nil { + prev.next = entry.next + } else { + h.table[hash] = entry.next + } + h.size-- + return + } + prev = entry + } +} + +// Hash returns the hash code for this hash map. +func (h *HashMap) Hash() int { + var hash int + h.Iter(func(k, v T) bool { + hash += h.hash(k) + h.hash(v) + return false + }) + return hash +} + +// Iter invokes the iter function for each element in the HashMap. +// If the iter function returns true, iteration stops and the return value is true. +// If the iter function never returns true, iteration proceeds through all elements +// and the return value is false. +func (h *HashMap) Iter(iter func(T, T) bool) bool { + for _, entry := range h.table { + for ; entry != nil; entry = entry.next { + if iter(entry.k, entry.v) { + return true + } + } + } + return false +} + +// Len returns the current size of this HashMap. +func (h *HashMap) Len() int { + return h.size +} + +// Put inserts a key/value pair into this HashMap. If the key is already present, the existing +// value is overwritten. +func (h *HashMap) Put(k T, v T) { + hash := h.hash(k) + head := h.table[hash] + for entry := head; entry != nil; entry = entry.next { + if h.eq(entry.k, k) { + entry.v = v + return + } + } + h.table[hash] = &hashEntry{k: k, v: v, next: head} + h.size++ +} + +func (h *HashMap) String() string { + var buf []string + h.Iter(func(k T, v T) bool { + buf = append(buf, fmt.Sprintf("%v: %v", k, v)) + return false + }) + return "{" + strings.Join(buf, ", ") + "}" +} + +// Update returns a new HashMap with elements from the other HashMap put into this HashMap. +// If the other HashMap contains elements with the same key as this HashMap, the value +// from the other HashMap overwrites the value from this HashMap. +func (h *HashMap) Update(other *HashMap) *HashMap { + updated := h.Copy() + other.Iter(func(k, v T) bool { + updated.Put(k, v) + return false + }) + return updated +} diff --git a/vendor/github.com/open-policy-agent/opa/util/json.go b/vendor/github.com/open-policy-agent/opa/util/json.go new file mode 100644 index 00000000..cfc94e06 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/util/json.go @@ -0,0 +1,113 @@ +// Copyright 2016 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package util + +import ( + "bytes" + "encoding/json" + "fmt" + "io" + "reflect" + + "github.com/ghodss/yaml" +) + +// UnmarshalJSON parses the JSON encoded data and stores the result in the value +// pointed to by x. +// +// This function is intended to be used in place of the standard json.Marshal +// function when json.Number is required. +func UnmarshalJSON(bs []byte, x interface{}) (err error) { + buf := bytes.NewBuffer(bs) + decoder := NewJSONDecoder(buf) + if err := decoder.Decode(x); err != nil { + return err + } + + // Since decoder.Decode validates only the first json structure in bytes, + // check if decoder has more bytes to consume to validate whole input bytes. + tok, err := decoder.Token() + if tok != nil { + return fmt.Errorf("error: invalid character '%s' after top-level value", tok) + } + if err != nil && err != io.EOF { + return err + } + return nil +} + +// NewJSONDecoder returns a new decoder that reads from r. +// +// This function is intended to be used in place of the standard json.NewDecoder +// when json.Number is required. +func NewJSONDecoder(r io.Reader) *json.Decoder { + decoder := json.NewDecoder(r) + decoder.UseNumber() + return decoder +} + +// MustUnmarshalJSON parse the JSON encoded data and returns the result. +// +// If the data cannot be decoded, this function will panic. This function is for +// test purposes. +func MustUnmarshalJSON(bs []byte) interface{} { + var x interface{} + if err := UnmarshalJSON(bs, &x); err != nil { + panic(err) + } + return x +} + +// MustMarshalJSON returns the JSON encoding of x +// +// If the data cannot be encoded, this function will panic. This function is for +// test purposes. +func MustMarshalJSON(x interface{}) []byte { + bs, err := json.Marshal(x) + if err != nil { + panic(err) + } + return bs +} + +// RoundTrip encodes to JSON, and decodes the result again. +// +// Thereby, it is converting its argument to the representation expected by +// rego.Input and inmem's Write operations. Works with both references and +// values. +func RoundTrip(x *interface{}) error { + bs, err := json.Marshal(x) + if err != nil { + return err + } + return UnmarshalJSON(bs, x) +} + +// Reference returns a pointer to its argument unless the argument already is +// a pointer. If the argument is **t, or ***t, etc, it will return *t. +// +// Used for preparing Go types (including pointers to structs) into values to be +// put through util.RoundTrip(). +func Reference(x interface{}) *interface{} { + var y interface{} + rv := reflect.ValueOf(x) + if rv.Kind() == reflect.Ptr { + return Reference(rv.Elem().Interface()) + } + if rv.Kind() != reflect.Invalid { + y = rv.Interface() + return &y + } + return &x +} + +// Unmarshal decodes a YAML or JSON value into the specified type. +func Unmarshal(bs []byte, v interface{}) error { + bs, err := yaml.YAMLToJSON(bs) + if err != nil { + return err + } + return UnmarshalJSON(bs, v) +} diff --git a/vendor/github.com/open-policy-agent/opa/util/queue.go b/vendor/github.com/open-policy-agent/opa/util/queue.go new file mode 100644 index 00000000..63a2ffc1 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/util/queue.go @@ -0,0 +1,113 @@ +// Copyright 2017 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package util + +// LIFO represents a simple LIFO queue. +type LIFO struct { + top *queueNode + size int +} + +type queueNode struct { + v T + next *queueNode +} + +// NewLIFO returns a new LIFO queue containing elements ts starting with the +// left-most argument at the bottom. +func NewLIFO(ts ...T) *LIFO { + s := &LIFO{} + for i := range ts { + s.Push(ts[i]) + } + return s +} + +// Push adds a new element onto the LIFO. +func (s *LIFO) Push(t T) { + node := &queueNode{v: t, next: s.top} + s.top = node + s.size++ +} + +// Peek returns the top of the LIFO. If LIFO is empty, returns nil, false. +func (s *LIFO) Peek() (T, bool) { + if s.top == nil { + return nil, false + } + return s.top.v, true +} + +// Pop returns the top of the LIFO and removes it. If LIFO is empty returns +// nil, false. +func (s *LIFO) Pop() (T, bool) { + if s.top == nil { + return nil, false + } + node := s.top + s.top = node.next + s.size-- + return node.v, true +} + +// Size returns the size of the LIFO. +func (s *LIFO) Size() int { + return s.size +} + +// FIFO represents a simple FIFO queue. +type FIFO struct { + front *queueNode + back *queueNode + size int +} + +// NewFIFO returns a new FIFO queue containing elements ts starting with the +// left-most argument at the front. +func NewFIFO(ts ...T) *FIFO { + s := &FIFO{} + for i := range ts { + s.Push(ts[i]) + } + return s +} + +// Push adds a new element onto the LIFO. +func (s *FIFO) Push(t T) { + node := &queueNode{v: t, next: nil} + if s.front == nil { + s.front = node + s.back = node + } else { + s.back.next = node + s.back = node + } + s.size++ +} + +// Peek returns the top of the LIFO. If LIFO is empty, returns nil, false. +func (s *FIFO) Peek() (T, bool) { + if s.front == nil { + return nil, false + } + return s.front.v, true +} + +// Pop returns the top of the LIFO and removes it. If LIFO is empty returns +// nil, false. +func (s *FIFO) Pop() (T, bool) { + if s.front == nil { + return nil, false + } + node := s.front + s.front = node.next + s.size-- + return node.v, true +} + +// Size returns the size of the LIFO. +func (s *FIFO) Size() int { + return s.size +} diff --git a/vendor/github.com/open-policy-agent/opa/util/wait.go b/vendor/github.com/open-policy-agent/opa/util/wait.go new file mode 100644 index 00000000..b70ab6fc --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/util/wait.go @@ -0,0 +1,34 @@ +// Copyright 2020 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package util + +import ( + "fmt" + "time" +) + +// WaitFunc will call passed function at an interval and return nil +// as soon this function returns true. +// If timeout is reached before the passed in function returns true +// an error is returned. +func WaitFunc(fun func() bool, interval, timeout time.Duration) error { + if fun() { + return nil + } + ticker := time.NewTicker(interval) + timer := time.NewTimer(timeout) + defer ticker.Stop() + defer timer.Stop() + for { + select { + case <-timer.C: + return fmt.Errorf("timeout") + case <-ticker.C: + if fun() { + return nil + } + } + } +} diff --git a/vendor/github.com/open-policy-agent/opa/version/version.go b/vendor/github.com/open-policy-agent/opa/version/version.go new file mode 100644 index 00000000..0b9c998e --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/version/version.go @@ -0,0 +1,24 @@ +// Copyright 2016 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +// Package version contains version information that is set at build time. +package version + +import ( + "runtime" +) + +// Version is the canonical version of OPA. +var Version = "0.36.1" + +// GoVersion is the version of Go this was built with +var GoVersion = runtime.Version() + +// Additional version information that is displayed by the "version" command and used to +// identify the version of running instances of OPA. +var ( + Vcs = "" + Timestamp = "" + Hostname = "" +) diff --git a/vendor/github.com/open-policy-agent/opa/version/wasm.go b/vendor/github.com/open-policy-agent/opa/version/wasm.go new file mode 100644 index 00000000..c274a782 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/version/wasm.go @@ -0,0 +1,13 @@ +// Copyright 2020 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package version + +import "github.com/open-policy-agent/opa/internal/rego/opa" + +// WasmRuntimeAvailable indicates if a wasm runtime is available in this OPA. +func WasmRuntimeAvailable() bool { + _, err := opa.LookupEngine("wasm") + return err == nil +} diff --git a/vendor/github.com/pkg/errors/.gitignore b/vendor/github.com/pkg/errors/.gitignore new file mode 100644 index 00000000..daf913b1 --- /dev/null +++ b/vendor/github.com/pkg/errors/.gitignore @@ -0,0 +1,24 @@ +# Compiled Object files, Static and Dynamic libs (Shared Objects) +*.o +*.a +*.so + +# Folders +_obj +_test + +# Architecture specific extensions/prefixes +*.[568vq] +[568vq].out + +*.cgo1.go +*.cgo2.c +_cgo_defun.c +_cgo_gotypes.go +_cgo_export.* + +_testmain.go + +*.exe +*.test +*.prof diff --git a/vendor/github.com/pkg/errors/.travis.yml b/vendor/github.com/pkg/errors/.travis.yml new file mode 100644 index 00000000..9159de03 --- /dev/null +++ b/vendor/github.com/pkg/errors/.travis.yml @@ -0,0 +1,10 @@ +language: go +go_import_path: github.com/pkg/errors +go: + - 1.11.x + - 1.12.x + - 1.13.x + - tip + +script: + - make check diff --git a/vendor/github.com/pkg/errors/LICENSE b/vendor/github.com/pkg/errors/LICENSE new file mode 100644 index 00000000..835ba3e7 --- /dev/null +++ b/vendor/github.com/pkg/errors/LICENSE @@ -0,0 +1,23 @@ +Copyright (c) 2015, Dave Cheney +All rights reserved. + +Redistribution and use in source and binary forms, with or without +modification, are permitted provided that the following conditions are met: + +* Redistributions of source code must retain the above copyright notice, this + list of conditions and the following disclaimer. + +* Redistributions in binary form must reproduce the above copyright notice, + this list of conditions and the following disclaimer in the documentation + and/or other materials provided with the distribution. + +THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" +AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE +IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE +DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE +FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL +DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR +SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER +CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, +OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. diff --git a/vendor/github.com/pkg/errors/Makefile b/vendor/github.com/pkg/errors/Makefile new file mode 100644 index 00000000..ce9d7cde --- /dev/null +++ b/vendor/github.com/pkg/errors/Makefile @@ -0,0 +1,44 @@ +PKGS := github.com/pkg/errors +SRCDIRS := $(shell go list -f '{{.Dir}}' $(PKGS)) +GO := go + +check: test vet gofmt misspell unconvert staticcheck ineffassign unparam + +test: + $(GO) test $(PKGS) + +vet: | test + $(GO) vet $(PKGS) + +staticcheck: + $(GO) get honnef.co/go/tools/cmd/staticcheck + staticcheck -checks all $(PKGS) + +misspell: + $(GO) get github.com/client9/misspell/cmd/misspell + misspell \ + -locale GB \ + -error \ + *.md *.go + +unconvert: + $(GO) get github.com/mdempsky/unconvert + unconvert -v $(PKGS) + +ineffassign: + $(GO) get github.com/gordonklaus/ineffassign + find $(SRCDIRS) -name '*.go' | xargs ineffassign + +pedantic: check errcheck + +unparam: + $(GO) get mvdan.cc/unparam + unparam ./... + +errcheck: + $(GO) get github.com/kisielk/errcheck + errcheck $(PKGS) + +gofmt: + @echo Checking code is gofmted + @test -z "$(shell gofmt -s -l -d -e $(SRCDIRS) | tee /dev/stderr)" diff --git a/vendor/github.com/pkg/errors/README.md b/vendor/github.com/pkg/errors/README.md new file mode 100644 index 00000000..54dfdcb1 --- /dev/null +++ b/vendor/github.com/pkg/errors/README.md @@ -0,0 +1,59 @@ +# errors [![Travis-CI](https://travis-ci.org/pkg/errors.svg)](https://travis-ci.org/pkg/errors) [![AppVeyor](https://ci.appveyor.com/api/projects/status/b98mptawhudj53ep/branch/master?svg=true)](https://ci.appveyor.com/project/davecheney/errors/branch/master) [![GoDoc](https://godoc.org/github.com/pkg/errors?status.svg)](http://godoc.org/github.com/pkg/errors) [![Report card](https://goreportcard.com/badge/github.com/pkg/errors)](https://goreportcard.com/report/github.com/pkg/errors) [![Sourcegraph](https://sourcegraph.com/github.com/pkg/errors/-/badge.svg)](https://sourcegraph.com/github.com/pkg/errors?badge) + +Package errors provides simple error handling primitives. + +`go get github.com/pkg/errors` + +The traditional error handling idiom in Go is roughly akin to +```go +if err != nil { + return err +} +``` +which applied recursively up the call stack results in error reports without context or debugging information. The errors package allows programmers to add context to the failure path in their code in a way that does not destroy the original value of the error. + +## Adding context to an error + +The errors.Wrap function returns a new error that adds context to the original error. For example +```go +_, err := ioutil.ReadAll(r) +if err != nil { + return errors.Wrap(err, "read failed") +} +``` +## Retrieving the cause of an error + +Using `errors.Wrap` constructs a stack of errors, adding context to the preceding error. Depending on the nature of the error it may be necessary to reverse the operation of errors.Wrap to retrieve the original error for inspection. Any error value which implements this interface can be inspected by `errors.Cause`. +```go +type causer interface { + Cause() error +} +``` +`errors.Cause` will recursively retrieve the topmost error which does not implement `causer`, which is assumed to be the original cause. For example: +```go +switch err := errors.Cause(err).(type) { +case *MyError: + // handle specifically +default: + // unknown error +} +``` + +[Read the package documentation for more information](https://godoc.org/github.com/pkg/errors). + +## Roadmap + +With the upcoming [Go2 error proposals](https://go.googlesource.com/proposal/+/master/design/go2draft.md) this package is moving into maintenance mode. The roadmap for a 1.0 release is as follows: + +- 0.9. Remove pre Go 1.9 and Go 1.10 support, address outstanding pull requests (if possible) +- 1.0. Final release. + +## Contributing + +Because of the Go2 errors changes, this package is not accepting proposals for new functionality. With that said, we welcome pull requests, bug fixes and issue reports. + +Before sending a PR, please discuss your change by raising an issue. + +## License + +BSD-2-Clause diff --git a/vendor/github.com/pkg/errors/appveyor.yml b/vendor/github.com/pkg/errors/appveyor.yml new file mode 100644 index 00000000..a932eade --- /dev/null +++ b/vendor/github.com/pkg/errors/appveyor.yml @@ -0,0 +1,32 @@ +version: build-{build}.{branch} + +clone_folder: C:\gopath\src\github.com\pkg\errors +shallow_clone: true # for startup speed + +environment: + GOPATH: C:\gopath + +platform: + - x64 + +# http://www.appveyor.com/docs/installed-software +install: + # some helpful output for debugging builds + - go version + - go env + # pre-installed MinGW at C:\MinGW is 32bit only + # but MSYS2 at C:\msys64 has mingw64 + - set PATH=C:\msys64\mingw64\bin;%PATH% + - gcc --version + - g++ --version + +build_script: + - go install -v ./... + +test_script: + - set PATH=C:\gopath\bin;%PATH% + - go test -v ./... + +#artifacts: +# - path: '%GOPATH%\bin\*.exe' +deploy: off diff --git a/vendor/github.com/pkg/errors/errors.go b/vendor/github.com/pkg/errors/errors.go new file mode 100644 index 00000000..161aea25 --- /dev/null +++ b/vendor/github.com/pkg/errors/errors.go @@ -0,0 +1,288 @@ +// Package errors provides simple error handling primitives. +// +// The traditional error handling idiom in Go is roughly akin to +// +// if err != nil { +// return err +// } +// +// which when applied recursively up the call stack results in error reports +// without context or debugging information. The errors package allows +// programmers to add context to the failure path in their code in a way +// that does not destroy the original value of the error. +// +// Adding context to an error +// +// The errors.Wrap function returns a new error that adds context to the +// original error by recording a stack trace at the point Wrap is called, +// together with the supplied message. For example +// +// _, err := ioutil.ReadAll(r) +// if err != nil { +// return errors.Wrap(err, "read failed") +// } +// +// If additional control is required, the errors.WithStack and +// errors.WithMessage functions destructure errors.Wrap into its component +// operations: annotating an error with a stack trace and with a message, +// respectively. +// +// Retrieving the cause of an error +// +// Using errors.Wrap constructs a stack of errors, adding context to the +// preceding error. Depending on the nature of the error it may be necessary +// to reverse the operation of errors.Wrap to retrieve the original error +// for inspection. Any error value which implements this interface +// +// type causer interface { +// Cause() error +// } +// +// can be inspected by errors.Cause. errors.Cause will recursively retrieve +// the topmost error that does not implement causer, which is assumed to be +// the original cause. For example: +// +// switch err := errors.Cause(err).(type) { +// case *MyError: +// // handle specifically +// default: +// // unknown error +// } +// +// Although the causer interface is not exported by this package, it is +// considered a part of its stable public interface. +// +// Formatted printing of errors +// +// All error values returned from this package implement fmt.Formatter and can +// be formatted by the fmt package. The following verbs are supported: +// +// %s print the error. If the error has a Cause it will be +// printed recursively. +// %v see %s +// %+v extended format. Each Frame of the error's StackTrace will +// be printed in detail. +// +// Retrieving the stack trace of an error or wrapper +// +// New, Errorf, Wrap, and Wrapf record a stack trace at the point they are +// invoked. This information can be retrieved with the following interface: +// +// type stackTracer interface { +// StackTrace() errors.StackTrace +// } +// +// The returned errors.StackTrace type is defined as +// +// type StackTrace []Frame +// +// The Frame type represents a call site in the stack trace. Frame supports +// the fmt.Formatter interface that can be used for printing information about +// the stack trace of this error. For example: +// +// if err, ok := err.(stackTracer); ok { +// for _, f := range err.StackTrace() { +// fmt.Printf("%+s:%d\n", f, f) +// } +// } +// +// Although the stackTracer interface is not exported by this package, it is +// considered a part of its stable public interface. +// +// See the documentation for Frame.Format for more details. +package errors + +import ( + "fmt" + "io" +) + +// New returns an error with the supplied message. +// New also records the stack trace at the point it was called. +func New(message string) error { + return &fundamental{ + msg: message, + stack: callers(), + } +} + +// Errorf formats according to a format specifier and returns the string +// as a value that satisfies error. +// Errorf also records the stack trace at the point it was called. +func Errorf(format string, args ...interface{}) error { + return &fundamental{ + msg: fmt.Sprintf(format, args...), + stack: callers(), + } +} + +// fundamental is an error that has a message and a stack, but no caller. +type fundamental struct { + msg string + *stack +} + +func (f *fundamental) Error() string { return f.msg } + +func (f *fundamental) Format(s fmt.State, verb rune) { + switch verb { + case 'v': + if s.Flag('+') { + io.WriteString(s, f.msg) + f.stack.Format(s, verb) + return + } + fallthrough + case 's': + io.WriteString(s, f.msg) + case 'q': + fmt.Fprintf(s, "%q", f.msg) + } +} + +// WithStack annotates err with a stack trace at the point WithStack was called. +// If err is nil, WithStack returns nil. +func WithStack(err error) error { + if err == nil { + return nil + } + return &withStack{ + err, + callers(), + } +} + +type withStack struct { + error + *stack +} + +func (w *withStack) Cause() error { return w.error } + +// Unwrap provides compatibility for Go 1.13 error chains. +func (w *withStack) Unwrap() error { return w.error } + +func (w *withStack) Format(s fmt.State, verb rune) { + switch verb { + case 'v': + if s.Flag('+') { + fmt.Fprintf(s, "%+v", w.Cause()) + w.stack.Format(s, verb) + return + } + fallthrough + case 's': + io.WriteString(s, w.Error()) + case 'q': + fmt.Fprintf(s, "%q", w.Error()) + } +} + +// Wrap returns an error annotating err with a stack trace +// at the point Wrap is called, and the supplied message. +// If err is nil, Wrap returns nil. +func Wrap(err error, message string) error { + if err == nil { + return nil + } + err = &withMessage{ + cause: err, + msg: message, + } + return &withStack{ + err, + callers(), + } +} + +// Wrapf returns an error annotating err with a stack trace +// at the point Wrapf is called, and the format specifier. +// If err is nil, Wrapf returns nil. +func Wrapf(err error, format string, args ...interface{}) error { + if err == nil { + return nil + } + err = &withMessage{ + cause: err, + msg: fmt.Sprintf(format, args...), + } + return &withStack{ + err, + callers(), + } +} + +// WithMessage annotates err with a new message. +// If err is nil, WithMessage returns nil. +func WithMessage(err error, message string) error { + if err == nil { + return nil + } + return &withMessage{ + cause: err, + msg: message, + } +} + +// WithMessagef annotates err with the format specifier. +// If err is nil, WithMessagef returns nil. +func WithMessagef(err error, format string, args ...interface{}) error { + if err == nil { + return nil + } + return &withMessage{ + cause: err, + msg: fmt.Sprintf(format, args...), + } +} + +type withMessage struct { + cause error + msg string +} + +func (w *withMessage) Error() string { return w.msg + ": " + w.cause.Error() } +func (w *withMessage) Cause() error { return w.cause } + +// Unwrap provides compatibility for Go 1.13 error chains. +func (w *withMessage) Unwrap() error { return w.cause } + +func (w *withMessage) Format(s fmt.State, verb rune) { + switch verb { + case 'v': + if s.Flag('+') { + fmt.Fprintf(s, "%+v\n", w.Cause()) + io.WriteString(s, w.msg) + return + } + fallthrough + case 's', 'q': + io.WriteString(s, w.Error()) + } +} + +// Cause returns the underlying cause of the error, if possible. +// An error value has a cause if it implements the following +// interface: +// +// type causer interface { +// Cause() error +// } +// +// If the error does not implement Cause, the original error will +// be returned. If the error is nil, nil will be returned without further +// investigation. +func Cause(err error) error { + type causer interface { + Cause() error + } + + for err != nil { + cause, ok := err.(causer) + if !ok { + break + } + err = cause.Cause() + } + return err +} diff --git a/vendor/github.com/pkg/errors/go113.go b/vendor/github.com/pkg/errors/go113.go new file mode 100644 index 00000000..be0d10d0 --- /dev/null +++ b/vendor/github.com/pkg/errors/go113.go @@ -0,0 +1,38 @@ +// +build go1.13 + +package errors + +import ( + stderrors "errors" +) + +// Is reports whether any error in err's chain matches target. +// +// The chain consists of err itself followed by the sequence of errors obtained by +// repeatedly calling Unwrap. +// +// An error is considered to match a target if it is equal to that target or if +// it implements a method Is(error) bool such that Is(target) returns true. +func Is(err, target error) bool { return stderrors.Is(err, target) } + +// As finds the first error in err's chain that matches target, and if so, sets +// target to that error value and returns true. +// +// The chain consists of err itself followed by the sequence of errors obtained by +// repeatedly calling Unwrap. +// +// An error matches target if the error's concrete value is assignable to the value +// pointed to by target, or if the error has a method As(interface{}) bool such that +// As(target) returns true. In the latter case, the As method is responsible for +// setting target. +// +// As will panic if target is not a non-nil pointer to either a type that implements +// error, or to any interface type. As returns false if err is nil. +func As(err error, target interface{}) bool { return stderrors.As(err, target) } + +// Unwrap returns the result of calling the Unwrap method on err, if err's +// type contains an Unwrap method returning error. +// Otherwise, Unwrap returns nil. +func Unwrap(err error) error { + return stderrors.Unwrap(err) +} diff --git a/vendor/github.com/pkg/errors/stack.go b/vendor/github.com/pkg/errors/stack.go new file mode 100644 index 00000000..779a8348 --- /dev/null +++ b/vendor/github.com/pkg/errors/stack.go @@ -0,0 +1,177 @@ +package errors + +import ( + "fmt" + "io" + "path" + "runtime" + "strconv" + "strings" +) + +// Frame represents a program counter inside a stack frame. +// For historical reasons if Frame is interpreted as a uintptr +// its value represents the program counter + 1. +type Frame uintptr + +// pc returns the program counter for this frame; +// multiple frames may have the same PC value. +func (f Frame) pc() uintptr { return uintptr(f) - 1 } + +// file returns the full path to the file that contains the +// function for this Frame's pc. +func (f Frame) file() string { + fn := runtime.FuncForPC(f.pc()) + if fn == nil { + return "unknown" + } + file, _ := fn.FileLine(f.pc()) + return file +} + +// line returns the line number of source code of the +// function for this Frame's pc. +func (f Frame) line() int { + fn := runtime.FuncForPC(f.pc()) + if fn == nil { + return 0 + } + _, line := fn.FileLine(f.pc()) + return line +} + +// name returns the name of this function, if known. +func (f Frame) name() string { + fn := runtime.FuncForPC(f.pc()) + if fn == nil { + return "unknown" + } + return fn.Name() +} + +// Format formats the frame according to the fmt.Formatter interface. +// +// %s source file +// %d source line +// %n function name +// %v equivalent to %s:%d +// +// Format accepts flags that alter the printing of some verbs, as follows: +// +// %+s function name and path of source file relative to the compile time +// GOPATH separated by \n\t (\n\t) +// %+v equivalent to %+s:%d +func (f Frame) Format(s fmt.State, verb rune) { + switch verb { + case 's': + switch { + case s.Flag('+'): + io.WriteString(s, f.name()) + io.WriteString(s, "\n\t") + io.WriteString(s, f.file()) + default: + io.WriteString(s, path.Base(f.file())) + } + case 'd': + io.WriteString(s, strconv.Itoa(f.line())) + case 'n': + io.WriteString(s, funcname(f.name())) + case 'v': + f.Format(s, 's') + io.WriteString(s, ":") + f.Format(s, 'd') + } +} + +// MarshalText formats a stacktrace Frame as a text string. The output is the +// same as that of fmt.Sprintf("%+v", f), but without newlines or tabs. +func (f Frame) MarshalText() ([]byte, error) { + name := f.name() + if name == "unknown" { + return []byte(name), nil + } + return []byte(fmt.Sprintf("%s %s:%d", name, f.file(), f.line())), nil +} + +// StackTrace is stack of Frames from innermost (newest) to outermost (oldest). +type StackTrace []Frame + +// Format formats the stack of Frames according to the fmt.Formatter interface. +// +// %s lists source files for each Frame in the stack +// %v lists the source file and line number for each Frame in the stack +// +// Format accepts flags that alter the printing of some verbs, as follows: +// +// %+v Prints filename, function, and line number for each Frame in the stack. +func (st StackTrace) Format(s fmt.State, verb rune) { + switch verb { + case 'v': + switch { + case s.Flag('+'): + for _, f := range st { + io.WriteString(s, "\n") + f.Format(s, verb) + } + case s.Flag('#'): + fmt.Fprintf(s, "%#v", []Frame(st)) + default: + st.formatSlice(s, verb) + } + case 's': + st.formatSlice(s, verb) + } +} + +// formatSlice will format this StackTrace into the given buffer as a slice of +// Frame, only valid when called with '%s' or '%v'. +func (st StackTrace) formatSlice(s fmt.State, verb rune) { + io.WriteString(s, "[") + for i, f := range st { + if i > 0 { + io.WriteString(s, " ") + } + f.Format(s, verb) + } + io.WriteString(s, "]") +} + +// stack represents a stack of program counters. +type stack []uintptr + +func (s *stack) Format(st fmt.State, verb rune) { + switch verb { + case 'v': + switch { + case st.Flag('+'): + for _, pc := range *s { + f := Frame(pc) + fmt.Fprintf(st, "\n%+v", f) + } + } + } +} + +func (s *stack) StackTrace() StackTrace { + f := make([]Frame, len(*s)) + for i := 0; i < len(f); i++ { + f[i] = Frame((*s)[i]) + } + return f +} + +func callers() *stack { + const depth = 32 + var pcs [depth]uintptr + n := runtime.Callers(3, pcs[:]) + var st stack = pcs[0:n] + return &st +} + +// funcname removes the path prefix component of a function's name reported by func.Name(). +func funcname(name string) string { + i := strings.LastIndex(name, "/") + name = name[i+1:] + i = strings.Index(name, ".") + return name[i+1:] +} diff --git a/vendor/github.com/pmezard/go-difflib/LICENSE b/vendor/github.com/pmezard/go-difflib/LICENSE new file mode 100644 index 00000000..c67dad61 --- /dev/null +++ b/vendor/github.com/pmezard/go-difflib/LICENSE @@ -0,0 +1,27 @@ +Copyright (c) 2013, Patrick Mezard +All rights reserved. + +Redistribution and use in source and binary forms, with or without +modification, are permitted provided that the following conditions are +met: + + Redistributions of source code must retain the above copyright +notice, this list of conditions and the following disclaimer. + Redistributions in binary form must reproduce the above copyright +notice, this list of conditions and the following disclaimer in the +documentation and/or other materials provided with the distribution. + The names of its contributors may not be used to endorse or promote +products derived from this software without specific prior written +permission. + +THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS +IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED +TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A +PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +HOLDER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED +TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR +PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF +LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING +NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS +SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. diff --git a/vendor/github.com/pmezard/go-difflib/difflib/difflib.go b/vendor/github.com/pmezard/go-difflib/difflib/difflib.go new file mode 100644 index 00000000..003e99fa --- /dev/null +++ b/vendor/github.com/pmezard/go-difflib/difflib/difflib.go @@ -0,0 +1,772 @@ +// Package difflib is a partial port of Python difflib module. +// +// It provides tools to compare sequences of strings and generate textual diffs. +// +// The following class and functions have been ported: +// +// - SequenceMatcher +// +// - unified_diff +// +// - context_diff +// +// Getting unified diffs was the main goal of the port. Keep in mind this code +// is mostly suitable to output text differences in a human friendly way, there +// are no guarantees generated diffs are consumable by patch(1). +package difflib + +import ( + "bufio" + "bytes" + "fmt" + "io" + "strings" +) + +func min(a, b int) int { + if a < b { + return a + } + return b +} + +func max(a, b int) int { + if a > b { + return a + } + return b +} + +func calculateRatio(matches, length int) float64 { + if length > 0 { + return 2.0 * float64(matches) / float64(length) + } + return 1.0 +} + +type Match struct { + A int + B int + Size int +} + +type OpCode struct { + Tag byte + I1 int + I2 int + J1 int + J2 int +} + +// SequenceMatcher compares sequence of strings. The basic +// algorithm predates, and is a little fancier than, an algorithm +// published in the late 1980's by Ratcliff and Obershelp under the +// hyperbolic name "gestalt pattern matching". The basic idea is to find +// the longest contiguous matching subsequence that contains no "junk" +// elements (R-O doesn't address junk). The same idea is then applied +// recursively to the pieces of the sequences to the left and to the right +// of the matching subsequence. This does not yield minimal edit +// sequences, but does tend to yield matches that "look right" to people. +// +// SequenceMatcher tries to compute a "human-friendly diff" between two +// sequences. Unlike e.g. UNIX(tm) diff, the fundamental notion is the +// longest *contiguous* & junk-free matching subsequence. That's what +// catches peoples' eyes. The Windows(tm) windiff has another interesting +// notion, pairing up elements that appear uniquely in each sequence. +// That, and the method here, appear to yield more intuitive difference +// reports than does diff. This method appears to be the least vulnerable +// to synching up on blocks of "junk lines", though (like blank lines in +// ordinary text files, or maybe "

" lines in HTML files). That may be +// because this is the only method of the 3 that has a *concept* of +// "junk" . +// +// Timing: Basic R-O is cubic time worst case and quadratic time expected +// case. SequenceMatcher is quadratic time for the worst case and has +// expected-case behavior dependent in a complicated way on how many +// elements the sequences have in common; best case time is linear. +type SequenceMatcher struct { + a []string + b []string + b2j map[string][]int + IsJunk func(string) bool + autoJunk bool + bJunk map[string]struct{} + matchingBlocks []Match + fullBCount map[string]int + bPopular map[string]struct{} + opCodes []OpCode +} + +func NewMatcher(a, b []string) *SequenceMatcher { + m := SequenceMatcher{autoJunk: true} + m.SetSeqs(a, b) + return &m +} + +func NewMatcherWithJunk(a, b []string, autoJunk bool, + isJunk func(string) bool) *SequenceMatcher { + + m := SequenceMatcher{IsJunk: isJunk, autoJunk: autoJunk} + m.SetSeqs(a, b) + return &m +} + +// Set two sequences to be compared. +func (m *SequenceMatcher) SetSeqs(a, b []string) { + m.SetSeq1(a) + m.SetSeq2(b) +} + +// Set the first sequence to be compared. The second sequence to be compared is +// not changed. +// +// SequenceMatcher computes and caches detailed information about the second +// sequence, so if you want to compare one sequence S against many sequences, +// use .SetSeq2(s) once and call .SetSeq1(x) repeatedly for each of the other +// sequences. +// +// See also SetSeqs() and SetSeq2(). +func (m *SequenceMatcher) SetSeq1(a []string) { + if &a == &m.a { + return + } + m.a = a + m.matchingBlocks = nil + m.opCodes = nil +} + +// Set the second sequence to be compared. The first sequence to be compared is +// not changed. +func (m *SequenceMatcher) SetSeq2(b []string) { + if &b == &m.b { + return + } + m.b = b + m.matchingBlocks = nil + m.opCodes = nil + m.fullBCount = nil + m.chainB() +} + +func (m *SequenceMatcher) chainB() { + // Populate line -> index mapping + b2j := map[string][]int{} + for i, s := range m.b { + indices := b2j[s] + indices = append(indices, i) + b2j[s] = indices + } + + // Purge junk elements + m.bJunk = map[string]struct{}{} + if m.IsJunk != nil { + junk := m.bJunk + for s, _ := range b2j { + if m.IsJunk(s) { + junk[s] = struct{}{} + } + } + for s, _ := range junk { + delete(b2j, s) + } + } + + // Purge remaining popular elements + popular := map[string]struct{}{} + n := len(m.b) + if m.autoJunk && n >= 200 { + ntest := n/100 + 1 + for s, indices := range b2j { + if len(indices) > ntest { + popular[s] = struct{}{} + } + } + for s, _ := range popular { + delete(b2j, s) + } + } + m.bPopular = popular + m.b2j = b2j +} + +func (m *SequenceMatcher) isBJunk(s string) bool { + _, ok := m.bJunk[s] + return ok +} + +// Find longest matching block in a[alo:ahi] and b[blo:bhi]. +// +// If IsJunk is not defined: +// +// Return (i,j,k) such that a[i:i+k] is equal to b[j:j+k], where +// alo <= i <= i+k <= ahi +// blo <= j <= j+k <= bhi +// and for all (i',j',k') meeting those conditions, +// k >= k' +// i <= i' +// and if i == i', j <= j' +// +// In other words, of all maximal matching blocks, return one that +// starts earliest in a, and of all those maximal matching blocks that +// start earliest in a, return the one that starts earliest in b. +// +// If IsJunk is defined, first the longest matching block is +// determined as above, but with the additional restriction that no +// junk element appears in the block. Then that block is extended as +// far as possible by matching (only) junk elements on both sides. So +// the resulting block never matches on junk except as identical junk +// happens to be adjacent to an "interesting" match. +// +// If no blocks match, return (alo, blo, 0). +func (m *SequenceMatcher) findLongestMatch(alo, ahi, blo, bhi int) Match { + // CAUTION: stripping common prefix or suffix would be incorrect. + // E.g., + // ab + // acab + // Longest matching block is "ab", but if common prefix is + // stripped, it's "a" (tied with "b"). UNIX(tm) diff does so + // strip, so ends up claiming that ab is changed to acab by + // inserting "ca" in the middle. That's minimal but unintuitive: + // "it's obvious" that someone inserted "ac" at the front. + // Windiff ends up at the same place as diff, but by pairing up + // the unique 'b's and then matching the first two 'a's. + besti, bestj, bestsize := alo, blo, 0 + + // find longest junk-free match + // during an iteration of the loop, j2len[j] = length of longest + // junk-free match ending with a[i-1] and b[j] + j2len := map[int]int{} + for i := alo; i != ahi; i++ { + // look at all instances of a[i] in b; note that because + // b2j has no junk keys, the loop is skipped if a[i] is junk + newj2len := map[int]int{} + for _, j := range m.b2j[m.a[i]] { + // a[i] matches b[j] + if j < blo { + continue + } + if j >= bhi { + break + } + k := j2len[j-1] + 1 + newj2len[j] = k + if k > bestsize { + besti, bestj, bestsize = i-k+1, j-k+1, k + } + } + j2len = newj2len + } + + // Extend the best by non-junk elements on each end. In particular, + // "popular" non-junk elements aren't in b2j, which greatly speeds + // the inner loop above, but also means "the best" match so far + // doesn't contain any junk *or* popular non-junk elements. + for besti > alo && bestj > blo && !m.isBJunk(m.b[bestj-1]) && + m.a[besti-1] == m.b[bestj-1] { + besti, bestj, bestsize = besti-1, bestj-1, bestsize+1 + } + for besti+bestsize < ahi && bestj+bestsize < bhi && + !m.isBJunk(m.b[bestj+bestsize]) && + m.a[besti+bestsize] == m.b[bestj+bestsize] { + bestsize += 1 + } + + // Now that we have a wholly interesting match (albeit possibly + // empty!), we may as well suck up the matching junk on each + // side of it too. Can't think of a good reason not to, and it + // saves post-processing the (possibly considerable) expense of + // figuring out what to do with it. In the case of an empty + // interesting match, this is clearly the right thing to do, + // because no other kind of match is possible in the regions. + for besti > alo && bestj > blo && m.isBJunk(m.b[bestj-1]) && + m.a[besti-1] == m.b[bestj-1] { + besti, bestj, bestsize = besti-1, bestj-1, bestsize+1 + } + for besti+bestsize < ahi && bestj+bestsize < bhi && + m.isBJunk(m.b[bestj+bestsize]) && + m.a[besti+bestsize] == m.b[bestj+bestsize] { + bestsize += 1 + } + + return Match{A: besti, B: bestj, Size: bestsize} +} + +// Return list of triples describing matching subsequences. +// +// Each triple is of the form (i, j, n), and means that +// a[i:i+n] == b[j:j+n]. The triples are monotonically increasing in +// i and in j. It's also guaranteed that if (i, j, n) and (i', j', n') are +// adjacent triples in the list, and the second is not the last triple in the +// list, then i+n != i' or j+n != j'. IOW, adjacent triples never describe +// adjacent equal blocks. +// +// The last triple is a dummy, (len(a), len(b), 0), and is the only +// triple with n==0. +func (m *SequenceMatcher) GetMatchingBlocks() []Match { + if m.matchingBlocks != nil { + return m.matchingBlocks + } + + var matchBlocks func(alo, ahi, blo, bhi int, matched []Match) []Match + matchBlocks = func(alo, ahi, blo, bhi int, matched []Match) []Match { + match := m.findLongestMatch(alo, ahi, blo, bhi) + i, j, k := match.A, match.B, match.Size + if match.Size > 0 { + if alo < i && blo < j { + matched = matchBlocks(alo, i, blo, j, matched) + } + matched = append(matched, match) + if i+k < ahi && j+k < bhi { + matched = matchBlocks(i+k, ahi, j+k, bhi, matched) + } + } + return matched + } + matched := matchBlocks(0, len(m.a), 0, len(m.b), nil) + + // It's possible that we have adjacent equal blocks in the + // matching_blocks list now. + nonAdjacent := []Match{} + i1, j1, k1 := 0, 0, 0 + for _, b := range matched { + // Is this block adjacent to i1, j1, k1? + i2, j2, k2 := b.A, b.B, b.Size + if i1+k1 == i2 && j1+k1 == j2 { + // Yes, so collapse them -- this just increases the length of + // the first block by the length of the second, and the first + // block so lengthened remains the block to compare against. + k1 += k2 + } else { + // Not adjacent. Remember the first block (k1==0 means it's + // the dummy we started with), and make the second block the + // new block to compare against. + if k1 > 0 { + nonAdjacent = append(nonAdjacent, Match{i1, j1, k1}) + } + i1, j1, k1 = i2, j2, k2 + } + } + if k1 > 0 { + nonAdjacent = append(nonAdjacent, Match{i1, j1, k1}) + } + + nonAdjacent = append(nonAdjacent, Match{len(m.a), len(m.b), 0}) + m.matchingBlocks = nonAdjacent + return m.matchingBlocks +} + +// Return list of 5-tuples describing how to turn a into b. +// +// Each tuple is of the form (tag, i1, i2, j1, j2). The first tuple +// has i1 == j1 == 0, and remaining tuples have i1 == the i2 from the +// tuple preceding it, and likewise for j1 == the previous j2. +// +// The tags are characters, with these meanings: +// +// 'r' (replace): a[i1:i2] should be replaced by b[j1:j2] +// +// 'd' (delete): a[i1:i2] should be deleted, j1==j2 in this case. +// +// 'i' (insert): b[j1:j2] should be inserted at a[i1:i1], i1==i2 in this case. +// +// 'e' (equal): a[i1:i2] == b[j1:j2] +func (m *SequenceMatcher) GetOpCodes() []OpCode { + if m.opCodes != nil { + return m.opCodes + } + i, j := 0, 0 + matching := m.GetMatchingBlocks() + opCodes := make([]OpCode, 0, len(matching)) + for _, m := range matching { + // invariant: we've pumped out correct diffs to change + // a[:i] into b[:j], and the next matching block is + // a[ai:ai+size] == b[bj:bj+size]. So we need to pump + // out a diff to change a[i:ai] into b[j:bj], pump out + // the matching block, and move (i,j) beyond the match + ai, bj, size := m.A, m.B, m.Size + tag := byte(0) + if i < ai && j < bj { + tag = 'r' + } else if i < ai { + tag = 'd' + } else if j < bj { + tag = 'i' + } + if tag > 0 { + opCodes = append(opCodes, OpCode{tag, i, ai, j, bj}) + } + i, j = ai+size, bj+size + // the list of matching blocks is terminated by a + // sentinel with size 0 + if size > 0 { + opCodes = append(opCodes, OpCode{'e', ai, i, bj, j}) + } + } + m.opCodes = opCodes + return m.opCodes +} + +// Isolate change clusters by eliminating ranges with no changes. +// +// Return a generator of groups with up to n lines of context. +// Each group is in the same format as returned by GetOpCodes(). +func (m *SequenceMatcher) GetGroupedOpCodes(n int) [][]OpCode { + if n < 0 { + n = 3 + } + codes := m.GetOpCodes() + if len(codes) == 0 { + codes = []OpCode{OpCode{'e', 0, 1, 0, 1}} + } + // Fixup leading and trailing groups if they show no changes. + if codes[0].Tag == 'e' { + c := codes[0] + i1, i2, j1, j2 := c.I1, c.I2, c.J1, c.J2 + codes[0] = OpCode{c.Tag, max(i1, i2-n), i2, max(j1, j2-n), j2} + } + if codes[len(codes)-1].Tag == 'e' { + c := codes[len(codes)-1] + i1, i2, j1, j2 := c.I1, c.I2, c.J1, c.J2 + codes[len(codes)-1] = OpCode{c.Tag, i1, min(i2, i1+n), j1, min(j2, j1+n)} + } + nn := n + n + groups := [][]OpCode{} + group := []OpCode{} + for _, c := range codes { + i1, i2, j1, j2 := c.I1, c.I2, c.J1, c.J2 + // End the current group and start a new one whenever + // there is a large range with no changes. + if c.Tag == 'e' && i2-i1 > nn { + group = append(group, OpCode{c.Tag, i1, min(i2, i1+n), + j1, min(j2, j1+n)}) + groups = append(groups, group) + group = []OpCode{} + i1, j1 = max(i1, i2-n), max(j1, j2-n) + } + group = append(group, OpCode{c.Tag, i1, i2, j1, j2}) + } + if len(group) > 0 && !(len(group) == 1 && group[0].Tag == 'e') { + groups = append(groups, group) + } + return groups +} + +// Return a measure of the sequences' similarity (float in [0,1]). +// +// Where T is the total number of elements in both sequences, and +// M is the number of matches, this is 2.0*M / T. +// Note that this is 1 if the sequences are identical, and 0 if +// they have nothing in common. +// +// .Ratio() is expensive to compute if you haven't already computed +// .GetMatchingBlocks() or .GetOpCodes(), in which case you may +// want to try .QuickRatio() or .RealQuickRation() first to get an +// upper bound. +func (m *SequenceMatcher) Ratio() float64 { + matches := 0 + for _, m := range m.GetMatchingBlocks() { + matches += m.Size + } + return calculateRatio(matches, len(m.a)+len(m.b)) +} + +// Return an upper bound on ratio() relatively quickly. +// +// This isn't defined beyond that it is an upper bound on .Ratio(), and +// is faster to compute. +func (m *SequenceMatcher) QuickRatio() float64 { + // viewing a and b as multisets, set matches to the cardinality + // of their intersection; this counts the number of matches + // without regard to order, so is clearly an upper bound + if m.fullBCount == nil { + m.fullBCount = map[string]int{} + for _, s := range m.b { + m.fullBCount[s] = m.fullBCount[s] + 1 + } + } + + // avail[x] is the number of times x appears in 'b' less the + // number of times we've seen it in 'a' so far ... kinda + avail := map[string]int{} + matches := 0 + for _, s := range m.a { + n, ok := avail[s] + if !ok { + n = m.fullBCount[s] + } + avail[s] = n - 1 + if n > 0 { + matches += 1 + } + } + return calculateRatio(matches, len(m.a)+len(m.b)) +} + +// Return an upper bound on ratio() very quickly. +// +// This isn't defined beyond that it is an upper bound on .Ratio(), and +// is faster to compute than either .Ratio() or .QuickRatio(). +func (m *SequenceMatcher) RealQuickRatio() float64 { + la, lb := len(m.a), len(m.b) + return calculateRatio(min(la, lb), la+lb) +} + +// Convert range to the "ed" format +func formatRangeUnified(start, stop int) string { + // Per the diff spec at http://www.unix.org/single_unix_specification/ + beginning := start + 1 // lines start numbering with one + length := stop - start + if length == 1 { + return fmt.Sprintf("%d", beginning) + } + if length == 0 { + beginning -= 1 // empty ranges begin at line just before the range + } + return fmt.Sprintf("%d,%d", beginning, length) +} + +// Unified diff parameters +type UnifiedDiff struct { + A []string // First sequence lines + FromFile string // First file name + FromDate string // First file time + B []string // Second sequence lines + ToFile string // Second file name + ToDate string // Second file time + Eol string // Headers end of line, defaults to LF + Context int // Number of context lines +} + +// Compare two sequences of lines; generate the delta as a unified diff. +// +// Unified diffs are a compact way of showing line changes and a few +// lines of context. The number of context lines is set by 'n' which +// defaults to three. +// +// By default, the diff control lines (those with ---, +++, or @@) are +// created with a trailing newline. This is helpful so that inputs +// created from file.readlines() result in diffs that are suitable for +// file.writelines() since both the inputs and outputs have trailing +// newlines. +// +// For inputs that do not have trailing newlines, set the lineterm +// argument to "" so that the output will be uniformly newline free. +// +// The unidiff format normally has a header for filenames and modification +// times. Any or all of these may be specified using strings for +// 'fromfile', 'tofile', 'fromfiledate', and 'tofiledate'. +// The modification times are normally expressed in the ISO 8601 format. +func WriteUnifiedDiff(writer io.Writer, diff UnifiedDiff) error { + buf := bufio.NewWriter(writer) + defer buf.Flush() + wf := func(format string, args ...interface{}) error { + _, err := buf.WriteString(fmt.Sprintf(format, args...)) + return err + } + ws := func(s string) error { + _, err := buf.WriteString(s) + return err + } + + if len(diff.Eol) == 0 { + diff.Eol = "\n" + } + + started := false + m := NewMatcher(diff.A, diff.B) + for _, g := range m.GetGroupedOpCodes(diff.Context) { + if !started { + started = true + fromDate := "" + if len(diff.FromDate) > 0 { + fromDate = "\t" + diff.FromDate + } + toDate := "" + if len(diff.ToDate) > 0 { + toDate = "\t" + diff.ToDate + } + if diff.FromFile != "" || diff.ToFile != "" { + err := wf("--- %s%s%s", diff.FromFile, fromDate, diff.Eol) + if err != nil { + return err + } + err = wf("+++ %s%s%s", diff.ToFile, toDate, diff.Eol) + if err != nil { + return err + } + } + } + first, last := g[0], g[len(g)-1] + range1 := formatRangeUnified(first.I1, last.I2) + range2 := formatRangeUnified(first.J1, last.J2) + if err := wf("@@ -%s +%s @@%s", range1, range2, diff.Eol); err != nil { + return err + } + for _, c := range g { + i1, i2, j1, j2 := c.I1, c.I2, c.J1, c.J2 + if c.Tag == 'e' { + for _, line := range diff.A[i1:i2] { + if err := ws(" " + line); err != nil { + return err + } + } + continue + } + if c.Tag == 'r' || c.Tag == 'd' { + for _, line := range diff.A[i1:i2] { + if err := ws("-" + line); err != nil { + return err + } + } + } + if c.Tag == 'r' || c.Tag == 'i' { + for _, line := range diff.B[j1:j2] { + if err := ws("+" + line); err != nil { + return err + } + } + } + } + } + return nil +} + +// Like WriteUnifiedDiff but returns the diff a string. +func GetUnifiedDiffString(diff UnifiedDiff) (string, error) { + w := &bytes.Buffer{} + err := WriteUnifiedDiff(w, diff) + return string(w.Bytes()), err +} + +// Convert range to the "ed" format. +func formatRangeContext(start, stop int) string { + // Per the diff spec at http://www.unix.org/single_unix_specification/ + beginning := start + 1 // lines start numbering with one + length := stop - start + if length == 0 { + beginning -= 1 // empty ranges begin at line just before the range + } + if length <= 1 { + return fmt.Sprintf("%d", beginning) + } + return fmt.Sprintf("%d,%d", beginning, beginning+length-1) +} + +type ContextDiff UnifiedDiff + +// Compare two sequences of lines; generate the delta as a context diff. +// +// Context diffs are a compact way of showing line changes and a few +// lines of context. The number of context lines is set by diff.Context +// which defaults to three. +// +// By default, the diff control lines (those with *** or ---) are +// created with a trailing newline. +// +// For inputs that do not have trailing newlines, set the diff.Eol +// argument to "" so that the output will be uniformly newline free. +// +// The context diff format normally has a header for filenames and +// modification times. Any or all of these may be specified using +// strings for diff.FromFile, diff.ToFile, diff.FromDate, diff.ToDate. +// The modification times are normally expressed in the ISO 8601 format. +// If not specified, the strings default to blanks. +func WriteContextDiff(writer io.Writer, diff ContextDiff) error { + buf := bufio.NewWriter(writer) + defer buf.Flush() + var diffErr error + wf := func(format string, args ...interface{}) { + _, err := buf.WriteString(fmt.Sprintf(format, args...)) + if diffErr == nil && err != nil { + diffErr = err + } + } + ws := func(s string) { + _, err := buf.WriteString(s) + if diffErr == nil && err != nil { + diffErr = err + } + } + + if len(diff.Eol) == 0 { + diff.Eol = "\n" + } + + prefix := map[byte]string{ + 'i': "+ ", + 'd': "- ", + 'r': "! ", + 'e': " ", + } + + started := false + m := NewMatcher(diff.A, diff.B) + for _, g := range m.GetGroupedOpCodes(diff.Context) { + if !started { + started = true + fromDate := "" + if len(diff.FromDate) > 0 { + fromDate = "\t" + diff.FromDate + } + toDate := "" + if len(diff.ToDate) > 0 { + toDate = "\t" + diff.ToDate + } + if diff.FromFile != "" || diff.ToFile != "" { + wf("*** %s%s%s", diff.FromFile, fromDate, diff.Eol) + wf("--- %s%s%s", diff.ToFile, toDate, diff.Eol) + } + } + + first, last := g[0], g[len(g)-1] + ws("***************" + diff.Eol) + + range1 := formatRangeContext(first.I1, last.I2) + wf("*** %s ****%s", range1, diff.Eol) + for _, c := range g { + if c.Tag == 'r' || c.Tag == 'd' { + for _, cc := range g { + if cc.Tag == 'i' { + continue + } + for _, line := range diff.A[cc.I1:cc.I2] { + ws(prefix[cc.Tag] + line) + } + } + break + } + } + + range2 := formatRangeContext(first.J1, last.J2) + wf("--- %s ----%s", range2, diff.Eol) + for _, c := range g { + if c.Tag == 'r' || c.Tag == 'i' { + for _, cc := range g { + if cc.Tag == 'd' { + continue + } + for _, line := range diff.B[cc.J1:cc.J2] { + ws(prefix[cc.Tag] + line) + } + } + break + } + } + } + return diffErr +} + +// Like WriteContextDiff but returns the diff a string. +func GetContextDiffString(diff ContextDiff) (string, error) { + w := &bytes.Buffer{} + err := WriteContextDiff(w, diff) + return string(w.Bytes()), err +} + +// Split a string on "\n" while preserving them. The output can be used +// as input for UnifiedDiff and ContextDiff structures. +func SplitLines(s string) []string { + lines := strings.SplitAfter(s, "\n") + lines[len(lines)-1] += "\n" + return lines +} diff --git a/vendor/github.com/rcrowley/go-metrics/.gitignore b/vendor/github.com/rcrowley/go-metrics/.gitignore new file mode 100644 index 00000000..83c8f823 --- /dev/null +++ b/vendor/github.com/rcrowley/go-metrics/.gitignore @@ -0,0 +1,9 @@ +*.[68] +*.a +*.out +*.swp +_obj +_testmain.go +cmd/metrics-bench/metrics-bench +cmd/metrics-example/metrics-example +cmd/never-read/never-read diff --git a/vendor/github.com/rcrowley/go-metrics/.travis.yml b/vendor/github.com/rcrowley/go-metrics/.travis.yml new file mode 100644 index 00000000..409a5b63 --- /dev/null +++ b/vendor/github.com/rcrowley/go-metrics/.travis.yml @@ -0,0 +1,22 @@ +language: go + +go: + - "1.3" + - "1.4" + - "1.5" + - "1.6" + - "1.7" + - "1.8" + - "1.9" + - "1.10" + - "1.11" + - "1.12" + - "1.13" + - "1.14" + +script: + - ./validate.sh + +# this should give us faster builds according to +# http://docs.travis-ci.com/user/migrating-from-legacy/ +sudo: false diff --git a/vendor/github.com/rcrowley/go-metrics/LICENSE b/vendor/github.com/rcrowley/go-metrics/LICENSE new file mode 100644 index 00000000..363fa9ee --- /dev/null +++ b/vendor/github.com/rcrowley/go-metrics/LICENSE @@ -0,0 +1,29 @@ +Copyright 2012 Richard Crowley. All rights reserved. + +Redistribution and use in source and binary forms, with or without +modification, are permitted provided that the following conditions are +met: + + 1. Redistributions of source code must retain the above copyright + notice, this list of conditions and the following disclaimer. + + 2. Redistributions in binary form must reproduce the above + copyright notice, this list of conditions and the following + disclaimer in the documentation and/or other materials provided + with the distribution. + +THIS SOFTWARE IS PROVIDED BY RICHARD CROWLEY ``AS IS'' AND ANY EXPRESS +OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED +WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE +DISCLAIMED. IN NO EVENT SHALL RICHARD CROWLEY OR CONTRIBUTORS BE LIABLE +FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR +CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF +SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS +INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN +CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) +ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF +THE POSSIBILITY OF SUCH DAMAGE. + +The views and conclusions contained in the software and documentation +are those of the authors and should not be interpreted as representing +official policies, either expressed or implied, of Richard Crowley. diff --git a/vendor/github.com/rcrowley/go-metrics/README.md b/vendor/github.com/rcrowley/go-metrics/README.md new file mode 100644 index 00000000..27ddfee8 --- /dev/null +++ b/vendor/github.com/rcrowley/go-metrics/README.md @@ -0,0 +1,171 @@ +go-metrics +========== + +![travis build status](https://travis-ci.org/rcrowley/go-metrics.svg?branch=master) + +Go port of Coda Hale's Metrics library: . + +Documentation: . + +Usage +----- + +Create and update metrics: + +```go +c := metrics.NewCounter() +metrics.Register("foo", c) +c.Inc(47) + +g := metrics.NewGauge() +metrics.Register("bar", g) +g.Update(47) + +r := NewRegistry() +g := metrics.NewRegisteredFunctionalGauge("cache-evictions", r, func() int64 { return cache.getEvictionsCount() }) + +s := metrics.NewExpDecaySample(1028, 0.015) // or metrics.NewUniformSample(1028) +h := metrics.NewHistogram(s) +metrics.Register("baz", h) +h.Update(47) + +m := metrics.NewMeter() +metrics.Register("quux", m) +m.Mark(47) + +t := metrics.NewTimer() +metrics.Register("bang", t) +t.Time(func() {}) +t.Update(47) +``` + +Register() is not threadsafe. For threadsafe metric registration use +GetOrRegister: + +```go +t := metrics.GetOrRegisterTimer("account.create.latency", nil) +t.Time(func() {}) +t.Update(47) +``` + +**NOTE:** Be sure to unregister short-lived meters and timers otherwise they will +leak memory: + +```go +// Will call Stop() on the Meter to allow for garbage collection +metrics.Unregister("quux") +// Or similarly for a Timer that embeds a Meter +metrics.Unregister("bang") +``` + +Periodically log every metric in human-readable form to standard error: + +```go +go metrics.Log(metrics.DefaultRegistry, 5 * time.Second, log.New(os.Stderr, "metrics: ", log.Lmicroseconds)) +``` + +Periodically log every metric in slightly-more-parseable form to syslog: + +```go +w, _ := syslog.Dial("unixgram", "/dev/log", syslog.LOG_INFO, "metrics") +go metrics.Syslog(metrics.DefaultRegistry, 60e9, w) +``` + +Periodically emit every metric to Graphite using the [Graphite client](https://github.com/cyberdelia/go-metrics-graphite): + +```go + +import "github.com/cyberdelia/go-metrics-graphite" + +addr, _ := net.ResolveTCPAddr("tcp", "127.0.0.1:2003") +go graphite.Graphite(metrics.DefaultRegistry, 10e9, "metrics", addr) +``` + +Periodically emit every metric into InfluxDB: + +**NOTE:** this has been pulled out of the library due to constant fluctuations +in the InfluxDB API. In fact, all client libraries are on their way out. see +issues [#121](https://github.com/rcrowley/go-metrics/issues/121) and +[#124](https://github.com/rcrowley/go-metrics/issues/124) for progress and details. + +```go +import "github.com/vrischmann/go-metrics-influxdb" + +go influxdb.InfluxDB(metrics.DefaultRegistry, + 10e9, + "127.0.0.1:8086", + "database-name", + "username", + "password" +) +``` + +Periodically upload every metric to Librato using the [Librato client](https://github.com/mihasya/go-metrics-librato): + +**Note**: the client included with this repository under the `librato` package +has been deprecated and moved to the repository linked above. + +```go +import "github.com/mihasya/go-metrics-librato" + +go librato.Librato(metrics.DefaultRegistry, + 10e9, // interval + "example@example.com", // account owner email address + "token", // Librato API token + "hostname", // source + []float64{0.95}, // percentiles to send + time.Millisecond, // time unit +) +``` + +Periodically emit every metric to StatHat: + +```go +import "github.com/rcrowley/go-metrics/stathat" + +go stathat.Stathat(metrics.DefaultRegistry, 10e9, "example@example.com") +``` + +Maintain all metrics along with expvars at `/debug/metrics`: + +This uses the same mechanism as [the official expvar](http://golang.org/pkg/expvar/) +but exposed under `/debug/metrics`, which shows a json representation of all your usual expvars +as well as all your go-metrics. + + +```go +import "github.com/rcrowley/go-metrics/exp" + +exp.Exp(metrics.DefaultRegistry) +``` + +Installation +------------ + +```sh +go get github.com/rcrowley/go-metrics +``` + +StatHat support additionally requires their Go client: + +```sh +go get github.com/stathat/go +``` + +Publishing Metrics +------------------ + +Clients are available for the following destinations: + +* AppOptics - https://github.com/ysamlan/go-metrics-appoptics +* Librato - https://github.com/mihasya/go-metrics-librato +* Graphite - https://github.com/cyberdelia/go-metrics-graphite +* InfluxDB - https://github.com/vrischmann/go-metrics-influxdb +* Ganglia - https://github.com/appscode/metlia +* Prometheus - https://github.com/deathowl/go-metrics-prometheus +* DataDog - https://github.com/syntaqx/go-metrics-datadog +* SignalFX - https://github.com/pascallouisperez/go-metrics-signalfx +* Honeycomb - https://github.com/getspine/go-metrics-honeycomb +* Wavefront - https://github.com/wavefrontHQ/go-metrics-wavefront +* Open-Falcon - https://github.com/g4zhuj/go-metrics-falcon +* AWS CloudWatch - [https://github.com/savaki/cloudmetrics](https://github.com/savaki/cloudmetrics) diff --git a/vendor/github.com/rcrowley/go-metrics/counter.go b/vendor/github.com/rcrowley/go-metrics/counter.go new file mode 100644 index 00000000..bb7b039c --- /dev/null +++ b/vendor/github.com/rcrowley/go-metrics/counter.go @@ -0,0 +1,112 @@ +package metrics + +import "sync/atomic" + +// Counters hold an int64 value that can be incremented and decremented. +type Counter interface { + Clear() + Count() int64 + Dec(int64) + Inc(int64) + Snapshot() Counter +} + +// GetOrRegisterCounter returns an existing Counter or constructs and registers +// a new StandardCounter. +func GetOrRegisterCounter(name string, r Registry) Counter { + if nil == r { + r = DefaultRegistry + } + return r.GetOrRegister(name, NewCounter).(Counter) +} + +// NewCounter constructs a new StandardCounter. +func NewCounter() Counter { + if UseNilMetrics { + return NilCounter{} + } + return &StandardCounter{0} +} + +// NewRegisteredCounter constructs and registers a new StandardCounter. +func NewRegisteredCounter(name string, r Registry) Counter { + c := NewCounter() + if nil == r { + r = DefaultRegistry + } + r.Register(name, c) + return c +} + +// CounterSnapshot is a read-only copy of another Counter. +type CounterSnapshot int64 + +// Clear panics. +func (CounterSnapshot) Clear() { + panic("Clear called on a CounterSnapshot") +} + +// Count returns the count at the time the snapshot was taken. +func (c CounterSnapshot) Count() int64 { return int64(c) } + +// Dec panics. +func (CounterSnapshot) Dec(int64) { + panic("Dec called on a CounterSnapshot") +} + +// Inc panics. +func (CounterSnapshot) Inc(int64) { + panic("Inc called on a CounterSnapshot") +} + +// Snapshot returns the snapshot. +func (c CounterSnapshot) Snapshot() Counter { return c } + +// NilCounter is a no-op Counter. +type NilCounter struct{} + +// Clear is a no-op. +func (NilCounter) Clear() {} + +// Count is a no-op. +func (NilCounter) Count() int64 { return 0 } + +// Dec is a no-op. +func (NilCounter) Dec(i int64) {} + +// Inc is a no-op. +func (NilCounter) Inc(i int64) {} + +// Snapshot is a no-op. +func (NilCounter) Snapshot() Counter { return NilCounter{} } + +// StandardCounter is the standard implementation of a Counter and uses the +// sync/atomic package to manage a single int64 value. +type StandardCounter struct { + count int64 +} + +// Clear sets the counter to zero. +func (c *StandardCounter) Clear() { + atomic.StoreInt64(&c.count, 0) +} + +// Count returns the current count. +func (c *StandardCounter) Count() int64 { + return atomic.LoadInt64(&c.count) +} + +// Dec decrements the counter by the given amount. +func (c *StandardCounter) Dec(i int64) { + atomic.AddInt64(&c.count, -i) +} + +// Inc increments the counter by the given amount. +func (c *StandardCounter) Inc(i int64) { + atomic.AddInt64(&c.count, i) +} + +// Snapshot returns a read-only copy of the counter. +func (c *StandardCounter) Snapshot() Counter { + return CounterSnapshot(c.Count()) +} diff --git a/vendor/github.com/rcrowley/go-metrics/debug.go b/vendor/github.com/rcrowley/go-metrics/debug.go new file mode 100644 index 00000000..179e5aae --- /dev/null +++ b/vendor/github.com/rcrowley/go-metrics/debug.go @@ -0,0 +1,80 @@ +package metrics + +import ( + "runtime/debug" + "sync" + "time" +) + +var ( + debugMetrics struct { + GCStats struct { + LastGC Gauge + NumGC Gauge + Pause Histogram + //PauseQuantiles Histogram + PauseTotal Gauge + } + ReadGCStats Timer + } + gcStats debug.GCStats + registerDebugMetricsOnce = sync.Once{} +) + +// Capture new values for the Go garbage collector statistics exported in +// debug.GCStats. This is designed to be called as a goroutine. +func CaptureDebugGCStats(r Registry, d time.Duration) { + for _ = range time.Tick(d) { + CaptureDebugGCStatsOnce(r) + } +} + +// Capture new values for the Go garbage collector statistics exported in +// debug.GCStats. This is designed to be called in a background goroutine. +// Giving a registry which has not been given to RegisterDebugGCStats will +// panic. +// +// Be careful (but much less so) with this because debug.ReadGCStats calls +// the C function runtime·lock(runtime·mheap) which, while not a stop-the-world +// operation, isn't something you want to be doing all the time. +func CaptureDebugGCStatsOnce(r Registry) { + lastGC := gcStats.LastGC + t := time.Now() + debug.ReadGCStats(&gcStats) + debugMetrics.ReadGCStats.UpdateSince(t) + + debugMetrics.GCStats.LastGC.Update(int64(gcStats.LastGC.UnixNano())) + debugMetrics.GCStats.NumGC.Update(int64(gcStats.NumGC)) + if lastGC != gcStats.LastGC && 0 < len(gcStats.Pause) { + debugMetrics.GCStats.Pause.Update(int64(gcStats.Pause[0])) + } + //debugMetrics.GCStats.PauseQuantiles.Update(gcStats.PauseQuantiles) + debugMetrics.GCStats.PauseTotal.Update(int64(gcStats.PauseTotal)) +} + +// Register metrics for the Go garbage collector statistics exported in +// debug.GCStats. The metrics are named by their fully-qualified Go symbols, +// i.e. debug.GCStats.PauseTotal. +func RegisterDebugGCStats(r Registry) { + registerDebugMetricsOnce.Do(func() { + debugMetrics.GCStats.LastGC = NewGauge() + debugMetrics.GCStats.NumGC = NewGauge() + debugMetrics.GCStats.Pause = NewHistogram(NewExpDecaySample(1028, 0.015)) + //debugMetrics.GCStats.PauseQuantiles = NewHistogram(NewExpDecaySample(1028, 0.015)) + debugMetrics.GCStats.PauseTotal = NewGauge() + debugMetrics.ReadGCStats = NewTimer() + + r.Register("debug.GCStats.LastGC", debugMetrics.GCStats.LastGC) + r.Register("debug.GCStats.NumGC", debugMetrics.GCStats.NumGC) + r.Register("debug.GCStats.Pause", debugMetrics.GCStats.Pause) + //r.Register("debug.GCStats.PauseQuantiles", debugMetrics.GCStats.PauseQuantiles) + r.Register("debug.GCStats.PauseTotal", debugMetrics.GCStats.PauseTotal) + r.Register("debug.ReadGCStats", debugMetrics.ReadGCStats) + }) +} + +// Allocate an initial slice for gcStats.Pause to avoid allocations during +// normal operation. +func init() { + gcStats.Pause = make([]time.Duration, 11) +} diff --git a/vendor/github.com/rcrowley/go-metrics/ewma.go b/vendor/github.com/rcrowley/go-metrics/ewma.go new file mode 100644 index 00000000..a8183dd7 --- /dev/null +++ b/vendor/github.com/rcrowley/go-metrics/ewma.go @@ -0,0 +1,138 @@ +package metrics + +import ( + "math" + "sync" + "sync/atomic" +) + +// EWMAs continuously calculate an exponentially-weighted moving average +// based on an outside source of clock ticks. +type EWMA interface { + Rate() float64 + Snapshot() EWMA + Tick() + Update(int64) +} + +// NewEWMA constructs a new EWMA with the given alpha. +func NewEWMA(alpha float64) EWMA { + if UseNilMetrics { + return NilEWMA{} + } + return &StandardEWMA{alpha: alpha} +} + +// NewEWMA1 constructs a new EWMA for a one-minute moving average. +func NewEWMA1() EWMA { + return NewEWMA(1 - math.Exp(-5.0/60.0/1)) +} + +// NewEWMA5 constructs a new EWMA for a five-minute moving average. +func NewEWMA5() EWMA { + return NewEWMA(1 - math.Exp(-5.0/60.0/5)) +} + +// NewEWMA15 constructs a new EWMA for a fifteen-minute moving average. +func NewEWMA15() EWMA { + return NewEWMA(1 - math.Exp(-5.0/60.0/15)) +} + +// EWMASnapshot is a read-only copy of another EWMA. +type EWMASnapshot float64 + +// Rate returns the rate of events per second at the time the snapshot was +// taken. +func (a EWMASnapshot) Rate() float64 { return float64(a) } + +// Snapshot returns the snapshot. +func (a EWMASnapshot) Snapshot() EWMA { return a } + +// Tick panics. +func (EWMASnapshot) Tick() { + panic("Tick called on an EWMASnapshot") +} + +// Update panics. +func (EWMASnapshot) Update(int64) { + panic("Update called on an EWMASnapshot") +} + +// NilEWMA is a no-op EWMA. +type NilEWMA struct{} + +// Rate is a no-op. +func (NilEWMA) Rate() float64 { return 0.0 } + +// Snapshot is a no-op. +func (NilEWMA) Snapshot() EWMA { return NilEWMA{} } + +// Tick is a no-op. +func (NilEWMA) Tick() {} + +// Update is a no-op. +func (NilEWMA) Update(n int64) {} + +// StandardEWMA is the standard implementation of an EWMA and tracks the number +// of uncounted events and processes them on each tick. It uses the +// sync/atomic package to manage uncounted events. +type StandardEWMA struct { + uncounted int64 // /!\ this should be the first member to ensure 64-bit alignment + alpha float64 + rate uint64 + init uint32 + mutex sync.Mutex +} + +// Rate returns the moving average rate of events per second. +func (a *StandardEWMA) Rate() float64 { + currentRate := math.Float64frombits(atomic.LoadUint64(&a.rate)) * float64(1e9) + return currentRate +} + +// Snapshot returns a read-only copy of the EWMA. +func (a *StandardEWMA) Snapshot() EWMA { + return EWMASnapshot(a.Rate()) +} + +// Tick ticks the clock to update the moving average. It assumes it is called +// every five seconds. +func (a *StandardEWMA) Tick() { + // Optimization to avoid mutex locking in the hot-path. + if atomic.LoadUint32(&a.init) == 1 { + a.updateRate(a.fetchInstantRate()) + } else { + // Slow-path: this is only needed on the first Tick() and preserves transactional updating + // of init and rate in the else block. The first conditional is needed below because + // a different thread could have set a.init = 1 between the time of the first atomic load and when + // the lock was acquired. + a.mutex.Lock() + if atomic.LoadUint32(&a.init) == 1 { + // The fetchInstantRate() uses atomic loading, which is unecessary in this critical section + // but again, this section is only invoked on the first successful Tick() operation. + a.updateRate(a.fetchInstantRate()) + } else { + atomic.StoreUint32(&a.init, 1) + atomic.StoreUint64(&a.rate, math.Float64bits(a.fetchInstantRate())) + } + a.mutex.Unlock() + } +} + +func (a *StandardEWMA) fetchInstantRate() float64 { + count := atomic.LoadInt64(&a.uncounted) + atomic.AddInt64(&a.uncounted, -count) + instantRate := float64(count) / float64(5e9) + return instantRate +} + +func (a *StandardEWMA) updateRate(instantRate float64) { + currentRate := math.Float64frombits(atomic.LoadUint64(&a.rate)) + currentRate += a.alpha * (instantRate - currentRate) + atomic.StoreUint64(&a.rate, math.Float64bits(currentRate)) +} + +// Update adds n uncounted events. +func (a *StandardEWMA) Update(n int64) { + atomic.AddInt64(&a.uncounted, n) +} diff --git a/vendor/github.com/rcrowley/go-metrics/gauge.go b/vendor/github.com/rcrowley/go-metrics/gauge.go new file mode 100644 index 00000000..cb57a938 --- /dev/null +++ b/vendor/github.com/rcrowley/go-metrics/gauge.go @@ -0,0 +1,120 @@ +package metrics + +import "sync/atomic" + +// Gauges hold an int64 value that can be set arbitrarily. +type Gauge interface { + Snapshot() Gauge + Update(int64) + Value() int64 +} + +// GetOrRegisterGauge returns an existing Gauge or constructs and registers a +// new StandardGauge. +func GetOrRegisterGauge(name string, r Registry) Gauge { + if nil == r { + r = DefaultRegistry + } + return r.GetOrRegister(name, NewGauge).(Gauge) +} + +// NewGauge constructs a new StandardGauge. +func NewGauge() Gauge { + if UseNilMetrics { + return NilGauge{} + } + return &StandardGauge{0} +} + +// NewRegisteredGauge constructs and registers a new StandardGauge. +func NewRegisteredGauge(name string, r Registry) Gauge { + c := NewGauge() + if nil == r { + r = DefaultRegistry + } + r.Register(name, c) + return c +} + +// NewFunctionalGauge constructs a new FunctionalGauge. +func NewFunctionalGauge(f func() int64) Gauge { + if UseNilMetrics { + return NilGauge{} + } + return &FunctionalGauge{value: f} +} + +// NewRegisteredFunctionalGauge constructs and registers a new StandardGauge. +func NewRegisteredFunctionalGauge(name string, r Registry, f func() int64) Gauge { + c := NewFunctionalGauge(f) + if nil == r { + r = DefaultRegistry + } + r.Register(name, c) + return c +} + +// GaugeSnapshot is a read-only copy of another Gauge. +type GaugeSnapshot int64 + +// Snapshot returns the snapshot. +func (g GaugeSnapshot) Snapshot() Gauge { return g } + +// Update panics. +func (GaugeSnapshot) Update(int64) { + panic("Update called on a GaugeSnapshot") +} + +// Value returns the value at the time the snapshot was taken. +func (g GaugeSnapshot) Value() int64 { return int64(g) } + +// NilGauge is a no-op Gauge. +type NilGauge struct{} + +// Snapshot is a no-op. +func (NilGauge) Snapshot() Gauge { return NilGauge{} } + +// Update is a no-op. +func (NilGauge) Update(v int64) {} + +// Value is a no-op. +func (NilGauge) Value() int64 { return 0 } + +// StandardGauge is the standard implementation of a Gauge and uses the +// sync/atomic package to manage a single int64 value. +type StandardGauge struct { + value int64 +} + +// Snapshot returns a read-only copy of the gauge. +func (g *StandardGauge) Snapshot() Gauge { + return GaugeSnapshot(g.Value()) +} + +// Update updates the gauge's value. +func (g *StandardGauge) Update(v int64) { + atomic.StoreInt64(&g.value, v) +} + +// Value returns the gauge's current value. +func (g *StandardGauge) Value() int64 { + return atomic.LoadInt64(&g.value) +} + +// FunctionalGauge returns value from given function +type FunctionalGauge struct { + value func() int64 +} + +// Value returns the gauge's current value. +func (g FunctionalGauge) Value() int64 { + return g.value() +} + +// Snapshot returns the snapshot. +func (g FunctionalGauge) Snapshot() Gauge { return GaugeSnapshot(g.Value()) } + +// Update panics. +func (FunctionalGauge) Update(int64) { + panic("Update called on a FunctionalGauge") +} diff --git a/vendor/github.com/rcrowley/go-metrics/gauge_float64.go b/vendor/github.com/rcrowley/go-metrics/gauge_float64.go new file mode 100644 index 00000000..3962e6db --- /dev/null +++ b/vendor/github.com/rcrowley/go-metrics/gauge_float64.go @@ -0,0 +1,125 @@ +package metrics + +import ( + "math" + "sync/atomic" +) + +// GaugeFloat64s hold a float64 value that can be set arbitrarily. +type GaugeFloat64 interface { + Snapshot() GaugeFloat64 + Update(float64) + Value() float64 +} + +// GetOrRegisterGaugeFloat64 returns an existing GaugeFloat64 or constructs and registers a +// new StandardGaugeFloat64. +func GetOrRegisterGaugeFloat64(name string, r Registry) GaugeFloat64 { + if nil == r { + r = DefaultRegistry + } + return r.GetOrRegister(name, NewGaugeFloat64()).(GaugeFloat64) +} + +// NewGaugeFloat64 constructs a new StandardGaugeFloat64. +func NewGaugeFloat64() GaugeFloat64 { + if UseNilMetrics { + return NilGaugeFloat64{} + } + return &StandardGaugeFloat64{ + value: 0.0, + } +} + +// NewRegisteredGaugeFloat64 constructs and registers a new StandardGaugeFloat64. +func NewRegisteredGaugeFloat64(name string, r Registry) GaugeFloat64 { + c := NewGaugeFloat64() + if nil == r { + r = DefaultRegistry + } + r.Register(name, c) + return c +} + +// NewFunctionalGauge constructs a new FunctionalGauge. +func NewFunctionalGaugeFloat64(f func() float64) GaugeFloat64 { + if UseNilMetrics { + return NilGaugeFloat64{} + } + return &FunctionalGaugeFloat64{value: f} +} + +// NewRegisteredFunctionalGauge constructs and registers a new StandardGauge. +func NewRegisteredFunctionalGaugeFloat64(name string, r Registry, f func() float64) GaugeFloat64 { + c := NewFunctionalGaugeFloat64(f) + if nil == r { + r = DefaultRegistry + } + r.Register(name, c) + return c +} + +// GaugeFloat64Snapshot is a read-only copy of another GaugeFloat64. +type GaugeFloat64Snapshot float64 + +// Snapshot returns the snapshot. +func (g GaugeFloat64Snapshot) Snapshot() GaugeFloat64 { return g } + +// Update panics. +func (GaugeFloat64Snapshot) Update(float64) { + panic("Update called on a GaugeFloat64Snapshot") +} + +// Value returns the value at the time the snapshot was taken. +func (g GaugeFloat64Snapshot) Value() float64 { return float64(g) } + +// NilGauge is a no-op Gauge. +type NilGaugeFloat64 struct{} + +// Snapshot is a no-op. +func (NilGaugeFloat64) Snapshot() GaugeFloat64 { return NilGaugeFloat64{} } + +// Update is a no-op. +func (NilGaugeFloat64) Update(v float64) {} + +// Value is a no-op. +func (NilGaugeFloat64) Value() float64 { return 0.0 } + +// StandardGaugeFloat64 is the standard implementation of a GaugeFloat64 and uses +// sync.Mutex to manage a single float64 value. +type StandardGaugeFloat64 struct { + value uint64 +} + +// Snapshot returns a read-only copy of the gauge. +func (g *StandardGaugeFloat64) Snapshot() GaugeFloat64 { + return GaugeFloat64Snapshot(g.Value()) +} + +// Update updates the gauge's value. +func (g *StandardGaugeFloat64) Update(v float64) { + atomic.StoreUint64(&g.value, math.Float64bits(v)) +} + +// Value returns the gauge's current value. +func (g *StandardGaugeFloat64) Value() float64 { + return math.Float64frombits(atomic.LoadUint64(&g.value)) +} + +// FunctionalGaugeFloat64 returns value from given function +type FunctionalGaugeFloat64 struct { + value func() float64 +} + +// Value returns the gauge's current value. +func (g FunctionalGaugeFloat64) Value() float64 { + return g.value() +} + +// Snapshot returns the snapshot. +func (g FunctionalGaugeFloat64) Snapshot() GaugeFloat64 { return GaugeFloat64Snapshot(g.Value()) } + +// Update panics. +func (FunctionalGaugeFloat64) Update(float64) { + panic("Update called on a FunctionalGaugeFloat64") +} diff --git a/vendor/github.com/rcrowley/go-metrics/graphite.go b/vendor/github.com/rcrowley/go-metrics/graphite.go new file mode 100644 index 00000000..abd0a7d2 --- /dev/null +++ b/vendor/github.com/rcrowley/go-metrics/graphite.go @@ -0,0 +1,113 @@ +package metrics + +import ( + "bufio" + "fmt" + "log" + "net" + "strconv" + "strings" + "time" +) + +// GraphiteConfig provides a container with configuration parameters for +// the Graphite exporter +type GraphiteConfig struct { + Addr *net.TCPAddr // Network address to connect to + Registry Registry // Registry to be exported + FlushInterval time.Duration // Flush interval + DurationUnit time.Duration // Time conversion unit for durations + Prefix string // Prefix to be prepended to metric names + Percentiles []float64 // Percentiles to export from timers and histograms +} + +// Graphite is a blocking exporter function which reports metrics in r +// to a graphite server located at addr, flushing them every d duration +// and prepending metric names with prefix. +func Graphite(r Registry, d time.Duration, prefix string, addr *net.TCPAddr) { + GraphiteWithConfig(GraphiteConfig{ + Addr: addr, + Registry: r, + FlushInterval: d, + DurationUnit: time.Nanosecond, + Prefix: prefix, + Percentiles: []float64{0.5, 0.75, 0.95, 0.99, 0.999}, + }) +} + +// GraphiteWithConfig is a blocking exporter function just like Graphite, +// but it takes a GraphiteConfig instead. +func GraphiteWithConfig(c GraphiteConfig) { + log.Printf("WARNING: This go-metrics client has been DEPRECATED! It has been moved to https://github.com/cyberdelia/go-metrics-graphite and will be removed from rcrowley/go-metrics on August 12th 2015") + for _ = range time.Tick(c.FlushInterval) { + if err := graphite(&c); nil != err { + log.Println(err) + } + } +} + +// GraphiteOnce performs a single submission to Graphite, returning a +// non-nil error on failed connections. This can be used in a loop +// similar to GraphiteWithConfig for custom error handling. +func GraphiteOnce(c GraphiteConfig) error { + log.Printf("WARNING: This go-metrics client has been DEPRECATED! It has been moved to https://github.com/cyberdelia/go-metrics-graphite and will be removed from rcrowley/go-metrics on August 12th 2015") + return graphite(&c) +} + +func graphite(c *GraphiteConfig) error { + now := time.Now().Unix() + du := float64(c.DurationUnit) + conn, err := net.DialTCP("tcp", nil, c.Addr) + if nil != err { + return err + } + defer conn.Close() + w := bufio.NewWriter(conn) + c.Registry.Each(func(name string, i interface{}) { + switch metric := i.(type) { + case Counter: + fmt.Fprintf(w, "%s.%s.count %d %d\n", c.Prefix, name, metric.Count(), now) + case Gauge: + fmt.Fprintf(w, "%s.%s.value %d %d\n", c.Prefix, name, metric.Value(), now) + case GaugeFloat64: + fmt.Fprintf(w, "%s.%s.value %f %d\n", c.Prefix, name, metric.Value(), now) + case Histogram: + h := metric.Snapshot() + ps := h.Percentiles(c.Percentiles) + fmt.Fprintf(w, "%s.%s.count %d %d\n", c.Prefix, name, h.Count(), now) + fmt.Fprintf(w, "%s.%s.min %d %d\n", c.Prefix, name, h.Min(), now) + fmt.Fprintf(w, "%s.%s.max %d %d\n", c.Prefix, name, h.Max(), now) + fmt.Fprintf(w, "%s.%s.mean %.2f %d\n", c.Prefix, name, h.Mean(), now) + fmt.Fprintf(w, "%s.%s.std-dev %.2f %d\n", c.Prefix, name, h.StdDev(), now) + for psIdx, psKey := range c.Percentiles { + key := strings.Replace(strconv.FormatFloat(psKey*100.0, 'f', -1, 64), ".", "", 1) + fmt.Fprintf(w, "%s.%s.%s-percentile %.2f %d\n", c.Prefix, name, key, ps[psIdx], now) + } + case Meter: + m := metric.Snapshot() + fmt.Fprintf(w, "%s.%s.count %d %d\n", c.Prefix, name, m.Count(), now) + fmt.Fprintf(w, "%s.%s.one-minute %.2f %d\n", c.Prefix, name, m.Rate1(), now) + fmt.Fprintf(w, "%s.%s.five-minute %.2f %d\n", c.Prefix, name, m.Rate5(), now) + fmt.Fprintf(w, "%s.%s.fifteen-minute %.2f %d\n", c.Prefix, name, m.Rate15(), now) + fmt.Fprintf(w, "%s.%s.mean %.2f %d\n", c.Prefix, name, m.RateMean(), now) + case Timer: + t := metric.Snapshot() + ps := t.Percentiles(c.Percentiles) + fmt.Fprintf(w, "%s.%s.count %d %d\n", c.Prefix, name, t.Count(), now) + fmt.Fprintf(w, "%s.%s.min %d %d\n", c.Prefix, name, t.Min()/int64(du), now) + fmt.Fprintf(w, "%s.%s.max %d %d\n", c.Prefix, name, t.Max()/int64(du), now) + fmt.Fprintf(w, "%s.%s.mean %.2f %d\n", c.Prefix, name, t.Mean()/du, now) + fmt.Fprintf(w, "%s.%s.std-dev %.2f %d\n", c.Prefix, name, t.StdDev()/du, now) + for psIdx, psKey := range c.Percentiles { + key := strings.Replace(strconv.FormatFloat(psKey*100.0, 'f', -1, 64), ".", "", 1) + fmt.Fprintf(w, "%s.%s.%s-percentile %.2f %d\n", c.Prefix, name, key, ps[psIdx], now) + } + fmt.Fprintf(w, "%s.%s.one-minute %.2f %d\n", c.Prefix, name, t.Rate1(), now) + fmt.Fprintf(w, "%s.%s.five-minute %.2f %d\n", c.Prefix, name, t.Rate5(), now) + fmt.Fprintf(w, "%s.%s.fifteen-minute %.2f %d\n", c.Prefix, name, t.Rate15(), now) + fmt.Fprintf(w, "%s.%s.mean-rate %.2f %d\n", c.Prefix, name, t.RateMean(), now) + } + w.Flush() + }) + return nil +} diff --git a/vendor/github.com/rcrowley/go-metrics/healthcheck.go b/vendor/github.com/rcrowley/go-metrics/healthcheck.go new file mode 100644 index 00000000..445131ca --- /dev/null +++ b/vendor/github.com/rcrowley/go-metrics/healthcheck.go @@ -0,0 +1,61 @@ +package metrics + +// Healthchecks hold an error value describing an arbitrary up/down status. +type Healthcheck interface { + Check() + Error() error + Healthy() + Unhealthy(error) +} + +// NewHealthcheck constructs a new Healthcheck which will use the given +// function to update its status. +func NewHealthcheck(f func(Healthcheck)) Healthcheck { + if UseNilMetrics { + return NilHealthcheck{} + } + return &StandardHealthcheck{nil, f} +} + +// NilHealthcheck is a no-op. +type NilHealthcheck struct{} + +// Check is a no-op. +func (NilHealthcheck) Check() {} + +// Error is a no-op. +func (NilHealthcheck) Error() error { return nil } + +// Healthy is a no-op. +func (NilHealthcheck) Healthy() {} + +// Unhealthy is a no-op. +func (NilHealthcheck) Unhealthy(error) {} + +// StandardHealthcheck is the standard implementation of a Healthcheck and +// stores the status and a function to call to update the status. +type StandardHealthcheck struct { + err error + f func(Healthcheck) +} + +// Check runs the healthcheck function to update the healthcheck's status. +func (h *StandardHealthcheck) Check() { + h.f(h) +} + +// Error returns the healthcheck's status, which will be nil if it is healthy. +func (h *StandardHealthcheck) Error() error { + return h.err +} + +// Healthy marks the healthcheck as healthy. +func (h *StandardHealthcheck) Healthy() { + h.err = nil +} + +// Unhealthy marks the healthcheck as unhealthy. The error is stored and +// may be retrieved by the Error method. +func (h *StandardHealthcheck) Unhealthy(err error) { + h.err = err +} diff --git a/vendor/github.com/rcrowley/go-metrics/histogram.go b/vendor/github.com/rcrowley/go-metrics/histogram.go new file mode 100644 index 00000000..dbc837fe --- /dev/null +++ b/vendor/github.com/rcrowley/go-metrics/histogram.go @@ -0,0 +1,202 @@ +package metrics + +// Histograms calculate distribution statistics from a series of int64 values. +type Histogram interface { + Clear() + Count() int64 + Max() int64 + Mean() float64 + Min() int64 + Percentile(float64) float64 + Percentiles([]float64) []float64 + Sample() Sample + Snapshot() Histogram + StdDev() float64 + Sum() int64 + Update(int64) + Variance() float64 +} + +// GetOrRegisterHistogram returns an existing Histogram or constructs and +// registers a new StandardHistogram. +func GetOrRegisterHistogram(name string, r Registry, s Sample) Histogram { + if nil == r { + r = DefaultRegistry + } + return r.GetOrRegister(name, func() Histogram { return NewHistogram(s) }).(Histogram) +} + +// NewHistogram constructs a new StandardHistogram from a Sample. +func NewHistogram(s Sample) Histogram { + if UseNilMetrics { + return NilHistogram{} + } + return &StandardHistogram{sample: s} +} + +// NewRegisteredHistogram constructs and registers a new StandardHistogram from +// a Sample. +func NewRegisteredHistogram(name string, r Registry, s Sample) Histogram { + c := NewHistogram(s) + if nil == r { + r = DefaultRegistry + } + r.Register(name, c) + return c +} + +// HistogramSnapshot is a read-only copy of another Histogram. +type HistogramSnapshot struct { + sample *SampleSnapshot +} + +// Clear panics. +func (*HistogramSnapshot) Clear() { + panic("Clear called on a HistogramSnapshot") +} + +// Count returns the number of samples recorded at the time the snapshot was +// taken. +func (h *HistogramSnapshot) Count() int64 { return h.sample.Count() } + +// Max returns the maximum value in the sample at the time the snapshot was +// taken. +func (h *HistogramSnapshot) Max() int64 { return h.sample.Max() } + +// Mean returns the mean of the values in the sample at the time the snapshot +// was taken. +func (h *HistogramSnapshot) Mean() float64 { return h.sample.Mean() } + +// Min returns the minimum value in the sample at the time the snapshot was +// taken. +func (h *HistogramSnapshot) Min() int64 { return h.sample.Min() } + +// Percentile returns an arbitrary percentile of values in the sample at the +// time the snapshot was taken. +func (h *HistogramSnapshot) Percentile(p float64) float64 { + return h.sample.Percentile(p) +} + +// Percentiles returns a slice of arbitrary percentiles of values in the sample +// at the time the snapshot was taken. +func (h *HistogramSnapshot) Percentiles(ps []float64) []float64 { + return h.sample.Percentiles(ps) +} + +// Sample returns the Sample underlying the histogram. +func (h *HistogramSnapshot) Sample() Sample { return h.sample } + +// Snapshot returns the snapshot. +func (h *HistogramSnapshot) Snapshot() Histogram { return h } + +// StdDev returns the standard deviation of the values in the sample at the +// time the snapshot was taken. +func (h *HistogramSnapshot) StdDev() float64 { return h.sample.StdDev() } + +// Sum returns the sum in the sample at the time the snapshot was taken. +func (h *HistogramSnapshot) Sum() int64 { return h.sample.Sum() } + +// Update panics. +func (*HistogramSnapshot) Update(int64) { + panic("Update called on a HistogramSnapshot") +} + +// Variance returns the variance of inputs at the time the snapshot was taken. +func (h *HistogramSnapshot) Variance() float64 { return h.sample.Variance() } + +// NilHistogram is a no-op Histogram. +type NilHistogram struct{} + +// Clear is a no-op. +func (NilHistogram) Clear() {} + +// Count is a no-op. +func (NilHistogram) Count() int64 { return 0 } + +// Max is a no-op. +func (NilHistogram) Max() int64 { return 0 } + +// Mean is a no-op. +func (NilHistogram) Mean() float64 { return 0.0 } + +// Min is a no-op. +func (NilHistogram) Min() int64 { return 0 } + +// Percentile is a no-op. +func (NilHistogram) Percentile(p float64) float64 { return 0.0 } + +// Percentiles is a no-op. +func (NilHistogram) Percentiles(ps []float64) []float64 { + return make([]float64, len(ps)) +} + +// Sample is a no-op. +func (NilHistogram) Sample() Sample { return NilSample{} } + +// Snapshot is a no-op. +func (NilHistogram) Snapshot() Histogram { return NilHistogram{} } + +// StdDev is a no-op. +func (NilHistogram) StdDev() float64 { return 0.0 } + +// Sum is a no-op. +func (NilHistogram) Sum() int64 { return 0 } + +// Update is a no-op. +func (NilHistogram) Update(v int64) {} + +// Variance is a no-op. +func (NilHistogram) Variance() float64 { return 0.0 } + +// StandardHistogram is the standard implementation of a Histogram and uses a +// Sample to bound its memory use. +type StandardHistogram struct { + sample Sample +} + +// Clear clears the histogram and its sample. +func (h *StandardHistogram) Clear() { h.sample.Clear() } + +// Count returns the number of samples recorded since the histogram was last +// cleared. +func (h *StandardHistogram) Count() int64 { return h.sample.Count() } + +// Max returns the maximum value in the sample. +func (h *StandardHistogram) Max() int64 { return h.sample.Max() } + +// Mean returns the mean of the values in the sample. +func (h *StandardHistogram) Mean() float64 { return h.sample.Mean() } + +// Min returns the minimum value in the sample. +func (h *StandardHistogram) Min() int64 { return h.sample.Min() } + +// Percentile returns an arbitrary percentile of the values in the sample. +func (h *StandardHistogram) Percentile(p float64) float64 { + return h.sample.Percentile(p) +} + +// Percentiles returns a slice of arbitrary percentiles of the values in the +// sample. +func (h *StandardHistogram) Percentiles(ps []float64) []float64 { + return h.sample.Percentiles(ps) +} + +// Sample returns the Sample underlying the histogram. +func (h *StandardHistogram) Sample() Sample { return h.sample } + +// Snapshot returns a read-only copy of the histogram. +func (h *StandardHistogram) Snapshot() Histogram { + return &HistogramSnapshot{sample: h.sample.Snapshot().(*SampleSnapshot)} +} + +// StdDev returns the standard deviation of the values in the sample. +func (h *StandardHistogram) StdDev() float64 { return h.sample.StdDev() } + +// Sum returns the sum in the sample. +func (h *StandardHistogram) Sum() int64 { return h.sample.Sum() } + +// Update samples a new value. +func (h *StandardHistogram) Update(v int64) { h.sample.Update(v) } + +// Variance returns the variance of the values in the sample. +func (h *StandardHistogram) Variance() float64 { return h.sample.Variance() } diff --git a/vendor/github.com/rcrowley/go-metrics/json.go b/vendor/github.com/rcrowley/go-metrics/json.go new file mode 100644 index 00000000..174b9477 --- /dev/null +++ b/vendor/github.com/rcrowley/go-metrics/json.go @@ -0,0 +1,31 @@ +package metrics + +import ( + "encoding/json" + "io" + "time" +) + +// MarshalJSON returns a byte slice containing a JSON representation of all +// the metrics in the Registry. +func (r *StandardRegistry) MarshalJSON() ([]byte, error) { + return json.Marshal(r.GetAll()) +} + +// WriteJSON writes metrics from the given registry periodically to the +// specified io.Writer as JSON. +func WriteJSON(r Registry, d time.Duration, w io.Writer) { + for _ = range time.Tick(d) { + WriteJSONOnce(r, w) + } +} + +// WriteJSONOnce writes metrics from the given registry to the specified +// io.Writer as JSON. +func WriteJSONOnce(r Registry, w io.Writer) { + json.NewEncoder(w).Encode(r) +} + +func (p *PrefixedRegistry) MarshalJSON() ([]byte, error) { + return json.Marshal(p.GetAll()) +} diff --git a/vendor/github.com/rcrowley/go-metrics/log.go b/vendor/github.com/rcrowley/go-metrics/log.go new file mode 100644 index 00000000..2614a0a3 --- /dev/null +++ b/vendor/github.com/rcrowley/go-metrics/log.go @@ -0,0 +1,100 @@ +package metrics + +import ( + "time" +) + +type Logger interface { + Printf(format string, v ...interface{}) +} + +// Log outputs each metric in the given registry periodically using the given logger. +func Log(r Registry, freq time.Duration, l Logger) { + LogScaled(r, freq, time.Nanosecond, l) +} + +// LogOnCue outputs each metric in the given registry on demand through the channel +// using the given logger +func LogOnCue(r Registry, ch chan interface{}, l Logger) { + LogScaledOnCue(r, ch, time.Nanosecond, l) +} + +// LogScaled outputs each metric in the given registry periodically using the given +// logger. Print timings in `scale` units (eg time.Millisecond) rather than nanos. +func LogScaled(r Registry, freq time.Duration, scale time.Duration, l Logger) { + ch := make(chan interface{}) + go func(channel chan interface{}) { + for _ = range time.Tick(freq) { + channel <- struct{}{} + } + }(ch) + LogScaledOnCue(r, ch, scale, l) +} + +// LogScaledOnCue outputs each metric in the given registry on demand through the channel +// using the given logger. Print timings in `scale` units (eg time.Millisecond) rather +// than nanos. +func LogScaledOnCue(r Registry, ch chan interface{}, scale time.Duration, l Logger) { + du := float64(scale) + duSuffix := scale.String()[1:] + + for _ = range ch { + r.Each(func(name string, i interface{}) { + switch metric := i.(type) { + case Counter: + l.Printf("counter %s\n", name) + l.Printf(" count: %9d\n", metric.Count()) + case Gauge: + l.Printf("gauge %s\n", name) + l.Printf(" value: %9d\n", metric.Value()) + case GaugeFloat64: + l.Printf("gauge %s\n", name) + l.Printf(" value: %f\n", metric.Value()) + case Healthcheck: + metric.Check() + l.Printf("healthcheck %s\n", name) + l.Printf(" error: %v\n", metric.Error()) + case Histogram: + h := metric.Snapshot() + ps := h.Percentiles([]float64{0.5, 0.75, 0.95, 0.99, 0.999}) + l.Printf("histogram %s\n", name) + l.Printf(" count: %9d\n", h.Count()) + l.Printf(" min: %9d\n", h.Min()) + l.Printf(" max: %9d\n", h.Max()) + l.Printf(" mean: %12.2f\n", h.Mean()) + l.Printf(" stddev: %12.2f\n", h.StdDev()) + l.Printf(" median: %12.2f\n", ps[0]) + l.Printf(" 75%%: %12.2f\n", ps[1]) + l.Printf(" 95%%: %12.2f\n", ps[2]) + l.Printf(" 99%%: %12.2f\n", ps[3]) + l.Printf(" 99.9%%: %12.2f\n", ps[4]) + case Meter: + m := metric.Snapshot() + l.Printf("meter %s\n", name) + l.Printf(" count: %9d\n", m.Count()) + l.Printf(" 1-min rate: %12.2f\n", m.Rate1()) + l.Printf(" 5-min rate: %12.2f\n", m.Rate5()) + l.Printf(" 15-min rate: %12.2f\n", m.Rate15()) + l.Printf(" mean rate: %12.2f\n", m.RateMean()) + case Timer: + t := metric.Snapshot() + ps := t.Percentiles([]float64{0.5, 0.75, 0.95, 0.99, 0.999}) + l.Printf("timer %s\n", name) + l.Printf(" count: %9d\n", t.Count()) + l.Printf(" min: %12.2f%s\n", float64(t.Min())/du, duSuffix) + l.Printf(" max: %12.2f%s\n", float64(t.Max())/du, duSuffix) + l.Printf(" mean: %12.2f%s\n", t.Mean()/du, duSuffix) + l.Printf(" stddev: %12.2f%s\n", t.StdDev()/du, duSuffix) + l.Printf(" median: %12.2f%s\n", ps[0]/du, duSuffix) + l.Printf(" 75%%: %12.2f%s\n", ps[1]/du, duSuffix) + l.Printf(" 95%%: %12.2f%s\n", ps[2]/du, duSuffix) + l.Printf(" 99%%: %12.2f%s\n", ps[3]/du, duSuffix) + l.Printf(" 99.9%%: %12.2f%s\n", ps[4]/du, duSuffix) + l.Printf(" 1-min rate: %12.2f\n", t.Rate1()) + l.Printf(" 5-min rate: %12.2f\n", t.Rate5()) + l.Printf(" 15-min rate: %12.2f\n", t.Rate15()) + l.Printf(" mean rate: %12.2f\n", t.RateMean()) + } + }) + } +} diff --git a/vendor/github.com/rcrowley/go-metrics/memory.md b/vendor/github.com/rcrowley/go-metrics/memory.md new file mode 100644 index 00000000..47454f54 --- /dev/null +++ b/vendor/github.com/rcrowley/go-metrics/memory.md @@ -0,0 +1,285 @@ +Memory usage +============ + +(Highly unscientific.) + +Command used to gather static memory usage: + +```sh +grep ^Vm "/proc/$(ps fax | grep [m]etrics-bench | awk '{print $1}')/status" +``` + +Program used to gather baseline memory usage: + +```go +package main + +import "time" + +func main() { + time.Sleep(600e9) +} +``` + +Baseline +-------- + +``` +VmPeak: 42604 kB +VmSize: 42604 kB +VmLck: 0 kB +VmHWM: 1120 kB +VmRSS: 1120 kB +VmData: 35460 kB +VmStk: 136 kB +VmExe: 1020 kB +VmLib: 1848 kB +VmPTE: 36 kB +VmSwap: 0 kB +``` + +Program used to gather metric memory usage (with other metrics being similar): + +```go +package main + +import ( + "fmt" + "metrics" + "time" +) + +func main() { + fmt.Sprintf("foo") + metrics.NewRegistry() + time.Sleep(600e9) +} +``` + +1000 counters registered +------------------------ + +``` +VmPeak: 44016 kB +VmSize: 44016 kB +VmLck: 0 kB +VmHWM: 1928 kB +VmRSS: 1928 kB +VmData: 36868 kB +VmStk: 136 kB +VmExe: 1024 kB +VmLib: 1848 kB +VmPTE: 40 kB +VmSwap: 0 kB +``` + +**1.412 kB virtual, TODO 0.808 kB resident per counter.** + +100000 counters registered +-------------------------- + +``` +VmPeak: 55024 kB +VmSize: 55024 kB +VmLck: 0 kB +VmHWM: 12440 kB +VmRSS: 12440 kB +VmData: 47876 kB +VmStk: 136 kB +VmExe: 1024 kB +VmLib: 1848 kB +VmPTE: 64 kB +VmSwap: 0 kB +``` + +**0.1242 kB virtual, 0.1132 kB resident per counter.** + +1000 gauges registered +---------------------- + +``` +VmPeak: 44012 kB +VmSize: 44012 kB +VmLck: 0 kB +VmHWM: 1928 kB +VmRSS: 1928 kB +VmData: 36868 kB +VmStk: 136 kB +VmExe: 1020 kB +VmLib: 1848 kB +VmPTE: 40 kB +VmSwap: 0 kB +``` + +**1.408 kB virtual, 0.808 kB resident per counter.** + +100000 gauges registered +------------------------ + +``` +VmPeak: 55020 kB +VmSize: 55020 kB +VmLck: 0 kB +VmHWM: 12432 kB +VmRSS: 12432 kB +VmData: 47876 kB +VmStk: 136 kB +VmExe: 1020 kB +VmLib: 1848 kB +VmPTE: 60 kB +VmSwap: 0 kB +``` + +**0.12416 kB virtual, 0.11312 resident per gauge.** + +1000 histograms with a uniform sample size of 1028 +-------------------------------------------------- + +``` +VmPeak: 72272 kB +VmSize: 72272 kB +VmLck: 0 kB +VmHWM: 16204 kB +VmRSS: 16204 kB +VmData: 65100 kB +VmStk: 136 kB +VmExe: 1048 kB +VmLib: 1848 kB +VmPTE: 80 kB +VmSwap: 0 kB +``` + +**29.668 kB virtual, TODO 15.084 resident per histogram.** + +10000 histograms with a uniform sample size of 1028 +--------------------------------------------------- + +``` +VmPeak: 256912 kB +VmSize: 256912 kB +VmLck: 0 kB +VmHWM: 146204 kB +VmRSS: 146204 kB +VmData: 249740 kB +VmStk: 136 kB +VmExe: 1048 kB +VmLib: 1848 kB +VmPTE: 448 kB +VmSwap: 0 kB +``` + +**21.4308 kB virtual, 14.5084 kB resident per histogram.** + +50000 histograms with a uniform sample size of 1028 +--------------------------------------------------- + +``` +VmPeak: 908112 kB +VmSize: 908112 kB +VmLck: 0 kB +VmHWM: 645832 kB +VmRSS: 645588 kB +VmData: 900940 kB +VmStk: 136 kB +VmExe: 1048 kB +VmLib: 1848 kB +VmPTE: 1716 kB +VmSwap: 1544 kB +``` + +**17.31016 kB virtual, 12.88936 kB resident per histogram.** + +1000 histograms with an exponentially-decaying sample size of 1028 and alpha of 0.015 +------------------------------------------------------------------------------------- + +``` +VmPeak: 62480 kB +VmSize: 62480 kB +VmLck: 0 kB +VmHWM: 11572 kB +VmRSS: 11572 kB +VmData: 55308 kB +VmStk: 136 kB +VmExe: 1048 kB +VmLib: 1848 kB +VmPTE: 64 kB +VmSwap: 0 kB +``` + +**19.876 kB virtual, 10.452 kB resident per histogram.** + +10000 histograms with an exponentially-decaying sample size of 1028 and alpha of 0.015 +-------------------------------------------------------------------------------------- + +``` +VmPeak: 153296 kB +VmSize: 153296 kB +VmLck: 0 kB +VmHWM: 101176 kB +VmRSS: 101176 kB +VmData: 146124 kB +VmStk: 136 kB +VmExe: 1048 kB +VmLib: 1848 kB +VmPTE: 240 kB +VmSwap: 0 kB +``` + +**11.0692 kB virtual, 10.0056 kB resident per histogram.** + +50000 histograms with an exponentially-decaying sample size of 1028 and alpha of 0.015 +-------------------------------------------------------------------------------------- + +``` +VmPeak: 557264 kB +VmSize: 557264 kB +VmLck: 0 kB +VmHWM: 501056 kB +VmRSS: 501056 kB +VmData: 550092 kB +VmStk: 136 kB +VmExe: 1048 kB +VmLib: 1848 kB +VmPTE: 1032 kB +VmSwap: 0 kB +``` + +**10.2932 kB virtual, 9.99872 kB resident per histogram.** + +1000 meters +----------- + +``` +VmPeak: 74504 kB +VmSize: 74504 kB +VmLck: 0 kB +VmHWM: 24124 kB +VmRSS: 24124 kB +VmData: 67340 kB +VmStk: 136 kB +VmExe: 1040 kB +VmLib: 1848 kB +VmPTE: 92 kB +VmSwap: 0 kB +``` + +**31.9 kB virtual, 23.004 kB resident per meter.** + +10000 meters +------------ + +``` +VmPeak: 278920 kB +VmSize: 278920 kB +VmLck: 0 kB +VmHWM: 227300 kB +VmRSS: 227300 kB +VmData: 271756 kB +VmStk: 136 kB +VmExe: 1040 kB +VmLib: 1848 kB +VmPTE: 488 kB +VmSwap: 0 kB +``` + +**23.6316 kB virtual, 22.618 kB resident per meter.** diff --git a/vendor/github.com/rcrowley/go-metrics/meter.go b/vendor/github.com/rcrowley/go-metrics/meter.go new file mode 100644 index 00000000..223669bc --- /dev/null +++ b/vendor/github.com/rcrowley/go-metrics/meter.go @@ -0,0 +1,251 @@ +package metrics + +import ( + "math" + "sync" + "sync/atomic" + "time" +) + +// Meters count events to produce exponentially-weighted moving average rates +// at one-, five-, and fifteen-minutes and a mean rate. +type Meter interface { + Count() int64 + Mark(int64) + Rate1() float64 + Rate5() float64 + Rate15() float64 + RateMean() float64 + Snapshot() Meter + Stop() +} + +// GetOrRegisterMeter returns an existing Meter or constructs and registers a +// new StandardMeter. +// Be sure to unregister the meter from the registry once it is of no use to +// allow for garbage collection. +func GetOrRegisterMeter(name string, r Registry) Meter { + if nil == r { + r = DefaultRegistry + } + return r.GetOrRegister(name, NewMeter).(Meter) +} + +// NewMeter constructs a new StandardMeter and launches a goroutine. +// Be sure to call Stop() once the meter is of no use to allow for garbage collection. +func NewMeter() Meter { + if UseNilMetrics { + return NilMeter{} + } + m := newStandardMeter() + arbiter.Lock() + defer arbiter.Unlock() + arbiter.meters[m] = struct{}{} + if !arbiter.started { + arbiter.started = true + go arbiter.tick() + } + return m +} + +// NewMeter constructs and registers a new StandardMeter and launches a +// goroutine. +// Be sure to unregister the meter from the registry once it is of no use to +// allow for garbage collection. +func NewRegisteredMeter(name string, r Registry) Meter { + c := NewMeter() + if nil == r { + r = DefaultRegistry + } + r.Register(name, c) + return c +} + +// MeterSnapshot is a read-only copy of another Meter. +type MeterSnapshot struct { + count int64 + rate1, rate5, rate15, rateMean uint64 +} + +// Count returns the count of events at the time the snapshot was taken. +func (m *MeterSnapshot) Count() int64 { return m.count } + +// Mark panics. +func (*MeterSnapshot) Mark(n int64) { + panic("Mark called on a MeterSnapshot") +} + +// Rate1 returns the one-minute moving average rate of events per second at the +// time the snapshot was taken. +func (m *MeterSnapshot) Rate1() float64 { return math.Float64frombits(m.rate1) } + +// Rate5 returns the five-minute moving average rate of events per second at +// the time the snapshot was taken. +func (m *MeterSnapshot) Rate5() float64 { return math.Float64frombits(m.rate5) } + +// Rate15 returns the fifteen-minute moving average rate of events per second +// at the time the snapshot was taken. +func (m *MeterSnapshot) Rate15() float64 { return math.Float64frombits(m.rate15) } + +// RateMean returns the meter's mean rate of events per second at the time the +// snapshot was taken. +func (m *MeterSnapshot) RateMean() float64 { return math.Float64frombits(m.rateMean) } + +// Snapshot returns the snapshot. +func (m *MeterSnapshot) Snapshot() Meter { return m } + +// Stop is a no-op. +func (m *MeterSnapshot) Stop() {} + +// NilMeter is a no-op Meter. +type NilMeter struct{} + +// Count is a no-op. +func (NilMeter) Count() int64 { return 0 } + +// Mark is a no-op. +func (NilMeter) Mark(n int64) {} + +// Rate1 is a no-op. +func (NilMeter) Rate1() float64 { return 0.0 } + +// Rate5 is a no-op. +func (NilMeter) Rate5() float64 { return 0.0 } + +// Rate15is a no-op. +func (NilMeter) Rate15() float64 { return 0.0 } + +// RateMean is a no-op. +func (NilMeter) RateMean() float64 { return 0.0 } + +// Snapshot is a no-op. +func (NilMeter) Snapshot() Meter { return NilMeter{} } + +// Stop is a no-op. +func (NilMeter) Stop() {} + +// StandardMeter is the standard implementation of a Meter. +type StandardMeter struct { + snapshot *MeterSnapshot + a1, a5, a15 EWMA + startTime time.Time + stopped uint32 +} + +func newStandardMeter() *StandardMeter { + return &StandardMeter{ + snapshot: &MeterSnapshot{}, + a1: NewEWMA1(), + a5: NewEWMA5(), + a15: NewEWMA15(), + startTime: time.Now(), + } +} + +// Stop stops the meter, Mark() will be a no-op if you use it after being stopped. +func (m *StandardMeter) Stop() { + if atomic.CompareAndSwapUint32(&m.stopped, 0, 1) { + arbiter.Lock() + delete(arbiter.meters, m) + arbiter.Unlock() + } +} + +// Count returns the number of events recorded. +func (m *StandardMeter) Count() int64 { + return atomic.LoadInt64(&m.snapshot.count) +} + +// Mark records the occurance of n events. +func (m *StandardMeter) Mark(n int64) { + if atomic.LoadUint32(&m.stopped) == 1 { + return + } + + atomic.AddInt64(&m.snapshot.count, n) + + m.a1.Update(n) + m.a5.Update(n) + m.a15.Update(n) + m.updateSnapshot() +} + +// Rate1 returns the one-minute moving average rate of events per second. +func (m *StandardMeter) Rate1() float64 { + return math.Float64frombits(atomic.LoadUint64(&m.snapshot.rate1)) +} + +// Rate5 returns the five-minute moving average rate of events per second. +func (m *StandardMeter) Rate5() float64 { + return math.Float64frombits(atomic.LoadUint64(&m.snapshot.rate5)) +} + +// Rate15 returns the fifteen-minute moving average rate of events per second. +func (m *StandardMeter) Rate15() float64 { + return math.Float64frombits(atomic.LoadUint64(&m.snapshot.rate15)) +} + +// RateMean returns the meter's mean rate of events per second. +func (m *StandardMeter) RateMean() float64 { + return math.Float64frombits(atomic.LoadUint64(&m.snapshot.rateMean)) +} + +// Snapshot returns a read-only copy of the meter. +func (m *StandardMeter) Snapshot() Meter { + copiedSnapshot := MeterSnapshot{ + count: atomic.LoadInt64(&m.snapshot.count), + rate1: atomic.LoadUint64(&m.snapshot.rate1), + rate5: atomic.LoadUint64(&m.snapshot.rate5), + rate15: atomic.LoadUint64(&m.snapshot.rate15), + rateMean: atomic.LoadUint64(&m.snapshot.rateMean), + } + return &copiedSnapshot +} + +func (m *StandardMeter) updateSnapshot() { + rate1 := math.Float64bits(m.a1.Rate()) + rate5 := math.Float64bits(m.a5.Rate()) + rate15 := math.Float64bits(m.a15.Rate()) + rateMean := math.Float64bits(float64(m.Count()) / time.Since(m.startTime).Seconds()) + + atomic.StoreUint64(&m.snapshot.rate1, rate1) + atomic.StoreUint64(&m.snapshot.rate5, rate5) + atomic.StoreUint64(&m.snapshot.rate15, rate15) + atomic.StoreUint64(&m.snapshot.rateMean, rateMean) +} + +func (m *StandardMeter) tick() { + m.a1.Tick() + m.a5.Tick() + m.a15.Tick() + m.updateSnapshot() +} + +// meterArbiter ticks meters every 5s from a single goroutine. +// meters are references in a set for future stopping. +type meterArbiter struct { + sync.RWMutex + started bool + meters map[*StandardMeter]struct{} + ticker *time.Ticker +} + +var arbiter = meterArbiter{ticker: time.NewTicker(5e9), meters: make(map[*StandardMeter]struct{})} + +// Ticks meters on the scheduled interval +func (ma *meterArbiter) tick() { + for { + select { + case <-ma.ticker.C: + ma.tickMeters() + } + } +} + +func (ma *meterArbiter) tickMeters() { + ma.RLock() + defer ma.RUnlock() + for meter := range ma.meters { + meter.tick() + } +} diff --git a/vendor/github.com/rcrowley/go-metrics/metrics.go b/vendor/github.com/rcrowley/go-metrics/metrics.go new file mode 100644 index 00000000..b97a49ed --- /dev/null +++ b/vendor/github.com/rcrowley/go-metrics/metrics.go @@ -0,0 +1,13 @@ +// Go port of Coda Hale's Metrics library +// +// +// +// Coda Hale's original work: +package metrics + +// UseNilMetrics is checked by the constructor functions for all of the +// standard metrics. If it is true, the metric returned is a stub. +// +// This global kill-switch helps quantify the observer effect and makes +// for less cluttered pprof profiles. +var UseNilMetrics bool = false diff --git a/vendor/github.com/rcrowley/go-metrics/opentsdb.go b/vendor/github.com/rcrowley/go-metrics/opentsdb.go new file mode 100644 index 00000000..266b6c93 --- /dev/null +++ b/vendor/github.com/rcrowley/go-metrics/opentsdb.go @@ -0,0 +1,119 @@ +package metrics + +import ( + "bufio" + "fmt" + "log" + "net" + "os" + "strings" + "time" +) + +var shortHostName string = "" + +// OpenTSDBConfig provides a container with configuration parameters for +// the OpenTSDB exporter +type OpenTSDBConfig struct { + Addr *net.TCPAddr // Network address to connect to + Registry Registry // Registry to be exported + FlushInterval time.Duration // Flush interval + DurationUnit time.Duration // Time conversion unit for durations + Prefix string // Prefix to be prepended to metric names +} + +// OpenTSDB is a blocking exporter function which reports metrics in r +// to a TSDB server located at addr, flushing them every d duration +// and prepending metric names with prefix. +func OpenTSDB(r Registry, d time.Duration, prefix string, addr *net.TCPAddr) { + OpenTSDBWithConfig(OpenTSDBConfig{ + Addr: addr, + Registry: r, + FlushInterval: d, + DurationUnit: time.Nanosecond, + Prefix: prefix, + }) +} + +// OpenTSDBWithConfig is a blocking exporter function just like OpenTSDB, +// but it takes a OpenTSDBConfig instead. +func OpenTSDBWithConfig(c OpenTSDBConfig) { + for _ = range time.Tick(c.FlushInterval) { + if err := openTSDB(&c); nil != err { + log.Println(err) + } + } +} + +func getShortHostname() string { + if shortHostName == "" { + host, _ := os.Hostname() + if index := strings.Index(host, "."); index > 0 { + shortHostName = host[:index] + } else { + shortHostName = host + } + } + return shortHostName +} + +func openTSDB(c *OpenTSDBConfig) error { + shortHostname := getShortHostname() + now := time.Now().Unix() + du := float64(c.DurationUnit) + conn, err := net.DialTCP("tcp", nil, c.Addr) + if nil != err { + return err + } + defer conn.Close() + w := bufio.NewWriter(conn) + c.Registry.Each(func(name string, i interface{}) { + switch metric := i.(type) { + case Counter: + fmt.Fprintf(w, "put %s.%s.count %d %d host=%s\n", c.Prefix, name, now, metric.Count(), shortHostname) + case Gauge: + fmt.Fprintf(w, "put %s.%s.value %d %d host=%s\n", c.Prefix, name, now, metric.Value(), shortHostname) + case GaugeFloat64: + fmt.Fprintf(w, "put %s.%s.value %d %f host=%s\n", c.Prefix, name, now, metric.Value(), shortHostname) + case Histogram: + h := metric.Snapshot() + ps := h.Percentiles([]float64{0.5, 0.75, 0.95, 0.99, 0.999}) + fmt.Fprintf(w, "put %s.%s.count %d %d host=%s\n", c.Prefix, name, now, h.Count(), shortHostname) + fmt.Fprintf(w, "put %s.%s.min %d %d host=%s\n", c.Prefix, name, now, h.Min(), shortHostname) + fmt.Fprintf(w, "put %s.%s.max %d %d host=%s\n", c.Prefix, name, now, h.Max(), shortHostname) + fmt.Fprintf(w, "put %s.%s.mean %d %.2f host=%s\n", c.Prefix, name, now, h.Mean(), shortHostname) + fmt.Fprintf(w, "put %s.%s.std-dev %d %.2f host=%s\n", c.Prefix, name, now, h.StdDev(), shortHostname) + fmt.Fprintf(w, "put %s.%s.50-percentile %d %.2f host=%s\n", c.Prefix, name, now, ps[0], shortHostname) + fmt.Fprintf(w, "put %s.%s.75-percentile %d %.2f host=%s\n", c.Prefix, name, now, ps[1], shortHostname) + fmt.Fprintf(w, "put %s.%s.95-percentile %d %.2f host=%s\n", c.Prefix, name, now, ps[2], shortHostname) + fmt.Fprintf(w, "put %s.%s.99-percentile %d %.2f host=%s\n", c.Prefix, name, now, ps[3], shortHostname) + fmt.Fprintf(w, "put %s.%s.999-percentile %d %.2f host=%s\n", c.Prefix, name, now, ps[4], shortHostname) + case Meter: + m := metric.Snapshot() + fmt.Fprintf(w, "put %s.%s.count %d %d host=%s\n", c.Prefix, name, now, m.Count(), shortHostname) + fmt.Fprintf(w, "put %s.%s.one-minute %d %.2f host=%s\n", c.Prefix, name, now, m.Rate1(), shortHostname) + fmt.Fprintf(w, "put %s.%s.five-minute %d %.2f host=%s\n", c.Prefix, name, now, m.Rate5(), shortHostname) + fmt.Fprintf(w, "put %s.%s.fifteen-minute %d %.2f host=%s\n", c.Prefix, name, now, m.Rate15(), shortHostname) + fmt.Fprintf(w, "put %s.%s.mean %d %.2f host=%s\n", c.Prefix, name, now, m.RateMean(), shortHostname) + case Timer: + t := metric.Snapshot() + ps := t.Percentiles([]float64{0.5, 0.75, 0.95, 0.99, 0.999}) + fmt.Fprintf(w, "put %s.%s.count %d %d host=%s\n", c.Prefix, name, now, t.Count(), shortHostname) + fmt.Fprintf(w, "put %s.%s.min %d %d host=%s\n", c.Prefix, name, now, t.Min()/int64(du), shortHostname) + fmt.Fprintf(w, "put %s.%s.max %d %d host=%s\n", c.Prefix, name, now, t.Max()/int64(du), shortHostname) + fmt.Fprintf(w, "put %s.%s.mean %d %.2f host=%s\n", c.Prefix, name, now, t.Mean()/du, shortHostname) + fmt.Fprintf(w, "put %s.%s.std-dev %d %.2f host=%s\n", c.Prefix, name, now, t.StdDev()/du, shortHostname) + fmt.Fprintf(w, "put %s.%s.50-percentile %d %.2f host=%s\n", c.Prefix, name, now, ps[0]/du, shortHostname) + fmt.Fprintf(w, "put %s.%s.75-percentile %d %.2f host=%s\n", c.Prefix, name, now, ps[1]/du, shortHostname) + fmt.Fprintf(w, "put %s.%s.95-percentile %d %.2f host=%s\n", c.Prefix, name, now, ps[2]/du, shortHostname) + fmt.Fprintf(w, "put %s.%s.99-percentile %d %.2f host=%s\n", c.Prefix, name, now, ps[3]/du, shortHostname) + fmt.Fprintf(w, "put %s.%s.999-percentile %d %.2f host=%s\n", c.Prefix, name, now, ps[4]/du, shortHostname) + fmt.Fprintf(w, "put %s.%s.one-minute %d %.2f host=%s\n", c.Prefix, name, now, t.Rate1(), shortHostname) + fmt.Fprintf(w, "put %s.%s.five-minute %d %.2f host=%s\n", c.Prefix, name, now, t.Rate5(), shortHostname) + fmt.Fprintf(w, "put %s.%s.fifteen-minute %d %.2f host=%s\n", c.Prefix, name, now, t.Rate15(), shortHostname) + fmt.Fprintf(w, "put %s.%s.mean-rate %d %.2f host=%s\n", c.Prefix, name, now, t.RateMean(), shortHostname) + } + w.Flush() + }) + return nil +} diff --git a/vendor/github.com/rcrowley/go-metrics/registry.go b/vendor/github.com/rcrowley/go-metrics/registry.go new file mode 100644 index 00000000..a8e67228 --- /dev/null +++ b/vendor/github.com/rcrowley/go-metrics/registry.go @@ -0,0 +1,373 @@ +package metrics + +import ( + "fmt" + "reflect" + "strings" + "sync" +) + +// DuplicateMetric is the error returned by Registry.Register when a metric +// already exists. If you mean to Register that metric you must first +// Unregister the existing metric. +type DuplicateMetric string + +func (err DuplicateMetric) Error() string { + return fmt.Sprintf("duplicate metric: %s", string(err)) +} + +// A Registry holds references to a set of metrics by name and can iterate +// over them, calling callback functions provided by the user. +// +// This is an interface so as to encourage other structs to implement +// the Registry API as appropriate. +type Registry interface { + + // Call the given function for each registered metric. + Each(func(string, interface{})) + + // Get the metric by the given name or nil if none is registered. + Get(string) interface{} + + // GetAll metrics in the Registry. + GetAll() map[string]map[string]interface{} + + // Gets an existing metric or registers the given one. + // The interface can be the metric to register if not found in registry, + // or a function returning the metric for lazy instantiation. + GetOrRegister(string, interface{}) interface{} + + // Register the given metric under the given name. + Register(string, interface{}) error + + // Run all registered healthchecks. + RunHealthchecks() + + // Unregister the metric with the given name. + Unregister(string) + + // Unregister all metrics. (Mostly for testing.) + UnregisterAll() +} + +// The standard implementation of a Registry is a mutex-protected map +// of names to metrics. +type StandardRegistry struct { + metrics map[string]interface{} + mutex sync.RWMutex +} + +// Create a new registry. +func NewRegistry() Registry { + return &StandardRegistry{metrics: make(map[string]interface{})} +} + +// Call the given function for each registered metric. +func (r *StandardRegistry) Each(f func(string, interface{})) { + metrics := r.registered() + for i := range metrics { + kv := &metrics[i] + f(kv.name, kv.value) + } +} + +// Get the metric by the given name or nil if none is registered. +func (r *StandardRegistry) Get(name string) interface{} { + r.mutex.RLock() + defer r.mutex.RUnlock() + return r.metrics[name] +} + +// Gets an existing metric or creates and registers a new one. Threadsafe +// alternative to calling Get and Register on failure. +// The interface can be the metric to register if not found in registry, +// or a function returning the metric for lazy instantiation. +func (r *StandardRegistry) GetOrRegister(name string, i interface{}) interface{} { + // access the read lock first which should be re-entrant + r.mutex.RLock() + metric, ok := r.metrics[name] + r.mutex.RUnlock() + if ok { + return metric + } + + // only take the write lock if we'll be modifying the metrics map + r.mutex.Lock() + defer r.mutex.Unlock() + if metric, ok := r.metrics[name]; ok { + return metric + } + if v := reflect.ValueOf(i); v.Kind() == reflect.Func { + i = v.Call(nil)[0].Interface() + } + r.register(name, i) + return i +} + +// Register the given metric under the given name. Returns a DuplicateMetric +// if a metric by the given name is already registered. +func (r *StandardRegistry) Register(name string, i interface{}) error { + r.mutex.Lock() + defer r.mutex.Unlock() + return r.register(name, i) +} + +// Run all registered healthchecks. +func (r *StandardRegistry) RunHealthchecks() { + r.mutex.RLock() + defer r.mutex.RUnlock() + for _, i := range r.metrics { + if h, ok := i.(Healthcheck); ok { + h.Check() + } + } +} + +// GetAll metrics in the Registry +func (r *StandardRegistry) GetAll() map[string]map[string]interface{} { + data := make(map[string]map[string]interface{}) + r.Each(func(name string, i interface{}) { + values := make(map[string]interface{}) + switch metric := i.(type) { + case Counter: + values["count"] = metric.Count() + case Gauge: + values["value"] = metric.Value() + case GaugeFloat64: + values["value"] = metric.Value() + case Healthcheck: + values["error"] = nil + metric.Check() + if err := metric.Error(); nil != err { + values["error"] = metric.Error().Error() + } + case Histogram: + h := metric.Snapshot() + ps := h.Percentiles([]float64{0.5, 0.75, 0.95, 0.99, 0.999}) + values["count"] = h.Count() + values["min"] = h.Min() + values["max"] = h.Max() + values["mean"] = h.Mean() + values["stddev"] = h.StdDev() + values["median"] = ps[0] + values["75%"] = ps[1] + values["95%"] = ps[2] + values["99%"] = ps[3] + values["99.9%"] = ps[4] + case Meter: + m := metric.Snapshot() + values["count"] = m.Count() + values["1m.rate"] = m.Rate1() + values["5m.rate"] = m.Rate5() + values["15m.rate"] = m.Rate15() + values["mean.rate"] = m.RateMean() + case Timer: + t := metric.Snapshot() + ps := t.Percentiles([]float64{0.5, 0.75, 0.95, 0.99, 0.999}) + values["count"] = t.Count() + values["min"] = t.Min() + values["max"] = t.Max() + values["mean"] = t.Mean() + values["stddev"] = t.StdDev() + values["median"] = ps[0] + values["75%"] = ps[1] + values["95%"] = ps[2] + values["99%"] = ps[3] + values["99.9%"] = ps[4] + values["1m.rate"] = t.Rate1() + values["5m.rate"] = t.Rate5() + values["15m.rate"] = t.Rate15() + values["mean.rate"] = t.RateMean() + } + data[name] = values + }) + return data +} + +// Unregister the metric with the given name. +func (r *StandardRegistry) Unregister(name string) { + r.mutex.Lock() + defer r.mutex.Unlock() + r.stop(name) + delete(r.metrics, name) +} + +// Unregister all metrics. (Mostly for testing.) +func (r *StandardRegistry) UnregisterAll() { + r.mutex.Lock() + defer r.mutex.Unlock() + for name, _ := range r.metrics { + r.stop(name) + delete(r.metrics, name) + } +} + +func (r *StandardRegistry) register(name string, i interface{}) error { + if _, ok := r.metrics[name]; ok { + return DuplicateMetric(name) + } + switch i.(type) { + case Counter, Gauge, GaugeFloat64, Healthcheck, Histogram, Meter, Timer: + r.metrics[name] = i + } + return nil +} + +type metricKV struct { + name string + value interface{} +} + +func (r *StandardRegistry) registered() []metricKV { + r.mutex.RLock() + defer r.mutex.RUnlock() + metrics := make([]metricKV, 0, len(r.metrics)) + for name, i := range r.metrics { + metrics = append(metrics, metricKV{ + name: name, + value: i, + }) + } + return metrics +} + +func (r *StandardRegistry) stop(name string) { + if i, ok := r.metrics[name]; ok { + if s, ok := i.(Stoppable); ok { + s.Stop() + } + } +} + +// Stoppable defines the metrics which has to be stopped. +type Stoppable interface { + Stop() +} + +type PrefixedRegistry struct { + underlying Registry + prefix string +} + +func NewPrefixedRegistry(prefix string) Registry { + return &PrefixedRegistry{ + underlying: NewRegistry(), + prefix: prefix, + } +} + +func NewPrefixedChildRegistry(parent Registry, prefix string) Registry { + return &PrefixedRegistry{ + underlying: parent, + prefix: prefix, + } +} + +// Call the given function for each registered metric. +func (r *PrefixedRegistry) Each(fn func(string, interface{})) { + wrappedFn := func(prefix string) func(string, interface{}) { + return func(name string, iface interface{}) { + if strings.HasPrefix(name, prefix) { + fn(name, iface) + } else { + return + } + } + } + + baseRegistry, prefix := findPrefix(r, "") + baseRegistry.Each(wrappedFn(prefix)) +} + +func findPrefix(registry Registry, prefix string) (Registry, string) { + switch r := registry.(type) { + case *PrefixedRegistry: + return findPrefix(r.underlying, r.prefix+prefix) + case *StandardRegistry: + return r, prefix + } + return nil, "" +} + +// Get the metric by the given name or nil if none is registered. +func (r *PrefixedRegistry) Get(name string) interface{} { + realName := r.prefix + name + return r.underlying.Get(realName) +} + +// Gets an existing metric or registers the given one. +// The interface can be the metric to register if not found in registry, +// or a function returning the metric for lazy instantiation. +func (r *PrefixedRegistry) GetOrRegister(name string, metric interface{}) interface{} { + realName := r.prefix + name + return r.underlying.GetOrRegister(realName, metric) +} + +// Register the given metric under the given name. The name will be prefixed. +func (r *PrefixedRegistry) Register(name string, metric interface{}) error { + realName := r.prefix + name + return r.underlying.Register(realName, metric) +} + +// Run all registered healthchecks. +func (r *PrefixedRegistry) RunHealthchecks() { + r.underlying.RunHealthchecks() +} + +// GetAll metrics in the Registry +func (r *PrefixedRegistry) GetAll() map[string]map[string]interface{} { + return r.underlying.GetAll() +} + +// Unregister the metric with the given name. The name will be prefixed. +func (r *PrefixedRegistry) Unregister(name string) { + realName := r.prefix + name + r.underlying.Unregister(realName) +} + +// Unregister all metrics. (Mostly for testing.) +func (r *PrefixedRegistry) UnregisterAll() { + r.underlying.UnregisterAll() +} + +var DefaultRegistry Registry = NewRegistry() + +// Call the given function for each registered metric. +func Each(f func(string, interface{})) { + DefaultRegistry.Each(f) +} + +// Get the metric by the given name or nil if none is registered. +func Get(name string) interface{} { + return DefaultRegistry.Get(name) +} + +// Gets an existing metric or creates and registers a new one. Threadsafe +// alternative to calling Get and Register on failure. +func GetOrRegister(name string, i interface{}) interface{} { + return DefaultRegistry.GetOrRegister(name, i) +} + +// Register the given metric under the given name. Returns a DuplicateMetric +// if a metric by the given name is already registered. +func Register(name string, i interface{}) error { + return DefaultRegistry.Register(name, i) +} + +// Register the given metric under the given name. Panics if a metric by the +// given name is already registered. +func MustRegister(name string, i interface{}) { + if err := Register(name, i); err != nil { + panic(err) + } +} + +// Run all registered healthchecks. +func RunHealthchecks() { + DefaultRegistry.RunHealthchecks() +} + +// Unregister the metric with the given name. +func Unregister(name string) { + DefaultRegistry.Unregister(name) +} diff --git a/vendor/github.com/rcrowley/go-metrics/runtime.go b/vendor/github.com/rcrowley/go-metrics/runtime.go new file mode 100644 index 00000000..4047ab3d --- /dev/null +++ b/vendor/github.com/rcrowley/go-metrics/runtime.go @@ -0,0 +1,216 @@ +package metrics + +import ( + "runtime" + "runtime/pprof" + "sync" + "time" +) + +var ( + memStats runtime.MemStats + runtimeMetrics struct { + MemStats struct { + Alloc Gauge + BuckHashSys Gauge + DebugGC Gauge + EnableGC Gauge + Frees Gauge + HeapAlloc Gauge + HeapIdle Gauge + HeapInuse Gauge + HeapObjects Gauge + HeapReleased Gauge + HeapSys Gauge + LastGC Gauge + Lookups Gauge + Mallocs Gauge + MCacheInuse Gauge + MCacheSys Gauge + MSpanInuse Gauge + MSpanSys Gauge + NextGC Gauge + NumGC Gauge + GCCPUFraction GaugeFloat64 + PauseNs Histogram + PauseTotalNs Gauge + StackInuse Gauge + StackSys Gauge + Sys Gauge + TotalAlloc Gauge + } + NumCgoCall Gauge + NumGoroutine Gauge + NumThread Gauge + ReadMemStats Timer + } + frees uint64 + lookups uint64 + mallocs uint64 + numGC uint32 + numCgoCalls int64 + + threadCreateProfile = pprof.Lookup("threadcreate") + registerRuntimeMetricsOnce = sync.Once{} +) + +// Capture new values for the Go runtime statistics exported in +// runtime.MemStats. This is designed to be called as a goroutine. +func CaptureRuntimeMemStats(r Registry, d time.Duration) { + for _ = range time.Tick(d) { + CaptureRuntimeMemStatsOnce(r) + } +} + +// Capture new values for the Go runtime statistics exported in +// runtime.MemStats. This is designed to be called in a background +// goroutine. Giving a registry which has not been given to +// RegisterRuntimeMemStats will panic. +// +// Be very careful with this because runtime.ReadMemStats calls the C +// functions runtime·semacquire(&runtime·worldsema) and runtime·stoptheworld() +// and that last one does what it says on the tin. +func CaptureRuntimeMemStatsOnce(r Registry) { + t := time.Now() + runtime.ReadMemStats(&memStats) // This takes 50-200us. + runtimeMetrics.ReadMemStats.UpdateSince(t) + + runtimeMetrics.MemStats.Alloc.Update(int64(memStats.Alloc)) + runtimeMetrics.MemStats.BuckHashSys.Update(int64(memStats.BuckHashSys)) + if memStats.DebugGC { + runtimeMetrics.MemStats.DebugGC.Update(1) + } else { + runtimeMetrics.MemStats.DebugGC.Update(0) + } + if memStats.EnableGC { + runtimeMetrics.MemStats.EnableGC.Update(1) + } else { + runtimeMetrics.MemStats.EnableGC.Update(0) + } + + runtimeMetrics.MemStats.Frees.Update(int64(memStats.Frees - frees)) + runtimeMetrics.MemStats.HeapAlloc.Update(int64(memStats.HeapAlloc)) + runtimeMetrics.MemStats.HeapIdle.Update(int64(memStats.HeapIdle)) + runtimeMetrics.MemStats.HeapInuse.Update(int64(memStats.HeapInuse)) + runtimeMetrics.MemStats.HeapObjects.Update(int64(memStats.HeapObjects)) + runtimeMetrics.MemStats.HeapReleased.Update(int64(memStats.HeapReleased)) + runtimeMetrics.MemStats.HeapSys.Update(int64(memStats.HeapSys)) + runtimeMetrics.MemStats.LastGC.Update(int64(memStats.LastGC)) + runtimeMetrics.MemStats.Lookups.Update(int64(memStats.Lookups - lookups)) + runtimeMetrics.MemStats.Mallocs.Update(int64(memStats.Mallocs - mallocs)) + runtimeMetrics.MemStats.MCacheInuse.Update(int64(memStats.MCacheInuse)) + runtimeMetrics.MemStats.MCacheSys.Update(int64(memStats.MCacheSys)) + runtimeMetrics.MemStats.MSpanInuse.Update(int64(memStats.MSpanInuse)) + runtimeMetrics.MemStats.MSpanSys.Update(int64(memStats.MSpanSys)) + runtimeMetrics.MemStats.NextGC.Update(int64(memStats.NextGC)) + runtimeMetrics.MemStats.NumGC.Update(int64(memStats.NumGC - numGC)) + runtimeMetrics.MemStats.GCCPUFraction.Update(gcCPUFraction(&memStats)) + + // + i := numGC % uint32(len(memStats.PauseNs)) + ii := memStats.NumGC % uint32(len(memStats.PauseNs)) + if memStats.NumGC-numGC >= uint32(len(memStats.PauseNs)) { + for i = 0; i < uint32(len(memStats.PauseNs)); i++ { + runtimeMetrics.MemStats.PauseNs.Update(int64(memStats.PauseNs[i])) + } + } else { + if i > ii { + for ; i < uint32(len(memStats.PauseNs)); i++ { + runtimeMetrics.MemStats.PauseNs.Update(int64(memStats.PauseNs[i])) + } + i = 0 + } + for ; i < ii; i++ { + runtimeMetrics.MemStats.PauseNs.Update(int64(memStats.PauseNs[i])) + } + } + frees = memStats.Frees + lookups = memStats.Lookups + mallocs = memStats.Mallocs + numGC = memStats.NumGC + + runtimeMetrics.MemStats.PauseTotalNs.Update(int64(memStats.PauseTotalNs)) + runtimeMetrics.MemStats.StackInuse.Update(int64(memStats.StackInuse)) + runtimeMetrics.MemStats.StackSys.Update(int64(memStats.StackSys)) + runtimeMetrics.MemStats.Sys.Update(int64(memStats.Sys)) + runtimeMetrics.MemStats.TotalAlloc.Update(int64(memStats.TotalAlloc)) + + currentNumCgoCalls := numCgoCall() + runtimeMetrics.NumCgoCall.Update(currentNumCgoCalls - numCgoCalls) + numCgoCalls = currentNumCgoCalls + + runtimeMetrics.NumGoroutine.Update(int64(runtime.NumGoroutine())) + + runtimeMetrics.NumThread.Update(int64(threadCreateProfile.Count())) +} + +// Register runtimeMetrics for the Go runtime statistics exported in runtime and +// specifically runtime.MemStats. The runtimeMetrics are named by their +// fully-qualified Go symbols, i.e. runtime.MemStats.Alloc. +func RegisterRuntimeMemStats(r Registry) { + registerRuntimeMetricsOnce.Do(func() { + runtimeMetrics.MemStats.Alloc = NewGauge() + runtimeMetrics.MemStats.BuckHashSys = NewGauge() + runtimeMetrics.MemStats.DebugGC = NewGauge() + runtimeMetrics.MemStats.EnableGC = NewGauge() + runtimeMetrics.MemStats.Frees = NewGauge() + runtimeMetrics.MemStats.HeapAlloc = NewGauge() + runtimeMetrics.MemStats.HeapIdle = NewGauge() + runtimeMetrics.MemStats.HeapInuse = NewGauge() + runtimeMetrics.MemStats.HeapObjects = NewGauge() + runtimeMetrics.MemStats.HeapReleased = NewGauge() + runtimeMetrics.MemStats.HeapSys = NewGauge() + runtimeMetrics.MemStats.LastGC = NewGauge() + runtimeMetrics.MemStats.Lookups = NewGauge() + runtimeMetrics.MemStats.Mallocs = NewGauge() + runtimeMetrics.MemStats.MCacheInuse = NewGauge() + runtimeMetrics.MemStats.MCacheSys = NewGauge() + runtimeMetrics.MemStats.MSpanInuse = NewGauge() + runtimeMetrics.MemStats.MSpanSys = NewGauge() + runtimeMetrics.MemStats.NextGC = NewGauge() + runtimeMetrics.MemStats.NumGC = NewGauge() + runtimeMetrics.MemStats.GCCPUFraction = NewGaugeFloat64() + runtimeMetrics.MemStats.PauseNs = NewHistogram(NewExpDecaySample(1028, 0.015)) + runtimeMetrics.MemStats.PauseTotalNs = NewGauge() + runtimeMetrics.MemStats.StackInuse = NewGauge() + runtimeMetrics.MemStats.StackSys = NewGauge() + runtimeMetrics.MemStats.Sys = NewGauge() + runtimeMetrics.MemStats.TotalAlloc = NewGauge() + runtimeMetrics.NumCgoCall = NewGauge() + runtimeMetrics.NumGoroutine = NewGauge() + runtimeMetrics.NumThread = NewGauge() + runtimeMetrics.ReadMemStats = NewTimer() + + r.Register("runtime.MemStats.Alloc", runtimeMetrics.MemStats.Alloc) + r.Register("runtime.MemStats.BuckHashSys", runtimeMetrics.MemStats.BuckHashSys) + r.Register("runtime.MemStats.DebugGC", runtimeMetrics.MemStats.DebugGC) + r.Register("runtime.MemStats.EnableGC", runtimeMetrics.MemStats.EnableGC) + r.Register("runtime.MemStats.Frees", runtimeMetrics.MemStats.Frees) + r.Register("runtime.MemStats.HeapAlloc", runtimeMetrics.MemStats.HeapAlloc) + r.Register("runtime.MemStats.HeapIdle", runtimeMetrics.MemStats.HeapIdle) + r.Register("runtime.MemStats.HeapInuse", runtimeMetrics.MemStats.HeapInuse) + r.Register("runtime.MemStats.HeapObjects", runtimeMetrics.MemStats.HeapObjects) + r.Register("runtime.MemStats.HeapReleased", runtimeMetrics.MemStats.HeapReleased) + r.Register("runtime.MemStats.HeapSys", runtimeMetrics.MemStats.HeapSys) + r.Register("runtime.MemStats.LastGC", runtimeMetrics.MemStats.LastGC) + r.Register("runtime.MemStats.Lookups", runtimeMetrics.MemStats.Lookups) + r.Register("runtime.MemStats.Mallocs", runtimeMetrics.MemStats.Mallocs) + r.Register("runtime.MemStats.MCacheInuse", runtimeMetrics.MemStats.MCacheInuse) + r.Register("runtime.MemStats.MCacheSys", runtimeMetrics.MemStats.MCacheSys) + r.Register("runtime.MemStats.MSpanInuse", runtimeMetrics.MemStats.MSpanInuse) + r.Register("runtime.MemStats.MSpanSys", runtimeMetrics.MemStats.MSpanSys) + r.Register("runtime.MemStats.NextGC", runtimeMetrics.MemStats.NextGC) + r.Register("runtime.MemStats.NumGC", runtimeMetrics.MemStats.NumGC) + r.Register("runtime.MemStats.GCCPUFraction", runtimeMetrics.MemStats.GCCPUFraction) + r.Register("runtime.MemStats.PauseNs", runtimeMetrics.MemStats.PauseNs) + r.Register("runtime.MemStats.PauseTotalNs", runtimeMetrics.MemStats.PauseTotalNs) + r.Register("runtime.MemStats.StackInuse", runtimeMetrics.MemStats.StackInuse) + r.Register("runtime.MemStats.StackSys", runtimeMetrics.MemStats.StackSys) + r.Register("runtime.MemStats.Sys", runtimeMetrics.MemStats.Sys) + r.Register("runtime.MemStats.TotalAlloc", runtimeMetrics.MemStats.TotalAlloc) + r.Register("runtime.NumCgoCall", runtimeMetrics.NumCgoCall) + r.Register("runtime.NumGoroutine", runtimeMetrics.NumGoroutine) + r.Register("runtime.NumThread", runtimeMetrics.NumThread) + r.Register("runtime.ReadMemStats", runtimeMetrics.ReadMemStats) + }) +} diff --git a/vendor/github.com/rcrowley/go-metrics/runtime_cgo.go b/vendor/github.com/rcrowley/go-metrics/runtime_cgo.go new file mode 100644 index 00000000..e3391f4e --- /dev/null +++ b/vendor/github.com/rcrowley/go-metrics/runtime_cgo.go @@ -0,0 +1,10 @@ +// +build cgo +// +build !appengine + +package metrics + +import "runtime" + +func numCgoCall() int64 { + return runtime.NumCgoCall() +} diff --git a/vendor/github.com/rcrowley/go-metrics/runtime_gccpufraction.go b/vendor/github.com/rcrowley/go-metrics/runtime_gccpufraction.go new file mode 100644 index 00000000..ca12c05b --- /dev/null +++ b/vendor/github.com/rcrowley/go-metrics/runtime_gccpufraction.go @@ -0,0 +1,9 @@ +// +build go1.5 + +package metrics + +import "runtime" + +func gcCPUFraction(memStats *runtime.MemStats) float64 { + return memStats.GCCPUFraction +} diff --git a/vendor/github.com/rcrowley/go-metrics/runtime_no_cgo.go b/vendor/github.com/rcrowley/go-metrics/runtime_no_cgo.go new file mode 100644 index 00000000..616a3b47 --- /dev/null +++ b/vendor/github.com/rcrowley/go-metrics/runtime_no_cgo.go @@ -0,0 +1,7 @@ +// +build !cgo appengine + +package metrics + +func numCgoCall() int64 { + return 0 +} diff --git a/vendor/github.com/rcrowley/go-metrics/runtime_no_gccpufraction.go b/vendor/github.com/rcrowley/go-metrics/runtime_no_gccpufraction.go new file mode 100644 index 00000000..be96aa6f --- /dev/null +++ b/vendor/github.com/rcrowley/go-metrics/runtime_no_gccpufraction.go @@ -0,0 +1,9 @@ +// +build !go1.5 + +package metrics + +import "runtime" + +func gcCPUFraction(memStats *runtime.MemStats) float64 { + return 0 +} diff --git a/vendor/github.com/rcrowley/go-metrics/sample.go b/vendor/github.com/rcrowley/go-metrics/sample.go new file mode 100644 index 00000000..fecee5ef --- /dev/null +++ b/vendor/github.com/rcrowley/go-metrics/sample.go @@ -0,0 +1,616 @@ +package metrics + +import ( + "math" + "math/rand" + "sort" + "sync" + "time" +) + +const rescaleThreshold = time.Hour + +// Samples maintain a statistically-significant selection of values from +// a stream. +type Sample interface { + Clear() + Count() int64 + Max() int64 + Mean() float64 + Min() int64 + Percentile(float64) float64 + Percentiles([]float64) []float64 + Size() int + Snapshot() Sample + StdDev() float64 + Sum() int64 + Update(int64) + Values() []int64 + Variance() float64 +} + +// ExpDecaySample is an exponentially-decaying sample using a forward-decaying +// priority reservoir. See Cormode et al's "Forward Decay: A Practical Time +// Decay Model for Streaming Systems". +// +// +type ExpDecaySample struct { + alpha float64 + count int64 + mutex sync.Mutex + reservoirSize int + t0, t1 time.Time + values *expDecaySampleHeap +} + +// NewExpDecaySample constructs a new exponentially-decaying sample with the +// given reservoir size and alpha. +func NewExpDecaySample(reservoirSize int, alpha float64) Sample { + if UseNilMetrics { + return NilSample{} + } + s := &ExpDecaySample{ + alpha: alpha, + reservoirSize: reservoirSize, + t0: time.Now(), + values: newExpDecaySampleHeap(reservoirSize), + } + s.t1 = s.t0.Add(rescaleThreshold) + return s +} + +// Clear clears all samples. +func (s *ExpDecaySample) Clear() { + s.mutex.Lock() + defer s.mutex.Unlock() + s.count = 0 + s.t0 = time.Now() + s.t1 = s.t0.Add(rescaleThreshold) + s.values.Clear() +} + +// Count returns the number of samples recorded, which may exceed the +// reservoir size. +func (s *ExpDecaySample) Count() int64 { + s.mutex.Lock() + defer s.mutex.Unlock() + return s.count +} + +// Max returns the maximum value in the sample, which may not be the maximum +// value ever to be part of the sample. +func (s *ExpDecaySample) Max() int64 { + return SampleMax(s.Values()) +} + +// Mean returns the mean of the values in the sample. +func (s *ExpDecaySample) Mean() float64 { + return SampleMean(s.Values()) +} + +// Min returns the minimum value in the sample, which may not be the minimum +// value ever to be part of the sample. +func (s *ExpDecaySample) Min() int64 { + return SampleMin(s.Values()) +} + +// Percentile returns an arbitrary percentile of values in the sample. +func (s *ExpDecaySample) Percentile(p float64) float64 { + return SamplePercentile(s.Values(), p) +} + +// Percentiles returns a slice of arbitrary percentiles of values in the +// sample. +func (s *ExpDecaySample) Percentiles(ps []float64) []float64 { + return SamplePercentiles(s.Values(), ps) +} + +// Size returns the size of the sample, which is at most the reservoir size. +func (s *ExpDecaySample) Size() int { + s.mutex.Lock() + defer s.mutex.Unlock() + return s.values.Size() +} + +// Snapshot returns a read-only copy of the sample. +func (s *ExpDecaySample) Snapshot() Sample { + s.mutex.Lock() + defer s.mutex.Unlock() + vals := s.values.Values() + values := make([]int64, len(vals)) + for i, v := range vals { + values[i] = v.v + } + return &SampleSnapshot{ + count: s.count, + values: values, + } +} + +// StdDev returns the standard deviation of the values in the sample. +func (s *ExpDecaySample) StdDev() float64 { + return SampleStdDev(s.Values()) +} + +// Sum returns the sum of the values in the sample. +func (s *ExpDecaySample) Sum() int64 { + return SampleSum(s.Values()) +} + +// Update samples a new value. +func (s *ExpDecaySample) Update(v int64) { + s.update(time.Now(), v) +} + +// Values returns a copy of the values in the sample. +func (s *ExpDecaySample) Values() []int64 { + s.mutex.Lock() + defer s.mutex.Unlock() + vals := s.values.Values() + values := make([]int64, len(vals)) + for i, v := range vals { + values[i] = v.v + } + return values +} + +// Variance returns the variance of the values in the sample. +func (s *ExpDecaySample) Variance() float64 { + return SampleVariance(s.Values()) +} + +// update samples a new value at a particular timestamp. This is a method all +// its own to facilitate testing. +func (s *ExpDecaySample) update(t time.Time, v int64) { + s.mutex.Lock() + defer s.mutex.Unlock() + s.count++ + if s.values.Size() == s.reservoirSize { + s.values.Pop() + } + s.values.Push(expDecaySample{ + k: math.Exp(t.Sub(s.t0).Seconds()*s.alpha) / rand.Float64(), + v: v, + }) + if t.After(s.t1) { + values := s.values.Values() + t0 := s.t0 + s.values.Clear() + s.t0 = t + s.t1 = s.t0.Add(rescaleThreshold) + for _, v := range values { + v.k = v.k * math.Exp(-s.alpha*s.t0.Sub(t0).Seconds()) + s.values.Push(v) + } + } +} + +// NilSample is a no-op Sample. +type NilSample struct{} + +// Clear is a no-op. +func (NilSample) Clear() {} + +// Count is a no-op. +func (NilSample) Count() int64 { return 0 } + +// Max is a no-op. +func (NilSample) Max() int64 { return 0 } + +// Mean is a no-op. +func (NilSample) Mean() float64 { return 0.0 } + +// Min is a no-op. +func (NilSample) Min() int64 { return 0 } + +// Percentile is a no-op. +func (NilSample) Percentile(p float64) float64 { return 0.0 } + +// Percentiles is a no-op. +func (NilSample) Percentiles(ps []float64) []float64 { + return make([]float64, len(ps)) +} + +// Size is a no-op. +func (NilSample) Size() int { return 0 } + +// Sample is a no-op. +func (NilSample) Snapshot() Sample { return NilSample{} } + +// StdDev is a no-op. +func (NilSample) StdDev() float64 { return 0.0 } + +// Sum is a no-op. +func (NilSample) Sum() int64 { return 0 } + +// Update is a no-op. +func (NilSample) Update(v int64) {} + +// Values is a no-op. +func (NilSample) Values() []int64 { return []int64{} } + +// Variance is a no-op. +func (NilSample) Variance() float64 { return 0.0 } + +// SampleMax returns the maximum value of the slice of int64. +func SampleMax(values []int64) int64 { + if 0 == len(values) { + return 0 + } + var max int64 = math.MinInt64 + for _, v := range values { + if max < v { + max = v + } + } + return max +} + +// SampleMean returns the mean value of the slice of int64. +func SampleMean(values []int64) float64 { + if 0 == len(values) { + return 0.0 + } + return float64(SampleSum(values)) / float64(len(values)) +} + +// SampleMin returns the minimum value of the slice of int64. +func SampleMin(values []int64) int64 { + if 0 == len(values) { + return 0 + } + var min int64 = math.MaxInt64 + for _, v := range values { + if min > v { + min = v + } + } + return min +} + +// SamplePercentiles returns an arbitrary percentile of the slice of int64. +func SamplePercentile(values int64Slice, p float64) float64 { + return SamplePercentiles(values, []float64{p})[0] +} + +// SamplePercentiles returns a slice of arbitrary percentiles of the slice of +// int64. +func SamplePercentiles(values int64Slice, ps []float64) []float64 { + scores := make([]float64, len(ps)) + size := len(values) + if size > 0 { + sort.Sort(values) + for i, p := range ps { + pos := p * float64(size+1) + if pos < 1.0 { + scores[i] = float64(values[0]) + } else if pos >= float64(size) { + scores[i] = float64(values[size-1]) + } else { + lower := float64(values[int(pos)-1]) + upper := float64(values[int(pos)]) + scores[i] = lower + (pos-math.Floor(pos))*(upper-lower) + } + } + } + return scores +} + +// SampleSnapshot is a read-only copy of another Sample. +type SampleSnapshot struct { + count int64 + values []int64 +} + +func NewSampleSnapshot(count int64, values []int64) *SampleSnapshot { + return &SampleSnapshot{ + count: count, + values: values, + } +} + +// Clear panics. +func (*SampleSnapshot) Clear() { + panic("Clear called on a SampleSnapshot") +} + +// Count returns the count of inputs at the time the snapshot was taken. +func (s *SampleSnapshot) Count() int64 { return s.count } + +// Max returns the maximal value at the time the snapshot was taken. +func (s *SampleSnapshot) Max() int64 { return SampleMax(s.values) } + +// Mean returns the mean value at the time the snapshot was taken. +func (s *SampleSnapshot) Mean() float64 { return SampleMean(s.values) } + +// Min returns the minimal value at the time the snapshot was taken. +func (s *SampleSnapshot) Min() int64 { return SampleMin(s.values) } + +// Percentile returns an arbitrary percentile of values at the time the +// snapshot was taken. +func (s *SampleSnapshot) Percentile(p float64) float64 { + return SamplePercentile(s.values, p) +} + +// Percentiles returns a slice of arbitrary percentiles of values at the time +// the snapshot was taken. +func (s *SampleSnapshot) Percentiles(ps []float64) []float64 { + return SamplePercentiles(s.values, ps) +} + +// Size returns the size of the sample at the time the snapshot was taken. +func (s *SampleSnapshot) Size() int { return len(s.values) } + +// Snapshot returns the snapshot. +func (s *SampleSnapshot) Snapshot() Sample { return s } + +// StdDev returns the standard deviation of values at the time the snapshot was +// taken. +func (s *SampleSnapshot) StdDev() float64 { return SampleStdDev(s.values) } + +// Sum returns the sum of values at the time the snapshot was taken. +func (s *SampleSnapshot) Sum() int64 { return SampleSum(s.values) } + +// Update panics. +func (*SampleSnapshot) Update(int64) { + panic("Update called on a SampleSnapshot") +} + +// Values returns a copy of the values in the sample. +func (s *SampleSnapshot) Values() []int64 { + values := make([]int64, len(s.values)) + copy(values, s.values) + return values +} + +// Variance returns the variance of values at the time the snapshot was taken. +func (s *SampleSnapshot) Variance() float64 { return SampleVariance(s.values) } + +// SampleStdDev returns the standard deviation of the slice of int64. +func SampleStdDev(values []int64) float64 { + return math.Sqrt(SampleVariance(values)) +} + +// SampleSum returns the sum of the slice of int64. +func SampleSum(values []int64) int64 { + var sum int64 + for _, v := range values { + sum += v + } + return sum +} + +// SampleVariance returns the variance of the slice of int64. +func SampleVariance(values []int64) float64 { + if 0 == len(values) { + return 0.0 + } + m := SampleMean(values) + var sum float64 + for _, v := range values { + d := float64(v) - m + sum += d * d + } + return sum / float64(len(values)) +} + +// A uniform sample using Vitter's Algorithm R. +// +// +type UniformSample struct { + count int64 + mutex sync.Mutex + reservoirSize int + values []int64 +} + +// NewUniformSample constructs a new uniform sample with the given reservoir +// size. +func NewUniformSample(reservoirSize int) Sample { + if UseNilMetrics { + return NilSample{} + } + return &UniformSample{ + reservoirSize: reservoirSize, + values: make([]int64, 0, reservoirSize), + } +} + +// Clear clears all samples. +func (s *UniformSample) Clear() { + s.mutex.Lock() + defer s.mutex.Unlock() + s.count = 0 + s.values = make([]int64, 0, s.reservoirSize) +} + +// Count returns the number of samples recorded, which may exceed the +// reservoir size. +func (s *UniformSample) Count() int64 { + s.mutex.Lock() + defer s.mutex.Unlock() + return s.count +} + +// Max returns the maximum value in the sample, which may not be the maximum +// value ever to be part of the sample. +func (s *UniformSample) Max() int64 { + s.mutex.Lock() + defer s.mutex.Unlock() + return SampleMax(s.values) +} + +// Mean returns the mean of the values in the sample. +func (s *UniformSample) Mean() float64 { + s.mutex.Lock() + defer s.mutex.Unlock() + return SampleMean(s.values) +} + +// Min returns the minimum value in the sample, which may not be the minimum +// value ever to be part of the sample. +func (s *UniformSample) Min() int64 { + s.mutex.Lock() + defer s.mutex.Unlock() + return SampleMin(s.values) +} + +// Percentile returns an arbitrary percentile of values in the sample. +func (s *UniformSample) Percentile(p float64) float64 { + s.mutex.Lock() + defer s.mutex.Unlock() + return SamplePercentile(s.values, p) +} + +// Percentiles returns a slice of arbitrary percentiles of values in the +// sample. +func (s *UniformSample) Percentiles(ps []float64) []float64 { + s.mutex.Lock() + defer s.mutex.Unlock() + return SamplePercentiles(s.values, ps) +} + +// Size returns the size of the sample, which is at most the reservoir size. +func (s *UniformSample) Size() int { + s.mutex.Lock() + defer s.mutex.Unlock() + return len(s.values) +} + +// Snapshot returns a read-only copy of the sample. +func (s *UniformSample) Snapshot() Sample { + s.mutex.Lock() + defer s.mutex.Unlock() + values := make([]int64, len(s.values)) + copy(values, s.values) + return &SampleSnapshot{ + count: s.count, + values: values, + } +} + +// StdDev returns the standard deviation of the values in the sample. +func (s *UniformSample) StdDev() float64 { + s.mutex.Lock() + defer s.mutex.Unlock() + return SampleStdDev(s.values) +} + +// Sum returns the sum of the values in the sample. +func (s *UniformSample) Sum() int64 { + s.mutex.Lock() + defer s.mutex.Unlock() + return SampleSum(s.values) +} + +// Update samples a new value. +func (s *UniformSample) Update(v int64) { + s.mutex.Lock() + defer s.mutex.Unlock() + s.count++ + if len(s.values) < s.reservoirSize { + s.values = append(s.values, v) + } else { + r := rand.Int63n(s.count) + if r < int64(len(s.values)) { + s.values[int(r)] = v + } + } +} + +// Values returns a copy of the values in the sample. +func (s *UniformSample) Values() []int64 { + s.mutex.Lock() + defer s.mutex.Unlock() + values := make([]int64, len(s.values)) + copy(values, s.values) + return values +} + +// Variance returns the variance of the values in the sample. +func (s *UniformSample) Variance() float64 { + s.mutex.Lock() + defer s.mutex.Unlock() + return SampleVariance(s.values) +} + +// expDecaySample represents an individual sample in a heap. +type expDecaySample struct { + k float64 + v int64 +} + +func newExpDecaySampleHeap(reservoirSize int) *expDecaySampleHeap { + return &expDecaySampleHeap{make([]expDecaySample, 0, reservoirSize)} +} + +// expDecaySampleHeap is a min-heap of expDecaySamples. +// The internal implementation is copied from the standard library's container/heap +type expDecaySampleHeap struct { + s []expDecaySample +} + +func (h *expDecaySampleHeap) Clear() { + h.s = h.s[:0] +} + +func (h *expDecaySampleHeap) Push(s expDecaySample) { + n := len(h.s) + h.s = h.s[0 : n+1] + h.s[n] = s + h.up(n) +} + +func (h *expDecaySampleHeap) Pop() expDecaySample { + n := len(h.s) - 1 + h.s[0], h.s[n] = h.s[n], h.s[0] + h.down(0, n) + + n = len(h.s) + s := h.s[n-1] + h.s = h.s[0 : n-1] + return s +} + +func (h *expDecaySampleHeap) Size() int { + return len(h.s) +} + +func (h *expDecaySampleHeap) Values() []expDecaySample { + return h.s +} + +func (h *expDecaySampleHeap) up(j int) { + for { + i := (j - 1) / 2 // parent + if i == j || !(h.s[j].k < h.s[i].k) { + break + } + h.s[i], h.s[j] = h.s[j], h.s[i] + j = i + } +} + +func (h *expDecaySampleHeap) down(i, n int) { + for { + j1 := 2*i + 1 + if j1 >= n || j1 < 0 { // j1 < 0 after int overflow + break + } + j := j1 // left child + if j2 := j1 + 1; j2 < n && !(h.s[j1].k < h.s[j2].k) { + j = j2 // = 2*i + 2 // right child + } + if !(h.s[j].k < h.s[i].k) { + break + } + h.s[i], h.s[j] = h.s[j], h.s[i] + i = j + } +} + +type int64Slice []int64 + +func (p int64Slice) Len() int { return len(p) } +func (p int64Slice) Less(i, j int) bool { return p[i] < p[j] } +func (p int64Slice) Swap(i, j int) { p[i], p[j] = p[j], p[i] } diff --git a/vendor/github.com/rcrowley/go-metrics/syslog.go b/vendor/github.com/rcrowley/go-metrics/syslog.go new file mode 100644 index 00000000..693f1908 --- /dev/null +++ b/vendor/github.com/rcrowley/go-metrics/syslog.go @@ -0,0 +1,78 @@ +// +build !windows + +package metrics + +import ( + "fmt" + "log/syslog" + "time" +) + +// Output each metric in the given registry to syslog periodically using +// the given syslogger. +func Syslog(r Registry, d time.Duration, w *syslog.Writer) { + for _ = range time.Tick(d) { + r.Each(func(name string, i interface{}) { + switch metric := i.(type) { + case Counter: + w.Info(fmt.Sprintf("counter %s: count: %d", name, metric.Count())) + case Gauge: + w.Info(fmt.Sprintf("gauge %s: value: %d", name, metric.Value())) + case GaugeFloat64: + w.Info(fmt.Sprintf("gauge %s: value: %f", name, metric.Value())) + case Healthcheck: + metric.Check() + w.Info(fmt.Sprintf("healthcheck %s: error: %v", name, metric.Error())) + case Histogram: + h := metric.Snapshot() + ps := h.Percentiles([]float64{0.5, 0.75, 0.95, 0.99, 0.999}) + w.Info(fmt.Sprintf( + "histogram %s: count: %d min: %d max: %d mean: %.2f stddev: %.2f median: %.2f 75%%: %.2f 95%%: %.2f 99%%: %.2f 99.9%%: %.2f", + name, + h.Count(), + h.Min(), + h.Max(), + h.Mean(), + h.StdDev(), + ps[0], + ps[1], + ps[2], + ps[3], + ps[4], + )) + case Meter: + m := metric.Snapshot() + w.Info(fmt.Sprintf( + "meter %s: count: %d 1-min: %.2f 5-min: %.2f 15-min: %.2f mean: %.2f", + name, + m.Count(), + m.Rate1(), + m.Rate5(), + m.Rate15(), + m.RateMean(), + )) + case Timer: + t := metric.Snapshot() + ps := t.Percentiles([]float64{0.5, 0.75, 0.95, 0.99, 0.999}) + w.Info(fmt.Sprintf( + "timer %s: count: %d min: %d max: %d mean: %.2f stddev: %.2f median: %.2f 75%%: %.2f 95%%: %.2f 99%%: %.2f 99.9%%: %.2f 1-min: %.2f 5-min: %.2f 15-min: %.2f mean-rate: %.2f", + name, + t.Count(), + t.Min(), + t.Max(), + t.Mean(), + t.StdDev(), + ps[0], + ps[1], + ps[2], + ps[3], + ps[4], + t.Rate1(), + t.Rate5(), + t.Rate15(), + t.RateMean(), + )) + } + }) + } +} diff --git a/vendor/github.com/rcrowley/go-metrics/timer.go b/vendor/github.com/rcrowley/go-metrics/timer.go new file mode 100644 index 00000000..d6ec4c62 --- /dev/null +++ b/vendor/github.com/rcrowley/go-metrics/timer.go @@ -0,0 +1,329 @@ +package metrics + +import ( + "sync" + "time" +) + +// Timers capture the duration and rate of events. +type Timer interface { + Count() int64 + Max() int64 + Mean() float64 + Min() int64 + Percentile(float64) float64 + Percentiles([]float64) []float64 + Rate1() float64 + Rate5() float64 + Rate15() float64 + RateMean() float64 + Snapshot() Timer + StdDev() float64 + Stop() + Sum() int64 + Time(func()) + Update(time.Duration) + UpdateSince(time.Time) + Variance() float64 +} + +// GetOrRegisterTimer returns an existing Timer or constructs and registers a +// new StandardTimer. +// Be sure to unregister the meter from the registry once it is of no use to +// allow for garbage collection. +func GetOrRegisterTimer(name string, r Registry) Timer { + if nil == r { + r = DefaultRegistry + } + return r.GetOrRegister(name, NewTimer).(Timer) +} + +// NewCustomTimer constructs a new StandardTimer from a Histogram and a Meter. +// Be sure to call Stop() once the timer is of no use to allow for garbage collection. +func NewCustomTimer(h Histogram, m Meter) Timer { + if UseNilMetrics { + return NilTimer{} + } + return &StandardTimer{ + histogram: h, + meter: m, + } +} + +// NewRegisteredTimer constructs and registers a new StandardTimer. +// Be sure to unregister the meter from the registry once it is of no use to +// allow for garbage collection. +func NewRegisteredTimer(name string, r Registry) Timer { + c := NewTimer() + if nil == r { + r = DefaultRegistry + } + r.Register(name, c) + return c +} + +// NewTimer constructs a new StandardTimer using an exponentially-decaying +// sample with the same reservoir size and alpha as UNIX load averages. +// Be sure to call Stop() once the timer is of no use to allow for garbage collection. +func NewTimer() Timer { + if UseNilMetrics { + return NilTimer{} + } + return &StandardTimer{ + histogram: NewHistogram(NewExpDecaySample(1028, 0.015)), + meter: NewMeter(), + } +} + +// NilTimer is a no-op Timer. +type NilTimer struct { + h Histogram + m Meter +} + +// Count is a no-op. +func (NilTimer) Count() int64 { return 0 } + +// Max is a no-op. +func (NilTimer) Max() int64 { return 0 } + +// Mean is a no-op. +func (NilTimer) Mean() float64 { return 0.0 } + +// Min is a no-op. +func (NilTimer) Min() int64 { return 0 } + +// Percentile is a no-op. +func (NilTimer) Percentile(p float64) float64 { return 0.0 } + +// Percentiles is a no-op. +func (NilTimer) Percentiles(ps []float64) []float64 { + return make([]float64, len(ps)) +} + +// Rate1 is a no-op. +func (NilTimer) Rate1() float64 { return 0.0 } + +// Rate5 is a no-op. +func (NilTimer) Rate5() float64 { return 0.0 } + +// Rate15 is a no-op. +func (NilTimer) Rate15() float64 { return 0.0 } + +// RateMean is a no-op. +func (NilTimer) RateMean() float64 { return 0.0 } + +// Snapshot is a no-op. +func (NilTimer) Snapshot() Timer { return NilTimer{} } + +// StdDev is a no-op. +func (NilTimer) StdDev() float64 { return 0.0 } + +// Stop is a no-op. +func (NilTimer) Stop() {} + +// Sum is a no-op. +func (NilTimer) Sum() int64 { return 0 } + +// Time is a no-op. +func (NilTimer) Time(func()) {} + +// Update is a no-op. +func (NilTimer) Update(time.Duration) {} + +// UpdateSince is a no-op. +func (NilTimer) UpdateSince(time.Time) {} + +// Variance is a no-op. +func (NilTimer) Variance() float64 { return 0.0 } + +// StandardTimer is the standard implementation of a Timer and uses a Histogram +// and Meter. +type StandardTimer struct { + histogram Histogram + meter Meter + mutex sync.Mutex +} + +// Count returns the number of events recorded. +func (t *StandardTimer) Count() int64 { + return t.histogram.Count() +} + +// Max returns the maximum value in the sample. +func (t *StandardTimer) Max() int64 { + return t.histogram.Max() +} + +// Mean returns the mean of the values in the sample. +func (t *StandardTimer) Mean() float64 { + return t.histogram.Mean() +} + +// Min returns the minimum value in the sample. +func (t *StandardTimer) Min() int64 { + return t.histogram.Min() +} + +// Percentile returns an arbitrary percentile of the values in the sample. +func (t *StandardTimer) Percentile(p float64) float64 { + return t.histogram.Percentile(p) +} + +// Percentiles returns a slice of arbitrary percentiles of the values in the +// sample. +func (t *StandardTimer) Percentiles(ps []float64) []float64 { + return t.histogram.Percentiles(ps) +} + +// Rate1 returns the one-minute moving average rate of events per second. +func (t *StandardTimer) Rate1() float64 { + return t.meter.Rate1() +} + +// Rate5 returns the five-minute moving average rate of events per second. +func (t *StandardTimer) Rate5() float64 { + return t.meter.Rate5() +} + +// Rate15 returns the fifteen-minute moving average rate of events per second. +func (t *StandardTimer) Rate15() float64 { + return t.meter.Rate15() +} + +// RateMean returns the meter's mean rate of events per second. +func (t *StandardTimer) RateMean() float64 { + return t.meter.RateMean() +} + +// Snapshot returns a read-only copy of the timer. +func (t *StandardTimer) Snapshot() Timer { + t.mutex.Lock() + defer t.mutex.Unlock() + return &TimerSnapshot{ + histogram: t.histogram.Snapshot().(*HistogramSnapshot), + meter: t.meter.Snapshot().(*MeterSnapshot), + } +} + +// StdDev returns the standard deviation of the values in the sample. +func (t *StandardTimer) StdDev() float64 { + return t.histogram.StdDev() +} + +// Stop stops the meter. +func (t *StandardTimer) Stop() { + t.meter.Stop() +} + +// Sum returns the sum in the sample. +func (t *StandardTimer) Sum() int64 { + return t.histogram.Sum() +} + +// Record the duration of the execution of the given function. +func (t *StandardTimer) Time(f func()) { + ts := time.Now() + f() + t.Update(time.Since(ts)) +} + +// Record the duration of an event. +func (t *StandardTimer) Update(d time.Duration) { + t.mutex.Lock() + defer t.mutex.Unlock() + t.histogram.Update(int64(d)) + t.meter.Mark(1) +} + +// Record the duration of an event that started at a time and ends now. +func (t *StandardTimer) UpdateSince(ts time.Time) { + t.mutex.Lock() + defer t.mutex.Unlock() + t.histogram.Update(int64(time.Since(ts))) + t.meter.Mark(1) +} + +// Variance returns the variance of the values in the sample. +func (t *StandardTimer) Variance() float64 { + return t.histogram.Variance() +} + +// TimerSnapshot is a read-only copy of another Timer. +type TimerSnapshot struct { + histogram *HistogramSnapshot + meter *MeterSnapshot +} + +// Count returns the number of events recorded at the time the snapshot was +// taken. +func (t *TimerSnapshot) Count() int64 { return t.histogram.Count() } + +// Max returns the maximum value at the time the snapshot was taken. +func (t *TimerSnapshot) Max() int64 { return t.histogram.Max() } + +// Mean returns the mean value at the time the snapshot was taken. +func (t *TimerSnapshot) Mean() float64 { return t.histogram.Mean() } + +// Min returns the minimum value at the time the snapshot was taken. +func (t *TimerSnapshot) Min() int64 { return t.histogram.Min() } + +// Percentile returns an arbitrary percentile of sampled values at the time the +// snapshot was taken. +func (t *TimerSnapshot) Percentile(p float64) float64 { + return t.histogram.Percentile(p) +} + +// Percentiles returns a slice of arbitrary percentiles of sampled values at +// the time the snapshot was taken. +func (t *TimerSnapshot) Percentiles(ps []float64) []float64 { + return t.histogram.Percentiles(ps) +} + +// Rate1 returns the one-minute moving average rate of events per second at the +// time the snapshot was taken. +func (t *TimerSnapshot) Rate1() float64 { return t.meter.Rate1() } + +// Rate5 returns the five-minute moving average rate of events per second at +// the time the snapshot was taken. +func (t *TimerSnapshot) Rate5() float64 { return t.meter.Rate5() } + +// Rate15 returns the fifteen-minute moving average rate of events per second +// at the time the snapshot was taken. +func (t *TimerSnapshot) Rate15() float64 { return t.meter.Rate15() } + +// RateMean returns the meter's mean rate of events per second at the time the +// snapshot was taken. +func (t *TimerSnapshot) RateMean() float64 { return t.meter.RateMean() } + +// Snapshot returns the snapshot. +func (t *TimerSnapshot) Snapshot() Timer { return t } + +// StdDev returns the standard deviation of the values at the time the snapshot +// was taken. +func (t *TimerSnapshot) StdDev() float64 { return t.histogram.StdDev() } + +// Stop is a no-op. +func (t *TimerSnapshot) Stop() {} + +// Sum returns the sum at the time the snapshot was taken. +func (t *TimerSnapshot) Sum() int64 { return t.histogram.Sum() } + +// Time panics. +func (*TimerSnapshot) Time(func()) { + panic("Time called on a TimerSnapshot") +} + +// Update panics. +func (*TimerSnapshot) Update(time.Duration) { + panic("Update called on a TimerSnapshot") +} + +// UpdateSince panics. +func (*TimerSnapshot) UpdateSince(time.Time) { + panic("UpdateSince called on a TimerSnapshot") +} + +// Variance returns the variance of the values at the time the snapshot was +// taken. +func (t *TimerSnapshot) Variance() float64 { return t.histogram.Variance() } diff --git a/vendor/github.com/rcrowley/go-metrics/validate.sh b/vendor/github.com/rcrowley/go-metrics/validate.sh new file mode 100644 index 00000000..c4ae91e6 --- /dev/null +++ b/vendor/github.com/rcrowley/go-metrics/validate.sh @@ -0,0 +1,10 @@ +#!/bin/bash + +set -e + +# check there are no formatting issues +GOFMT_LINES=`gofmt -l . | wc -l | xargs` +test $GOFMT_LINES -eq 0 || echo "gofmt needs to be run, ${GOFMT_LINES} files have issues" + +# run the tests for the root package +go test -race . diff --git a/vendor/github.com/rcrowley/go-metrics/writer.go b/vendor/github.com/rcrowley/go-metrics/writer.go new file mode 100644 index 00000000..091e971d --- /dev/null +++ b/vendor/github.com/rcrowley/go-metrics/writer.go @@ -0,0 +1,100 @@ +package metrics + +import ( + "fmt" + "io" + "sort" + "time" +) + +// Write sorts writes each metric in the given registry periodically to the +// given io.Writer. +func Write(r Registry, d time.Duration, w io.Writer) { + for _ = range time.Tick(d) { + WriteOnce(r, w) + } +} + +// WriteOnce sorts and writes metrics in the given registry to the given +// io.Writer. +func WriteOnce(r Registry, w io.Writer) { + var namedMetrics namedMetricSlice + r.Each(func(name string, i interface{}) { + namedMetrics = append(namedMetrics, namedMetric{name, i}) + }) + + sort.Sort(namedMetrics) + for _, namedMetric := range namedMetrics { + switch metric := namedMetric.m.(type) { + case Counter: + fmt.Fprintf(w, "counter %s\n", namedMetric.name) + fmt.Fprintf(w, " count: %9d\n", metric.Count()) + case Gauge: + fmt.Fprintf(w, "gauge %s\n", namedMetric.name) + fmt.Fprintf(w, " value: %9d\n", metric.Value()) + case GaugeFloat64: + fmt.Fprintf(w, "gauge %s\n", namedMetric.name) + fmt.Fprintf(w, " value: %f\n", metric.Value()) + case Healthcheck: + metric.Check() + fmt.Fprintf(w, "healthcheck %s\n", namedMetric.name) + fmt.Fprintf(w, " error: %v\n", metric.Error()) + case Histogram: + h := metric.Snapshot() + ps := h.Percentiles([]float64{0.5, 0.75, 0.95, 0.99, 0.999}) + fmt.Fprintf(w, "histogram %s\n", namedMetric.name) + fmt.Fprintf(w, " count: %9d\n", h.Count()) + fmt.Fprintf(w, " min: %9d\n", h.Min()) + fmt.Fprintf(w, " max: %9d\n", h.Max()) + fmt.Fprintf(w, " mean: %12.2f\n", h.Mean()) + fmt.Fprintf(w, " stddev: %12.2f\n", h.StdDev()) + fmt.Fprintf(w, " median: %12.2f\n", ps[0]) + fmt.Fprintf(w, " 75%%: %12.2f\n", ps[1]) + fmt.Fprintf(w, " 95%%: %12.2f\n", ps[2]) + fmt.Fprintf(w, " 99%%: %12.2f\n", ps[3]) + fmt.Fprintf(w, " 99.9%%: %12.2f\n", ps[4]) + case Meter: + m := metric.Snapshot() + fmt.Fprintf(w, "meter %s\n", namedMetric.name) + fmt.Fprintf(w, " count: %9d\n", m.Count()) + fmt.Fprintf(w, " 1-min rate: %12.2f\n", m.Rate1()) + fmt.Fprintf(w, " 5-min rate: %12.2f\n", m.Rate5()) + fmt.Fprintf(w, " 15-min rate: %12.2f\n", m.Rate15()) + fmt.Fprintf(w, " mean rate: %12.2f\n", m.RateMean()) + case Timer: + t := metric.Snapshot() + ps := t.Percentiles([]float64{0.5, 0.75, 0.95, 0.99, 0.999}) + fmt.Fprintf(w, "timer %s\n", namedMetric.name) + fmt.Fprintf(w, " count: %9d\n", t.Count()) + fmt.Fprintf(w, " min: %9d\n", t.Min()) + fmt.Fprintf(w, " max: %9d\n", t.Max()) + fmt.Fprintf(w, " mean: %12.2f\n", t.Mean()) + fmt.Fprintf(w, " stddev: %12.2f\n", t.StdDev()) + fmt.Fprintf(w, " median: %12.2f\n", ps[0]) + fmt.Fprintf(w, " 75%%: %12.2f\n", ps[1]) + fmt.Fprintf(w, " 95%%: %12.2f\n", ps[2]) + fmt.Fprintf(w, " 99%%: %12.2f\n", ps[3]) + fmt.Fprintf(w, " 99.9%%: %12.2f\n", ps[4]) + fmt.Fprintf(w, " 1-min rate: %12.2f\n", t.Rate1()) + fmt.Fprintf(w, " 5-min rate: %12.2f\n", t.Rate5()) + fmt.Fprintf(w, " 15-min rate: %12.2f\n", t.Rate15()) + fmt.Fprintf(w, " mean rate: %12.2f\n", t.RateMean()) + } + } +} + +type namedMetric struct { + name string + m interface{} +} + +// namedMetricSlice is a slice of namedMetrics that implements sort.Interface. +type namedMetricSlice []namedMetric + +func (nms namedMetricSlice) Len() int { return len(nms) } + +func (nms namedMetricSlice) Swap(i, j int) { nms[i], nms[j] = nms[j], nms[i] } + +func (nms namedMetricSlice) Less(i, j int) bool { + return nms[i].name < nms[j].name +} diff --git a/vendor/github.com/stretchr/testify/LICENSE b/vendor/github.com/stretchr/testify/LICENSE new file mode 100644 index 00000000..4b0421cf --- /dev/null +++ b/vendor/github.com/stretchr/testify/LICENSE @@ -0,0 +1,21 @@ +MIT License + +Copyright (c) 2012-2020 Mat Ryer, Tyler Bunnell and contributors. + +Permission is hereby granted, free of charge, to any person obtaining a copy +of this software and associated documentation files (the "Software"), to deal +in the Software without restriction, including without limitation the rights +to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the Software is +furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in all +copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE +SOFTWARE. diff --git a/vendor/github.com/stretchr/testify/assert/assertion_compare.go b/vendor/github.com/stretchr/testify/assert/assertion_compare.go new file mode 100644 index 00000000..41649d26 --- /dev/null +++ b/vendor/github.com/stretchr/testify/assert/assertion_compare.go @@ -0,0 +1,394 @@ +package assert + +import ( + "fmt" + "reflect" +) + +type CompareType int + +const ( + compareLess CompareType = iota - 1 + compareEqual + compareGreater +) + +var ( + intType = reflect.TypeOf(int(1)) + int8Type = reflect.TypeOf(int8(1)) + int16Type = reflect.TypeOf(int16(1)) + int32Type = reflect.TypeOf(int32(1)) + int64Type = reflect.TypeOf(int64(1)) + + uintType = reflect.TypeOf(uint(1)) + uint8Type = reflect.TypeOf(uint8(1)) + uint16Type = reflect.TypeOf(uint16(1)) + uint32Type = reflect.TypeOf(uint32(1)) + uint64Type = reflect.TypeOf(uint64(1)) + + float32Type = reflect.TypeOf(float32(1)) + float64Type = reflect.TypeOf(float64(1)) + + stringType = reflect.TypeOf("") +) + +func compare(obj1, obj2 interface{}, kind reflect.Kind) (CompareType, bool) { + obj1Value := reflect.ValueOf(obj1) + obj2Value := reflect.ValueOf(obj2) + + // throughout this switch we try and avoid calling .Convert() if possible, + // as this has a pretty big performance impact + switch kind { + case reflect.Int: + { + intobj1, ok := obj1.(int) + if !ok { + intobj1 = obj1Value.Convert(intType).Interface().(int) + } + intobj2, ok := obj2.(int) + if !ok { + intobj2 = obj2Value.Convert(intType).Interface().(int) + } + if intobj1 > intobj2 { + return compareGreater, true + } + if intobj1 == intobj2 { + return compareEqual, true + } + if intobj1 < intobj2 { + return compareLess, true + } + } + case reflect.Int8: + { + int8obj1, ok := obj1.(int8) + if !ok { + int8obj1 = obj1Value.Convert(int8Type).Interface().(int8) + } + int8obj2, ok := obj2.(int8) + if !ok { + int8obj2 = obj2Value.Convert(int8Type).Interface().(int8) + } + if int8obj1 > int8obj2 { + return compareGreater, true + } + if int8obj1 == int8obj2 { + return compareEqual, true + } + if int8obj1 < int8obj2 { + return compareLess, true + } + } + case reflect.Int16: + { + int16obj1, ok := obj1.(int16) + if !ok { + int16obj1 = obj1Value.Convert(int16Type).Interface().(int16) + } + int16obj2, ok := obj2.(int16) + if !ok { + int16obj2 = obj2Value.Convert(int16Type).Interface().(int16) + } + if int16obj1 > int16obj2 { + return compareGreater, true + } + if int16obj1 == int16obj2 { + return compareEqual, true + } + if int16obj1 < int16obj2 { + return compareLess, true + } + } + case reflect.Int32: + { + int32obj1, ok := obj1.(int32) + if !ok { + int32obj1 = obj1Value.Convert(int32Type).Interface().(int32) + } + int32obj2, ok := obj2.(int32) + if !ok { + int32obj2 = obj2Value.Convert(int32Type).Interface().(int32) + } + if int32obj1 > int32obj2 { + return compareGreater, true + } + if int32obj1 == int32obj2 { + return compareEqual, true + } + if int32obj1 < int32obj2 { + return compareLess, true + } + } + case reflect.Int64: + { + int64obj1, ok := obj1.(int64) + if !ok { + int64obj1 = obj1Value.Convert(int64Type).Interface().(int64) + } + int64obj2, ok := obj2.(int64) + if !ok { + int64obj2 = obj2Value.Convert(int64Type).Interface().(int64) + } + if int64obj1 > int64obj2 { + return compareGreater, true + } + if int64obj1 == int64obj2 { + return compareEqual, true + } + if int64obj1 < int64obj2 { + return compareLess, true + } + } + case reflect.Uint: + { + uintobj1, ok := obj1.(uint) + if !ok { + uintobj1 = obj1Value.Convert(uintType).Interface().(uint) + } + uintobj2, ok := obj2.(uint) + if !ok { + uintobj2 = obj2Value.Convert(uintType).Interface().(uint) + } + if uintobj1 > uintobj2 { + return compareGreater, true + } + if uintobj1 == uintobj2 { + return compareEqual, true + } + if uintobj1 < uintobj2 { + return compareLess, true + } + } + case reflect.Uint8: + { + uint8obj1, ok := obj1.(uint8) + if !ok { + uint8obj1 = obj1Value.Convert(uint8Type).Interface().(uint8) + } + uint8obj2, ok := obj2.(uint8) + if !ok { + uint8obj2 = obj2Value.Convert(uint8Type).Interface().(uint8) + } + if uint8obj1 > uint8obj2 { + return compareGreater, true + } + if uint8obj1 == uint8obj2 { + return compareEqual, true + } + if uint8obj1 < uint8obj2 { + return compareLess, true + } + } + case reflect.Uint16: + { + uint16obj1, ok := obj1.(uint16) + if !ok { + uint16obj1 = obj1Value.Convert(uint16Type).Interface().(uint16) + } + uint16obj2, ok := obj2.(uint16) + if !ok { + uint16obj2 = obj2Value.Convert(uint16Type).Interface().(uint16) + } + if uint16obj1 > uint16obj2 { + return compareGreater, true + } + if uint16obj1 == uint16obj2 { + return compareEqual, true + } + if uint16obj1 < uint16obj2 { + return compareLess, true + } + } + case reflect.Uint32: + { + uint32obj1, ok := obj1.(uint32) + if !ok { + uint32obj1 = obj1Value.Convert(uint32Type).Interface().(uint32) + } + uint32obj2, ok := obj2.(uint32) + if !ok { + uint32obj2 = obj2Value.Convert(uint32Type).Interface().(uint32) + } + if uint32obj1 > uint32obj2 { + return compareGreater, true + } + if uint32obj1 == uint32obj2 { + return compareEqual, true + } + if uint32obj1 < uint32obj2 { + return compareLess, true + } + } + case reflect.Uint64: + { + uint64obj1, ok := obj1.(uint64) + if !ok { + uint64obj1 = obj1Value.Convert(uint64Type).Interface().(uint64) + } + uint64obj2, ok := obj2.(uint64) + if !ok { + uint64obj2 = obj2Value.Convert(uint64Type).Interface().(uint64) + } + if uint64obj1 > uint64obj2 { + return compareGreater, true + } + if uint64obj1 == uint64obj2 { + return compareEqual, true + } + if uint64obj1 < uint64obj2 { + return compareLess, true + } + } + case reflect.Float32: + { + float32obj1, ok := obj1.(float32) + if !ok { + float32obj1 = obj1Value.Convert(float32Type).Interface().(float32) + } + float32obj2, ok := obj2.(float32) + if !ok { + float32obj2 = obj2Value.Convert(float32Type).Interface().(float32) + } + if float32obj1 > float32obj2 { + return compareGreater, true + } + if float32obj1 == float32obj2 { + return compareEqual, true + } + if float32obj1 < float32obj2 { + return compareLess, true + } + } + case reflect.Float64: + { + float64obj1, ok := obj1.(float64) + if !ok { + float64obj1 = obj1Value.Convert(float64Type).Interface().(float64) + } + float64obj2, ok := obj2.(float64) + if !ok { + float64obj2 = obj2Value.Convert(float64Type).Interface().(float64) + } + if float64obj1 > float64obj2 { + return compareGreater, true + } + if float64obj1 == float64obj2 { + return compareEqual, true + } + if float64obj1 < float64obj2 { + return compareLess, true + } + } + case reflect.String: + { + stringobj1, ok := obj1.(string) + if !ok { + stringobj1 = obj1Value.Convert(stringType).Interface().(string) + } + stringobj2, ok := obj2.(string) + if !ok { + stringobj2 = obj2Value.Convert(stringType).Interface().(string) + } + if stringobj1 > stringobj2 { + return compareGreater, true + } + if stringobj1 == stringobj2 { + return compareEqual, true + } + if stringobj1 < stringobj2 { + return compareLess, true + } + } + } + + return compareEqual, false +} + +// Greater asserts that the first element is greater than the second +// +// assert.Greater(t, 2, 1) +// assert.Greater(t, float64(2), float64(1)) +// assert.Greater(t, "b", "a") +func Greater(t TestingT, e1 interface{}, e2 interface{}, msgAndArgs ...interface{}) bool { + return compareTwoValues(t, e1, e2, []CompareType{compareGreater}, "\"%v\" is not greater than \"%v\"", msgAndArgs) +} + +// GreaterOrEqual asserts that the first element is greater than or equal to the second +// +// assert.GreaterOrEqual(t, 2, 1) +// assert.GreaterOrEqual(t, 2, 2) +// assert.GreaterOrEqual(t, "b", "a") +// assert.GreaterOrEqual(t, "b", "b") +func GreaterOrEqual(t TestingT, e1 interface{}, e2 interface{}, msgAndArgs ...interface{}) bool { + return compareTwoValues(t, e1, e2, []CompareType{compareGreater, compareEqual}, "\"%v\" is not greater than or equal to \"%v\"", msgAndArgs) +} + +// Less asserts that the first element is less than the second +// +// assert.Less(t, 1, 2) +// assert.Less(t, float64(1), float64(2)) +// assert.Less(t, "a", "b") +func Less(t TestingT, e1 interface{}, e2 interface{}, msgAndArgs ...interface{}) bool { + return compareTwoValues(t, e1, e2, []CompareType{compareLess}, "\"%v\" is not less than \"%v\"", msgAndArgs) +} + +// LessOrEqual asserts that the first element is less than or equal to the second +// +// assert.LessOrEqual(t, 1, 2) +// assert.LessOrEqual(t, 2, 2) +// assert.LessOrEqual(t, "a", "b") +// assert.LessOrEqual(t, "b", "b") +func LessOrEqual(t TestingT, e1 interface{}, e2 interface{}, msgAndArgs ...interface{}) bool { + return compareTwoValues(t, e1, e2, []CompareType{compareLess, compareEqual}, "\"%v\" is not less than or equal to \"%v\"", msgAndArgs) +} + +// Positive asserts that the specified element is positive +// +// assert.Positive(t, 1) +// assert.Positive(t, 1.23) +func Positive(t TestingT, e interface{}, msgAndArgs ...interface{}) bool { + zero := reflect.Zero(reflect.TypeOf(e)) + return compareTwoValues(t, e, zero.Interface(), []CompareType{compareGreater}, "\"%v\" is not positive", msgAndArgs) +} + +// Negative asserts that the specified element is negative +// +// assert.Negative(t, -1) +// assert.Negative(t, -1.23) +func Negative(t TestingT, e interface{}, msgAndArgs ...interface{}) bool { + zero := reflect.Zero(reflect.TypeOf(e)) + return compareTwoValues(t, e, zero.Interface(), []CompareType{compareLess}, "\"%v\" is not negative", msgAndArgs) +} + +func compareTwoValues(t TestingT, e1 interface{}, e2 interface{}, allowedComparesResults []CompareType, failMessage string, msgAndArgs ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + + e1Kind := reflect.ValueOf(e1).Kind() + e2Kind := reflect.ValueOf(e2).Kind() + if e1Kind != e2Kind { + return Fail(t, "Elements should be the same type", msgAndArgs...) + } + + compareResult, isComparable := compare(e1, e2, e1Kind) + if !isComparable { + return Fail(t, fmt.Sprintf("Can not compare type \"%s\"", reflect.TypeOf(e1)), msgAndArgs...) + } + + if !containsValue(allowedComparesResults, compareResult) { + return Fail(t, fmt.Sprintf(failMessage, e1, e2), msgAndArgs...) + } + + return true +} + +func containsValue(values []CompareType, value CompareType) bool { + for _, v := range values { + if v == value { + return true + } + } + + return false +} diff --git a/vendor/github.com/stretchr/testify/assert/assertion_format.go b/vendor/github.com/stretchr/testify/assert/assertion_format.go new file mode 100644 index 00000000..4dfd1229 --- /dev/null +++ b/vendor/github.com/stretchr/testify/assert/assertion_format.go @@ -0,0 +1,741 @@ +/* +* CODE GENERATED AUTOMATICALLY WITH github.com/stretchr/testify/_codegen +* THIS FILE MUST NOT BE EDITED BY HAND + */ + +package assert + +import ( + http "net/http" + url "net/url" + time "time" +) + +// Conditionf uses a Comparison to assert a complex condition. +func Conditionf(t TestingT, comp Comparison, msg string, args ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + return Condition(t, comp, append([]interface{}{msg}, args...)...) +} + +// Containsf asserts that the specified string, list(array, slice...) or map contains the +// specified substring or element. +// +// assert.Containsf(t, "Hello World", "World", "error message %s", "formatted") +// assert.Containsf(t, ["Hello", "World"], "World", "error message %s", "formatted") +// assert.Containsf(t, {"Hello": "World"}, "Hello", "error message %s", "formatted") +func Containsf(t TestingT, s interface{}, contains interface{}, msg string, args ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + return Contains(t, s, contains, append([]interface{}{msg}, args...)...) +} + +// DirExistsf checks whether a directory exists in the given path. It also fails +// if the path is a file rather a directory or there is an error checking whether it exists. +func DirExistsf(t TestingT, path string, msg string, args ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + return DirExists(t, path, append([]interface{}{msg}, args...)...) +} + +// ElementsMatchf asserts that the specified listA(array, slice...) is equal to specified +// listB(array, slice...) ignoring the order of the elements. If there are duplicate elements, +// the number of appearances of each of them in both lists should match. +// +// assert.ElementsMatchf(t, [1, 3, 2, 3], [1, 3, 3, 2], "error message %s", "formatted") +func ElementsMatchf(t TestingT, listA interface{}, listB interface{}, msg string, args ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + return ElementsMatch(t, listA, listB, append([]interface{}{msg}, args...)...) +} + +// Emptyf asserts that the specified object is empty. I.e. nil, "", false, 0 or either +// a slice or a channel with len == 0. +// +// assert.Emptyf(t, obj, "error message %s", "formatted") +func Emptyf(t TestingT, object interface{}, msg string, args ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + return Empty(t, object, append([]interface{}{msg}, args...)...) +} + +// Equalf asserts that two objects are equal. +// +// assert.Equalf(t, 123, 123, "error message %s", "formatted") +// +// Pointer variable equality is determined based on the equality of the +// referenced values (as opposed to the memory addresses). Function equality +// cannot be determined and will always fail. +func Equalf(t TestingT, expected interface{}, actual interface{}, msg string, args ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + return Equal(t, expected, actual, append([]interface{}{msg}, args...)...) +} + +// EqualErrorf asserts that a function returned an error (i.e. not `nil`) +// and that it is equal to the provided error. +// +// actualObj, err := SomeFunction() +// assert.EqualErrorf(t, err, expectedErrorString, "error message %s", "formatted") +func EqualErrorf(t TestingT, theError error, errString string, msg string, args ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + return EqualError(t, theError, errString, append([]interface{}{msg}, args...)...) +} + +// EqualValuesf asserts that two objects are equal or convertable to the same types +// and equal. +// +// assert.EqualValuesf(t, uint32(123), int32(123), "error message %s", "formatted") +func EqualValuesf(t TestingT, expected interface{}, actual interface{}, msg string, args ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + return EqualValues(t, expected, actual, append([]interface{}{msg}, args...)...) +} + +// Errorf asserts that a function returned an error (i.e. not `nil`). +// +// actualObj, err := SomeFunction() +// if assert.Errorf(t, err, "error message %s", "formatted") { +// assert.Equal(t, expectedErrorf, err) +// } +func Errorf(t TestingT, err error, msg string, args ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + return Error(t, err, append([]interface{}{msg}, args...)...) +} + +// ErrorAsf asserts that at least one of the errors in err's chain matches target, and if so, sets target to that error value. +// This is a wrapper for errors.As. +func ErrorAsf(t TestingT, err error, target interface{}, msg string, args ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + return ErrorAs(t, err, target, append([]interface{}{msg}, args...)...) +} + +// ErrorIsf asserts that at least one of the errors in err's chain matches target. +// This is a wrapper for errors.Is. +func ErrorIsf(t TestingT, err error, target error, msg string, args ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + return ErrorIs(t, err, target, append([]interface{}{msg}, args...)...) +} + +// Eventuallyf asserts that given condition will be met in waitFor time, +// periodically checking target function each tick. +// +// assert.Eventuallyf(t, func() bool { return true; }, time.Second, 10*time.Millisecond, "error message %s", "formatted") +func Eventuallyf(t TestingT, condition func() bool, waitFor time.Duration, tick time.Duration, msg string, args ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + return Eventually(t, condition, waitFor, tick, append([]interface{}{msg}, args...)...) +} + +// Exactlyf asserts that two objects are equal in value and type. +// +// assert.Exactlyf(t, int32(123), int64(123), "error message %s", "formatted") +func Exactlyf(t TestingT, expected interface{}, actual interface{}, msg string, args ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + return Exactly(t, expected, actual, append([]interface{}{msg}, args...)...) +} + +// Failf reports a failure through +func Failf(t TestingT, failureMessage string, msg string, args ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + return Fail(t, failureMessage, append([]interface{}{msg}, args...)...) +} + +// FailNowf fails test +func FailNowf(t TestingT, failureMessage string, msg string, args ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + return FailNow(t, failureMessage, append([]interface{}{msg}, args...)...) +} + +// Falsef asserts that the specified value is false. +// +// assert.Falsef(t, myBool, "error message %s", "formatted") +func Falsef(t TestingT, value bool, msg string, args ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + return False(t, value, append([]interface{}{msg}, args...)...) +} + +// FileExistsf checks whether a file exists in the given path. It also fails if +// the path points to a directory or there is an error when trying to check the file. +func FileExistsf(t TestingT, path string, msg string, args ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + return FileExists(t, path, append([]interface{}{msg}, args...)...) +} + +// Greaterf asserts that the first element is greater than the second +// +// assert.Greaterf(t, 2, 1, "error message %s", "formatted") +// assert.Greaterf(t, float64(2), float64(1), "error message %s", "formatted") +// assert.Greaterf(t, "b", "a", "error message %s", "formatted") +func Greaterf(t TestingT, e1 interface{}, e2 interface{}, msg string, args ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + return Greater(t, e1, e2, append([]interface{}{msg}, args...)...) +} + +// GreaterOrEqualf asserts that the first element is greater than or equal to the second +// +// assert.GreaterOrEqualf(t, 2, 1, "error message %s", "formatted") +// assert.GreaterOrEqualf(t, 2, 2, "error message %s", "formatted") +// assert.GreaterOrEqualf(t, "b", "a", "error message %s", "formatted") +// assert.GreaterOrEqualf(t, "b", "b", "error message %s", "formatted") +func GreaterOrEqualf(t TestingT, e1 interface{}, e2 interface{}, msg string, args ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + return GreaterOrEqual(t, e1, e2, append([]interface{}{msg}, args...)...) +} + +// HTTPBodyContainsf asserts that a specified handler returns a +// body that contains a string. +// +// assert.HTTPBodyContainsf(t, myHandler, "GET", "www.google.com", nil, "I'm Feeling Lucky", "error message %s", "formatted") +// +// Returns whether the assertion was successful (true) or not (false). +func HTTPBodyContainsf(t TestingT, handler http.HandlerFunc, method string, url string, values url.Values, str interface{}, msg string, args ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + return HTTPBodyContains(t, handler, method, url, values, str, append([]interface{}{msg}, args...)...) +} + +// HTTPBodyNotContainsf asserts that a specified handler returns a +// body that does not contain a string. +// +// assert.HTTPBodyNotContainsf(t, myHandler, "GET", "www.google.com", nil, "I'm Feeling Lucky", "error message %s", "formatted") +// +// Returns whether the assertion was successful (true) or not (false). +func HTTPBodyNotContainsf(t TestingT, handler http.HandlerFunc, method string, url string, values url.Values, str interface{}, msg string, args ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + return HTTPBodyNotContains(t, handler, method, url, values, str, append([]interface{}{msg}, args...)...) +} + +// HTTPErrorf asserts that a specified handler returns an error status code. +// +// assert.HTTPErrorf(t, myHandler, "POST", "/a/b/c", url.Values{"a": []string{"b", "c"}} +// +// Returns whether the assertion was successful (true) or not (false). +func HTTPErrorf(t TestingT, handler http.HandlerFunc, method string, url string, values url.Values, msg string, args ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + return HTTPError(t, handler, method, url, values, append([]interface{}{msg}, args...)...) +} + +// HTTPRedirectf asserts that a specified handler returns a redirect status code. +// +// assert.HTTPRedirectf(t, myHandler, "GET", "/a/b/c", url.Values{"a": []string{"b", "c"}} +// +// Returns whether the assertion was successful (true) or not (false). +func HTTPRedirectf(t TestingT, handler http.HandlerFunc, method string, url string, values url.Values, msg string, args ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + return HTTPRedirect(t, handler, method, url, values, append([]interface{}{msg}, args...)...) +} + +// HTTPStatusCodef asserts that a specified handler returns a specified status code. +// +// assert.HTTPStatusCodef(t, myHandler, "GET", "/notImplemented", nil, 501, "error message %s", "formatted") +// +// Returns whether the assertion was successful (true) or not (false). +func HTTPStatusCodef(t TestingT, handler http.HandlerFunc, method string, url string, values url.Values, statuscode int, msg string, args ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + return HTTPStatusCode(t, handler, method, url, values, statuscode, append([]interface{}{msg}, args...)...) +} + +// HTTPSuccessf asserts that a specified handler returns a success status code. +// +// assert.HTTPSuccessf(t, myHandler, "POST", "http://www.google.com", nil, "error message %s", "formatted") +// +// Returns whether the assertion was successful (true) or not (false). +func HTTPSuccessf(t TestingT, handler http.HandlerFunc, method string, url string, values url.Values, msg string, args ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + return HTTPSuccess(t, handler, method, url, values, append([]interface{}{msg}, args...)...) +} + +// Implementsf asserts that an object is implemented by the specified interface. +// +// assert.Implementsf(t, (*MyInterface)(nil), new(MyObject), "error message %s", "formatted") +func Implementsf(t TestingT, interfaceObject interface{}, object interface{}, msg string, args ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + return Implements(t, interfaceObject, object, append([]interface{}{msg}, args...)...) +} + +// InDeltaf asserts that the two numerals are within delta of each other. +// +// assert.InDeltaf(t, math.Pi, 22/7.0, 0.01, "error message %s", "formatted") +func InDeltaf(t TestingT, expected interface{}, actual interface{}, delta float64, msg string, args ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + return InDelta(t, expected, actual, delta, append([]interface{}{msg}, args...)...) +} + +// InDeltaMapValuesf is the same as InDelta, but it compares all values between two maps. Both maps must have exactly the same keys. +func InDeltaMapValuesf(t TestingT, expected interface{}, actual interface{}, delta float64, msg string, args ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + return InDeltaMapValues(t, expected, actual, delta, append([]interface{}{msg}, args...)...) +} + +// InDeltaSlicef is the same as InDelta, except it compares two slices. +func InDeltaSlicef(t TestingT, expected interface{}, actual interface{}, delta float64, msg string, args ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + return InDeltaSlice(t, expected, actual, delta, append([]interface{}{msg}, args...)...) +} + +// InEpsilonf asserts that expected and actual have a relative error less than epsilon +func InEpsilonf(t TestingT, expected interface{}, actual interface{}, epsilon float64, msg string, args ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + return InEpsilon(t, expected, actual, epsilon, append([]interface{}{msg}, args...)...) +} + +// InEpsilonSlicef is the same as InEpsilon, except it compares each value from two slices. +func InEpsilonSlicef(t TestingT, expected interface{}, actual interface{}, epsilon float64, msg string, args ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + return InEpsilonSlice(t, expected, actual, epsilon, append([]interface{}{msg}, args...)...) +} + +// IsDecreasingf asserts that the collection is decreasing +// +// assert.IsDecreasingf(t, []int{2, 1, 0}, "error message %s", "formatted") +// assert.IsDecreasingf(t, []float{2, 1}, "error message %s", "formatted") +// assert.IsDecreasingf(t, []string{"b", "a"}, "error message %s", "formatted") +func IsDecreasingf(t TestingT, object interface{}, msg string, args ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + return IsDecreasing(t, object, append([]interface{}{msg}, args...)...) +} + +// IsIncreasingf asserts that the collection is increasing +// +// assert.IsIncreasingf(t, []int{1, 2, 3}, "error message %s", "formatted") +// assert.IsIncreasingf(t, []float{1, 2}, "error message %s", "formatted") +// assert.IsIncreasingf(t, []string{"a", "b"}, "error message %s", "formatted") +func IsIncreasingf(t TestingT, object interface{}, msg string, args ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + return IsIncreasing(t, object, append([]interface{}{msg}, args...)...) +} + +// IsNonDecreasingf asserts that the collection is not decreasing +// +// assert.IsNonDecreasingf(t, []int{1, 1, 2}, "error message %s", "formatted") +// assert.IsNonDecreasingf(t, []float{1, 2}, "error message %s", "formatted") +// assert.IsNonDecreasingf(t, []string{"a", "b"}, "error message %s", "formatted") +func IsNonDecreasingf(t TestingT, object interface{}, msg string, args ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + return IsNonDecreasing(t, object, append([]interface{}{msg}, args...)...) +} + +// IsNonIncreasingf asserts that the collection is not increasing +// +// assert.IsNonIncreasingf(t, []int{2, 1, 1}, "error message %s", "formatted") +// assert.IsNonIncreasingf(t, []float{2, 1}, "error message %s", "formatted") +// assert.IsNonIncreasingf(t, []string{"b", "a"}, "error message %s", "formatted") +func IsNonIncreasingf(t TestingT, object interface{}, msg string, args ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + return IsNonIncreasing(t, object, append([]interface{}{msg}, args...)...) +} + +// IsTypef asserts that the specified objects are of the same type. +func IsTypef(t TestingT, expectedType interface{}, object interface{}, msg string, args ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + return IsType(t, expectedType, object, append([]interface{}{msg}, args...)...) +} + +// JSONEqf asserts that two JSON strings are equivalent. +// +// assert.JSONEqf(t, `{"hello": "world", "foo": "bar"}`, `{"foo": "bar", "hello": "world"}`, "error message %s", "formatted") +func JSONEqf(t TestingT, expected string, actual string, msg string, args ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + return JSONEq(t, expected, actual, append([]interface{}{msg}, args...)...) +} + +// Lenf asserts that the specified object has specific length. +// Lenf also fails if the object has a type that len() not accept. +// +// assert.Lenf(t, mySlice, 3, "error message %s", "formatted") +func Lenf(t TestingT, object interface{}, length int, msg string, args ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + return Len(t, object, length, append([]interface{}{msg}, args...)...) +} + +// Lessf asserts that the first element is less than the second +// +// assert.Lessf(t, 1, 2, "error message %s", "formatted") +// assert.Lessf(t, float64(1), float64(2), "error message %s", "formatted") +// assert.Lessf(t, "a", "b", "error message %s", "formatted") +func Lessf(t TestingT, e1 interface{}, e2 interface{}, msg string, args ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + return Less(t, e1, e2, append([]interface{}{msg}, args...)...) +} + +// LessOrEqualf asserts that the first element is less than or equal to the second +// +// assert.LessOrEqualf(t, 1, 2, "error message %s", "formatted") +// assert.LessOrEqualf(t, 2, 2, "error message %s", "formatted") +// assert.LessOrEqualf(t, "a", "b", "error message %s", "formatted") +// assert.LessOrEqualf(t, "b", "b", "error message %s", "formatted") +func LessOrEqualf(t TestingT, e1 interface{}, e2 interface{}, msg string, args ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + return LessOrEqual(t, e1, e2, append([]interface{}{msg}, args...)...) +} + +// Negativef asserts that the specified element is negative +// +// assert.Negativef(t, -1, "error message %s", "formatted") +// assert.Negativef(t, -1.23, "error message %s", "formatted") +func Negativef(t TestingT, e interface{}, msg string, args ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + return Negative(t, e, append([]interface{}{msg}, args...)...) +} + +// Neverf asserts that the given condition doesn't satisfy in waitFor time, +// periodically checking the target function each tick. +// +// assert.Neverf(t, func() bool { return false; }, time.Second, 10*time.Millisecond, "error message %s", "formatted") +func Neverf(t TestingT, condition func() bool, waitFor time.Duration, tick time.Duration, msg string, args ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + return Never(t, condition, waitFor, tick, append([]interface{}{msg}, args...)...) +} + +// Nilf asserts that the specified object is nil. +// +// assert.Nilf(t, err, "error message %s", "formatted") +func Nilf(t TestingT, object interface{}, msg string, args ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + return Nil(t, object, append([]interface{}{msg}, args...)...) +} + +// NoDirExistsf checks whether a directory does not exist in the given path. +// It fails if the path points to an existing _directory_ only. +func NoDirExistsf(t TestingT, path string, msg string, args ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + return NoDirExists(t, path, append([]interface{}{msg}, args...)...) +} + +// NoErrorf asserts that a function returned no error (i.e. `nil`). +// +// actualObj, err := SomeFunction() +// if assert.NoErrorf(t, err, "error message %s", "formatted") { +// assert.Equal(t, expectedObj, actualObj) +// } +func NoErrorf(t TestingT, err error, msg string, args ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + return NoError(t, err, append([]interface{}{msg}, args...)...) +} + +// NoFileExistsf checks whether a file does not exist in a given path. It fails +// if the path points to an existing _file_ only. +func NoFileExistsf(t TestingT, path string, msg string, args ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + return NoFileExists(t, path, append([]interface{}{msg}, args...)...) +} + +// NotContainsf asserts that the specified string, list(array, slice...) or map does NOT contain the +// specified substring or element. +// +// assert.NotContainsf(t, "Hello World", "Earth", "error message %s", "formatted") +// assert.NotContainsf(t, ["Hello", "World"], "Earth", "error message %s", "formatted") +// assert.NotContainsf(t, {"Hello": "World"}, "Earth", "error message %s", "formatted") +func NotContainsf(t TestingT, s interface{}, contains interface{}, msg string, args ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + return NotContains(t, s, contains, append([]interface{}{msg}, args...)...) +} + +// NotEmptyf asserts that the specified object is NOT empty. I.e. not nil, "", false, 0 or either +// a slice or a channel with len == 0. +// +// if assert.NotEmptyf(t, obj, "error message %s", "formatted") { +// assert.Equal(t, "two", obj[1]) +// } +func NotEmptyf(t TestingT, object interface{}, msg string, args ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + return NotEmpty(t, object, append([]interface{}{msg}, args...)...) +} + +// NotEqualf asserts that the specified values are NOT equal. +// +// assert.NotEqualf(t, obj1, obj2, "error message %s", "formatted") +// +// Pointer variable equality is determined based on the equality of the +// referenced values (as opposed to the memory addresses). +func NotEqualf(t TestingT, expected interface{}, actual interface{}, msg string, args ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + return NotEqual(t, expected, actual, append([]interface{}{msg}, args...)...) +} + +// NotEqualValuesf asserts that two objects are not equal even when converted to the same type +// +// assert.NotEqualValuesf(t, obj1, obj2, "error message %s", "formatted") +func NotEqualValuesf(t TestingT, expected interface{}, actual interface{}, msg string, args ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + return NotEqualValues(t, expected, actual, append([]interface{}{msg}, args...)...) +} + +// NotErrorIsf asserts that at none of the errors in err's chain matches target. +// This is a wrapper for errors.Is. +func NotErrorIsf(t TestingT, err error, target error, msg string, args ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + return NotErrorIs(t, err, target, append([]interface{}{msg}, args...)...) +} + +// NotNilf asserts that the specified object is not nil. +// +// assert.NotNilf(t, err, "error message %s", "formatted") +func NotNilf(t TestingT, object interface{}, msg string, args ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + return NotNil(t, object, append([]interface{}{msg}, args...)...) +} + +// NotPanicsf asserts that the code inside the specified PanicTestFunc does NOT panic. +// +// assert.NotPanicsf(t, func(){ RemainCalm() }, "error message %s", "formatted") +func NotPanicsf(t TestingT, f PanicTestFunc, msg string, args ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + return NotPanics(t, f, append([]interface{}{msg}, args...)...) +} + +// NotRegexpf asserts that a specified regexp does not match a string. +// +// assert.NotRegexpf(t, regexp.MustCompile("starts"), "it's starting", "error message %s", "formatted") +// assert.NotRegexpf(t, "^start", "it's not starting", "error message %s", "formatted") +func NotRegexpf(t TestingT, rx interface{}, str interface{}, msg string, args ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + return NotRegexp(t, rx, str, append([]interface{}{msg}, args...)...) +} + +// NotSamef asserts that two pointers do not reference the same object. +// +// assert.NotSamef(t, ptr1, ptr2, "error message %s", "formatted") +// +// Both arguments must be pointer variables. Pointer variable sameness is +// determined based on the equality of both type and value. +func NotSamef(t TestingT, expected interface{}, actual interface{}, msg string, args ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + return NotSame(t, expected, actual, append([]interface{}{msg}, args...)...) +} + +// NotSubsetf asserts that the specified list(array, slice...) contains not all +// elements given in the specified subset(array, slice...). +// +// assert.NotSubsetf(t, [1, 3, 4], [1, 2], "But [1, 3, 4] does not contain [1, 2]", "error message %s", "formatted") +func NotSubsetf(t TestingT, list interface{}, subset interface{}, msg string, args ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + return NotSubset(t, list, subset, append([]interface{}{msg}, args...)...) +} + +// NotZerof asserts that i is not the zero value for its type. +func NotZerof(t TestingT, i interface{}, msg string, args ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + return NotZero(t, i, append([]interface{}{msg}, args...)...) +} + +// Panicsf asserts that the code inside the specified PanicTestFunc panics. +// +// assert.Panicsf(t, func(){ GoCrazy() }, "error message %s", "formatted") +func Panicsf(t TestingT, f PanicTestFunc, msg string, args ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + return Panics(t, f, append([]interface{}{msg}, args...)...) +} + +// PanicsWithErrorf asserts that the code inside the specified PanicTestFunc +// panics, and that the recovered panic value is an error that satisfies the +// EqualError comparison. +// +// assert.PanicsWithErrorf(t, "crazy error", func(){ GoCrazy() }, "error message %s", "formatted") +func PanicsWithErrorf(t TestingT, errString string, f PanicTestFunc, msg string, args ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + return PanicsWithError(t, errString, f, append([]interface{}{msg}, args...)...) +} + +// PanicsWithValuef asserts that the code inside the specified PanicTestFunc panics, and that +// the recovered panic value equals the expected panic value. +// +// assert.PanicsWithValuef(t, "crazy error", func(){ GoCrazy() }, "error message %s", "formatted") +func PanicsWithValuef(t TestingT, expected interface{}, f PanicTestFunc, msg string, args ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + return PanicsWithValue(t, expected, f, append([]interface{}{msg}, args...)...) +} + +// Positivef asserts that the specified element is positive +// +// assert.Positivef(t, 1, "error message %s", "formatted") +// assert.Positivef(t, 1.23, "error message %s", "formatted") +func Positivef(t TestingT, e interface{}, msg string, args ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + return Positive(t, e, append([]interface{}{msg}, args...)...) +} + +// Regexpf asserts that a specified regexp matches a string. +// +// assert.Regexpf(t, regexp.MustCompile("start"), "it's starting", "error message %s", "formatted") +// assert.Regexpf(t, "start...$", "it's not starting", "error message %s", "formatted") +func Regexpf(t TestingT, rx interface{}, str interface{}, msg string, args ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + return Regexp(t, rx, str, append([]interface{}{msg}, args...)...) +} + +// Samef asserts that two pointers reference the same object. +// +// assert.Samef(t, ptr1, ptr2, "error message %s", "formatted") +// +// Both arguments must be pointer variables. Pointer variable sameness is +// determined based on the equality of both type and value. +func Samef(t TestingT, expected interface{}, actual interface{}, msg string, args ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + return Same(t, expected, actual, append([]interface{}{msg}, args...)...) +} + +// Subsetf asserts that the specified list(array, slice...) contains all +// elements given in the specified subset(array, slice...). +// +// assert.Subsetf(t, [1, 2, 3], [1, 2], "But [1, 2, 3] does contain [1, 2]", "error message %s", "formatted") +func Subsetf(t TestingT, list interface{}, subset interface{}, msg string, args ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + return Subset(t, list, subset, append([]interface{}{msg}, args...)...) +} + +// Truef asserts that the specified value is true. +// +// assert.Truef(t, myBool, "error message %s", "formatted") +func Truef(t TestingT, value bool, msg string, args ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + return True(t, value, append([]interface{}{msg}, args...)...) +} + +// WithinDurationf asserts that the two times are within duration delta of each other. +// +// assert.WithinDurationf(t, time.Now(), time.Now(), 10*time.Second, "error message %s", "formatted") +func WithinDurationf(t TestingT, expected time.Time, actual time.Time, delta time.Duration, msg string, args ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + return WithinDuration(t, expected, actual, delta, append([]interface{}{msg}, args...)...) +} + +// YAMLEqf asserts that two YAML strings are equivalent. +func YAMLEqf(t TestingT, expected string, actual string, msg string, args ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + return YAMLEq(t, expected, actual, append([]interface{}{msg}, args...)...) +} + +// Zerof asserts that i is the zero value for its type. +func Zerof(t TestingT, i interface{}, msg string, args ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + return Zero(t, i, append([]interface{}{msg}, args...)...) +} diff --git a/vendor/github.com/stretchr/testify/assert/assertion_format.go.tmpl b/vendor/github.com/stretchr/testify/assert/assertion_format.go.tmpl new file mode 100644 index 00000000..d2bb0b81 --- /dev/null +++ b/vendor/github.com/stretchr/testify/assert/assertion_format.go.tmpl @@ -0,0 +1,5 @@ +{{.CommentFormat}} +func {{.DocInfo.Name}}f(t TestingT, {{.ParamsFormat}}) bool { + if h, ok := t.(tHelper); ok { h.Helper() } + return {{.DocInfo.Name}}(t, {{.ForwardedParamsFormat}}) +} diff --git a/vendor/github.com/stretchr/testify/assert/assertion_forward.go b/vendor/github.com/stretchr/testify/assert/assertion_forward.go new file mode 100644 index 00000000..25337a6f --- /dev/null +++ b/vendor/github.com/stretchr/testify/assert/assertion_forward.go @@ -0,0 +1,1470 @@ +/* +* CODE GENERATED AUTOMATICALLY WITH github.com/stretchr/testify/_codegen +* THIS FILE MUST NOT BE EDITED BY HAND + */ + +package assert + +import ( + http "net/http" + url "net/url" + time "time" +) + +// Condition uses a Comparison to assert a complex condition. +func (a *Assertions) Condition(comp Comparison, msgAndArgs ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return Condition(a.t, comp, msgAndArgs...) +} + +// Conditionf uses a Comparison to assert a complex condition. +func (a *Assertions) Conditionf(comp Comparison, msg string, args ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return Conditionf(a.t, comp, msg, args...) +} + +// Contains asserts that the specified string, list(array, slice...) or map contains the +// specified substring or element. +// +// a.Contains("Hello World", "World") +// a.Contains(["Hello", "World"], "World") +// a.Contains({"Hello": "World"}, "Hello") +func (a *Assertions) Contains(s interface{}, contains interface{}, msgAndArgs ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return Contains(a.t, s, contains, msgAndArgs...) +} + +// Containsf asserts that the specified string, list(array, slice...) or map contains the +// specified substring or element. +// +// a.Containsf("Hello World", "World", "error message %s", "formatted") +// a.Containsf(["Hello", "World"], "World", "error message %s", "formatted") +// a.Containsf({"Hello": "World"}, "Hello", "error message %s", "formatted") +func (a *Assertions) Containsf(s interface{}, contains interface{}, msg string, args ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return Containsf(a.t, s, contains, msg, args...) +} + +// DirExists checks whether a directory exists in the given path. It also fails +// if the path is a file rather a directory or there is an error checking whether it exists. +func (a *Assertions) DirExists(path string, msgAndArgs ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return DirExists(a.t, path, msgAndArgs...) +} + +// DirExistsf checks whether a directory exists in the given path. It also fails +// if the path is a file rather a directory or there is an error checking whether it exists. +func (a *Assertions) DirExistsf(path string, msg string, args ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return DirExistsf(a.t, path, msg, args...) +} + +// ElementsMatch asserts that the specified listA(array, slice...) is equal to specified +// listB(array, slice...) ignoring the order of the elements. If there are duplicate elements, +// the number of appearances of each of them in both lists should match. +// +// a.ElementsMatch([1, 3, 2, 3], [1, 3, 3, 2]) +func (a *Assertions) ElementsMatch(listA interface{}, listB interface{}, msgAndArgs ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return ElementsMatch(a.t, listA, listB, msgAndArgs...) +} + +// ElementsMatchf asserts that the specified listA(array, slice...) is equal to specified +// listB(array, slice...) ignoring the order of the elements. If there are duplicate elements, +// the number of appearances of each of them in both lists should match. +// +// a.ElementsMatchf([1, 3, 2, 3], [1, 3, 3, 2], "error message %s", "formatted") +func (a *Assertions) ElementsMatchf(listA interface{}, listB interface{}, msg string, args ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return ElementsMatchf(a.t, listA, listB, msg, args...) +} + +// Empty asserts that the specified object is empty. I.e. nil, "", false, 0 or either +// a slice or a channel with len == 0. +// +// a.Empty(obj) +func (a *Assertions) Empty(object interface{}, msgAndArgs ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return Empty(a.t, object, msgAndArgs...) +} + +// Emptyf asserts that the specified object is empty. I.e. nil, "", false, 0 or either +// a slice or a channel with len == 0. +// +// a.Emptyf(obj, "error message %s", "formatted") +func (a *Assertions) Emptyf(object interface{}, msg string, args ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return Emptyf(a.t, object, msg, args...) +} + +// Equal asserts that two objects are equal. +// +// a.Equal(123, 123) +// +// Pointer variable equality is determined based on the equality of the +// referenced values (as opposed to the memory addresses). Function equality +// cannot be determined and will always fail. +func (a *Assertions) Equal(expected interface{}, actual interface{}, msgAndArgs ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return Equal(a.t, expected, actual, msgAndArgs...) +} + +// EqualError asserts that a function returned an error (i.e. not `nil`) +// and that it is equal to the provided error. +// +// actualObj, err := SomeFunction() +// a.EqualError(err, expectedErrorString) +func (a *Assertions) EqualError(theError error, errString string, msgAndArgs ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return EqualError(a.t, theError, errString, msgAndArgs...) +} + +// EqualErrorf asserts that a function returned an error (i.e. not `nil`) +// and that it is equal to the provided error. +// +// actualObj, err := SomeFunction() +// a.EqualErrorf(err, expectedErrorString, "error message %s", "formatted") +func (a *Assertions) EqualErrorf(theError error, errString string, msg string, args ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return EqualErrorf(a.t, theError, errString, msg, args...) +} + +// EqualValues asserts that two objects are equal or convertable to the same types +// and equal. +// +// a.EqualValues(uint32(123), int32(123)) +func (a *Assertions) EqualValues(expected interface{}, actual interface{}, msgAndArgs ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return EqualValues(a.t, expected, actual, msgAndArgs...) +} + +// EqualValuesf asserts that two objects are equal or convertable to the same types +// and equal. +// +// a.EqualValuesf(uint32(123), int32(123), "error message %s", "formatted") +func (a *Assertions) EqualValuesf(expected interface{}, actual interface{}, msg string, args ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return EqualValuesf(a.t, expected, actual, msg, args...) +} + +// Equalf asserts that two objects are equal. +// +// a.Equalf(123, 123, "error message %s", "formatted") +// +// Pointer variable equality is determined based on the equality of the +// referenced values (as opposed to the memory addresses). Function equality +// cannot be determined and will always fail. +func (a *Assertions) Equalf(expected interface{}, actual interface{}, msg string, args ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return Equalf(a.t, expected, actual, msg, args...) +} + +// Error asserts that a function returned an error (i.e. not `nil`). +// +// actualObj, err := SomeFunction() +// if a.Error(err) { +// assert.Equal(t, expectedError, err) +// } +func (a *Assertions) Error(err error, msgAndArgs ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return Error(a.t, err, msgAndArgs...) +} + +// ErrorAs asserts that at least one of the errors in err's chain matches target, and if so, sets target to that error value. +// This is a wrapper for errors.As. +func (a *Assertions) ErrorAs(err error, target interface{}, msgAndArgs ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return ErrorAs(a.t, err, target, msgAndArgs...) +} + +// ErrorAsf asserts that at least one of the errors in err's chain matches target, and if so, sets target to that error value. +// This is a wrapper for errors.As. +func (a *Assertions) ErrorAsf(err error, target interface{}, msg string, args ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return ErrorAsf(a.t, err, target, msg, args...) +} + +// ErrorIs asserts that at least one of the errors in err's chain matches target. +// This is a wrapper for errors.Is. +func (a *Assertions) ErrorIs(err error, target error, msgAndArgs ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return ErrorIs(a.t, err, target, msgAndArgs...) +} + +// ErrorIsf asserts that at least one of the errors in err's chain matches target. +// This is a wrapper for errors.Is. +func (a *Assertions) ErrorIsf(err error, target error, msg string, args ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return ErrorIsf(a.t, err, target, msg, args...) +} + +// Errorf asserts that a function returned an error (i.e. not `nil`). +// +// actualObj, err := SomeFunction() +// if a.Errorf(err, "error message %s", "formatted") { +// assert.Equal(t, expectedErrorf, err) +// } +func (a *Assertions) Errorf(err error, msg string, args ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return Errorf(a.t, err, msg, args...) +} + +// Eventually asserts that given condition will be met in waitFor time, +// periodically checking target function each tick. +// +// a.Eventually(func() bool { return true; }, time.Second, 10*time.Millisecond) +func (a *Assertions) Eventually(condition func() bool, waitFor time.Duration, tick time.Duration, msgAndArgs ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return Eventually(a.t, condition, waitFor, tick, msgAndArgs...) +} + +// Eventuallyf asserts that given condition will be met in waitFor time, +// periodically checking target function each tick. +// +// a.Eventuallyf(func() bool { return true; }, time.Second, 10*time.Millisecond, "error message %s", "formatted") +func (a *Assertions) Eventuallyf(condition func() bool, waitFor time.Duration, tick time.Duration, msg string, args ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return Eventuallyf(a.t, condition, waitFor, tick, msg, args...) +} + +// Exactly asserts that two objects are equal in value and type. +// +// a.Exactly(int32(123), int64(123)) +func (a *Assertions) Exactly(expected interface{}, actual interface{}, msgAndArgs ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return Exactly(a.t, expected, actual, msgAndArgs...) +} + +// Exactlyf asserts that two objects are equal in value and type. +// +// a.Exactlyf(int32(123), int64(123), "error message %s", "formatted") +func (a *Assertions) Exactlyf(expected interface{}, actual interface{}, msg string, args ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return Exactlyf(a.t, expected, actual, msg, args...) +} + +// Fail reports a failure through +func (a *Assertions) Fail(failureMessage string, msgAndArgs ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return Fail(a.t, failureMessage, msgAndArgs...) +} + +// FailNow fails test +func (a *Assertions) FailNow(failureMessage string, msgAndArgs ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return FailNow(a.t, failureMessage, msgAndArgs...) +} + +// FailNowf fails test +func (a *Assertions) FailNowf(failureMessage string, msg string, args ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return FailNowf(a.t, failureMessage, msg, args...) +} + +// Failf reports a failure through +func (a *Assertions) Failf(failureMessage string, msg string, args ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return Failf(a.t, failureMessage, msg, args...) +} + +// False asserts that the specified value is false. +// +// a.False(myBool) +func (a *Assertions) False(value bool, msgAndArgs ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return False(a.t, value, msgAndArgs...) +} + +// Falsef asserts that the specified value is false. +// +// a.Falsef(myBool, "error message %s", "formatted") +func (a *Assertions) Falsef(value bool, msg string, args ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return Falsef(a.t, value, msg, args...) +} + +// FileExists checks whether a file exists in the given path. It also fails if +// the path points to a directory or there is an error when trying to check the file. +func (a *Assertions) FileExists(path string, msgAndArgs ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return FileExists(a.t, path, msgAndArgs...) +} + +// FileExistsf checks whether a file exists in the given path. It also fails if +// the path points to a directory or there is an error when trying to check the file. +func (a *Assertions) FileExistsf(path string, msg string, args ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return FileExistsf(a.t, path, msg, args...) +} + +// Greater asserts that the first element is greater than the second +// +// a.Greater(2, 1) +// a.Greater(float64(2), float64(1)) +// a.Greater("b", "a") +func (a *Assertions) Greater(e1 interface{}, e2 interface{}, msgAndArgs ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return Greater(a.t, e1, e2, msgAndArgs...) +} + +// GreaterOrEqual asserts that the first element is greater than or equal to the second +// +// a.GreaterOrEqual(2, 1) +// a.GreaterOrEqual(2, 2) +// a.GreaterOrEqual("b", "a") +// a.GreaterOrEqual("b", "b") +func (a *Assertions) GreaterOrEqual(e1 interface{}, e2 interface{}, msgAndArgs ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return GreaterOrEqual(a.t, e1, e2, msgAndArgs...) +} + +// GreaterOrEqualf asserts that the first element is greater than or equal to the second +// +// a.GreaterOrEqualf(2, 1, "error message %s", "formatted") +// a.GreaterOrEqualf(2, 2, "error message %s", "formatted") +// a.GreaterOrEqualf("b", "a", "error message %s", "formatted") +// a.GreaterOrEqualf("b", "b", "error message %s", "formatted") +func (a *Assertions) GreaterOrEqualf(e1 interface{}, e2 interface{}, msg string, args ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return GreaterOrEqualf(a.t, e1, e2, msg, args...) +} + +// Greaterf asserts that the first element is greater than the second +// +// a.Greaterf(2, 1, "error message %s", "formatted") +// a.Greaterf(float64(2), float64(1), "error message %s", "formatted") +// a.Greaterf("b", "a", "error message %s", "formatted") +func (a *Assertions) Greaterf(e1 interface{}, e2 interface{}, msg string, args ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return Greaterf(a.t, e1, e2, msg, args...) +} + +// HTTPBodyContains asserts that a specified handler returns a +// body that contains a string. +// +// a.HTTPBodyContains(myHandler, "GET", "www.google.com", nil, "I'm Feeling Lucky") +// +// Returns whether the assertion was successful (true) or not (false). +func (a *Assertions) HTTPBodyContains(handler http.HandlerFunc, method string, url string, values url.Values, str interface{}, msgAndArgs ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return HTTPBodyContains(a.t, handler, method, url, values, str, msgAndArgs...) +} + +// HTTPBodyContainsf asserts that a specified handler returns a +// body that contains a string. +// +// a.HTTPBodyContainsf(myHandler, "GET", "www.google.com", nil, "I'm Feeling Lucky", "error message %s", "formatted") +// +// Returns whether the assertion was successful (true) or not (false). +func (a *Assertions) HTTPBodyContainsf(handler http.HandlerFunc, method string, url string, values url.Values, str interface{}, msg string, args ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return HTTPBodyContainsf(a.t, handler, method, url, values, str, msg, args...) +} + +// HTTPBodyNotContains asserts that a specified handler returns a +// body that does not contain a string. +// +// a.HTTPBodyNotContains(myHandler, "GET", "www.google.com", nil, "I'm Feeling Lucky") +// +// Returns whether the assertion was successful (true) or not (false). +func (a *Assertions) HTTPBodyNotContains(handler http.HandlerFunc, method string, url string, values url.Values, str interface{}, msgAndArgs ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return HTTPBodyNotContains(a.t, handler, method, url, values, str, msgAndArgs...) +} + +// HTTPBodyNotContainsf asserts that a specified handler returns a +// body that does not contain a string. +// +// a.HTTPBodyNotContainsf(myHandler, "GET", "www.google.com", nil, "I'm Feeling Lucky", "error message %s", "formatted") +// +// Returns whether the assertion was successful (true) or not (false). +func (a *Assertions) HTTPBodyNotContainsf(handler http.HandlerFunc, method string, url string, values url.Values, str interface{}, msg string, args ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return HTTPBodyNotContainsf(a.t, handler, method, url, values, str, msg, args...) +} + +// HTTPError asserts that a specified handler returns an error status code. +// +// a.HTTPError(myHandler, "POST", "/a/b/c", url.Values{"a": []string{"b", "c"}} +// +// Returns whether the assertion was successful (true) or not (false). +func (a *Assertions) HTTPError(handler http.HandlerFunc, method string, url string, values url.Values, msgAndArgs ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return HTTPError(a.t, handler, method, url, values, msgAndArgs...) +} + +// HTTPErrorf asserts that a specified handler returns an error status code. +// +// a.HTTPErrorf(myHandler, "POST", "/a/b/c", url.Values{"a": []string{"b", "c"}} +// +// Returns whether the assertion was successful (true) or not (false). +func (a *Assertions) HTTPErrorf(handler http.HandlerFunc, method string, url string, values url.Values, msg string, args ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return HTTPErrorf(a.t, handler, method, url, values, msg, args...) +} + +// HTTPRedirect asserts that a specified handler returns a redirect status code. +// +// a.HTTPRedirect(myHandler, "GET", "/a/b/c", url.Values{"a": []string{"b", "c"}} +// +// Returns whether the assertion was successful (true) or not (false). +func (a *Assertions) HTTPRedirect(handler http.HandlerFunc, method string, url string, values url.Values, msgAndArgs ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return HTTPRedirect(a.t, handler, method, url, values, msgAndArgs...) +} + +// HTTPRedirectf asserts that a specified handler returns a redirect status code. +// +// a.HTTPRedirectf(myHandler, "GET", "/a/b/c", url.Values{"a": []string{"b", "c"}} +// +// Returns whether the assertion was successful (true) or not (false). +func (a *Assertions) HTTPRedirectf(handler http.HandlerFunc, method string, url string, values url.Values, msg string, args ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return HTTPRedirectf(a.t, handler, method, url, values, msg, args...) +} + +// HTTPStatusCode asserts that a specified handler returns a specified status code. +// +// a.HTTPStatusCode(myHandler, "GET", "/notImplemented", nil, 501) +// +// Returns whether the assertion was successful (true) or not (false). +func (a *Assertions) HTTPStatusCode(handler http.HandlerFunc, method string, url string, values url.Values, statuscode int, msgAndArgs ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return HTTPStatusCode(a.t, handler, method, url, values, statuscode, msgAndArgs...) +} + +// HTTPStatusCodef asserts that a specified handler returns a specified status code. +// +// a.HTTPStatusCodef(myHandler, "GET", "/notImplemented", nil, 501, "error message %s", "formatted") +// +// Returns whether the assertion was successful (true) or not (false). +func (a *Assertions) HTTPStatusCodef(handler http.HandlerFunc, method string, url string, values url.Values, statuscode int, msg string, args ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return HTTPStatusCodef(a.t, handler, method, url, values, statuscode, msg, args...) +} + +// HTTPSuccess asserts that a specified handler returns a success status code. +// +// a.HTTPSuccess(myHandler, "POST", "http://www.google.com", nil) +// +// Returns whether the assertion was successful (true) or not (false). +func (a *Assertions) HTTPSuccess(handler http.HandlerFunc, method string, url string, values url.Values, msgAndArgs ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return HTTPSuccess(a.t, handler, method, url, values, msgAndArgs...) +} + +// HTTPSuccessf asserts that a specified handler returns a success status code. +// +// a.HTTPSuccessf(myHandler, "POST", "http://www.google.com", nil, "error message %s", "formatted") +// +// Returns whether the assertion was successful (true) or not (false). +func (a *Assertions) HTTPSuccessf(handler http.HandlerFunc, method string, url string, values url.Values, msg string, args ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return HTTPSuccessf(a.t, handler, method, url, values, msg, args...) +} + +// Implements asserts that an object is implemented by the specified interface. +// +// a.Implements((*MyInterface)(nil), new(MyObject)) +func (a *Assertions) Implements(interfaceObject interface{}, object interface{}, msgAndArgs ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return Implements(a.t, interfaceObject, object, msgAndArgs...) +} + +// Implementsf asserts that an object is implemented by the specified interface. +// +// a.Implementsf((*MyInterface)(nil), new(MyObject), "error message %s", "formatted") +func (a *Assertions) Implementsf(interfaceObject interface{}, object interface{}, msg string, args ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return Implementsf(a.t, interfaceObject, object, msg, args...) +} + +// InDelta asserts that the two numerals are within delta of each other. +// +// a.InDelta(math.Pi, 22/7.0, 0.01) +func (a *Assertions) InDelta(expected interface{}, actual interface{}, delta float64, msgAndArgs ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return InDelta(a.t, expected, actual, delta, msgAndArgs...) +} + +// InDeltaMapValues is the same as InDelta, but it compares all values between two maps. Both maps must have exactly the same keys. +func (a *Assertions) InDeltaMapValues(expected interface{}, actual interface{}, delta float64, msgAndArgs ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return InDeltaMapValues(a.t, expected, actual, delta, msgAndArgs...) +} + +// InDeltaMapValuesf is the same as InDelta, but it compares all values between two maps. Both maps must have exactly the same keys. +func (a *Assertions) InDeltaMapValuesf(expected interface{}, actual interface{}, delta float64, msg string, args ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return InDeltaMapValuesf(a.t, expected, actual, delta, msg, args...) +} + +// InDeltaSlice is the same as InDelta, except it compares two slices. +func (a *Assertions) InDeltaSlice(expected interface{}, actual interface{}, delta float64, msgAndArgs ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return InDeltaSlice(a.t, expected, actual, delta, msgAndArgs...) +} + +// InDeltaSlicef is the same as InDelta, except it compares two slices. +func (a *Assertions) InDeltaSlicef(expected interface{}, actual interface{}, delta float64, msg string, args ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return InDeltaSlicef(a.t, expected, actual, delta, msg, args...) +} + +// InDeltaf asserts that the two numerals are within delta of each other. +// +// a.InDeltaf(math.Pi, 22/7.0, 0.01, "error message %s", "formatted") +func (a *Assertions) InDeltaf(expected interface{}, actual interface{}, delta float64, msg string, args ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return InDeltaf(a.t, expected, actual, delta, msg, args...) +} + +// InEpsilon asserts that expected and actual have a relative error less than epsilon +func (a *Assertions) InEpsilon(expected interface{}, actual interface{}, epsilon float64, msgAndArgs ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return InEpsilon(a.t, expected, actual, epsilon, msgAndArgs...) +} + +// InEpsilonSlice is the same as InEpsilon, except it compares each value from two slices. +func (a *Assertions) InEpsilonSlice(expected interface{}, actual interface{}, epsilon float64, msgAndArgs ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return InEpsilonSlice(a.t, expected, actual, epsilon, msgAndArgs...) +} + +// InEpsilonSlicef is the same as InEpsilon, except it compares each value from two slices. +func (a *Assertions) InEpsilonSlicef(expected interface{}, actual interface{}, epsilon float64, msg string, args ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return InEpsilonSlicef(a.t, expected, actual, epsilon, msg, args...) +} + +// InEpsilonf asserts that expected and actual have a relative error less than epsilon +func (a *Assertions) InEpsilonf(expected interface{}, actual interface{}, epsilon float64, msg string, args ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return InEpsilonf(a.t, expected, actual, epsilon, msg, args...) +} + +// IsDecreasing asserts that the collection is decreasing +// +// a.IsDecreasing([]int{2, 1, 0}) +// a.IsDecreasing([]float{2, 1}) +// a.IsDecreasing([]string{"b", "a"}) +func (a *Assertions) IsDecreasing(object interface{}, msgAndArgs ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return IsDecreasing(a.t, object, msgAndArgs...) +} + +// IsDecreasingf asserts that the collection is decreasing +// +// a.IsDecreasingf([]int{2, 1, 0}, "error message %s", "formatted") +// a.IsDecreasingf([]float{2, 1}, "error message %s", "formatted") +// a.IsDecreasingf([]string{"b", "a"}, "error message %s", "formatted") +func (a *Assertions) IsDecreasingf(object interface{}, msg string, args ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return IsDecreasingf(a.t, object, msg, args...) +} + +// IsIncreasing asserts that the collection is increasing +// +// a.IsIncreasing([]int{1, 2, 3}) +// a.IsIncreasing([]float{1, 2}) +// a.IsIncreasing([]string{"a", "b"}) +func (a *Assertions) IsIncreasing(object interface{}, msgAndArgs ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return IsIncreasing(a.t, object, msgAndArgs...) +} + +// IsIncreasingf asserts that the collection is increasing +// +// a.IsIncreasingf([]int{1, 2, 3}, "error message %s", "formatted") +// a.IsIncreasingf([]float{1, 2}, "error message %s", "formatted") +// a.IsIncreasingf([]string{"a", "b"}, "error message %s", "formatted") +func (a *Assertions) IsIncreasingf(object interface{}, msg string, args ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return IsIncreasingf(a.t, object, msg, args...) +} + +// IsNonDecreasing asserts that the collection is not decreasing +// +// a.IsNonDecreasing([]int{1, 1, 2}) +// a.IsNonDecreasing([]float{1, 2}) +// a.IsNonDecreasing([]string{"a", "b"}) +func (a *Assertions) IsNonDecreasing(object interface{}, msgAndArgs ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return IsNonDecreasing(a.t, object, msgAndArgs...) +} + +// IsNonDecreasingf asserts that the collection is not decreasing +// +// a.IsNonDecreasingf([]int{1, 1, 2}, "error message %s", "formatted") +// a.IsNonDecreasingf([]float{1, 2}, "error message %s", "formatted") +// a.IsNonDecreasingf([]string{"a", "b"}, "error message %s", "formatted") +func (a *Assertions) IsNonDecreasingf(object interface{}, msg string, args ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return IsNonDecreasingf(a.t, object, msg, args...) +} + +// IsNonIncreasing asserts that the collection is not increasing +// +// a.IsNonIncreasing([]int{2, 1, 1}) +// a.IsNonIncreasing([]float{2, 1}) +// a.IsNonIncreasing([]string{"b", "a"}) +func (a *Assertions) IsNonIncreasing(object interface{}, msgAndArgs ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return IsNonIncreasing(a.t, object, msgAndArgs...) +} + +// IsNonIncreasingf asserts that the collection is not increasing +// +// a.IsNonIncreasingf([]int{2, 1, 1}, "error message %s", "formatted") +// a.IsNonIncreasingf([]float{2, 1}, "error message %s", "formatted") +// a.IsNonIncreasingf([]string{"b", "a"}, "error message %s", "formatted") +func (a *Assertions) IsNonIncreasingf(object interface{}, msg string, args ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return IsNonIncreasingf(a.t, object, msg, args...) +} + +// IsType asserts that the specified objects are of the same type. +func (a *Assertions) IsType(expectedType interface{}, object interface{}, msgAndArgs ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return IsType(a.t, expectedType, object, msgAndArgs...) +} + +// IsTypef asserts that the specified objects are of the same type. +func (a *Assertions) IsTypef(expectedType interface{}, object interface{}, msg string, args ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return IsTypef(a.t, expectedType, object, msg, args...) +} + +// JSONEq asserts that two JSON strings are equivalent. +// +// a.JSONEq(`{"hello": "world", "foo": "bar"}`, `{"foo": "bar", "hello": "world"}`) +func (a *Assertions) JSONEq(expected string, actual string, msgAndArgs ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return JSONEq(a.t, expected, actual, msgAndArgs...) +} + +// JSONEqf asserts that two JSON strings are equivalent. +// +// a.JSONEqf(`{"hello": "world", "foo": "bar"}`, `{"foo": "bar", "hello": "world"}`, "error message %s", "formatted") +func (a *Assertions) JSONEqf(expected string, actual string, msg string, args ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return JSONEqf(a.t, expected, actual, msg, args...) +} + +// Len asserts that the specified object has specific length. +// Len also fails if the object has a type that len() not accept. +// +// a.Len(mySlice, 3) +func (a *Assertions) Len(object interface{}, length int, msgAndArgs ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return Len(a.t, object, length, msgAndArgs...) +} + +// Lenf asserts that the specified object has specific length. +// Lenf also fails if the object has a type that len() not accept. +// +// a.Lenf(mySlice, 3, "error message %s", "formatted") +func (a *Assertions) Lenf(object interface{}, length int, msg string, args ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return Lenf(a.t, object, length, msg, args...) +} + +// Less asserts that the first element is less than the second +// +// a.Less(1, 2) +// a.Less(float64(1), float64(2)) +// a.Less("a", "b") +func (a *Assertions) Less(e1 interface{}, e2 interface{}, msgAndArgs ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return Less(a.t, e1, e2, msgAndArgs...) +} + +// LessOrEqual asserts that the first element is less than or equal to the second +// +// a.LessOrEqual(1, 2) +// a.LessOrEqual(2, 2) +// a.LessOrEqual("a", "b") +// a.LessOrEqual("b", "b") +func (a *Assertions) LessOrEqual(e1 interface{}, e2 interface{}, msgAndArgs ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return LessOrEqual(a.t, e1, e2, msgAndArgs...) +} + +// LessOrEqualf asserts that the first element is less than or equal to the second +// +// a.LessOrEqualf(1, 2, "error message %s", "formatted") +// a.LessOrEqualf(2, 2, "error message %s", "formatted") +// a.LessOrEqualf("a", "b", "error message %s", "formatted") +// a.LessOrEqualf("b", "b", "error message %s", "formatted") +func (a *Assertions) LessOrEqualf(e1 interface{}, e2 interface{}, msg string, args ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return LessOrEqualf(a.t, e1, e2, msg, args...) +} + +// Lessf asserts that the first element is less than the second +// +// a.Lessf(1, 2, "error message %s", "formatted") +// a.Lessf(float64(1), float64(2), "error message %s", "formatted") +// a.Lessf("a", "b", "error message %s", "formatted") +func (a *Assertions) Lessf(e1 interface{}, e2 interface{}, msg string, args ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return Lessf(a.t, e1, e2, msg, args...) +} + +// Negative asserts that the specified element is negative +// +// a.Negative(-1) +// a.Negative(-1.23) +func (a *Assertions) Negative(e interface{}, msgAndArgs ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return Negative(a.t, e, msgAndArgs...) +} + +// Negativef asserts that the specified element is negative +// +// a.Negativef(-1, "error message %s", "formatted") +// a.Negativef(-1.23, "error message %s", "formatted") +func (a *Assertions) Negativef(e interface{}, msg string, args ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return Negativef(a.t, e, msg, args...) +} + +// Never asserts that the given condition doesn't satisfy in waitFor time, +// periodically checking the target function each tick. +// +// a.Never(func() bool { return false; }, time.Second, 10*time.Millisecond) +func (a *Assertions) Never(condition func() bool, waitFor time.Duration, tick time.Duration, msgAndArgs ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return Never(a.t, condition, waitFor, tick, msgAndArgs...) +} + +// Neverf asserts that the given condition doesn't satisfy in waitFor time, +// periodically checking the target function each tick. +// +// a.Neverf(func() bool { return false; }, time.Second, 10*time.Millisecond, "error message %s", "formatted") +func (a *Assertions) Neverf(condition func() bool, waitFor time.Duration, tick time.Duration, msg string, args ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return Neverf(a.t, condition, waitFor, tick, msg, args...) +} + +// Nil asserts that the specified object is nil. +// +// a.Nil(err) +func (a *Assertions) Nil(object interface{}, msgAndArgs ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return Nil(a.t, object, msgAndArgs...) +} + +// Nilf asserts that the specified object is nil. +// +// a.Nilf(err, "error message %s", "formatted") +func (a *Assertions) Nilf(object interface{}, msg string, args ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return Nilf(a.t, object, msg, args...) +} + +// NoDirExists checks whether a directory does not exist in the given path. +// It fails if the path points to an existing _directory_ only. +func (a *Assertions) NoDirExists(path string, msgAndArgs ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return NoDirExists(a.t, path, msgAndArgs...) +} + +// NoDirExistsf checks whether a directory does not exist in the given path. +// It fails if the path points to an existing _directory_ only. +func (a *Assertions) NoDirExistsf(path string, msg string, args ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return NoDirExistsf(a.t, path, msg, args...) +} + +// NoError asserts that a function returned no error (i.e. `nil`). +// +// actualObj, err := SomeFunction() +// if a.NoError(err) { +// assert.Equal(t, expectedObj, actualObj) +// } +func (a *Assertions) NoError(err error, msgAndArgs ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return NoError(a.t, err, msgAndArgs...) +} + +// NoErrorf asserts that a function returned no error (i.e. `nil`). +// +// actualObj, err := SomeFunction() +// if a.NoErrorf(err, "error message %s", "formatted") { +// assert.Equal(t, expectedObj, actualObj) +// } +func (a *Assertions) NoErrorf(err error, msg string, args ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return NoErrorf(a.t, err, msg, args...) +} + +// NoFileExists checks whether a file does not exist in a given path. It fails +// if the path points to an existing _file_ only. +func (a *Assertions) NoFileExists(path string, msgAndArgs ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return NoFileExists(a.t, path, msgAndArgs...) +} + +// NoFileExistsf checks whether a file does not exist in a given path. It fails +// if the path points to an existing _file_ only. +func (a *Assertions) NoFileExistsf(path string, msg string, args ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return NoFileExistsf(a.t, path, msg, args...) +} + +// NotContains asserts that the specified string, list(array, slice...) or map does NOT contain the +// specified substring or element. +// +// a.NotContains("Hello World", "Earth") +// a.NotContains(["Hello", "World"], "Earth") +// a.NotContains({"Hello": "World"}, "Earth") +func (a *Assertions) NotContains(s interface{}, contains interface{}, msgAndArgs ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return NotContains(a.t, s, contains, msgAndArgs...) +} + +// NotContainsf asserts that the specified string, list(array, slice...) or map does NOT contain the +// specified substring or element. +// +// a.NotContainsf("Hello World", "Earth", "error message %s", "formatted") +// a.NotContainsf(["Hello", "World"], "Earth", "error message %s", "formatted") +// a.NotContainsf({"Hello": "World"}, "Earth", "error message %s", "formatted") +func (a *Assertions) NotContainsf(s interface{}, contains interface{}, msg string, args ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return NotContainsf(a.t, s, contains, msg, args...) +} + +// NotEmpty asserts that the specified object is NOT empty. I.e. not nil, "", false, 0 or either +// a slice or a channel with len == 0. +// +// if a.NotEmpty(obj) { +// assert.Equal(t, "two", obj[1]) +// } +func (a *Assertions) NotEmpty(object interface{}, msgAndArgs ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return NotEmpty(a.t, object, msgAndArgs...) +} + +// NotEmptyf asserts that the specified object is NOT empty. I.e. not nil, "", false, 0 or either +// a slice or a channel with len == 0. +// +// if a.NotEmptyf(obj, "error message %s", "formatted") { +// assert.Equal(t, "two", obj[1]) +// } +func (a *Assertions) NotEmptyf(object interface{}, msg string, args ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return NotEmptyf(a.t, object, msg, args...) +} + +// NotEqual asserts that the specified values are NOT equal. +// +// a.NotEqual(obj1, obj2) +// +// Pointer variable equality is determined based on the equality of the +// referenced values (as opposed to the memory addresses). +func (a *Assertions) NotEqual(expected interface{}, actual interface{}, msgAndArgs ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return NotEqual(a.t, expected, actual, msgAndArgs...) +} + +// NotEqualValues asserts that two objects are not equal even when converted to the same type +// +// a.NotEqualValues(obj1, obj2) +func (a *Assertions) NotEqualValues(expected interface{}, actual interface{}, msgAndArgs ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return NotEqualValues(a.t, expected, actual, msgAndArgs...) +} + +// NotEqualValuesf asserts that two objects are not equal even when converted to the same type +// +// a.NotEqualValuesf(obj1, obj2, "error message %s", "formatted") +func (a *Assertions) NotEqualValuesf(expected interface{}, actual interface{}, msg string, args ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return NotEqualValuesf(a.t, expected, actual, msg, args...) +} + +// NotEqualf asserts that the specified values are NOT equal. +// +// a.NotEqualf(obj1, obj2, "error message %s", "formatted") +// +// Pointer variable equality is determined based on the equality of the +// referenced values (as opposed to the memory addresses). +func (a *Assertions) NotEqualf(expected interface{}, actual interface{}, msg string, args ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return NotEqualf(a.t, expected, actual, msg, args...) +} + +// NotErrorIs asserts that at none of the errors in err's chain matches target. +// This is a wrapper for errors.Is. +func (a *Assertions) NotErrorIs(err error, target error, msgAndArgs ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return NotErrorIs(a.t, err, target, msgAndArgs...) +} + +// NotErrorIsf asserts that at none of the errors in err's chain matches target. +// This is a wrapper for errors.Is. +func (a *Assertions) NotErrorIsf(err error, target error, msg string, args ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return NotErrorIsf(a.t, err, target, msg, args...) +} + +// NotNil asserts that the specified object is not nil. +// +// a.NotNil(err) +func (a *Assertions) NotNil(object interface{}, msgAndArgs ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return NotNil(a.t, object, msgAndArgs...) +} + +// NotNilf asserts that the specified object is not nil. +// +// a.NotNilf(err, "error message %s", "formatted") +func (a *Assertions) NotNilf(object interface{}, msg string, args ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return NotNilf(a.t, object, msg, args...) +} + +// NotPanics asserts that the code inside the specified PanicTestFunc does NOT panic. +// +// a.NotPanics(func(){ RemainCalm() }) +func (a *Assertions) NotPanics(f PanicTestFunc, msgAndArgs ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return NotPanics(a.t, f, msgAndArgs...) +} + +// NotPanicsf asserts that the code inside the specified PanicTestFunc does NOT panic. +// +// a.NotPanicsf(func(){ RemainCalm() }, "error message %s", "formatted") +func (a *Assertions) NotPanicsf(f PanicTestFunc, msg string, args ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return NotPanicsf(a.t, f, msg, args...) +} + +// NotRegexp asserts that a specified regexp does not match a string. +// +// a.NotRegexp(regexp.MustCompile("starts"), "it's starting") +// a.NotRegexp("^start", "it's not starting") +func (a *Assertions) NotRegexp(rx interface{}, str interface{}, msgAndArgs ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return NotRegexp(a.t, rx, str, msgAndArgs...) +} + +// NotRegexpf asserts that a specified regexp does not match a string. +// +// a.NotRegexpf(regexp.MustCompile("starts"), "it's starting", "error message %s", "formatted") +// a.NotRegexpf("^start", "it's not starting", "error message %s", "formatted") +func (a *Assertions) NotRegexpf(rx interface{}, str interface{}, msg string, args ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return NotRegexpf(a.t, rx, str, msg, args...) +} + +// NotSame asserts that two pointers do not reference the same object. +// +// a.NotSame(ptr1, ptr2) +// +// Both arguments must be pointer variables. Pointer variable sameness is +// determined based on the equality of both type and value. +func (a *Assertions) NotSame(expected interface{}, actual interface{}, msgAndArgs ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return NotSame(a.t, expected, actual, msgAndArgs...) +} + +// NotSamef asserts that two pointers do not reference the same object. +// +// a.NotSamef(ptr1, ptr2, "error message %s", "formatted") +// +// Both arguments must be pointer variables. Pointer variable sameness is +// determined based on the equality of both type and value. +func (a *Assertions) NotSamef(expected interface{}, actual interface{}, msg string, args ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return NotSamef(a.t, expected, actual, msg, args...) +} + +// NotSubset asserts that the specified list(array, slice...) contains not all +// elements given in the specified subset(array, slice...). +// +// a.NotSubset([1, 3, 4], [1, 2], "But [1, 3, 4] does not contain [1, 2]") +func (a *Assertions) NotSubset(list interface{}, subset interface{}, msgAndArgs ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return NotSubset(a.t, list, subset, msgAndArgs...) +} + +// NotSubsetf asserts that the specified list(array, slice...) contains not all +// elements given in the specified subset(array, slice...). +// +// a.NotSubsetf([1, 3, 4], [1, 2], "But [1, 3, 4] does not contain [1, 2]", "error message %s", "formatted") +func (a *Assertions) NotSubsetf(list interface{}, subset interface{}, msg string, args ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return NotSubsetf(a.t, list, subset, msg, args...) +} + +// NotZero asserts that i is not the zero value for its type. +func (a *Assertions) NotZero(i interface{}, msgAndArgs ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return NotZero(a.t, i, msgAndArgs...) +} + +// NotZerof asserts that i is not the zero value for its type. +func (a *Assertions) NotZerof(i interface{}, msg string, args ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return NotZerof(a.t, i, msg, args...) +} + +// Panics asserts that the code inside the specified PanicTestFunc panics. +// +// a.Panics(func(){ GoCrazy() }) +func (a *Assertions) Panics(f PanicTestFunc, msgAndArgs ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return Panics(a.t, f, msgAndArgs...) +} + +// PanicsWithError asserts that the code inside the specified PanicTestFunc +// panics, and that the recovered panic value is an error that satisfies the +// EqualError comparison. +// +// a.PanicsWithError("crazy error", func(){ GoCrazy() }) +func (a *Assertions) PanicsWithError(errString string, f PanicTestFunc, msgAndArgs ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return PanicsWithError(a.t, errString, f, msgAndArgs...) +} + +// PanicsWithErrorf asserts that the code inside the specified PanicTestFunc +// panics, and that the recovered panic value is an error that satisfies the +// EqualError comparison. +// +// a.PanicsWithErrorf("crazy error", func(){ GoCrazy() }, "error message %s", "formatted") +func (a *Assertions) PanicsWithErrorf(errString string, f PanicTestFunc, msg string, args ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return PanicsWithErrorf(a.t, errString, f, msg, args...) +} + +// PanicsWithValue asserts that the code inside the specified PanicTestFunc panics, and that +// the recovered panic value equals the expected panic value. +// +// a.PanicsWithValue("crazy error", func(){ GoCrazy() }) +func (a *Assertions) PanicsWithValue(expected interface{}, f PanicTestFunc, msgAndArgs ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return PanicsWithValue(a.t, expected, f, msgAndArgs...) +} + +// PanicsWithValuef asserts that the code inside the specified PanicTestFunc panics, and that +// the recovered panic value equals the expected panic value. +// +// a.PanicsWithValuef("crazy error", func(){ GoCrazy() }, "error message %s", "formatted") +func (a *Assertions) PanicsWithValuef(expected interface{}, f PanicTestFunc, msg string, args ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return PanicsWithValuef(a.t, expected, f, msg, args...) +} + +// Panicsf asserts that the code inside the specified PanicTestFunc panics. +// +// a.Panicsf(func(){ GoCrazy() }, "error message %s", "formatted") +func (a *Assertions) Panicsf(f PanicTestFunc, msg string, args ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return Panicsf(a.t, f, msg, args...) +} + +// Positive asserts that the specified element is positive +// +// a.Positive(1) +// a.Positive(1.23) +func (a *Assertions) Positive(e interface{}, msgAndArgs ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return Positive(a.t, e, msgAndArgs...) +} + +// Positivef asserts that the specified element is positive +// +// a.Positivef(1, "error message %s", "formatted") +// a.Positivef(1.23, "error message %s", "formatted") +func (a *Assertions) Positivef(e interface{}, msg string, args ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return Positivef(a.t, e, msg, args...) +} + +// Regexp asserts that a specified regexp matches a string. +// +// a.Regexp(regexp.MustCompile("start"), "it's starting") +// a.Regexp("start...$", "it's not starting") +func (a *Assertions) Regexp(rx interface{}, str interface{}, msgAndArgs ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return Regexp(a.t, rx, str, msgAndArgs...) +} + +// Regexpf asserts that a specified regexp matches a string. +// +// a.Regexpf(regexp.MustCompile("start"), "it's starting", "error message %s", "formatted") +// a.Regexpf("start...$", "it's not starting", "error message %s", "formatted") +func (a *Assertions) Regexpf(rx interface{}, str interface{}, msg string, args ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return Regexpf(a.t, rx, str, msg, args...) +} + +// Same asserts that two pointers reference the same object. +// +// a.Same(ptr1, ptr2) +// +// Both arguments must be pointer variables. Pointer variable sameness is +// determined based on the equality of both type and value. +func (a *Assertions) Same(expected interface{}, actual interface{}, msgAndArgs ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return Same(a.t, expected, actual, msgAndArgs...) +} + +// Samef asserts that two pointers reference the same object. +// +// a.Samef(ptr1, ptr2, "error message %s", "formatted") +// +// Both arguments must be pointer variables. Pointer variable sameness is +// determined based on the equality of both type and value. +func (a *Assertions) Samef(expected interface{}, actual interface{}, msg string, args ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return Samef(a.t, expected, actual, msg, args...) +} + +// Subset asserts that the specified list(array, slice...) contains all +// elements given in the specified subset(array, slice...). +// +// a.Subset([1, 2, 3], [1, 2], "But [1, 2, 3] does contain [1, 2]") +func (a *Assertions) Subset(list interface{}, subset interface{}, msgAndArgs ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return Subset(a.t, list, subset, msgAndArgs...) +} + +// Subsetf asserts that the specified list(array, slice...) contains all +// elements given in the specified subset(array, slice...). +// +// a.Subsetf([1, 2, 3], [1, 2], "But [1, 2, 3] does contain [1, 2]", "error message %s", "formatted") +func (a *Assertions) Subsetf(list interface{}, subset interface{}, msg string, args ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return Subsetf(a.t, list, subset, msg, args...) +} + +// True asserts that the specified value is true. +// +// a.True(myBool) +func (a *Assertions) True(value bool, msgAndArgs ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return True(a.t, value, msgAndArgs...) +} + +// Truef asserts that the specified value is true. +// +// a.Truef(myBool, "error message %s", "formatted") +func (a *Assertions) Truef(value bool, msg string, args ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return Truef(a.t, value, msg, args...) +} + +// WithinDuration asserts that the two times are within duration delta of each other. +// +// a.WithinDuration(time.Now(), time.Now(), 10*time.Second) +func (a *Assertions) WithinDuration(expected time.Time, actual time.Time, delta time.Duration, msgAndArgs ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return WithinDuration(a.t, expected, actual, delta, msgAndArgs...) +} + +// WithinDurationf asserts that the two times are within duration delta of each other. +// +// a.WithinDurationf(time.Now(), time.Now(), 10*time.Second, "error message %s", "formatted") +func (a *Assertions) WithinDurationf(expected time.Time, actual time.Time, delta time.Duration, msg string, args ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return WithinDurationf(a.t, expected, actual, delta, msg, args...) +} + +// YAMLEq asserts that two YAML strings are equivalent. +func (a *Assertions) YAMLEq(expected string, actual string, msgAndArgs ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return YAMLEq(a.t, expected, actual, msgAndArgs...) +} + +// YAMLEqf asserts that two YAML strings are equivalent. +func (a *Assertions) YAMLEqf(expected string, actual string, msg string, args ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return YAMLEqf(a.t, expected, actual, msg, args...) +} + +// Zero asserts that i is the zero value for its type. +func (a *Assertions) Zero(i interface{}, msgAndArgs ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return Zero(a.t, i, msgAndArgs...) +} + +// Zerof asserts that i is the zero value for its type. +func (a *Assertions) Zerof(i interface{}, msg string, args ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return Zerof(a.t, i, msg, args...) +} diff --git a/vendor/github.com/stretchr/testify/assert/assertion_forward.go.tmpl b/vendor/github.com/stretchr/testify/assert/assertion_forward.go.tmpl new file mode 100644 index 00000000..188bb9e1 --- /dev/null +++ b/vendor/github.com/stretchr/testify/assert/assertion_forward.go.tmpl @@ -0,0 +1,5 @@ +{{.CommentWithoutT "a"}} +func (a *Assertions) {{.DocInfo.Name}}({{.Params}}) bool { + if h, ok := a.t.(tHelper); ok { h.Helper() } + return {{.DocInfo.Name}}(a.t, {{.ForwardedParams}}) +} diff --git a/vendor/github.com/stretchr/testify/assert/assertion_order.go b/vendor/github.com/stretchr/testify/assert/assertion_order.go new file mode 100644 index 00000000..1c3b4718 --- /dev/null +++ b/vendor/github.com/stretchr/testify/assert/assertion_order.go @@ -0,0 +1,81 @@ +package assert + +import ( + "fmt" + "reflect" +) + +// isOrdered checks that collection contains orderable elements. +func isOrdered(t TestingT, object interface{}, allowedComparesResults []CompareType, failMessage string, msgAndArgs ...interface{}) bool { + objKind := reflect.TypeOf(object).Kind() + if objKind != reflect.Slice && objKind != reflect.Array { + return false + } + + objValue := reflect.ValueOf(object) + objLen := objValue.Len() + + if objLen <= 1 { + return true + } + + value := objValue.Index(0) + valueInterface := value.Interface() + firstValueKind := value.Kind() + + for i := 1; i < objLen; i++ { + prevValue := value + prevValueInterface := valueInterface + + value = objValue.Index(i) + valueInterface = value.Interface() + + compareResult, isComparable := compare(prevValueInterface, valueInterface, firstValueKind) + + if !isComparable { + return Fail(t, fmt.Sprintf("Can not compare type \"%s\" and \"%s\"", reflect.TypeOf(value), reflect.TypeOf(prevValue)), msgAndArgs...) + } + + if !containsValue(allowedComparesResults, compareResult) { + return Fail(t, fmt.Sprintf(failMessage, prevValue, value), msgAndArgs...) + } + } + + return true +} + +// IsIncreasing asserts that the collection is increasing +// +// assert.IsIncreasing(t, []int{1, 2, 3}) +// assert.IsIncreasing(t, []float{1, 2}) +// assert.IsIncreasing(t, []string{"a", "b"}) +func IsIncreasing(t TestingT, object interface{}, msgAndArgs ...interface{}) bool { + return isOrdered(t, object, []CompareType{compareLess}, "\"%v\" is not less than \"%v\"", msgAndArgs) +} + +// IsNonIncreasing asserts that the collection is not increasing +// +// assert.IsNonIncreasing(t, []int{2, 1, 1}) +// assert.IsNonIncreasing(t, []float{2, 1}) +// assert.IsNonIncreasing(t, []string{"b", "a"}) +func IsNonIncreasing(t TestingT, object interface{}, msgAndArgs ...interface{}) bool { + return isOrdered(t, object, []CompareType{compareEqual, compareGreater}, "\"%v\" is not greater than or equal to \"%v\"", msgAndArgs) +} + +// IsDecreasing asserts that the collection is decreasing +// +// assert.IsDecreasing(t, []int{2, 1, 0}) +// assert.IsDecreasing(t, []float{2, 1}) +// assert.IsDecreasing(t, []string{"b", "a"}) +func IsDecreasing(t TestingT, object interface{}, msgAndArgs ...interface{}) bool { + return isOrdered(t, object, []CompareType{compareGreater}, "\"%v\" is not greater than \"%v\"", msgAndArgs) +} + +// IsNonDecreasing asserts that the collection is not decreasing +// +// assert.IsNonDecreasing(t, []int{1, 1, 2}) +// assert.IsNonDecreasing(t, []float{1, 2}) +// assert.IsNonDecreasing(t, []string{"a", "b"}) +func IsNonDecreasing(t TestingT, object interface{}, msgAndArgs ...interface{}) bool { + return isOrdered(t, object, []CompareType{compareLess, compareEqual}, "\"%v\" is not less than or equal to \"%v\"", msgAndArgs) +} diff --git a/vendor/github.com/stretchr/testify/assert/assertions.go b/vendor/github.com/stretchr/testify/assert/assertions.go new file mode 100644 index 00000000..bcac4401 --- /dev/null +++ b/vendor/github.com/stretchr/testify/assert/assertions.go @@ -0,0 +1,1774 @@ +package assert + +import ( + "bufio" + "bytes" + "encoding/json" + "errors" + "fmt" + "math" + "os" + "reflect" + "regexp" + "runtime" + "runtime/debug" + "strings" + "time" + "unicode" + "unicode/utf8" + + "github.com/davecgh/go-spew/spew" + "github.com/pmezard/go-difflib/difflib" + yaml "gopkg.in/yaml.v3" +) + +//go:generate sh -c "cd ../_codegen && go build && cd - && ../_codegen/_codegen -output-package=assert -template=assertion_format.go.tmpl" + +// TestingT is an interface wrapper around *testing.T +type TestingT interface { + Errorf(format string, args ...interface{}) +} + +// ComparisonAssertionFunc is a common function prototype when comparing two values. Can be useful +// for table driven tests. +type ComparisonAssertionFunc func(TestingT, interface{}, interface{}, ...interface{}) bool + +// ValueAssertionFunc is a common function prototype when validating a single value. Can be useful +// for table driven tests. +type ValueAssertionFunc func(TestingT, interface{}, ...interface{}) bool + +// BoolAssertionFunc is a common function prototype when validating a bool value. Can be useful +// for table driven tests. +type BoolAssertionFunc func(TestingT, bool, ...interface{}) bool + +// ErrorAssertionFunc is a common function prototype when validating an error value. Can be useful +// for table driven tests. +type ErrorAssertionFunc func(TestingT, error, ...interface{}) bool + +// Comparison is a custom function that returns true on success and false on failure +type Comparison func() (success bool) + +/* + Helper functions +*/ + +// ObjectsAreEqual determines if two objects are considered equal. +// +// This function does no assertion of any kind. +func ObjectsAreEqual(expected, actual interface{}) bool { + if expected == nil || actual == nil { + return expected == actual + } + + exp, ok := expected.([]byte) + if !ok { + return reflect.DeepEqual(expected, actual) + } + + act, ok := actual.([]byte) + if !ok { + return false + } + if exp == nil || act == nil { + return exp == nil && act == nil + } + return bytes.Equal(exp, act) +} + +// ObjectsAreEqualValues gets whether two objects are equal, or if their +// values are equal. +func ObjectsAreEqualValues(expected, actual interface{}) bool { + if ObjectsAreEqual(expected, actual) { + return true + } + + actualType := reflect.TypeOf(actual) + if actualType == nil { + return false + } + expectedValue := reflect.ValueOf(expected) + if expectedValue.IsValid() && expectedValue.Type().ConvertibleTo(actualType) { + // Attempt comparison after type conversion + return reflect.DeepEqual(expectedValue.Convert(actualType).Interface(), actual) + } + + return false +} + +/* CallerInfo is necessary because the assert functions use the testing object +internally, causing it to print the file:line of the assert method, rather than where +the problem actually occurred in calling code.*/ + +// CallerInfo returns an array of strings containing the file and line number +// of each stack frame leading from the current test to the assert call that +// failed. +func CallerInfo() []string { + + var pc uintptr + var ok bool + var file string + var line int + var name string + + callers := []string{} + for i := 0; ; i++ { + pc, file, line, ok = runtime.Caller(i) + if !ok { + // The breaks below failed to terminate the loop, and we ran off the + // end of the call stack. + break + } + + // This is a huge edge case, but it will panic if this is the case, see #180 + if file == "" { + break + } + + f := runtime.FuncForPC(pc) + if f == nil { + break + } + name = f.Name() + + // testing.tRunner is the standard library function that calls + // tests. Subtests are called directly by tRunner, without going through + // the Test/Benchmark/Example function that contains the t.Run calls, so + // with subtests we should break when we hit tRunner, without adding it + // to the list of callers. + if name == "testing.tRunner" { + break + } + + parts := strings.Split(file, "/") + file = parts[len(parts)-1] + if len(parts) > 1 { + dir := parts[len(parts)-2] + if (dir != "assert" && dir != "mock" && dir != "require") || file == "mock_test.go" { + callers = append(callers, fmt.Sprintf("%s:%d", file, line)) + } + } + + // Drop the package + segments := strings.Split(name, ".") + name = segments[len(segments)-1] + if isTest(name, "Test") || + isTest(name, "Benchmark") || + isTest(name, "Example") { + break + } + } + + return callers +} + +// Stolen from the `go test` tool. +// isTest tells whether name looks like a test (or benchmark, according to prefix). +// It is a Test (say) if there is a character after Test that is not a lower-case letter. +// We don't want TesticularCancer. +func isTest(name, prefix string) bool { + if !strings.HasPrefix(name, prefix) { + return false + } + if len(name) == len(prefix) { // "Test" is ok + return true + } + r, _ := utf8.DecodeRuneInString(name[len(prefix):]) + return !unicode.IsLower(r) +} + +func messageFromMsgAndArgs(msgAndArgs ...interface{}) string { + if len(msgAndArgs) == 0 || msgAndArgs == nil { + return "" + } + if len(msgAndArgs) == 1 { + msg := msgAndArgs[0] + if msgAsStr, ok := msg.(string); ok { + return msgAsStr + } + return fmt.Sprintf("%+v", msg) + } + if len(msgAndArgs) > 1 { + return fmt.Sprintf(msgAndArgs[0].(string), msgAndArgs[1:]...) + } + return "" +} + +// Aligns the provided message so that all lines after the first line start at the same location as the first line. +// Assumes that the first line starts at the correct location (after carriage return, tab, label, spacer and tab). +// The longestLabelLen parameter specifies the length of the longest label in the output (required becaues this is the +// basis on which the alignment occurs). +func indentMessageLines(message string, longestLabelLen int) string { + outBuf := new(bytes.Buffer) + + for i, scanner := 0, bufio.NewScanner(strings.NewReader(message)); scanner.Scan(); i++ { + // no need to align first line because it starts at the correct location (after the label) + if i != 0 { + // append alignLen+1 spaces to align with "{{longestLabel}}:" before adding tab + outBuf.WriteString("\n\t" + strings.Repeat(" ", longestLabelLen+1) + "\t") + } + outBuf.WriteString(scanner.Text()) + } + + return outBuf.String() +} + +type failNower interface { + FailNow() +} + +// FailNow fails test +func FailNow(t TestingT, failureMessage string, msgAndArgs ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + Fail(t, failureMessage, msgAndArgs...) + + // We cannot extend TestingT with FailNow() and + // maintain backwards compatibility, so we fallback + // to panicking when FailNow is not available in + // TestingT. + // See issue #263 + + if t, ok := t.(failNower); ok { + t.FailNow() + } else { + panic("test failed and t is missing `FailNow()`") + } + return false +} + +// Fail reports a failure through +func Fail(t TestingT, failureMessage string, msgAndArgs ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + content := []labeledContent{ + {"Error Trace", strings.Join(CallerInfo(), "\n\t\t\t")}, + {"Error", failureMessage}, + } + + // Add test name if the Go version supports it + if n, ok := t.(interface { + Name() string + }); ok { + content = append(content, labeledContent{"Test", n.Name()}) + } + + message := messageFromMsgAndArgs(msgAndArgs...) + if len(message) > 0 { + content = append(content, labeledContent{"Messages", message}) + } + + t.Errorf("\n%s", ""+labeledOutput(content...)) + + return false +} + +type labeledContent struct { + label string + content string +} + +// labeledOutput returns a string consisting of the provided labeledContent. Each labeled output is appended in the following manner: +// +// \t{{label}}:{{align_spaces}}\t{{content}}\n +// +// The initial carriage return is required to undo/erase any padding added by testing.T.Errorf. The "\t{{label}}:" is for the label. +// If a label is shorter than the longest label provided, padding spaces are added to make all the labels match in length. Once this +// alignment is achieved, "\t{{content}}\n" is added for the output. +// +// If the content of the labeledOutput contains line breaks, the subsequent lines are aligned so that they start at the same location as the first line. +func labeledOutput(content ...labeledContent) string { + longestLabel := 0 + for _, v := range content { + if len(v.label) > longestLabel { + longestLabel = len(v.label) + } + } + var output string + for _, v := range content { + output += "\t" + v.label + ":" + strings.Repeat(" ", longestLabel-len(v.label)) + "\t" + indentMessageLines(v.content, longestLabel) + "\n" + } + return output +} + +// Implements asserts that an object is implemented by the specified interface. +// +// assert.Implements(t, (*MyInterface)(nil), new(MyObject)) +func Implements(t TestingT, interfaceObject interface{}, object interface{}, msgAndArgs ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + interfaceType := reflect.TypeOf(interfaceObject).Elem() + + if object == nil { + return Fail(t, fmt.Sprintf("Cannot check if nil implements %v", interfaceType), msgAndArgs...) + } + if !reflect.TypeOf(object).Implements(interfaceType) { + return Fail(t, fmt.Sprintf("%T must implement %v", object, interfaceType), msgAndArgs...) + } + + return true +} + +// IsType asserts that the specified objects are of the same type. +func IsType(t TestingT, expectedType interface{}, object interface{}, msgAndArgs ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + + if !ObjectsAreEqual(reflect.TypeOf(object), reflect.TypeOf(expectedType)) { + return Fail(t, fmt.Sprintf("Object expected to be of type %v, but was %v", reflect.TypeOf(expectedType), reflect.TypeOf(object)), msgAndArgs...) + } + + return true +} + +// Equal asserts that two objects are equal. +// +// assert.Equal(t, 123, 123) +// +// Pointer variable equality is determined based on the equality of the +// referenced values (as opposed to the memory addresses). Function equality +// cannot be determined and will always fail. +func Equal(t TestingT, expected, actual interface{}, msgAndArgs ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if err := validateEqualArgs(expected, actual); err != nil { + return Fail(t, fmt.Sprintf("Invalid operation: %#v == %#v (%s)", + expected, actual, err), msgAndArgs...) + } + + if !ObjectsAreEqual(expected, actual) { + diff := diff(expected, actual) + expected, actual = formatUnequalValues(expected, actual) + return Fail(t, fmt.Sprintf("Not equal: \n"+ + "expected: %s\n"+ + "actual : %s%s", expected, actual, diff), msgAndArgs...) + } + + return true + +} + +// validateEqualArgs checks whether provided arguments can be safely used in the +// Equal/NotEqual functions. +func validateEqualArgs(expected, actual interface{}) error { + if expected == nil && actual == nil { + return nil + } + + if isFunction(expected) || isFunction(actual) { + return errors.New("cannot take func type as argument") + } + return nil +} + +// Same asserts that two pointers reference the same object. +// +// assert.Same(t, ptr1, ptr2) +// +// Both arguments must be pointer variables. Pointer variable sameness is +// determined based on the equality of both type and value. +func Same(t TestingT, expected, actual interface{}, msgAndArgs ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + + if !samePointers(expected, actual) { + return Fail(t, fmt.Sprintf("Not same: \n"+ + "expected: %p %#v\n"+ + "actual : %p %#v", expected, expected, actual, actual), msgAndArgs...) + } + + return true +} + +// NotSame asserts that two pointers do not reference the same object. +// +// assert.NotSame(t, ptr1, ptr2) +// +// Both arguments must be pointer variables. Pointer variable sameness is +// determined based on the equality of both type and value. +func NotSame(t TestingT, expected, actual interface{}, msgAndArgs ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + + if samePointers(expected, actual) { + return Fail(t, fmt.Sprintf( + "Expected and actual point to the same object: %p %#v", + expected, expected), msgAndArgs...) + } + return true +} + +// samePointers compares two generic interface objects and returns whether +// they point to the same object +func samePointers(first, second interface{}) bool { + firstPtr, secondPtr := reflect.ValueOf(first), reflect.ValueOf(second) + if firstPtr.Kind() != reflect.Ptr || secondPtr.Kind() != reflect.Ptr { + return false + } + + firstType, secondType := reflect.TypeOf(first), reflect.TypeOf(second) + if firstType != secondType { + return false + } + + // compare pointer addresses + return first == second +} + +// formatUnequalValues takes two values of arbitrary types and returns string +// representations appropriate to be presented to the user. +// +// If the values are not of like type, the returned strings will be prefixed +// with the type name, and the value will be enclosed in parenthesis similar +// to a type conversion in the Go grammar. +func formatUnequalValues(expected, actual interface{}) (e string, a string) { + if reflect.TypeOf(expected) != reflect.TypeOf(actual) { + return fmt.Sprintf("%T(%s)", expected, truncatingFormat(expected)), + fmt.Sprintf("%T(%s)", actual, truncatingFormat(actual)) + } + switch expected.(type) { + case time.Duration: + return fmt.Sprintf("%v", expected), fmt.Sprintf("%v", actual) + } + return truncatingFormat(expected), truncatingFormat(actual) +} + +// truncatingFormat formats the data and truncates it if it's too long. +// +// This helps keep formatted error messages lines from exceeding the +// bufio.MaxScanTokenSize max line length that the go testing framework imposes. +func truncatingFormat(data interface{}) string { + value := fmt.Sprintf("%#v", data) + max := bufio.MaxScanTokenSize - 100 // Give us some space the type info too if needed. + if len(value) > max { + value = value[0:max] + "<... truncated>" + } + return value +} + +// EqualValues asserts that two objects are equal or convertable to the same types +// and equal. +// +// assert.EqualValues(t, uint32(123), int32(123)) +func EqualValues(t TestingT, expected, actual interface{}, msgAndArgs ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + + if !ObjectsAreEqualValues(expected, actual) { + diff := diff(expected, actual) + expected, actual = formatUnequalValues(expected, actual) + return Fail(t, fmt.Sprintf("Not equal: \n"+ + "expected: %s\n"+ + "actual : %s%s", expected, actual, diff), msgAndArgs...) + } + + return true + +} + +// Exactly asserts that two objects are equal in value and type. +// +// assert.Exactly(t, int32(123), int64(123)) +func Exactly(t TestingT, expected, actual interface{}, msgAndArgs ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + + aType := reflect.TypeOf(expected) + bType := reflect.TypeOf(actual) + + if aType != bType { + return Fail(t, fmt.Sprintf("Types expected to match exactly\n\t%v != %v", aType, bType), msgAndArgs...) + } + + return Equal(t, expected, actual, msgAndArgs...) + +} + +// NotNil asserts that the specified object is not nil. +// +// assert.NotNil(t, err) +func NotNil(t TestingT, object interface{}, msgAndArgs ...interface{}) bool { + if !isNil(object) { + return true + } + if h, ok := t.(tHelper); ok { + h.Helper() + } + return Fail(t, "Expected value not to be nil.", msgAndArgs...) +} + +// containsKind checks if a specified kind in the slice of kinds. +func containsKind(kinds []reflect.Kind, kind reflect.Kind) bool { + for i := 0; i < len(kinds); i++ { + if kind == kinds[i] { + return true + } + } + + return false +} + +// isNil checks if a specified object is nil or not, without Failing. +func isNil(object interface{}) bool { + if object == nil { + return true + } + + value := reflect.ValueOf(object) + kind := value.Kind() + isNilableKind := containsKind( + []reflect.Kind{ + reflect.Chan, reflect.Func, + reflect.Interface, reflect.Map, + reflect.Ptr, reflect.Slice}, + kind) + + if isNilableKind && value.IsNil() { + return true + } + + return false +} + +// Nil asserts that the specified object is nil. +// +// assert.Nil(t, err) +func Nil(t TestingT, object interface{}, msgAndArgs ...interface{}) bool { + if isNil(object) { + return true + } + if h, ok := t.(tHelper); ok { + h.Helper() + } + return Fail(t, fmt.Sprintf("Expected nil, but got: %#v", object), msgAndArgs...) +} + +// isEmpty gets whether the specified object is considered empty or not. +func isEmpty(object interface{}) bool { + + // get nil case out of the way + if object == nil { + return true + } + + objValue := reflect.ValueOf(object) + + switch objValue.Kind() { + // collection types are empty when they have no element + case reflect.Array, reflect.Chan, reflect.Map, reflect.Slice: + return objValue.Len() == 0 + // pointers are empty if nil or if the value they point to is empty + case reflect.Ptr: + if objValue.IsNil() { + return true + } + deref := objValue.Elem().Interface() + return isEmpty(deref) + // for all other types, compare against the zero value + default: + zero := reflect.Zero(objValue.Type()) + return reflect.DeepEqual(object, zero.Interface()) + } +} + +// Empty asserts that the specified object is empty. I.e. nil, "", false, 0 or either +// a slice or a channel with len == 0. +// +// assert.Empty(t, obj) +func Empty(t TestingT, object interface{}, msgAndArgs ...interface{}) bool { + pass := isEmpty(object) + if !pass { + if h, ok := t.(tHelper); ok { + h.Helper() + } + Fail(t, fmt.Sprintf("Should be empty, but was %v", object), msgAndArgs...) + } + + return pass + +} + +// NotEmpty asserts that the specified object is NOT empty. I.e. not nil, "", false, 0 or either +// a slice or a channel with len == 0. +// +// if assert.NotEmpty(t, obj) { +// assert.Equal(t, "two", obj[1]) +// } +func NotEmpty(t TestingT, object interface{}, msgAndArgs ...interface{}) bool { + pass := !isEmpty(object) + if !pass { + if h, ok := t.(tHelper); ok { + h.Helper() + } + Fail(t, fmt.Sprintf("Should NOT be empty, but was %v", object), msgAndArgs...) + } + + return pass + +} + +// getLen try to get length of object. +// return (false, 0) if impossible. +func getLen(x interface{}) (ok bool, length int) { + v := reflect.ValueOf(x) + defer func() { + if e := recover(); e != nil { + ok = false + } + }() + return true, v.Len() +} + +// Len asserts that the specified object has specific length. +// Len also fails if the object has a type that len() not accept. +// +// assert.Len(t, mySlice, 3) +func Len(t TestingT, object interface{}, length int, msgAndArgs ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + ok, l := getLen(object) + if !ok { + return Fail(t, fmt.Sprintf("\"%s\" could not be applied builtin len()", object), msgAndArgs...) + } + + if l != length { + return Fail(t, fmt.Sprintf("\"%s\" should have %d item(s), but has %d", object, length, l), msgAndArgs...) + } + return true +} + +// True asserts that the specified value is true. +// +// assert.True(t, myBool) +func True(t TestingT, value bool, msgAndArgs ...interface{}) bool { + if !value { + if h, ok := t.(tHelper); ok { + h.Helper() + } + return Fail(t, "Should be true", msgAndArgs...) + } + + return true + +} + +// False asserts that the specified value is false. +// +// assert.False(t, myBool) +func False(t TestingT, value bool, msgAndArgs ...interface{}) bool { + if value { + if h, ok := t.(tHelper); ok { + h.Helper() + } + return Fail(t, "Should be false", msgAndArgs...) + } + + return true + +} + +// NotEqual asserts that the specified values are NOT equal. +// +// assert.NotEqual(t, obj1, obj2) +// +// Pointer variable equality is determined based on the equality of the +// referenced values (as opposed to the memory addresses). +func NotEqual(t TestingT, expected, actual interface{}, msgAndArgs ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if err := validateEqualArgs(expected, actual); err != nil { + return Fail(t, fmt.Sprintf("Invalid operation: %#v != %#v (%s)", + expected, actual, err), msgAndArgs...) + } + + if ObjectsAreEqual(expected, actual) { + return Fail(t, fmt.Sprintf("Should not be: %#v\n", actual), msgAndArgs...) + } + + return true + +} + +// NotEqualValues asserts that two objects are not equal even when converted to the same type +// +// assert.NotEqualValues(t, obj1, obj2) +func NotEqualValues(t TestingT, expected, actual interface{}, msgAndArgs ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + + if ObjectsAreEqualValues(expected, actual) { + return Fail(t, fmt.Sprintf("Should not be: %#v\n", actual), msgAndArgs...) + } + + return true +} + +// containsElement try loop over the list check if the list includes the element. +// return (false, false) if impossible. +// return (true, false) if element was not found. +// return (true, true) if element was found. +func includeElement(list interface{}, element interface{}) (ok, found bool) { + + listValue := reflect.ValueOf(list) + listKind := reflect.TypeOf(list).Kind() + defer func() { + if e := recover(); e != nil { + ok = false + found = false + } + }() + + if listKind == reflect.String { + elementValue := reflect.ValueOf(element) + return true, strings.Contains(listValue.String(), elementValue.String()) + } + + if listKind == reflect.Map { + mapKeys := listValue.MapKeys() + for i := 0; i < len(mapKeys); i++ { + if ObjectsAreEqual(mapKeys[i].Interface(), element) { + return true, true + } + } + return true, false + } + + for i := 0; i < listValue.Len(); i++ { + if ObjectsAreEqual(listValue.Index(i).Interface(), element) { + return true, true + } + } + return true, false + +} + +// Contains asserts that the specified string, list(array, slice...) or map contains the +// specified substring or element. +// +// assert.Contains(t, "Hello World", "World") +// assert.Contains(t, ["Hello", "World"], "World") +// assert.Contains(t, {"Hello": "World"}, "Hello") +func Contains(t TestingT, s, contains interface{}, msgAndArgs ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + + ok, found := includeElement(s, contains) + if !ok { + return Fail(t, fmt.Sprintf("%#v could not be applied builtin len()", s), msgAndArgs...) + } + if !found { + return Fail(t, fmt.Sprintf("%#v does not contain %#v", s, contains), msgAndArgs...) + } + + return true + +} + +// NotContains asserts that the specified string, list(array, slice...) or map does NOT contain the +// specified substring or element. +// +// assert.NotContains(t, "Hello World", "Earth") +// assert.NotContains(t, ["Hello", "World"], "Earth") +// assert.NotContains(t, {"Hello": "World"}, "Earth") +func NotContains(t TestingT, s, contains interface{}, msgAndArgs ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + + ok, found := includeElement(s, contains) + if !ok { + return Fail(t, fmt.Sprintf("\"%s\" could not be applied builtin len()", s), msgAndArgs...) + } + if found { + return Fail(t, fmt.Sprintf("\"%s\" should not contain \"%s\"", s, contains), msgAndArgs...) + } + + return true + +} + +// Subset asserts that the specified list(array, slice...) contains all +// elements given in the specified subset(array, slice...). +// +// assert.Subset(t, [1, 2, 3], [1, 2], "But [1, 2, 3] does contain [1, 2]") +func Subset(t TestingT, list, subset interface{}, msgAndArgs ...interface{}) (ok bool) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if subset == nil { + return true // we consider nil to be equal to the nil set + } + + subsetValue := reflect.ValueOf(subset) + defer func() { + if e := recover(); e != nil { + ok = false + } + }() + + listKind := reflect.TypeOf(list).Kind() + subsetKind := reflect.TypeOf(subset).Kind() + + if listKind != reflect.Array && listKind != reflect.Slice { + return Fail(t, fmt.Sprintf("%q has an unsupported type %s", list, listKind), msgAndArgs...) + } + + if subsetKind != reflect.Array && subsetKind != reflect.Slice { + return Fail(t, fmt.Sprintf("%q has an unsupported type %s", subset, subsetKind), msgAndArgs...) + } + + for i := 0; i < subsetValue.Len(); i++ { + element := subsetValue.Index(i).Interface() + ok, found := includeElement(list, element) + if !ok { + return Fail(t, fmt.Sprintf("\"%s\" could not be applied builtin len()", list), msgAndArgs...) + } + if !found { + return Fail(t, fmt.Sprintf("\"%s\" does not contain \"%s\"", list, element), msgAndArgs...) + } + } + + return true +} + +// NotSubset asserts that the specified list(array, slice...) contains not all +// elements given in the specified subset(array, slice...). +// +// assert.NotSubset(t, [1, 3, 4], [1, 2], "But [1, 3, 4] does not contain [1, 2]") +func NotSubset(t TestingT, list, subset interface{}, msgAndArgs ...interface{}) (ok bool) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if subset == nil { + return Fail(t, fmt.Sprintf("nil is the empty set which is a subset of every set"), msgAndArgs...) + } + + subsetValue := reflect.ValueOf(subset) + defer func() { + if e := recover(); e != nil { + ok = false + } + }() + + listKind := reflect.TypeOf(list).Kind() + subsetKind := reflect.TypeOf(subset).Kind() + + if listKind != reflect.Array && listKind != reflect.Slice { + return Fail(t, fmt.Sprintf("%q has an unsupported type %s", list, listKind), msgAndArgs...) + } + + if subsetKind != reflect.Array && subsetKind != reflect.Slice { + return Fail(t, fmt.Sprintf("%q has an unsupported type %s", subset, subsetKind), msgAndArgs...) + } + + for i := 0; i < subsetValue.Len(); i++ { + element := subsetValue.Index(i).Interface() + ok, found := includeElement(list, element) + if !ok { + return Fail(t, fmt.Sprintf("\"%s\" could not be applied builtin len()", list), msgAndArgs...) + } + if !found { + return true + } + } + + return Fail(t, fmt.Sprintf("%q is a subset of %q", subset, list), msgAndArgs...) +} + +// ElementsMatch asserts that the specified listA(array, slice...) is equal to specified +// listB(array, slice...) ignoring the order of the elements. If there are duplicate elements, +// the number of appearances of each of them in both lists should match. +// +// assert.ElementsMatch(t, [1, 3, 2, 3], [1, 3, 3, 2]) +func ElementsMatch(t TestingT, listA, listB interface{}, msgAndArgs ...interface{}) (ok bool) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if isEmpty(listA) && isEmpty(listB) { + return true + } + + if !isList(t, listA, msgAndArgs...) || !isList(t, listB, msgAndArgs...) { + return false + } + + extraA, extraB := diffLists(listA, listB) + + if len(extraA) == 0 && len(extraB) == 0 { + return true + } + + return Fail(t, formatListDiff(listA, listB, extraA, extraB), msgAndArgs...) +} + +// isList checks that the provided value is array or slice. +func isList(t TestingT, list interface{}, msgAndArgs ...interface{}) (ok bool) { + kind := reflect.TypeOf(list).Kind() + if kind != reflect.Array && kind != reflect.Slice { + return Fail(t, fmt.Sprintf("%q has an unsupported type %s, expecting array or slice", list, kind), + msgAndArgs...) + } + return true +} + +// diffLists diffs two arrays/slices and returns slices of elements that are only in A and only in B. +// If some element is present multiple times, each instance is counted separately (e.g. if something is 2x in A and +// 5x in B, it will be 0x in extraA and 3x in extraB). The order of items in both lists is ignored. +func diffLists(listA, listB interface{}) (extraA, extraB []interface{}) { + aValue := reflect.ValueOf(listA) + bValue := reflect.ValueOf(listB) + + aLen := aValue.Len() + bLen := bValue.Len() + + // Mark indexes in bValue that we already used + visited := make([]bool, bLen) + for i := 0; i < aLen; i++ { + element := aValue.Index(i).Interface() + found := false + for j := 0; j < bLen; j++ { + if visited[j] { + continue + } + if ObjectsAreEqual(bValue.Index(j).Interface(), element) { + visited[j] = true + found = true + break + } + } + if !found { + extraA = append(extraA, element) + } + } + + for j := 0; j < bLen; j++ { + if visited[j] { + continue + } + extraB = append(extraB, bValue.Index(j).Interface()) + } + + return +} + +func formatListDiff(listA, listB interface{}, extraA, extraB []interface{}) string { + var msg bytes.Buffer + + msg.WriteString("elements differ") + if len(extraA) > 0 { + msg.WriteString("\n\nextra elements in list A:\n") + msg.WriteString(spewConfig.Sdump(extraA)) + } + if len(extraB) > 0 { + msg.WriteString("\n\nextra elements in list B:\n") + msg.WriteString(spewConfig.Sdump(extraB)) + } + msg.WriteString("\n\nlistA:\n") + msg.WriteString(spewConfig.Sdump(listA)) + msg.WriteString("\n\nlistB:\n") + msg.WriteString(spewConfig.Sdump(listB)) + + return msg.String() +} + +// Condition uses a Comparison to assert a complex condition. +func Condition(t TestingT, comp Comparison, msgAndArgs ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + result := comp() + if !result { + Fail(t, "Condition failed!", msgAndArgs...) + } + return result +} + +// PanicTestFunc defines a func that should be passed to the assert.Panics and assert.NotPanics +// methods, and represents a simple func that takes no arguments, and returns nothing. +type PanicTestFunc func() + +// didPanic returns true if the function passed to it panics. Otherwise, it returns false. +func didPanic(f PanicTestFunc) (bool, interface{}, string) { + + didPanic := false + var message interface{} + var stack string + func() { + + defer func() { + if message = recover(); message != nil { + didPanic = true + stack = string(debug.Stack()) + } + }() + + // call the target function + f() + + }() + + return didPanic, message, stack + +} + +// Panics asserts that the code inside the specified PanicTestFunc panics. +// +// assert.Panics(t, func(){ GoCrazy() }) +func Panics(t TestingT, f PanicTestFunc, msgAndArgs ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + + if funcDidPanic, panicValue, _ := didPanic(f); !funcDidPanic { + return Fail(t, fmt.Sprintf("func %#v should panic\n\tPanic value:\t%#v", f, panicValue), msgAndArgs...) + } + + return true +} + +// PanicsWithValue asserts that the code inside the specified PanicTestFunc panics, and that +// the recovered panic value equals the expected panic value. +// +// assert.PanicsWithValue(t, "crazy error", func(){ GoCrazy() }) +func PanicsWithValue(t TestingT, expected interface{}, f PanicTestFunc, msgAndArgs ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + + funcDidPanic, panicValue, panickedStack := didPanic(f) + if !funcDidPanic { + return Fail(t, fmt.Sprintf("func %#v should panic\n\tPanic value:\t%#v", f, panicValue), msgAndArgs...) + } + if panicValue != expected { + return Fail(t, fmt.Sprintf("func %#v should panic with value:\t%#v\n\tPanic value:\t%#v\n\tPanic stack:\t%s", f, expected, panicValue, panickedStack), msgAndArgs...) + } + + return true +} + +// PanicsWithError asserts that the code inside the specified PanicTestFunc +// panics, and that the recovered panic value is an error that satisfies the +// EqualError comparison. +// +// assert.PanicsWithError(t, "crazy error", func(){ GoCrazy() }) +func PanicsWithError(t TestingT, errString string, f PanicTestFunc, msgAndArgs ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + + funcDidPanic, panicValue, panickedStack := didPanic(f) + if !funcDidPanic { + return Fail(t, fmt.Sprintf("func %#v should panic\n\tPanic value:\t%#v", f, panicValue), msgAndArgs...) + } + panicErr, ok := panicValue.(error) + if !ok || panicErr.Error() != errString { + return Fail(t, fmt.Sprintf("func %#v should panic with error message:\t%#v\n\tPanic value:\t%#v\n\tPanic stack:\t%s", f, errString, panicValue, panickedStack), msgAndArgs...) + } + + return true +} + +// NotPanics asserts that the code inside the specified PanicTestFunc does NOT panic. +// +// assert.NotPanics(t, func(){ RemainCalm() }) +func NotPanics(t TestingT, f PanicTestFunc, msgAndArgs ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + + if funcDidPanic, panicValue, panickedStack := didPanic(f); funcDidPanic { + return Fail(t, fmt.Sprintf("func %#v should not panic\n\tPanic value:\t%v\n\tPanic stack:\t%s", f, panicValue, panickedStack), msgAndArgs...) + } + + return true +} + +// WithinDuration asserts that the two times are within duration delta of each other. +// +// assert.WithinDuration(t, time.Now(), time.Now(), 10*time.Second) +func WithinDuration(t TestingT, expected, actual time.Time, delta time.Duration, msgAndArgs ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + + dt := expected.Sub(actual) + if dt < -delta || dt > delta { + return Fail(t, fmt.Sprintf("Max difference between %v and %v allowed is %v, but difference was %v", expected, actual, delta, dt), msgAndArgs...) + } + + return true +} + +func toFloat(x interface{}) (float64, bool) { + var xf float64 + xok := true + + switch xn := x.(type) { + case uint: + xf = float64(xn) + case uint8: + xf = float64(xn) + case uint16: + xf = float64(xn) + case uint32: + xf = float64(xn) + case uint64: + xf = float64(xn) + case int: + xf = float64(xn) + case int8: + xf = float64(xn) + case int16: + xf = float64(xn) + case int32: + xf = float64(xn) + case int64: + xf = float64(xn) + case float32: + xf = float64(xn) + case float64: + xf = xn + case time.Duration: + xf = float64(xn) + default: + xok = false + } + + return xf, xok +} + +// InDelta asserts that the two numerals are within delta of each other. +// +// assert.InDelta(t, math.Pi, 22/7.0, 0.01) +func InDelta(t TestingT, expected, actual interface{}, delta float64, msgAndArgs ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + + af, aok := toFloat(expected) + bf, bok := toFloat(actual) + + if !aok || !bok { + return Fail(t, fmt.Sprintf("Parameters must be numerical"), msgAndArgs...) + } + + if math.IsNaN(af) { + return Fail(t, fmt.Sprintf("Expected must not be NaN"), msgAndArgs...) + } + + if math.IsNaN(bf) { + return Fail(t, fmt.Sprintf("Expected %v with delta %v, but was NaN", expected, delta), msgAndArgs...) + } + + dt := af - bf + if dt < -delta || dt > delta { + return Fail(t, fmt.Sprintf("Max difference between %v and %v allowed is %v, but difference was %v", expected, actual, delta, dt), msgAndArgs...) + } + + return true +} + +// InDeltaSlice is the same as InDelta, except it compares two slices. +func InDeltaSlice(t TestingT, expected, actual interface{}, delta float64, msgAndArgs ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if expected == nil || actual == nil || + reflect.TypeOf(actual).Kind() != reflect.Slice || + reflect.TypeOf(expected).Kind() != reflect.Slice { + return Fail(t, fmt.Sprintf("Parameters must be slice"), msgAndArgs...) + } + + actualSlice := reflect.ValueOf(actual) + expectedSlice := reflect.ValueOf(expected) + + for i := 0; i < actualSlice.Len(); i++ { + result := InDelta(t, actualSlice.Index(i).Interface(), expectedSlice.Index(i).Interface(), delta, msgAndArgs...) + if !result { + return result + } + } + + return true +} + +// InDeltaMapValues is the same as InDelta, but it compares all values between two maps. Both maps must have exactly the same keys. +func InDeltaMapValues(t TestingT, expected, actual interface{}, delta float64, msgAndArgs ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if expected == nil || actual == nil || + reflect.TypeOf(actual).Kind() != reflect.Map || + reflect.TypeOf(expected).Kind() != reflect.Map { + return Fail(t, "Arguments must be maps", msgAndArgs...) + } + + expectedMap := reflect.ValueOf(expected) + actualMap := reflect.ValueOf(actual) + + if expectedMap.Len() != actualMap.Len() { + return Fail(t, "Arguments must have the same number of keys", msgAndArgs...) + } + + for _, k := range expectedMap.MapKeys() { + ev := expectedMap.MapIndex(k) + av := actualMap.MapIndex(k) + + if !ev.IsValid() { + return Fail(t, fmt.Sprintf("missing key %q in expected map", k), msgAndArgs...) + } + + if !av.IsValid() { + return Fail(t, fmt.Sprintf("missing key %q in actual map", k), msgAndArgs...) + } + + if !InDelta( + t, + ev.Interface(), + av.Interface(), + delta, + msgAndArgs..., + ) { + return false + } + } + + return true +} + +func calcRelativeError(expected, actual interface{}) (float64, error) { + af, aok := toFloat(expected) + if !aok { + return 0, fmt.Errorf("expected value %q cannot be converted to float", expected) + } + if math.IsNaN(af) { + return 0, errors.New("expected value must not be NaN") + } + if af == 0 { + return 0, fmt.Errorf("expected value must have a value other than zero to calculate the relative error") + } + bf, bok := toFloat(actual) + if !bok { + return 0, fmt.Errorf("actual value %q cannot be converted to float", actual) + } + if math.IsNaN(bf) { + return 0, errors.New("actual value must not be NaN") + } + + return math.Abs(af-bf) / math.Abs(af), nil +} + +// InEpsilon asserts that expected and actual have a relative error less than epsilon +func InEpsilon(t TestingT, expected, actual interface{}, epsilon float64, msgAndArgs ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if math.IsNaN(epsilon) { + return Fail(t, "epsilon must not be NaN") + } + actualEpsilon, err := calcRelativeError(expected, actual) + if err != nil { + return Fail(t, err.Error(), msgAndArgs...) + } + if actualEpsilon > epsilon { + return Fail(t, fmt.Sprintf("Relative error is too high: %#v (expected)\n"+ + " < %#v (actual)", epsilon, actualEpsilon), msgAndArgs...) + } + + return true +} + +// InEpsilonSlice is the same as InEpsilon, except it compares each value from two slices. +func InEpsilonSlice(t TestingT, expected, actual interface{}, epsilon float64, msgAndArgs ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if expected == nil || actual == nil || + reflect.TypeOf(actual).Kind() != reflect.Slice || + reflect.TypeOf(expected).Kind() != reflect.Slice { + return Fail(t, fmt.Sprintf("Parameters must be slice"), msgAndArgs...) + } + + actualSlice := reflect.ValueOf(actual) + expectedSlice := reflect.ValueOf(expected) + + for i := 0; i < actualSlice.Len(); i++ { + result := InEpsilon(t, actualSlice.Index(i).Interface(), expectedSlice.Index(i).Interface(), epsilon) + if !result { + return result + } + } + + return true +} + +/* + Errors +*/ + +// NoError asserts that a function returned no error (i.e. `nil`). +// +// actualObj, err := SomeFunction() +// if assert.NoError(t, err) { +// assert.Equal(t, expectedObj, actualObj) +// } +func NoError(t TestingT, err error, msgAndArgs ...interface{}) bool { + if err != nil { + if h, ok := t.(tHelper); ok { + h.Helper() + } + return Fail(t, fmt.Sprintf("Received unexpected error:\n%+v", err), msgAndArgs...) + } + + return true +} + +// Error asserts that a function returned an error (i.e. not `nil`). +// +// actualObj, err := SomeFunction() +// if assert.Error(t, err) { +// assert.Equal(t, expectedError, err) +// } +func Error(t TestingT, err error, msgAndArgs ...interface{}) bool { + if err == nil { + if h, ok := t.(tHelper); ok { + h.Helper() + } + return Fail(t, "An error is expected but got nil.", msgAndArgs...) + } + + return true +} + +// EqualError asserts that a function returned an error (i.e. not `nil`) +// and that it is equal to the provided error. +// +// actualObj, err := SomeFunction() +// assert.EqualError(t, err, expectedErrorString) +func EqualError(t TestingT, theError error, errString string, msgAndArgs ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if !Error(t, theError, msgAndArgs...) { + return false + } + expected := errString + actual := theError.Error() + // don't need to use deep equals here, we know they are both strings + if expected != actual { + return Fail(t, fmt.Sprintf("Error message not equal:\n"+ + "expected: %q\n"+ + "actual : %q", expected, actual), msgAndArgs...) + } + return true +} + +// matchRegexp return true if a specified regexp matches a string. +func matchRegexp(rx interface{}, str interface{}) bool { + + var r *regexp.Regexp + if rr, ok := rx.(*regexp.Regexp); ok { + r = rr + } else { + r = regexp.MustCompile(fmt.Sprint(rx)) + } + + return (r.FindStringIndex(fmt.Sprint(str)) != nil) + +} + +// Regexp asserts that a specified regexp matches a string. +// +// assert.Regexp(t, regexp.MustCompile("start"), "it's starting") +// assert.Regexp(t, "start...$", "it's not starting") +func Regexp(t TestingT, rx interface{}, str interface{}, msgAndArgs ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + + match := matchRegexp(rx, str) + + if !match { + Fail(t, fmt.Sprintf("Expect \"%v\" to match \"%v\"", str, rx), msgAndArgs...) + } + + return match +} + +// NotRegexp asserts that a specified regexp does not match a string. +// +// assert.NotRegexp(t, regexp.MustCompile("starts"), "it's starting") +// assert.NotRegexp(t, "^start", "it's not starting") +func NotRegexp(t TestingT, rx interface{}, str interface{}, msgAndArgs ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + match := matchRegexp(rx, str) + + if match { + Fail(t, fmt.Sprintf("Expect \"%v\" to NOT match \"%v\"", str, rx), msgAndArgs...) + } + + return !match + +} + +// Zero asserts that i is the zero value for its type. +func Zero(t TestingT, i interface{}, msgAndArgs ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if i != nil && !reflect.DeepEqual(i, reflect.Zero(reflect.TypeOf(i)).Interface()) { + return Fail(t, fmt.Sprintf("Should be zero, but was %v", i), msgAndArgs...) + } + return true +} + +// NotZero asserts that i is not the zero value for its type. +func NotZero(t TestingT, i interface{}, msgAndArgs ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if i == nil || reflect.DeepEqual(i, reflect.Zero(reflect.TypeOf(i)).Interface()) { + return Fail(t, fmt.Sprintf("Should not be zero, but was %v", i), msgAndArgs...) + } + return true +} + +// FileExists checks whether a file exists in the given path. It also fails if +// the path points to a directory or there is an error when trying to check the file. +func FileExists(t TestingT, path string, msgAndArgs ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + info, err := os.Lstat(path) + if err != nil { + if os.IsNotExist(err) { + return Fail(t, fmt.Sprintf("unable to find file %q", path), msgAndArgs...) + } + return Fail(t, fmt.Sprintf("error when running os.Lstat(%q): %s", path, err), msgAndArgs...) + } + if info.IsDir() { + return Fail(t, fmt.Sprintf("%q is a directory", path), msgAndArgs...) + } + return true +} + +// NoFileExists checks whether a file does not exist in a given path. It fails +// if the path points to an existing _file_ only. +func NoFileExists(t TestingT, path string, msgAndArgs ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + info, err := os.Lstat(path) + if err != nil { + return true + } + if info.IsDir() { + return true + } + return Fail(t, fmt.Sprintf("file %q exists", path), msgAndArgs...) +} + +// DirExists checks whether a directory exists in the given path. It also fails +// if the path is a file rather a directory or there is an error checking whether it exists. +func DirExists(t TestingT, path string, msgAndArgs ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + info, err := os.Lstat(path) + if err != nil { + if os.IsNotExist(err) { + return Fail(t, fmt.Sprintf("unable to find file %q", path), msgAndArgs...) + } + return Fail(t, fmt.Sprintf("error when running os.Lstat(%q): %s", path, err), msgAndArgs...) + } + if !info.IsDir() { + return Fail(t, fmt.Sprintf("%q is a file", path), msgAndArgs...) + } + return true +} + +// NoDirExists checks whether a directory does not exist in the given path. +// It fails if the path points to an existing _directory_ only. +func NoDirExists(t TestingT, path string, msgAndArgs ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + info, err := os.Lstat(path) + if err != nil { + if os.IsNotExist(err) { + return true + } + return true + } + if !info.IsDir() { + return true + } + return Fail(t, fmt.Sprintf("directory %q exists", path), msgAndArgs...) +} + +// JSONEq asserts that two JSON strings are equivalent. +// +// assert.JSONEq(t, `{"hello": "world", "foo": "bar"}`, `{"foo": "bar", "hello": "world"}`) +func JSONEq(t TestingT, expected string, actual string, msgAndArgs ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + var expectedJSONAsInterface, actualJSONAsInterface interface{} + + if err := json.Unmarshal([]byte(expected), &expectedJSONAsInterface); err != nil { + return Fail(t, fmt.Sprintf("Expected value ('%s') is not valid json.\nJSON parsing error: '%s'", expected, err.Error()), msgAndArgs...) + } + + if err := json.Unmarshal([]byte(actual), &actualJSONAsInterface); err != nil { + return Fail(t, fmt.Sprintf("Input ('%s') needs to be valid json.\nJSON parsing error: '%s'", actual, err.Error()), msgAndArgs...) + } + + return Equal(t, expectedJSONAsInterface, actualJSONAsInterface, msgAndArgs...) +} + +// YAMLEq asserts that two YAML strings are equivalent. +func YAMLEq(t TestingT, expected string, actual string, msgAndArgs ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + var expectedYAMLAsInterface, actualYAMLAsInterface interface{} + + if err := yaml.Unmarshal([]byte(expected), &expectedYAMLAsInterface); err != nil { + return Fail(t, fmt.Sprintf("Expected value ('%s') is not valid yaml.\nYAML parsing error: '%s'", expected, err.Error()), msgAndArgs...) + } + + if err := yaml.Unmarshal([]byte(actual), &actualYAMLAsInterface); err != nil { + return Fail(t, fmt.Sprintf("Input ('%s') needs to be valid yaml.\nYAML error: '%s'", actual, err.Error()), msgAndArgs...) + } + + return Equal(t, expectedYAMLAsInterface, actualYAMLAsInterface, msgAndArgs...) +} + +func typeAndKind(v interface{}) (reflect.Type, reflect.Kind) { + t := reflect.TypeOf(v) + k := t.Kind() + + if k == reflect.Ptr { + t = t.Elem() + k = t.Kind() + } + return t, k +} + +// diff returns a diff of both values as long as both are of the same type and +// are a struct, map, slice, array or string. Otherwise it returns an empty string. +func diff(expected interface{}, actual interface{}) string { + if expected == nil || actual == nil { + return "" + } + + et, ek := typeAndKind(expected) + at, _ := typeAndKind(actual) + + if et != at { + return "" + } + + if ek != reflect.Struct && ek != reflect.Map && ek != reflect.Slice && ek != reflect.Array && ek != reflect.String { + return "" + } + + var e, a string + if et != reflect.TypeOf("") { + e = spewConfig.Sdump(expected) + a = spewConfig.Sdump(actual) + } else { + e = reflect.ValueOf(expected).String() + a = reflect.ValueOf(actual).String() + } + + diff, _ := difflib.GetUnifiedDiffString(difflib.UnifiedDiff{ + A: difflib.SplitLines(e), + B: difflib.SplitLines(a), + FromFile: "Expected", + FromDate: "", + ToFile: "Actual", + ToDate: "", + Context: 1, + }) + + return "\n\nDiff:\n" + diff +} + +func isFunction(arg interface{}) bool { + if arg == nil { + return false + } + return reflect.TypeOf(arg).Kind() == reflect.Func +} + +var spewConfig = spew.ConfigState{ + Indent: " ", + DisablePointerAddresses: true, + DisableCapacities: true, + SortKeys: true, + DisableMethods: true, + MaxDepth: 10, +} + +type tHelper interface { + Helper() +} + +// Eventually asserts that given condition will be met in waitFor time, +// periodically checking target function each tick. +// +// assert.Eventually(t, func() bool { return true; }, time.Second, 10*time.Millisecond) +func Eventually(t TestingT, condition func() bool, waitFor time.Duration, tick time.Duration, msgAndArgs ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + + ch := make(chan bool, 1) + + timer := time.NewTimer(waitFor) + defer timer.Stop() + + ticker := time.NewTicker(tick) + defer ticker.Stop() + + for tick := ticker.C; ; { + select { + case <-timer.C: + return Fail(t, "Condition never satisfied", msgAndArgs...) + case <-tick: + tick = nil + go func() { ch <- condition() }() + case v := <-ch: + if v { + return true + } + tick = ticker.C + } + } +} + +// Never asserts that the given condition doesn't satisfy in waitFor time, +// periodically checking the target function each tick. +// +// assert.Never(t, func() bool { return false; }, time.Second, 10*time.Millisecond) +func Never(t TestingT, condition func() bool, waitFor time.Duration, tick time.Duration, msgAndArgs ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + + ch := make(chan bool, 1) + + timer := time.NewTimer(waitFor) + defer timer.Stop() + + ticker := time.NewTicker(tick) + defer ticker.Stop() + + for tick := ticker.C; ; { + select { + case <-timer.C: + return true + case <-tick: + tick = nil + go func() { ch <- condition() }() + case v := <-ch: + if v { + return Fail(t, "Condition satisfied", msgAndArgs...) + } + tick = ticker.C + } + } +} + +// ErrorIs asserts that at least one of the errors in err's chain matches target. +// This is a wrapper for errors.Is. +func ErrorIs(t TestingT, err, target error, msgAndArgs ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if errors.Is(err, target) { + return true + } + + var expectedText string + if target != nil { + expectedText = target.Error() + } + + chain := buildErrorChainString(err) + + return Fail(t, fmt.Sprintf("Target error should be in err chain:\n"+ + "expected: %q\n"+ + "in chain: %s", expectedText, chain, + ), msgAndArgs...) +} + +// NotErrorIs asserts that at none of the errors in err's chain matches target. +// This is a wrapper for errors.Is. +func NotErrorIs(t TestingT, err, target error, msgAndArgs ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if !errors.Is(err, target) { + return true + } + + var expectedText string + if target != nil { + expectedText = target.Error() + } + + chain := buildErrorChainString(err) + + return Fail(t, fmt.Sprintf("Target error should not be in err chain:\n"+ + "found: %q\n"+ + "in chain: %s", expectedText, chain, + ), msgAndArgs...) +} + +// ErrorAs asserts that at least one of the errors in err's chain matches target, and if so, sets target to that error value. +// This is a wrapper for errors.As. +func ErrorAs(t TestingT, err error, target interface{}, msgAndArgs ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if errors.As(err, target) { + return true + } + + chain := buildErrorChainString(err) + + return Fail(t, fmt.Sprintf("Should be in error chain:\n"+ + "expected: %q\n"+ + "in chain: %s", target, chain, + ), msgAndArgs...) +} + +func buildErrorChainString(err error) string { + if err == nil { + return "" + } + + e := errors.Unwrap(err) + chain := fmt.Sprintf("%q", err.Error()) + for e != nil { + chain += fmt.Sprintf("\n\t%q", e.Error()) + e = errors.Unwrap(e) + } + return chain +} diff --git a/vendor/github.com/stretchr/testify/assert/doc.go b/vendor/github.com/stretchr/testify/assert/doc.go new file mode 100644 index 00000000..c9dccc4d --- /dev/null +++ b/vendor/github.com/stretchr/testify/assert/doc.go @@ -0,0 +1,45 @@ +// Package assert provides a set of comprehensive testing tools for use with the normal Go testing system. +// +// Example Usage +// +// The following is a complete example using assert in a standard test function: +// import ( +// "testing" +// "github.com/stretchr/testify/assert" +// ) +// +// func TestSomething(t *testing.T) { +// +// var a string = "Hello" +// var b string = "Hello" +// +// assert.Equal(t, a, b, "The two words should be the same.") +// +// } +// +// if you assert many times, use the format below: +// +// import ( +// "testing" +// "github.com/stretchr/testify/assert" +// ) +// +// func TestSomething(t *testing.T) { +// assert := assert.New(t) +// +// var a string = "Hello" +// var b string = "Hello" +// +// assert.Equal(a, b, "The two words should be the same.") +// } +// +// Assertions +// +// Assertions allow you to easily write test code, and are global funcs in the `assert` package. +// All assertion functions take, as the first argument, the `*testing.T` object provided by the +// testing framework. This allows the assertion funcs to write the failings and other details to +// the correct place. +// +// Every assertion function also takes an optional string message as the final argument, +// allowing custom error messages to be appended to the message the assertion method outputs. +package assert diff --git a/vendor/github.com/stretchr/testify/assert/errors.go b/vendor/github.com/stretchr/testify/assert/errors.go new file mode 100644 index 00000000..ac9dc9d1 --- /dev/null +++ b/vendor/github.com/stretchr/testify/assert/errors.go @@ -0,0 +1,10 @@ +package assert + +import ( + "errors" +) + +// AnError is an error instance useful for testing. If the code does not care +// about error specifics, and only needs to return the error for example, this +// error should be used to make the test code more readable. +var AnError = errors.New("assert.AnError general error for testing") diff --git a/vendor/github.com/stretchr/testify/assert/forward_assertions.go b/vendor/github.com/stretchr/testify/assert/forward_assertions.go new file mode 100644 index 00000000..df189d23 --- /dev/null +++ b/vendor/github.com/stretchr/testify/assert/forward_assertions.go @@ -0,0 +1,16 @@ +package assert + +// Assertions provides assertion methods around the +// TestingT interface. +type Assertions struct { + t TestingT +} + +// New makes a new Assertions object for the specified TestingT. +func New(t TestingT) *Assertions { + return &Assertions{ + t: t, + } +} + +//go:generate sh -c "cd ../_codegen && go build && cd - && ../_codegen/_codegen -output-package=assert -template=assertion_forward.go.tmpl -include-format-funcs" diff --git a/vendor/github.com/stretchr/testify/assert/http_assertions.go b/vendor/github.com/stretchr/testify/assert/http_assertions.go new file mode 100644 index 00000000..4ed341dd --- /dev/null +++ b/vendor/github.com/stretchr/testify/assert/http_assertions.go @@ -0,0 +1,162 @@ +package assert + +import ( + "fmt" + "net/http" + "net/http/httptest" + "net/url" + "strings" +) + +// httpCode is a helper that returns HTTP code of the response. It returns -1 and +// an error if building a new request fails. +func httpCode(handler http.HandlerFunc, method, url string, values url.Values) (int, error) { + w := httptest.NewRecorder() + req, err := http.NewRequest(method, url, nil) + if err != nil { + return -1, err + } + req.URL.RawQuery = values.Encode() + handler(w, req) + return w.Code, nil +} + +// HTTPSuccess asserts that a specified handler returns a success status code. +// +// assert.HTTPSuccess(t, myHandler, "POST", "http://www.google.com", nil) +// +// Returns whether the assertion was successful (true) or not (false). +func HTTPSuccess(t TestingT, handler http.HandlerFunc, method, url string, values url.Values, msgAndArgs ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + code, err := httpCode(handler, method, url, values) + if err != nil { + Fail(t, fmt.Sprintf("Failed to build test request, got error: %s", err)) + } + + isSuccessCode := code >= http.StatusOK && code <= http.StatusPartialContent + if !isSuccessCode { + Fail(t, fmt.Sprintf("Expected HTTP success status code for %q but received %d", url+"?"+values.Encode(), code)) + } + + return isSuccessCode +} + +// HTTPRedirect asserts that a specified handler returns a redirect status code. +// +// assert.HTTPRedirect(t, myHandler, "GET", "/a/b/c", url.Values{"a": []string{"b", "c"}} +// +// Returns whether the assertion was successful (true) or not (false). +func HTTPRedirect(t TestingT, handler http.HandlerFunc, method, url string, values url.Values, msgAndArgs ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + code, err := httpCode(handler, method, url, values) + if err != nil { + Fail(t, fmt.Sprintf("Failed to build test request, got error: %s", err)) + } + + isRedirectCode := code >= http.StatusMultipleChoices && code <= http.StatusTemporaryRedirect + if !isRedirectCode { + Fail(t, fmt.Sprintf("Expected HTTP redirect status code for %q but received %d", url+"?"+values.Encode(), code)) + } + + return isRedirectCode +} + +// HTTPError asserts that a specified handler returns an error status code. +// +// assert.HTTPError(t, myHandler, "POST", "/a/b/c", url.Values{"a": []string{"b", "c"}} +// +// Returns whether the assertion was successful (true) or not (false). +func HTTPError(t TestingT, handler http.HandlerFunc, method, url string, values url.Values, msgAndArgs ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + code, err := httpCode(handler, method, url, values) + if err != nil { + Fail(t, fmt.Sprintf("Failed to build test request, got error: %s", err)) + } + + isErrorCode := code >= http.StatusBadRequest + if !isErrorCode { + Fail(t, fmt.Sprintf("Expected HTTP error status code for %q but received %d", url+"?"+values.Encode(), code)) + } + + return isErrorCode +} + +// HTTPStatusCode asserts that a specified handler returns a specified status code. +// +// assert.HTTPStatusCode(t, myHandler, "GET", "/notImplemented", nil, 501) +// +// Returns whether the assertion was successful (true) or not (false). +func HTTPStatusCode(t TestingT, handler http.HandlerFunc, method, url string, values url.Values, statuscode int, msgAndArgs ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + code, err := httpCode(handler, method, url, values) + if err != nil { + Fail(t, fmt.Sprintf("Failed to build test request, got error: %s", err)) + } + + successful := code == statuscode + if !successful { + Fail(t, fmt.Sprintf("Expected HTTP status code %d for %q but received %d", statuscode, url+"?"+values.Encode(), code)) + } + + return successful +} + +// HTTPBody is a helper that returns HTTP body of the response. It returns +// empty string if building a new request fails. +func HTTPBody(handler http.HandlerFunc, method, url string, values url.Values) string { + w := httptest.NewRecorder() + req, err := http.NewRequest(method, url+"?"+values.Encode(), nil) + if err != nil { + return "" + } + handler(w, req) + return w.Body.String() +} + +// HTTPBodyContains asserts that a specified handler returns a +// body that contains a string. +// +// assert.HTTPBodyContains(t, myHandler, "GET", "www.google.com", nil, "I'm Feeling Lucky") +// +// Returns whether the assertion was successful (true) or not (false). +func HTTPBodyContains(t TestingT, handler http.HandlerFunc, method, url string, values url.Values, str interface{}, msgAndArgs ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + body := HTTPBody(handler, method, url, values) + + contains := strings.Contains(body, fmt.Sprint(str)) + if !contains { + Fail(t, fmt.Sprintf("Expected response body for \"%s\" to contain \"%s\" but found \"%s\"", url+"?"+values.Encode(), str, body)) + } + + return contains +} + +// HTTPBodyNotContains asserts that a specified handler returns a +// body that does not contain a string. +// +// assert.HTTPBodyNotContains(t, myHandler, "GET", "www.google.com", nil, "I'm Feeling Lucky") +// +// Returns whether the assertion was successful (true) or not (false). +func HTTPBodyNotContains(t TestingT, handler http.HandlerFunc, method, url string, values url.Values, str interface{}, msgAndArgs ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + body := HTTPBody(handler, method, url, values) + + contains := strings.Contains(body, fmt.Sprint(str)) + if contains { + Fail(t, fmt.Sprintf("Expected response body for \"%s\" to NOT contain \"%s\" but found \"%s\"", url+"?"+values.Encode(), str, body)) + } + + return !contains +} diff --git a/vendor/github.com/stretchr/testify/require/doc.go b/vendor/github.com/stretchr/testify/require/doc.go new file mode 100644 index 00000000..169de392 --- /dev/null +++ b/vendor/github.com/stretchr/testify/require/doc.go @@ -0,0 +1,28 @@ +// Package require implements the same assertions as the `assert` package but +// stops test execution when a test fails. +// +// Example Usage +// +// The following is a complete example using require in a standard test function: +// import ( +// "testing" +// "github.com/stretchr/testify/require" +// ) +// +// func TestSomething(t *testing.T) { +// +// var a string = "Hello" +// var b string = "Hello" +// +// require.Equal(t, a, b, "The two words should be the same.") +// +// } +// +// Assertions +// +// The `require` package have same global functions as in the `assert` package, +// but instead of returning a boolean result they call `t.FailNow()`. +// +// Every assertion function also takes an optional string message as the final argument, +// allowing custom error messages to be appended to the message the assertion method outputs. +package require diff --git a/vendor/github.com/stretchr/testify/require/forward_requirements.go b/vendor/github.com/stretchr/testify/require/forward_requirements.go new file mode 100644 index 00000000..1dcb2338 --- /dev/null +++ b/vendor/github.com/stretchr/testify/require/forward_requirements.go @@ -0,0 +1,16 @@ +package require + +// Assertions provides assertion methods around the +// TestingT interface. +type Assertions struct { + t TestingT +} + +// New makes a new Assertions object for the specified TestingT. +func New(t TestingT) *Assertions { + return &Assertions{ + t: t, + } +} + +//go:generate sh -c "cd ../_codegen && go build && cd - && ../_codegen/_codegen -output-package=require -template=require_forward.go.tmpl -include-format-funcs" diff --git a/vendor/github.com/stretchr/testify/require/require.go b/vendor/github.com/stretchr/testify/require/require.go new file mode 100644 index 00000000..51820df2 --- /dev/null +++ b/vendor/github.com/stretchr/testify/require/require.go @@ -0,0 +1,1879 @@ +/* +* CODE GENERATED AUTOMATICALLY WITH github.com/stretchr/testify/_codegen +* THIS FILE MUST NOT BE EDITED BY HAND + */ + +package require + +import ( + assert "github.com/stretchr/testify/assert" + http "net/http" + url "net/url" + time "time" +) + +// Condition uses a Comparison to assert a complex condition. +func Condition(t TestingT, comp assert.Comparison, msgAndArgs ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.Condition(t, comp, msgAndArgs...) { + return + } + t.FailNow() +} + +// Conditionf uses a Comparison to assert a complex condition. +func Conditionf(t TestingT, comp assert.Comparison, msg string, args ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.Conditionf(t, comp, msg, args...) { + return + } + t.FailNow() +} + +// Contains asserts that the specified string, list(array, slice...) or map contains the +// specified substring or element. +// +// assert.Contains(t, "Hello World", "World") +// assert.Contains(t, ["Hello", "World"], "World") +// assert.Contains(t, {"Hello": "World"}, "Hello") +func Contains(t TestingT, s interface{}, contains interface{}, msgAndArgs ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.Contains(t, s, contains, msgAndArgs...) { + return + } + t.FailNow() +} + +// Containsf asserts that the specified string, list(array, slice...) or map contains the +// specified substring or element. +// +// assert.Containsf(t, "Hello World", "World", "error message %s", "formatted") +// assert.Containsf(t, ["Hello", "World"], "World", "error message %s", "formatted") +// assert.Containsf(t, {"Hello": "World"}, "Hello", "error message %s", "formatted") +func Containsf(t TestingT, s interface{}, contains interface{}, msg string, args ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.Containsf(t, s, contains, msg, args...) { + return + } + t.FailNow() +} + +// DirExists checks whether a directory exists in the given path. It also fails +// if the path is a file rather a directory or there is an error checking whether it exists. +func DirExists(t TestingT, path string, msgAndArgs ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.DirExists(t, path, msgAndArgs...) { + return + } + t.FailNow() +} + +// DirExistsf checks whether a directory exists in the given path. It also fails +// if the path is a file rather a directory or there is an error checking whether it exists. +func DirExistsf(t TestingT, path string, msg string, args ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.DirExistsf(t, path, msg, args...) { + return + } + t.FailNow() +} + +// ElementsMatch asserts that the specified listA(array, slice...) is equal to specified +// listB(array, slice...) ignoring the order of the elements. If there are duplicate elements, +// the number of appearances of each of them in both lists should match. +// +// assert.ElementsMatch(t, [1, 3, 2, 3], [1, 3, 3, 2]) +func ElementsMatch(t TestingT, listA interface{}, listB interface{}, msgAndArgs ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.ElementsMatch(t, listA, listB, msgAndArgs...) { + return + } + t.FailNow() +} + +// ElementsMatchf asserts that the specified listA(array, slice...) is equal to specified +// listB(array, slice...) ignoring the order of the elements. If there are duplicate elements, +// the number of appearances of each of them in both lists should match. +// +// assert.ElementsMatchf(t, [1, 3, 2, 3], [1, 3, 3, 2], "error message %s", "formatted") +func ElementsMatchf(t TestingT, listA interface{}, listB interface{}, msg string, args ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.ElementsMatchf(t, listA, listB, msg, args...) { + return + } + t.FailNow() +} + +// Empty asserts that the specified object is empty. I.e. nil, "", false, 0 or either +// a slice or a channel with len == 0. +// +// assert.Empty(t, obj) +func Empty(t TestingT, object interface{}, msgAndArgs ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.Empty(t, object, msgAndArgs...) { + return + } + t.FailNow() +} + +// Emptyf asserts that the specified object is empty. I.e. nil, "", false, 0 or either +// a slice or a channel with len == 0. +// +// assert.Emptyf(t, obj, "error message %s", "formatted") +func Emptyf(t TestingT, object interface{}, msg string, args ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.Emptyf(t, object, msg, args...) { + return + } + t.FailNow() +} + +// Equal asserts that two objects are equal. +// +// assert.Equal(t, 123, 123) +// +// Pointer variable equality is determined based on the equality of the +// referenced values (as opposed to the memory addresses). Function equality +// cannot be determined and will always fail. +func Equal(t TestingT, expected interface{}, actual interface{}, msgAndArgs ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.Equal(t, expected, actual, msgAndArgs...) { + return + } + t.FailNow() +} + +// EqualError asserts that a function returned an error (i.e. not `nil`) +// and that it is equal to the provided error. +// +// actualObj, err := SomeFunction() +// assert.EqualError(t, err, expectedErrorString) +func EqualError(t TestingT, theError error, errString string, msgAndArgs ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.EqualError(t, theError, errString, msgAndArgs...) { + return + } + t.FailNow() +} + +// EqualErrorf asserts that a function returned an error (i.e. not `nil`) +// and that it is equal to the provided error. +// +// actualObj, err := SomeFunction() +// assert.EqualErrorf(t, err, expectedErrorString, "error message %s", "formatted") +func EqualErrorf(t TestingT, theError error, errString string, msg string, args ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.EqualErrorf(t, theError, errString, msg, args...) { + return + } + t.FailNow() +} + +// EqualValues asserts that two objects are equal or convertable to the same types +// and equal. +// +// assert.EqualValues(t, uint32(123), int32(123)) +func EqualValues(t TestingT, expected interface{}, actual interface{}, msgAndArgs ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.EqualValues(t, expected, actual, msgAndArgs...) { + return + } + t.FailNow() +} + +// EqualValuesf asserts that two objects are equal or convertable to the same types +// and equal. +// +// assert.EqualValuesf(t, uint32(123), int32(123), "error message %s", "formatted") +func EqualValuesf(t TestingT, expected interface{}, actual interface{}, msg string, args ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.EqualValuesf(t, expected, actual, msg, args...) { + return + } + t.FailNow() +} + +// Equalf asserts that two objects are equal. +// +// assert.Equalf(t, 123, 123, "error message %s", "formatted") +// +// Pointer variable equality is determined based on the equality of the +// referenced values (as opposed to the memory addresses). Function equality +// cannot be determined and will always fail. +func Equalf(t TestingT, expected interface{}, actual interface{}, msg string, args ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.Equalf(t, expected, actual, msg, args...) { + return + } + t.FailNow() +} + +// Error asserts that a function returned an error (i.e. not `nil`). +// +// actualObj, err := SomeFunction() +// if assert.Error(t, err) { +// assert.Equal(t, expectedError, err) +// } +func Error(t TestingT, err error, msgAndArgs ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.Error(t, err, msgAndArgs...) { + return + } + t.FailNow() +} + +// ErrorAs asserts that at least one of the errors in err's chain matches target, and if so, sets target to that error value. +// This is a wrapper for errors.As. +func ErrorAs(t TestingT, err error, target interface{}, msgAndArgs ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.ErrorAs(t, err, target, msgAndArgs...) { + return + } + t.FailNow() +} + +// ErrorAsf asserts that at least one of the errors in err's chain matches target, and if so, sets target to that error value. +// This is a wrapper for errors.As. +func ErrorAsf(t TestingT, err error, target interface{}, msg string, args ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.ErrorAsf(t, err, target, msg, args...) { + return + } + t.FailNow() +} + +// ErrorIs asserts that at least one of the errors in err's chain matches target. +// This is a wrapper for errors.Is. +func ErrorIs(t TestingT, err error, target error, msgAndArgs ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.ErrorIs(t, err, target, msgAndArgs...) { + return + } + t.FailNow() +} + +// ErrorIsf asserts that at least one of the errors in err's chain matches target. +// This is a wrapper for errors.Is. +func ErrorIsf(t TestingT, err error, target error, msg string, args ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.ErrorIsf(t, err, target, msg, args...) { + return + } + t.FailNow() +} + +// Errorf asserts that a function returned an error (i.e. not `nil`). +// +// actualObj, err := SomeFunction() +// if assert.Errorf(t, err, "error message %s", "formatted") { +// assert.Equal(t, expectedErrorf, err) +// } +func Errorf(t TestingT, err error, msg string, args ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.Errorf(t, err, msg, args...) { + return + } + t.FailNow() +} + +// Eventually asserts that given condition will be met in waitFor time, +// periodically checking target function each tick. +// +// assert.Eventually(t, func() bool { return true; }, time.Second, 10*time.Millisecond) +func Eventually(t TestingT, condition func() bool, waitFor time.Duration, tick time.Duration, msgAndArgs ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.Eventually(t, condition, waitFor, tick, msgAndArgs...) { + return + } + t.FailNow() +} + +// Eventuallyf asserts that given condition will be met in waitFor time, +// periodically checking target function each tick. +// +// assert.Eventuallyf(t, func() bool { return true; }, time.Second, 10*time.Millisecond, "error message %s", "formatted") +func Eventuallyf(t TestingT, condition func() bool, waitFor time.Duration, tick time.Duration, msg string, args ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.Eventuallyf(t, condition, waitFor, tick, msg, args...) { + return + } + t.FailNow() +} + +// Exactly asserts that two objects are equal in value and type. +// +// assert.Exactly(t, int32(123), int64(123)) +func Exactly(t TestingT, expected interface{}, actual interface{}, msgAndArgs ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.Exactly(t, expected, actual, msgAndArgs...) { + return + } + t.FailNow() +} + +// Exactlyf asserts that two objects are equal in value and type. +// +// assert.Exactlyf(t, int32(123), int64(123), "error message %s", "formatted") +func Exactlyf(t TestingT, expected interface{}, actual interface{}, msg string, args ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.Exactlyf(t, expected, actual, msg, args...) { + return + } + t.FailNow() +} + +// Fail reports a failure through +func Fail(t TestingT, failureMessage string, msgAndArgs ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.Fail(t, failureMessage, msgAndArgs...) { + return + } + t.FailNow() +} + +// FailNow fails test +func FailNow(t TestingT, failureMessage string, msgAndArgs ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.FailNow(t, failureMessage, msgAndArgs...) { + return + } + t.FailNow() +} + +// FailNowf fails test +func FailNowf(t TestingT, failureMessage string, msg string, args ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.FailNowf(t, failureMessage, msg, args...) { + return + } + t.FailNow() +} + +// Failf reports a failure through +func Failf(t TestingT, failureMessage string, msg string, args ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.Failf(t, failureMessage, msg, args...) { + return + } + t.FailNow() +} + +// False asserts that the specified value is false. +// +// assert.False(t, myBool) +func False(t TestingT, value bool, msgAndArgs ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.False(t, value, msgAndArgs...) { + return + } + t.FailNow() +} + +// Falsef asserts that the specified value is false. +// +// assert.Falsef(t, myBool, "error message %s", "formatted") +func Falsef(t TestingT, value bool, msg string, args ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.Falsef(t, value, msg, args...) { + return + } + t.FailNow() +} + +// FileExists checks whether a file exists in the given path. It also fails if +// the path points to a directory or there is an error when trying to check the file. +func FileExists(t TestingT, path string, msgAndArgs ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.FileExists(t, path, msgAndArgs...) { + return + } + t.FailNow() +} + +// FileExistsf checks whether a file exists in the given path. It also fails if +// the path points to a directory or there is an error when trying to check the file. +func FileExistsf(t TestingT, path string, msg string, args ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.FileExistsf(t, path, msg, args...) { + return + } + t.FailNow() +} + +// Greater asserts that the first element is greater than the second +// +// assert.Greater(t, 2, 1) +// assert.Greater(t, float64(2), float64(1)) +// assert.Greater(t, "b", "a") +func Greater(t TestingT, e1 interface{}, e2 interface{}, msgAndArgs ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.Greater(t, e1, e2, msgAndArgs...) { + return + } + t.FailNow() +} + +// GreaterOrEqual asserts that the first element is greater than or equal to the second +// +// assert.GreaterOrEqual(t, 2, 1) +// assert.GreaterOrEqual(t, 2, 2) +// assert.GreaterOrEqual(t, "b", "a") +// assert.GreaterOrEqual(t, "b", "b") +func GreaterOrEqual(t TestingT, e1 interface{}, e2 interface{}, msgAndArgs ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.GreaterOrEqual(t, e1, e2, msgAndArgs...) { + return + } + t.FailNow() +} + +// GreaterOrEqualf asserts that the first element is greater than or equal to the second +// +// assert.GreaterOrEqualf(t, 2, 1, "error message %s", "formatted") +// assert.GreaterOrEqualf(t, 2, 2, "error message %s", "formatted") +// assert.GreaterOrEqualf(t, "b", "a", "error message %s", "formatted") +// assert.GreaterOrEqualf(t, "b", "b", "error message %s", "formatted") +func GreaterOrEqualf(t TestingT, e1 interface{}, e2 interface{}, msg string, args ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.GreaterOrEqualf(t, e1, e2, msg, args...) { + return + } + t.FailNow() +} + +// Greaterf asserts that the first element is greater than the second +// +// assert.Greaterf(t, 2, 1, "error message %s", "formatted") +// assert.Greaterf(t, float64(2), float64(1), "error message %s", "formatted") +// assert.Greaterf(t, "b", "a", "error message %s", "formatted") +func Greaterf(t TestingT, e1 interface{}, e2 interface{}, msg string, args ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.Greaterf(t, e1, e2, msg, args...) { + return + } + t.FailNow() +} + +// HTTPBodyContains asserts that a specified handler returns a +// body that contains a string. +// +// assert.HTTPBodyContains(t, myHandler, "GET", "www.google.com", nil, "I'm Feeling Lucky") +// +// Returns whether the assertion was successful (true) or not (false). +func HTTPBodyContains(t TestingT, handler http.HandlerFunc, method string, url string, values url.Values, str interface{}, msgAndArgs ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.HTTPBodyContains(t, handler, method, url, values, str, msgAndArgs...) { + return + } + t.FailNow() +} + +// HTTPBodyContainsf asserts that a specified handler returns a +// body that contains a string. +// +// assert.HTTPBodyContainsf(t, myHandler, "GET", "www.google.com", nil, "I'm Feeling Lucky", "error message %s", "formatted") +// +// Returns whether the assertion was successful (true) or not (false). +func HTTPBodyContainsf(t TestingT, handler http.HandlerFunc, method string, url string, values url.Values, str interface{}, msg string, args ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.HTTPBodyContainsf(t, handler, method, url, values, str, msg, args...) { + return + } + t.FailNow() +} + +// HTTPBodyNotContains asserts that a specified handler returns a +// body that does not contain a string. +// +// assert.HTTPBodyNotContains(t, myHandler, "GET", "www.google.com", nil, "I'm Feeling Lucky") +// +// Returns whether the assertion was successful (true) or not (false). +func HTTPBodyNotContains(t TestingT, handler http.HandlerFunc, method string, url string, values url.Values, str interface{}, msgAndArgs ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.HTTPBodyNotContains(t, handler, method, url, values, str, msgAndArgs...) { + return + } + t.FailNow() +} + +// HTTPBodyNotContainsf asserts that a specified handler returns a +// body that does not contain a string. +// +// assert.HTTPBodyNotContainsf(t, myHandler, "GET", "www.google.com", nil, "I'm Feeling Lucky", "error message %s", "formatted") +// +// Returns whether the assertion was successful (true) or not (false). +func HTTPBodyNotContainsf(t TestingT, handler http.HandlerFunc, method string, url string, values url.Values, str interface{}, msg string, args ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.HTTPBodyNotContainsf(t, handler, method, url, values, str, msg, args...) { + return + } + t.FailNow() +} + +// HTTPError asserts that a specified handler returns an error status code. +// +// assert.HTTPError(t, myHandler, "POST", "/a/b/c", url.Values{"a": []string{"b", "c"}} +// +// Returns whether the assertion was successful (true) or not (false). +func HTTPError(t TestingT, handler http.HandlerFunc, method string, url string, values url.Values, msgAndArgs ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.HTTPError(t, handler, method, url, values, msgAndArgs...) { + return + } + t.FailNow() +} + +// HTTPErrorf asserts that a specified handler returns an error status code. +// +// assert.HTTPErrorf(t, myHandler, "POST", "/a/b/c", url.Values{"a": []string{"b", "c"}} +// +// Returns whether the assertion was successful (true) or not (false). +func HTTPErrorf(t TestingT, handler http.HandlerFunc, method string, url string, values url.Values, msg string, args ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.HTTPErrorf(t, handler, method, url, values, msg, args...) { + return + } + t.FailNow() +} + +// HTTPRedirect asserts that a specified handler returns a redirect status code. +// +// assert.HTTPRedirect(t, myHandler, "GET", "/a/b/c", url.Values{"a": []string{"b", "c"}} +// +// Returns whether the assertion was successful (true) or not (false). +func HTTPRedirect(t TestingT, handler http.HandlerFunc, method string, url string, values url.Values, msgAndArgs ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.HTTPRedirect(t, handler, method, url, values, msgAndArgs...) { + return + } + t.FailNow() +} + +// HTTPRedirectf asserts that a specified handler returns a redirect status code. +// +// assert.HTTPRedirectf(t, myHandler, "GET", "/a/b/c", url.Values{"a": []string{"b", "c"}} +// +// Returns whether the assertion was successful (true) or not (false). +func HTTPRedirectf(t TestingT, handler http.HandlerFunc, method string, url string, values url.Values, msg string, args ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.HTTPRedirectf(t, handler, method, url, values, msg, args...) { + return + } + t.FailNow() +} + +// HTTPStatusCode asserts that a specified handler returns a specified status code. +// +// assert.HTTPStatusCode(t, myHandler, "GET", "/notImplemented", nil, 501) +// +// Returns whether the assertion was successful (true) or not (false). +func HTTPStatusCode(t TestingT, handler http.HandlerFunc, method string, url string, values url.Values, statuscode int, msgAndArgs ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.HTTPStatusCode(t, handler, method, url, values, statuscode, msgAndArgs...) { + return + } + t.FailNow() +} + +// HTTPStatusCodef asserts that a specified handler returns a specified status code. +// +// assert.HTTPStatusCodef(t, myHandler, "GET", "/notImplemented", nil, 501, "error message %s", "formatted") +// +// Returns whether the assertion was successful (true) or not (false). +func HTTPStatusCodef(t TestingT, handler http.HandlerFunc, method string, url string, values url.Values, statuscode int, msg string, args ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.HTTPStatusCodef(t, handler, method, url, values, statuscode, msg, args...) { + return + } + t.FailNow() +} + +// HTTPSuccess asserts that a specified handler returns a success status code. +// +// assert.HTTPSuccess(t, myHandler, "POST", "http://www.google.com", nil) +// +// Returns whether the assertion was successful (true) or not (false). +func HTTPSuccess(t TestingT, handler http.HandlerFunc, method string, url string, values url.Values, msgAndArgs ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.HTTPSuccess(t, handler, method, url, values, msgAndArgs...) { + return + } + t.FailNow() +} + +// HTTPSuccessf asserts that a specified handler returns a success status code. +// +// assert.HTTPSuccessf(t, myHandler, "POST", "http://www.google.com", nil, "error message %s", "formatted") +// +// Returns whether the assertion was successful (true) or not (false). +func HTTPSuccessf(t TestingT, handler http.HandlerFunc, method string, url string, values url.Values, msg string, args ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.HTTPSuccessf(t, handler, method, url, values, msg, args...) { + return + } + t.FailNow() +} + +// Implements asserts that an object is implemented by the specified interface. +// +// assert.Implements(t, (*MyInterface)(nil), new(MyObject)) +func Implements(t TestingT, interfaceObject interface{}, object interface{}, msgAndArgs ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.Implements(t, interfaceObject, object, msgAndArgs...) { + return + } + t.FailNow() +} + +// Implementsf asserts that an object is implemented by the specified interface. +// +// assert.Implementsf(t, (*MyInterface)(nil), new(MyObject), "error message %s", "formatted") +func Implementsf(t TestingT, interfaceObject interface{}, object interface{}, msg string, args ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.Implementsf(t, interfaceObject, object, msg, args...) { + return + } + t.FailNow() +} + +// InDelta asserts that the two numerals are within delta of each other. +// +// assert.InDelta(t, math.Pi, 22/7.0, 0.01) +func InDelta(t TestingT, expected interface{}, actual interface{}, delta float64, msgAndArgs ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.InDelta(t, expected, actual, delta, msgAndArgs...) { + return + } + t.FailNow() +} + +// InDeltaMapValues is the same as InDelta, but it compares all values between two maps. Both maps must have exactly the same keys. +func InDeltaMapValues(t TestingT, expected interface{}, actual interface{}, delta float64, msgAndArgs ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.InDeltaMapValues(t, expected, actual, delta, msgAndArgs...) { + return + } + t.FailNow() +} + +// InDeltaMapValuesf is the same as InDelta, but it compares all values between two maps. Both maps must have exactly the same keys. +func InDeltaMapValuesf(t TestingT, expected interface{}, actual interface{}, delta float64, msg string, args ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.InDeltaMapValuesf(t, expected, actual, delta, msg, args...) { + return + } + t.FailNow() +} + +// InDeltaSlice is the same as InDelta, except it compares two slices. +func InDeltaSlice(t TestingT, expected interface{}, actual interface{}, delta float64, msgAndArgs ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.InDeltaSlice(t, expected, actual, delta, msgAndArgs...) { + return + } + t.FailNow() +} + +// InDeltaSlicef is the same as InDelta, except it compares two slices. +func InDeltaSlicef(t TestingT, expected interface{}, actual interface{}, delta float64, msg string, args ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.InDeltaSlicef(t, expected, actual, delta, msg, args...) { + return + } + t.FailNow() +} + +// InDeltaf asserts that the two numerals are within delta of each other. +// +// assert.InDeltaf(t, math.Pi, 22/7.0, 0.01, "error message %s", "formatted") +func InDeltaf(t TestingT, expected interface{}, actual interface{}, delta float64, msg string, args ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.InDeltaf(t, expected, actual, delta, msg, args...) { + return + } + t.FailNow() +} + +// InEpsilon asserts that expected and actual have a relative error less than epsilon +func InEpsilon(t TestingT, expected interface{}, actual interface{}, epsilon float64, msgAndArgs ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.InEpsilon(t, expected, actual, epsilon, msgAndArgs...) { + return + } + t.FailNow() +} + +// InEpsilonSlice is the same as InEpsilon, except it compares each value from two slices. +func InEpsilonSlice(t TestingT, expected interface{}, actual interface{}, epsilon float64, msgAndArgs ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.InEpsilonSlice(t, expected, actual, epsilon, msgAndArgs...) { + return + } + t.FailNow() +} + +// InEpsilonSlicef is the same as InEpsilon, except it compares each value from two slices. +func InEpsilonSlicef(t TestingT, expected interface{}, actual interface{}, epsilon float64, msg string, args ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.InEpsilonSlicef(t, expected, actual, epsilon, msg, args...) { + return + } + t.FailNow() +} + +// InEpsilonf asserts that expected and actual have a relative error less than epsilon +func InEpsilonf(t TestingT, expected interface{}, actual interface{}, epsilon float64, msg string, args ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.InEpsilonf(t, expected, actual, epsilon, msg, args...) { + return + } + t.FailNow() +} + +// IsDecreasing asserts that the collection is decreasing +// +// assert.IsDecreasing(t, []int{2, 1, 0}) +// assert.IsDecreasing(t, []float{2, 1}) +// assert.IsDecreasing(t, []string{"b", "a"}) +func IsDecreasing(t TestingT, object interface{}, msgAndArgs ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.IsDecreasing(t, object, msgAndArgs...) { + return + } + t.FailNow() +} + +// IsDecreasingf asserts that the collection is decreasing +// +// assert.IsDecreasingf(t, []int{2, 1, 0}, "error message %s", "formatted") +// assert.IsDecreasingf(t, []float{2, 1}, "error message %s", "formatted") +// assert.IsDecreasingf(t, []string{"b", "a"}, "error message %s", "formatted") +func IsDecreasingf(t TestingT, object interface{}, msg string, args ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.IsDecreasingf(t, object, msg, args...) { + return + } + t.FailNow() +} + +// IsIncreasing asserts that the collection is increasing +// +// assert.IsIncreasing(t, []int{1, 2, 3}) +// assert.IsIncreasing(t, []float{1, 2}) +// assert.IsIncreasing(t, []string{"a", "b"}) +func IsIncreasing(t TestingT, object interface{}, msgAndArgs ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.IsIncreasing(t, object, msgAndArgs...) { + return + } + t.FailNow() +} + +// IsIncreasingf asserts that the collection is increasing +// +// assert.IsIncreasingf(t, []int{1, 2, 3}, "error message %s", "formatted") +// assert.IsIncreasingf(t, []float{1, 2}, "error message %s", "formatted") +// assert.IsIncreasingf(t, []string{"a", "b"}, "error message %s", "formatted") +func IsIncreasingf(t TestingT, object interface{}, msg string, args ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.IsIncreasingf(t, object, msg, args...) { + return + } + t.FailNow() +} + +// IsNonDecreasing asserts that the collection is not decreasing +// +// assert.IsNonDecreasing(t, []int{1, 1, 2}) +// assert.IsNonDecreasing(t, []float{1, 2}) +// assert.IsNonDecreasing(t, []string{"a", "b"}) +func IsNonDecreasing(t TestingT, object interface{}, msgAndArgs ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.IsNonDecreasing(t, object, msgAndArgs...) { + return + } + t.FailNow() +} + +// IsNonDecreasingf asserts that the collection is not decreasing +// +// assert.IsNonDecreasingf(t, []int{1, 1, 2}, "error message %s", "formatted") +// assert.IsNonDecreasingf(t, []float{1, 2}, "error message %s", "formatted") +// assert.IsNonDecreasingf(t, []string{"a", "b"}, "error message %s", "formatted") +func IsNonDecreasingf(t TestingT, object interface{}, msg string, args ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.IsNonDecreasingf(t, object, msg, args...) { + return + } + t.FailNow() +} + +// IsNonIncreasing asserts that the collection is not increasing +// +// assert.IsNonIncreasing(t, []int{2, 1, 1}) +// assert.IsNonIncreasing(t, []float{2, 1}) +// assert.IsNonIncreasing(t, []string{"b", "a"}) +func IsNonIncreasing(t TestingT, object interface{}, msgAndArgs ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.IsNonIncreasing(t, object, msgAndArgs...) { + return + } + t.FailNow() +} + +// IsNonIncreasingf asserts that the collection is not increasing +// +// assert.IsNonIncreasingf(t, []int{2, 1, 1}, "error message %s", "formatted") +// assert.IsNonIncreasingf(t, []float{2, 1}, "error message %s", "formatted") +// assert.IsNonIncreasingf(t, []string{"b", "a"}, "error message %s", "formatted") +func IsNonIncreasingf(t TestingT, object interface{}, msg string, args ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.IsNonIncreasingf(t, object, msg, args...) { + return + } + t.FailNow() +} + +// IsType asserts that the specified objects are of the same type. +func IsType(t TestingT, expectedType interface{}, object interface{}, msgAndArgs ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.IsType(t, expectedType, object, msgAndArgs...) { + return + } + t.FailNow() +} + +// IsTypef asserts that the specified objects are of the same type. +func IsTypef(t TestingT, expectedType interface{}, object interface{}, msg string, args ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.IsTypef(t, expectedType, object, msg, args...) { + return + } + t.FailNow() +} + +// JSONEq asserts that two JSON strings are equivalent. +// +// assert.JSONEq(t, `{"hello": "world", "foo": "bar"}`, `{"foo": "bar", "hello": "world"}`) +func JSONEq(t TestingT, expected string, actual string, msgAndArgs ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.JSONEq(t, expected, actual, msgAndArgs...) { + return + } + t.FailNow() +} + +// JSONEqf asserts that two JSON strings are equivalent. +// +// assert.JSONEqf(t, `{"hello": "world", "foo": "bar"}`, `{"foo": "bar", "hello": "world"}`, "error message %s", "formatted") +func JSONEqf(t TestingT, expected string, actual string, msg string, args ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.JSONEqf(t, expected, actual, msg, args...) { + return + } + t.FailNow() +} + +// Len asserts that the specified object has specific length. +// Len also fails if the object has a type that len() not accept. +// +// assert.Len(t, mySlice, 3) +func Len(t TestingT, object interface{}, length int, msgAndArgs ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.Len(t, object, length, msgAndArgs...) { + return + } + t.FailNow() +} + +// Lenf asserts that the specified object has specific length. +// Lenf also fails if the object has a type that len() not accept. +// +// assert.Lenf(t, mySlice, 3, "error message %s", "formatted") +func Lenf(t TestingT, object interface{}, length int, msg string, args ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.Lenf(t, object, length, msg, args...) { + return + } + t.FailNow() +} + +// Less asserts that the first element is less than the second +// +// assert.Less(t, 1, 2) +// assert.Less(t, float64(1), float64(2)) +// assert.Less(t, "a", "b") +func Less(t TestingT, e1 interface{}, e2 interface{}, msgAndArgs ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.Less(t, e1, e2, msgAndArgs...) { + return + } + t.FailNow() +} + +// LessOrEqual asserts that the first element is less than or equal to the second +// +// assert.LessOrEqual(t, 1, 2) +// assert.LessOrEqual(t, 2, 2) +// assert.LessOrEqual(t, "a", "b") +// assert.LessOrEqual(t, "b", "b") +func LessOrEqual(t TestingT, e1 interface{}, e2 interface{}, msgAndArgs ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.LessOrEqual(t, e1, e2, msgAndArgs...) { + return + } + t.FailNow() +} + +// LessOrEqualf asserts that the first element is less than or equal to the second +// +// assert.LessOrEqualf(t, 1, 2, "error message %s", "formatted") +// assert.LessOrEqualf(t, 2, 2, "error message %s", "formatted") +// assert.LessOrEqualf(t, "a", "b", "error message %s", "formatted") +// assert.LessOrEqualf(t, "b", "b", "error message %s", "formatted") +func LessOrEqualf(t TestingT, e1 interface{}, e2 interface{}, msg string, args ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.LessOrEqualf(t, e1, e2, msg, args...) { + return + } + t.FailNow() +} + +// Lessf asserts that the first element is less than the second +// +// assert.Lessf(t, 1, 2, "error message %s", "formatted") +// assert.Lessf(t, float64(1), float64(2), "error message %s", "formatted") +// assert.Lessf(t, "a", "b", "error message %s", "formatted") +func Lessf(t TestingT, e1 interface{}, e2 interface{}, msg string, args ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.Lessf(t, e1, e2, msg, args...) { + return + } + t.FailNow() +} + +// Negative asserts that the specified element is negative +// +// assert.Negative(t, -1) +// assert.Negative(t, -1.23) +func Negative(t TestingT, e interface{}, msgAndArgs ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.Negative(t, e, msgAndArgs...) { + return + } + t.FailNow() +} + +// Negativef asserts that the specified element is negative +// +// assert.Negativef(t, -1, "error message %s", "formatted") +// assert.Negativef(t, -1.23, "error message %s", "formatted") +func Negativef(t TestingT, e interface{}, msg string, args ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.Negativef(t, e, msg, args...) { + return + } + t.FailNow() +} + +// Never asserts that the given condition doesn't satisfy in waitFor time, +// periodically checking the target function each tick. +// +// assert.Never(t, func() bool { return false; }, time.Second, 10*time.Millisecond) +func Never(t TestingT, condition func() bool, waitFor time.Duration, tick time.Duration, msgAndArgs ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.Never(t, condition, waitFor, tick, msgAndArgs...) { + return + } + t.FailNow() +} + +// Neverf asserts that the given condition doesn't satisfy in waitFor time, +// periodically checking the target function each tick. +// +// assert.Neverf(t, func() bool { return false; }, time.Second, 10*time.Millisecond, "error message %s", "formatted") +func Neverf(t TestingT, condition func() bool, waitFor time.Duration, tick time.Duration, msg string, args ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.Neverf(t, condition, waitFor, tick, msg, args...) { + return + } + t.FailNow() +} + +// Nil asserts that the specified object is nil. +// +// assert.Nil(t, err) +func Nil(t TestingT, object interface{}, msgAndArgs ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.Nil(t, object, msgAndArgs...) { + return + } + t.FailNow() +} + +// Nilf asserts that the specified object is nil. +// +// assert.Nilf(t, err, "error message %s", "formatted") +func Nilf(t TestingT, object interface{}, msg string, args ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.Nilf(t, object, msg, args...) { + return + } + t.FailNow() +} + +// NoDirExists checks whether a directory does not exist in the given path. +// It fails if the path points to an existing _directory_ only. +func NoDirExists(t TestingT, path string, msgAndArgs ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.NoDirExists(t, path, msgAndArgs...) { + return + } + t.FailNow() +} + +// NoDirExistsf checks whether a directory does not exist in the given path. +// It fails if the path points to an existing _directory_ only. +func NoDirExistsf(t TestingT, path string, msg string, args ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.NoDirExistsf(t, path, msg, args...) { + return + } + t.FailNow() +} + +// NoError asserts that a function returned no error (i.e. `nil`). +// +// actualObj, err := SomeFunction() +// if assert.NoError(t, err) { +// assert.Equal(t, expectedObj, actualObj) +// } +func NoError(t TestingT, err error, msgAndArgs ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.NoError(t, err, msgAndArgs...) { + return + } + t.FailNow() +} + +// NoErrorf asserts that a function returned no error (i.e. `nil`). +// +// actualObj, err := SomeFunction() +// if assert.NoErrorf(t, err, "error message %s", "formatted") { +// assert.Equal(t, expectedObj, actualObj) +// } +func NoErrorf(t TestingT, err error, msg string, args ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.NoErrorf(t, err, msg, args...) { + return + } + t.FailNow() +} + +// NoFileExists checks whether a file does not exist in a given path. It fails +// if the path points to an existing _file_ only. +func NoFileExists(t TestingT, path string, msgAndArgs ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.NoFileExists(t, path, msgAndArgs...) { + return + } + t.FailNow() +} + +// NoFileExistsf checks whether a file does not exist in a given path. It fails +// if the path points to an existing _file_ only. +func NoFileExistsf(t TestingT, path string, msg string, args ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.NoFileExistsf(t, path, msg, args...) { + return + } + t.FailNow() +} + +// NotContains asserts that the specified string, list(array, slice...) or map does NOT contain the +// specified substring or element. +// +// assert.NotContains(t, "Hello World", "Earth") +// assert.NotContains(t, ["Hello", "World"], "Earth") +// assert.NotContains(t, {"Hello": "World"}, "Earth") +func NotContains(t TestingT, s interface{}, contains interface{}, msgAndArgs ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.NotContains(t, s, contains, msgAndArgs...) { + return + } + t.FailNow() +} + +// NotContainsf asserts that the specified string, list(array, slice...) or map does NOT contain the +// specified substring or element. +// +// assert.NotContainsf(t, "Hello World", "Earth", "error message %s", "formatted") +// assert.NotContainsf(t, ["Hello", "World"], "Earth", "error message %s", "formatted") +// assert.NotContainsf(t, {"Hello": "World"}, "Earth", "error message %s", "formatted") +func NotContainsf(t TestingT, s interface{}, contains interface{}, msg string, args ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.NotContainsf(t, s, contains, msg, args...) { + return + } + t.FailNow() +} + +// NotEmpty asserts that the specified object is NOT empty. I.e. not nil, "", false, 0 or either +// a slice or a channel with len == 0. +// +// if assert.NotEmpty(t, obj) { +// assert.Equal(t, "two", obj[1]) +// } +func NotEmpty(t TestingT, object interface{}, msgAndArgs ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.NotEmpty(t, object, msgAndArgs...) { + return + } + t.FailNow() +} + +// NotEmptyf asserts that the specified object is NOT empty. I.e. not nil, "", false, 0 or either +// a slice or a channel with len == 0. +// +// if assert.NotEmptyf(t, obj, "error message %s", "formatted") { +// assert.Equal(t, "two", obj[1]) +// } +func NotEmptyf(t TestingT, object interface{}, msg string, args ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.NotEmptyf(t, object, msg, args...) { + return + } + t.FailNow() +} + +// NotEqual asserts that the specified values are NOT equal. +// +// assert.NotEqual(t, obj1, obj2) +// +// Pointer variable equality is determined based on the equality of the +// referenced values (as opposed to the memory addresses). +func NotEqual(t TestingT, expected interface{}, actual interface{}, msgAndArgs ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.NotEqual(t, expected, actual, msgAndArgs...) { + return + } + t.FailNow() +} + +// NotEqualValues asserts that two objects are not equal even when converted to the same type +// +// assert.NotEqualValues(t, obj1, obj2) +func NotEqualValues(t TestingT, expected interface{}, actual interface{}, msgAndArgs ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.NotEqualValues(t, expected, actual, msgAndArgs...) { + return + } + t.FailNow() +} + +// NotEqualValuesf asserts that two objects are not equal even when converted to the same type +// +// assert.NotEqualValuesf(t, obj1, obj2, "error message %s", "formatted") +func NotEqualValuesf(t TestingT, expected interface{}, actual interface{}, msg string, args ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.NotEqualValuesf(t, expected, actual, msg, args...) { + return + } + t.FailNow() +} + +// NotEqualf asserts that the specified values are NOT equal. +// +// assert.NotEqualf(t, obj1, obj2, "error message %s", "formatted") +// +// Pointer variable equality is determined based on the equality of the +// referenced values (as opposed to the memory addresses). +func NotEqualf(t TestingT, expected interface{}, actual interface{}, msg string, args ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.NotEqualf(t, expected, actual, msg, args...) { + return + } + t.FailNow() +} + +// NotErrorIs asserts that at none of the errors in err's chain matches target. +// This is a wrapper for errors.Is. +func NotErrorIs(t TestingT, err error, target error, msgAndArgs ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.NotErrorIs(t, err, target, msgAndArgs...) { + return + } + t.FailNow() +} + +// NotErrorIsf asserts that at none of the errors in err's chain matches target. +// This is a wrapper for errors.Is. +func NotErrorIsf(t TestingT, err error, target error, msg string, args ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.NotErrorIsf(t, err, target, msg, args...) { + return + } + t.FailNow() +} + +// NotNil asserts that the specified object is not nil. +// +// assert.NotNil(t, err) +func NotNil(t TestingT, object interface{}, msgAndArgs ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.NotNil(t, object, msgAndArgs...) { + return + } + t.FailNow() +} + +// NotNilf asserts that the specified object is not nil. +// +// assert.NotNilf(t, err, "error message %s", "formatted") +func NotNilf(t TestingT, object interface{}, msg string, args ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.NotNilf(t, object, msg, args...) { + return + } + t.FailNow() +} + +// NotPanics asserts that the code inside the specified PanicTestFunc does NOT panic. +// +// assert.NotPanics(t, func(){ RemainCalm() }) +func NotPanics(t TestingT, f assert.PanicTestFunc, msgAndArgs ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.NotPanics(t, f, msgAndArgs...) { + return + } + t.FailNow() +} + +// NotPanicsf asserts that the code inside the specified PanicTestFunc does NOT panic. +// +// assert.NotPanicsf(t, func(){ RemainCalm() }, "error message %s", "formatted") +func NotPanicsf(t TestingT, f assert.PanicTestFunc, msg string, args ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.NotPanicsf(t, f, msg, args...) { + return + } + t.FailNow() +} + +// NotRegexp asserts that a specified regexp does not match a string. +// +// assert.NotRegexp(t, regexp.MustCompile("starts"), "it's starting") +// assert.NotRegexp(t, "^start", "it's not starting") +func NotRegexp(t TestingT, rx interface{}, str interface{}, msgAndArgs ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.NotRegexp(t, rx, str, msgAndArgs...) { + return + } + t.FailNow() +} + +// NotRegexpf asserts that a specified regexp does not match a string. +// +// assert.NotRegexpf(t, regexp.MustCompile("starts"), "it's starting", "error message %s", "formatted") +// assert.NotRegexpf(t, "^start", "it's not starting", "error message %s", "formatted") +func NotRegexpf(t TestingT, rx interface{}, str interface{}, msg string, args ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.NotRegexpf(t, rx, str, msg, args...) { + return + } + t.FailNow() +} + +// NotSame asserts that two pointers do not reference the same object. +// +// assert.NotSame(t, ptr1, ptr2) +// +// Both arguments must be pointer variables. Pointer variable sameness is +// determined based on the equality of both type and value. +func NotSame(t TestingT, expected interface{}, actual interface{}, msgAndArgs ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.NotSame(t, expected, actual, msgAndArgs...) { + return + } + t.FailNow() +} + +// NotSamef asserts that two pointers do not reference the same object. +// +// assert.NotSamef(t, ptr1, ptr2, "error message %s", "formatted") +// +// Both arguments must be pointer variables. Pointer variable sameness is +// determined based on the equality of both type and value. +func NotSamef(t TestingT, expected interface{}, actual interface{}, msg string, args ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.NotSamef(t, expected, actual, msg, args...) { + return + } + t.FailNow() +} + +// NotSubset asserts that the specified list(array, slice...) contains not all +// elements given in the specified subset(array, slice...). +// +// assert.NotSubset(t, [1, 3, 4], [1, 2], "But [1, 3, 4] does not contain [1, 2]") +func NotSubset(t TestingT, list interface{}, subset interface{}, msgAndArgs ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.NotSubset(t, list, subset, msgAndArgs...) { + return + } + t.FailNow() +} + +// NotSubsetf asserts that the specified list(array, slice...) contains not all +// elements given in the specified subset(array, slice...). +// +// assert.NotSubsetf(t, [1, 3, 4], [1, 2], "But [1, 3, 4] does not contain [1, 2]", "error message %s", "formatted") +func NotSubsetf(t TestingT, list interface{}, subset interface{}, msg string, args ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.NotSubsetf(t, list, subset, msg, args...) { + return + } + t.FailNow() +} + +// NotZero asserts that i is not the zero value for its type. +func NotZero(t TestingT, i interface{}, msgAndArgs ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.NotZero(t, i, msgAndArgs...) { + return + } + t.FailNow() +} + +// NotZerof asserts that i is not the zero value for its type. +func NotZerof(t TestingT, i interface{}, msg string, args ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.NotZerof(t, i, msg, args...) { + return + } + t.FailNow() +} + +// Panics asserts that the code inside the specified PanicTestFunc panics. +// +// assert.Panics(t, func(){ GoCrazy() }) +func Panics(t TestingT, f assert.PanicTestFunc, msgAndArgs ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.Panics(t, f, msgAndArgs...) { + return + } + t.FailNow() +} + +// PanicsWithError asserts that the code inside the specified PanicTestFunc +// panics, and that the recovered panic value is an error that satisfies the +// EqualError comparison. +// +// assert.PanicsWithError(t, "crazy error", func(){ GoCrazy() }) +func PanicsWithError(t TestingT, errString string, f assert.PanicTestFunc, msgAndArgs ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.PanicsWithError(t, errString, f, msgAndArgs...) { + return + } + t.FailNow() +} + +// PanicsWithErrorf asserts that the code inside the specified PanicTestFunc +// panics, and that the recovered panic value is an error that satisfies the +// EqualError comparison. +// +// assert.PanicsWithErrorf(t, "crazy error", func(){ GoCrazy() }, "error message %s", "formatted") +func PanicsWithErrorf(t TestingT, errString string, f assert.PanicTestFunc, msg string, args ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.PanicsWithErrorf(t, errString, f, msg, args...) { + return + } + t.FailNow() +} + +// PanicsWithValue asserts that the code inside the specified PanicTestFunc panics, and that +// the recovered panic value equals the expected panic value. +// +// assert.PanicsWithValue(t, "crazy error", func(){ GoCrazy() }) +func PanicsWithValue(t TestingT, expected interface{}, f assert.PanicTestFunc, msgAndArgs ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.PanicsWithValue(t, expected, f, msgAndArgs...) { + return + } + t.FailNow() +} + +// PanicsWithValuef asserts that the code inside the specified PanicTestFunc panics, and that +// the recovered panic value equals the expected panic value. +// +// assert.PanicsWithValuef(t, "crazy error", func(){ GoCrazy() }, "error message %s", "formatted") +func PanicsWithValuef(t TestingT, expected interface{}, f assert.PanicTestFunc, msg string, args ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.PanicsWithValuef(t, expected, f, msg, args...) { + return + } + t.FailNow() +} + +// Panicsf asserts that the code inside the specified PanicTestFunc panics. +// +// assert.Panicsf(t, func(){ GoCrazy() }, "error message %s", "formatted") +func Panicsf(t TestingT, f assert.PanicTestFunc, msg string, args ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.Panicsf(t, f, msg, args...) { + return + } + t.FailNow() +} + +// Positive asserts that the specified element is positive +// +// assert.Positive(t, 1) +// assert.Positive(t, 1.23) +func Positive(t TestingT, e interface{}, msgAndArgs ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.Positive(t, e, msgAndArgs...) { + return + } + t.FailNow() +} + +// Positivef asserts that the specified element is positive +// +// assert.Positivef(t, 1, "error message %s", "formatted") +// assert.Positivef(t, 1.23, "error message %s", "formatted") +func Positivef(t TestingT, e interface{}, msg string, args ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.Positivef(t, e, msg, args...) { + return + } + t.FailNow() +} + +// Regexp asserts that a specified regexp matches a string. +// +// assert.Regexp(t, regexp.MustCompile("start"), "it's starting") +// assert.Regexp(t, "start...$", "it's not starting") +func Regexp(t TestingT, rx interface{}, str interface{}, msgAndArgs ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.Regexp(t, rx, str, msgAndArgs...) { + return + } + t.FailNow() +} + +// Regexpf asserts that a specified regexp matches a string. +// +// assert.Regexpf(t, regexp.MustCompile("start"), "it's starting", "error message %s", "formatted") +// assert.Regexpf(t, "start...$", "it's not starting", "error message %s", "formatted") +func Regexpf(t TestingT, rx interface{}, str interface{}, msg string, args ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.Regexpf(t, rx, str, msg, args...) { + return + } + t.FailNow() +} + +// Same asserts that two pointers reference the same object. +// +// assert.Same(t, ptr1, ptr2) +// +// Both arguments must be pointer variables. Pointer variable sameness is +// determined based on the equality of both type and value. +func Same(t TestingT, expected interface{}, actual interface{}, msgAndArgs ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.Same(t, expected, actual, msgAndArgs...) { + return + } + t.FailNow() +} + +// Samef asserts that two pointers reference the same object. +// +// assert.Samef(t, ptr1, ptr2, "error message %s", "formatted") +// +// Both arguments must be pointer variables. Pointer variable sameness is +// determined based on the equality of both type and value. +func Samef(t TestingT, expected interface{}, actual interface{}, msg string, args ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.Samef(t, expected, actual, msg, args...) { + return + } + t.FailNow() +} + +// Subset asserts that the specified list(array, slice...) contains all +// elements given in the specified subset(array, slice...). +// +// assert.Subset(t, [1, 2, 3], [1, 2], "But [1, 2, 3] does contain [1, 2]") +func Subset(t TestingT, list interface{}, subset interface{}, msgAndArgs ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.Subset(t, list, subset, msgAndArgs...) { + return + } + t.FailNow() +} + +// Subsetf asserts that the specified list(array, slice...) contains all +// elements given in the specified subset(array, slice...). +// +// assert.Subsetf(t, [1, 2, 3], [1, 2], "But [1, 2, 3] does contain [1, 2]", "error message %s", "formatted") +func Subsetf(t TestingT, list interface{}, subset interface{}, msg string, args ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.Subsetf(t, list, subset, msg, args...) { + return + } + t.FailNow() +} + +// True asserts that the specified value is true. +// +// assert.True(t, myBool) +func True(t TestingT, value bool, msgAndArgs ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.True(t, value, msgAndArgs...) { + return + } + t.FailNow() +} + +// Truef asserts that the specified value is true. +// +// assert.Truef(t, myBool, "error message %s", "formatted") +func Truef(t TestingT, value bool, msg string, args ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.Truef(t, value, msg, args...) { + return + } + t.FailNow() +} + +// WithinDuration asserts that the two times are within duration delta of each other. +// +// assert.WithinDuration(t, time.Now(), time.Now(), 10*time.Second) +func WithinDuration(t TestingT, expected time.Time, actual time.Time, delta time.Duration, msgAndArgs ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.WithinDuration(t, expected, actual, delta, msgAndArgs...) { + return + } + t.FailNow() +} + +// WithinDurationf asserts that the two times are within duration delta of each other. +// +// assert.WithinDurationf(t, time.Now(), time.Now(), 10*time.Second, "error message %s", "formatted") +func WithinDurationf(t TestingT, expected time.Time, actual time.Time, delta time.Duration, msg string, args ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.WithinDurationf(t, expected, actual, delta, msg, args...) { + return + } + t.FailNow() +} + +// YAMLEq asserts that two YAML strings are equivalent. +func YAMLEq(t TestingT, expected string, actual string, msgAndArgs ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.YAMLEq(t, expected, actual, msgAndArgs...) { + return + } + t.FailNow() +} + +// YAMLEqf asserts that two YAML strings are equivalent. +func YAMLEqf(t TestingT, expected string, actual string, msg string, args ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.YAMLEqf(t, expected, actual, msg, args...) { + return + } + t.FailNow() +} + +// Zero asserts that i is the zero value for its type. +func Zero(t TestingT, i interface{}, msgAndArgs ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.Zero(t, i, msgAndArgs...) { + return + } + t.FailNow() +} + +// Zerof asserts that i is the zero value for its type. +func Zerof(t TestingT, i interface{}, msg string, args ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.Zerof(t, i, msg, args...) { + return + } + t.FailNow() +} diff --git a/vendor/github.com/stretchr/testify/require/require.go.tmpl b/vendor/github.com/stretchr/testify/require/require.go.tmpl new file mode 100644 index 00000000..55e42dde --- /dev/null +++ b/vendor/github.com/stretchr/testify/require/require.go.tmpl @@ -0,0 +1,6 @@ +{{.Comment}} +func {{.DocInfo.Name}}(t TestingT, {{.Params}}) { + if h, ok := t.(tHelper); ok { h.Helper() } + if assert.{{.DocInfo.Name}}(t, {{.ForwardedParams}}) { return } + t.FailNow() +} diff --git a/vendor/github.com/stretchr/testify/require/require_forward.go b/vendor/github.com/stretchr/testify/require/require_forward.go new file mode 100644 index 00000000..ed54a9d8 --- /dev/null +++ b/vendor/github.com/stretchr/testify/require/require_forward.go @@ -0,0 +1,1471 @@ +/* +* CODE GENERATED AUTOMATICALLY WITH github.com/stretchr/testify/_codegen +* THIS FILE MUST NOT BE EDITED BY HAND + */ + +package require + +import ( + assert "github.com/stretchr/testify/assert" + http "net/http" + url "net/url" + time "time" +) + +// Condition uses a Comparison to assert a complex condition. +func (a *Assertions) Condition(comp assert.Comparison, msgAndArgs ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + Condition(a.t, comp, msgAndArgs...) +} + +// Conditionf uses a Comparison to assert a complex condition. +func (a *Assertions) Conditionf(comp assert.Comparison, msg string, args ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + Conditionf(a.t, comp, msg, args...) +} + +// Contains asserts that the specified string, list(array, slice...) or map contains the +// specified substring or element. +// +// a.Contains("Hello World", "World") +// a.Contains(["Hello", "World"], "World") +// a.Contains({"Hello": "World"}, "Hello") +func (a *Assertions) Contains(s interface{}, contains interface{}, msgAndArgs ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + Contains(a.t, s, contains, msgAndArgs...) +} + +// Containsf asserts that the specified string, list(array, slice...) or map contains the +// specified substring or element. +// +// a.Containsf("Hello World", "World", "error message %s", "formatted") +// a.Containsf(["Hello", "World"], "World", "error message %s", "formatted") +// a.Containsf({"Hello": "World"}, "Hello", "error message %s", "formatted") +func (a *Assertions) Containsf(s interface{}, contains interface{}, msg string, args ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + Containsf(a.t, s, contains, msg, args...) +} + +// DirExists checks whether a directory exists in the given path. It also fails +// if the path is a file rather a directory or there is an error checking whether it exists. +func (a *Assertions) DirExists(path string, msgAndArgs ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + DirExists(a.t, path, msgAndArgs...) +} + +// DirExistsf checks whether a directory exists in the given path. It also fails +// if the path is a file rather a directory or there is an error checking whether it exists. +func (a *Assertions) DirExistsf(path string, msg string, args ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + DirExistsf(a.t, path, msg, args...) +} + +// ElementsMatch asserts that the specified listA(array, slice...) is equal to specified +// listB(array, slice...) ignoring the order of the elements. If there are duplicate elements, +// the number of appearances of each of them in both lists should match. +// +// a.ElementsMatch([1, 3, 2, 3], [1, 3, 3, 2]) +func (a *Assertions) ElementsMatch(listA interface{}, listB interface{}, msgAndArgs ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + ElementsMatch(a.t, listA, listB, msgAndArgs...) +} + +// ElementsMatchf asserts that the specified listA(array, slice...) is equal to specified +// listB(array, slice...) ignoring the order of the elements. If there are duplicate elements, +// the number of appearances of each of them in both lists should match. +// +// a.ElementsMatchf([1, 3, 2, 3], [1, 3, 3, 2], "error message %s", "formatted") +func (a *Assertions) ElementsMatchf(listA interface{}, listB interface{}, msg string, args ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + ElementsMatchf(a.t, listA, listB, msg, args...) +} + +// Empty asserts that the specified object is empty. I.e. nil, "", false, 0 or either +// a slice or a channel with len == 0. +// +// a.Empty(obj) +func (a *Assertions) Empty(object interface{}, msgAndArgs ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + Empty(a.t, object, msgAndArgs...) +} + +// Emptyf asserts that the specified object is empty. I.e. nil, "", false, 0 or either +// a slice or a channel with len == 0. +// +// a.Emptyf(obj, "error message %s", "formatted") +func (a *Assertions) Emptyf(object interface{}, msg string, args ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + Emptyf(a.t, object, msg, args...) +} + +// Equal asserts that two objects are equal. +// +// a.Equal(123, 123) +// +// Pointer variable equality is determined based on the equality of the +// referenced values (as opposed to the memory addresses). Function equality +// cannot be determined and will always fail. +func (a *Assertions) Equal(expected interface{}, actual interface{}, msgAndArgs ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + Equal(a.t, expected, actual, msgAndArgs...) +} + +// EqualError asserts that a function returned an error (i.e. not `nil`) +// and that it is equal to the provided error. +// +// actualObj, err := SomeFunction() +// a.EqualError(err, expectedErrorString) +func (a *Assertions) EqualError(theError error, errString string, msgAndArgs ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + EqualError(a.t, theError, errString, msgAndArgs...) +} + +// EqualErrorf asserts that a function returned an error (i.e. not `nil`) +// and that it is equal to the provided error. +// +// actualObj, err := SomeFunction() +// a.EqualErrorf(err, expectedErrorString, "error message %s", "formatted") +func (a *Assertions) EqualErrorf(theError error, errString string, msg string, args ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + EqualErrorf(a.t, theError, errString, msg, args...) +} + +// EqualValues asserts that two objects are equal or convertable to the same types +// and equal. +// +// a.EqualValues(uint32(123), int32(123)) +func (a *Assertions) EqualValues(expected interface{}, actual interface{}, msgAndArgs ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + EqualValues(a.t, expected, actual, msgAndArgs...) +} + +// EqualValuesf asserts that two objects are equal or convertable to the same types +// and equal. +// +// a.EqualValuesf(uint32(123), int32(123), "error message %s", "formatted") +func (a *Assertions) EqualValuesf(expected interface{}, actual interface{}, msg string, args ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + EqualValuesf(a.t, expected, actual, msg, args...) +} + +// Equalf asserts that two objects are equal. +// +// a.Equalf(123, 123, "error message %s", "formatted") +// +// Pointer variable equality is determined based on the equality of the +// referenced values (as opposed to the memory addresses). Function equality +// cannot be determined and will always fail. +func (a *Assertions) Equalf(expected interface{}, actual interface{}, msg string, args ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + Equalf(a.t, expected, actual, msg, args...) +} + +// Error asserts that a function returned an error (i.e. not `nil`). +// +// actualObj, err := SomeFunction() +// if a.Error(err) { +// assert.Equal(t, expectedError, err) +// } +func (a *Assertions) Error(err error, msgAndArgs ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + Error(a.t, err, msgAndArgs...) +} + +// ErrorAs asserts that at least one of the errors in err's chain matches target, and if so, sets target to that error value. +// This is a wrapper for errors.As. +func (a *Assertions) ErrorAs(err error, target interface{}, msgAndArgs ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + ErrorAs(a.t, err, target, msgAndArgs...) +} + +// ErrorAsf asserts that at least one of the errors in err's chain matches target, and if so, sets target to that error value. +// This is a wrapper for errors.As. +func (a *Assertions) ErrorAsf(err error, target interface{}, msg string, args ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + ErrorAsf(a.t, err, target, msg, args...) +} + +// ErrorIs asserts that at least one of the errors in err's chain matches target. +// This is a wrapper for errors.Is. +func (a *Assertions) ErrorIs(err error, target error, msgAndArgs ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + ErrorIs(a.t, err, target, msgAndArgs...) +} + +// ErrorIsf asserts that at least one of the errors in err's chain matches target. +// This is a wrapper for errors.Is. +func (a *Assertions) ErrorIsf(err error, target error, msg string, args ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + ErrorIsf(a.t, err, target, msg, args...) +} + +// Errorf asserts that a function returned an error (i.e. not `nil`). +// +// actualObj, err := SomeFunction() +// if a.Errorf(err, "error message %s", "formatted") { +// assert.Equal(t, expectedErrorf, err) +// } +func (a *Assertions) Errorf(err error, msg string, args ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + Errorf(a.t, err, msg, args...) +} + +// Eventually asserts that given condition will be met in waitFor time, +// periodically checking target function each tick. +// +// a.Eventually(func() bool { return true; }, time.Second, 10*time.Millisecond) +func (a *Assertions) Eventually(condition func() bool, waitFor time.Duration, tick time.Duration, msgAndArgs ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + Eventually(a.t, condition, waitFor, tick, msgAndArgs...) +} + +// Eventuallyf asserts that given condition will be met in waitFor time, +// periodically checking target function each tick. +// +// a.Eventuallyf(func() bool { return true; }, time.Second, 10*time.Millisecond, "error message %s", "formatted") +func (a *Assertions) Eventuallyf(condition func() bool, waitFor time.Duration, tick time.Duration, msg string, args ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + Eventuallyf(a.t, condition, waitFor, tick, msg, args...) +} + +// Exactly asserts that two objects are equal in value and type. +// +// a.Exactly(int32(123), int64(123)) +func (a *Assertions) Exactly(expected interface{}, actual interface{}, msgAndArgs ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + Exactly(a.t, expected, actual, msgAndArgs...) +} + +// Exactlyf asserts that two objects are equal in value and type. +// +// a.Exactlyf(int32(123), int64(123), "error message %s", "formatted") +func (a *Assertions) Exactlyf(expected interface{}, actual interface{}, msg string, args ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + Exactlyf(a.t, expected, actual, msg, args...) +} + +// Fail reports a failure through +func (a *Assertions) Fail(failureMessage string, msgAndArgs ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + Fail(a.t, failureMessage, msgAndArgs...) +} + +// FailNow fails test +func (a *Assertions) FailNow(failureMessage string, msgAndArgs ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + FailNow(a.t, failureMessage, msgAndArgs...) +} + +// FailNowf fails test +func (a *Assertions) FailNowf(failureMessage string, msg string, args ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + FailNowf(a.t, failureMessage, msg, args...) +} + +// Failf reports a failure through +func (a *Assertions) Failf(failureMessage string, msg string, args ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + Failf(a.t, failureMessage, msg, args...) +} + +// False asserts that the specified value is false. +// +// a.False(myBool) +func (a *Assertions) False(value bool, msgAndArgs ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + False(a.t, value, msgAndArgs...) +} + +// Falsef asserts that the specified value is false. +// +// a.Falsef(myBool, "error message %s", "formatted") +func (a *Assertions) Falsef(value bool, msg string, args ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + Falsef(a.t, value, msg, args...) +} + +// FileExists checks whether a file exists in the given path. It also fails if +// the path points to a directory or there is an error when trying to check the file. +func (a *Assertions) FileExists(path string, msgAndArgs ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + FileExists(a.t, path, msgAndArgs...) +} + +// FileExistsf checks whether a file exists in the given path. It also fails if +// the path points to a directory or there is an error when trying to check the file. +func (a *Assertions) FileExistsf(path string, msg string, args ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + FileExistsf(a.t, path, msg, args...) +} + +// Greater asserts that the first element is greater than the second +// +// a.Greater(2, 1) +// a.Greater(float64(2), float64(1)) +// a.Greater("b", "a") +func (a *Assertions) Greater(e1 interface{}, e2 interface{}, msgAndArgs ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + Greater(a.t, e1, e2, msgAndArgs...) +} + +// GreaterOrEqual asserts that the first element is greater than or equal to the second +// +// a.GreaterOrEqual(2, 1) +// a.GreaterOrEqual(2, 2) +// a.GreaterOrEqual("b", "a") +// a.GreaterOrEqual("b", "b") +func (a *Assertions) GreaterOrEqual(e1 interface{}, e2 interface{}, msgAndArgs ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + GreaterOrEqual(a.t, e1, e2, msgAndArgs...) +} + +// GreaterOrEqualf asserts that the first element is greater than or equal to the second +// +// a.GreaterOrEqualf(2, 1, "error message %s", "formatted") +// a.GreaterOrEqualf(2, 2, "error message %s", "formatted") +// a.GreaterOrEqualf("b", "a", "error message %s", "formatted") +// a.GreaterOrEqualf("b", "b", "error message %s", "formatted") +func (a *Assertions) GreaterOrEqualf(e1 interface{}, e2 interface{}, msg string, args ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + GreaterOrEqualf(a.t, e1, e2, msg, args...) +} + +// Greaterf asserts that the first element is greater than the second +// +// a.Greaterf(2, 1, "error message %s", "formatted") +// a.Greaterf(float64(2), float64(1), "error message %s", "formatted") +// a.Greaterf("b", "a", "error message %s", "formatted") +func (a *Assertions) Greaterf(e1 interface{}, e2 interface{}, msg string, args ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + Greaterf(a.t, e1, e2, msg, args...) +} + +// HTTPBodyContains asserts that a specified handler returns a +// body that contains a string. +// +// a.HTTPBodyContains(myHandler, "GET", "www.google.com", nil, "I'm Feeling Lucky") +// +// Returns whether the assertion was successful (true) or not (false). +func (a *Assertions) HTTPBodyContains(handler http.HandlerFunc, method string, url string, values url.Values, str interface{}, msgAndArgs ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + HTTPBodyContains(a.t, handler, method, url, values, str, msgAndArgs...) +} + +// HTTPBodyContainsf asserts that a specified handler returns a +// body that contains a string. +// +// a.HTTPBodyContainsf(myHandler, "GET", "www.google.com", nil, "I'm Feeling Lucky", "error message %s", "formatted") +// +// Returns whether the assertion was successful (true) or not (false). +func (a *Assertions) HTTPBodyContainsf(handler http.HandlerFunc, method string, url string, values url.Values, str interface{}, msg string, args ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + HTTPBodyContainsf(a.t, handler, method, url, values, str, msg, args...) +} + +// HTTPBodyNotContains asserts that a specified handler returns a +// body that does not contain a string. +// +// a.HTTPBodyNotContains(myHandler, "GET", "www.google.com", nil, "I'm Feeling Lucky") +// +// Returns whether the assertion was successful (true) or not (false). +func (a *Assertions) HTTPBodyNotContains(handler http.HandlerFunc, method string, url string, values url.Values, str interface{}, msgAndArgs ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + HTTPBodyNotContains(a.t, handler, method, url, values, str, msgAndArgs...) +} + +// HTTPBodyNotContainsf asserts that a specified handler returns a +// body that does not contain a string. +// +// a.HTTPBodyNotContainsf(myHandler, "GET", "www.google.com", nil, "I'm Feeling Lucky", "error message %s", "formatted") +// +// Returns whether the assertion was successful (true) or not (false). +func (a *Assertions) HTTPBodyNotContainsf(handler http.HandlerFunc, method string, url string, values url.Values, str interface{}, msg string, args ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + HTTPBodyNotContainsf(a.t, handler, method, url, values, str, msg, args...) +} + +// HTTPError asserts that a specified handler returns an error status code. +// +// a.HTTPError(myHandler, "POST", "/a/b/c", url.Values{"a": []string{"b", "c"}} +// +// Returns whether the assertion was successful (true) or not (false). +func (a *Assertions) HTTPError(handler http.HandlerFunc, method string, url string, values url.Values, msgAndArgs ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + HTTPError(a.t, handler, method, url, values, msgAndArgs...) +} + +// HTTPErrorf asserts that a specified handler returns an error status code. +// +// a.HTTPErrorf(myHandler, "POST", "/a/b/c", url.Values{"a": []string{"b", "c"}} +// +// Returns whether the assertion was successful (true) or not (false). +func (a *Assertions) HTTPErrorf(handler http.HandlerFunc, method string, url string, values url.Values, msg string, args ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + HTTPErrorf(a.t, handler, method, url, values, msg, args...) +} + +// HTTPRedirect asserts that a specified handler returns a redirect status code. +// +// a.HTTPRedirect(myHandler, "GET", "/a/b/c", url.Values{"a": []string{"b", "c"}} +// +// Returns whether the assertion was successful (true) or not (false). +func (a *Assertions) HTTPRedirect(handler http.HandlerFunc, method string, url string, values url.Values, msgAndArgs ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + HTTPRedirect(a.t, handler, method, url, values, msgAndArgs...) +} + +// HTTPRedirectf asserts that a specified handler returns a redirect status code. +// +// a.HTTPRedirectf(myHandler, "GET", "/a/b/c", url.Values{"a": []string{"b", "c"}} +// +// Returns whether the assertion was successful (true) or not (false). +func (a *Assertions) HTTPRedirectf(handler http.HandlerFunc, method string, url string, values url.Values, msg string, args ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + HTTPRedirectf(a.t, handler, method, url, values, msg, args...) +} + +// HTTPStatusCode asserts that a specified handler returns a specified status code. +// +// a.HTTPStatusCode(myHandler, "GET", "/notImplemented", nil, 501) +// +// Returns whether the assertion was successful (true) or not (false). +func (a *Assertions) HTTPStatusCode(handler http.HandlerFunc, method string, url string, values url.Values, statuscode int, msgAndArgs ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + HTTPStatusCode(a.t, handler, method, url, values, statuscode, msgAndArgs...) +} + +// HTTPStatusCodef asserts that a specified handler returns a specified status code. +// +// a.HTTPStatusCodef(myHandler, "GET", "/notImplemented", nil, 501, "error message %s", "formatted") +// +// Returns whether the assertion was successful (true) or not (false). +func (a *Assertions) HTTPStatusCodef(handler http.HandlerFunc, method string, url string, values url.Values, statuscode int, msg string, args ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + HTTPStatusCodef(a.t, handler, method, url, values, statuscode, msg, args...) +} + +// HTTPSuccess asserts that a specified handler returns a success status code. +// +// a.HTTPSuccess(myHandler, "POST", "http://www.google.com", nil) +// +// Returns whether the assertion was successful (true) or not (false). +func (a *Assertions) HTTPSuccess(handler http.HandlerFunc, method string, url string, values url.Values, msgAndArgs ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + HTTPSuccess(a.t, handler, method, url, values, msgAndArgs...) +} + +// HTTPSuccessf asserts that a specified handler returns a success status code. +// +// a.HTTPSuccessf(myHandler, "POST", "http://www.google.com", nil, "error message %s", "formatted") +// +// Returns whether the assertion was successful (true) or not (false). +func (a *Assertions) HTTPSuccessf(handler http.HandlerFunc, method string, url string, values url.Values, msg string, args ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + HTTPSuccessf(a.t, handler, method, url, values, msg, args...) +} + +// Implements asserts that an object is implemented by the specified interface. +// +// a.Implements((*MyInterface)(nil), new(MyObject)) +func (a *Assertions) Implements(interfaceObject interface{}, object interface{}, msgAndArgs ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + Implements(a.t, interfaceObject, object, msgAndArgs...) +} + +// Implementsf asserts that an object is implemented by the specified interface. +// +// a.Implementsf((*MyInterface)(nil), new(MyObject), "error message %s", "formatted") +func (a *Assertions) Implementsf(interfaceObject interface{}, object interface{}, msg string, args ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + Implementsf(a.t, interfaceObject, object, msg, args...) +} + +// InDelta asserts that the two numerals are within delta of each other. +// +// a.InDelta(math.Pi, 22/7.0, 0.01) +func (a *Assertions) InDelta(expected interface{}, actual interface{}, delta float64, msgAndArgs ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + InDelta(a.t, expected, actual, delta, msgAndArgs...) +} + +// InDeltaMapValues is the same as InDelta, but it compares all values between two maps. Both maps must have exactly the same keys. +func (a *Assertions) InDeltaMapValues(expected interface{}, actual interface{}, delta float64, msgAndArgs ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + InDeltaMapValues(a.t, expected, actual, delta, msgAndArgs...) +} + +// InDeltaMapValuesf is the same as InDelta, but it compares all values between two maps. Both maps must have exactly the same keys. +func (a *Assertions) InDeltaMapValuesf(expected interface{}, actual interface{}, delta float64, msg string, args ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + InDeltaMapValuesf(a.t, expected, actual, delta, msg, args...) +} + +// InDeltaSlice is the same as InDelta, except it compares two slices. +func (a *Assertions) InDeltaSlice(expected interface{}, actual interface{}, delta float64, msgAndArgs ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + InDeltaSlice(a.t, expected, actual, delta, msgAndArgs...) +} + +// InDeltaSlicef is the same as InDelta, except it compares two slices. +func (a *Assertions) InDeltaSlicef(expected interface{}, actual interface{}, delta float64, msg string, args ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + InDeltaSlicef(a.t, expected, actual, delta, msg, args...) +} + +// InDeltaf asserts that the two numerals are within delta of each other. +// +// a.InDeltaf(math.Pi, 22/7.0, 0.01, "error message %s", "formatted") +func (a *Assertions) InDeltaf(expected interface{}, actual interface{}, delta float64, msg string, args ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + InDeltaf(a.t, expected, actual, delta, msg, args...) +} + +// InEpsilon asserts that expected and actual have a relative error less than epsilon +func (a *Assertions) InEpsilon(expected interface{}, actual interface{}, epsilon float64, msgAndArgs ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + InEpsilon(a.t, expected, actual, epsilon, msgAndArgs...) +} + +// InEpsilonSlice is the same as InEpsilon, except it compares each value from two slices. +func (a *Assertions) InEpsilonSlice(expected interface{}, actual interface{}, epsilon float64, msgAndArgs ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + InEpsilonSlice(a.t, expected, actual, epsilon, msgAndArgs...) +} + +// InEpsilonSlicef is the same as InEpsilon, except it compares each value from two slices. +func (a *Assertions) InEpsilonSlicef(expected interface{}, actual interface{}, epsilon float64, msg string, args ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + InEpsilonSlicef(a.t, expected, actual, epsilon, msg, args...) +} + +// InEpsilonf asserts that expected and actual have a relative error less than epsilon +func (a *Assertions) InEpsilonf(expected interface{}, actual interface{}, epsilon float64, msg string, args ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + InEpsilonf(a.t, expected, actual, epsilon, msg, args...) +} + +// IsDecreasing asserts that the collection is decreasing +// +// a.IsDecreasing([]int{2, 1, 0}) +// a.IsDecreasing([]float{2, 1}) +// a.IsDecreasing([]string{"b", "a"}) +func (a *Assertions) IsDecreasing(object interface{}, msgAndArgs ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + IsDecreasing(a.t, object, msgAndArgs...) +} + +// IsDecreasingf asserts that the collection is decreasing +// +// a.IsDecreasingf([]int{2, 1, 0}, "error message %s", "formatted") +// a.IsDecreasingf([]float{2, 1}, "error message %s", "formatted") +// a.IsDecreasingf([]string{"b", "a"}, "error message %s", "formatted") +func (a *Assertions) IsDecreasingf(object interface{}, msg string, args ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + IsDecreasingf(a.t, object, msg, args...) +} + +// IsIncreasing asserts that the collection is increasing +// +// a.IsIncreasing([]int{1, 2, 3}) +// a.IsIncreasing([]float{1, 2}) +// a.IsIncreasing([]string{"a", "b"}) +func (a *Assertions) IsIncreasing(object interface{}, msgAndArgs ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + IsIncreasing(a.t, object, msgAndArgs...) +} + +// IsIncreasingf asserts that the collection is increasing +// +// a.IsIncreasingf([]int{1, 2, 3}, "error message %s", "formatted") +// a.IsIncreasingf([]float{1, 2}, "error message %s", "formatted") +// a.IsIncreasingf([]string{"a", "b"}, "error message %s", "formatted") +func (a *Assertions) IsIncreasingf(object interface{}, msg string, args ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + IsIncreasingf(a.t, object, msg, args...) +} + +// IsNonDecreasing asserts that the collection is not decreasing +// +// a.IsNonDecreasing([]int{1, 1, 2}) +// a.IsNonDecreasing([]float{1, 2}) +// a.IsNonDecreasing([]string{"a", "b"}) +func (a *Assertions) IsNonDecreasing(object interface{}, msgAndArgs ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + IsNonDecreasing(a.t, object, msgAndArgs...) +} + +// IsNonDecreasingf asserts that the collection is not decreasing +// +// a.IsNonDecreasingf([]int{1, 1, 2}, "error message %s", "formatted") +// a.IsNonDecreasingf([]float{1, 2}, "error message %s", "formatted") +// a.IsNonDecreasingf([]string{"a", "b"}, "error message %s", "formatted") +func (a *Assertions) IsNonDecreasingf(object interface{}, msg string, args ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + IsNonDecreasingf(a.t, object, msg, args...) +} + +// IsNonIncreasing asserts that the collection is not increasing +// +// a.IsNonIncreasing([]int{2, 1, 1}) +// a.IsNonIncreasing([]float{2, 1}) +// a.IsNonIncreasing([]string{"b", "a"}) +func (a *Assertions) IsNonIncreasing(object interface{}, msgAndArgs ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + IsNonIncreasing(a.t, object, msgAndArgs...) +} + +// IsNonIncreasingf asserts that the collection is not increasing +// +// a.IsNonIncreasingf([]int{2, 1, 1}, "error message %s", "formatted") +// a.IsNonIncreasingf([]float{2, 1}, "error message %s", "formatted") +// a.IsNonIncreasingf([]string{"b", "a"}, "error message %s", "formatted") +func (a *Assertions) IsNonIncreasingf(object interface{}, msg string, args ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + IsNonIncreasingf(a.t, object, msg, args...) +} + +// IsType asserts that the specified objects are of the same type. +func (a *Assertions) IsType(expectedType interface{}, object interface{}, msgAndArgs ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + IsType(a.t, expectedType, object, msgAndArgs...) +} + +// IsTypef asserts that the specified objects are of the same type. +func (a *Assertions) IsTypef(expectedType interface{}, object interface{}, msg string, args ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + IsTypef(a.t, expectedType, object, msg, args...) +} + +// JSONEq asserts that two JSON strings are equivalent. +// +// a.JSONEq(`{"hello": "world", "foo": "bar"}`, `{"foo": "bar", "hello": "world"}`) +func (a *Assertions) JSONEq(expected string, actual string, msgAndArgs ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + JSONEq(a.t, expected, actual, msgAndArgs...) +} + +// JSONEqf asserts that two JSON strings are equivalent. +// +// a.JSONEqf(`{"hello": "world", "foo": "bar"}`, `{"foo": "bar", "hello": "world"}`, "error message %s", "formatted") +func (a *Assertions) JSONEqf(expected string, actual string, msg string, args ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + JSONEqf(a.t, expected, actual, msg, args...) +} + +// Len asserts that the specified object has specific length. +// Len also fails if the object has a type that len() not accept. +// +// a.Len(mySlice, 3) +func (a *Assertions) Len(object interface{}, length int, msgAndArgs ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + Len(a.t, object, length, msgAndArgs...) +} + +// Lenf asserts that the specified object has specific length. +// Lenf also fails if the object has a type that len() not accept. +// +// a.Lenf(mySlice, 3, "error message %s", "formatted") +func (a *Assertions) Lenf(object interface{}, length int, msg string, args ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + Lenf(a.t, object, length, msg, args...) +} + +// Less asserts that the first element is less than the second +// +// a.Less(1, 2) +// a.Less(float64(1), float64(2)) +// a.Less("a", "b") +func (a *Assertions) Less(e1 interface{}, e2 interface{}, msgAndArgs ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + Less(a.t, e1, e2, msgAndArgs...) +} + +// LessOrEqual asserts that the first element is less than or equal to the second +// +// a.LessOrEqual(1, 2) +// a.LessOrEqual(2, 2) +// a.LessOrEqual("a", "b") +// a.LessOrEqual("b", "b") +func (a *Assertions) LessOrEqual(e1 interface{}, e2 interface{}, msgAndArgs ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + LessOrEqual(a.t, e1, e2, msgAndArgs...) +} + +// LessOrEqualf asserts that the first element is less than or equal to the second +// +// a.LessOrEqualf(1, 2, "error message %s", "formatted") +// a.LessOrEqualf(2, 2, "error message %s", "formatted") +// a.LessOrEqualf("a", "b", "error message %s", "formatted") +// a.LessOrEqualf("b", "b", "error message %s", "formatted") +func (a *Assertions) LessOrEqualf(e1 interface{}, e2 interface{}, msg string, args ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + LessOrEqualf(a.t, e1, e2, msg, args...) +} + +// Lessf asserts that the first element is less than the second +// +// a.Lessf(1, 2, "error message %s", "formatted") +// a.Lessf(float64(1), float64(2), "error message %s", "formatted") +// a.Lessf("a", "b", "error message %s", "formatted") +func (a *Assertions) Lessf(e1 interface{}, e2 interface{}, msg string, args ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + Lessf(a.t, e1, e2, msg, args...) +} + +// Negative asserts that the specified element is negative +// +// a.Negative(-1) +// a.Negative(-1.23) +func (a *Assertions) Negative(e interface{}, msgAndArgs ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + Negative(a.t, e, msgAndArgs...) +} + +// Negativef asserts that the specified element is negative +// +// a.Negativef(-1, "error message %s", "formatted") +// a.Negativef(-1.23, "error message %s", "formatted") +func (a *Assertions) Negativef(e interface{}, msg string, args ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + Negativef(a.t, e, msg, args...) +} + +// Never asserts that the given condition doesn't satisfy in waitFor time, +// periodically checking the target function each tick. +// +// a.Never(func() bool { return false; }, time.Second, 10*time.Millisecond) +func (a *Assertions) Never(condition func() bool, waitFor time.Duration, tick time.Duration, msgAndArgs ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + Never(a.t, condition, waitFor, tick, msgAndArgs...) +} + +// Neverf asserts that the given condition doesn't satisfy in waitFor time, +// periodically checking the target function each tick. +// +// a.Neverf(func() bool { return false; }, time.Second, 10*time.Millisecond, "error message %s", "formatted") +func (a *Assertions) Neverf(condition func() bool, waitFor time.Duration, tick time.Duration, msg string, args ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + Neverf(a.t, condition, waitFor, tick, msg, args...) +} + +// Nil asserts that the specified object is nil. +// +// a.Nil(err) +func (a *Assertions) Nil(object interface{}, msgAndArgs ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + Nil(a.t, object, msgAndArgs...) +} + +// Nilf asserts that the specified object is nil. +// +// a.Nilf(err, "error message %s", "formatted") +func (a *Assertions) Nilf(object interface{}, msg string, args ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + Nilf(a.t, object, msg, args...) +} + +// NoDirExists checks whether a directory does not exist in the given path. +// It fails if the path points to an existing _directory_ only. +func (a *Assertions) NoDirExists(path string, msgAndArgs ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + NoDirExists(a.t, path, msgAndArgs...) +} + +// NoDirExistsf checks whether a directory does not exist in the given path. +// It fails if the path points to an existing _directory_ only. +func (a *Assertions) NoDirExistsf(path string, msg string, args ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + NoDirExistsf(a.t, path, msg, args...) +} + +// NoError asserts that a function returned no error (i.e. `nil`). +// +// actualObj, err := SomeFunction() +// if a.NoError(err) { +// assert.Equal(t, expectedObj, actualObj) +// } +func (a *Assertions) NoError(err error, msgAndArgs ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + NoError(a.t, err, msgAndArgs...) +} + +// NoErrorf asserts that a function returned no error (i.e. `nil`). +// +// actualObj, err := SomeFunction() +// if a.NoErrorf(err, "error message %s", "formatted") { +// assert.Equal(t, expectedObj, actualObj) +// } +func (a *Assertions) NoErrorf(err error, msg string, args ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + NoErrorf(a.t, err, msg, args...) +} + +// NoFileExists checks whether a file does not exist in a given path. It fails +// if the path points to an existing _file_ only. +func (a *Assertions) NoFileExists(path string, msgAndArgs ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + NoFileExists(a.t, path, msgAndArgs...) +} + +// NoFileExistsf checks whether a file does not exist in a given path. It fails +// if the path points to an existing _file_ only. +func (a *Assertions) NoFileExistsf(path string, msg string, args ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + NoFileExistsf(a.t, path, msg, args...) +} + +// NotContains asserts that the specified string, list(array, slice...) or map does NOT contain the +// specified substring or element. +// +// a.NotContains("Hello World", "Earth") +// a.NotContains(["Hello", "World"], "Earth") +// a.NotContains({"Hello": "World"}, "Earth") +func (a *Assertions) NotContains(s interface{}, contains interface{}, msgAndArgs ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + NotContains(a.t, s, contains, msgAndArgs...) +} + +// NotContainsf asserts that the specified string, list(array, slice...) or map does NOT contain the +// specified substring or element. +// +// a.NotContainsf("Hello World", "Earth", "error message %s", "formatted") +// a.NotContainsf(["Hello", "World"], "Earth", "error message %s", "formatted") +// a.NotContainsf({"Hello": "World"}, "Earth", "error message %s", "formatted") +func (a *Assertions) NotContainsf(s interface{}, contains interface{}, msg string, args ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + NotContainsf(a.t, s, contains, msg, args...) +} + +// NotEmpty asserts that the specified object is NOT empty. I.e. not nil, "", false, 0 or either +// a slice or a channel with len == 0. +// +// if a.NotEmpty(obj) { +// assert.Equal(t, "two", obj[1]) +// } +func (a *Assertions) NotEmpty(object interface{}, msgAndArgs ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + NotEmpty(a.t, object, msgAndArgs...) +} + +// NotEmptyf asserts that the specified object is NOT empty. I.e. not nil, "", false, 0 or either +// a slice or a channel with len == 0. +// +// if a.NotEmptyf(obj, "error message %s", "formatted") { +// assert.Equal(t, "two", obj[1]) +// } +func (a *Assertions) NotEmptyf(object interface{}, msg string, args ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + NotEmptyf(a.t, object, msg, args...) +} + +// NotEqual asserts that the specified values are NOT equal. +// +// a.NotEqual(obj1, obj2) +// +// Pointer variable equality is determined based on the equality of the +// referenced values (as opposed to the memory addresses). +func (a *Assertions) NotEqual(expected interface{}, actual interface{}, msgAndArgs ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + NotEqual(a.t, expected, actual, msgAndArgs...) +} + +// NotEqualValues asserts that two objects are not equal even when converted to the same type +// +// a.NotEqualValues(obj1, obj2) +func (a *Assertions) NotEqualValues(expected interface{}, actual interface{}, msgAndArgs ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + NotEqualValues(a.t, expected, actual, msgAndArgs...) +} + +// NotEqualValuesf asserts that two objects are not equal even when converted to the same type +// +// a.NotEqualValuesf(obj1, obj2, "error message %s", "formatted") +func (a *Assertions) NotEqualValuesf(expected interface{}, actual interface{}, msg string, args ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + NotEqualValuesf(a.t, expected, actual, msg, args...) +} + +// NotEqualf asserts that the specified values are NOT equal. +// +// a.NotEqualf(obj1, obj2, "error message %s", "formatted") +// +// Pointer variable equality is determined based on the equality of the +// referenced values (as opposed to the memory addresses). +func (a *Assertions) NotEqualf(expected interface{}, actual interface{}, msg string, args ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + NotEqualf(a.t, expected, actual, msg, args...) +} + +// NotErrorIs asserts that at none of the errors in err's chain matches target. +// This is a wrapper for errors.Is. +func (a *Assertions) NotErrorIs(err error, target error, msgAndArgs ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + NotErrorIs(a.t, err, target, msgAndArgs...) +} + +// NotErrorIsf asserts that at none of the errors in err's chain matches target. +// This is a wrapper for errors.Is. +func (a *Assertions) NotErrorIsf(err error, target error, msg string, args ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + NotErrorIsf(a.t, err, target, msg, args...) +} + +// NotNil asserts that the specified object is not nil. +// +// a.NotNil(err) +func (a *Assertions) NotNil(object interface{}, msgAndArgs ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + NotNil(a.t, object, msgAndArgs...) +} + +// NotNilf asserts that the specified object is not nil. +// +// a.NotNilf(err, "error message %s", "formatted") +func (a *Assertions) NotNilf(object interface{}, msg string, args ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + NotNilf(a.t, object, msg, args...) +} + +// NotPanics asserts that the code inside the specified PanicTestFunc does NOT panic. +// +// a.NotPanics(func(){ RemainCalm() }) +func (a *Assertions) NotPanics(f assert.PanicTestFunc, msgAndArgs ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + NotPanics(a.t, f, msgAndArgs...) +} + +// NotPanicsf asserts that the code inside the specified PanicTestFunc does NOT panic. +// +// a.NotPanicsf(func(){ RemainCalm() }, "error message %s", "formatted") +func (a *Assertions) NotPanicsf(f assert.PanicTestFunc, msg string, args ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + NotPanicsf(a.t, f, msg, args...) +} + +// NotRegexp asserts that a specified regexp does not match a string. +// +// a.NotRegexp(regexp.MustCompile("starts"), "it's starting") +// a.NotRegexp("^start", "it's not starting") +func (a *Assertions) NotRegexp(rx interface{}, str interface{}, msgAndArgs ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + NotRegexp(a.t, rx, str, msgAndArgs...) +} + +// NotRegexpf asserts that a specified regexp does not match a string. +// +// a.NotRegexpf(regexp.MustCompile("starts"), "it's starting", "error message %s", "formatted") +// a.NotRegexpf("^start", "it's not starting", "error message %s", "formatted") +func (a *Assertions) NotRegexpf(rx interface{}, str interface{}, msg string, args ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + NotRegexpf(a.t, rx, str, msg, args...) +} + +// NotSame asserts that two pointers do not reference the same object. +// +// a.NotSame(ptr1, ptr2) +// +// Both arguments must be pointer variables. Pointer variable sameness is +// determined based on the equality of both type and value. +func (a *Assertions) NotSame(expected interface{}, actual interface{}, msgAndArgs ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + NotSame(a.t, expected, actual, msgAndArgs...) +} + +// NotSamef asserts that two pointers do not reference the same object. +// +// a.NotSamef(ptr1, ptr2, "error message %s", "formatted") +// +// Both arguments must be pointer variables. Pointer variable sameness is +// determined based on the equality of both type and value. +func (a *Assertions) NotSamef(expected interface{}, actual interface{}, msg string, args ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + NotSamef(a.t, expected, actual, msg, args...) +} + +// NotSubset asserts that the specified list(array, slice...) contains not all +// elements given in the specified subset(array, slice...). +// +// a.NotSubset([1, 3, 4], [1, 2], "But [1, 3, 4] does not contain [1, 2]") +func (a *Assertions) NotSubset(list interface{}, subset interface{}, msgAndArgs ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + NotSubset(a.t, list, subset, msgAndArgs...) +} + +// NotSubsetf asserts that the specified list(array, slice...) contains not all +// elements given in the specified subset(array, slice...). +// +// a.NotSubsetf([1, 3, 4], [1, 2], "But [1, 3, 4] does not contain [1, 2]", "error message %s", "formatted") +func (a *Assertions) NotSubsetf(list interface{}, subset interface{}, msg string, args ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + NotSubsetf(a.t, list, subset, msg, args...) +} + +// NotZero asserts that i is not the zero value for its type. +func (a *Assertions) NotZero(i interface{}, msgAndArgs ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + NotZero(a.t, i, msgAndArgs...) +} + +// NotZerof asserts that i is not the zero value for its type. +func (a *Assertions) NotZerof(i interface{}, msg string, args ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + NotZerof(a.t, i, msg, args...) +} + +// Panics asserts that the code inside the specified PanicTestFunc panics. +// +// a.Panics(func(){ GoCrazy() }) +func (a *Assertions) Panics(f assert.PanicTestFunc, msgAndArgs ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + Panics(a.t, f, msgAndArgs...) +} + +// PanicsWithError asserts that the code inside the specified PanicTestFunc +// panics, and that the recovered panic value is an error that satisfies the +// EqualError comparison. +// +// a.PanicsWithError("crazy error", func(){ GoCrazy() }) +func (a *Assertions) PanicsWithError(errString string, f assert.PanicTestFunc, msgAndArgs ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + PanicsWithError(a.t, errString, f, msgAndArgs...) +} + +// PanicsWithErrorf asserts that the code inside the specified PanicTestFunc +// panics, and that the recovered panic value is an error that satisfies the +// EqualError comparison. +// +// a.PanicsWithErrorf("crazy error", func(){ GoCrazy() }, "error message %s", "formatted") +func (a *Assertions) PanicsWithErrorf(errString string, f assert.PanicTestFunc, msg string, args ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + PanicsWithErrorf(a.t, errString, f, msg, args...) +} + +// PanicsWithValue asserts that the code inside the specified PanicTestFunc panics, and that +// the recovered panic value equals the expected panic value. +// +// a.PanicsWithValue("crazy error", func(){ GoCrazy() }) +func (a *Assertions) PanicsWithValue(expected interface{}, f assert.PanicTestFunc, msgAndArgs ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + PanicsWithValue(a.t, expected, f, msgAndArgs...) +} + +// PanicsWithValuef asserts that the code inside the specified PanicTestFunc panics, and that +// the recovered panic value equals the expected panic value. +// +// a.PanicsWithValuef("crazy error", func(){ GoCrazy() }, "error message %s", "formatted") +func (a *Assertions) PanicsWithValuef(expected interface{}, f assert.PanicTestFunc, msg string, args ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + PanicsWithValuef(a.t, expected, f, msg, args...) +} + +// Panicsf asserts that the code inside the specified PanicTestFunc panics. +// +// a.Panicsf(func(){ GoCrazy() }, "error message %s", "formatted") +func (a *Assertions) Panicsf(f assert.PanicTestFunc, msg string, args ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + Panicsf(a.t, f, msg, args...) +} + +// Positive asserts that the specified element is positive +// +// a.Positive(1) +// a.Positive(1.23) +func (a *Assertions) Positive(e interface{}, msgAndArgs ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + Positive(a.t, e, msgAndArgs...) +} + +// Positivef asserts that the specified element is positive +// +// a.Positivef(1, "error message %s", "formatted") +// a.Positivef(1.23, "error message %s", "formatted") +func (a *Assertions) Positivef(e interface{}, msg string, args ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + Positivef(a.t, e, msg, args...) +} + +// Regexp asserts that a specified regexp matches a string. +// +// a.Regexp(regexp.MustCompile("start"), "it's starting") +// a.Regexp("start...$", "it's not starting") +func (a *Assertions) Regexp(rx interface{}, str interface{}, msgAndArgs ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + Regexp(a.t, rx, str, msgAndArgs...) +} + +// Regexpf asserts that a specified regexp matches a string. +// +// a.Regexpf(regexp.MustCompile("start"), "it's starting", "error message %s", "formatted") +// a.Regexpf("start...$", "it's not starting", "error message %s", "formatted") +func (a *Assertions) Regexpf(rx interface{}, str interface{}, msg string, args ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + Regexpf(a.t, rx, str, msg, args...) +} + +// Same asserts that two pointers reference the same object. +// +// a.Same(ptr1, ptr2) +// +// Both arguments must be pointer variables. Pointer variable sameness is +// determined based on the equality of both type and value. +func (a *Assertions) Same(expected interface{}, actual interface{}, msgAndArgs ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + Same(a.t, expected, actual, msgAndArgs...) +} + +// Samef asserts that two pointers reference the same object. +// +// a.Samef(ptr1, ptr2, "error message %s", "formatted") +// +// Both arguments must be pointer variables. Pointer variable sameness is +// determined based on the equality of both type and value. +func (a *Assertions) Samef(expected interface{}, actual interface{}, msg string, args ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + Samef(a.t, expected, actual, msg, args...) +} + +// Subset asserts that the specified list(array, slice...) contains all +// elements given in the specified subset(array, slice...). +// +// a.Subset([1, 2, 3], [1, 2], "But [1, 2, 3] does contain [1, 2]") +func (a *Assertions) Subset(list interface{}, subset interface{}, msgAndArgs ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + Subset(a.t, list, subset, msgAndArgs...) +} + +// Subsetf asserts that the specified list(array, slice...) contains all +// elements given in the specified subset(array, slice...). +// +// a.Subsetf([1, 2, 3], [1, 2], "But [1, 2, 3] does contain [1, 2]", "error message %s", "formatted") +func (a *Assertions) Subsetf(list interface{}, subset interface{}, msg string, args ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + Subsetf(a.t, list, subset, msg, args...) +} + +// True asserts that the specified value is true. +// +// a.True(myBool) +func (a *Assertions) True(value bool, msgAndArgs ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + True(a.t, value, msgAndArgs...) +} + +// Truef asserts that the specified value is true. +// +// a.Truef(myBool, "error message %s", "formatted") +func (a *Assertions) Truef(value bool, msg string, args ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + Truef(a.t, value, msg, args...) +} + +// WithinDuration asserts that the two times are within duration delta of each other. +// +// a.WithinDuration(time.Now(), time.Now(), 10*time.Second) +func (a *Assertions) WithinDuration(expected time.Time, actual time.Time, delta time.Duration, msgAndArgs ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + WithinDuration(a.t, expected, actual, delta, msgAndArgs...) +} + +// WithinDurationf asserts that the two times are within duration delta of each other. +// +// a.WithinDurationf(time.Now(), time.Now(), 10*time.Second, "error message %s", "formatted") +func (a *Assertions) WithinDurationf(expected time.Time, actual time.Time, delta time.Duration, msg string, args ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + WithinDurationf(a.t, expected, actual, delta, msg, args...) +} + +// YAMLEq asserts that two YAML strings are equivalent. +func (a *Assertions) YAMLEq(expected string, actual string, msgAndArgs ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + YAMLEq(a.t, expected, actual, msgAndArgs...) +} + +// YAMLEqf asserts that two YAML strings are equivalent. +func (a *Assertions) YAMLEqf(expected string, actual string, msg string, args ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + YAMLEqf(a.t, expected, actual, msg, args...) +} + +// Zero asserts that i is the zero value for its type. +func (a *Assertions) Zero(i interface{}, msgAndArgs ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + Zero(a.t, i, msgAndArgs...) +} + +// Zerof asserts that i is the zero value for its type. +func (a *Assertions) Zerof(i interface{}, msg string, args ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + Zerof(a.t, i, msg, args...) +} diff --git a/vendor/github.com/stretchr/testify/require/require_forward.go.tmpl b/vendor/github.com/stretchr/testify/require/require_forward.go.tmpl new file mode 100644 index 00000000..54124df1 --- /dev/null +++ b/vendor/github.com/stretchr/testify/require/require_forward.go.tmpl @@ -0,0 +1,5 @@ +{{.CommentWithoutT "a"}} +func (a *Assertions) {{.DocInfo.Name}}({{.Params}}) { + if h, ok := a.t.(tHelper); ok { h.Helper() } + {{.DocInfo.Name}}(a.t, {{.ForwardedParams}}) +} diff --git a/vendor/github.com/stretchr/testify/require/requirements.go b/vendor/github.com/stretchr/testify/require/requirements.go new file mode 100644 index 00000000..91772dfe --- /dev/null +++ b/vendor/github.com/stretchr/testify/require/requirements.go @@ -0,0 +1,29 @@ +package require + +// TestingT is an interface wrapper around *testing.T +type TestingT interface { + Errorf(format string, args ...interface{}) + FailNow() +} + +type tHelper interface { + Helper() +} + +// ComparisonAssertionFunc is a common function prototype when comparing two values. Can be useful +// for table driven tests. +type ComparisonAssertionFunc func(TestingT, interface{}, interface{}, ...interface{}) + +// ValueAssertionFunc is a common function prototype when validating a single value. Can be useful +// for table driven tests. +type ValueAssertionFunc func(TestingT, interface{}, ...interface{}) + +// BoolAssertionFunc is a common function prototype when validating a bool value. Can be useful +// for table driven tests. +type BoolAssertionFunc func(TestingT, bool, ...interface{}) + +// ErrorAssertionFunc is a common function prototype when validating an error value. Can be useful +// for table driven tests. +type ErrorAssertionFunc func(TestingT, error, ...interface{}) + +//go:generate sh -c "cd ../_codegen && go build && cd - && ../_codegen/_codegen -output-package=require -template=require.go.tmpl -include-format-funcs" diff --git a/vendor/github.com/xeipuuv/gojsonpointer/LICENSE-APACHE-2.0.txt b/vendor/github.com/xeipuuv/gojsonpointer/LICENSE-APACHE-2.0.txt new file mode 100644 index 00000000..55ede8a4 --- /dev/null +++ b/vendor/github.com/xeipuuv/gojsonpointer/LICENSE-APACHE-2.0.txt @@ -0,0 +1,202 @@ + + Apache License + Version 2.0, January 2004 + http://www.apache.org/licenses/ + + TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION + + 1. Definitions. + + "License" shall mean the terms and conditions for use, reproduction, + and distribution as defined by Sections 1 through 9 of this document. + + "Licensor" shall mean the copyright owner or entity authorized by + the copyright owner that is granting the License. + + "Legal Entity" shall mean the union of the acting entity and all + other entities that control, are controlled by, or are under common + control with that entity. For the purposes of this definition, + "control" means (i) the power, direct or indirect, to cause the + direction or management of such entity, whether by contract or + otherwise, or (ii) ownership of fifty percent (50%) or more of the + outstanding shares, or (iii) beneficial ownership of such entity. + + "You" (or "Your") shall mean an individual or Legal Entity + exercising permissions granted by this License. + + "Source" form shall mean the preferred form for making modifications, + including but not limited to software source code, documentation + source, and configuration files. + + "Object" form shall mean any form resulting from mechanical + transformation or translation of a Source form, including but + not limited to compiled object code, generated documentation, + and conversions to other media types. + + "Work" shall mean the work of authorship, whether in Source or + Object form, made available under the License, as indicated by a + copyright notice that is included in or attached to the work + (an example is provided in the Appendix below). + + "Derivative Works" shall mean any work, whether in Source or Object + form, that is based on (or derived from) the Work and for which the + editorial revisions, annotations, elaborations, or other modifications + represent, as a whole, an original work of authorship. For the purposes + of this License, Derivative Works shall not include works that remain + separable from, or merely link (or bind by name) to the interfaces of, + the Work and Derivative Works thereof. + + "Contribution" shall mean any work of authorship, including + the original version of the Work and any modifications or additions + to that Work or Derivative Works thereof, that is intentionally + submitted to Licensor for inclusion in the Work by the copyright owner + or by an individual or Legal Entity authorized to submit on behalf of + the copyright owner. For the purposes of this definition, "submitted" + means any form of electronic, verbal, or written communication sent + to the Licensor or its representatives, including but not limited to + communication on electronic mailing lists, source code control systems, + and issue tracking systems that are managed by, or on behalf of, the + Licensor for the purpose of discussing and improving the Work, but + excluding communication that is conspicuously marked or otherwise + designated in writing by the copyright owner as "Not a Contribution." + + "Contributor" shall mean Licensor and any individual or Legal Entity + on behalf of whom a Contribution has been received by Licensor and + subsequently incorporated within the Work. + + 2. Grant of Copyright License. Subject to the terms and conditions of + this License, each Contributor hereby grants to You a perpetual, + worldwide, non-exclusive, no-charge, royalty-free, irrevocable + copyright license to reproduce, prepare Derivative Works of, + publicly display, publicly perform, sublicense, and distribute the + Work and such Derivative Works in Source or Object form. + + 3. Grant of Patent License. Subject to the terms and conditions of + this License, each Contributor hereby grants to You a perpetual, + worldwide, non-exclusive, no-charge, royalty-free, irrevocable + (except as stated in this section) patent license to make, have made, + use, offer to sell, sell, import, and otherwise transfer the Work, + where such license applies only to those patent claims licensable + by such Contributor that are necessarily infringed by their + Contribution(s) alone or by combination of their Contribution(s) + with the Work to which such Contribution(s) was submitted. If You + institute patent litigation against any entity (including a + cross-claim or counterclaim in a lawsuit) alleging that the Work + or a Contribution incorporated within the Work constitutes direct + or contributory patent infringement, then any patent licenses + granted to You under this License for that Work shall terminate + as of the date such litigation is filed. + + 4. Redistribution. You may reproduce and distribute copies of the + Work or Derivative Works thereof in any medium, with or without + modifications, and in Source or Object form, provided that You + meet the following conditions: + + (a) You must give any other recipients of the Work or + Derivative Works a copy of this License; and + + (b) You must cause any modified files to carry prominent notices + stating that You changed the files; and + + (c) You must retain, in the Source form of any Derivative Works + that You distribute, all copyright, patent, trademark, and + attribution notices from the Source form of the Work, + excluding those notices that do not pertain to any part of + the Derivative Works; and + + (d) If the Work includes a "NOTICE" text file as part of its + distribution, then any Derivative Works that You distribute must + include a readable copy of the attribution notices contained + within such NOTICE file, excluding those notices that do not + pertain to any part of the Derivative Works, in at least one + of the following places: within a NOTICE text file distributed + as part of the Derivative Works; within the Source form or + documentation, if provided along with the Derivative Works; or, + within a display generated by the Derivative Works, if and + wherever such third-party notices normally appear. The contents + of the NOTICE file are for informational purposes only and + do not modify the License. You may add Your own attribution + notices within Derivative Works that You distribute, alongside + or as an addendum to the NOTICE text from the Work, provided + that such additional attribution notices cannot be construed + as modifying the License. + + You may add Your own copyright statement to Your modifications and + may provide additional or different license terms and conditions + for use, reproduction, or distribution of Your modifications, or + for any such Derivative Works as a whole, provided Your use, + reproduction, and distribution of the Work otherwise complies with + the conditions stated in this License. + + 5. Submission of Contributions. Unless You explicitly state otherwise, + any Contribution intentionally submitted for inclusion in the Work + by You to the Licensor shall be under the terms and conditions of + this License, without any additional terms or conditions. + Notwithstanding the above, nothing herein shall supersede or modify + the terms of any separate license agreement you may have executed + with Licensor regarding such Contributions. + + 6. Trademarks. This License does not grant permission to use the trade + names, trademarks, service marks, or product names of the Licensor, + except as required for reasonable and customary use in describing the + origin of the Work and reproducing the content of the NOTICE file. + + 7. Disclaimer of Warranty. Unless required by applicable law or + agreed to in writing, Licensor provides the Work (and each + Contributor provides its Contributions) on an "AS IS" BASIS, + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or + implied, including, without limitation, any warranties or conditions + of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A + PARTICULAR PURPOSE. You are solely responsible for determining the + appropriateness of using or redistributing the Work and assume any + risks associated with Your exercise of permissions under this License. + + 8. Limitation of Liability. In no event and under no legal theory, + whether in tort (including negligence), contract, or otherwise, + unless required by applicable law (such as deliberate and grossly + negligent acts) or agreed to in writing, shall any Contributor be + liable to You for damages, including any direct, indirect, special, + incidental, or consequential damages of any character arising as a + result of this License or out of the use or inability to use the + Work (including but not limited to damages for loss of goodwill, + work stoppage, computer failure or malfunction, or any and all + other commercial damages or losses), even if such Contributor + has been advised of the possibility of such damages. + + 9. Accepting Warranty or Additional Liability. While redistributing + the Work or Derivative Works thereof, You may choose to offer, + and charge a fee for, acceptance of support, warranty, indemnity, + or other liability obligations and/or rights consistent with this + License. However, in accepting such obligations, You may act only + on Your own behalf and on Your sole responsibility, not on behalf + of any other Contributor, and only if You agree to indemnify, + defend, and hold each Contributor harmless for any liability + incurred by, or claims asserted against, such Contributor by reason + of your accepting any such warranty or additional liability. + + END OF TERMS AND CONDITIONS + + APPENDIX: How to apply the Apache License to your work. + + To apply the Apache License to your work, attach the following + boilerplate notice, with the fields enclosed by brackets "[]" + replaced with your own identifying information. (Don't include + the brackets!) The text should be enclosed in the appropriate + comment syntax for the file format. We also recommend that a + file or class name and description of purpose be included on the + same "printed page" as the copyright notice for easier + identification within third-party archives. + + Copyright 2015 xeipuuv + + Licensed under the Apache License, Version 2.0 (the "License"); + you may not use this file except in compliance with the License. + You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + + Unless required by applicable law or agreed to in writing, software + distributed under the License is distributed on an "AS IS" BASIS, + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + See the License for the specific language governing permissions and + limitations under the License. diff --git a/vendor/github.com/xeipuuv/gojsonpointer/README.md b/vendor/github.com/xeipuuv/gojsonpointer/README.md new file mode 100644 index 00000000..a4f5f145 --- /dev/null +++ b/vendor/github.com/xeipuuv/gojsonpointer/README.md @@ -0,0 +1,41 @@ +# gojsonpointer +An implementation of JSON Pointer - Go language + +## Usage + jsonText := `{ + "name": "Bobby B", + "occupation": { + "title" : "King", + "years" : 15, + "heir" : "Joffrey B" + } + }` + + var jsonDocument map[string]interface{} + json.Unmarshal([]byte(jsonText), &jsonDocument) + + //create a JSON pointer + pointerString := "/occupation/title" + pointer, _ := NewJsonPointer(pointerString) + + //SET a new value for the "title" in the document + pointer.Set(jsonDocument, "Supreme Leader of Westeros") + + //GET the new "title" from the document + title, _, _ := pointer.Get(jsonDocument) + fmt.Println(title) //outputs "Supreme Leader of Westeros" + + //DELETE the "heir" from the document + deletePointer := NewJsonPointer("/occupation/heir") + deletePointer.Delete(jsonDocument) + + b, _ := json.Marshal(jsonDocument) + fmt.Println(string(b)) + //outputs `{"name":"Bobby B","occupation":{"title":"Supreme Leader of Westeros","years":15}}` + + +## References +https://tools.ietf.org/html/rfc6901 + +### Note +The 4.Evaluation part of the previous reference, starting with 'If the currently referenced value is a JSON array, the reference token MUST contain either...' is not implemented. diff --git a/vendor/github.com/xeipuuv/gojsonpointer/pointer.go b/vendor/github.com/xeipuuv/gojsonpointer/pointer.go new file mode 100644 index 00000000..798c1f1c --- /dev/null +++ b/vendor/github.com/xeipuuv/gojsonpointer/pointer.go @@ -0,0 +1,211 @@ +// Copyright 2015 xeipuuv ( https://github.com/xeipuuv ) +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. + +// author xeipuuv +// author-github https://github.com/xeipuuv +// author-mail xeipuuv@gmail.com +// +// repository-name gojsonpointer +// repository-desc An implementation of JSON Pointer - Go language +// +// description Main and unique file. +// +// created 25-02-2013 + +package gojsonpointer + +import ( + "errors" + "fmt" + "reflect" + "strconv" + "strings" +) + +const ( + const_empty_pointer = `` + const_pointer_separator = `/` + + const_invalid_start = `JSON pointer must be empty or start with a "` + const_pointer_separator + `"` +) + +type implStruct struct { + mode string // "SET" or "GET" + + inDocument interface{} + + setInValue interface{} + + getOutNode interface{} + getOutKind reflect.Kind + outError error +} + +type JsonPointer struct { + referenceTokens []string +} + +// NewJsonPointer parses the given string JSON pointer and returns an object +func NewJsonPointer(jsonPointerString string) (p JsonPointer, err error) { + + // Pointer to the root of the document + if len(jsonPointerString) == 0 { + // Keep referenceTokens nil + return + } + if jsonPointerString[0] != '/' { + return p, errors.New(const_invalid_start) + } + + p.referenceTokens = strings.Split(jsonPointerString[1:], const_pointer_separator) + return +} + +// Uses the pointer to retrieve a value from a JSON document +func (p *JsonPointer) Get(document interface{}) (interface{}, reflect.Kind, error) { + + is := &implStruct{mode: "GET", inDocument: document} + p.implementation(is) + return is.getOutNode, is.getOutKind, is.outError + +} + +// Uses the pointer to update a value from a JSON document +func (p *JsonPointer) Set(document interface{}, value interface{}) (interface{}, error) { + + is := &implStruct{mode: "SET", inDocument: document, setInValue: value} + p.implementation(is) + return document, is.outError + +} + +// Uses the pointer to delete a value from a JSON document +func (p *JsonPointer) Delete(document interface{}) (interface{}, error) { + is := &implStruct{mode: "DEL", inDocument: document} + p.implementation(is) + return document, is.outError +} + +// Both Get and Set functions use the same implementation to avoid code duplication +func (p *JsonPointer) implementation(i *implStruct) { + + kind := reflect.Invalid + + // Full document when empty + if len(p.referenceTokens) == 0 { + i.getOutNode = i.inDocument + i.outError = nil + i.getOutKind = kind + i.outError = nil + return + } + + node := i.inDocument + + previousNodes := make([]interface{}, len(p.referenceTokens)) + previousTokens := make([]string, len(p.referenceTokens)) + + for ti, token := range p.referenceTokens { + + isLastToken := ti == len(p.referenceTokens)-1 + previousNodes[ti] = node + previousTokens[ti] = token + + switch v := node.(type) { + + case map[string]interface{}: + decodedToken := decodeReferenceToken(token) + if _, ok := v[decodedToken]; ok { + node = v[decodedToken] + if isLastToken && i.mode == "SET" { + v[decodedToken] = i.setInValue + } else if isLastToken && i.mode == "DEL" { + delete(v, decodedToken) + } + } else if isLastToken && i.mode == "SET" { + v[decodedToken] = i.setInValue + } else { + i.outError = fmt.Errorf("Object has no key '%s'", decodedToken) + i.getOutKind = reflect.Map + i.getOutNode = nil + return + } + + case []interface{}: + tokenIndex, err := strconv.Atoi(token) + if err != nil { + i.outError = fmt.Errorf("Invalid array index '%s'", token) + i.getOutKind = reflect.Slice + i.getOutNode = nil + return + } + if tokenIndex < 0 || tokenIndex >= len(v) { + i.outError = fmt.Errorf("Out of bound array[0,%d] index '%d'", len(v), tokenIndex) + i.getOutKind = reflect.Slice + i.getOutNode = nil + return + } + + node = v[tokenIndex] + if isLastToken && i.mode == "SET" { + v[tokenIndex] = i.setInValue + } else if isLastToken && i.mode == "DEL" { + v[tokenIndex] = v[len(v)-1] + v[len(v)-1] = nil + v = v[:len(v)-1] + previousNodes[ti-1].(map[string]interface{})[previousTokens[ti-1]] = v + } + + default: + i.outError = fmt.Errorf("Invalid token reference '%s'", token) + i.getOutKind = reflect.ValueOf(node).Kind() + i.getOutNode = nil + return + } + + } + + i.getOutNode = node + i.getOutKind = reflect.ValueOf(node).Kind() + i.outError = nil +} + +// Pointer to string representation function +func (p *JsonPointer) String() string { + + if len(p.referenceTokens) == 0 { + return const_empty_pointer + } + + pointerString := const_pointer_separator + strings.Join(p.referenceTokens, const_pointer_separator) + + return pointerString +} + +// Specific JSON pointer encoding here +// ~0 => ~ +// ~1 => / +// ... and vice versa + +func decodeReferenceToken(token string) string { + step1 := strings.Replace(token, `~1`, `/`, -1) + step2 := strings.Replace(step1, `~0`, `~`, -1) + return step2 +} + +func encodeReferenceToken(token string) string { + step1 := strings.Replace(token, `~`, `~0`, -1) + step2 := strings.Replace(step1, `/`, `~1`, -1) + return step2 +} diff --git a/vendor/github.com/xeipuuv/gojsonreference/LICENSE-APACHE-2.0.txt b/vendor/github.com/xeipuuv/gojsonreference/LICENSE-APACHE-2.0.txt new file mode 100644 index 00000000..55ede8a4 --- /dev/null +++ b/vendor/github.com/xeipuuv/gojsonreference/LICENSE-APACHE-2.0.txt @@ -0,0 +1,202 @@ + + Apache License + Version 2.0, January 2004 + http://www.apache.org/licenses/ + + TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION + + 1. Definitions. + + "License" shall mean the terms and conditions for use, reproduction, + and distribution as defined by Sections 1 through 9 of this document. + + "Licensor" shall mean the copyright owner or entity authorized by + the copyright owner that is granting the License. + + "Legal Entity" shall mean the union of the acting entity and all + other entities that control, are controlled by, or are under common + control with that entity. For the purposes of this definition, + "control" means (i) the power, direct or indirect, to cause the + direction or management of such entity, whether by contract or + otherwise, or (ii) ownership of fifty percent (50%) or more of the + outstanding shares, or (iii) beneficial ownership of such entity. + + "You" (or "Your") shall mean an individual or Legal Entity + exercising permissions granted by this License. + + "Source" form shall mean the preferred form for making modifications, + including but not limited to software source code, documentation + source, and configuration files. + + "Object" form shall mean any form resulting from mechanical + transformation or translation of a Source form, including but + not limited to compiled object code, generated documentation, + and conversions to other media types. + + "Work" shall mean the work of authorship, whether in Source or + Object form, made available under the License, as indicated by a + copyright notice that is included in or attached to the work + (an example is provided in the Appendix below). + + "Derivative Works" shall mean any work, whether in Source or Object + form, that is based on (or derived from) the Work and for which the + editorial revisions, annotations, elaborations, or other modifications + represent, as a whole, an original work of authorship. For the purposes + of this License, Derivative Works shall not include works that remain + separable from, or merely link (or bind by name) to the interfaces of, + the Work and Derivative Works thereof. + + "Contribution" shall mean any work of authorship, including + the original version of the Work and any modifications or additions + to that Work or Derivative Works thereof, that is intentionally + submitted to Licensor for inclusion in the Work by the copyright owner + or by an individual or Legal Entity authorized to submit on behalf of + the copyright owner. For the purposes of this definition, "submitted" + means any form of electronic, verbal, or written communication sent + to the Licensor or its representatives, including but not limited to + communication on electronic mailing lists, source code control systems, + and issue tracking systems that are managed by, or on behalf of, the + Licensor for the purpose of discussing and improving the Work, but + excluding communication that is conspicuously marked or otherwise + designated in writing by the copyright owner as "Not a Contribution." + + "Contributor" shall mean Licensor and any individual or Legal Entity + on behalf of whom a Contribution has been received by Licensor and + subsequently incorporated within the Work. + + 2. Grant of Copyright License. Subject to the terms and conditions of + this License, each Contributor hereby grants to You a perpetual, + worldwide, non-exclusive, no-charge, royalty-free, irrevocable + copyright license to reproduce, prepare Derivative Works of, + publicly display, publicly perform, sublicense, and distribute the + Work and such Derivative Works in Source or Object form. + + 3. Grant of Patent License. Subject to the terms and conditions of + this License, each Contributor hereby grants to You a perpetual, + worldwide, non-exclusive, no-charge, royalty-free, irrevocable + (except as stated in this section) patent license to make, have made, + use, offer to sell, sell, import, and otherwise transfer the Work, + where such license applies only to those patent claims licensable + by such Contributor that are necessarily infringed by their + Contribution(s) alone or by combination of their Contribution(s) + with the Work to which such Contribution(s) was submitted. If You + institute patent litigation against any entity (including a + cross-claim or counterclaim in a lawsuit) alleging that the Work + or a Contribution incorporated within the Work constitutes direct + or contributory patent infringement, then any patent licenses + granted to You under this License for that Work shall terminate + as of the date such litigation is filed. + + 4. Redistribution. You may reproduce and distribute copies of the + Work or Derivative Works thereof in any medium, with or without + modifications, and in Source or Object form, provided that You + meet the following conditions: + + (a) You must give any other recipients of the Work or + Derivative Works a copy of this License; and + + (b) You must cause any modified files to carry prominent notices + stating that You changed the files; and + + (c) You must retain, in the Source form of any Derivative Works + that You distribute, all copyright, patent, trademark, and + attribution notices from the Source form of the Work, + excluding those notices that do not pertain to any part of + the Derivative Works; and + + (d) If the Work includes a "NOTICE" text file as part of its + distribution, then any Derivative Works that You distribute must + include a readable copy of the attribution notices contained + within such NOTICE file, excluding those notices that do not + pertain to any part of the Derivative Works, in at least one + of the following places: within a NOTICE text file distributed + as part of the Derivative Works; within the Source form or + documentation, if provided along with the Derivative Works; or, + within a display generated by the Derivative Works, if and + wherever such third-party notices normally appear. The contents + of the NOTICE file are for informational purposes only and + do not modify the License. You may add Your own attribution + notices within Derivative Works that You distribute, alongside + or as an addendum to the NOTICE text from the Work, provided + that such additional attribution notices cannot be construed + as modifying the License. + + You may add Your own copyright statement to Your modifications and + may provide additional or different license terms and conditions + for use, reproduction, or distribution of Your modifications, or + for any such Derivative Works as a whole, provided Your use, + reproduction, and distribution of the Work otherwise complies with + the conditions stated in this License. + + 5. Submission of Contributions. Unless You explicitly state otherwise, + any Contribution intentionally submitted for inclusion in the Work + by You to the Licensor shall be under the terms and conditions of + this License, without any additional terms or conditions. + Notwithstanding the above, nothing herein shall supersede or modify + the terms of any separate license agreement you may have executed + with Licensor regarding such Contributions. + + 6. Trademarks. This License does not grant permission to use the trade + names, trademarks, service marks, or product names of the Licensor, + except as required for reasonable and customary use in describing the + origin of the Work and reproducing the content of the NOTICE file. + + 7. Disclaimer of Warranty. Unless required by applicable law or + agreed to in writing, Licensor provides the Work (and each + Contributor provides its Contributions) on an "AS IS" BASIS, + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or + implied, including, without limitation, any warranties or conditions + of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A + PARTICULAR PURPOSE. You are solely responsible for determining the + appropriateness of using or redistributing the Work and assume any + risks associated with Your exercise of permissions under this License. + + 8. Limitation of Liability. In no event and under no legal theory, + whether in tort (including negligence), contract, or otherwise, + unless required by applicable law (such as deliberate and grossly + negligent acts) or agreed to in writing, shall any Contributor be + liable to You for damages, including any direct, indirect, special, + incidental, or consequential damages of any character arising as a + result of this License or out of the use or inability to use the + Work (including but not limited to damages for loss of goodwill, + work stoppage, computer failure or malfunction, or any and all + other commercial damages or losses), even if such Contributor + has been advised of the possibility of such damages. + + 9. Accepting Warranty or Additional Liability. While redistributing + the Work or Derivative Works thereof, You may choose to offer, + and charge a fee for, acceptance of support, warranty, indemnity, + or other liability obligations and/or rights consistent with this + License. However, in accepting such obligations, You may act only + on Your own behalf and on Your sole responsibility, not on behalf + of any other Contributor, and only if You agree to indemnify, + defend, and hold each Contributor harmless for any liability + incurred by, or claims asserted against, such Contributor by reason + of your accepting any such warranty or additional liability. + + END OF TERMS AND CONDITIONS + + APPENDIX: How to apply the Apache License to your work. + + To apply the Apache License to your work, attach the following + boilerplate notice, with the fields enclosed by brackets "[]" + replaced with your own identifying information. (Don't include + the brackets!) The text should be enclosed in the appropriate + comment syntax for the file format. We also recommend that a + file or class name and description of purpose be included on the + same "printed page" as the copyright notice for easier + identification within third-party archives. + + Copyright 2015 xeipuuv + + Licensed under the Apache License, Version 2.0 (the "License"); + you may not use this file except in compliance with the License. + You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + + Unless required by applicable law or agreed to in writing, software + distributed under the License is distributed on an "AS IS" BASIS, + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + See the License for the specific language governing permissions and + limitations under the License. diff --git a/vendor/github.com/xeipuuv/gojsonreference/README.md b/vendor/github.com/xeipuuv/gojsonreference/README.md new file mode 100644 index 00000000..9ab6e1eb --- /dev/null +++ b/vendor/github.com/xeipuuv/gojsonreference/README.md @@ -0,0 +1,10 @@ +# gojsonreference +An implementation of JSON Reference - Go language + +## Dependencies +https://github.com/xeipuuv/gojsonpointer + +## References +http://tools.ietf.org/html/draft-ietf-appsawg-json-pointer-07 + +http://tools.ietf.org/html/draft-pbryan-zyp-json-ref-03 diff --git a/vendor/github.com/xeipuuv/gojsonreference/reference.go b/vendor/github.com/xeipuuv/gojsonreference/reference.go new file mode 100644 index 00000000..64572913 --- /dev/null +++ b/vendor/github.com/xeipuuv/gojsonreference/reference.go @@ -0,0 +1,147 @@ +// Copyright 2015 xeipuuv ( https://github.com/xeipuuv ) +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. + +// author xeipuuv +// author-github https://github.com/xeipuuv +// author-mail xeipuuv@gmail.com +// +// repository-name gojsonreference +// repository-desc An implementation of JSON Reference - Go language +// +// description Main and unique file. +// +// created 26-02-2013 + +package gojsonreference + +import ( + "errors" + "net/url" + "path/filepath" + "runtime" + "strings" + + "github.com/xeipuuv/gojsonpointer" +) + +const ( + const_fragment_char = `#` +) + +func NewJsonReference(jsonReferenceString string) (JsonReference, error) { + + var r JsonReference + err := r.parse(jsonReferenceString) + return r, err + +} + +type JsonReference struct { + referenceUrl *url.URL + referencePointer gojsonpointer.JsonPointer + + HasFullUrl bool + HasUrlPathOnly bool + HasFragmentOnly bool + HasFileScheme bool + HasFullFilePath bool +} + +func (r *JsonReference) GetUrl() *url.URL { + return r.referenceUrl +} + +func (r *JsonReference) GetPointer() *gojsonpointer.JsonPointer { + return &r.referencePointer +} + +func (r *JsonReference) String() string { + + if r.referenceUrl != nil { + return r.referenceUrl.String() + } + + if r.HasFragmentOnly { + return const_fragment_char + r.referencePointer.String() + } + + return r.referencePointer.String() +} + +func (r *JsonReference) IsCanonical() bool { + return (r.HasFileScheme && r.HasFullFilePath) || (!r.HasFileScheme && r.HasFullUrl) +} + +// "Constructor", parses the given string JSON reference +func (r *JsonReference) parse(jsonReferenceString string) (err error) { + + r.referenceUrl, err = url.Parse(jsonReferenceString) + if err != nil { + return + } + refUrl := r.referenceUrl + + if refUrl.Scheme != "" && refUrl.Host != "" { + r.HasFullUrl = true + } else { + if refUrl.Path != "" { + r.HasUrlPathOnly = true + } else if refUrl.RawQuery == "" && refUrl.Fragment != "" { + r.HasFragmentOnly = true + } + } + + r.HasFileScheme = refUrl.Scheme == "file" + if runtime.GOOS == "windows" { + // on Windows, a file URL may have an extra leading slash, and if it + // doesn't then its first component will be treated as the host by the + // Go runtime + if refUrl.Host == "" && strings.HasPrefix(refUrl.Path, "/") { + r.HasFullFilePath = filepath.IsAbs(refUrl.Path[1:]) + } else { + r.HasFullFilePath = filepath.IsAbs(refUrl.Host + refUrl.Path) + } + } else { + r.HasFullFilePath = filepath.IsAbs(refUrl.Path) + } + + // invalid json-pointer error means url has no json-pointer fragment. simply ignore error + r.referencePointer, _ = gojsonpointer.NewJsonPointer(refUrl.Fragment) + + return +} + +// Creates a new reference from a parent and a child +// If the child cannot inherit from the parent, an error is returned +func (r *JsonReference) Inherits(child JsonReference) (*JsonReference, error) { + if child.GetUrl() == nil { + return nil, errors.New("childUrl is nil!") + } + + if r.GetUrl() == nil { + return nil, errors.New("parentUrl is nil!") + } + + // Get a copy of the parent url to make sure we do not modify the original. + // URL reference resolving fails if the fragment of the child is empty, but the parent's is not. + // The fragment of the child must be used, so the fragment of the parent is manually removed. + parentUrl := *r.GetUrl() + parentUrl.Fragment = "" + + ref, err := NewJsonReference(parentUrl.ResolveReference(child.GetUrl()).String()) + if err != nil { + return nil, err + } + return &ref, err +} diff --git a/vendor/github.com/yashtewari/glob-intersection/LICENSE b/vendor/github.com/yashtewari/glob-intersection/LICENSE new file mode 100644 index 00000000..8dada3ed --- /dev/null +++ b/vendor/github.com/yashtewari/glob-intersection/LICENSE @@ -0,0 +1,201 @@ + Apache License + Version 2.0, January 2004 + http://www.apache.org/licenses/ + + TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION + + 1. Definitions. + + "License" shall mean the terms and conditions for use, reproduction, + and distribution as defined by Sections 1 through 9 of this document. + + "Licensor" shall mean the copyright owner or entity authorized by + the copyright owner that is granting the License. + + "Legal Entity" shall mean the union of the acting entity and all + other entities that control, are controlled by, or are under common + control with that entity. For the purposes of this definition, + "control" means (i) the power, direct or indirect, to cause the + direction or management of such entity, whether by contract or + otherwise, or (ii) ownership of fifty percent (50%) or more of the + outstanding shares, or (iii) beneficial ownership of such entity. + + "You" (or "Your") shall mean an individual or Legal Entity + exercising permissions granted by this License. + + "Source" form shall mean the preferred form for making modifications, + including but not limited to software source code, documentation + source, and configuration files. + + "Object" form shall mean any form resulting from mechanical + transformation or translation of a Source form, including but + not limited to compiled object code, generated documentation, + and conversions to other media types. + + "Work" shall mean the work of authorship, whether in Source or + Object form, made available under the License, as indicated by a + copyright notice that is included in or attached to the work + (an example is provided in the Appendix below). + + "Derivative Works" shall mean any work, whether in Source or Object + form, that is based on (or derived from) the Work and for which the + editorial revisions, annotations, elaborations, or other modifications + represent, as a whole, an original work of authorship. For the purposes + of this License, Derivative Works shall not include works that remain + separable from, or merely link (or bind by name) to the interfaces of, + the Work and Derivative Works thereof. + + "Contribution" shall mean any work of authorship, including + the original version of the Work and any modifications or additions + to that Work or Derivative Works thereof, that is intentionally + submitted to Licensor for inclusion in the Work by the copyright owner + or by an individual or Legal Entity authorized to submit on behalf of + the copyright owner. For the purposes of this definition, "submitted" + means any form of electronic, verbal, or written communication sent + to the Licensor or its representatives, including but not limited to + communication on electronic mailing lists, source code control systems, + and issue tracking systems that are managed by, or on behalf of, the + Licensor for the purpose of discussing and improving the Work, but + excluding communication that is conspicuously marked or otherwise + designated in writing by the copyright owner as "Not a Contribution." + + "Contributor" shall mean Licensor and any individual or Legal Entity + on behalf of whom a Contribution has been received by Licensor and + subsequently incorporated within the Work. + + 2. Grant of Copyright License. Subject to the terms and conditions of + this License, each Contributor hereby grants to You a perpetual, + worldwide, non-exclusive, no-charge, royalty-free, irrevocable + copyright license to reproduce, prepare Derivative Works of, + publicly display, publicly perform, sublicense, and distribute the + Work and such Derivative Works in Source or Object form. + + 3. Grant of Patent License. Subject to the terms and conditions of + this License, each Contributor hereby grants to You a perpetual, + worldwide, non-exclusive, no-charge, royalty-free, irrevocable + (except as stated in this section) patent license to make, have made, + use, offer to sell, sell, import, and otherwise transfer the Work, + where such license applies only to those patent claims licensable + by such Contributor that are necessarily infringed by their + Contribution(s) alone or by combination of their Contribution(s) + with the Work to which such Contribution(s) was submitted. If You + institute patent litigation against any entity (including a + cross-claim or counterclaim in a lawsuit) alleging that the Work + or a Contribution incorporated within the Work constitutes direct + or contributory patent infringement, then any patent licenses + granted to You under this License for that Work shall terminate + as of the date such litigation is filed. + + 4. Redistribution. You may reproduce and distribute copies of the + Work or Derivative Works thereof in any medium, with or without + modifications, and in Source or Object form, provided that You + meet the following conditions: + + (a) You must give any other recipients of the Work or + Derivative Works a copy of this License; and + + (b) You must cause any modified files to carry prominent notices + stating that You changed the files; and + + (c) You must retain, in the Source form of any Derivative Works + that You distribute, all copyright, patent, trademark, and + attribution notices from the Source form of the Work, + excluding those notices that do not pertain to any part of + the Derivative Works; and + + (d) If the Work includes a "NOTICE" text file as part of its + distribution, then any Derivative Works that You distribute must + include a readable copy of the attribution notices contained + within such NOTICE file, excluding those notices that do not + pertain to any part of the Derivative Works, in at least one + of the following places: within a NOTICE text file distributed + as part of the Derivative Works; within the Source form or + documentation, if provided along with the Derivative Works; or, + within a display generated by the Derivative Works, if and + wherever such third-party notices normally appear. The contents + of the NOTICE file are for informational purposes only and + do not modify the License. You may add Your own attribution + notices within Derivative Works that You distribute, alongside + or as an addendum to the NOTICE text from the Work, provided + that such additional attribution notices cannot be construed + as modifying the License. + + You may add Your own copyright statement to Your modifications and + may provide additional or different license terms and conditions + for use, reproduction, or distribution of Your modifications, or + for any such Derivative Works as a whole, provided Your use, + reproduction, and distribution of the Work otherwise complies with + the conditions stated in this License. + + 5. Submission of Contributions. Unless You explicitly state otherwise, + any Contribution intentionally submitted for inclusion in the Work + by You to the Licensor shall be under the terms and conditions of + this License, without any additional terms or conditions. + Notwithstanding the above, nothing herein shall supersede or modify + the terms of any separate license agreement you may have executed + with Licensor regarding such Contributions. + + 6. Trademarks. This License does not grant permission to use the trade + names, trademarks, service marks, or product names of the Licensor, + except as required for reasonable and customary use in describing the + origin of the Work and reproducing the content of the NOTICE file. + + 7. Disclaimer of Warranty. Unless required by applicable law or + agreed to in writing, Licensor provides the Work (and each + Contributor provides its Contributions) on an "AS IS" BASIS, + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or + implied, including, without limitation, any warranties or conditions + of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A + PARTICULAR PURPOSE. You are solely responsible for determining the + appropriateness of using or redistributing the Work and assume any + risks associated with Your exercise of permissions under this License. + + 8. Limitation of Liability. In no event and under no legal theory, + whether in tort (including negligence), contract, or otherwise, + unless required by applicable law (such as deliberate and grossly + negligent acts) or agreed to in writing, shall any Contributor be + liable to You for damages, including any direct, indirect, special, + incidental, or consequential damages of any character arising as a + result of this License or out of the use or inability to use the + Work (including but not limited to damages for loss of goodwill, + work stoppage, computer failure or malfunction, or any and all + other commercial damages or losses), even if such Contributor + has been advised of the possibility of such damages. + + 9. Accepting Warranty or Additional Liability. While redistributing + the Work or Derivative Works thereof, You may choose to offer, + and charge a fee for, acceptance of support, warranty, indemnity, + or other liability obligations and/or rights consistent with this + License. However, in accepting such obligations, You may act only + on Your own behalf and on Your sole responsibility, not on behalf + of any other Contributor, and only if You agree to indemnify, + defend, and hold each Contributor harmless for any liability + incurred by, or claims asserted against, such Contributor by reason + of your accepting any such warranty or additional liability. + + END OF TERMS AND CONDITIONS + + APPENDIX: How to apply the Apache License to your work. + + To apply the Apache License to your work, attach the following + boilerplate notice, with the fields enclosed by brackets "{}" + replaced with your own identifying information. (Don't include + the brackets!) The text should be enclosed in the appropriate + comment syntax for the file format. We also recommend that a + file or class name and description of purpose be included on the + same "printed page" as the copyright notice for easier + identification within third-party archives. + + Copyright {yyyy} {name of copyright owner} + + Licensed under the Apache License, Version 2.0 (the "License"); + you may not use this file except in compliance with the License. + You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + + Unless required by applicable law or agreed to in writing, software + distributed under the License is distributed on an "AS IS" BASIS, + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + See the License for the specific language governing permissions and + limitations under the License. diff --git a/vendor/github.com/yashtewari/glob-intersection/README.md b/vendor/github.com/yashtewari/glob-intersection/README.md new file mode 100644 index 00000000..618a8506 --- /dev/null +++ b/vendor/github.com/yashtewari/glob-intersection/README.md @@ -0,0 +1,26 @@ +# glob-intersection +Go package to check if the set of non-empty strings matched by the intersection of two regexp-style globs is non-empty. + +### Examples +- `gintersect.NonEmpty("a.a.", ".b.b")` is `true` because both globs match the string `abab`. +- `gintersect.NonEmpty("[a-z]+", "[0-9]*)` is `false` because there are no non-empty strings that both globs match. + +### Limitations + +- It is assumed that all input is rooted at the beginning and the end, i.e, starts and ends with the regexp symbols `^` and `$` respectively. This is done because any non-rooted expressions will always match a non-empty set of non-empty strings. +- The only special symbols are: + - `.` for any character. + - `+` for 1 or more of the preceding expression. + - `*` for 0 or more of the preceding expression. + - `[` and `]` to define regexp-style character classes. + - `-` to specify Unicode ranges inside character class definitions. + - `\` escapes any special symbol, including itself. + +### Complexity + +Complexity is exponential in the number of flags (`+` or `*`) present in the glob with the smaller flag count. +Benchmarks (see [`non_empty_bench_test.go`](/non_empty_bench_test.go)) reveal that inputs where one of the globs has <= 10 flags, and both globs have 100s of characters, will run in less than a nanosecond. This should be ok for most use cases. + +### Acknowledgements + +[This StackOverflow discussion](https://stackoverflow.com/questions/18695727/algorithm-to-find-out-whether-the-matches-for-two-glob-patterns-or-regular-expr) for fleshing out the logic. diff --git a/vendor/github.com/yashtewari/glob-intersection/glob.go b/vendor/github.com/yashtewari/glob-intersection/glob.go new file mode 100644 index 00000000..54d4729b --- /dev/null +++ b/vendor/github.com/yashtewari/glob-intersection/glob.go @@ -0,0 +1,182 @@ +// Package gintersect provides methods to check whether the intersection of several globs matches a non-empty set of strings. +package gintersect + +import ( + "fmt" + "strings" +) + +// Glob represents a glob. +type Glob []Token + +// NewGlob constructs a Glob from the given string by tokenizing and then simplifying it, or reports errors if any. +func NewGlob(input string) (Glob, error) { + tokens, err := Tokenize([]rune(input)) + if err != nil { + return nil, err + } + + tokens = Simplify(tokens) + + return Glob(tokens), nil +} + +// TokenType is the type of a Token. +type TokenType uint + +const ( + TTCharacter TokenType = iota + TTDot + TTSet +) + +// Flag applies to a token. +type Flag uint + +func (f Flag) String() (s string) { + for r, flag := range flagRunes { + if f == flag { + s = string(r) + break + } + } + return +} + +const ( + FlagNone = iota + FlagPlus + FlagStar +) + +// Token is the element that makes up a Glob. +type Token interface { + Type() TokenType + Flag() Flag + SetFlag(Flag) + // Equal describes whether the given Token is exactly equal to this one, barring differences in flags. + Equal(Token) bool + String() string +} + +// token is the base for all structs implementing Token. +type token struct { + ttype TokenType + flag Flag +} + +func (t token) Type() TokenType { + return t.ttype +} + +func (t token) Flag() Flag { + return t.flag +} + +func (t *token) SetFlag(f Flag) { + t.flag = f +} + +// character is a specific rune. It implements Token. +type character struct { + token + r rune +} + +func NewCharacter(r rune) Token { + return &character{ + token: token{ttype: TTCharacter}, + r: r, + } +} + +func (c character) Equal(other Token) bool { + if c.Type() != other.Type() { + return false + } + + o := other.(*character) + return c.Rune() == o.Rune() +} + +func (c character) String() string { + return fmt.Sprintf("{character: %s flag: %s}", string(c.Rune()), c.Flag().String()) +} + +func (c character) Rune() rune { + return c.r +} + +// dot is any character. It implements Token. +type dot struct { + token +} + +func NewDot() Token { + return &dot{ + token: token{ttype: TTDot}, + } +} + +func (d dot) Equal(other Token) bool { + if d.Type() != other.Type() { + return false + } + + return true +} + +func (d dot) String() string { + return fmt.Sprintf("{dot flag: %s}", d.Flag().String()) +} + +// set is a set of characters (similar to regexp character class). +// It implements Token. +type set struct { + token + runes map[rune]bool +} + +func NewSet(runes []rune) Token { + m := map[rune]bool{} + for _, r := range runes { + m[r] = true + } + return &set{ + token: token{ttype: TTSet}, + runes: m, + } +} + +func (s set) Equal(other Token) bool { + if s.Type() != other.Type() { + return false + } + + o := other.(*set) + r1, r2 := s.Runes(), o.Runes() + + if len(r1) != len(r2) { + return false + } + + for k, _ := range r1 { + if _, ok := r2[k]; !ok { + return false + } + } + + return true +} + +func (s set) String() string { + rs := make([]string, 0, 30) + for r, _ := range s.Runes() { + rs = append(rs, string(r)) + } + return fmt.Sprintf("{set: %s flag: %s}", strings.Join(rs, ""), s.Flag().String()) +} + +func (s set) Runes() map[rune]bool { + return s.runes +} diff --git a/vendor/github.com/yashtewari/glob-intersection/match.go b/vendor/github.com/yashtewari/glob-intersection/match.go new file mode 100644 index 00000000..45a988a8 --- /dev/null +++ b/vendor/github.com/yashtewari/glob-intersection/match.go @@ -0,0 +1,91 @@ +package gintersect + +import ( + "github.com/pkg/errors" +) + +var ( + errBadImplementation = errors.New("this logical path is invalid") +) + +// Match implements single-Token matching, ignoring flags. +// Example: [a-d] and [b-e] match, while [a-z] and [0-9] do not. +func Match(t1 Token, t2 Token) bool { + var temp Token + if t1.Type() > t2.Type() { + temp = t1 + t1 = t2 + t2 = temp + } + + switch t1.Type() { + case TTCharacter: + ch := t1.(*character) + + switch t2.Type() { + case TTCharacter: + return matchCharacters(ch, t2.(*character)) + case TTDot: + return matchCharacterDot(ch, t2.(*dot)) + case TTSet: + return matchCharacterSet(ch, t2.(*set)) + default: + panic(errBadImplementation) + } + + case TTDot: + d := t1.(*dot) + + switch t2.Type() { + case TTDot: + return matchDots(d, t2.(*dot)) + case TTSet: + return matchDotSet(d, t2.(*set)) + default: + panic(errBadImplementation) + } + + case TTSet: + switch t2.Type() { + case TTSet: + return matchSets(t1.(*set), t2.(*set)) + default: + panic(errBadImplementation) + } + + default: + panic(errBadImplementation) + + } +} + +func matchCharacters(a *character, b *character) bool { + return a.Rune() == b.Rune() +} + +func matchCharacterDot(a *character, b *dot) bool { + return true +} + +func matchCharacterSet(a *character, b *set) bool { + _, ok := b.Runes()[a.Rune()] + return ok +} + +func matchDots(a *dot, b *dot) bool { + return true +} + +func matchDotSet(a *dot, b *set) bool { + return true +} + +func matchSets(a *set, b *set) bool { + for k, _ := range a.Runes() { + if _, ok := b.Runes()[k]; ok { + return true + } + } + + return false +} diff --git a/vendor/github.com/yashtewari/glob-intersection/non_empty.go b/vendor/github.com/yashtewari/glob-intersection/non_empty.go new file mode 100644 index 00000000..91cbdbde --- /dev/null +++ b/vendor/github.com/yashtewari/glob-intersection/non_empty.go @@ -0,0 +1,154 @@ +package gintersect + +// NonEmpty is true if the intersection of lhs and rhs matches a non-empty set of non-empty str1ngs. +func NonEmpty(lhs string, rhs string) (bool, error) { + g1, err := NewGlob(lhs) + if err != nil { + return false, err + } + + g2, err := NewGlob(rhs) + if err != nil { + return false, err + } + + var match bool + g1, g2, match = trimGlobs(g1, g2) + if !match { + return false, nil + } + + return intersectNormal(g1, g2), nil +} + +// trimGlobs removes matching prefixes and suffixes from g1, g2, or returns false if prefixes/suffixes don't match. +func trimGlobs(g1, g2 Glob) (Glob, Glob, bool) { + var l, r1, r2 int + + // Trim from the beginning until a flagged Token or a mismatch is found. + for l = 0; l < len(g1) && l < len(g2) && g1[l].Flag() == FlagNone && g2[l].Flag() == FlagNone; l++ { + if !Match(g1[l], g2[l]) { + return nil, nil, false + } + } + + // Leave one prefix Token untrimmed to avoid empty Globs because those will break the algorithm. + if l > 0 { + l-- + } + + // Trim from the end until a flagged Token or a mismatch is found. + for r1, r2 = len(g1)-1, len(g2)-1; r1 >= 0 && r1 >= l && r2 >= 0 && r2 >= l && g1[r1].Flag() == FlagNone && g2[r2].Flag() == FlagNone; r1, r2 = r1-1, r2-1 { + if !Match(g1[r1], g2[r2]) { + return nil, nil, false + } + } + + // Leave one suffix Token untrimmed to avoid empty Globs because those will break the algorithm. + if r1 < len(g1)-1 { + r1++ + r2++ + } + + return g1[l : r1+1], g2[l : r2+1], true +} + +// All uses of `intersection exists` below mean that the intersection of the globs matches a non-empty set of non-empty strings. + +// intersectNormal accepts two globs and returns a boolean describing whether their intersection exists. +// It traverses g1, g2 while ensuring that their Tokens match. +// If a flagged Token is encountered, flow of control is handed off to intersectSpecial. +func intersectNormal(g1, g2 Glob) bool { + var i, j int + for i, j = 0, 0; i < len(g1) && j < len(g2); i, j = i+1, j+1 { + if g1[i].Flag() == FlagNone && g2[j].Flag() == FlagNone { + if !Match(g1[i], g2[j]) { + return false + } + } else { + return intersectSpecial(g1[i:], g2[j:]) + } + } + + if i == len(g1) && j == len(g2) { + return true + } + + return false +} + +// intersectSpecial accepts two globs such that at least one starts with a flagged Token. +// It returns a boolean describing whether their intersection exists. +// It hands flow of control to intersectPlus or intersectStar correctly. +func intersectSpecial(g1, g2 Glob) bool { + if g1[0].Flag() != FlagNone { // If g1 starts with a Token having a Flag. + switch g1[0].Flag() { + case FlagPlus: + return intersectPlus(g1, g2) + case FlagStar: + return intersectStar(g1, g2) + } + } else { // If g2 starts with a Token having a Flag. + switch g2[0].Flag() { + case FlagPlus: + return intersectPlus(g2, g1) + case FlagStar: + return intersectStar(g2, g1) + } + } + + return false +} + +// intersectPlus accepts two globs such that plussed[0].Flag() == FlagPlus. +// It returns a boolean describing whether their intersection exists. +// It ensures that at least one token in other maches plussed[0], before handing flow of control to intersectSpecial. +func intersectPlus(plussed, other Glob) bool { + if !Match(plussed[0], other[0]) { + return false + } + return intersectStar(plussed, other[1:]) +} + +// intersectStar accepts two globs such that starred[0].Flag() == FlagStar. +// It returns a boolean describing whether their intersection exists. +// It gobbles up Tokens from other until the Tokens remaining in other intersect with starred[1:] +func intersectStar(starred, other Glob) bool { + // starToken, nextToken are the token having FlagStar and the one that follows immediately after, respectively. + var starToken, nextToken Token + + starToken = starred[0] + if len(starred) > 1 { + nextToken = starred[1] + } + + for i, t := range other { + // Start gobbl1ng up tokens in other while they match starToken. + if nextToken != nil && Match(t, nextToken) { + // When a token in other matches the token after starToken, stop gobbl1ng and try to match the two all the way. + allTheWay := intersectNormal(starred[1:], other[i:]) + // If they match all the way, the Globs intersect. + if allTheWay { + return true + } else { + // If they don't match all the way, then the current token from other should still match starToken. + if !Match(t, starToken) { + return false + } + } + } else { + // Only move forward if this token can be gobbled up by starToken. + if !Match(t, starToken) { + return false + } + } + } + + // If there was no token following starToken, and everything from other was gobbled, the Globs intersect. + if nextToken == nil { + return true + } + + // If everything from other was gobbles but there was a nextToken to match, they don't intersect. + return false +} diff --git a/vendor/github.com/yashtewari/glob-intersection/simplify.go b/vendor/github.com/yashtewari/glob-intersection/simplify.go new file mode 100644 index 00000000..7704fb10 --- /dev/null +++ b/vendor/github.com/yashtewari/glob-intersection/simplify.go @@ -0,0 +1,43 @@ +package gintersect + +// Simplify accepts a Token slice and returns a equivalient Token slice that is shorter/simpler. +// The only simplification currently applied is removing redundant flagged Tokens. +// TODO: Remove unflagged Tokens next to equivalen Tokens with FlagPlus. Example: tt+t == t+ +func Simplify(tokens []Token) []Token { + if len(tokens) == 0 { + return tokens + } + simple := make([]Token, 1, len(tokens)) + simple[0] = tokens[0] + + latest := simple[0] + + for i := 1; i < len(tokens); i++ { + handled := false + // Possible simplifications to apply if there is a flag. + if tokens[i].Flag() != FlagNone && latest.Flag() != FlagNone { + // If the token contents are the same, then apply simplification. + if tokens[i].Equal(latest) { + var flag Flag + // FlagPlus takes precedence, because: + // t+t* == t+ + // t*t+ == t+ + if tokens[i].Flag() == FlagPlus || latest.Flag() == FlagPlus { + flag = FlagPlus + } else { + flag = FlagStar + } + + simple[len(simple)-1].SetFlag(flag) + handled = true + } + } + + if !handled { + latest = tokens[i] + simple = append(simple, tokens[i]) + } + } + + return simple +} diff --git a/vendor/github.com/yashtewari/glob-intersection/test_samples.go b/vendor/github.com/yashtewari/glob-intersection/test_samples.go new file mode 100644 index 00000000..5d2922c7 --- /dev/null +++ b/vendor/github.com/yashtewari/glob-intersection/test_samples.go @@ -0,0 +1,84 @@ +package gintersect + +var ( + samplesInitialized = false + + testCharacters map[rune]Token + testCharactersPlus map[rune]Token + testCharactersStar map[rune]Token + + testDot, testDotPlus, testDotStar Token + + testLowerAlphaSet, testLowerAlphaSetPlus, lowerAplhaSetStar Token + testUpperAlphaSet, testUpperAlphaSetPlus, testUpperAlphaSetStar Token + testNumSet, testNumSetPlus, testNumSetStar Token + testSymbolSet, testSymbolSetPlus, testSymbolSetStar Token + + testEmptySet Token +) + +func initializeTestSamples() { + if samplesInitialized { + return + } + + testCharacters, testCharactersPlus, testCharactersStar = make(map[rune]Token), make(map[rune]Token), make(map[rune]Token) + + testDot, testDotPlus, testDotStar = NewDot(), NewDot(), NewDot() + testDotPlus.SetFlag(FlagPlus) + testDotStar.SetFlag(FlagStar) + + var runes []rune + runes = makeRunes('a', 'z') + + testLowerAlphaSet, testLowerAlphaSetPlus, lowerAplhaSetStar = NewSet(runes), NewSet(runes), NewSet(runes) + testLowerAlphaSetPlus.SetFlag(FlagPlus) + lowerAplhaSetStar.SetFlag(FlagStar) + + runes = makeRunes('A', 'Z') + + testUpperAlphaSet, testUpperAlphaSetPlus, testUpperAlphaSetStar = NewSet(runes), NewSet(runes), NewSet(runes) + testUpperAlphaSetPlus.SetFlag(FlagPlus) + testUpperAlphaSetStar.SetFlag(FlagStar) + + runes = makeRunes('0', '9') + + testNumSet, testNumSetPlus, testNumSetStar = NewSet(runes), NewSet(runes), NewSet(runes) + testNumSetPlus.SetFlag(FlagPlus) + testNumSetStar.SetFlag(FlagStar) + + runes = makeRunes('!', '/') + + testSymbolSet, testSymbolSetPlus, testSymbolSetStar = NewSet(runes), NewSet(runes), NewSet(runes) + testSymbolSetPlus.SetFlag(FlagPlus) + testSymbolSetStar.SetFlag(FlagStar) + + testEmptySet = NewSet([]rune{}) + + samplesInitialized = true +} + +func makeRunes(from rune, to rune) []rune { + runes := make([]rune, 0, 30) + for r := from; r <= to; r++ { + runes = append(runes, r) + addToCharacters(r) + } + + return runes +} + +func addToCharacters(r rune) { + var t Token + + t = NewCharacter(r) + testCharacters[r] = t + + t = NewCharacter(r) + t.SetFlag(FlagPlus) + testCharactersPlus[r] = t + + t = NewCharacter(r) + t.SetFlag(FlagStar) + testCharactersStar[r] = t +} diff --git a/vendor/github.com/yashtewari/glob-intersection/tokenize.go b/vendor/github.com/yashtewari/glob-intersection/tokenize.go new file mode 100644 index 00000000..0b674316 --- /dev/null +++ b/vendor/github.com/yashtewari/glob-intersection/tokenize.go @@ -0,0 +1,251 @@ +package gintersect + +import ( + "fmt" + + "github.com/pkg/errors" +) + +// Modifier is a special character that affects lexical analysis. +type Modifier uint + +const ( + ModifierBackslash Modifier = iota +) + +var ( + // Special runes. + tokenTypeRunes = map[rune]TokenType{ + '.': TTDot, + '[': TTSet, + ']': TTSet, + } + flagRunes = map[rune]Flag{ + '+': FlagPlus, + '*': FlagStar, + } + modifierRunes = map[rune]Modifier{ + '\\': ModifierBackslash, + } + + // Errors. + ErrInvalidInput = errors.New("the input provided is invalid") + errEndOfInput = errors.New("reached end of input") +) + +// Tokenize converts a rune slice into a Token slice. +func Tokenize(input []rune) ([]Token, error) { + tokens := []Token{} + for i, t, err := nextToken(0, input); err != errEndOfInput; i, t, err = nextToken(i, input) { + if err != nil { + return nil, err + } + + tokens = append(tokens, t) + } + + return tokens, nil +} + +// nextToken yields the Token starting at the given index of input, and newIndex at which the next Token should start. +func nextToken(index int, input []rune) (newIndex int, token Token, err error) { + var r rune + var escaped bool + + newIndex, r, escaped, err = nextRune(index, input) + if err != nil { + return + } + + if !escaped { + if ttype, ok := tokenTypeRunes[r]; ok { + switch ttype { + case TTDot: + token = NewDot() + + case TTSet: + if r == ']' { + err = errors.Wrap(ErrInvalidInput, invalidInputMessage(input, newIndex, "set-close ']' with no preceding '['")) + return + } + + newIndex, token, err = nextTokenSet(newIndex, input) + if err != nil { + return + } + + default: + panic(errors.Wrapf(errBadImplementation, "encountered unhandled token type: %v", ttype)) + } + + } else if _, ok := flagRunes[r]; ok { + err = errors.Wrap(ErrInvalidInput, invalidInputMessage(input, newIndex, "flag '%s' must be preceded by a non-flag", string(r))) + return + + } else if m, ok := modifierRunes[r]; ok { + panic(errors.Wrapf(errBadImplementation, "encountered unhandled modifier: %v", m)) + } else { + // Nothing special to do. + token = NewCharacter(r) + } + } else { + // Nothing special to do. + token = NewCharacter(r) + } + + var f Flag + newIndex, f, err = nextFlag(newIndex, input) + if err == errEndOfInput { + // Let this err be passed in the next cycle, after the current token is consumed. + err = nil + } else if err != nil { + return + } + + token.SetFlag(f) + + return +} + +// nextTokenSet yields a Token having type TokenSet and starting at the given index of input. +// The next Token/Flag should start at newIndex. +func nextTokenSet(index int, input []rune) (newIndex int, t Token, err error) { + var r, prev rune + var escaped bool + + runes := make([]rune, 0, 30) + complete, prevExists := false, false + + newIndex, r, escaped, err = nextRune(index, input) + // If errEndOfInput is encountered, flow of control proceeds to the end of the function, + // where the error is handled. + if err != nil && err != errEndOfInput { + return + } + + for ; err != errEndOfInput; newIndex, r, escaped, err = nextRune(newIndex, input) { + if err != nil { + return + } + + if !escaped { + // Handle symbols. + switch r { + case '-': + if !prevExists { + err = errors.Wrap(ErrInvalidInput, invalidInputMessage(input, newIndex, "range character '-' must be preceded by a Unicode character")) + return + } + if newIndex >= len(input)-1 { + err = errors.Wrap(ErrInvalidInput, invalidInputMessage(input, newIndex, "range character '-' must be followed by a Unicode character")) + return + } + + // Get the next rune to know the extent of the range. + newIndex, r, escaped, err = nextRune(newIndex, input) + + if !escaped { + if r == ']' || r == '-' { + err = errors.Wrap(ErrInvalidInput, invalidInputMessage(input, newIndex, "range character '-' cannot be followed by a special symbol")) + return + } + } + if r < prev { + err = errors.Wrap(ErrInvalidInput, invalidInputMessage(input, newIndex, "range is out of order: '%s' comes before '%s' in Unicode", string(r), string(prev))) + return + } + + for x := prev; x <= r; x++ { + runes = append(runes, x) + } + + prevExists = false + + case ']': + complete = true + + // Nothing special to do. + default: + runes = append(runes, r) + prev, prevExists = r, true + } + } else { + // Nothing special to do. + runes = append(runes, r) + prev, prevExists = r, true + } + + // Don't move the index forward if the set is complete. + if complete { + break + } + } + + // End of input is reached before the set completes. + if !complete { + err = errors.Wrap(ErrInvalidInput, invalidInputMessage(input, newIndex, "found [ without matching ]")) + } else { + t = NewSet(runes) + } + + return +} + +// nextFlag yields the Flag starting at the given index of input, if any. +// The next Token should start at newIndex. +func nextFlag(index int, input []rune) (newIndex int, f Flag, err error) { + var escaped, ok bool + var r rune + + f = FlagNone + + newIndex, r, escaped, err = nextRune(index, input) + if err != nil { + return + } + + if !escaped { + // Revert back to index for later consumption. + if f, ok = flagRunes[r]; !ok { + newIndex = index + } + } else { + // Revert back to index for later consumption. + newIndex = index + } + + return +} + +// nextRune yields the rune starting (with modifiers) at the given index of input, with boolean escaped describing whether the rune is escaped. +// The next rune should start at newIndex. +func nextRune(index int, input []rune) (newIndex int, r rune, escaped bool, err error) { + if index >= len(input) { + newIndex = index + err = errEndOfInput + return + } + + if m, ok := modifierRunes[input[index]]; ok { + switch m { + + case ModifierBackslash: + if index < len(input)-1 { + newIndex, r, escaped = index+2, input[index+1], true + } else if index == len(input)-1 { + err = errors.Wrap(ErrInvalidInput, invalidInputMessage(input, index, "input ends with a \\ (escape) character")) + } + default: + panic(errors.Wrapf(errBadImplementation, "encountered unhandled modifier: %v", m)) + } + } else { + newIndex, r, escaped = index+1, input[index], false + } + + return +} + +// invalidInputMessage wraps the message describing invalid input with the input itself and index at which it is invalid. +func invalidInputMessage(input []rune, index int, message string, args ...interface{}) string { + return fmt.Sprintf("input:%s, pos:%d, %s", string(input), index, fmt.Sprintf(message, args...)) +} diff --git a/vendor/gopkg.in/yaml.v2/.travis.yml b/vendor/gopkg.in/yaml.v2/.travis.yml new file mode 100644 index 00000000..7348c50c --- /dev/null +++ b/vendor/gopkg.in/yaml.v2/.travis.yml @@ -0,0 +1,17 @@ +language: go + +go: + - "1.4.x" + - "1.5.x" + - "1.6.x" + - "1.7.x" + - "1.8.x" + - "1.9.x" + - "1.10.x" + - "1.11.x" + - "1.12.x" + - "1.13.x" + - "1.14.x" + - "tip" + +go_import_path: gopkg.in/yaml.v2 diff --git a/vendor/gopkg.in/yaml.v2/LICENSE b/vendor/gopkg.in/yaml.v2/LICENSE new file mode 100644 index 00000000..8dada3ed --- /dev/null +++ b/vendor/gopkg.in/yaml.v2/LICENSE @@ -0,0 +1,201 @@ + Apache License + Version 2.0, January 2004 + http://www.apache.org/licenses/ + + TERMS AND CONDITIONS FOR USE, REPRODUCTION, AND DISTRIBUTION + + 1. Definitions. + + "License" shall mean the terms and conditions for use, reproduction, + and distribution as defined by Sections 1 through 9 of this document. + + "Licensor" shall mean the copyright owner or entity authorized by + the copyright owner that is granting the License. + + "Legal Entity" shall mean the union of the acting entity and all + other entities that control, are controlled by, or are under common + control with that entity. For the purposes of this definition, + "control" means (i) the power, direct or indirect, to cause the + direction or management of such entity, whether by contract or + otherwise, or (ii) ownership of fifty percent (50%) or more of the + outstanding shares, or (iii) beneficial ownership of such entity. + + "You" (or "Your") shall mean an individual or Legal Entity + exercising permissions granted by this License. + + "Source" form shall mean the preferred form for making modifications, + including but not limited to software source code, documentation + source, and configuration files. + + "Object" form shall mean any form resulting from mechanical + transformation or translation of a Source form, including but + not limited to compiled object code, generated documentation, + and conversions to other media types. + + "Work" shall mean the work of authorship, whether in Source or + Object form, made available under the License, as indicated by a + copyright notice that is included in or attached to the work + (an example is provided in the Appendix below). + + "Derivative Works" shall mean any work, whether in Source or Object + form, that is based on (or derived from) the Work and for which the + editorial revisions, annotations, elaborations, or other modifications + represent, as a whole, an original work of authorship. For the purposes + of this License, Derivative Works shall not include works that remain + separable from, or merely link (or bind by name) to the interfaces of, + the Work and Derivative Works thereof. + + "Contribution" shall mean any work of authorship, including + the original version of the Work and any modifications or additions + to that Work or Derivative Works thereof, that is intentionally + submitted to Licensor for inclusion in the Work by the copyright owner + or by an individual or Legal Entity authorized to submit on behalf of + the copyright owner. For the purposes of this definition, "submitted" + means any form of electronic, verbal, or written communication sent + to the Licensor or its representatives, including but not limited to + communication on electronic mailing lists, source code control systems, + and issue tracking systems that are managed by, or on behalf of, the + Licensor for the purpose of discussing and improving the Work, but + excluding communication that is conspicuously marked or otherwise + designated in writing by the copyright owner as "Not a Contribution." + + "Contributor" shall mean Licensor and any individual or Legal Entity + on behalf of whom a Contribution has been received by Licensor and + subsequently incorporated within the Work. + + 2. Grant of Copyright License. Subject to the terms and conditions of + this License, each Contributor hereby grants to You a perpetual, + worldwide, non-exclusive, no-charge, royalty-free, irrevocable + copyright license to reproduce, prepare Derivative Works of, + publicly display, publicly perform, sublicense, and distribute the + Work and such Derivative Works in Source or Object form. + + 3. Grant of Patent License. Subject to the terms and conditions of + this License, each Contributor hereby grants to You a perpetual, + worldwide, non-exclusive, no-charge, royalty-free, irrevocable + (except as stated in this section) patent license to make, have made, + use, offer to sell, sell, import, and otherwise transfer the Work, + where such license applies only to those patent claims licensable + by such Contributor that are necessarily infringed by their + Contribution(s) alone or by combination of their Contribution(s) + with the Work to which such Contribution(s) was submitted. If You + institute patent litigation against any entity (including a + cross-claim or counterclaim in a lawsuit) alleging that the Work + or a Contribution incorporated within the Work constitutes direct + or contributory patent infringement, then any patent licenses + granted to You under this License for that Work shall terminate + as of the date such litigation is filed. + + 4. Redistribution. You may reproduce and distribute copies of the + Work or Derivative Works thereof in any medium, with or without + modifications, and in Source or Object form, provided that You + meet the following conditions: + + (a) You must give any other recipients of the Work or + Derivative Works a copy of this License; and + + (b) You must cause any modified files to carry prominent notices + stating that You changed the files; and + + (c) You must retain, in the Source form of any Derivative Works + that You distribute, all copyright, patent, trademark, and + attribution notices from the Source form of the Work, + excluding those notices that do not pertain to any part of + the Derivative Works; and + + (d) If the Work includes a "NOTICE" text file as part of its + distribution, then any Derivative Works that You distribute must + include a readable copy of the attribution notices contained + within such NOTICE file, excluding those notices that do not + pertain to any part of the Derivative Works, in at least one + of the following places: within a NOTICE text file distributed + as part of the Derivative Works; within the Source form or + documentation, if provided along with the Derivative Works; or, + within a display generated by the Derivative Works, if and + wherever such third-party notices normally appear. The contents + of the NOTICE file are for informational purposes only and + do not modify the License. You may add Your own attribution + notices within Derivative Works that You distribute, alongside + or as an addendum to the NOTICE text from the Work, provided + that such additional attribution notices cannot be construed + as modifying the License. + + You may add Your own copyright statement to Your modifications and + may provide additional or different license terms and conditions + for use, reproduction, or distribution of Your modifications, or + for any such Derivative Works as a whole, provided Your use, + reproduction, and distribution of the Work otherwise complies with + the conditions stated in this License. + + 5. Submission of Contributions. Unless You explicitly state otherwise, + any Contribution intentionally submitted for inclusion in the Work + by You to the Licensor shall be under the terms and conditions of + this License, without any additional terms or conditions. + Notwithstanding the above, nothing herein shall supersede or modify + the terms of any separate license agreement you may have executed + with Licensor regarding such Contributions. + + 6. Trademarks. This License does not grant permission to use the trade + names, trademarks, service marks, or product names of the Licensor, + except as required for reasonable and customary use in describing the + origin of the Work and reproducing the content of the NOTICE file. + + 7. Disclaimer of Warranty. Unless required by applicable law or + agreed to in writing, Licensor provides the Work (and each + Contributor provides its Contributions) on an "AS IS" BASIS, + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or + implied, including, without limitation, any warranties or conditions + of TITLE, NON-INFRINGEMENT, MERCHANTABILITY, or FITNESS FOR A + PARTICULAR PURPOSE. You are solely responsible for determining the + appropriateness of using or redistributing the Work and assume any + risks associated with Your exercise of permissions under this License. + + 8. Limitation of Liability. In no event and under no legal theory, + whether in tort (including negligence), contract, or otherwise, + unless required by applicable law (such as deliberate and grossly + negligent acts) or agreed to in writing, shall any Contributor be + liable to You for damages, including any direct, indirect, special, + incidental, or consequential damages of any character arising as a + result of this License or out of the use or inability to use the + Work (including but not limited to damages for loss of goodwill, + work stoppage, computer failure or malfunction, or any and all + other commercial damages or losses), even if such Contributor + has been advised of the possibility of such damages. + + 9. Accepting Warranty or Additional Liability. While redistributing + the Work or Derivative Works thereof, You may choose to offer, + and charge a fee for, acceptance of support, warranty, indemnity, + or other liability obligations and/or rights consistent with this + License. However, in accepting such obligations, You may act only + on Your own behalf and on Your sole responsibility, not on behalf + of any other Contributor, and only if You agree to indemnify, + defend, and hold each Contributor harmless for any liability + incurred by, or claims asserted against, such Contributor by reason + of your accepting any such warranty or additional liability. + + END OF TERMS AND CONDITIONS + + APPENDIX: How to apply the Apache License to your work. + + To apply the Apache License to your work, attach the following + boilerplate notice, with the fields enclosed by brackets "{}" + replaced with your own identifying information. (Don't include + the brackets!) The text should be enclosed in the appropriate + comment syntax for the file format. We also recommend that a + file or class name and description of purpose be included on the + same "printed page" as the copyright notice for easier + identification within third-party archives. + + Copyright {yyyy} {name of copyright owner} + + Licensed under the Apache License, Version 2.0 (the "License"); + you may not use this file except in compliance with the License. + You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + + Unless required by applicable law or agreed to in writing, software + distributed under the License is distributed on an "AS IS" BASIS, + WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. + See the License for the specific language governing permissions and + limitations under the License. diff --git a/vendor/gopkg.in/yaml.v2/LICENSE.libyaml b/vendor/gopkg.in/yaml.v2/LICENSE.libyaml new file mode 100644 index 00000000..8da58fbf --- /dev/null +++ b/vendor/gopkg.in/yaml.v2/LICENSE.libyaml @@ -0,0 +1,31 @@ +The following files were ported to Go from C files of libyaml, and thus +are still covered by their original copyright and license: + + apic.go + emitterc.go + parserc.go + readerc.go + scannerc.go + writerc.go + yamlh.go + yamlprivateh.go + +Copyright (c) 2006 Kirill Simonov + +Permission is hereby granted, free of charge, to any person obtaining a copy of +this software and associated documentation files (the "Software"), to deal in +the Software without restriction, including without limitation the rights to +use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies +of the Software, and to permit persons to whom the Software is furnished to do +so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in all +copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE +SOFTWARE. diff --git a/vendor/gopkg.in/yaml.v2/NOTICE b/vendor/gopkg.in/yaml.v2/NOTICE new file mode 100644 index 00000000..866d74a7 --- /dev/null +++ b/vendor/gopkg.in/yaml.v2/NOTICE @@ -0,0 +1,13 @@ +Copyright 2011-2016 Canonical Ltd. + +Licensed under the Apache License, Version 2.0 (the "License"); +you may not use this file except in compliance with the License. +You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + +Unless required by applicable law or agreed to in writing, software +distributed under the License is distributed on an "AS IS" BASIS, +WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +See the License for the specific language governing permissions and +limitations under the License. diff --git a/vendor/gopkg.in/yaml.v2/README.md b/vendor/gopkg.in/yaml.v2/README.md new file mode 100644 index 00000000..b50c6e87 --- /dev/null +++ b/vendor/gopkg.in/yaml.v2/README.md @@ -0,0 +1,133 @@ +# YAML support for the Go language + +Introduction +------------ + +The yaml package enables Go programs to comfortably encode and decode YAML +values. It was developed within [Canonical](https://www.canonical.com) as +part of the [juju](https://juju.ubuntu.com) project, and is based on a +pure Go port of the well-known [libyaml](http://pyyaml.org/wiki/LibYAML) +C library to parse and generate YAML data quickly and reliably. + +Compatibility +------------- + +The yaml package supports most of YAML 1.1 and 1.2, including support for +anchors, tags, map merging, etc. Multi-document unmarshalling is not yet +implemented, and base-60 floats from YAML 1.1 are purposefully not +supported since they're a poor design and are gone in YAML 1.2. + +Installation and usage +---------------------- + +The import path for the package is *gopkg.in/yaml.v2*. + +To install it, run: + + go get gopkg.in/yaml.v2 + +API documentation +----------------- + +If opened in a browser, the import path itself leads to the API documentation: + + * [https://gopkg.in/yaml.v2](https://gopkg.in/yaml.v2) + +API stability +------------- + +The package API for yaml v2 will remain stable as described in [gopkg.in](https://gopkg.in). + + +License +------- + +The yaml package is licensed under the Apache License 2.0. Please see the LICENSE file for details. + + +Example +------- + +```Go +package main + +import ( + "fmt" + "log" + + "gopkg.in/yaml.v2" +) + +var data = ` +a: Easy! +b: + c: 2 + d: [3, 4] +` + +// Note: struct fields must be public in order for unmarshal to +// correctly populate the data. +type T struct { + A string + B struct { + RenamedC int `yaml:"c"` + D []int `yaml:",flow"` + } +} + +func main() { + t := T{} + + err := yaml.Unmarshal([]byte(data), &t) + if err != nil { + log.Fatalf("error: %v", err) + } + fmt.Printf("--- t:\n%v\n\n", t) + + d, err := yaml.Marshal(&t) + if err != nil { + log.Fatalf("error: %v", err) + } + fmt.Printf("--- t dump:\n%s\n\n", string(d)) + + m := make(map[interface{}]interface{}) + + err = yaml.Unmarshal([]byte(data), &m) + if err != nil { + log.Fatalf("error: %v", err) + } + fmt.Printf("--- m:\n%v\n\n", m) + + d, err = yaml.Marshal(&m) + if err != nil { + log.Fatalf("error: %v", err) + } + fmt.Printf("--- m dump:\n%s\n\n", string(d)) +} +``` + +This example will generate the following output: + +``` +--- t: +{Easy! {2 [3 4]}} + +--- t dump: +a: Easy! +b: + c: 2 + d: [3, 4] + + +--- m: +map[a:Easy! b:map[c:2 d:[3 4]]] + +--- m dump: +a: Easy! +b: + c: 2 + d: + - 3 + - 4 +``` + diff --git a/vendor/gopkg.in/yaml.v2/apic.go b/vendor/gopkg.in/yaml.v2/apic.go new file mode 100644 index 00000000..acf71402 --- /dev/null +++ b/vendor/gopkg.in/yaml.v2/apic.go @@ -0,0 +1,744 @@ +package yaml + +import ( + "io" +) + +func yaml_insert_token(parser *yaml_parser_t, pos int, token *yaml_token_t) { + //fmt.Println("yaml_insert_token", "pos:", pos, "typ:", token.typ, "head:", parser.tokens_head, "len:", len(parser.tokens)) + + // Check if we can move the queue at the beginning of the buffer. + if parser.tokens_head > 0 && len(parser.tokens) == cap(parser.tokens) { + if parser.tokens_head != len(parser.tokens) { + copy(parser.tokens, parser.tokens[parser.tokens_head:]) + } + parser.tokens = parser.tokens[:len(parser.tokens)-parser.tokens_head] + parser.tokens_head = 0 + } + parser.tokens = append(parser.tokens, *token) + if pos < 0 { + return + } + copy(parser.tokens[parser.tokens_head+pos+1:], parser.tokens[parser.tokens_head+pos:]) + parser.tokens[parser.tokens_head+pos] = *token +} + +// Create a new parser object. +func yaml_parser_initialize(parser *yaml_parser_t) bool { + *parser = yaml_parser_t{ + raw_buffer: make([]byte, 0, input_raw_buffer_size), + buffer: make([]byte, 0, input_buffer_size), + } + return true +} + +// Destroy a parser object. +func yaml_parser_delete(parser *yaml_parser_t) { + *parser = yaml_parser_t{} +} + +// String read handler. +func yaml_string_read_handler(parser *yaml_parser_t, buffer []byte) (n int, err error) { + if parser.input_pos == len(parser.input) { + return 0, io.EOF + } + n = copy(buffer, parser.input[parser.input_pos:]) + parser.input_pos += n + return n, nil +} + +// Reader read handler. +func yaml_reader_read_handler(parser *yaml_parser_t, buffer []byte) (n int, err error) { + return parser.input_reader.Read(buffer) +} + +// Set a string input. +func yaml_parser_set_input_string(parser *yaml_parser_t, input []byte) { + if parser.read_handler != nil { + panic("must set the input source only once") + } + parser.read_handler = yaml_string_read_handler + parser.input = input + parser.input_pos = 0 +} + +// Set a file input. +func yaml_parser_set_input_reader(parser *yaml_parser_t, r io.Reader) { + if parser.read_handler != nil { + panic("must set the input source only once") + } + parser.read_handler = yaml_reader_read_handler + parser.input_reader = r +} + +// Set the source encoding. +func yaml_parser_set_encoding(parser *yaml_parser_t, encoding yaml_encoding_t) { + if parser.encoding != yaml_ANY_ENCODING { + panic("must set the encoding only once") + } + parser.encoding = encoding +} + +var disableLineWrapping = false + +// Create a new emitter object. +func yaml_emitter_initialize(emitter *yaml_emitter_t) { + *emitter = yaml_emitter_t{ + buffer: make([]byte, output_buffer_size), + raw_buffer: make([]byte, 0, output_raw_buffer_size), + states: make([]yaml_emitter_state_t, 0, initial_stack_size), + events: make([]yaml_event_t, 0, initial_queue_size), + } + if disableLineWrapping { + emitter.best_width = -1 + } +} + +// Destroy an emitter object. +func yaml_emitter_delete(emitter *yaml_emitter_t) { + *emitter = yaml_emitter_t{} +} + +// String write handler. +func yaml_string_write_handler(emitter *yaml_emitter_t, buffer []byte) error { + *emitter.output_buffer = append(*emitter.output_buffer, buffer...) + return nil +} + +// yaml_writer_write_handler uses emitter.output_writer to write the +// emitted text. +func yaml_writer_write_handler(emitter *yaml_emitter_t, buffer []byte) error { + _, err := emitter.output_writer.Write(buffer) + return err +} + +// Set a string output. +func yaml_emitter_set_output_string(emitter *yaml_emitter_t, output_buffer *[]byte) { + if emitter.write_handler != nil { + panic("must set the output target only once") + } + emitter.write_handler = yaml_string_write_handler + emitter.output_buffer = output_buffer +} + +// Set a file output. +func yaml_emitter_set_output_writer(emitter *yaml_emitter_t, w io.Writer) { + if emitter.write_handler != nil { + panic("must set the output target only once") + } + emitter.write_handler = yaml_writer_write_handler + emitter.output_writer = w +} + +// Set the output encoding. +func yaml_emitter_set_encoding(emitter *yaml_emitter_t, encoding yaml_encoding_t) { + if emitter.encoding != yaml_ANY_ENCODING { + panic("must set the output encoding only once") + } + emitter.encoding = encoding +} + +// Set the canonical output style. +func yaml_emitter_set_canonical(emitter *yaml_emitter_t, canonical bool) { + emitter.canonical = canonical +} + +//// Set the indentation increment. +func yaml_emitter_set_indent(emitter *yaml_emitter_t, indent int) { + if indent < 2 || indent > 9 { + indent = 2 + } + emitter.best_indent = indent +} + +// Set the preferred line width. +func yaml_emitter_set_width(emitter *yaml_emitter_t, width int) { + if width < 0 { + width = -1 + } + emitter.best_width = width +} + +// Set if unescaped non-ASCII characters are allowed. +func yaml_emitter_set_unicode(emitter *yaml_emitter_t, unicode bool) { + emitter.unicode = unicode +} + +// Set the preferred line break character. +func yaml_emitter_set_break(emitter *yaml_emitter_t, line_break yaml_break_t) { + emitter.line_break = line_break +} + +///* +// * Destroy a token object. +// */ +// +//YAML_DECLARE(void) +//yaml_token_delete(yaml_token_t *token) +//{ +// assert(token); // Non-NULL token object expected. +// +// switch (token.type) +// { +// case YAML_TAG_DIRECTIVE_TOKEN: +// yaml_free(token.data.tag_directive.handle); +// yaml_free(token.data.tag_directive.prefix); +// break; +// +// case YAML_ALIAS_TOKEN: +// yaml_free(token.data.alias.value); +// break; +// +// case YAML_ANCHOR_TOKEN: +// yaml_free(token.data.anchor.value); +// break; +// +// case YAML_TAG_TOKEN: +// yaml_free(token.data.tag.handle); +// yaml_free(token.data.tag.suffix); +// break; +// +// case YAML_SCALAR_TOKEN: +// yaml_free(token.data.scalar.value); +// break; +// +// default: +// break; +// } +// +// memset(token, 0, sizeof(yaml_token_t)); +//} +// +///* +// * Check if a string is a valid UTF-8 sequence. +// * +// * Check 'reader.c' for more details on UTF-8 encoding. +// */ +// +//static int +//yaml_check_utf8(yaml_char_t *start, size_t length) +//{ +// yaml_char_t *end = start+length; +// yaml_char_t *pointer = start; +// +// while (pointer < end) { +// unsigned char octet; +// unsigned int width; +// unsigned int value; +// size_t k; +// +// octet = pointer[0]; +// width = (octet & 0x80) == 0x00 ? 1 : +// (octet & 0xE0) == 0xC0 ? 2 : +// (octet & 0xF0) == 0xE0 ? 3 : +// (octet & 0xF8) == 0xF0 ? 4 : 0; +// value = (octet & 0x80) == 0x00 ? octet & 0x7F : +// (octet & 0xE0) == 0xC0 ? octet & 0x1F : +// (octet & 0xF0) == 0xE0 ? octet & 0x0F : +// (octet & 0xF8) == 0xF0 ? octet & 0x07 : 0; +// if (!width) return 0; +// if (pointer+width > end) return 0; +// for (k = 1; k < width; k ++) { +// octet = pointer[k]; +// if ((octet & 0xC0) != 0x80) return 0; +// value = (value << 6) + (octet & 0x3F); +// } +// if (!((width == 1) || +// (width == 2 && value >= 0x80) || +// (width == 3 && value >= 0x800) || +// (width == 4 && value >= 0x10000))) return 0; +// +// pointer += width; +// } +// +// return 1; +//} +// + +// Create STREAM-START. +func yaml_stream_start_event_initialize(event *yaml_event_t, encoding yaml_encoding_t) { + *event = yaml_event_t{ + typ: yaml_STREAM_START_EVENT, + encoding: encoding, + } +} + +// Create STREAM-END. +func yaml_stream_end_event_initialize(event *yaml_event_t) { + *event = yaml_event_t{ + typ: yaml_STREAM_END_EVENT, + } +} + +// Create DOCUMENT-START. +func yaml_document_start_event_initialize( + event *yaml_event_t, + version_directive *yaml_version_directive_t, + tag_directives []yaml_tag_directive_t, + implicit bool, +) { + *event = yaml_event_t{ + typ: yaml_DOCUMENT_START_EVENT, + version_directive: version_directive, + tag_directives: tag_directives, + implicit: implicit, + } +} + +// Create DOCUMENT-END. +func yaml_document_end_event_initialize(event *yaml_event_t, implicit bool) { + *event = yaml_event_t{ + typ: yaml_DOCUMENT_END_EVENT, + implicit: implicit, + } +} + +///* +// * Create ALIAS. +// */ +// +//YAML_DECLARE(int) +//yaml_alias_event_initialize(event *yaml_event_t, anchor *yaml_char_t) +//{ +// mark yaml_mark_t = { 0, 0, 0 } +// anchor_copy *yaml_char_t = NULL +// +// assert(event) // Non-NULL event object is expected. +// assert(anchor) // Non-NULL anchor is expected. +// +// if (!yaml_check_utf8(anchor, strlen((char *)anchor))) return 0 +// +// anchor_copy = yaml_strdup(anchor) +// if (!anchor_copy) +// return 0 +// +// ALIAS_EVENT_INIT(*event, anchor_copy, mark, mark) +// +// return 1 +//} + +// Create SCALAR. +func yaml_scalar_event_initialize(event *yaml_event_t, anchor, tag, value []byte, plain_implicit, quoted_implicit bool, style yaml_scalar_style_t) bool { + *event = yaml_event_t{ + typ: yaml_SCALAR_EVENT, + anchor: anchor, + tag: tag, + value: value, + implicit: plain_implicit, + quoted_implicit: quoted_implicit, + style: yaml_style_t(style), + } + return true +} + +// Create SEQUENCE-START. +func yaml_sequence_start_event_initialize(event *yaml_event_t, anchor, tag []byte, implicit bool, style yaml_sequence_style_t) bool { + *event = yaml_event_t{ + typ: yaml_SEQUENCE_START_EVENT, + anchor: anchor, + tag: tag, + implicit: implicit, + style: yaml_style_t(style), + } + return true +} + +// Create SEQUENCE-END. +func yaml_sequence_end_event_initialize(event *yaml_event_t) bool { + *event = yaml_event_t{ + typ: yaml_SEQUENCE_END_EVENT, + } + return true +} + +// Create MAPPING-START. +func yaml_mapping_start_event_initialize(event *yaml_event_t, anchor, tag []byte, implicit bool, style yaml_mapping_style_t) { + *event = yaml_event_t{ + typ: yaml_MAPPING_START_EVENT, + anchor: anchor, + tag: tag, + implicit: implicit, + style: yaml_style_t(style), + } +} + +// Create MAPPING-END. +func yaml_mapping_end_event_initialize(event *yaml_event_t) { + *event = yaml_event_t{ + typ: yaml_MAPPING_END_EVENT, + } +} + +// Destroy an event object. +func yaml_event_delete(event *yaml_event_t) { + *event = yaml_event_t{} +} + +///* +// * Create a document object. +// */ +// +//YAML_DECLARE(int) +//yaml_document_initialize(document *yaml_document_t, +// version_directive *yaml_version_directive_t, +// tag_directives_start *yaml_tag_directive_t, +// tag_directives_end *yaml_tag_directive_t, +// start_implicit int, end_implicit int) +//{ +// struct { +// error yaml_error_type_t +// } context +// struct { +// start *yaml_node_t +// end *yaml_node_t +// top *yaml_node_t +// } nodes = { NULL, NULL, NULL } +// version_directive_copy *yaml_version_directive_t = NULL +// struct { +// start *yaml_tag_directive_t +// end *yaml_tag_directive_t +// top *yaml_tag_directive_t +// } tag_directives_copy = { NULL, NULL, NULL } +// value yaml_tag_directive_t = { NULL, NULL } +// mark yaml_mark_t = { 0, 0, 0 } +// +// assert(document) // Non-NULL document object is expected. +// assert((tag_directives_start && tag_directives_end) || +// (tag_directives_start == tag_directives_end)) +// // Valid tag directives are expected. +// +// if (!STACK_INIT(&context, nodes, INITIAL_STACK_SIZE)) goto error +// +// if (version_directive) { +// version_directive_copy = yaml_malloc(sizeof(yaml_version_directive_t)) +// if (!version_directive_copy) goto error +// version_directive_copy.major = version_directive.major +// version_directive_copy.minor = version_directive.minor +// } +// +// if (tag_directives_start != tag_directives_end) { +// tag_directive *yaml_tag_directive_t +// if (!STACK_INIT(&context, tag_directives_copy, INITIAL_STACK_SIZE)) +// goto error +// for (tag_directive = tag_directives_start +// tag_directive != tag_directives_end; tag_directive ++) { +// assert(tag_directive.handle) +// assert(tag_directive.prefix) +// if (!yaml_check_utf8(tag_directive.handle, +// strlen((char *)tag_directive.handle))) +// goto error +// if (!yaml_check_utf8(tag_directive.prefix, +// strlen((char *)tag_directive.prefix))) +// goto error +// value.handle = yaml_strdup(tag_directive.handle) +// value.prefix = yaml_strdup(tag_directive.prefix) +// if (!value.handle || !value.prefix) goto error +// if (!PUSH(&context, tag_directives_copy, value)) +// goto error +// value.handle = NULL +// value.prefix = NULL +// } +// } +// +// DOCUMENT_INIT(*document, nodes.start, nodes.end, version_directive_copy, +// tag_directives_copy.start, tag_directives_copy.top, +// start_implicit, end_implicit, mark, mark) +// +// return 1 +// +//error: +// STACK_DEL(&context, nodes) +// yaml_free(version_directive_copy) +// while (!STACK_EMPTY(&context, tag_directives_copy)) { +// value yaml_tag_directive_t = POP(&context, tag_directives_copy) +// yaml_free(value.handle) +// yaml_free(value.prefix) +// } +// STACK_DEL(&context, tag_directives_copy) +// yaml_free(value.handle) +// yaml_free(value.prefix) +// +// return 0 +//} +// +///* +// * Destroy a document object. +// */ +// +//YAML_DECLARE(void) +//yaml_document_delete(document *yaml_document_t) +//{ +// struct { +// error yaml_error_type_t +// } context +// tag_directive *yaml_tag_directive_t +// +// context.error = YAML_NO_ERROR // Eliminate a compiler warning. +// +// assert(document) // Non-NULL document object is expected. +// +// while (!STACK_EMPTY(&context, document.nodes)) { +// node yaml_node_t = POP(&context, document.nodes) +// yaml_free(node.tag) +// switch (node.type) { +// case YAML_SCALAR_NODE: +// yaml_free(node.data.scalar.value) +// break +// case YAML_SEQUENCE_NODE: +// STACK_DEL(&context, node.data.sequence.items) +// break +// case YAML_MAPPING_NODE: +// STACK_DEL(&context, node.data.mapping.pairs) +// break +// default: +// assert(0) // Should not happen. +// } +// } +// STACK_DEL(&context, document.nodes) +// +// yaml_free(document.version_directive) +// for (tag_directive = document.tag_directives.start +// tag_directive != document.tag_directives.end +// tag_directive++) { +// yaml_free(tag_directive.handle) +// yaml_free(tag_directive.prefix) +// } +// yaml_free(document.tag_directives.start) +// +// memset(document, 0, sizeof(yaml_document_t)) +//} +// +///** +// * Get a document node. +// */ +// +//YAML_DECLARE(yaml_node_t *) +//yaml_document_get_node(document *yaml_document_t, index int) +//{ +// assert(document) // Non-NULL document object is expected. +// +// if (index > 0 && document.nodes.start + index <= document.nodes.top) { +// return document.nodes.start + index - 1 +// } +// return NULL +//} +// +///** +// * Get the root object. +// */ +// +//YAML_DECLARE(yaml_node_t *) +//yaml_document_get_root_node(document *yaml_document_t) +//{ +// assert(document) // Non-NULL document object is expected. +// +// if (document.nodes.top != document.nodes.start) { +// return document.nodes.start +// } +// return NULL +//} +// +///* +// * Add a scalar node to a document. +// */ +// +//YAML_DECLARE(int) +//yaml_document_add_scalar(document *yaml_document_t, +// tag *yaml_char_t, value *yaml_char_t, length int, +// style yaml_scalar_style_t) +//{ +// struct { +// error yaml_error_type_t +// } context +// mark yaml_mark_t = { 0, 0, 0 } +// tag_copy *yaml_char_t = NULL +// value_copy *yaml_char_t = NULL +// node yaml_node_t +// +// assert(document) // Non-NULL document object is expected. +// assert(value) // Non-NULL value is expected. +// +// if (!tag) { +// tag = (yaml_char_t *)YAML_DEFAULT_SCALAR_TAG +// } +// +// if (!yaml_check_utf8(tag, strlen((char *)tag))) goto error +// tag_copy = yaml_strdup(tag) +// if (!tag_copy) goto error +// +// if (length < 0) { +// length = strlen((char *)value) +// } +// +// if (!yaml_check_utf8(value, length)) goto error +// value_copy = yaml_malloc(length+1) +// if (!value_copy) goto error +// memcpy(value_copy, value, length) +// value_copy[length] = '\0' +// +// SCALAR_NODE_INIT(node, tag_copy, value_copy, length, style, mark, mark) +// if (!PUSH(&context, document.nodes, node)) goto error +// +// return document.nodes.top - document.nodes.start +// +//error: +// yaml_free(tag_copy) +// yaml_free(value_copy) +// +// return 0 +//} +// +///* +// * Add a sequence node to a document. +// */ +// +//YAML_DECLARE(int) +//yaml_document_add_sequence(document *yaml_document_t, +// tag *yaml_char_t, style yaml_sequence_style_t) +//{ +// struct { +// error yaml_error_type_t +// } context +// mark yaml_mark_t = { 0, 0, 0 } +// tag_copy *yaml_char_t = NULL +// struct { +// start *yaml_node_item_t +// end *yaml_node_item_t +// top *yaml_node_item_t +// } items = { NULL, NULL, NULL } +// node yaml_node_t +// +// assert(document) // Non-NULL document object is expected. +// +// if (!tag) { +// tag = (yaml_char_t *)YAML_DEFAULT_SEQUENCE_TAG +// } +// +// if (!yaml_check_utf8(tag, strlen((char *)tag))) goto error +// tag_copy = yaml_strdup(tag) +// if (!tag_copy) goto error +// +// if (!STACK_INIT(&context, items, INITIAL_STACK_SIZE)) goto error +// +// SEQUENCE_NODE_INIT(node, tag_copy, items.start, items.end, +// style, mark, mark) +// if (!PUSH(&context, document.nodes, node)) goto error +// +// return document.nodes.top - document.nodes.start +// +//error: +// STACK_DEL(&context, items) +// yaml_free(tag_copy) +// +// return 0 +//} +// +///* +// * Add a mapping node to a document. +// */ +// +//YAML_DECLARE(int) +//yaml_document_add_mapping(document *yaml_document_t, +// tag *yaml_char_t, style yaml_mapping_style_t) +//{ +// struct { +// error yaml_error_type_t +// } context +// mark yaml_mark_t = { 0, 0, 0 } +// tag_copy *yaml_char_t = NULL +// struct { +// start *yaml_node_pair_t +// end *yaml_node_pair_t +// top *yaml_node_pair_t +// } pairs = { NULL, NULL, NULL } +// node yaml_node_t +// +// assert(document) // Non-NULL document object is expected. +// +// if (!tag) { +// tag = (yaml_char_t *)YAML_DEFAULT_MAPPING_TAG +// } +// +// if (!yaml_check_utf8(tag, strlen((char *)tag))) goto error +// tag_copy = yaml_strdup(tag) +// if (!tag_copy) goto error +// +// if (!STACK_INIT(&context, pairs, INITIAL_STACK_SIZE)) goto error +// +// MAPPING_NODE_INIT(node, tag_copy, pairs.start, pairs.end, +// style, mark, mark) +// if (!PUSH(&context, document.nodes, node)) goto error +// +// return document.nodes.top - document.nodes.start +// +//error: +// STACK_DEL(&context, pairs) +// yaml_free(tag_copy) +// +// return 0 +//} +// +///* +// * Append an item to a sequence node. +// */ +// +//YAML_DECLARE(int) +//yaml_document_append_sequence_item(document *yaml_document_t, +// sequence int, item int) +//{ +// struct { +// error yaml_error_type_t +// } context +// +// assert(document) // Non-NULL document is required. +// assert(sequence > 0 +// && document.nodes.start + sequence <= document.nodes.top) +// // Valid sequence id is required. +// assert(document.nodes.start[sequence-1].type == YAML_SEQUENCE_NODE) +// // A sequence node is required. +// assert(item > 0 && document.nodes.start + item <= document.nodes.top) +// // Valid item id is required. +// +// if (!PUSH(&context, +// document.nodes.start[sequence-1].data.sequence.items, item)) +// return 0 +// +// return 1 +//} +// +///* +// * Append a pair of a key and a value to a mapping node. +// */ +// +//YAML_DECLARE(int) +//yaml_document_append_mapping_pair(document *yaml_document_t, +// mapping int, key int, value int) +//{ +// struct { +// error yaml_error_type_t +// } context +// +// pair yaml_node_pair_t +// +// assert(document) // Non-NULL document is required. +// assert(mapping > 0 +// && document.nodes.start + mapping <= document.nodes.top) +// // Valid mapping id is required. +// assert(document.nodes.start[mapping-1].type == YAML_MAPPING_NODE) +// // A mapping node is required. +// assert(key > 0 && document.nodes.start + key <= document.nodes.top) +// // Valid key id is required. +// assert(value > 0 && document.nodes.start + value <= document.nodes.top) +// // Valid value id is required. +// +// pair.key = key +// pair.value = value +// +// if (!PUSH(&context, +// document.nodes.start[mapping-1].data.mapping.pairs, pair)) +// return 0 +// +// return 1 +//} +// +// diff --git a/vendor/gopkg.in/yaml.v2/decode.go b/vendor/gopkg.in/yaml.v2/decode.go new file mode 100644 index 00000000..129bc2a9 --- /dev/null +++ b/vendor/gopkg.in/yaml.v2/decode.go @@ -0,0 +1,815 @@ +package yaml + +import ( + "encoding" + "encoding/base64" + "fmt" + "io" + "math" + "reflect" + "strconv" + "time" +) + +const ( + documentNode = 1 << iota + mappingNode + sequenceNode + scalarNode + aliasNode +) + +type node struct { + kind int + line, column int + tag string + // For an alias node, alias holds the resolved alias. + alias *node + value string + implicit bool + children []*node + anchors map[string]*node +} + +// ---------------------------------------------------------------------------- +// Parser, produces a node tree out of a libyaml event stream. + +type parser struct { + parser yaml_parser_t + event yaml_event_t + doc *node + doneInit bool +} + +func newParser(b []byte) *parser { + p := parser{} + if !yaml_parser_initialize(&p.parser) { + panic("failed to initialize YAML emitter") + } + if len(b) == 0 { + b = []byte{'\n'} + } + yaml_parser_set_input_string(&p.parser, b) + return &p +} + +func newParserFromReader(r io.Reader) *parser { + p := parser{} + if !yaml_parser_initialize(&p.parser) { + panic("failed to initialize YAML emitter") + } + yaml_parser_set_input_reader(&p.parser, r) + return &p +} + +func (p *parser) init() { + if p.doneInit { + return + } + p.expect(yaml_STREAM_START_EVENT) + p.doneInit = true +} + +func (p *parser) destroy() { + if p.event.typ != yaml_NO_EVENT { + yaml_event_delete(&p.event) + } + yaml_parser_delete(&p.parser) +} + +// expect consumes an event from the event stream and +// checks that it's of the expected type. +func (p *parser) expect(e yaml_event_type_t) { + if p.event.typ == yaml_NO_EVENT { + if !yaml_parser_parse(&p.parser, &p.event) { + p.fail() + } + } + if p.event.typ == yaml_STREAM_END_EVENT { + failf("attempted to go past the end of stream; corrupted value?") + } + if p.event.typ != e { + p.parser.problem = fmt.Sprintf("expected %s event but got %s", e, p.event.typ) + p.fail() + } + yaml_event_delete(&p.event) + p.event.typ = yaml_NO_EVENT +} + +// peek peeks at the next event in the event stream, +// puts the results into p.event and returns the event type. +func (p *parser) peek() yaml_event_type_t { + if p.event.typ != yaml_NO_EVENT { + return p.event.typ + } + if !yaml_parser_parse(&p.parser, &p.event) { + p.fail() + } + return p.event.typ +} + +func (p *parser) fail() { + var where string + var line int + if p.parser.problem_mark.line != 0 { + line = p.parser.problem_mark.line + // Scanner errors don't iterate line before returning error + if p.parser.error == yaml_SCANNER_ERROR { + line++ + } + } else if p.parser.context_mark.line != 0 { + line = p.parser.context_mark.line + } + if line != 0 { + where = "line " + strconv.Itoa(line) + ": " + } + var msg string + if len(p.parser.problem) > 0 { + msg = p.parser.problem + } else { + msg = "unknown problem parsing YAML content" + } + failf("%s%s", where, msg) +} + +func (p *parser) anchor(n *node, anchor []byte) { + if anchor != nil { + p.doc.anchors[string(anchor)] = n + } +} + +func (p *parser) parse() *node { + p.init() + switch p.peek() { + case yaml_SCALAR_EVENT: + return p.scalar() + case yaml_ALIAS_EVENT: + return p.alias() + case yaml_MAPPING_START_EVENT: + return p.mapping() + case yaml_SEQUENCE_START_EVENT: + return p.sequence() + case yaml_DOCUMENT_START_EVENT: + return p.document() + case yaml_STREAM_END_EVENT: + // Happens when attempting to decode an empty buffer. + return nil + default: + panic("attempted to parse unknown event: " + p.event.typ.String()) + } +} + +func (p *parser) node(kind int) *node { + return &node{ + kind: kind, + line: p.event.start_mark.line, + column: p.event.start_mark.column, + } +} + +func (p *parser) document() *node { + n := p.node(documentNode) + n.anchors = make(map[string]*node) + p.doc = n + p.expect(yaml_DOCUMENT_START_EVENT) + n.children = append(n.children, p.parse()) + p.expect(yaml_DOCUMENT_END_EVENT) + return n +} + +func (p *parser) alias() *node { + n := p.node(aliasNode) + n.value = string(p.event.anchor) + n.alias = p.doc.anchors[n.value] + if n.alias == nil { + failf("unknown anchor '%s' referenced", n.value) + } + p.expect(yaml_ALIAS_EVENT) + return n +} + +func (p *parser) scalar() *node { + n := p.node(scalarNode) + n.value = string(p.event.value) + n.tag = string(p.event.tag) + n.implicit = p.event.implicit + p.anchor(n, p.event.anchor) + p.expect(yaml_SCALAR_EVENT) + return n +} + +func (p *parser) sequence() *node { + n := p.node(sequenceNode) + p.anchor(n, p.event.anchor) + p.expect(yaml_SEQUENCE_START_EVENT) + for p.peek() != yaml_SEQUENCE_END_EVENT { + n.children = append(n.children, p.parse()) + } + p.expect(yaml_SEQUENCE_END_EVENT) + return n +} + +func (p *parser) mapping() *node { + n := p.node(mappingNode) + p.anchor(n, p.event.anchor) + p.expect(yaml_MAPPING_START_EVENT) + for p.peek() != yaml_MAPPING_END_EVENT { + n.children = append(n.children, p.parse(), p.parse()) + } + p.expect(yaml_MAPPING_END_EVENT) + return n +} + +// ---------------------------------------------------------------------------- +// Decoder, unmarshals a node into a provided value. + +type decoder struct { + doc *node + aliases map[*node]bool + mapType reflect.Type + terrors []string + strict bool + + decodeCount int + aliasCount int + aliasDepth int +} + +var ( + mapItemType = reflect.TypeOf(MapItem{}) + durationType = reflect.TypeOf(time.Duration(0)) + defaultMapType = reflect.TypeOf(map[interface{}]interface{}{}) + ifaceType = defaultMapType.Elem() + timeType = reflect.TypeOf(time.Time{}) + ptrTimeType = reflect.TypeOf(&time.Time{}) +) + +func newDecoder(strict bool) *decoder { + d := &decoder{mapType: defaultMapType, strict: strict} + d.aliases = make(map[*node]bool) + return d +} + +func (d *decoder) terror(n *node, tag string, out reflect.Value) { + if n.tag != "" { + tag = n.tag + } + value := n.value + if tag != yaml_SEQ_TAG && tag != yaml_MAP_TAG { + if len(value) > 10 { + value = " `" + value[:7] + "...`" + } else { + value = " `" + value + "`" + } + } + d.terrors = append(d.terrors, fmt.Sprintf("line %d: cannot unmarshal %s%s into %s", n.line+1, shortTag(tag), value, out.Type())) +} + +func (d *decoder) callUnmarshaler(n *node, u Unmarshaler) (good bool) { + terrlen := len(d.terrors) + err := u.UnmarshalYAML(func(v interface{}) (err error) { + defer handleErr(&err) + d.unmarshal(n, reflect.ValueOf(v)) + if len(d.terrors) > terrlen { + issues := d.terrors[terrlen:] + d.terrors = d.terrors[:terrlen] + return &TypeError{issues} + } + return nil + }) + if e, ok := err.(*TypeError); ok { + d.terrors = append(d.terrors, e.Errors...) + return false + } + if err != nil { + fail(err) + } + return true +} + +// d.prepare initializes and dereferences pointers and calls UnmarshalYAML +// if a value is found to implement it. +// It returns the initialized and dereferenced out value, whether +// unmarshalling was already done by UnmarshalYAML, and if so whether +// its types unmarshalled appropriately. +// +// If n holds a null value, prepare returns before doing anything. +func (d *decoder) prepare(n *node, out reflect.Value) (newout reflect.Value, unmarshaled, good bool) { + if n.tag == yaml_NULL_TAG || n.kind == scalarNode && n.tag == "" && (n.value == "null" || n.value == "~" || n.value == "" && n.implicit) { + return out, false, false + } + again := true + for again { + again = false + if out.Kind() == reflect.Ptr { + if out.IsNil() { + out.Set(reflect.New(out.Type().Elem())) + } + out = out.Elem() + again = true + } + if out.CanAddr() { + if u, ok := out.Addr().Interface().(Unmarshaler); ok { + good = d.callUnmarshaler(n, u) + return out, true, good + } + } + } + return out, false, false +} + +const ( + // 400,000 decode operations is ~500kb of dense object declarations, or + // ~5kb of dense object declarations with 10000% alias expansion + alias_ratio_range_low = 400000 + + // 4,000,000 decode operations is ~5MB of dense object declarations, or + // ~4.5MB of dense object declarations with 10% alias expansion + alias_ratio_range_high = 4000000 + + // alias_ratio_range is the range over which we scale allowed alias ratios + alias_ratio_range = float64(alias_ratio_range_high - alias_ratio_range_low) +) + +func allowedAliasRatio(decodeCount int) float64 { + switch { + case decodeCount <= alias_ratio_range_low: + // allow 99% to come from alias expansion for small-to-medium documents + return 0.99 + case decodeCount >= alias_ratio_range_high: + // allow 10% to come from alias expansion for very large documents + return 0.10 + default: + // scale smoothly from 99% down to 10% over the range. + // this maps to 396,000 - 400,000 allowed alias-driven decodes over the range. + // 400,000 decode operations is ~100MB of allocations in worst-case scenarios (single-item maps). + return 0.99 - 0.89*(float64(decodeCount-alias_ratio_range_low)/alias_ratio_range) + } +} + +func (d *decoder) unmarshal(n *node, out reflect.Value) (good bool) { + d.decodeCount++ + if d.aliasDepth > 0 { + d.aliasCount++ + } + if d.aliasCount > 100 && d.decodeCount > 1000 && float64(d.aliasCount)/float64(d.decodeCount) > allowedAliasRatio(d.decodeCount) { + failf("document contains excessive aliasing") + } + switch n.kind { + case documentNode: + return d.document(n, out) + case aliasNode: + return d.alias(n, out) + } + out, unmarshaled, good := d.prepare(n, out) + if unmarshaled { + return good + } + switch n.kind { + case scalarNode: + good = d.scalar(n, out) + case mappingNode: + good = d.mapping(n, out) + case sequenceNode: + good = d.sequence(n, out) + default: + panic("internal error: unknown node kind: " + strconv.Itoa(n.kind)) + } + return good +} + +func (d *decoder) document(n *node, out reflect.Value) (good bool) { + if len(n.children) == 1 { + d.doc = n + d.unmarshal(n.children[0], out) + return true + } + return false +} + +func (d *decoder) alias(n *node, out reflect.Value) (good bool) { + if d.aliases[n] { + // TODO this could actually be allowed in some circumstances. + failf("anchor '%s' value contains itself", n.value) + } + d.aliases[n] = true + d.aliasDepth++ + good = d.unmarshal(n.alias, out) + d.aliasDepth-- + delete(d.aliases, n) + return good +} + +var zeroValue reflect.Value + +func resetMap(out reflect.Value) { + for _, k := range out.MapKeys() { + out.SetMapIndex(k, zeroValue) + } +} + +func (d *decoder) scalar(n *node, out reflect.Value) bool { + var tag string + var resolved interface{} + if n.tag == "" && !n.implicit { + tag = yaml_STR_TAG + resolved = n.value + } else { + tag, resolved = resolve(n.tag, n.value) + if tag == yaml_BINARY_TAG { + data, err := base64.StdEncoding.DecodeString(resolved.(string)) + if err != nil { + failf("!!binary value contains invalid base64 data") + } + resolved = string(data) + } + } + if resolved == nil { + if out.Kind() == reflect.Map && !out.CanAddr() { + resetMap(out) + } else { + out.Set(reflect.Zero(out.Type())) + } + return true + } + if resolvedv := reflect.ValueOf(resolved); out.Type() == resolvedv.Type() { + // We've resolved to exactly the type we want, so use that. + out.Set(resolvedv) + return true + } + // Perhaps we can use the value as a TextUnmarshaler to + // set its value. + if out.CanAddr() { + u, ok := out.Addr().Interface().(encoding.TextUnmarshaler) + if ok { + var text []byte + if tag == yaml_BINARY_TAG { + text = []byte(resolved.(string)) + } else { + // We let any value be unmarshaled into TextUnmarshaler. + // That might be more lax than we'd like, but the + // TextUnmarshaler itself should bowl out any dubious values. + text = []byte(n.value) + } + err := u.UnmarshalText(text) + if err != nil { + fail(err) + } + return true + } + } + switch out.Kind() { + case reflect.String: + if tag == yaml_BINARY_TAG { + out.SetString(resolved.(string)) + return true + } + if resolved != nil { + out.SetString(n.value) + return true + } + case reflect.Interface: + if resolved == nil { + out.Set(reflect.Zero(out.Type())) + } else if tag == yaml_TIMESTAMP_TAG { + // It looks like a timestamp but for backward compatibility + // reasons we set it as a string, so that code that unmarshals + // timestamp-like values into interface{} will continue to + // see a string and not a time.Time. + // TODO(v3) Drop this. + out.Set(reflect.ValueOf(n.value)) + } else { + out.Set(reflect.ValueOf(resolved)) + } + return true + case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64: + switch resolved := resolved.(type) { + case int: + if !out.OverflowInt(int64(resolved)) { + out.SetInt(int64(resolved)) + return true + } + case int64: + if !out.OverflowInt(resolved) { + out.SetInt(resolved) + return true + } + case uint64: + if resolved <= math.MaxInt64 && !out.OverflowInt(int64(resolved)) { + out.SetInt(int64(resolved)) + return true + } + case float64: + if resolved <= math.MaxInt64 && !out.OverflowInt(int64(resolved)) { + out.SetInt(int64(resolved)) + return true + } + case string: + if out.Type() == durationType { + d, err := time.ParseDuration(resolved) + if err == nil { + out.SetInt(int64(d)) + return true + } + } + } + case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64, reflect.Uintptr: + switch resolved := resolved.(type) { + case int: + if resolved >= 0 && !out.OverflowUint(uint64(resolved)) { + out.SetUint(uint64(resolved)) + return true + } + case int64: + if resolved >= 0 && !out.OverflowUint(uint64(resolved)) { + out.SetUint(uint64(resolved)) + return true + } + case uint64: + if !out.OverflowUint(uint64(resolved)) { + out.SetUint(uint64(resolved)) + return true + } + case float64: + if resolved <= math.MaxUint64 && !out.OverflowUint(uint64(resolved)) { + out.SetUint(uint64(resolved)) + return true + } + } + case reflect.Bool: + switch resolved := resolved.(type) { + case bool: + out.SetBool(resolved) + return true + } + case reflect.Float32, reflect.Float64: + switch resolved := resolved.(type) { + case int: + out.SetFloat(float64(resolved)) + return true + case int64: + out.SetFloat(float64(resolved)) + return true + case uint64: + out.SetFloat(float64(resolved)) + return true + case float64: + out.SetFloat(resolved) + return true + } + case reflect.Struct: + if resolvedv := reflect.ValueOf(resolved); out.Type() == resolvedv.Type() { + out.Set(resolvedv) + return true + } + case reflect.Ptr: + if out.Type().Elem() == reflect.TypeOf(resolved) { + // TODO DOes this make sense? When is out a Ptr except when decoding a nil value? + elem := reflect.New(out.Type().Elem()) + elem.Elem().Set(reflect.ValueOf(resolved)) + out.Set(elem) + return true + } + } + d.terror(n, tag, out) + return false +} + +func settableValueOf(i interface{}) reflect.Value { + v := reflect.ValueOf(i) + sv := reflect.New(v.Type()).Elem() + sv.Set(v) + return sv +} + +func (d *decoder) sequence(n *node, out reflect.Value) (good bool) { + l := len(n.children) + + var iface reflect.Value + switch out.Kind() { + case reflect.Slice: + out.Set(reflect.MakeSlice(out.Type(), l, l)) + case reflect.Array: + if l != out.Len() { + failf("invalid array: want %d elements but got %d", out.Len(), l) + } + case reflect.Interface: + // No type hints. Will have to use a generic sequence. + iface = out + out = settableValueOf(make([]interface{}, l)) + default: + d.terror(n, yaml_SEQ_TAG, out) + return false + } + et := out.Type().Elem() + + j := 0 + for i := 0; i < l; i++ { + e := reflect.New(et).Elem() + if ok := d.unmarshal(n.children[i], e); ok { + out.Index(j).Set(e) + j++ + } + } + if out.Kind() != reflect.Array { + out.Set(out.Slice(0, j)) + } + if iface.IsValid() { + iface.Set(out) + } + return true +} + +func (d *decoder) mapping(n *node, out reflect.Value) (good bool) { + switch out.Kind() { + case reflect.Struct: + return d.mappingStruct(n, out) + case reflect.Slice: + return d.mappingSlice(n, out) + case reflect.Map: + // okay + case reflect.Interface: + if d.mapType.Kind() == reflect.Map { + iface := out + out = reflect.MakeMap(d.mapType) + iface.Set(out) + } else { + slicev := reflect.New(d.mapType).Elem() + if !d.mappingSlice(n, slicev) { + return false + } + out.Set(slicev) + return true + } + default: + d.terror(n, yaml_MAP_TAG, out) + return false + } + outt := out.Type() + kt := outt.Key() + et := outt.Elem() + + mapType := d.mapType + if outt.Key() == ifaceType && outt.Elem() == ifaceType { + d.mapType = outt + } + + if out.IsNil() { + out.Set(reflect.MakeMap(outt)) + } + l := len(n.children) + for i := 0; i < l; i += 2 { + if isMerge(n.children[i]) { + d.merge(n.children[i+1], out) + continue + } + k := reflect.New(kt).Elem() + if d.unmarshal(n.children[i], k) { + kkind := k.Kind() + if kkind == reflect.Interface { + kkind = k.Elem().Kind() + } + if kkind == reflect.Map || kkind == reflect.Slice { + failf("invalid map key: %#v", k.Interface()) + } + e := reflect.New(et).Elem() + if d.unmarshal(n.children[i+1], e) { + d.setMapIndex(n.children[i+1], out, k, e) + } + } + } + d.mapType = mapType + return true +} + +func (d *decoder) setMapIndex(n *node, out, k, v reflect.Value) { + if d.strict && out.MapIndex(k) != zeroValue { + d.terrors = append(d.terrors, fmt.Sprintf("line %d: key %#v already set in map", n.line+1, k.Interface())) + return + } + out.SetMapIndex(k, v) +} + +func (d *decoder) mappingSlice(n *node, out reflect.Value) (good bool) { + outt := out.Type() + if outt.Elem() != mapItemType { + d.terror(n, yaml_MAP_TAG, out) + return false + } + + mapType := d.mapType + d.mapType = outt + + var slice []MapItem + var l = len(n.children) + for i := 0; i < l; i += 2 { + if isMerge(n.children[i]) { + d.merge(n.children[i+1], out) + continue + } + item := MapItem{} + k := reflect.ValueOf(&item.Key).Elem() + if d.unmarshal(n.children[i], k) { + v := reflect.ValueOf(&item.Value).Elem() + if d.unmarshal(n.children[i+1], v) { + slice = append(slice, item) + } + } + } + out.Set(reflect.ValueOf(slice)) + d.mapType = mapType + return true +} + +func (d *decoder) mappingStruct(n *node, out reflect.Value) (good bool) { + sinfo, err := getStructInfo(out.Type()) + if err != nil { + panic(err) + } + name := settableValueOf("") + l := len(n.children) + + var inlineMap reflect.Value + var elemType reflect.Type + if sinfo.InlineMap != -1 { + inlineMap = out.Field(sinfo.InlineMap) + inlineMap.Set(reflect.New(inlineMap.Type()).Elem()) + elemType = inlineMap.Type().Elem() + } + + var doneFields []bool + if d.strict { + doneFields = make([]bool, len(sinfo.FieldsList)) + } + for i := 0; i < l; i += 2 { + ni := n.children[i] + if isMerge(ni) { + d.merge(n.children[i+1], out) + continue + } + if !d.unmarshal(ni, name) { + continue + } + if info, ok := sinfo.FieldsMap[name.String()]; ok { + if d.strict { + if doneFields[info.Id] { + d.terrors = append(d.terrors, fmt.Sprintf("line %d: field %s already set in type %s", ni.line+1, name.String(), out.Type())) + continue + } + doneFields[info.Id] = true + } + var field reflect.Value + if info.Inline == nil { + field = out.Field(info.Num) + } else { + field = out.FieldByIndex(info.Inline) + } + d.unmarshal(n.children[i+1], field) + } else if sinfo.InlineMap != -1 { + if inlineMap.IsNil() { + inlineMap.Set(reflect.MakeMap(inlineMap.Type())) + } + value := reflect.New(elemType).Elem() + d.unmarshal(n.children[i+1], value) + d.setMapIndex(n.children[i+1], inlineMap, name, value) + } else if d.strict { + d.terrors = append(d.terrors, fmt.Sprintf("line %d: field %s not found in type %s", ni.line+1, name.String(), out.Type())) + } + } + return true +} + +func failWantMap() { + failf("map merge requires map or sequence of maps as the value") +} + +func (d *decoder) merge(n *node, out reflect.Value) { + switch n.kind { + case mappingNode: + d.unmarshal(n, out) + case aliasNode: + if n.alias != nil && n.alias.kind != mappingNode { + failWantMap() + } + d.unmarshal(n, out) + case sequenceNode: + // Step backwards as earlier nodes take precedence. + for i := len(n.children) - 1; i >= 0; i-- { + ni := n.children[i] + if ni.kind == aliasNode { + if ni.alias != nil && ni.alias.kind != mappingNode { + failWantMap() + } + } else if ni.kind != mappingNode { + failWantMap() + } + d.unmarshal(ni, out) + } + default: + failWantMap() + } +} + +func isMerge(n *node) bool { + return n.kind == scalarNode && n.value == "<<" && (n.implicit == true || n.tag == yaml_MERGE_TAG) +} diff --git a/vendor/gopkg.in/yaml.v2/emitterc.go b/vendor/gopkg.in/yaml.v2/emitterc.go new file mode 100644 index 00000000..a1c2cc52 --- /dev/null +++ b/vendor/gopkg.in/yaml.v2/emitterc.go @@ -0,0 +1,1685 @@ +package yaml + +import ( + "bytes" + "fmt" +) + +// Flush the buffer if needed. +func flush(emitter *yaml_emitter_t) bool { + if emitter.buffer_pos+5 >= len(emitter.buffer) { + return yaml_emitter_flush(emitter) + } + return true +} + +// Put a character to the output buffer. +func put(emitter *yaml_emitter_t, value byte) bool { + if emitter.buffer_pos+5 >= len(emitter.buffer) && !yaml_emitter_flush(emitter) { + return false + } + emitter.buffer[emitter.buffer_pos] = value + emitter.buffer_pos++ + emitter.column++ + return true +} + +// Put a line break to the output buffer. +func put_break(emitter *yaml_emitter_t) bool { + if emitter.buffer_pos+5 >= len(emitter.buffer) && !yaml_emitter_flush(emitter) { + return false + } + switch emitter.line_break { + case yaml_CR_BREAK: + emitter.buffer[emitter.buffer_pos] = '\r' + emitter.buffer_pos += 1 + case yaml_LN_BREAK: + emitter.buffer[emitter.buffer_pos] = '\n' + emitter.buffer_pos += 1 + case yaml_CRLN_BREAK: + emitter.buffer[emitter.buffer_pos+0] = '\r' + emitter.buffer[emitter.buffer_pos+1] = '\n' + emitter.buffer_pos += 2 + default: + panic("unknown line break setting") + } + emitter.column = 0 + emitter.line++ + return true +} + +// Copy a character from a string into buffer. +func write(emitter *yaml_emitter_t, s []byte, i *int) bool { + if emitter.buffer_pos+5 >= len(emitter.buffer) && !yaml_emitter_flush(emitter) { + return false + } + p := emitter.buffer_pos + w := width(s[*i]) + switch w { + case 4: + emitter.buffer[p+3] = s[*i+3] + fallthrough + case 3: + emitter.buffer[p+2] = s[*i+2] + fallthrough + case 2: + emitter.buffer[p+1] = s[*i+1] + fallthrough + case 1: + emitter.buffer[p+0] = s[*i+0] + default: + panic("unknown character width") + } + emitter.column++ + emitter.buffer_pos += w + *i += w + return true +} + +// Write a whole string into buffer. +func write_all(emitter *yaml_emitter_t, s []byte) bool { + for i := 0; i < len(s); { + if !write(emitter, s, &i) { + return false + } + } + return true +} + +// Copy a line break character from a string into buffer. +func write_break(emitter *yaml_emitter_t, s []byte, i *int) bool { + if s[*i] == '\n' { + if !put_break(emitter) { + return false + } + *i++ + } else { + if !write(emitter, s, i) { + return false + } + emitter.column = 0 + emitter.line++ + } + return true +} + +// Set an emitter error and return false. +func yaml_emitter_set_emitter_error(emitter *yaml_emitter_t, problem string) bool { + emitter.error = yaml_EMITTER_ERROR + emitter.problem = problem + return false +} + +// Emit an event. +func yaml_emitter_emit(emitter *yaml_emitter_t, event *yaml_event_t) bool { + emitter.events = append(emitter.events, *event) + for !yaml_emitter_need_more_events(emitter) { + event := &emitter.events[emitter.events_head] + if !yaml_emitter_analyze_event(emitter, event) { + return false + } + if !yaml_emitter_state_machine(emitter, event) { + return false + } + yaml_event_delete(event) + emitter.events_head++ + } + return true +} + +// Check if we need to accumulate more events before emitting. +// +// We accumulate extra +// - 1 event for DOCUMENT-START +// - 2 events for SEQUENCE-START +// - 3 events for MAPPING-START +// +func yaml_emitter_need_more_events(emitter *yaml_emitter_t) bool { + if emitter.events_head == len(emitter.events) { + return true + } + var accumulate int + switch emitter.events[emitter.events_head].typ { + case yaml_DOCUMENT_START_EVENT: + accumulate = 1 + break + case yaml_SEQUENCE_START_EVENT: + accumulate = 2 + break + case yaml_MAPPING_START_EVENT: + accumulate = 3 + break + default: + return false + } + if len(emitter.events)-emitter.events_head > accumulate { + return false + } + var level int + for i := emitter.events_head; i < len(emitter.events); i++ { + switch emitter.events[i].typ { + case yaml_STREAM_START_EVENT, yaml_DOCUMENT_START_EVENT, yaml_SEQUENCE_START_EVENT, yaml_MAPPING_START_EVENT: + level++ + case yaml_STREAM_END_EVENT, yaml_DOCUMENT_END_EVENT, yaml_SEQUENCE_END_EVENT, yaml_MAPPING_END_EVENT: + level-- + } + if level == 0 { + return false + } + } + return true +} + +// Append a directive to the directives stack. +func yaml_emitter_append_tag_directive(emitter *yaml_emitter_t, value *yaml_tag_directive_t, allow_duplicates bool) bool { + for i := 0; i < len(emitter.tag_directives); i++ { + if bytes.Equal(value.handle, emitter.tag_directives[i].handle) { + if allow_duplicates { + return true + } + return yaml_emitter_set_emitter_error(emitter, "duplicate %TAG directive") + } + } + + // [Go] Do we actually need to copy this given garbage collection + // and the lack of deallocating destructors? + tag_copy := yaml_tag_directive_t{ + handle: make([]byte, len(value.handle)), + prefix: make([]byte, len(value.prefix)), + } + copy(tag_copy.handle, value.handle) + copy(tag_copy.prefix, value.prefix) + emitter.tag_directives = append(emitter.tag_directives, tag_copy) + return true +} + +// Increase the indentation level. +func yaml_emitter_increase_indent(emitter *yaml_emitter_t, flow, indentless bool) bool { + emitter.indents = append(emitter.indents, emitter.indent) + if emitter.indent < 0 { + if flow { + emitter.indent = emitter.best_indent + } else { + emitter.indent = 0 + } + } else if !indentless { + emitter.indent += emitter.best_indent + } + return true +} + +// State dispatcher. +func yaml_emitter_state_machine(emitter *yaml_emitter_t, event *yaml_event_t) bool { + switch emitter.state { + default: + case yaml_EMIT_STREAM_START_STATE: + return yaml_emitter_emit_stream_start(emitter, event) + + case yaml_EMIT_FIRST_DOCUMENT_START_STATE: + return yaml_emitter_emit_document_start(emitter, event, true) + + case yaml_EMIT_DOCUMENT_START_STATE: + return yaml_emitter_emit_document_start(emitter, event, false) + + case yaml_EMIT_DOCUMENT_CONTENT_STATE: + return yaml_emitter_emit_document_content(emitter, event) + + case yaml_EMIT_DOCUMENT_END_STATE: + return yaml_emitter_emit_document_end(emitter, event) + + case yaml_EMIT_FLOW_SEQUENCE_FIRST_ITEM_STATE: + return yaml_emitter_emit_flow_sequence_item(emitter, event, true) + + case yaml_EMIT_FLOW_SEQUENCE_ITEM_STATE: + return yaml_emitter_emit_flow_sequence_item(emitter, event, false) + + case yaml_EMIT_FLOW_MAPPING_FIRST_KEY_STATE: + return yaml_emitter_emit_flow_mapping_key(emitter, event, true) + + case yaml_EMIT_FLOW_MAPPING_KEY_STATE: + return yaml_emitter_emit_flow_mapping_key(emitter, event, false) + + case yaml_EMIT_FLOW_MAPPING_SIMPLE_VALUE_STATE: + return yaml_emitter_emit_flow_mapping_value(emitter, event, true) + + case yaml_EMIT_FLOW_MAPPING_VALUE_STATE: + return yaml_emitter_emit_flow_mapping_value(emitter, event, false) + + case yaml_EMIT_BLOCK_SEQUENCE_FIRST_ITEM_STATE: + return yaml_emitter_emit_block_sequence_item(emitter, event, true) + + case yaml_EMIT_BLOCK_SEQUENCE_ITEM_STATE: + return yaml_emitter_emit_block_sequence_item(emitter, event, false) + + case yaml_EMIT_BLOCK_MAPPING_FIRST_KEY_STATE: + return yaml_emitter_emit_block_mapping_key(emitter, event, true) + + case yaml_EMIT_BLOCK_MAPPING_KEY_STATE: + return yaml_emitter_emit_block_mapping_key(emitter, event, false) + + case yaml_EMIT_BLOCK_MAPPING_SIMPLE_VALUE_STATE: + return yaml_emitter_emit_block_mapping_value(emitter, event, true) + + case yaml_EMIT_BLOCK_MAPPING_VALUE_STATE: + return yaml_emitter_emit_block_mapping_value(emitter, event, false) + + case yaml_EMIT_END_STATE: + return yaml_emitter_set_emitter_error(emitter, "expected nothing after STREAM-END") + } + panic("invalid emitter state") +} + +// Expect STREAM-START. +func yaml_emitter_emit_stream_start(emitter *yaml_emitter_t, event *yaml_event_t) bool { + if event.typ != yaml_STREAM_START_EVENT { + return yaml_emitter_set_emitter_error(emitter, "expected STREAM-START") + } + if emitter.encoding == yaml_ANY_ENCODING { + emitter.encoding = event.encoding + if emitter.encoding == yaml_ANY_ENCODING { + emitter.encoding = yaml_UTF8_ENCODING + } + } + if emitter.best_indent < 2 || emitter.best_indent > 9 { + emitter.best_indent = 2 + } + if emitter.best_width >= 0 && emitter.best_width <= emitter.best_indent*2 { + emitter.best_width = 80 + } + if emitter.best_width < 0 { + emitter.best_width = 1<<31 - 1 + } + if emitter.line_break == yaml_ANY_BREAK { + emitter.line_break = yaml_LN_BREAK + } + + emitter.indent = -1 + emitter.line = 0 + emitter.column = 0 + emitter.whitespace = true + emitter.indention = true + + if emitter.encoding != yaml_UTF8_ENCODING { + if !yaml_emitter_write_bom(emitter) { + return false + } + } + emitter.state = yaml_EMIT_FIRST_DOCUMENT_START_STATE + return true +} + +// Expect DOCUMENT-START or STREAM-END. +func yaml_emitter_emit_document_start(emitter *yaml_emitter_t, event *yaml_event_t, first bool) bool { + + if event.typ == yaml_DOCUMENT_START_EVENT { + + if event.version_directive != nil { + if !yaml_emitter_analyze_version_directive(emitter, event.version_directive) { + return false + } + } + + for i := 0; i < len(event.tag_directives); i++ { + tag_directive := &event.tag_directives[i] + if !yaml_emitter_analyze_tag_directive(emitter, tag_directive) { + return false + } + if !yaml_emitter_append_tag_directive(emitter, tag_directive, false) { + return false + } + } + + for i := 0; i < len(default_tag_directives); i++ { + tag_directive := &default_tag_directives[i] + if !yaml_emitter_append_tag_directive(emitter, tag_directive, true) { + return false + } + } + + implicit := event.implicit + if !first || emitter.canonical { + implicit = false + } + + if emitter.open_ended && (event.version_directive != nil || len(event.tag_directives) > 0) { + if !yaml_emitter_write_indicator(emitter, []byte("..."), true, false, false) { + return false + } + if !yaml_emitter_write_indent(emitter) { + return false + } + } + + if event.version_directive != nil { + implicit = false + if !yaml_emitter_write_indicator(emitter, []byte("%YAML"), true, false, false) { + return false + } + if !yaml_emitter_write_indicator(emitter, []byte("1.1"), true, false, false) { + return false + } + if !yaml_emitter_write_indent(emitter) { + return false + } + } + + if len(event.tag_directives) > 0 { + implicit = false + for i := 0; i < len(event.tag_directives); i++ { + tag_directive := &event.tag_directives[i] + if !yaml_emitter_write_indicator(emitter, []byte("%TAG"), true, false, false) { + return false + } + if !yaml_emitter_write_tag_handle(emitter, tag_directive.handle) { + return false + } + if !yaml_emitter_write_tag_content(emitter, tag_directive.prefix, true) { + return false + } + if !yaml_emitter_write_indent(emitter) { + return false + } + } + } + + if yaml_emitter_check_empty_document(emitter) { + implicit = false + } + if !implicit { + if !yaml_emitter_write_indent(emitter) { + return false + } + if !yaml_emitter_write_indicator(emitter, []byte("---"), true, false, false) { + return false + } + if emitter.canonical { + if !yaml_emitter_write_indent(emitter) { + return false + } + } + } + + emitter.state = yaml_EMIT_DOCUMENT_CONTENT_STATE + return true + } + + if event.typ == yaml_STREAM_END_EVENT { + if emitter.open_ended { + if !yaml_emitter_write_indicator(emitter, []byte("..."), true, false, false) { + return false + } + if !yaml_emitter_write_indent(emitter) { + return false + } + } + if !yaml_emitter_flush(emitter) { + return false + } + emitter.state = yaml_EMIT_END_STATE + return true + } + + return yaml_emitter_set_emitter_error(emitter, "expected DOCUMENT-START or STREAM-END") +} + +// Expect the root node. +func yaml_emitter_emit_document_content(emitter *yaml_emitter_t, event *yaml_event_t) bool { + emitter.states = append(emitter.states, yaml_EMIT_DOCUMENT_END_STATE) + return yaml_emitter_emit_node(emitter, event, true, false, false, false) +} + +// Expect DOCUMENT-END. +func yaml_emitter_emit_document_end(emitter *yaml_emitter_t, event *yaml_event_t) bool { + if event.typ != yaml_DOCUMENT_END_EVENT { + return yaml_emitter_set_emitter_error(emitter, "expected DOCUMENT-END") + } + if !yaml_emitter_write_indent(emitter) { + return false + } + if !event.implicit { + // [Go] Allocate the slice elsewhere. + if !yaml_emitter_write_indicator(emitter, []byte("..."), true, false, false) { + return false + } + if !yaml_emitter_write_indent(emitter) { + return false + } + } + if !yaml_emitter_flush(emitter) { + return false + } + emitter.state = yaml_EMIT_DOCUMENT_START_STATE + emitter.tag_directives = emitter.tag_directives[:0] + return true +} + +// Expect a flow item node. +func yaml_emitter_emit_flow_sequence_item(emitter *yaml_emitter_t, event *yaml_event_t, first bool) bool { + if first { + if !yaml_emitter_write_indicator(emitter, []byte{'['}, true, true, false) { + return false + } + if !yaml_emitter_increase_indent(emitter, true, false) { + return false + } + emitter.flow_level++ + } + + if event.typ == yaml_SEQUENCE_END_EVENT { + emitter.flow_level-- + emitter.indent = emitter.indents[len(emitter.indents)-1] + emitter.indents = emitter.indents[:len(emitter.indents)-1] + if emitter.canonical && !first { + if !yaml_emitter_write_indicator(emitter, []byte{','}, false, false, false) { + return false + } + if !yaml_emitter_write_indent(emitter) { + return false + } + } + if !yaml_emitter_write_indicator(emitter, []byte{']'}, false, false, false) { + return false + } + emitter.state = emitter.states[len(emitter.states)-1] + emitter.states = emitter.states[:len(emitter.states)-1] + + return true + } + + if !first { + if !yaml_emitter_write_indicator(emitter, []byte{','}, false, false, false) { + return false + } + } + + if emitter.canonical || emitter.column > emitter.best_width { + if !yaml_emitter_write_indent(emitter) { + return false + } + } + emitter.states = append(emitter.states, yaml_EMIT_FLOW_SEQUENCE_ITEM_STATE) + return yaml_emitter_emit_node(emitter, event, false, true, false, false) +} + +// Expect a flow key node. +func yaml_emitter_emit_flow_mapping_key(emitter *yaml_emitter_t, event *yaml_event_t, first bool) bool { + if first { + if !yaml_emitter_write_indicator(emitter, []byte{'{'}, true, true, false) { + return false + } + if !yaml_emitter_increase_indent(emitter, true, false) { + return false + } + emitter.flow_level++ + } + + if event.typ == yaml_MAPPING_END_EVENT { + emitter.flow_level-- + emitter.indent = emitter.indents[len(emitter.indents)-1] + emitter.indents = emitter.indents[:len(emitter.indents)-1] + if emitter.canonical && !first { + if !yaml_emitter_write_indicator(emitter, []byte{','}, false, false, false) { + return false + } + if !yaml_emitter_write_indent(emitter) { + return false + } + } + if !yaml_emitter_write_indicator(emitter, []byte{'}'}, false, false, false) { + return false + } + emitter.state = emitter.states[len(emitter.states)-1] + emitter.states = emitter.states[:len(emitter.states)-1] + return true + } + + if !first { + if !yaml_emitter_write_indicator(emitter, []byte{','}, false, false, false) { + return false + } + } + if emitter.canonical || emitter.column > emitter.best_width { + if !yaml_emitter_write_indent(emitter) { + return false + } + } + + if !emitter.canonical && yaml_emitter_check_simple_key(emitter) { + emitter.states = append(emitter.states, yaml_EMIT_FLOW_MAPPING_SIMPLE_VALUE_STATE) + return yaml_emitter_emit_node(emitter, event, false, false, true, true) + } + if !yaml_emitter_write_indicator(emitter, []byte{'?'}, true, false, false) { + return false + } + emitter.states = append(emitter.states, yaml_EMIT_FLOW_MAPPING_VALUE_STATE) + return yaml_emitter_emit_node(emitter, event, false, false, true, false) +} + +// Expect a flow value node. +func yaml_emitter_emit_flow_mapping_value(emitter *yaml_emitter_t, event *yaml_event_t, simple bool) bool { + if simple { + if !yaml_emitter_write_indicator(emitter, []byte{':'}, false, false, false) { + return false + } + } else { + if emitter.canonical || emitter.column > emitter.best_width { + if !yaml_emitter_write_indent(emitter) { + return false + } + } + if !yaml_emitter_write_indicator(emitter, []byte{':'}, true, false, false) { + return false + } + } + emitter.states = append(emitter.states, yaml_EMIT_FLOW_MAPPING_KEY_STATE) + return yaml_emitter_emit_node(emitter, event, false, false, true, false) +} + +// Expect a block item node. +func yaml_emitter_emit_block_sequence_item(emitter *yaml_emitter_t, event *yaml_event_t, first bool) bool { + if first { + if !yaml_emitter_increase_indent(emitter, false, emitter.mapping_context && !emitter.indention) { + return false + } + } + if event.typ == yaml_SEQUENCE_END_EVENT { + emitter.indent = emitter.indents[len(emitter.indents)-1] + emitter.indents = emitter.indents[:len(emitter.indents)-1] + emitter.state = emitter.states[len(emitter.states)-1] + emitter.states = emitter.states[:len(emitter.states)-1] + return true + } + if !yaml_emitter_write_indent(emitter) { + return false + } + if !yaml_emitter_write_indicator(emitter, []byte{'-'}, true, false, true) { + return false + } + emitter.states = append(emitter.states, yaml_EMIT_BLOCK_SEQUENCE_ITEM_STATE) + return yaml_emitter_emit_node(emitter, event, false, true, false, false) +} + +// Expect a block key node. +func yaml_emitter_emit_block_mapping_key(emitter *yaml_emitter_t, event *yaml_event_t, first bool) bool { + if first { + if !yaml_emitter_increase_indent(emitter, false, false) { + return false + } + } + if event.typ == yaml_MAPPING_END_EVENT { + emitter.indent = emitter.indents[len(emitter.indents)-1] + emitter.indents = emitter.indents[:len(emitter.indents)-1] + emitter.state = emitter.states[len(emitter.states)-1] + emitter.states = emitter.states[:len(emitter.states)-1] + return true + } + if !yaml_emitter_write_indent(emitter) { + return false + } + if yaml_emitter_check_simple_key(emitter) { + emitter.states = append(emitter.states, yaml_EMIT_BLOCK_MAPPING_SIMPLE_VALUE_STATE) + return yaml_emitter_emit_node(emitter, event, false, false, true, true) + } + if !yaml_emitter_write_indicator(emitter, []byte{'?'}, true, false, true) { + return false + } + emitter.states = append(emitter.states, yaml_EMIT_BLOCK_MAPPING_VALUE_STATE) + return yaml_emitter_emit_node(emitter, event, false, false, true, false) +} + +// Expect a block value node. +func yaml_emitter_emit_block_mapping_value(emitter *yaml_emitter_t, event *yaml_event_t, simple bool) bool { + if simple { + if !yaml_emitter_write_indicator(emitter, []byte{':'}, false, false, false) { + return false + } + } else { + if !yaml_emitter_write_indent(emitter) { + return false + } + if !yaml_emitter_write_indicator(emitter, []byte{':'}, true, false, true) { + return false + } + } + emitter.states = append(emitter.states, yaml_EMIT_BLOCK_MAPPING_KEY_STATE) + return yaml_emitter_emit_node(emitter, event, false, false, true, false) +} + +// Expect a node. +func yaml_emitter_emit_node(emitter *yaml_emitter_t, event *yaml_event_t, + root bool, sequence bool, mapping bool, simple_key bool) bool { + + emitter.root_context = root + emitter.sequence_context = sequence + emitter.mapping_context = mapping + emitter.simple_key_context = simple_key + + switch event.typ { + case yaml_ALIAS_EVENT: + return yaml_emitter_emit_alias(emitter, event) + case yaml_SCALAR_EVENT: + return yaml_emitter_emit_scalar(emitter, event) + case yaml_SEQUENCE_START_EVENT: + return yaml_emitter_emit_sequence_start(emitter, event) + case yaml_MAPPING_START_EVENT: + return yaml_emitter_emit_mapping_start(emitter, event) + default: + return yaml_emitter_set_emitter_error(emitter, + fmt.Sprintf("expected SCALAR, SEQUENCE-START, MAPPING-START, or ALIAS, but got %v", event.typ)) + } +} + +// Expect ALIAS. +func yaml_emitter_emit_alias(emitter *yaml_emitter_t, event *yaml_event_t) bool { + if !yaml_emitter_process_anchor(emitter) { + return false + } + emitter.state = emitter.states[len(emitter.states)-1] + emitter.states = emitter.states[:len(emitter.states)-1] + return true +} + +// Expect SCALAR. +func yaml_emitter_emit_scalar(emitter *yaml_emitter_t, event *yaml_event_t) bool { + if !yaml_emitter_select_scalar_style(emitter, event) { + return false + } + if !yaml_emitter_process_anchor(emitter) { + return false + } + if !yaml_emitter_process_tag(emitter) { + return false + } + if !yaml_emitter_increase_indent(emitter, true, false) { + return false + } + if !yaml_emitter_process_scalar(emitter) { + return false + } + emitter.indent = emitter.indents[len(emitter.indents)-1] + emitter.indents = emitter.indents[:len(emitter.indents)-1] + emitter.state = emitter.states[len(emitter.states)-1] + emitter.states = emitter.states[:len(emitter.states)-1] + return true +} + +// Expect SEQUENCE-START. +func yaml_emitter_emit_sequence_start(emitter *yaml_emitter_t, event *yaml_event_t) bool { + if !yaml_emitter_process_anchor(emitter) { + return false + } + if !yaml_emitter_process_tag(emitter) { + return false + } + if emitter.flow_level > 0 || emitter.canonical || event.sequence_style() == yaml_FLOW_SEQUENCE_STYLE || + yaml_emitter_check_empty_sequence(emitter) { + emitter.state = yaml_EMIT_FLOW_SEQUENCE_FIRST_ITEM_STATE + } else { + emitter.state = yaml_EMIT_BLOCK_SEQUENCE_FIRST_ITEM_STATE + } + return true +} + +// Expect MAPPING-START. +func yaml_emitter_emit_mapping_start(emitter *yaml_emitter_t, event *yaml_event_t) bool { + if !yaml_emitter_process_anchor(emitter) { + return false + } + if !yaml_emitter_process_tag(emitter) { + return false + } + if emitter.flow_level > 0 || emitter.canonical || event.mapping_style() == yaml_FLOW_MAPPING_STYLE || + yaml_emitter_check_empty_mapping(emitter) { + emitter.state = yaml_EMIT_FLOW_MAPPING_FIRST_KEY_STATE + } else { + emitter.state = yaml_EMIT_BLOCK_MAPPING_FIRST_KEY_STATE + } + return true +} + +// Check if the document content is an empty scalar. +func yaml_emitter_check_empty_document(emitter *yaml_emitter_t) bool { + return false // [Go] Huh? +} + +// Check if the next events represent an empty sequence. +func yaml_emitter_check_empty_sequence(emitter *yaml_emitter_t) bool { + if len(emitter.events)-emitter.events_head < 2 { + return false + } + return emitter.events[emitter.events_head].typ == yaml_SEQUENCE_START_EVENT && + emitter.events[emitter.events_head+1].typ == yaml_SEQUENCE_END_EVENT +} + +// Check if the next events represent an empty mapping. +func yaml_emitter_check_empty_mapping(emitter *yaml_emitter_t) bool { + if len(emitter.events)-emitter.events_head < 2 { + return false + } + return emitter.events[emitter.events_head].typ == yaml_MAPPING_START_EVENT && + emitter.events[emitter.events_head+1].typ == yaml_MAPPING_END_EVENT +} + +// Check if the next node can be expressed as a simple key. +func yaml_emitter_check_simple_key(emitter *yaml_emitter_t) bool { + length := 0 + switch emitter.events[emitter.events_head].typ { + case yaml_ALIAS_EVENT: + length += len(emitter.anchor_data.anchor) + case yaml_SCALAR_EVENT: + if emitter.scalar_data.multiline { + return false + } + length += len(emitter.anchor_data.anchor) + + len(emitter.tag_data.handle) + + len(emitter.tag_data.suffix) + + len(emitter.scalar_data.value) + case yaml_SEQUENCE_START_EVENT: + if !yaml_emitter_check_empty_sequence(emitter) { + return false + } + length += len(emitter.anchor_data.anchor) + + len(emitter.tag_data.handle) + + len(emitter.tag_data.suffix) + case yaml_MAPPING_START_EVENT: + if !yaml_emitter_check_empty_mapping(emitter) { + return false + } + length += len(emitter.anchor_data.anchor) + + len(emitter.tag_data.handle) + + len(emitter.tag_data.suffix) + default: + return false + } + return length <= 128 +} + +// Determine an acceptable scalar style. +func yaml_emitter_select_scalar_style(emitter *yaml_emitter_t, event *yaml_event_t) bool { + + no_tag := len(emitter.tag_data.handle) == 0 && len(emitter.tag_data.suffix) == 0 + if no_tag && !event.implicit && !event.quoted_implicit { + return yaml_emitter_set_emitter_error(emitter, "neither tag nor implicit flags are specified") + } + + style := event.scalar_style() + if style == yaml_ANY_SCALAR_STYLE { + style = yaml_PLAIN_SCALAR_STYLE + } + if emitter.canonical { + style = yaml_DOUBLE_QUOTED_SCALAR_STYLE + } + if emitter.simple_key_context && emitter.scalar_data.multiline { + style = yaml_DOUBLE_QUOTED_SCALAR_STYLE + } + + if style == yaml_PLAIN_SCALAR_STYLE { + if emitter.flow_level > 0 && !emitter.scalar_data.flow_plain_allowed || + emitter.flow_level == 0 && !emitter.scalar_data.block_plain_allowed { + style = yaml_SINGLE_QUOTED_SCALAR_STYLE + } + if len(emitter.scalar_data.value) == 0 && (emitter.flow_level > 0 || emitter.simple_key_context) { + style = yaml_SINGLE_QUOTED_SCALAR_STYLE + } + if no_tag && !event.implicit { + style = yaml_SINGLE_QUOTED_SCALAR_STYLE + } + } + if style == yaml_SINGLE_QUOTED_SCALAR_STYLE { + if !emitter.scalar_data.single_quoted_allowed { + style = yaml_DOUBLE_QUOTED_SCALAR_STYLE + } + } + if style == yaml_LITERAL_SCALAR_STYLE || style == yaml_FOLDED_SCALAR_STYLE { + if !emitter.scalar_data.block_allowed || emitter.flow_level > 0 || emitter.simple_key_context { + style = yaml_DOUBLE_QUOTED_SCALAR_STYLE + } + } + + if no_tag && !event.quoted_implicit && style != yaml_PLAIN_SCALAR_STYLE { + emitter.tag_data.handle = []byte{'!'} + } + emitter.scalar_data.style = style + return true +} + +// Write an anchor. +func yaml_emitter_process_anchor(emitter *yaml_emitter_t) bool { + if emitter.anchor_data.anchor == nil { + return true + } + c := []byte{'&'} + if emitter.anchor_data.alias { + c[0] = '*' + } + if !yaml_emitter_write_indicator(emitter, c, true, false, false) { + return false + } + return yaml_emitter_write_anchor(emitter, emitter.anchor_data.anchor) +} + +// Write a tag. +func yaml_emitter_process_tag(emitter *yaml_emitter_t) bool { + if len(emitter.tag_data.handle) == 0 && len(emitter.tag_data.suffix) == 0 { + return true + } + if len(emitter.tag_data.handle) > 0 { + if !yaml_emitter_write_tag_handle(emitter, emitter.tag_data.handle) { + return false + } + if len(emitter.tag_data.suffix) > 0 { + if !yaml_emitter_write_tag_content(emitter, emitter.tag_data.suffix, false) { + return false + } + } + } else { + // [Go] Allocate these slices elsewhere. + if !yaml_emitter_write_indicator(emitter, []byte("!<"), true, false, false) { + return false + } + if !yaml_emitter_write_tag_content(emitter, emitter.tag_data.suffix, false) { + return false + } + if !yaml_emitter_write_indicator(emitter, []byte{'>'}, false, false, false) { + return false + } + } + return true +} + +// Write a scalar. +func yaml_emitter_process_scalar(emitter *yaml_emitter_t) bool { + switch emitter.scalar_data.style { + case yaml_PLAIN_SCALAR_STYLE: + return yaml_emitter_write_plain_scalar(emitter, emitter.scalar_data.value, !emitter.simple_key_context) + + case yaml_SINGLE_QUOTED_SCALAR_STYLE: + return yaml_emitter_write_single_quoted_scalar(emitter, emitter.scalar_data.value, !emitter.simple_key_context) + + case yaml_DOUBLE_QUOTED_SCALAR_STYLE: + return yaml_emitter_write_double_quoted_scalar(emitter, emitter.scalar_data.value, !emitter.simple_key_context) + + case yaml_LITERAL_SCALAR_STYLE: + return yaml_emitter_write_literal_scalar(emitter, emitter.scalar_data.value) + + case yaml_FOLDED_SCALAR_STYLE: + return yaml_emitter_write_folded_scalar(emitter, emitter.scalar_data.value) + } + panic("unknown scalar style") +} + +// Check if a %YAML directive is valid. +func yaml_emitter_analyze_version_directive(emitter *yaml_emitter_t, version_directive *yaml_version_directive_t) bool { + if version_directive.major != 1 || version_directive.minor != 1 { + return yaml_emitter_set_emitter_error(emitter, "incompatible %YAML directive") + } + return true +} + +// Check if a %TAG directive is valid. +func yaml_emitter_analyze_tag_directive(emitter *yaml_emitter_t, tag_directive *yaml_tag_directive_t) bool { + handle := tag_directive.handle + prefix := tag_directive.prefix + if len(handle) == 0 { + return yaml_emitter_set_emitter_error(emitter, "tag handle must not be empty") + } + if handle[0] != '!' { + return yaml_emitter_set_emitter_error(emitter, "tag handle must start with '!'") + } + if handle[len(handle)-1] != '!' { + return yaml_emitter_set_emitter_error(emitter, "tag handle must end with '!'") + } + for i := 1; i < len(handle)-1; i += width(handle[i]) { + if !is_alpha(handle, i) { + return yaml_emitter_set_emitter_error(emitter, "tag handle must contain alphanumerical characters only") + } + } + if len(prefix) == 0 { + return yaml_emitter_set_emitter_error(emitter, "tag prefix must not be empty") + } + return true +} + +// Check if an anchor is valid. +func yaml_emitter_analyze_anchor(emitter *yaml_emitter_t, anchor []byte, alias bool) bool { + if len(anchor) == 0 { + problem := "anchor value must not be empty" + if alias { + problem = "alias value must not be empty" + } + return yaml_emitter_set_emitter_error(emitter, problem) + } + for i := 0; i < len(anchor); i += width(anchor[i]) { + if !is_alpha(anchor, i) { + problem := "anchor value must contain alphanumerical characters only" + if alias { + problem = "alias value must contain alphanumerical characters only" + } + return yaml_emitter_set_emitter_error(emitter, problem) + } + } + emitter.anchor_data.anchor = anchor + emitter.anchor_data.alias = alias + return true +} + +// Check if a tag is valid. +func yaml_emitter_analyze_tag(emitter *yaml_emitter_t, tag []byte) bool { + if len(tag) == 0 { + return yaml_emitter_set_emitter_error(emitter, "tag value must not be empty") + } + for i := 0; i < len(emitter.tag_directives); i++ { + tag_directive := &emitter.tag_directives[i] + if bytes.HasPrefix(tag, tag_directive.prefix) { + emitter.tag_data.handle = tag_directive.handle + emitter.tag_data.suffix = tag[len(tag_directive.prefix):] + return true + } + } + emitter.tag_data.suffix = tag + return true +} + +// Check if a scalar is valid. +func yaml_emitter_analyze_scalar(emitter *yaml_emitter_t, value []byte) bool { + var ( + block_indicators = false + flow_indicators = false + line_breaks = false + special_characters = false + + leading_space = false + leading_break = false + trailing_space = false + trailing_break = false + break_space = false + space_break = false + + preceded_by_whitespace = false + followed_by_whitespace = false + previous_space = false + previous_break = false + ) + + emitter.scalar_data.value = value + + if len(value) == 0 { + emitter.scalar_data.multiline = false + emitter.scalar_data.flow_plain_allowed = false + emitter.scalar_data.block_plain_allowed = true + emitter.scalar_data.single_quoted_allowed = true + emitter.scalar_data.block_allowed = false + return true + } + + if len(value) >= 3 && ((value[0] == '-' && value[1] == '-' && value[2] == '-') || (value[0] == '.' && value[1] == '.' && value[2] == '.')) { + block_indicators = true + flow_indicators = true + } + + preceded_by_whitespace = true + for i, w := 0, 0; i < len(value); i += w { + w = width(value[i]) + followed_by_whitespace = i+w >= len(value) || is_blank(value, i+w) + + if i == 0 { + switch value[i] { + case '#', ',', '[', ']', '{', '}', '&', '*', '!', '|', '>', '\'', '"', '%', '@', '`': + flow_indicators = true + block_indicators = true + case '?', ':': + flow_indicators = true + if followed_by_whitespace { + block_indicators = true + } + case '-': + if followed_by_whitespace { + flow_indicators = true + block_indicators = true + } + } + } else { + switch value[i] { + case ',', '?', '[', ']', '{', '}': + flow_indicators = true + case ':': + flow_indicators = true + if followed_by_whitespace { + block_indicators = true + } + case '#': + if preceded_by_whitespace { + flow_indicators = true + block_indicators = true + } + } + } + + if !is_printable(value, i) || !is_ascii(value, i) && !emitter.unicode { + special_characters = true + } + if is_space(value, i) { + if i == 0 { + leading_space = true + } + if i+width(value[i]) == len(value) { + trailing_space = true + } + if previous_break { + break_space = true + } + previous_space = true + previous_break = false + } else if is_break(value, i) { + line_breaks = true + if i == 0 { + leading_break = true + } + if i+width(value[i]) == len(value) { + trailing_break = true + } + if previous_space { + space_break = true + } + previous_space = false + previous_break = true + } else { + previous_space = false + previous_break = false + } + + // [Go]: Why 'z'? Couldn't be the end of the string as that's the loop condition. + preceded_by_whitespace = is_blankz(value, i) + } + + emitter.scalar_data.multiline = line_breaks + emitter.scalar_data.flow_plain_allowed = true + emitter.scalar_data.block_plain_allowed = true + emitter.scalar_data.single_quoted_allowed = true + emitter.scalar_data.block_allowed = true + + if leading_space || leading_break || trailing_space || trailing_break { + emitter.scalar_data.flow_plain_allowed = false + emitter.scalar_data.block_plain_allowed = false + } + if trailing_space { + emitter.scalar_data.block_allowed = false + } + if break_space { + emitter.scalar_data.flow_plain_allowed = false + emitter.scalar_data.block_plain_allowed = false + emitter.scalar_data.single_quoted_allowed = false + } + if space_break || special_characters { + emitter.scalar_data.flow_plain_allowed = false + emitter.scalar_data.block_plain_allowed = false + emitter.scalar_data.single_quoted_allowed = false + emitter.scalar_data.block_allowed = false + } + if line_breaks { + emitter.scalar_data.flow_plain_allowed = false + emitter.scalar_data.block_plain_allowed = false + } + if flow_indicators { + emitter.scalar_data.flow_plain_allowed = false + } + if block_indicators { + emitter.scalar_data.block_plain_allowed = false + } + return true +} + +// Check if the event data is valid. +func yaml_emitter_analyze_event(emitter *yaml_emitter_t, event *yaml_event_t) bool { + + emitter.anchor_data.anchor = nil + emitter.tag_data.handle = nil + emitter.tag_data.suffix = nil + emitter.scalar_data.value = nil + + switch event.typ { + case yaml_ALIAS_EVENT: + if !yaml_emitter_analyze_anchor(emitter, event.anchor, true) { + return false + } + + case yaml_SCALAR_EVENT: + if len(event.anchor) > 0 { + if !yaml_emitter_analyze_anchor(emitter, event.anchor, false) { + return false + } + } + if len(event.tag) > 0 && (emitter.canonical || (!event.implicit && !event.quoted_implicit)) { + if !yaml_emitter_analyze_tag(emitter, event.tag) { + return false + } + } + if !yaml_emitter_analyze_scalar(emitter, event.value) { + return false + } + + case yaml_SEQUENCE_START_EVENT: + if len(event.anchor) > 0 { + if !yaml_emitter_analyze_anchor(emitter, event.anchor, false) { + return false + } + } + if len(event.tag) > 0 && (emitter.canonical || !event.implicit) { + if !yaml_emitter_analyze_tag(emitter, event.tag) { + return false + } + } + + case yaml_MAPPING_START_EVENT: + if len(event.anchor) > 0 { + if !yaml_emitter_analyze_anchor(emitter, event.anchor, false) { + return false + } + } + if len(event.tag) > 0 && (emitter.canonical || !event.implicit) { + if !yaml_emitter_analyze_tag(emitter, event.tag) { + return false + } + } + } + return true +} + +// Write the BOM character. +func yaml_emitter_write_bom(emitter *yaml_emitter_t) bool { + if !flush(emitter) { + return false + } + pos := emitter.buffer_pos + emitter.buffer[pos+0] = '\xEF' + emitter.buffer[pos+1] = '\xBB' + emitter.buffer[pos+2] = '\xBF' + emitter.buffer_pos += 3 + return true +} + +func yaml_emitter_write_indent(emitter *yaml_emitter_t) bool { + indent := emitter.indent + if indent < 0 { + indent = 0 + } + if !emitter.indention || emitter.column > indent || (emitter.column == indent && !emitter.whitespace) { + if !put_break(emitter) { + return false + } + } + for emitter.column < indent { + if !put(emitter, ' ') { + return false + } + } + emitter.whitespace = true + emitter.indention = true + return true +} + +func yaml_emitter_write_indicator(emitter *yaml_emitter_t, indicator []byte, need_whitespace, is_whitespace, is_indention bool) bool { + if need_whitespace && !emitter.whitespace { + if !put(emitter, ' ') { + return false + } + } + if !write_all(emitter, indicator) { + return false + } + emitter.whitespace = is_whitespace + emitter.indention = (emitter.indention && is_indention) + emitter.open_ended = false + return true +} + +func yaml_emitter_write_anchor(emitter *yaml_emitter_t, value []byte) bool { + if !write_all(emitter, value) { + return false + } + emitter.whitespace = false + emitter.indention = false + return true +} + +func yaml_emitter_write_tag_handle(emitter *yaml_emitter_t, value []byte) bool { + if !emitter.whitespace { + if !put(emitter, ' ') { + return false + } + } + if !write_all(emitter, value) { + return false + } + emitter.whitespace = false + emitter.indention = false + return true +} + +func yaml_emitter_write_tag_content(emitter *yaml_emitter_t, value []byte, need_whitespace bool) bool { + if need_whitespace && !emitter.whitespace { + if !put(emitter, ' ') { + return false + } + } + for i := 0; i < len(value); { + var must_write bool + switch value[i] { + case ';', '/', '?', ':', '@', '&', '=', '+', '$', ',', '_', '.', '~', '*', '\'', '(', ')', '[', ']': + must_write = true + default: + must_write = is_alpha(value, i) + } + if must_write { + if !write(emitter, value, &i) { + return false + } + } else { + w := width(value[i]) + for k := 0; k < w; k++ { + octet := value[i] + i++ + if !put(emitter, '%') { + return false + } + + c := octet >> 4 + if c < 10 { + c += '0' + } else { + c += 'A' - 10 + } + if !put(emitter, c) { + return false + } + + c = octet & 0x0f + if c < 10 { + c += '0' + } else { + c += 'A' - 10 + } + if !put(emitter, c) { + return false + } + } + } + } + emitter.whitespace = false + emitter.indention = false + return true +} + +func yaml_emitter_write_plain_scalar(emitter *yaml_emitter_t, value []byte, allow_breaks bool) bool { + if !emitter.whitespace { + if !put(emitter, ' ') { + return false + } + } + + spaces := false + breaks := false + for i := 0; i < len(value); { + if is_space(value, i) { + if allow_breaks && !spaces && emitter.column > emitter.best_width && !is_space(value, i+1) { + if !yaml_emitter_write_indent(emitter) { + return false + } + i += width(value[i]) + } else { + if !write(emitter, value, &i) { + return false + } + } + spaces = true + } else if is_break(value, i) { + if !breaks && value[i] == '\n' { + if !put_break(emitter) { + return false + } + } + if !write_break(emitter, value, &i) { + return false + } + emitter.indention = true + breaks = true + } else { + if breaks { + if !yaml_emitter_write_indent(emitter) { + return false + } + } + if !write(emitter, value, &i) { + return false + } + emitter.indention = false + spaces = false + breaks = false + } + } + + emitter.whitespace = false + emitter.indention = false + if emitter.root_context { + emitter.open_ended = true + } + + return true +} + +func yaml_emitter_write_single_quoted_scalar(emitter *yaml_emitter_t, value []byte, allow_breaks bool) bool { + + if !yaml_emitter_write_indicator(emitter, []byte{'\''}, true, false, false) { + return false + } + + spaces := false + breaks := false + for i := 0; i < len(value); { + if is_space(value, i) { + if allow_breaks && !spaces && emitter.column > emitter.best_width && i > 0 && i < len(value)-1 && !is_space(value, i+1) { + if !yaml_emitter_write_indent(emitter) { + return false + } + i += width(value[i]) + } else { + if !write(emitter, value, &i) { + return false + } + } + spaces = true + } else if is_break(value, i) { + if !breaks && value[i] == '\n' { + if !put_break(emitter) { + return false + } + } + if !write_break(emitter, value, &i) { + return false + } + emitter.indention = true + breaks = true + } else { + if breaks { + if !yaml_emitter_write_indent(emitter) { + return false + } + } + if value[i] == '\'' { + if !put(emitter, '\'') { + return false + } + } + if !write(emitter, value, &i) { + return false + } + emitter.indention = false + spaces = false + breaks = false + } + } + if !yaml_emitter_write_indicator(emitter, []byte{'\''}, false, false, false) { + return false + } + emitter.whitespace = false + emitter.indention = false + return true +} + +func yaml_emitter_write_double_quoted_scalar(emitter *yaml_emitter_t, value []byte, allow_breaks bool) bool { + spaces := false + if !yaml_emitter_write_indicator(emitter, []byte{'"'}, true, false, false) { + return false + } + + for i := 0; i < len(value); { + if !is_printable(value, i) || (!emitter.unicode && !is_ascii(value, i)) || + is_bom(value, i) || is_break(value, i) || + value[i] == '"' || value[i] == '\\' { + + octet := value[i] + + var w int + var v rune + switch { + case octet&0x80 == 0x00: + w, v = 1, rune(octet&0x7F) + case octet&0xE0 == 0xC0: + w, v = 2, rune(octet&0x1F) + case octet&0xF0 == 0xE0: + w, v = 3, rune(octet&0x0F) + case octet&0xF8 == 0xF0: + w, v = 4, rune(octet&0x07) + } + for k := 1; k < w; k++ { + octet = value[i+k] + v = (v << 6) + (rune(octet) & 0x3F) + } + i += w + + if !put(emitter, '\\') { + return false + } + + var ok bool + switch v { + case 0x00: + ok = put(emitter, '0') + case 0x07: + ok = put(emitter, 'a') + case 0x08: + ok = put(emitter, 'b') + case 0x09: + ok = put(emitter, 't') + case 0x0A: + ok = put(emitter, 'n') + case 0x0b: + ok = put(emitter, 'v') + case 0x0c: + ok = put(emitter, 'f') + case 0x0d: + ok = put(emitter, 'r') + case 0x1b: + ok = put(emitter, 'e') + case 0x22: + ok = put(emitter, '"') + case 0x5c: + ok = put(emitter, '\\') + case 0x85: + ok = put(emitter, 'N') + case 0xA0: + ok = put(emitter, '_') + case 0x2028: + ok = put(emitter, 'L') + case 0x2029: + ok = put(emitter, 'P') + default: + if v <= 0xFF { + ok = put(emitter, 'x') + w = 2 + } else if v <= 0xFFFF { + ok = put(emitter, 'u') + w = 4 + } else { + ok = put(emitter, 'U') + w = 8 + } + for k := (w - 1) * 4; ok && k >= 0; k -= 4 { + digit := byte((v >> uint(k)) & 0x0F) + if digit < 10 { + ok = put(emitter, digit+'0') + } else { + ok = put(emitter, digit+'A'-10) + } + } + } + if !ok { + return false + } + spaces = false + } else if is_space(value, i) { + if allow_breaks && !spaces && emitter.column > emitter.best_width && i > 0 && i < len(value)-1 { + if !yaml_emitter_write_indent(emitter) { + return false + } + if is_space(value, i+1) { + if !put(emitter, '\\') { + return false + } + } + i += width(value[i]) + } else if !write(emitter, value, &i) { + return false + } + spaces = true + } else { + if !write(emitter, value, &i) { + return false + } + spaces = false + } + } + if !yaml_emitter_write_indicator(emitter, []byte{'"'}, false, false, false) { + return false + } + emitter.whitespace = false + emitter.indention = false + return true +} + +func yaml_emitter_write_block_scalar_hints(emitter *yaml_emitter_t, value []byte) bool { + if is_space(value, 0) || is_break(value, 0) { + indent_hint := []byte{'0' + byte(emitter.best_indent)} + if !yaml_emitter_write_indicator(emitter, indent_hint, false, false, false) { + return false + } + } + + emitter.open_ended = false + + var chomp_hint [1]byte + if len(value) == 0 { + chomp_hint[0] = '-' + } else { + i := len(value) - 1 + for value[i]&0xC0 == 0x80 { + i-- + } + if !is_break(value, i) { + chomp_hint[0] = '-' + } else if i == 0 { + chomp_hint[0] = '+' + emitter.open_ended = true + } else { + i-- + for value[i]&0xC0 == 0x80 { + i-- + } + if is_break(value, i) { + chomp_hint[0] = '+' + emitter.open_ended = true + } + } + } + if chomp_hint[0] != 0 { + if !yaml_emitter_write_indicator(emitter, chomp_hint[:], false, false, false) { + return false + } + } + return true +} + +func yaml_emitter_write_literal_scalar(emitter *yaml_emitter_t, value []byte) bool { + if !yaml_emitter_write_indicator(emitter, []byte{'|'}, true, false, false) { + return false + } + if !yaml_emitter_write_block_scalar_hints(emitter, value) { + return false + } + if !put_break(emitter) { + return false + } + emitter.indention = true + emitter.whitespace = true + breaks := true + for i := 0; i < len(value); { + if is_break(value, i) { + if !write_break(emitter, value, &i) { + return false + } + emitter.indention = true + breaks = true + } else { + if breaks { + if !yaml_emitter_write_indent(emitter) { + return false + } + } + if !write(emitter, value, &i) { + return false + } + emitter.indention = false + breaks = false + } + } + + return true +} + +func yaml_emitter_write_folded_scalar(emitter *yaml_emitter_t, value []byte) bool { + if !yaml_emitter_write_indicator(emitter, []byte{'>'}, true, false, false) { + return false + } + if !yaml_emitter_write_block_scalar_hints(emitter, value) { + return false + } + + if !put_break(emitter) { + return false + } + emitter.indention = true + emitter.whitespace = true + + breaks := true + leading_spaces := true + for i := 0; i < len(value); { + if is_break(value, i) { + if !breaks && !leading_spaces && value[i] == '\n' { + k := 0 + for is_break(value, k) { + k += width(value[k]) + } + if !is_blankz(value, k) { + if !put_break(emitter) { + return false + } + } + } + if !write_break(emitter, value, &i) { + return false + } + emitter.indention = true + breaks = true + } else { + if breaks { + if !yaml_emitter_write_indent(emitter) { + return false + } + leading_spaces = is_blank(value, i) + } + if !breaks && is_space(value, i) && !is_space(value, i+1) && emitter.column > emitter.best_width { + if !yaml_emitter_write_indent(emitter) { + return false + } + i += width(value[i]) + } else { + if !write(emitter, value, &i) { + return false + } + } + emitter.indention = false + breaks = false + } + } + return true +} diff --git a/vendor/gopkg.in/yaml.v2/encode.go b/vendor/gopkg.in/yaml.v2/encode.go new file mode 100644 index 00000000..0ee738e1 --- /dev/null +++ b/vendor/gopkg.in/yaml.v2/encode.go @@ -0,0 +1,390 @@ +package yaml + +import ( + "encoding" + "fmt" + "io" + "reflect" + "regexp" + "sort" + "strconv" + "strings" + "time" + "unicode/utf8" +) + +// jsonNumber is the interface of the encoding/json.Number datatype. +// Repeating the interface here avoids a dependency on encoding/json, and also +// supports other libraries like jsoniter, which use a similar datatype with +// the same interface. Detecting this interface is useful when dealing with +// structures containing json.Number, which is a string under the hood. The +// encoder should prefer the use of Int64(), Float64() and string(), in that +// order, when encoding this type. +type jsonNumber interface { + Float64() (float64, error) + Int64() (int64, error) + String() string +} + +type encoder struct { + emitter yaml_emitter_t + event yaml_event_t + out []byte + flow bool + // doneInit holds whether the initial stream_start_event has been + // emitted. + doneInit bool +} + +func newEncoder() *encoder { + e := &encoder{} + yaml_emitter_initialize(&e.emitter) + yaml_emitter_set_output_string(&e.emitter, &e.out) + yaml_emitter_set_unicode(&e.emitter, true) + return e +} + +func newEncoderWithWriter(w io.Writer) *encoder { + e := &encoder{} + yaml_emitter_initialize(&e.emitter) + yaml_emitter_set_output_writer(&e.emitter, w) + yaml_emitter_set_unicode(&e.emitter, true) + return e +} + +func (e *encoder) init() { + if e.doneInit { + return + } + yaml_stream_start_event_initialize(&e.event, yaml_UTF8_ENCODING) + e.emit() + e.doneInit = true +} + +func (e *encoder) finish() { + e.emitter.open_ended = false + yaml_stream_end_event_initialize(&e.event) + e.emit() +} + +func (e *encoder) destroy() { + yaml_emitter_delete(&e.emitter) +} + +func (e *encoder) emit() { + // This will internally delete the e.event value. + e.must(yaml_emitter_emit(&e.emitter, &e.event)) +} + +func (e *encoder) must(ok bool) { + if !ok { + msg := e.emitter.problem + if msg == "" { + msg = "unknown problem generating YAML content" + } + failf("%s", msg) + } +} + +func (e *encoder) marshalDoc(tag string, in reflect.Value) { + e.init() + yaml_document_start_event_initialize(&e.event, nil, nil, true) + e.emit() + e.marshal(tag, in) + yaml_document_end_event_initialize(&e.event, true) + e.emit() +} + +func (e *encoder) marshal(tag string, in reflect.Value) { + if !in.IsValid() || in.Kind() == reflect.Ptr && in.IsNil() { + e.nilv() + return + } + iface := in.Interface() + switch m := iface.(type) { + case jsonNumber: + integer, err := m.Int64() + if err == nil { + // In this case the json.Number is a valid int64 + in = reflect.ValueOf(integer) + break + } + float, err := m.Float64() + if err == nil { + // In this case the json.Number is a valid float64 + in = reflect.ValueOf(float) + break + } + // fallback case - no number could be obtained + in = reflect.ValueOf(m.String()) + case time.Time, *time.Time: + // Although time.Time implements TextMarshaler, + // we don't want to treat it as a string for YAML + // purposes because YAML has special support for + // timestamps. + case Marshaler: + v, err := m.MarshalYAML() + if err != nil { + fail(err) + } + if v == nil { + e.nilv() + return + } + in = reflect.ValueOf(v) + case encoding.TextMarshaler: + text, err := m.MarshalText() + if err != nil { + fail(err) + } + in = reflect.ValueOf(string(text)) + case nil: + e.nilv() + return + } + switch in.Kind() { + case reflect.Interface: + e.marshal(tag, in.Elem()) + case reflect.Map: + e.mapv(tag, in) + case reflect.Ptr: + if in.Type() == ptrTimeType { + e.timev(tag, in.Elem()) + } else { + e.marshal(tag, in.Elem()) + } + case reflect.Struct: + if in.Type() == timeType { + e.timev(tag, in) + } else { + e.structv(tag, in) + } + case reflect.Slice, reflect.Array: + if in.Type().Elem() == mapItemType { + e.itemsv(tag, in) + } else { + e.slicev(tag, in) + } + case reflect.String: + e.stringv(tag, in) + case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64: + if in.Type() == durationType { + e.stringv(tag, reflect.ValueOf(iface.(time.Duration).String())) + } else { + e.intv(tag, in) + } + case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64, reflect.Uintptr: + e.uintv(tag, in) + case reflect.Float32, reflect.Float64: + e.floatv(tag, in) + case reflect.Bool: + e.boolv(tag, in) + default: + panic("cannot marshal type: " + in.Type().String()) + } +} + +func (e *encoder) mapv(tag string, in reflect.Value) { + e.mappingv(tag, func() { + keys := keyList(in.MapKeys()) + sort.Sort(keys) + for _, k := range keys { + e.marshal("", k) + e.marshal("", in.MapIndex(k)) + } + }) +} + +func (e *encoder) itemsv(tag string, in reflect.Value) { + e.mappingv(tag, func() { + slice := in.Convert(reflect.TypeOf([]MapItem{})).Interface().([]MapItem) + for _, item := range slice { + e.marshal("", reflect.ValueOf(item.Key)) + e.marshal("", reflect.ValueOf(item.Value)) + } + }) +} + +func (e *encoder) structv(tag string, in reflect.Value) { + sinfo, err := getStructInfo(in.Type()) + if err != nil { + panic(err) + } + e.mappingv(tag, func() { + for _, info := range sinfo.FieldsList { + var value reflect.Value + if info.Inline == nil { + value = in.Field(info.Num) + } else { + value = in.FieldByIndex(info.Inline) + } + if info.OmitEmpty && isZero(value) { + continue + } + e.marshal("", reflect.ValueOf(info.Key)) + e.flow = info.Flow + e.marshal("", value) + } + if sinfo.InlineMap >= 0 { + m := in.Field(sinfo.InlineMap) + if m.Len() > 0 { + e.flow = false + keys := keyList(m.MapKeys()) + sort.Sort(keys) + for _, k := range keys { + if _, found := sinfo.FieldsMap[k.String()]; found { + panic(fmt.Sprintf("Can't have key %q in inlined map; conflicts with struct field", k.String())) + } + e.marshal("", k) + e.flow = false + e.marshal("", m.MapIndex(k)) + } + } + } + }) +} + +func (e *encoder) mappingv(tag string, f func()) { + implicit := tag == "" + style := yaml_BLOCK_MAPPING_STYLE + if e.flow { + e.flow = false + style = yaml_FLOW_MAPPING_STYLE + } + yaml_mapping_start_event_initialize(&e.event, nil, []byte(tag), implicit, style) + e.emit() + f() + yaml_mapping_end_event_initialize(&e.event) + e.emit() +} + +func (e *encoder) slicev(tag string, in reflect.Value) { + implicit := tag == "" + style := yaml_BLOCK_SEQUENCE_STYLE + if e.flow { + e.flow = false + style = yaml_FLOW_SEQUENCE_STYLE + } + e.must(yaml_sequence_start_event_initialize(&e.event, nil, []byte(tag), implicit, style)) + e.emit() + n := in.Len() + for i := 0; i < n; i++ { + e.marshal("", in.Index(i)) + } + e.must(yaml_sequence_end_event_initialize(&e.event)) + e.emit() +} + +// isBase60 returns whether s is in base 60 notation as defined in YAML 1.1. +// +// The base 60 float notation in YAML 1.1 is a terrible idea and is unsupported +// in YAML 1.2 and by this package, but these should be marshalled quoted for +// the time being for compatibility with other parsers. +func isBase60Float(s string) (result bool) { + // Fast path. + if s == "" { + return false + } + c := s[0] + if !(c == '+' || c == '-' || c >= '0' && c <= '9') || strings.IndexByte(s, ':') < 0 { + return false + } + // Do the full match. + return base60float.MatchString(s) +} + +// From http://yaml.org/type/float.html, except the regular expression there +// is bogus. In practice parsers do not enforce the "\.[0-9_]*" suffix. +var base60float = regexp.MustCompile(`^[-+]?[0-9][0-9_]*(?::[0-5]?[0-9])+(?:\.[0-9_]*)?$`) + +func (e *encoder) stringv(tag string, in reflect.Value) { + var style yaml_scalar_style_t + s := in.String() + canUsePlain := true + switch { + case !utf8.ValidString(s): + if tag == yaml_BINARY_TAG { + failf("explicitly tagged !!binary data must be base64-encoded") + } + if tag != "" { + failf("cannot marshal invalid UTF-8 data as %s", shortTag(tag)) + } + // It can't be encoded directly as YAML so use a binary tag + // and encode it as base64. + tag = yaml_BINARY_TAG + s = encodeBase64(s) + case tag == "": + // Check to see if it would resolve to a specific + // tag when encoded unquoted. If it doesn't, + // there's no need to quote it. + rtag, _ := resolve("", s) + canUsePlain = rtag == yaml_STR_TAG && !isBase60Float(s) + } + // Note: it's possible for user code to emit invalid YAML + // if they explicitly specify a tag and a string containing + // text that's incompatible with that tag. + switch { + case strings.Contains(s, "\n"): + style = yaml_LITERAL_SCALAR_STYLE + case canUsePlain: + style = yaml_PLAIN_SCALAR_STYLE + default: + style = yaml_DOUBLE_QUOTED_SCALAR_STYLE + } + e.emitScalar(s, "", tag, style) +} + +func (e *encoder) boolv(tag string, in reflect.Value) { + var s string + if in.Bool() { + s = "true" + } else { + s = "false" + } + e.emitScalar(s, "", tag, yaml_PLAIN_SCALAR_STYLE) +} + +func (e *encoder) intv(tag string, in reflect.Value) { + s := strconv.FormatInt(in.Int(), 10) + e.emitScalar(s, "", tag, yaml_PLAIN_SCALAR_STYLE) +} + +func (e *encoder) uintv(tag string, in reflect.Value) { + s := strconv.FormatUint(in.Uint(), 10) + e.emitScalar(s, "", tag, yaml_PLAIN_SCALAR_STYLE) +} + +func (e *encoder) timev(tag string, in reflect.Value) { + t := in.Interface().(time.Time) + s := t.Format(time.RFC3339Nano) + e.emitScalar(s, "", tag, yaml_PLAIN_SCALAR_STYLE) +} + +func (e *encoder) floatv(tag string, in reflect.Value) { + // Issue #352: When formatting, use the precision of the underlying value + precision := 64 + if in.Kind() == reflect.Float32 { + precision = 32 + } + + s := strconv.FormatFloat(in.Float(), 'g', -1, precision) + switch s { + case "+Inf": + s = ".inf" + case "-Inf": + s = "-.inf" + case "NaN": + s = ".nan" + } + e.emitScalar(s, "", tag, yaml_PLAIN_SCALAR_STYLE) +} + +func (e *encoder) nilv() { + e.emitScalar("null", "", "", yaml_PLAIN_SCALAR_STYLE) +} + +func (e *encoder) emitScalar(value, anchor, tag string, style yaml_scalar_style_t) { + implicit := tag == "" + e.must(yaml_scalar_event_initialize(&e.event, []byte(anchor), []byte(tag), []byte(value), implicit, implicit, style)) + e.emit() +} diff --git a/vendor/gopkg.in/yaml.v2/parserc.go b/vendor/gopkg.in/yaml.v2/parserc.go new file mode 100644 index 00000000..81d05dfe --- /dev/null +++ b/vendor/gopkg.in/yaml.v2/parserc.go @@ -0,0 +1,1095 @@ +package yaml + +import ( + "bytes" +) + +// The parser implements the following grammar: +// +// stream ::= STREAM-START implicit_document? explicit_document* STREAM-END +// implicit_document ::= block_node DOCUMENT-END* +// explicit_document ::= DIRECTIVE* DOCUMENT-START block_node? DOCUMENT-END* +// block_node_or_indentless_sequence ::= +// ALIAS +// | properties (block_content | indentless_block_sequence)? +// | block_content +// | indentless_block_sequence +// block_node ::= ALIAS +// | properties block_content? +// | block_content +// flow_node ::= ALIAS +// | properties flow_content? +// | flow_content +// properties ::= TAG ANCHOR? | ANCHOR TAG? +// block_content ::= block_collection | flow_collection | SCALAR +// flow_content ::= flow_collection | SCALAR +// block_collection ::= block_sequence | block_mapping +// flow_collection ::= flow_sequence | flow_mapping +// block_sequence ::= BLOCK-SEQUENCE-START (BLOCK-ENTRY block_node?)* BLOCK-END +// indentless_sequence ::= (BLOCK-ENTRY block_node?)+ +// block_mapping ::= BLOCK-MAPPING_START +// ((KEY block_node_or_indentless_sequence?)? +// (VALUE block_node_or_indentless_sequence?)?)* +// BLOCK-END +// flow_sequence ::= FLOW-SEQUENCE-START +// (flow_sequence_entry FLOW-ENTRY)* +// flow_sequence_entry? +// FLOW-SEQUENCE-END +// flow_sequence_entry ::= flow_node | KEY flow_node? (VALUE flow_node?)? +// flow_mapping ::= FLOW-MAPPING-START +// (flow_mapping_entry FLOW-ENTRY)* +// flow_mapping_entry? +// FLOW-MAPPING-END +// flow_mapping_entry ::= flow_node | KEY flow_node? (VALUE flow_node?)? + +// Peek the next token in the token queue. +func peek_token(parser *yaml_parser_t) *yaml_token_t { + if parser.token_available || yaml_parser_fetch_more_tokens(parser) { + return &parser.tokens[parser.tokens_head] + } + return nil +} + +// Remove the next token from the queue (must be called after peek_token). +func skip_token(parser *yaml_parser_t) { + parser.token_available = false + parser.tokens_parsed++ + parser.stream_end_produced = parser.tokens[parser.tokens_head].typ == yaml_STREAM_END_TOKEN + parser.tokens_head++ +} + +// Get the next event. +func yaml_parser_parse(parser *yaml_parser_t, event *yaml_event_t) bool { + // Erase the event object. + *event = yaml_event_t{} + + // No events after the end of the stream or error. + if parser.stream_end_produced || parser.error != yaml_NO_ERROR || parser.state == yaml_PARSE_END_STATE { + return true + } + + // Generate the next event. + return yaml_parser_state_machine(parser, event) +} + +// Set parser error. +func yaml_parser_set_parser_error(parser *yaml_parser_t, problem string, problem_mark yaml_mark_t) bool { + parser.error = yaml_PARSER_ERROR + parser.problem = problem + parser.problem_mark = problem_mark + return false +} + +func yaml_parser_set_parser_error_context(parser *yaml_parser_t, context string, context_mark yaml_mark_t, problem string, problem_mark yaml_mark_t) bool { + parser.error = yaml_PARSER_ERROR + parser.context = context + parser.context_mark = context_mark + parser.problem = problem + parser.problem_mark = problem_mark + return false +} + +// State dispatcher. +func yaml_parser_state_machine(parser *yaml_parser_t, event *yaml_event_t) bool { + //trace("yaml_parser_state_machine", "state:", parser.state.String()) + + switch parser.state { + case yaml_PARSE_STREAM_START_STATE: + return yaml_parser_parse_stream_start(parser, event) + + case yaml_PARSE_IMPLICIT_DOCUMENT_START_STATE: + return yaml_parser_parse_document_start(parser, event, true) + + case yaml_PARSE_DOCUMENT_START_STATE: + return yaml_parser_parse_document_start(parser, event, false) + + case yaml_PARSE_DOCUMENT_CONTENT_STATE: + return yaml_parser_parse_document_content(parser, event) + + case yaml_PARSE_DOCUMENT_END_STATE: + return yaml_parser_parse_document_end(parser, event) + + case yaml_PARSE_BLOCK_NODE_STATE: + return yaml_parser_parse_node(parser, event, true, false) + + case yaml_PARSE_BLOCK_NODE_OR_INDENTLESS_SEQUENCE_STATE: + return yaml_parser_parse_node(parser, event, true, true) + + case yaml_PARSE_FLOW_NODE_STATE: + return yaml_parser_parse_node(parser, event, false, false) + + case yaml_PARSE_BLOCK_SEQUENCE_FIRST_ENTRY_STATE: + return yaml_parser_parse_block_sequence_entry(parser, event, true) + + case yaml_PARSE_BLOCK_SEQUENCE_ENTRY_STATE: + return yaml_parser_parse_block_sequence_entry(parser, event, false) + + case yaml_PARSE_INDENTLESS_SEQUENCE_ENTRY_STATE: + return yaml_parser_parse_indentless_sequence_entry(parser, event) + + case yaml_PARSE_BLOCK_MAPPING_FIRST_KEY_STATE: + return yaml_parser_parse_block_mapping_key(parser, event, true) + + case yaml_PARSE_BLOCK_MAPPING_KEY_STATE: + return yaml_parser_parse_block_mapping_key(parser, event, false) + + case yaml_PARSE_BLOCK_MAPPING_VALUE_STATE: + return yaml_parser_parse_block_mapping_value(parser, event) + + case yaml_PARSE_FLOW_SEQUENCE_FIRST_ENTRY_STATE: + return yaml_parser_parse_flow_sequence_entry(parser, event, true) + + case yaml_PARSE_FLOW_SEQUENCE_ENTRY_STATE: + return yaml_parser_parse_flow_sequence_entry(parser, event, false) + + case yaml_PARSE_FLOW_SEQUENCE_ENTRY_MAPPING_KEY_STATE: + return yaml_parser_parse_flow_sequence_entry_mapping_key(parser, event) + + case yaml_PARSE_FLOW_SEQUENCE_ENTRY_MAPPING_VALUE_STATE: + return yaml_parser_parse_flow_sequence_entry_mapping_value(parser, event) + + case yaml_PARSE_FLOW_SEQUENCE_ENTRY_MAPPING_END_STATE: + return yaml_parser_parse_flow_sequence_entry_mapping_end(parser, event) + + case yaml_PARSE_FLOW_MAPPING_FIRST_KEY_STATE: + return yaml_parser_parse_flow_mapping_key(parser, event, true) + + case yaml_PARSE_FLOW_MAPPING_KEY_STATE: + return yaml_parser_parse_flow_mapping_key(parser, event, false) + + case yaml_PARSE_FLOW_MAPPING_VALUE_STATE: + return yaml_parser_parse_flow_mapping_value(parser, event, false) + + case yaml_PARSE_FLOW_MAPPING_EMPTY_VALUE_STATE: + return yaml_parser_parse_flow_mapping_value(parser, event, true) + + default: + panic("invalid parser state") + } +} + +// Parse the production: +// stream ::= STREAM-START implicit_document? explicit_document* STREAM-END +// ************ +func yaml_parser_parse_stream_start(parser *yaml_parser_t, event *yaml_event_t) bool { + token := peek_token(parser) + if token == nil { + return false + } + if token.typ != yaml_STREAM_START_TOKEN { + return yaml_parser_set_parser_error(parser, "did not find expected ", token.start_mark) + } + parser.state = yaml_PARSE_IMPLICIT_DOCUMENT_START_STATE + *event = yaml_event_t{ + typ: yaml_STREAM_START_EVENT, + start_mark: token.start_mark, + end_mark: token.end_mark, + encoding: token.encoding, + } + skip_token(parser) + return true +} + +// Parse the productions: +// implicit_document ::= block_node DOCUMENT-END* +// * +// explicit_document ::= DIRECTIVE* DOCUMENT-START block_node? DOCUMENT-END* +// ************************* +func yaml_parser_parse_document_start(parser *yaml_parser_t, event *yaml_event_t, implicit bool) bool { + + token := peek_token(parser) + if token == nil { + return false + } + + // Parse extra document end indicators. + if !implicit { + for token.typ == yaml_DOCUMENT_END_TOKEN { + skip_token(parser) + token = peek_token(parser) + if token == nil { + return false + } + } + } + + if implicit && token.typ != yaml_VERSION_DIRECTIVE_TOKEN && + token.typ != yaml_TAG_DIRECTIVE_TOKEN && + token.typ != yaml_DOCUMENT_START_TOKEN && + token.typ != yaml_STREAM_END_TOKEN { + // Parse an implicit document. + if !yaml_parser_process_directives(parser, nil, nil) { + return false + } + parser.states = append(parser.states, yaml_PARSE_DOCUMENT_END_STATE) + parser.state = yaml_PARSE_BLOCK_NODE_STATE + + *event = yaml_event_t{ + typ: yaml_DOCUMENT_START_EVENT, + start_mark: token.start_mark, + end_mark: token.end_mark, + } + + } else if token.typ != yaml_STREAM_END_TOKEN { + // Parse an explicit document. + var version_directive *yaml_version_directive_t + var tag_directives []yaml_tag_directive_t + start_mark := token.start_mark + if !yaml_parser_process_directives(parser, &version_directive, &tag_directives) { + return false + } + token = peek_token(parser) + if token == nil { + return false + } + if token.typ != yaml_DOCUMENT_START_TOKEN { + yaml_parser_set_parser_error(parser, + "did not find expected ", token.start_mark) + return false + } + parser.states = append(parser.states, yaml_PARSE_DOCUMENT_END_STATE) + parser.state = yaml_PARSE_DOCUMENT_CONTENT_STATE + end_mark := token.end_mark + + *event = yaml_event_t{ + typ: yaml_DOCUMENT_START_EVENT, + start_mark: start_mark, + end_mark: end_mark, + version_directive: version_directive, + tag_directives: tag_directives, + implicit: false, + } + skip_token(parser) + + } else { + // Parse the stream end. + parser.state = yaml_PARSE_END_STATE + *event = yaml_event_t{ + typ: yaml_STREAM_END_EVENT, + start_mark: token.start_mark, + end_mark: token.end_mark, + } + skip_token(parser) + } + + return true +} + +// Parse the productions: +// explicit_document ::= DIRECTIVE* DOCUMENT-START block_node? DOCUMENT-END* +// *********** +// +func yaml_parser_parse_document_content(parser *yaml_parser_t, event *yaml_event_t) bool { + token := peek_token(parser) + if token == nil { + return false + } + if token.typ == yaml_VERSION_DIRECTIVE_TOKEN || + token.typ == yaml_TAG_DIRECTIVE_TOKEN || + token.typ == yaml_DOCUMENT_START_TOKEN || + token.typ == yaml_DOCUMENT_END_TOKEN || + token.typ == yaml_STREAM_END_TOKEN { + parser.state = parser.states[len(parser.states)-1] + parser.states = parser.states[:len(parser.states)-1] + return yaml_parser_process_empty_scalar(parser, event, + token.start_mark) + } + return yaml_parser_parse_node(parser, event, true, false) +} + +// Parse the productions: +// implicit_document ::= block_node DOCUMENT-END* +// ************* +// explicit_document ::= DIRECTIVE* DOCUMENT-START block_node? DOCUMENT-END* +// +func yaml_parser_parse_document_end(parser *yaml_parser_t, event *yaml_event_t) bool { + token := peek_token(parser) + if token == nil { + return false + } + + start_mark := token.start_mark + end_mark := token.start_mark + + implicit := true + if token.typ == yaml_DOCUMENT_END_TOKEN { + end_mark = token.end_mark + skip_token(parser) + implicit = false + } + + parser.tag_directives = parser.tag_directives[:0] + + parser.state = yaml_PARSE_DOCUMENT_START_STATE + *event = yaml_event_t{ + typ: yaml_DOCUMENT_END_EVENT, + start_mark: start_mark, + end_mark: end_mark, + implicit: implicit, + } + return true +} + +// Parse the productions: +// block_node_or_indentless_sequence ::= +// ALIAS +// ***** +// | properties (block_content | indentless_block_sequence)? +// ********** * +// | block_content | indentless_block_sequence +// * +// block_node ::= ALIAS +// ***** +// | properties block_content? +// ********** * +// | block_content +// * +// flow_node ::= ALIAS +// ***** +// | properties flow_content? +// ********** * +// | flow_content +// * +// properties ::= TAG ANCHOR? | ANCHOR TAG? +// ************************* +// block_content ::= block_collection | flow_collection | SCALAR +// ****** +// flow_content ::= flow_collection | SCALAR +// ****** +func yaml_parser_parse_node(parser *yaml_parser_t, event *yaml_event_t, block, indentless_sequence bool) bool { + //defer trace("yaml_parser_parse_node", "block:", block, "indentless_sequence:", indentless_sequence)() + + token := peek_token(parser) + if token == nil { + return false + } + + if token.typ == yaml_ALIAS_TOKEN { + parser.state = parser.states[len(parser.states)-1] + parser.states = parser.states[:len(parser.states)-1] + *event = yaml_event_t{ + typ: yaml_ALIAS_EVENT, + start_mark: token.start_mark, + end_mark: token.end_mark, + anchor: token.value, + } + skip_token(parser) + return true + } + + start_mark := token.start_mark + end_mark := token.start_mark + + var tag_token bool + var tag_handle, tag_suffix, anchor []byte + var tag_mark yaml_mark_t + if token.typ == yaml_ANCHOR_TOKEN { + anchor = token.value + start_mark = token.start_mark + end_mark = token.end_mark + skip_token(parser) + token = peek_token(parser) + if token == nil { + return false + } + if token.typ == yaml_TAG_TOKEN { + tag_token = true + tag_handle = token.value + tag_suffix = token.suffix + tag_mark = token.start_mark + end_mark = token.end_mark + skip_token(parser) + token = peek_token(parser) + if token == nil { + return false + } + } + } else if token.typ == yaml_TAG_TOKEN { + tag_token = true + tag_handle = token.value + tag_suffix = token.suffix + start_mark = token.start_mark + tag_mark = token.start_mark + end_mark = token.end_mark + skip_token(parser) + token = peek_token(parser) + if token == nil { + return false + } + if token.typ == yaml_ANCHOR_TOKEN { + anchor = token.value + end_mark = token.end_mark + skip_token(parser) + token = peek_token(parser) + if token == nil { + return false + } + } + } + + var tag []byte + if tag_token { + if len(tag_handle) == 0 { + tag = tag_suffix + tag_suffix = nil + } else { + for i := range parser.tag_directives { + if bytes.Equal(parser.tag_directives[i].handle, tag_handle) { + tag = append([]byte(nil), parser.tag_directives[i].prefix...) + tag = append(tag, tag_suffix...) + break + } + } + if len(tag) == 0 { + yaml_parser_set_parser_error_context(parser, + "while parsing a node", start_mark, + "found undefined tag handle", tag_mark) + return false + } + } + } + + implicit := len(tag) == 0 + if indentless_sequence && token.typ == yaml_BLOCK_ENTRY_TOKEN { + end_mark = token.end_mark + parser.state = yaml_PARSE_INDENTLESS_SEQUENCE_ENTRY_STATE + *event = yaml_event_t{ + typ: yaml_SEQUENCE_START_EVENT, + start_mark: start_mark, + end_mark: end_mark, + anchor: anchor, + tag: tag, + implicit: implicit, + style: yaml_style_t(yaml_BLOCK_SEQUENCE_STYLE), + } + return true + } + if token.typ == yaml_SCALAR_TOKEN { + var plain_implicit, quoted_implicit bool + end_mark = token.end_mark + if (len(tag) == 0 && token.style == yaml_PLAIN_SCALAR_STYLE) || (len(tag) == 1 && tag[0] == '!') { + plain_implicit = true + } else if len(tag) == 0 { + quoted_implicit = true + } + parser.state = parser.states[len(parser.states)-1] + parser.states = parser.states[:len(parser.states)-1] + + *event = yaml_event_t{ + typ: yaml_SCALAR_EVENT, + start_mark: start_mark, + end_mark: end_mark, + anchor: anchor, + tag: tag, + value: token.value, + implicit: plain_implicit, + quoted_implicit: quoted_implicit, + style: yaml_style_t(token.style), + } + skip_token(parser) + return true + } + if token.typ == yaml_FLOW_SEQUENCE_START_TOKEN { + // [Go] Some of the events below can be merged as they differ only on style. + end_mark = token.end_mark + parser.state = yaml_PARSE_FLOW_SEQUENCE_FIRST_ENTRY_STATE + *event = yaml_event_t{ + typ: yaml_SEQUENCE_START_EVENT, + start_mark: start_mark, + end_mark: end_mark, + anchor: anchor, + tag: tag, + implicit: implicit, + style: yaml_style_t(yaml_FLOW_SEQUENCE_STYLE), + } + return true + } + if token.typ == yaml_FLOW_MAPPING_START_TOKEN { + end_mark = token.end_mark + parser.state = yaml_PARSE_FLOW_MAPPING_FIRST_KEY_STATE + *event = yaml_event_t{ + typ: yaml_MAPPING_START_EVENT, + start_mark: start_mark, + end_mark: end_mark, + anchor: anchor, + tag: tag, + implicit: implicit, + style: yaml_style_t(yaml_FLOW_MAPPING_STYLE), + } + return true + } + if block && token.typ == yaml_BLOCK_SEQUENCE_START_TOKEN { + end_mark = token.end_mark + parser.state = yaml_PARSE_BLOCK_SEQUENCE_FIRST_ENTRY_STATE + *event = yaml_event_t{ + typ: yaml_SEQUENCE_START_EVENT, + start_mark: start_mark, + end_mark: end_mark, + anchor: anchor, + tag: tag, + implicit: implicit, + style: yaml_style_t(yaml_BLOCK_SEQUENCE_STYLE), + } + return true + } + if block && token.typ == yaml_BLOCK_MAPPING_START_TOKEN { + end_mark = token.end_mark + parser.state = yaml_PARSE_BLOCK_MAPPING_FIRST_KEY_STATE + *event = yaml_event_t{ + typ: yaml_MAPPING_START_EVENT, + start_mark: start_mark, + end_mark: end_mark, + anchor: anchor, + tag: tag, + implicit: implicit, + style: yaml_style_t(yaml_BLOCK_MAPPING_STYLE), + } + return true + } + if len(anchor) > 0 || len(tag) > 0 { + parser.state = parser.states[len(parser.states)-1] + parser.states = parser.states[:len(parser.states)-1] + + *event = yaml_event_t{ + typ: yaml_SCALAR_EVENT, + start_mark: start_mark, + end_mark: end_mark, + anchor: anchor, + tag: tag, + implicit: implicit, + quoted_implicit: false, + style: yaml_style_t(yaml_PLAIN_SCALAR_STYLE), + } + return true + } + + context := "while parsing a flow node" + if block { + context = "while parsing a block node" + } + yaml_parser_set_parser_error_context(parser, context, start_mark, + "did not find expected node content", token.start_mark) + return false +} + +// Parse the productions: +// block_sequence ::= BLOCK-SEQUENCE-START (BLOCK-ENTRY block_node?)* BLOCK-END +// ******************** *********** * ********* +// +func yaml_parser_parse_block_sequence_entry(parser *yaml_parser_t, event *yaml_event_t, first bool) bool { + if first { + token := peek_token(parser) + parser.marks = append(parser.marks, token.start_mark) + skip_token(parser) + } + + token := peek_token(parser) + if token == nil { + return false + } + + if token.typ == yaml_BLOCK_ENTRY_TOKEN { + mark := token.end_mark + skip_token(parser) + token = peek_token(parser) + if token == nil { + return false + } + if token.typ != yaml_BLOCK_ENTRY_TOKEN && token.typ != yaml_BLOCK_END_TOKEN { + parser.states = append(parser.states, yaml_PARSE_BLOCK_SEQUENCE_ENTRY_STATE) + return yaml_parser_parse_node(parser, event, true, false) + } else { + parser.state = yaml_PARSE_BLOCK_SEQUENCE_ENTRY_STATE + return yaml_parser_process_empty_scalar(parser, event, mark) + } + } + if token.typ == yaml_BLOCK_END_TOKEN { + parser.state = parser.states[len(parser.states)-1] + parser.states = parser.states[:len(parser.states)-1] + parser.marks = parser.marks[:len(parser.marks)-1] + + *event = yaml_event_t{ + typ: yaml_SEQUENCE_END_EVENT, + start_mark: token.start_mark, + end_mark: token.end_mark, + } + + skip_token(parser) + return true + } + + context_mark := parser.marks[len(parser.marks)-1] + parser.marks = parser.marks[:len(parser.marks)-1] + return yaml_parser_set_parser_error_context(parser, + "while parsing a block collection", context_mark, + "did not find expected '-' indicator", token.start_mark) +} + +// Parse the productions: +// indentless_sequence ::= (BLOCK-ENTRY block_node?)+ +// *********** * +func yaml_parser_parse_indentless_sequence_entry(parser *yaml_parser_t, event *yaml_event_t) bool { + token := peek_token(parser) + if token == nil { + return false + } + + if token.typ == yaml_BLOCK_ENTRY_TOKEN { + mark := token.end_mark + skip_token(parser) + token = peek_token(parser) + if token == nil { + return false + } + if token.typ != yaml_BLOCK_ENTRY_TOKEN && + token.typ != yaml_KEY_TOKEN && + token.typ != yaml_VALUE_TOKEN && + token.typ != yaml_BLOCK_END_TOKEN { + parser.states = append(parser.states, yaml_PARSE_INDENTLESS_SEQUENCE_ENTRY_STATE) + return yaml_parser_parse_node(parser, event, true, false) + } + parser.state = yaml_PARSE_INDENTLESS_SEQUENCE_ENTRY_STATE + return yaml_parser_process_empty_scalar(parser, event, mark) + } + parser.state = parser.states[len(parser.states)-1] + parser.states = parser.states[:len(parser.states)-1] + + *event = yaml_event_t{ + typ: yaml_SEQUENCE_END_EVENT, + start_mark: token.start_mark, + end_mark: token.start_mark, // [Go] Shouldn't this be token.end_mark? + } + return true +} + +// Parse the productions: +// block_mapping ::= BLOCK-MAPPING_START +// ******************* +// ((KEY block_node_or_indentless_sequence?)? +// *** * +// (VALUE block_node_or_indentless_sequence?)?)* +// +// BLOCK-END +// ********* +// +func yaml_parser_parse_block_mapping_key(parser *yaml_parser_t, event *yaml_event_t, first bool) bool { + if first { + token := peek_token(parser) + parser.marks = append(parser.marks, token.start_mark) + skip_token(parser) + } + + token := peek_token(parser) + if token == nil { + return false + } + + if token.typ == yaml_KEY_TOKEN { + mark := token.end_mark + skip_token(parser) + token = peek_token(parser) + if token == nil { + return false + } + if token.typ != yaml_KEY_TOKEN && + token.typ != yaml_VALUE_TOKEN && + token.typ != yaml_BLOCK_END_TOKEN { + parser.states = append(parser.states, yaml_PARSE_BLOCK_MAPPING_VALUE_STATE) + return yaml_parser_parse_node(parser, event, true, true) + } else { + parser.state = yaml_PARSE_BLOCK_MAPPING_VALUE_STATE + return yaml_parser_process_empty_scalar(parser, event, mark) + } + } else if token.typ == yaml_BLOCK_END_TOKEN { + parser.state = parser.states[len(parser.states)-1] + parser.states = parser.states[:len(parser.states)-1] + parser.marks = parser.marks[:len(parser.marks)-1] + *event = yaml_event_t{ + typ: yaml_MAPPING_END_EVENT, + start_mark: token.start_mark, + end_mark: token.end_mark, + } + skip_token(parser) + return true + } + + context_mark := parser.marks[len(parser.marks)-1] + parser.marks = parser.marks[:len(parser.marks)-1] + return yaml_parser_set_parser_error_context(parser, + "while parsing a block mapping", context_mark, + "did not find expected key", token.start_mark) +} + +// Parse the productions: +// block_mapping ::= BLOCK-MAPPING_START +// +// ((KEY block_node_or_indentless_sequence?)? +// +// (VALUE block_node_or_indentless_sequence?)?)* +// ***** * +// BLOCK-END +// +// +func yaml_parser_parse_block_mapping_value(parser *yaml_parser_t, event *yaml_event_t) bool { + token := peek_token(parser) + if token == nil { + return false + } + if token.typ == yaml_VALUE_TOKEN { + mark := token.end_mark + skip_token(parser) + token = peek_token(parser) + if token == nil { + return false + } + if token.typ != yaml_KEY_TOKEN && + token.typ != yaml_VALUE_TOKEN && + token.typ != yaml_BLOCK_END_TOKEN { + parser.states = append(parser.states, yaml_PARSE_BLOCK_MAPPING_KEY_STATE) + return yaml_parser_parse_node(parser, event, true, true) + } + parser.state = yaml_PARSE_BLOCK_MAPPING_KEY_STATE + return yaml_parser_process_empty_scalar(parser, event, mark) + } + parser.state = yaml_PARSE_BLOCK_MAPPING_KEY_STATE + return yaml_parser_process_empty_scalar(parser, event, token.start_mark) +} + +// Parse the productions: +// flow_sequence ::= FLOW-SEQUENCE-START +// ******************* +// (flow_sequence_entry FLOW-ENTRY)* +// * ********** +// flow_sequence_entry? +// * +// FLOW-SEQUENCE-END +// ***************** +// flow_sequence_entry ::= flow_node | KEY flow_node? (VALUE flow_node?)? +// * +// +func yaml_parser_parse_flow_sequence_entry(parser *yaml_parser_t, event *yaml_event_t, first bool) bool { + if first { + token := peek_token(parser) + parser.marks = append(parser.marks, token.start_mark) + skip_token(parser) + } + token := peek_token(parser) + if token == nil { + return false + } + if token.typ != yaml_FLOW_SEQUENCE_END_TOKEN { + if !first { + if token.typ == yaml_FLOW_ENTRY_TOKEN { + skip_token(parser) + token = peek_token(parser) + if token == nil { + return false + } + } else { + context_mark := parser.marks[len(parser.marks)-1] + parser.marks = parser.marks[:len(parser.marks)-1] + return yaml_parser_set_parser_error_context(parser, + "while parsing a flow sequence", context_mark, + "did not find expected ',' or ']'", token.start_mark) + } + } + + if token.typ == yaml_KEY_TOKEN { + parser.state = yaml_PARSE_FLOW_SEQUENCE_ENTRY_MAPPING_KEY_STATE + *event = yaml_event_t{ + typ: yaml_MAPPING_START_EVENT, + start_mark: token.start_mark, + end_mark: token.end_mark, + implicit: true, + style: yaml_style_t(yaml_FLOW_MAPPING_STYLE), + } + skip_token(parser) + return true + } else if token.typ != yaml_FLOW_SEQUENCE_END_TOKEN { + parser.states = append(parser.states, yaml_PARSE_FLOW_SEQUENCE_ENTRY_STATE) + return yaml_parser_parse_node(parser, event, false, false) + } + } + + parser.state = parser.states[len(parser.states)-1] + parser.states = parser.states[:len(parser.states)-1] + parser.marks = parser.marks[:len(parser.marks)-1] + + *event = yaml_event_t{ + typ: yaml_SEQUENCE_END_EVENT, + start_mark: token.start_mark, + end_mark: token.end_mark, + } + + skip_token(parser) + return true +} + +// +// Parse the productions: +// flow_sequence_entry ::= flow_node | KEY flow_node? (VALUE flow_node?)? +// *** * +// +func yaml_parser_parse_flow_sequence_entry_mapping_key(parser *yaml_parser_t, event *yaml_event_t) bool { + token := peek_token(parser) + if token == nil { + return false + } + if token.typ != yaml_VALUE_TOKEN && + token.typ != yaml_FLOW_ENTRY_TOKEN && + token.typ != yaml_FLOW_SEQUENCE_END_TOKEN { + parser.states = append(parser.states, yaml_PARSE_FLOW_SEQUENCE_ENTRY_MAPPING_VALUE_STATE) + return yaml_parser_parse_node(parser, event, false, false) + } + mark := token.end_mark + skip_token(parser) + parser.state = yaml_PARSE_FLOW_SEQUENCE_ENTRY_MAPPING_VALUE_STATE + return yaml_parser_process_empty_scalar(parser, event, mark) +} + +// Parse the productions: +// flow_sequence_entry ::= flow_node | KEY flow_node? (VALUE flow_node?)? +// ***** * +// +func yaml_parser_parse_flow_sequence_entry_mapping_value(parser *yaml_parser_t, event *yaml_event_t) bool { + token := peek_token(parser) + if token == nil { + return false + } + if token.typ == yaml_VALUE_TOKEN { + skip_token(parser) + token := peek_token(parser) + if token == nil { + return false + } + if token.typ != yaml_FLOW_ENTRY_TOKEN && token.typ != yaml_FLOW_SEQUENCE_END_TOKEN { + parser.states = append(parser.states, yaml_PARSE_FLOW_SEQUENCE_ENTRY_MAPPING_END_STATE) + return yaml_parser_parse_node(parser, event, false, false) + } + } + parser.state = yaml_PARSE_FLOW_SEQUENCE_ENTRY_MAPPING_END_STATE + return yaml_parser_process_empty_scalar(parser, event, token.start_mark) +} + +// Parse the productions: +// flow_sequence_entry ::= flow_node | KEY flow_node? (VALUE flow_node?)? +// * +// +func yaml_parser_parse_flow_sequence_entry_mapping_end(parser *yaml_parser_t, event *yaml_event_t) bool { + token := peek_token(parser) + if token == nil { + return false + } + parser.state = yaml_PARSE_FLOW_SEQUENCE_ENTRY_STATE + *event = yaml_event_t{ + typ: yaml_MAPPING_END_EVENT, + start_mark: token.start_mark, + end_mark: token.start_mark, // [Go] Shouldn't this be end_mark? + } + return true +} + +// Parse the productions: +// flow_mapping ::= FLOW-MAPPING-START +// ****************** +// (flow_mapping_entry FLOW-ENTRY)* +// * ********** +// flow_mapping_entry? +// ****************** +// FLOW-MAPPING-END +// **************** +// flow_mapping_entry ::= flow_node | KEY flow_node? (VALUE flow_node?)? +// * *** * +// +func yaml_parser_parse_flow_mapping_key(parser *yaml_parser_t, event *yaml_event_t, first bool) bool { + if first { + token := peek_token(parser) + parser.marks = append(parser.marks, token.start_mark) + skip_token(parser) + } + + token := peek_token(parser) + if token == nil { + return false + } + + if token.typ != yaml_FLOW_MAPPING_END_TOKEN { + if !first { + if token.typ == yaml_FLOW_ENTRY_TOKEN { + skip_token(parser) + token = peek_token(parser) + if token == nil { + return false + } + } else { + context_mark := parser.marks[len(parser.marks)-1] + parser.marks = parser.marks[:len(parser.marks)-1] + return yaml_parser_set_parser_error_context(parser, + "while parsing a flow mapping", context_mark, + "did not find expected ',' or '}'", token.start_mark) + } + } + + if token.typ == yaml_KEY_TOKEN { + skip_token(parser) + token = peek_token(parser) + if token == nil { + return false + } + if token.typ != yaml_VALUE_TOKEN && + token.typ != yaml_FLOW_ENTRY_TOKEN && + token.typ != yaml_FLOW_MAPPING_END_TOKEN { + parser.states = append(parser.states, yaml_PARSE_FLOW_MAPPING_VALUE_STATE) + return yaml_parser_parse_node(parser, event, false, false) + } else { + parser.state = yaml_PARSE_FLOW_MAPPING_VALUE_STATE + return yaml_parser_process_empty_scalar(parser, event, token.start_mark) + } + } else if token.typ != yaml_FLOW_MAPPING_END_TOKEN { + parser.states = append(parser.states, yaml_PARSE_FLOW_MAPPING_EMPTY_VALUE_STATE) + return yaml_parser_parse_node(parser, event, false, false) + } + } + + parser.state = parser.states[len(parser.states)-1] + parser.states = parser.states[:len(parser.states)-1] + parser.marks = parser.marks[:len(parser.marks)-1] + *event = yaml_event_t{ + typ: yaml_MAPPING_END_EVENT, + start_mark: token.start_mark, + end_mark: token.end_mark, + } + skip_token(parser) + return true +} + +// Parse the productions: +// flow_mapping_entry ::= flow_node | KEY flow_node? (VALUE flow_node?)? +// * ***** * +// +func yaml_parser_parse_flow_mapping_value(parser *yaml_parser_t, event *yaml_event_t, empty bool) bool { + token := peek_token(parser) + if token == nil { + return false + } + if empty { + parser.state = yaml_PARSE_FLOW_MAPPING_KEY_STATE + return yaml_parser_process_empty_scalar(parser, event, token.start_mark) + } + if token.typ == yaml_VALUE_TOKEN { + skip_token(parser) + token = peek_token(parser) + if token == nil { + return false + } + if token.typ != yaml_FLOW_ENTRY_TOKEN && token.typ != yaml_FLOW_MAPPING_END_TOKEN { + parser.states = append(parser.states, yaml_PARSE_FLOW_MAPPING_KEY_STATE) + return yaml_parser_parse_node(parser, event, false, false) + } + } + parser.state = yaml_PARSE_FLOW_MAPPING_KEY_STATE + return yaml_parser_process_empty_scalar(parser, event, token.start_mark) +} + +// Generate an empty scalar event. +func yaml_parser_process_empty_scalar(parser *yaml_parser_t, event *yaml_event_t, mark yaml_mark_t) bool { + *event = yaml_event_t{ + typ: yaml_SCALAR_EVENT, + start_mark: mark, + end_mark: mark, + value: nil, // Empty + implicit: true, + style: yaml_style_t(yaml_PLAIN_SCALAR_STYLE), + } + return true +} + +var default_tag_directives = []yaml_tag_directive_t{ + {[]byte("!"), []byte("!")}, + {[]byte("!!"), []byte("tag:yaml.org,2002:")}, +} + +// Parse directives. +func yaml_parser_process_directives(parser *yaml_parser_t, + version_directive_ref **yaml_version_directive_t, + tag_directives_ref *[]yaml_tag_directive_t) bool { + + var version_directive *yaml_version_directive_t + var tag_directives []yaml_tag_directive_t + + token := peek_token(parser) + if token == nil { + return false + } + + for token.typ == yaml_VERSION_DIRECTIVE_TOKEN || token.typ == yaml_TAG_DIRECTIVE_TOKEN { + if token.typ == yaml_VERSION_DIRECTIVE_TOKEN { + if version_directive != nil { + yaml_parser_set_parser_error(parser, + "found duplicate %YAML directive", token.start_mark) + return false + } + if token.major != 1 || token.minor != 1 { + yaml_parser_set_parser_error(parser, + "found incompatible YAML document", token.start_mark) + return false + } + version_directive = &yaml_version_directive_t{ + major: token.major, + minor: token.minor, + } + } else if token.typ == yaml_TAG_DIRECTIVE_TOKEN { + value := yaml_tag_directive_t{ + handle: token.value, + prefix: token.prefix, + } + if !yaml_parser_append_tag_directive(parser, value, false, token.start_mark) { + return false + } + tag_directives = append(tag_directives, value) + } + + skip_token(parser) + token = peek_token(parser) + if token == nil { + return false + } + } + + for i := range default_tag_directives { + if !yaml_parser_append_tag_directive(parser, default_tag_directives[i], true, token.start_mark) { + return false + } + } + + if version_directive_ref != nil { + *version_directive_ref = version_directive + } + if tag_directives_ref != nil { + *tag_directives_ref = tag_directives + } + return true +} + +// Append a tag directive to the directives stack. +func yaml_parser_append_tag_directive(parser *yaml_parser_t, value yaml_tag_directive_t, allow_duplicates bool, mark yaml_mark_t) bool { + for i := range parser.tag_directives { + if bytes.Equal(value.handle, parser.tag_directives[i].handle) { + if allow_duplicates { + return true + } + return yaml_parser_set_parser_error(parser, "found duplicate %TAG directive", mark) + } + } + + // [Go] I suspect the copy is unnecessary. This was likely done + // because there was no way to track ownership of the data. + value_copy := yaml_tag_directive_t{ + handle: make([]byte, len(value.handle)), + prefix: make([]byte, len(value.prefix)), + } + copy(value_copy.handle, value.handle) + copy(value_copy.prefix, value.prefix) + parser.tag_directives = append(parser.tag_directives, value_copy) + return true +} diff --git a/vendor/gopkg.in/yaml.v2/readerc.go b/vendor/gopkg.in/yaml.v2/readerc.go new file mode 100644 index 00000000..7c1f5fac --- /dev/null +++ b/vendor/gopkg.in/yaml.v2/readerc.go @@ -0,0 +1,412 @@ +package yaml + +import ( + "io" +) + +// Set the reader error and return 0. +func yaml_parser_set_reader_error(parser *yaml_parser_t, problem string, offset int, value int) bool { + parser.error = yaml_READER_ERROR + parser.problem = problem + parser.problem_offset = offset + parser.problem_value = value + return false +} + +// Byte order marks. +const ( + bom_UTF8 = "\xef\xbb\xbf" + bom_UTF16LE = "\xff\xfe" + bom_UTF16BE = "\xfe\xff" +) + +// Determine the input stream encoding by checking the BOM symbol. If no BOM is +// found, the UTF-8 encoding is assumed. Return 1 on success, 0 on failure. +func yaml_parser_determine_encoding(parser *yaml_parser_t) bool { + // Ensure that we had enough bytes in the raw buffer. + for !parser.eof && len(parser.raw_buffer)-parser.raw_buffer_pos < 3 { + if !yaml_parser_update_raw_buffer(parser) { + return false + } + } + + // Determine the encoding. + buf := parser.raw_buffer + pos := parser.raw_buffer_pos + avail := len(buf) - pos + if avail >= 2 && buf[pos] == bom_UTF16LE[0] && buf[pos+1] == bom_UTF16LE[1] { + parser.encoding = yaml_UTF16LE_ENCODING + parser.raw_buffer_pos += 2 + parser.offset += 2 + } else if avail >= 2 && buf[pos] == bom_UTF16BE[0] && buf[pos+1] == bom_UTF16BE[1] { + parser.encoding = yaml_UTF16BE_ENCODING + parser.raw_buffer_pos += 2 + parser.offset += 2 + } else if avail >= 3 && buf[pos] == bom_UTF8[0] && buf[pos+1] == bom_UTF8[1] && buf[pos+2] == bom_UTF8[2] { + parser.encoding = yaml_UTF8_ENCODING + parser.raw_buffer_pos += 3 + parser.offset += 3 + } else { + parser.encoding = yaml_UTF8_ENCODING + } + return true +} + +// Update the raw buffer. +func yaml_parser_update_raw_buffer(parser *yaml_parser_t) bool { + size_read := 0 + + // Return if the raw buffer is full. + if parser.raw_buffer_pos == 0 && len(parser.raw_buffer) == cap(parser.raw_buffer) { + return true + } + + // Return on EOF. + if parser.eof { + return true + } + + // Move the remaining bytes in the raw buffer to the beginning. + if parser.raw_buffer_pos > 0 && parser.raw_buffer_pos < len(parser.raw_buffer) { + copy(parser.raw_buffer, parser.raw_buffer[parser.raw_buffer_pos:]) + } + parser.raw_buffer = parser.raw_buffer[:len(parser.raw_buffer)-parser.raw_buffer_pos] + parser.raw_buffer_pos = 0 + + // Call the read handler to fill the buffer. + size_read, err := parser.read_handler(parser, parser.raw_buffer[len(parser.raw_buffer):cap(parser.raw_buffer)]) + parser.raw_buffer = parser.raw_buffer[:len(parser.raw_buffer)+size_read] + if err == io.EOF { + parser.eof = true + } else if err != nil { + return yaml_parser_set_reader_error(parser, "input error: "+err.Error(), parser.offset, -1) + } + return true +} + +// Ensure that the buffer contains at least `length` characters. +// Return true on success, false on failure. +// +// The length is supposed to be significantly less that the buffer size. +func yaml_parser_update_buffer(parser *yaml_parser_t, length int) bool { + if parser.read_handler == nil { + panic("read handler must be set") + } + + // [Go] This function was changed to guarantee the requested length size at EOF. + // The fact we need to do this is pretty awful, but the description above implies + // for that to be the case, and there are tests + + // If the EOF flag is set and the raw buffer is empty, do nothing. + if parser.eof && parser.raw_buffer_pos == len(parser.raw_buffer) { + // [Go] ACTUALLY! Read the documentation of this function above. + // This is just broken. To return true, we need to have the + // given length in the buffer. Not doing that means every single + // check that calls this function to make sure the buffer has a + // given length is Go) panicking; or C) accessing invalid memory. + //return true + } + + // Return if the buffer contains enough characters. + if parser.unread >= length { + return true + } + + // Determine the input encoding if it is not known yet. + if parser.encoding == yaml_ANY_ENCODING { + if !yaml_parser_determine_encoding(parser) { + return false + } + } + + // Move the unread characters to the beginning of the buffer. + buffer_len := len(parser.buffer) + if parser.buffer_pos > 0 && parser.buffer_pos < buffer_len { + copy(parser.buffer, parser.buffer[parser.buffer_pos:]) + buffer_len -= parser.buffer_pos + parser.buffer_pos = 0 + } else if parser.buffer_pos == buffer_len { + buffer_len = 0 + parser.buffer_pos = 0 + } + + // Open the whole buffer for writing, and cut it before returning. + parser.buffer = parser.buffer[:cap(parser.buffer)] + + // Fill the buffer until it has enough characters. + first := true + for parser.unread < length { + + // Fill the raw buffer if necessary. + if !first || parser.raw_buffer_pos == len(parser.raw_buffer) { + if !yaml_parser_update_raw_buffer(parser) { + parser.buffer = parser.buffer[:buffer_len] + return false + } + } + first = false + + // Decode the raw buffer. + inner: + for parser.raw_buffer_pos != len(parser.raw_buffer) { + var value rune + var width int + + raw_unread := len(parser.raw_buffer) - parser.raw_buffer_pos + + // Decode the next character. + switch parser.encoding { + case yaml_UTF8_ENCODING: + // Decode a UTF-8 character. Check RFC 3629 + // (http://www.ietf.org/rfc/rfc3629.txt) for more details. + // + // The following table (taken from the RFC) is used for + // decoding. + // + // Char. number range | UTF-8 octet sequence + // (hexadecimal) | (binary) + // --------------------+------------------------------------ + // 0000 0000-0000 007F | 0xxxxxxx + // 0000 0080-0000 07FF | 110xxxxx 10xxxxxx + // 0000 0800-0000 FFFF | 1110xxxx 10xxxxxx 10xxxxxx + // 0001 0000-0010 FFFF | 11110xxx 10xxxxxx 10xxxxxx 10xxxxxx + // + // Additionally, the characters in the range 0xD800-0xDFFF + // are prohibited as they are reserved for use with UTF-16 + // surrogate pairs. + + // Determine the length of the UTF-8 sequence. + octet := parser.raw_buffer[parser.raw_buffer_pos] + switch { + case octet&0x80 == 0x00: + width = 1 + case octet&0xE0 == 0xC0: + width = 2 + case octet&0xF0 == 0xE0: + width = 3 + case octet&0xF8 == 0xF0: + width = 4 + default: + // The leading octet is invalid. + return yaml_parser_set_reader_error(parser, + "invalid leading UTF-8 octet", + parser.offset, int(octet)) + } + + // Check if the raw buffer contains an incomplete character. + if width > raw_unread { + if parser.eof { + return yaml_parser_set_reader_error(parser, + "incomplete UTF-8 octet sequence", + parser.offset, -1) + } + break inner + } + + // Decode the leading octet. + switch { + case octet&0x80 == 0x00: + value = rune(octet & 0x7F) + case octet&0xE0 == 0xC0: + value = rune(octet & 0x1F) + case octet&0xF0 == 0xE0: + value = rune(octet & 0x0F) + case octet&0xF8 == 0xF0: + value = rune(octet & 0x07) + default: + value = 0 + } + + // Check and decode the trailing octets. + for k := 1; k < width; k++ { + octet = parser.raw_buffer[parser.raw_buffer_pos+k] + + // Check if the octet is valid. + if (octet & 0xC0) != 0x80 { + return yaml_parser_set_reader_error(parser, + "invalid trailing UTF-8 octet", + parser.offset+k, int(octet)) + } + + // Decode the octet. + value = (value << 6) + rune(octet&0x3F) + } + + // Check the length of the sequence against the value. + switch { + case width == 1: + case width == 2 && value >= 0x80: + case width == 3 && value >= 0x800: + case width == 4 && value >= 0x10000: + default: + return yaml_parser_set_reader_error(parser, + "invalid length of a UTF-8 sequence", + parser.offset, -1) + } + + // Check the range of the value. + if value >= 0xD800 && value <= 0xDFFF || value > 0x10FFFF { + return yaml_parser_set_reader_error(parser, + "invalid Unicode character", + parser.offset, int(value)) + } + + case yaml_UTF16LE_ENCODING, yaml_UTF16BE_ENCODING: + var low, high int + if parser.encoding == yaml_UTF16LE_ENCODING { + low, high = 0, 1 + } else { + low, high = 1, 0 + } + + // The UTF-16 encoding is not as simple as one might + // naively think. Check RFC 2781 + // (http://www.ietf.org/rfc/rfc2781.txt). + // + // Normally, two subsequent bytes describe a Unicode + // character. However a special technique (called a + // surrogate pair) is used for specifying character + // values larger than 0xFFFF. + // + // A surrogate pair consists of two pseudo-characters: + // high surrogate area (0xD800-0xDBFF) + // low surrogate area (0xDC00-0xDFFF) + // + // The following formulas are used for decoding + // and encoding characters using surrogate pairs: + // + // U = U' + 0x10000 (0x01 00 00 <= U <= 0x10 FF FF) + // U' = yyyyyyyyyyxxxxxxxxxx (0 <= U' <= 0x0F FF FF) + // W1 = 110110yyyyyyyyyy + // W2 = 110111xxxxxxxxxx + // + // where U is the character value, W1 is the high surrogate + // area, W2 is the low surrogate area. + + // Check for incomplete UTF-16 character. + if raw_unread < 2 { + if parser.eof { + return yaml_parser_set_reader_error(parser, + "incomplete UTF-16 character", + parser.offset, -1) + } + break inner + } + + // Get the character. + value = rune(parser.raw_buffer[parser.raw_buffer_pos+low]) + + (rune(parser.raw_buffer[parser.raw_buffer_pos+high]) << 8) + + // Check for unexpected low surrogate area. + if value&0xFC00 == 0xDC00 { + return yaml_parser_set_reader_error(parser, + "unexpected low surrogate area", + parser.offset, int(value)) + } + + // Check for a high surrogate area. + if value&0xFC00 == 0xD800 { + width = 4 + + // Check for incomplete surrogate pair. + if raw_unread < 4 { + if parser.eof { + return yaml_parser_set_reader_error(parser, + "incomplete UTF-16 surrogate pair", + parser.offset, -1) + } + break inner + } + + // Get the next character. + value2 := rune(parser.raw_buffer[parser.raw_buffer_pos+low+2]) + + (rune(parser.raw_buffer[parser.raw_buffer_pos+high+2]) << 8) + + // Check for a low surrogate area. + if value2&0xFC00 != 0xDC00 { + return yaml_parser_set_reader_error(parser, + "expected low surrogate area", + parser.offset+2, int(value2)) + } + + // Generate the value of the surrogate pair. + value = 0x10000 + ((value & 0x3FF) << 10) + (value2 & 0x3FF) + } else { + width = 2 + } + + default: + panic("impossible") + } + + // Check if the character is in the allowed range: + // #x9 | #xA | #xD | [#x20-#x7E] (8 bit) + // | #x85 | [#xA0-#xD7FF] | [#xE000-#xFFFD] (16 bit) + // | [#x10000-#x10FFFF] (32 bit) + switch { + case value == 0x09: + case value == 0x0A: + case value == 0x0D: + case value >= 0x20 && value <= 0x7E: + case value == 0x85: + case value >= 0xA0 && value <= 0xD7FF: + case value >= 0xE000 && value <= 0xFFFD: + case value >= 0x10000 && value <= 0x10FFFF: + default: + return yaml_parser_set_reader_error(parser, + "control characters are not allowed", + parser.offset, int(value)) + } + + // Move the raw pointers. + parser.raw_buffer_pos += width + parser.offset += width + + // Finally put the character into the buffer. + if value <= 0x7F { + // 0000 0000-0000 007F . 0xxxxxxx + parser.buffer[buffer_len+0] = byte(value) + buffer_len += 1 + } else if value <= 0x7FF { + // 0000 0080-0000 07FF . 110xxxxx 10xxxxxx + parser.buffer[buffer_len+0] = byte(0xC0 + (value >> 6)) + parser.buffer[buffer_len+1] = byte(0x80 + (value & 0x3F)) + buffer_len += 2 + } else if value <= 0xFFFF { + // 0000 0800-0000 FFFF . 1110xxxx 10xxxxxx 10xxxxxx + parser.buffer[buffer_len+0] = byte(0xE0 + (value >> 12)) + parser.buffer[buffer_len+1] = byte(0x80 + ((value >> 6) & 0x3F)) + parser.buffer[buffer_len+2] = byte(0x80 + (value & 0x3F)) + buffer_len += 3 + } else { + // 0001 0000-0010 FFFF . 11110xxx 10xxxxxx 10xxxxxx 10xxxxxx + parser.buffer[buffer_len+0] = byte(0xF0 + (value >> 18)) + parser.buffer[buffer_len+1] = byte(0x80 + ((value >> 12) & 0x3F)) + parser.buffer[buffer_len+2] = byte(0x80 + ((value >> 6) & 0x3F)) + parser.buffer[buffer_len+3] = byte(0x80 + (value & 0x3F)) + buffer_len += 4 + } + + parser.unread++ + } + + // On EOF, put NUL into the buffer and return. + if parser.eof { + parser.buffer[buffer_len] = 0 + buffer_len++ + parser.unread++ + break + } + } + // [Go] Read the documentation of this function above. To return true, + // we need to have the given length in the buffer. Not doing that means + // every single check that calls this function to make sure the buffer + // has a given length is Go) panicking; or C) accessing invalid memory. + // This happens here due to the EOF above breaking early. + for buffer_len < length { + parser.buffer[buffer_len] = 0 + buffer_len++ + } + parser.buffer = parser.buffer[:buffer_len] + return true +} diff --git a/vendor/gopkg.in/yaml.v2/resolve.go b/vendor/gopkg.in/yaml.v2/resolve.go new file mode 100644 index 00000000..4120e0c9 --- /dev/null +++ b/vendor/gopkg.in/yaml.v2/resolve.go @@ -0,0 +1,258 @@ +package yaml + +import ( + "encoding/base64" + "math" + "regexp" + "strconv" + "strings" + "time" +) + +type resolveMapItem struct { + value interface{} + tag string +} + +var resolveTable = make([]byte, 256) +var resolveMap = make(map[string]resolveMapItem) + +func init() { + t := resolveTable + t[int('+')] = 'S' // Sign + t[int('-')] = 'S' + for _, c := range "0123456789" { + t[int(c)] = 'D' // Digit + } + for _, c := range "yYnNtTfFoO~" { + t[int(c)] = 'M' // In map + } + t[int('.')] = '.' // Float (potentially in map) + + var resolveMapList = []struct { + v interface{} + tag string + l []string + }{ + {true, yaml_BOOL_TAG, []string{"y", "Y", "yes", "Yes", "YES"}}, + {true, yaml_BOOL_TAG, []string{"true", "True", "TRUE"}}, + {true, yaml_BOOL_TAG, []string{"on", "On", "ON"}}, + {false, yaml_BOOL_TAG, []string{"n", "N", "no", "No", "NO"}}, + {false, yaml_BOOL_TAG, []string{"false", "False", "FALSE"}}, + {false, yaml_BOOL_TAG, []string{"off", "Off", "OFF"}}, + {nil, yaml_NULL_TAG, []string{"", "~", "null", "Null", "NULL"}}, + {math.NaN(), yaml_FLOAT_TAG, []string{".nan", ".NaN", ".NAN"}}, + {math.Inf(+1), yaml_FLOAT_TAG, []string{".inf", ".Inf", ".INF"}}, + {math.Inf(+1), yaml_FLOAT_TAG, []string{"+.inf", "+.Inf", "+.INF"}}, + {math.Inf(-1), yaml_FLOAT_TAG, []string{"-.inf", "-.Inf", "-.INF"}}, + {"<<", yaml_MERGE_TAG, []string{"<<"}}, + } + + m := resolveMap + for _, item := range resolveMapList { + for _, s := range item.l { + m[s] = resolveMapItem{item.v, item.tag} + } + } +} + +const longTagPrefix = "tag:yaml.org,2002:" + +func shortTag(tag string) string { + // TODO This can easily be made faster and produce less garbage. + if strings.HasPrefix(tag, longTagPrefix) { + return "!!" + tag[len(longTagPrefix):] + } + return tag +} + +func longTag(tag string) string { + if strings.HasPrefix(tag, "!!") { + return longTagPrefix + tag[2:] + } + return tag +} + +func resolvableTag(tag string) bool { + switch tag { + case "", yaml_STR_TAG, yaml_BOOL_TAG, yaml_INT_TAG, yaml_FLOAT_TAG, yaml_NULL_TAG, yaml_TIMESTAMP_TAG: + return true + } + return false +} + +var yamlStyleFloat = regexp.MustCompile(`^[-+]?(\.[0-9]+|[0-9]+(\.[0-9]*)?)([eE][-+]?[0-9]+)?$`) + +func resolve(tag string, in string) (rtag string, out interface{}) { + if !resolvableTag(tag) { + return tag, in + } + + defer func() { + switch tag { + case "", rtag, yaml_STR_TAG, yaml_BINARY_TAG: + return + case yaml_FLOAT_TAG: + if rtag == yaml_INT_TAG { + switch v := out.(type) { + case int64: + rtag = yaml_FLOAT_TAG + out = float64(v) + return + case int: + rtag = yaml_FLOAT_TAG + out = float64(v) + return + } + } + } + failf("cannot decode %s `%s` as a %s", shortTag(rtag), in, shortTag(tag)) + }() + + // Any data is accepted as a !!str or !!binary. + // Otherwise, the prefix is enough of a hint about what it might be. + hint := byte('N') + if in != "" { + hint = resolveTable[in[0]] + } + if hint != 0 && tag != yaml_STR_TAG && tag != yaml_BINARY_TAG { + // Handle things we can lookup in a map. + if item, ok := resolveMap[in]; ok { + return item.tag, item.value + } + + // Base 60 floats are a bad idea, were dropped in YAML 1.2, and + // are purposefully unsupported here. They're still quoted on + // the way out for compatibility with other parser, though. + + switch hint { + case 'M': + // We've already checked the map above. + + case '.': + // Not in the map, so maybe a normal float. + floatv, err := strconv.ParseFloat(in, 64) + if err == nil { + return yaml_FLOAT_TAG, floatv + } + + case 'D', 'S': + // Int, float, or timestamp. + // Only try values as a timestamp if the value is unquoted or there's an explicit + // !!timestamp tag. + if tag == "" || tag == yaml_TIMESTAMP_TAG { + t, ok := parseTimestamp(in) + if ok { + return yaml_TIMESTAMP_TAG, t + } + } + + plain := strings.Replace(in, "_", "", -1) + intv, err := strconv.ParseInt(plain, 0, 64) + if err == nil { + if intv == int64(int(intv)) { + return yaml_INT_TAG, int(intv) + } else { + return yaml_INT_TAG, intv + } + } + uintv, err := strconv.ParseUint(plain, 0, 64) + if err == nil { + return yaml_INT_TAG, uintv + } + if yamlStyleFloat.MatchString(plain) { + floatv, err := strconv.ParseFloat(plain, 64) + if err == nil { + return yaml_FLOAT_TAG, floatv + } + } + if strings.HasPrefix(plain, "0b") { + intv, err := strconv.ParseInt(plain[2:], 2, 64) + if err == nil { + if intv == int64(int(intv)) { + return yaml_INT_TAG, int(intv) + } else { + return yaml_INT_TAG, intv + } + } + uintv, err := strconv.ParseUint(plain[2:], 2, 64) + if err == nil { + return yaml_INT_TAG, uintv + } + } else if strings.HasPrefix(plain, "-0b") { + intv, err := strconv.ParseInt("-" + plain[3:], 2, 64) + if err == nil { + if true || intv == int64(int(intv)) { + return yaml_INT_TAG, int(intv) + } else { + return yaml_INT_TAG, intv + } + } + } + default: + panic("resolveTable item not yet handled: " + string(rune(hint)) + " (with " + in + ")") + } + } + return yaml_STR_TAG, in +} + +// encodeBase64 encodes s as base64 that is broken up into multiple lines +// as appropriate for the resulting length. +func encodeBase64(s string) string { + const lineLen = 70 + encLen := base64.StdEncoding.EncodedLen(len(s)) + lines := encLen/lineLen + 1 + buf := make([]byte, encLen*2+lines) + in := buf[0:encLen] + out := buf[encLen:] + base64.StdEncoding.Encode(in, []byte(s)) + k := 0 + for i := 0; i < len(in); i += lineLen { + j := i + lineLen + if j > len(in) { + j = len(in) + } + k += copy(out[k:], in[i:j]) + if lines > 1 { + out[k] = '\n' + k++ + } + } + return string(out[:k]) +} + +// This is a subset of the formats allowed by the regular expression +// defined at http://yaml.org/type/timestamp.html. +var allowedTimestampFormats = []string{ + "2006-1-2T15:4:5.999999999Z07:00", // RCF3339Nano with short date fields. + "2006-1-2t15:4:5.999999999Z07:00", // RFC3339Nano with short date fields and lower-case "t". + "2006-1-2 15:4:5.999999999", // space separated with no time zone + "2006-1-2", // date only + // Notable exception: time.Parse cannot handle: "2001-12-14 21:59:43.10 -5" + // from the set of examples. +} + +// parseTimestamp parses s as a timestamp string and +// returns the timestamp and reports whether it succeeded. +// Timestamp formats are defined at http://yaml.org/type/timestamp.html +func parseTimestamp(s string) (time.Time, bool) { + // TODO write code to check all the formats supported by + // http://yaml.org/type/timestamp.html instead of using time.Parse. + + // Quick check: all date formats start with YYYY-. + i := 0 + for ; i < len(s); i++ { + if c := s[i]; c < '0' || c > '9' { + break + } + } + if i != 4 || i == len(s) || s[i] != '-' { + return time.Time{}, false + } + for _, format := range allowedTimestampFormats { + if t, err := time.Parse(format, s); err == nil { + return t, true + } + } + return time.Time{}, false +} diff --git a/vendor/gopkg.in/yaml.v2/scannerc.go b/vendor/gopkg.in/yaml.v2/scannerc.go new file mode 100644 index 00000000..0b9bb603 --- /dev/null +++ b/vendor/gopkg.in/yaml.v2/scannerc.go @@ -0,0 +1,2711 @@ +package yaml + +import ( + "bytes" + "fmt" +) + +// Introduction +// ************ +// +// The following notes assume that you are familiar with the YAML specification +// (http://yaml.org/spec/1.2/spec.html). We mostly follow it, although in +// some cases we are less restrictive that it requires. +// +// The process of transforming a YAML stream into a sequence of events is +// divided on two steps: Scanning and Parsing. +// +// The Scanner transforms the input stream into a sequence of tokens, while the +// parser transform the sequence of tokens produced by the Scanner into a +// sequence of parsing events. +// +// The Scanner is rather clever and complicated. The Parser, on the contrary, +// is a straightforward implementation of a recursive-descendant parser (or, +// LL(1) parser, as it is usually called). +// +// Actually there are two issues of Scanning that might be called "clever", the +// rest is quite straightforward. The issues are "block collection start" and +// "simple keys". Both issues are explained below in details. +// +// Here the Scanning step is explained and implemented. We start with the list +// of all the tokens produced by the Scanner together with short descriptions. +// +// Now, tokens: +// +// STREAM-START(encoding) # The stream start. +// STREAM-END # The stream end. +// VERSION-DIRECTIVE(major,minor) # The '%YAML' directive. +// TAG-DIRECTIVE(handle,prefix) # The '%TAG' directive. +// DOCUMENT-START # '---' +// DOCUMENT-END # '...' +// BLOCK-SEQUENCE-START # Indentation increase denoting a block +// BLOCK-MAPPING-START # sequence or a block mapping. +// BLOCK-END # Indentation decrease. +// FLOW-SEQUENCE-START # '[' +// FLOW-SEQUENCE-END # ']' +// BLOCK-SEQUENCE-START # '{' +// BLOCK-SEQUENCE-END # '}' +// BLOCK-ENTRY # '-' +// FLOW-ENTRY # ',' +// KEY # '?' or nothing (simple keys). +// VALUE # ':' +// ALIAS(anchor) # '*anchor' +// ANCHOR(anchor) # '&anchor' +// TAG(handle,suffix) # '!handle!suffix' +// SCALAR(value,style) # A scalar. +// +// The following two tokens are "virtual" tokens denoting the beginning and the +// end of the stream: +// +// STREAM-START(encoding) +// STREAM-END +// +// We pass the information about the input stream encoding with the +// STREAM-START token. +// +// The next two tokens are responsible for tags: +// +// VERSION-DIRECTIVE(major,minor) +// TAG-DIRECTIVE(handle,prefix) +// +// Example: +// +// %YAML 1.1 +// %TAG ! !foo +// %TAG !yaml! tag:yaml.org,2002: +// --- +// +// The correspoding sequence of tokens: +// +// STREAM-START(utf-8) +// VERSION-DIRECTIVE(1,1) +// TAG-DIRECTIVE("!","!foo") +// TAG-DIRECTIVE("!yaml","tag:yaml.org,2002:") +// DOCUMENT-START +// STREAM-END +// +// Note that the VERSION-DIRECTIVE and TAG-DIRECTIVE tokens occupy a whole +// line. +// +// The document start and end indicators are represented by: +// +// DOCUMENT-START +// DOCUMENT-END +// +// Note that if a YAML stream contains an implicit document (without '---' +// and '...' indicators), no DOCUMENT-START and DOCUMENT-END tokens will be +// produced. +// +// In the following examples, we present whole documents together with the +// produced tokens. +// +// 1. An implicit document: +// +// 'a scalar' +// +// Tokens: +// +// STREAM-START(utf-8) +// SCALAR("a scalar",single-quoted) +// STREAM-END +// +// 2. An explicit document: +// +// --- +// 'a scalar' +// ... +// +// Tokens: +// +// STREAM-START(utf-8) +// DOCUMENT-START +// SCALAR("a scalar",single-quoted) +// DOCUMENT-END +// STREAM-END +// +// 3. Several documents in a stream: +// +// 'a scalar' +// --- +// 'another scalar' +// --- +// 'yet another scalar' +// +// Tokens: +// +// STREAM-START(utf-8) +// SCALAR("a scalar",single-quoted) +// DOCUMENT-START +// SCALAR("another scalar",single-quoted) +// DOCUMENT-START +// SCALAR("yet another scalar",single-quoted) +// STREAM-END +// +// We have already introduced the SCALAR token above. The following tokens are +// used to describe aliases, anchors, tag, and scalars: +// +// ALIAS(anchor) +// ANCHOR(anchor) +// TAG(handle,suffix) +// SCALAR(value,style) +// +// The following series of examples illustrate the usage of these tokens: +// +// 1. A recursive sequence: +// +// &A [ *A ] +// +// Tokens: +// +// STREAM-START(utf-8) +// ANCHOR("A") +// FLOW-SEQUENCE-START +// ALIAS("A") +// FLOW-SEQUENCE-END +// STREAM-END +// +// 2. A tagged scalar: +// +// !!float "3.14" # A good approximation. +// +// Tokens: +// +// STREAM-START(utf-8) +// TAG("!!","float") +// SCALAR("3.14",double-quoted) +// STREAM-END +// +// 3. Various scalar styles: +// +// --- # Implicit empty plain scalars do not produce tokens. +// --- a plain scalar +// --- 'a single-quoted scalar' +// --- "a double-quoted scalar" +// --- |- +// a literal scalar +// --- >- +// a folded +// scalar +// +// Tokens: +// +// STREAM-START(utf-8) +// DOCUMENT-START +// DOCUMENT-START +// SCALAR("a plain scalar",plain) +// DOCUMENT-START +// SCALAR("a single-quoted scalar",single-quoted) +// DOCUMENT-START +// SCALAR("a double-quoted scalar",double-quoted) +// DOCUMENT-START +// SCALAR("a literal scalar",literal) +// DOCUMENT-START +// SCALAR("a folded scalar",folded) +// STREAM-END +// +// Now it's time to review collection-related tokens. We will start with +// flow collections: +// +// FLOW-SEQUENCE-START +// FLOW-SEQUENCE-END +// FLOW-MAPPING-START +// FLOW-MAPPING-END +// FLOW-ENTRY +// KEY +// VALUE +// +// The tokens FLOW-SEQUENCE-START, FLOW-SEQUENCE-END, FLOW-MAPPING-START, and +// FLOW-MAPPING-END represent the indicators '[', ']', '{', and '}' +// correspondingly. FLOW-ENTRY represent the ',' indicator. Finally the +// indicators '?' and ':', which are used for denoting mapping keys and values, +// are represented by the KEY and VALUE tokens. +// +// The following examples show flow collections: +// +// 1. A flow sequence: +// +// [item 1, item 2, item 3] +// +// Tokens: +// +// STREAM-START(utf-8) +// FLOW-SEQUENCE-START +// SCALAR("item 1",plain) +// FLOW-ENTRY +// SCALAR("item 2",plain) +// FLOW-ENTRY +// SCALAR("item 3",plain) +// FLOW-SEQUENCE-END +// STREAM-END +// +// 2. A flow mapping: +// +// { +// a simple key: a value, # Note that the KEY token is produced. +// ? a complex key: another value, +// } +// +// Tokens: +// +// STREAM-START(utf-8) +// FLOW-MAPPING-START +// KEY +// SCALAR("a simple key",plain) +// VALUE +// SCALAR("a value",plain) +// FLOW-ENTRY +// KEY +// SCALAR("a complex key",plain) +// VALUE +// SCALAR("another value",plain) +// FLOW-ENTRY +// FLOW-MAPPING-END +// STREAM-END +// +// A simple key is a key which is not denoted by the '?' indicator. Note that +// the Scanner still produce the KEY token whenever it encounters a simple key. +// +// For scanning block collections, the following tokens are used (note that we +// repeat KEY and VALUE here): +// +// BLOCK-SEQUENCE-START +// BLOCK-MAPPING-START +// BLOCK-END +// BLOCK-ENTRY +// KEY +// VALUE +// +// The tokens BLOCK-SEQUENCE-START and BLOCK-MAPPING-START denote indentation +// increase that precedes a block collection (cf. the INDENT token in Python). +// The token BLOCK-END denote indentation decrease that ends a block collection +// (cf. the DEDENT token in Python). However YAML has some syntax pecularities +// that makes detections of these tokens more complex. +// +// The tokens BLOCK-ENTRY, KEY, and VALUE are used to represent the indicators +// '-', '?', and ':' correspondingly. +// +// The following examples show how the tokens BLOCK-SEQUENCE-START, +// BLOCK-MAPPING-START, and BLOCK-END are emitted by the Scanner: +// +// 1. Block sequences: +// +// - item 1 +// - item 2 +// - +// - item 3.1 +// - item 3.2 +// - +// key 1: value 1 +// key 2: value 2 +// +// Tokens: +// +// STREAM-START(utf-8) +// BLOCK-SEQUENCE-START +// BLOCK-ENTRY +// SCALAR("item 1",plain) +// BLOCK-ENTRY +// SCALAR("item 2",plain) +// BLOCK-ENTRY +// BLOCK-SEQUENCE-START +// BLOCK-ENTRY +// SCALAR("item 3.1",plain) +// BLOCK-ENTRY +// SCALAR("item 3.2",plain) +// BLOCK-END +// BLOCK-ENTRY +// BLOCK-MAPPING-START +// KEY +// SCALAR("key 1",plain) +// VALUE +// SCALAR("value 1",plain) +// KEY +// SCALAR("key 2",plain) +// VALUE +// SCALAR("value 2",plain) +// BLOCK-END +// BLOCK-END +// STREAM-END +// +// 2. Block mappings: +// +// a simple key: a value # The KEY token is produced here. +// ? a complex key +// : another value +// a mapping: +// key 1: value 1 +// key 2: value 2 +// a sequence: +// - item 1 +// - item 2 +// +// Tokens: +// +// STREAM-START(utf-8) +// BLOCK-MAPPING-START +// KEY +// SCALAR("a simple key",plain) +// VALUE +// SCALAR("a value",plain) +// KEY +// SCALAR("a complex key",plain) +// VALUE +// SCALAR("another value",plain) +// KEY +// SCALAR("a mapping",plain) +// BLOCK-MAPPING-START +// KEY +// SCALAR("key 1",plain) +// VALUE +// SCALAR("value 1",plain) +// KEY +// SCALAR("key 2",plain) +// VALUE +// SCALAR("value 2",plain) +// BLOCK-END +// KEY +// SCALAR("a sequence",plain) +// VALUE +// BLOCK-SEQUENCE-START +// BLOCK-ENTRY +// SCALAR("item 1",plain) +// BLOCK-ENTRY +// SCALAR("item 2",plain) +// BLOCK-END +// BLOCK-END +// STREAM-END +// +// YAML does not always require to start a new block collection from a new +// line. If the current line contains only '-', '?', and ':' indicators, a new +// block collection may start at the current line. The following examples +// illustrate this case: +// +// 1. Collections in a sequence: +// +// - - item 1 +// - item 2 +// - key 1: value 1 +// key 2: value 2 +// - ? complex key +// : complex value +// +// Tokens: +// +// STREAM-START(utf-8) +// BLOCK-SEQUENCE-START +// BLOCK-ENTRY +// BLOCK-SEQUENCE-START +// BLOCK-ENTRY +// SCALAR("item 1",plain) +// BLOCK-ENTRY +// SCALAR("item 2",plain) +// BLOCK-END +// BLOCK-ENTRY +// BLOCK-MAPPING-START +// KEY +// SCALAR("key 1",plain) +// VALUE +// SCALAR("value 1",plain) +// KEY +// SCALAR("key 2",plain) +// VALUE +// SCALAR("value 2",plain) +// BLOCK-END +// BLOCK-ENTRY +// BLOCK-MAPPING-START +// KEY +// SCALAR("complex key") +// VALUE +// SCALAR("complex value") +// BLOCK-END +// BLOCK-END +// STREAM-END +// +// 2. Collections in a mapping: +// +// ? a sequence +// : - item 1 +// - item 2 +// ? a mapping +// : key 1: value 1 +// key 2: value 2 +// +// Tokens: +// +// STREAM-START(utf-8) +// BLOCK-MAPPING-START +// KEY +// SCALAR("a sequence",plain) +// VALUE +// BLOCK-SEQUENCE-START +// BLOCK-ENTRY +// SCALAR("item 1",plain) +// BLOCK-ENTRY +// SCALAR("item 2",plain) +// BLOCK-END +// KEY +// SCALAR("a mapping",plain) +// VALUE +// BLOCK-MAPPING-START +// KEY +// SCALAR("key 1",plain) +// VALUE +// SCALAR("value 1",plain) +// KEY +// SCALAR("key 2",plain) +// VALUE +// SCALAR("value 2",plain) +// BLOCK-END +// BLOCK-END +// STREAM-END +// +// YAML also permits non-indented sequences if they are included into a block +// mapping. In this case, the token BLOCK-SEQUENCE-START is not produced: +// +// key: +// - item 1 # BLOCK-SEQUENCE-START is NOT produced here. +// - item 2 +// +// Tokens: +// +// STREAM-START(utf-8) +// BLOCK-MAPPING-START +// KEY +// SCALAR("key",plain) +// VALUE +// BLOCK-ENTRY +// SCALAR("item 1",plain) +// BLOCK-ENTRY +// SCALAR("item 2",plain) +// BLOCK-END +// + +// Ensure that the buffer contains the required number of characters. +// Return true on success, false on failure (reader error or memory error). +func cache(parser *yaml_parser_t, length int) bool { + // [Go] This was inlined: !cache(A, B) -> unread < B && !update(A, B) + return parser.unread >= length || yaml_parser_update_buffer(parser, length) +} + +// Advance the buffer pointer. +func skip(parser *yaml_parser_t) { + parser.mark.index++ + parser.mark.column++ + parser.unread-- + parser.buffer_pos += width(parser.buffer[parser.buffer_pos]) +} + +func skip_line(parser *yaml_parser_t) { + if is_crlf(parser.buffer, parser.buffer_pos) { + parser.mark.index += 2 + parser.mark.column = 0 + parser.mark.line++ + parser.unread -= 2 + parser.buffer_pos += 2 + } else if is_break(parser.buffer, parser.buffer_pos) { + parser.mark.index++ + parser.mark.column = 0 + parser.mark.line++ + parser.unread-- + parser.buffer_pos += width(parser.buffer[parser.buffer_pos]) + } +} + +// Copy a character to a string buffer and advance pointers. +func read(parser *yaml_parser_t, s []byte) []byte { + w := width(parser.buffer[parser.buffer_pos]) + if w == 0 { + panic("invalid character sequence") + } + if len(s) == 0 { + s = make([]byte, 0, 32) + } + if w == 1 && len(s)+w <= cap(s) { + s = s[:len(s)+1] + s[len(s)-1] = parser.buffer[parser.buffer_pos] + parser.buffer_pos++ + } else { + s = append(s, parser.buffer[parser.buffer_pos:parser.buffer_pos+w]...) + parser.buffer_pos += w + } + parser.mark.index++ + parser.mark.column++ + parser.unread-- + return s +} + +// Copy a line break character to a string buffer and advance pointers. +func read_line(parser *yaml_parser_t, s []byte) []byte { + buf := parser.buffer + pos := parser.buffer_pos + switch { + case buf[pos] == '\r' && buf[pos+1] == '\n': + // CR LF . LF + s = append(s, '\n') + parser.buffer_pos += 2 + parser.mark.index++ + parser.unread-- + case buf[pos] == '\r' || buf[pos] == '\n': + // CR|LF . LF + s = append(s, '\n') + parser.buffer_pos += 1 + case buf[pos] == '\xC2' && buf[pos+1] == '\x85': + // NEL . LF + s = append(s, '\n') + parser.buffer_pos += 2 + case buf[pos] == '\xE2' && buf[pos+1] == '\x80' && (buf[pos+2] == '\xA8' || buf[pos+2] == '\xA9'): + // LS|PS . LS|PS + s = append(s, buf[parser.buffer_pos:pos+3]...) + parser.buffer_pos += 3 + default: + return s + } + parser.mark.index++ + parser.mark.column = 0 + parser.mark.line++ + parser.unread-- + return s +} + +// Get the next token. +func yaml_parser_scan(parser *yaml_parser_t, token *yaml_token_t) bool { + // Erase the token object. + *token = yaml_token_t{} // [Go] Is this necessary? + + // No tokens after STREAM-END or error. + if parser.stream_end_produced || parser.error != yaml_NO_ERROR { + return true + } + + // Ensure that the tokens queue contains enough tokens. + if !parser.token_available { + if !yaml_parser_fetch_more_tokens(parser) { + return false + } + } + + // Fetch the next token from the queue. + *token = parser.tokens[parser.tokens_head] + parser.tokens_head++ + parser.tokens_parsed++ + parser.token_available = false + + if token.typ == yaml_STREAM_END_TOKEN { + parser.stream_end_produced = true + } + return true +} + +// Set the scanner error and return false. +func yaml_parser_set_scanner_error(parser *yaml_parser_t, context string, context_mark yaml_mark_t, problem string) bool { + parser.error = yaml_SCANNER_ERROR + parser.context = context + parser.context_mark = context_mark + parser.problem = problem + parser.problem_mark = parser.mark + return false +} + +func yaml_parser_set_scanner_tag_error(parser *yaml_parser_t, directive bool, context_mark yaml_mark_t, problem string) bool { + context := "while parsing a tag" + if directive { + context = "while parsing a %TAG directive" + } + return yaml_parser_set_scanner_error(parser, context, context_mark, problem) +} + +func trace(args ...interface{}) func() { + pargs := append([]interface{}{"+++"}, args...) + fmt.Println(pargs...) + pargs = append([]interface{}{"---"}, args...) + return func() { fmt.Println(pargs...) } +} + +// Ensure that the tokens queue contains at least one token which can be +// returned to the Parser. +func yaml_parser_fetch_more_tokens(parser *yaml_parser_t) bool { + // While we need more tokens to fetch, do it. + for { + if parser.tokens_head != len(parser.tokens) { + // If queue is non-empty, check if any potential simple key may + // occupy the head position. + head_tok_idx, ok := parser.simple_keys_by_tok[parser.tokens_parsed] + if !ok { + break + } else if valid, ok := yaml_simple_key_is_valid(parser, &parser.simple_keys[head_tok_idx]); !ok { + return false + } else if !valid { + break + } + } + // Fetch the next token. + if !yaml_parser_fetch_next_token(parser) { + return false + } + } + + parser.token_available = true + return true +} + +// The dispatcher for token fetchers. +func yaml_parser_fetch_next_token(parser *yaml_parser_t) bool { + // Ensure that the buffer is initialized. + if parser.unread < 1 && !yaml_parser_update_buffer(parser, 1) { + return false + } + + // Check if we just started scanning. Fetch STREAM-START then. + if !parser.stream_start_produced { + return yaml_parser_fetch_stream_start(parser) + } + + // Eat whitespaces and comments until we reach the next token. + if !yaml_parser_scan_to_next_token(parser) { + return false + } + + // Check the indentation level against the current column. + if !yaml_parser_unroll_indent(parser, parser.mark.column) { + return false + } + + // Ensure that the buffer contains at least 4 characters. 4 is the length + // of the longest indicators ('--- ' and '... '). + if parser.unread < 4 && !yaml_parser_update_buffer(parser, 4) { + return false + } + + // Is it the end of the stream? + if is_z(parser.buffer, parser.buffer_pos) { + return yaml_parser_fetch_stream_end(parser) + } + + // Is it a directive? + if parser.mark.column == 0 && parser.buffer[parser.buffer_pos] == '%' { + return yaml_parser_fetch_directive(parser) + } + + buf := parser.buffer + pos := parser.buffer_pos + + // Is it the document start indicator? + if parser.mark.column == 0 && buf[pos] == '-' && buf[pos+1] == '-' && buf[pos+2] == '-' && is_blankz(buf, pos+3) { + return yaml_parser_fetch_document_indicator(parser, yaml_DOCUMENT_START_TOKEN) + } + + // Is it the document end indicator? + if parser.mark.column == 0 && buf[pos] == '.' && buf[pos+1] == '.' && buf[pos+2] == '.' && is_blankz(buf, pos+3) { + return yaml_parser_fetch_document_indicator(parser, yaml_DOCUMENT_END_TOKEN) + } + + // Is it the flow sequence start indicator? + if buf[pos] == '[' { + return yaml_parser_fetch_flow_collection_start(parser, yaml_FLOW_SEQUENCE_START_TOKEN) + } + + // Is it the flow mapping start indicator? + if parser.buffer[parser.buffer_pos] == '{' { + return yaml_parser_fetch_flow_collection_start(parser, yaml_FLOW_MAPPING_START_TOKEN) + } + + // Is it the flow sequence end indicator? + if parser.buffer[parser.buffer_pos] == ']' { + return yaml_parser_fetch_flow_collection_end(parser, + yaml_FLOW_SEQUENCE_END_TOKEN) + } + + // Is it the flow mapping end indicator? + if parser.buffer[parser.buffer_pos] == '}' { + return yaml_parser_fetch_flow_collection_end(parser, + yaml_FLOW_MAPPING_END_TOKEN) + } + + // Is it the flow entry indicator? + if parser.buffer[parser.buffer_pos] == ',' { + return yaml_parser_fetch_flow_entry(parser) + } + + // Is it the block entry indicator? + if parser.buffer[parser.buffer_pos] == '-' && is_blankz(parser.buffer, parser.buffer_pos+1) { + return yaml_parser_fetch_block_entry(parser) + } + + // Is it the key indicator? + if parser.buffer[parser.buffer_pos] == '?' && (parser.flow_level > 0 || is_blankz(parser.buffer, parser.buffer_pos+1)) { + return yaml_parser_fetch_key(parser) + } + + // Is it the value indicator? + if parser.buffer[parser.buffer_pos] == ':' && (parser.flow_level > 0 || is_blankz(parser.buffer, parser.buffer_pos+1)) { + return yaml_parser_fetch_value(parser) + } + + // Is it an alias? + if parser.buffer[parser.buffer_pos] == '*' { + return yaml_parser_fetch_anchor(parser, yaml_ALIAS_TOKEN) + } + + // Is it an anchor? + if parser.buffer[parser.buffer_pos] == '&' { + return yaml_parser_fetch_anchor(parser, yaml_ANCHOR_TOKEN) + } + + // Is it a tag? + if parser.buffer[parser.buffer_pos] == '!' { + return yaml_parser_fetch_tag(parser) + } + + // Is it a literal scalar? + if parser.buffer[parser.buffer_pos] == '|' && parser.flow_level == 0 { + return yaml_parser_fetch_block_scalar(parser, true) + } + + // Is it a folded scalar? + if parser.buffer[parser.buffer_pos] == '>' && parser.flow_level == 0 { + return yaml_parser_fetch_block_scalar(parser, false) + } + + // Is it a single-quoted scalar? + if parser.buffer[parser.buffer_pos] == '\'' { + return yaml_parser_fetch_flow_scalar(parser, true) + } + + // Is it a double-quoted scalar? + if parser.buffer[parser.buffer_pos] == '"' { + return yaml_parser_fetch_flow_scalar(parser, false) + } + + // Is it a plain scalar? + // + // A plain scalar may start with any non-blank characters except + // + // '-', '?', ':', ',', '[', ']', '{', '}', + // '#', '&', '*', '!', '|', '>', '\'', '\"', + // '%', '@', '`'. + // + // In the block context (and, for the '-' indicator, in the flow context + // too), it may also start with the characters + // + // '-', '?', ':' + // + // if it is followed by a non-space character. + // + // The last rule is more restrictive than the specification requires. + // [Go] Make this logic more reasonable. + //switch parser.buffer[parser.buffer_pos] { + //case '-', '?', ':', ',', '?', '-', ',', ':', ']', '[', '}', '{', '&', '#', '!', '*', '>', '|', '"', '\'', '@', '%', '-', '`': + //} + if !(is_blankz(parser.buffer, parser.buffer_pos) || parser.buffer[parser.buffer_pos] == '-' || + parser.buffer[parser.buffer_pos] == '?' || parser.buffer[parser.buffer_pos] == ':' || + parser.buffer[parser.buffer_pos] == ',' || parser.buffer[parser.buffer_pos] == '[' || + parser.buffer[parser.buffer_pos] == ']' || parser.buffer[parser.buffer_pos] == '{' || + parser.buffer[parser.buffer_pos] == '}' || parser.buffer[parser.buffer_pos] == '#' || + parser.buffer[parser.buffer_pos] == '&' || parser.buffer[parser.buffer_pos] == '*' || + parser.buffer[parser.buffer_pos] == '!' || parser.buffer[parser.buffer_pos] == '|' || + parser.buffer[parser.buffer_pos] == '>' || parser.buffer[parser.buffer_pos] == '\'' || + parser.buffer[parser.buffer_pos] == '"' || parser.buffer[parser.buffer_pos] == '%' || + parser.buffer[parser.buffer_pos] == '@' || parser.buffer[parser.buffer_pos] == '`') || + (parser.buffer[parser.buffer_pos] == '-' && !is_blank(parser.buffer, parser.buffer_pos+1)) || + (parser.flow_level == 0 && + (parser.buffer[parser.buffer_pos] == '?' || parser.buffer[parser.buffer_pos] == ':') && + !is_blankz(parser.buffer, parser.buffer_pos+1)) { + return yaml_parser_fetch_plain_scalar(parser) + } + + // If we don't determine the token type so far, it is an error. + return yaml_parser_set_scanner_error(parser, + "while scanning for the next token", parser.mark, + "found character that cannot start any token") +} + +func yaml_simple_key_is_valid(parser *yaml_parser_t, simple_key *yaml_simple_key_t) (valid, ok bool) { + if !simple_key.possible { + return false, true + } + + // The 1.2 specification says: + // + // "If the ? indicator is omitted, parsing needs to see past the + // implicit key to recognize it as such. To limit the amount of + // lookahead required, the “:” indicator must appear at most 1024 + // Unicode characters beyond the start of the key. In addition, the key + // is restricted to a single line." + // + if simple_key.mark.line < parser.mark.line || simple_key.mark.index+1024 < parser.mark.index { + // Check if the potential simple key to be removed is required. + if simple_key.required { + return false, yaml_parser_set_scanner_error(parser, + "while scanning a simple key", simple_key.mark, + "could not find expected ':'") + } + simple_key.possible = false + return false, true + } + return true, true +} + +// Check if a simple key may start at the current position and add it if +// needed. +func yaml_parser_save_simple_key(parser *yaml_parser_t) bool { + // A simple key is required at the current position if the scanner is in + // the block context and the current column coincides with the indentation + // level. + + required := parser.flow_level == 0 && parser.indent == parser.mark.column + + // + // If the current position may start a simple key, save it. + // + if parser.simple_key_allowed { + simple_key := yaml_simple_key_t{ + possible: true, + required: required, + token_number: parser.tokens_parsed + (len(parser.tokens) - parser.tokens_head), + mark: parser.mark, + } + + if !yaml_parser_remove_simple_key(parser) { + return false + } + parser.simple_keys[len(parser.simple_keys)-1] = simple_key + parser.simple_keys_by_tok[simple_key.token_number] = len(parser.simple_keys) - 1 + } + return true +} + +// Remove a potential simple key at the current flow level. +func yaml_parser_remove_simple_key(parser *yaml_parser_t) bool { + i := len(parser.simple_keys) - 1 + if parser.simple_keys[i].possible { + // If the key is required, it is an error. + if parser.simple_keys[i].required { + return yaml_parser_set_scanner_error(parser, + "while scanning a simple key", parser.simple_keys[i].mark, + "could not find expected ':'") + } + // Remove the key from the stack. + parser.simple_keys[i].possible = false + delete(parser.simple_keys_by_tok, parser.simple_keys[i].token_number) + } + return true +} + +// max_flow_level limits the flow_level +const max_flow_level = 10000 + +// Increase the flow level and resize the simple key list if needed. +func yaml_parser_increase_flow_level(parser *yaml_parser_t) bool { + // Reset the simple key on the next level. + parser.simple_keys = append(parser.simple_keys, yaml_simple_key_t{ + possible: false, + required: false, + token_number: parser.tokens_parsed + (len(parser.tokens) - parser.tokens_head), + mark: parser.mark, + }) + + // Increase the flow level. + parser.flow_level++ + if parser.flow_level > max_flow_level { + return yaml_parser_set_scanner_error(parser, + "while increasing flow level", parser.simple_keys[len(parser.simple_keys)-1].mark, + fmt.Sprintf("exceeded max depth of %d", max_flow_level)) + } + return true +} + +// Decrease the flow level. +func yaml_parser_decrease_flow_level(parser *yaml_parser_t) bool { + if parser.flow_level > 0 { + parser.flow_level-- + last := len(parser.simple_keys) - 1 + delete(parser.simple_keys_by_tok, parser.simple_keys[last].token_number) + parser.simple_keys = parser.simple_keys[:last] + } + return true +} + +// max_indents limits the indents stack size +const max_indents = 10000 + +// Push the current indentation level to the stack and set the new level +// the current column is greater than the indentation level. In this case, +// append or insert the specified token into the token queue. +func yaml_parser_roll_indent(parser *yaml_parser_t, column, number int, typ yaml_token_type_t, mark yaml_mark_t) bool { + // In the flow context, do nothing. + if parser.flow_level > 0 { + return true + } + + if parser.indent < column { + // Push the current indentation level to the stack and set the new + // indentation level. + parser.indents = append(parser.indents, parser.indent) + parser.indent = column + if len(parser.indents) > max_indents { + return yaml_parser_set_scanner_error(parser, + "while increasing indent level", parser.simple_keys[len(parser.simple_keys)-1].mark, + fmt.Sprintf("exceeded max depth of %d", max_indents)) + } + + // Create a token and insert it into the queue. + token := yaml_token_t{ + typ: typ, + start_mark: mark, + end_mark: mark, + } + if number > -1 { + number -= parser.tokens_parsed + } + yaml_insert_token(parser, number, &token) + } + return true +} + +// Pop indentation levels from the indents stack until the current level +// becomes less or equal to the column. For each indentation level, append +// the BLOCK-END token. +func yaml_parser_unroll_indent(parser *yaml_parser_t, column int) bool { + // In the flow context, do nothing. + if parser.flow_level > 0 { + return true + } + + // Loop through the indentation levels in the stack. + for parser.indent > column { + // Create a token and append it to the queue. + token := yaml_token_t{ + typ: yaml_BLOCK_END_TOKEN, + start_mark: parser.mark, + end_mark: parser.mark, + } + yaml_insert_token(parser, -1, &token) + + // Pop the indentation level. + parser.indent = parser.indents[len(parser.indents)-1] + parser.indents = parser.indents[:len(parser.indents)-1] + } + return true +} + +// Initialize the scanner and produce the STREAM-START token. +func yaml_parser_fetch_stream_start(parser *yaml_parser_t) bool { + + // Set the initial indentation. + parser.indent = -1 + + // Initialize the simple key stack. + parser.simple_keys = append(parser.simple_keys, yaml_simple_key_t{}) + + parser.simple_keys_by_tok = make(map[int]int) + + // A simple key is allowed at the beginning of the stream. + parser.simple_key_allowed = true + + // We have started. + parser.stream_start_produced = true + + // Create the STREAM-START token and append it to the queue. + token := yaml_token_t{ + typ: yaml_STREAM_START_TOKEN, + start_mark: parser.mark, + end_mark: parser.mark, + encoding: parser.encoding, + } + yaml_insert_token(parser, -1, &token) + return true +} + +// Produce the STREAM-END token and shut down the scanner. +func yaml_parser_fetch_stream_end(parser *yaml_parser_t) bool { + + // Force new line. + if parser.mark.column != 0 { + parser.mark.column = 0 + parser.mark.line++ + } + + // Reset the indentation level. + if !yaml_parser_unroll_indent(parser, -1) { + return false + } + + // Reset simple keys. + if !yaml_parser_remove_simple_key(parser) { + return false + } + + parser.simple_key_allowed = false + + // Create the STREAM-END token and append it to the queue. + token := yaml_token_t{ + typ: yaml_STREAM_END_TOKEN, + start_mark: parser.mark, + end_mark: parser.mark, + } + yaml_insert_token(parser, -1, &token) + return true +} + +// Produce a VERSION-DIRECTIVE or TAG-DIRECTIVE token. +func yaml_parser_fetch_directive(parser *yaml_parser_t) bool { + // Reset the indentation level. + if !yaml_parser_unroll_indent(parser, -1) { + return false + } + + // Reset simple keys. + if !yaml_parser_remove_simple_key(parser) { + return false + } + + parser.simple_key_allowed = false + + // Create the YAML-DIRECTIVE or TAG-DIRECTIVE token. + token := yaml_token_t{} + if !yaml_parser_scan_directive(parser, &token) { + return false + } + // Append the token to the queue. + yaml_insert_token(parser, -1, &token) + return true +} + +// Produce the DOCUMENT-START or DOCUMENT-END token. +func yaml_parser_fetch_document_indicator(parser *yaml_parser_t, typ yaml_token_type_t) bool { + // Reset the indentation level. + if !yaml_parser_unroll_indent(parser, -1) { + return false + } + + // Reset simple keys. + if !yaml_parser_remove_simple_key(parser) { + return false + } + + parser.simple_key_allowed = false + + // Consume the token. + start_mark := parser.mark + + skip(parser) + skip(parser) + skip(parser) + + end_mark := parser.mark + + // Create the DOCUMENT-START or DOCUMENT-END token. + token := yaml_token_t{ + typ: typ, + start_mark: start_mark, + end_mark: end_mark, + } + // Append the token to the queue. + yaml_insert_token(parser, -1, &token) + return true +} + +// Produce the FLOW-SEQUENCE-START or FLOW-MAPPING-START token. +func yaml_parser_fetch_flow_collection_start(parser *yaml_parser_t, typ yaml_token_type_t) bool { + // The indicators '[' and '{' may start a simple key. + if !yaml_parser_save_simple_key(parser) { + return false + } + + // Increase the flow level. + if !yaml_parser_increase_flow_level(parser) { + return false + } + + // A simple key may follow the indicators '[' and '{'. + parser.simple_key_allowed = true + + // Consume the token. + start_mark := parser.mark + skip(parser) + end_mark := parser.mark + + // Create the FLOW-SEQUENCE-START of FLOW-MAPPING-START token. + token := yaml_token_t{ + typ: typ, + start_mark: start_mark, + end_mark: end_mark, + } + // Append the token to the queue. + yaml_insert_token(parser, -1, &token) + return true +} + +// Produce the FLOW-SEQUENCE-END or FLOW-MAPPING-END token. +func yaml_parser_fetch_flow_collection_end(parser *yaml_parser_t, typ yaml_token_type_t) bool { + // Reset any potential simple key on the current flow level. + if !yaml_parser_remove_simple_key(parser) { + return false + } + + // Decrease the flow level. + if !yaml_parser_decrease_flow_level(parser) { + return false + } + + // No simple keys after the indicators ']' and '}'. + parser.simple_key_allowed = false + + // Consume the token. + + start_mark := parser.mark + skip(parser) + end_mark := parser.mark + + // Create the FLOW-SEQUENCE-END of FLOW-MAPPING-END token. + token := yaml_token_t{ + typ: typ, + start_mark: start_mark, + end_mark: end_mark, + } + // Append the token to the queue. + yaml_insert_token(parser, -1, &token) + return true +} + +// Produce the FLOW-ENTRY token. +func yaml_parser_fetch_flow_entry(parser *yaml_parser_t) bool { + // Reset any potential simple keys on the current flow level. + if !yaml_parser_remove_simple_key(parser) { + return false + } + + // Simple keys are allowed after ','. + parser.simple_key_allowed = true + + // Consume the token. + start_mark := parser.mark + skip(parser) + end_mark := parser.mark + + // Create the FLOW-ENTRY token and append it to the queue. + token := yaml_token_t{ + typ: yaml_FLOW_ENTRY_TOKEN, + start_mark: start_mark, + end_mark: end_mark, + } + yaml_insert_token(parser, -1, &token) + return true +} + +// Produce the BLOCK-ENTRY token. +func yaml_parser_fetch_block_entry(parser *yaml_parser_t) bool { + // Check if the scanner is in the block context. + if parser.flow_level == 0 { + // Check if we are allowed to start a new entry. + if !parser.simple_key_allowed { + return yaml_parser_set_scanner_error(parser, "", parser.mark, + "block sequence entries are not allowed in this context") + } + // Add the BLOCK-SEQUENCE-START token if needed. + if !yaml_parser_roll_indent(parser, parser.mark.column, -1, yaml_BLOCK_SEQUENCE_START_TOKEN, parser.mark) { + return false + } + } else { + // It is an error for the '-' indicator to occur in the flow context, + // but we let the Parser detect and report about it because the Parser + // is able to point to the context. + } + + // Reset any potential simple keys on the current flow level. + if !yaml_parser_remove_simple_key(parser) { + return false + } + + // Simple keys are allowed after '-'. + parser.simple_key_allowed = true + + // Consume the token. + start_mark := parser.mark + skip(parser) + end_mark := parser.mark + + // Create the BLOCK-ENTRY token and append it to the queue. + token := yaml_token_t{ + typ: yaml_BLOCK_ENTRY_TOKEN, + start_mark: start_mark, + end_mark: end_mark, + } + yaml_insert_token(parser, -1, &token) + return true +} + +// Produce the KEY token. +func yaml_parser_fetch_key(parser *yaml_parser_t) bool { + + // In the block context, additional checks are required. + if parser.flow_level == 0 { + // Check if we are allowed to start a new key (not nessesary simple). + if !parser.simple_key_allowed { + return yaml_parser_set_scanner_error(parser, "", parser.mark, + "mapping keys are not allowed in this context") + } + // Add the BLOCK-MAPPING-START token if needed. + if !yaml_parser_roll_indent(parser, parser.mark.column, -1, yaml_BLOCK_MAPPING_START_TOKEN, parser.mark) { + return false + } + } + + // Reset any potential simple keys on the current flow level. + if !yaml_parser_remove_simple_key(parser) { + return false + } + + // Simple keys are allowed after '?' in the block context. + parser.simple_key_allowed = parser.flow_level == 0 + + // Consume the token. + start_mark := parser.mark + skip(parser) + end_mark := parser.mark + + // Create the KEY token and append it to the queue. + token := yaml_token_t{ + typ: yaml_KEY_TOKEN, + start_mark: start_mark, + end_mark: end_mark, + } + yaml_insert_token(parser, -1, &token) + return true +} + +// Produce the VALUE token. +func yaml_parser_fetch_value(parser *yaml_parser_t) bool { + + simple_key := &parser.simple_keys[len(parser.simple_keys)-1] + + // Have we found a simple key? + if valid, ok := yaml_simple_key_is_valid(parser, simple_key); !ok { + return false + + } else if valid { + + // Create the KEY token and insert it into the queue. + token := yaml_token_t{ + typ: yaml_KEY_TOKEN, + start_mark: simple_key.mark, + end_mark: simple_key.mark, + } + yaml_insert_token(parser, simple_key.token_number-parser.tokens_parsed, &token) + + // In the block context, we may need to add the BLOCK-MAPPING-START token. + if !yaml_parser_roll_indent(parser, simple_key.mark.column, + simple_key.token_number, + yaml_BLOCK_MAPPING_START_TOKEN, simple_key.mark) { + return false + } + + // Remove the simple key. + simple_key.possible = false + delete(parser.simple_keys_by_tok, simple_key.token_number) + + // A simple key cannot follow another simple key. + parser.simple_key_allowed = false + + } else { + // The ':' indicator follows a complex key. + + // In the block context, extra checks are required. + if parser.flow_level == 0 { + + // Check if we are allowed to start a complex value. + if !parser.simple_key_allowed { + return yaml_parser_set_scanner_error(parser, "", parser.mark, + "mapping values are not allowed in this context") + } + + // Add the BLOCK-MAPPING-START token if needed. + if !yaml_parser_roll_indent(parser, parser.mark.column, -1, yaml_BLOCK_MAPPING_START_TOKEN, parser.mark) { + return false + } + } + + // Simple keys after ':' are allowed in the block context. + parser.simple_key_allowed = parser.flow_level == 0 + } + + // Consume the token. + start_mark := parser.mark + skip(parser) + end_mark := parser.mark + + // Create the VALUE token and append it to the queue. + token := yaml_token_t{ + typ: yaml_VALUE_TOKEN, + start_mark: start_mark, + end_mark: end_mark, + } + yaml_insert_token(parser, -1, &token) + return true +} + +// Produce the ALIAS or ANCHOR token. +func yaml_parser_fetch_anchor(parser *yaml_parser_t, typ yaml_token_type_t) bool { + // An anchor or an alias could be a simple key. + if !yaml_parser_save_simple_key(parser) { + return false + } + + // A simple key cannot follow an anchor or an alias. + parser.simple_key_allowed = false + + // Create the ALIAS or ANCHOR token and append it to the queue. + var token yaml_token_t + if !yaml_parser_scan_anchor(parser, &token, typ) { + return false + } + yaml_insert_token(parser, -1, &token) + return true +} + +// Produce the TAG token. +func yaml_parser_fetch_tag(parser *yaml_parser_t) bool { + // A tag could be a simple key. + if !yaml_parser_save_simple_key(parser) { + return false + } + + // A simple key cannot follow a tag. + parser.simple_key_allowed = false + + // Create the TAG token and append it to the queue. + var token yaml_token_t + if !yaml_parser_scan_tag(parser, &token) { + return false + } + yaml_insert_token(parser, -1, &token) + return true +} + +// Produce the SCALAR(...,literal) or SCALAR(...,folded) tokens. +func yaml_parser_fetch_block_scalar(parser *yaml_parser_t, literal bool) bool { + // Remove any potential simple keys. + if !yaml_parser_remove_simple_key(parser) { + return false + } + + // A simple key may follow a block scalar. + parser.simple_key_allowed = true + + // Create the SCALAR token and append it to the queue. + var token yaml_token_t + if !yaml_parser_scan_block_scalar(parser, &token, literal) { + return false + } + yaml_insert_token(parser, -1, &token) + return true +} + +// Produce the SCALAR(...,single-quoted) or SCALAR(...,double-quoted) tokens. +func yaml_parser_fetch_flow_scalar(parser *yaml_parser_t, single bool) bool { + // A plain scalar could be a simple key. + if !yaml_parser_save_simple_key(parser) { + return false + } + + // A simple key cannot follow a flow scalar. + parser.simple_key_allowed = false + + // Create the SCALAR token and append it to the queue. + var token yaml_token_t + if !yaml_parser_scan_flow_scalar(parser, &token, single) { + return false + } + yaml_insert_token(parser, -1, &token) + return true +} + +// Produce the SCALAR(...,plain) token. +func yaml_parser_fetch_plain_scalar(parser *yaml_parser_t) bool { + // A plain scalar could be a simple key. + if !yaml_parser_save_simple_key(parser) { + return false + } + + // A simple key cannot follow a flow scalar. + parser.simple_key_allowed = false + + // Create the SCALAR token and append it to the queue. + var token yaml_token_t + if !yaml_parser_scan_plain_scalar(parser, &token) { + return false + } + yaml_insert_token(parser, -1, &token) + return true +} + +// Eat whitespaces and comments until the next token is found. +func yaml_parser_scan_to_next_token(parser *yaml_parser_t) bool { + + // Until the next token is not found. + for { + // Allow the BOM mark to start a line. + if parser.unread < 1 && !yaml_parser_update_buffer(parser, 1) { + return false + } + if parser.mark.column == 0 && is_bom(parser.buffer, parser.buffer_pos) { + skip(parser) + } + + // Eat whitespaces. + // Tabs are allowed: + // - in the flow context + // - in the block context, but not at the beginning of the line or + // after '-', '?', or ':' (complex value). + if parser.unread < 1 && !yaml_parser_update_buffer(parser, 1) { + return false + } + + for parser.buffer[parser.buffer_pos] == ' ' || ((parser.flow_level > 0 || !parser.simple_key_allowed) && parser.buffer[parser.buffer_pos] == '\t') { + skip(parser) + if parser.unread < 1 && !yaml_parser_update_buffer(parser, 1) { + return false + } + } + + // Eat a comment until a line break. + if parser.buffer[parser.buffer_pos] == '#' { + for !is_breakz(parser.buffer, parser.buffer_pos) { + skip(parser) + if parser.unread < 1 && !yaml_parser_update_buffer(parser, 1) { + return false + } + } + } + + // If it is a line break, eat it. + if is_break(parser.buffer, parser.buffer_pos) { + if parser.unread < 2 && !yaml_parser_update_buffer(parser, 2) { + return false + } + skip_line(parser) + + // In the block context, a new line may start a simple key. + if parser.flow_level == 0 { + parser.simple_key_allowed = true + } + } else { + break // We have found a token. + } + } + + return true +} + +// Scan a YAML-DIRECTIVE or TAG-DIRECTIVE token. +// +// Scope: +// %YAML 1.1 # a comment \n +// ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ +// %TAG !yaml! tag:yaml.org,2002: \n +// ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ +// +func yaml_parser_scan_directive(parser *yaml_parser_t, token *yaml_token_t) bool { + // Eat '%'. + start_mark := parser.mark + skip(parser) + + // Scan the directive name. + var name []byte + if !yaml_parser_scan_directive_name(parser, start_mark, &name) { + return false + } + + // Is it a YAML directive? + if bytes.Equal(name, []byte("YAML")) { + // Scan the VERSION directive value. + var major, minor int8 + if !yaml_parser_scan_version_directive_value(parser, start_mark, &major, &minor) { + return false + } + end_mark := parser.mark + + // Create a VERSION-DIRECTIVE token. + *token = yaml_token_t{ + typ: yaml_VERSION_DIRECTIVE_TOKEN, + start_mark: start_mark, + end_mark: end_mark, + major: major, + minor: minor, + } + + // Is it a TAG directive? + } else if bytes.Equal(name, []byte("TAG")) { + // Scan the TAG directive value. + var handle, prefix []byte + if !yaml_parser_scan_tag_directive_value(parser, start_mark, &handle, &prefix) { + return false + } + end_mark := parser.mark + + // Create a TAG-DIRECTIVE token. + *token = yaml_token_t{ + typ: yaml_TAG_DIRECTIVE_TOKEN, + start_mark: start_mark, + end_mark: end_mark, + value: handle, + prefix: prefix, + } + + // Unknown directive. + } else { + yaml_parser_set_scanner_error(parser, "while scanning a directive", + start_mark, "found unknown directive name") + return false + } + + // Eat the rest of the line including any comments. + if parser.unread < 1 && !yaml_parser_update_buffer(parser, 1) { + return false + } + + for is_blank(parser.buffer, parser.buffer_pos) { + skip(parser) + if parser.unread < 1 && !yaml_parser_update_buffer(parser, 1) { + return false + } + } + + if parser.buffer[parser.buffer_pos] == '#' { + for !is_breakz(parser.buffer, parser.buffer_pos) { + skip(parser) + if parser.unread < 1 && !yaml_parser_update_buffer(parser, 1) { + return false + } + } + } + + // Check if we are at the end of the line. + if !is_breakz(parser.buffer, parser.buffer_pos) { + yaml_parser_set_scanner_error(parser, "while scanning a directive", + start_mark, "did not find expected comment or line break") + return false + } + + // Eat a line break. + if is_break(parser.buffer, parser.buffer_pos) { + if parser.unread < 2 && !yaml_parser_update_buffer(parser, 2) { + return false + } + skip_line(parser) + } + + return true +} + +// Scan the directive name. +// +// Scope: +// %YAML 1.1 # a comment \n +// ^^^^ +// %TAG !yaml! tag:yaml.org,2002: \n +// ^^^ +// +func yaml_parser_scan_directive_name(parser *yaml_parser_t, start_mark yaml_mark_t, name *[]byte) bool { + // Consume the directive name. + if parser.unread < 1 && !yaml_parser_update_buffer(parser, 1) { + return false + } + + var s []byte + for is_alpha(parser.buffer, parser.buffer_pos) { + s = read(parser, s) + if parser.unread < 1 && !yaml_parser_update_buffer(parser, 1) { + return false + } + } + + // Check if the name is empty. + if len(s) == 0 { + yaml_parser_set_scanner_error(parser, "while scanning a directive", + start_mark, "could not find expected directive name") + return false + } + + // Check for an blank character after the name. + if !is_blankz(parser.buffer, parser.buffer_pos) { + yaml_parser_set_scanner_error(parser, "while scanning a directive", + start_mark, "found unexpected non-alphabetical character") + return false + } + *name = s + return true +} + +// Scan the value of VERSION-DIRECTIVE. +// +// Scope: +// %YAML 1.1 # a comment \n +// ^^^^^^ +func yaml_parser_scan_version_directive_value(parser *yaml_parser_t, start_mark yaml_mark_t, major, minor *int8) bool { + // Eat whitespaces. + if parser.unread < 1 && !yaml_parser_update_buffer(parser, 1) { + return false + } + for is_blank(parser.buffer, parser.buffer_pos) { + skip(parser) + if parser.unread < 1 && !yaml_parser_update_buffer(parser, 1) { + return false + } + } + + // Consume the major version number. + if !yaml_parser_scan_version_directive_number(parser, start_mark, major) { + return false + } + + // Eat '.'. + if parser.buffer[parser.buffer_pos] != '.' { + return yaml_parser_set_scanner_error(parser, "while scanning a %YAML directive", + start_mark, "did not find expected digit or '.' character") + } + + skip(parser) + + // Consume the minor version number. + if !yaml_parser_scan_version_directive_number(parser, start_mark, minor) { + return false + } + return true +} + +const max_number_length = 2 + +// Scan the version number of VERSION-DIRECTIVE. +// +// Scope: +// %YAML 1.1 # a comment \n +// ^ +// %YAML 1.1 # a comment \n +// ^ +func yaml_parser_scan_version_directive_number(parser *yaml_parser_t, start_mark yaml_mark_t, number *int8) bool { + + // Repeat while the next character is digit. + if parser.unread < 1 && !yaml_parser_update_buffer(parser, 1) { + return false + } + var value, length int8 + for is_digit(parser.buffer, parser.buffer_pos) { + // Check if the number is too long. + length++ + if length > max_number_length { + return yaml_parser_set_scanner_error(parser, "while scanning a %YAML directive", + start_mark, "found extremely long version number") + } + value = value*10 + int8(as_digit(parser.buffer, parser.buffer_pos)) + skip(parser) + if parser.unread < 1 && !yaml_parser_update_buffer(parser, 1) { + return false + } + } + + // Check if the number was present. + if length == 0 { + return yaml_parser_set_scanner_error(parser, "while scanning a %YAML directive", + start_mark, "did not find expected version number") + } + *number = value + return true +} + +// Scan the value of a TAG-DIRECTIVE token. +// +// Scope: +// %TAG !yaml! tag:yaml.org,2002: \n +// ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ +// +func yaml_parser_scan_tag_directive_value(parser *yaml_parser_t, start_mark yaml_mark_t, handle, prefix *[]byte) bool { + var handle_value, prefix_value []byte + + // Eat whitespaces. + if parser.unread < 1 && !yaml_parser_update_buffer(parser, 1) { + return false + } + + for is_blank(parser.buffer, parser.buffer_pos) { + skip(parser) + if parser.unread < 1 && !yaml_parser_update_buffer(parser, 1) { + return false + } + } + + // Scan a handle. + if !yaml_parser_scan_tag_handle(parser, true, start_mark, &handle_value) { + return false + } + + // Expect a whitespace. + if parser.unread < 1 && !yaml_parser_update_buffer(parser, 1) { + return false + } + if !is_blank(parser.buffer, parser.buffer_pos) { + yaml_parser_set_scanner_error(parser, "while scanning a %TAG directive", + start_mark, "did not find expected whitespace") + return false + } + + // Eat whitespaces. + for is_blank(parser.buffer, parser.buffer_pos) { + skip(parser) + if parser.unread < 1 && !yaml_parser_update_buffer(parser, 1) { + return false + } + } + + // Scan a prefix. + if !yaml_parser_scan_tag_uri(parser, true, nil, start_mark, &prefix_value) { + return false + } + + // Expect a whitespace or line break. + if parser.unread < 1 && !yaml_parser_update_buffer(parser, 1) { + return false + } + if !is_blankz(parser.buffer, parser.buffer_pos) { + yaml_parser_set_scanner_error(parser, "while scanning a %TAG directive", + start_mark, "did not find expected whitespace or line break") + return false + } + + *handle = handle_value + *prefix = prefix_value + return true +} + +func yaml_parser_scan_anchor(parser *yaml_parser_t, token *yaml_token_t, typ yaml_token_type_t) bool { + var s []byte + + // Eat the indicator character. + start_mark := parser.mark + skip(parser) + + // Consume the value. + if parser.unread < 1 && !yaml_parser_update_buffer(parser, 1) { + return false + } + + for is_alpha(parser.buffer, parser.buffer_pos) { + s = read(parser, s) + if parser.unread < 1 && !yaml_parser_update_buffer(parser, 1) { + return false + } + } + + end_mark := parser.mark + + /* + * Check if length of the anchor is greater than 0 and it is followed by + * a whitespace character or one of the indicators: + * + * '?', ':', ',', ']', '}', '%', '@', '`'. + */ + + if len(s) == 0 || + !(is_blankz(parser.buffer, parser.buffer_pos) || parser.buffer[parser.buffer_pos] == '?' || + parser.buffer[parser.buffer_pos] == ':' || parser.buffer[parser.buffer_pos] == ',' || + parser.buffer[parser.buffer_pos] == ']' || parser.buffer[parser.buffer_pos] == '}' || + parser.buffer[parser.buffer_pos] == '%' || parser.buffer[parser.buffer_pos] == '@' || + parser.buffer[parser.buffer_pos] == '`') { + context := "while scanning an alias" + if typ == yaml_ANCHOR_TOKEN { + context = "while scanning an anchor" + } + yaml_parser_set_scanner_error(parser, context, start_mark, + "did not find expected alphabetic or numeric character") + return false + } + + // Create a token. + *token = yaml_token_t{ + typ: typ, + start_mark: start_mark, + end_mark: end_mark, + value: s, + } + + return true +} + +/* + * Scan a TAG token. + */ + +func yaml_parser_scan_tag(parser *yaml_parser_t, token *yaml_token_t) bool { + var handle, suffix []byte + + start_mark := parser.mark + + // Check if the tag is in the canonical form. + if parser.unread < 2 && !yaml_parser_update_buffer(parser, 2) { + return false + } + + if parser.buffer[parser.buffer_pos+1] == '<' { + // Keep the handle as '' + + // Eat '!<' + skip(parser) + skip(parser) + + // Consume the tag value. + if !yaml_parser_scan_tag_uri(parser, false, nil, start_mark, &suffix) { + return false + } + + // Check for '>' and eat it. + if parser.buffer[parser.buffer_pos] != '>' { + yaml_parser_set_scanner_error(parser, "while scanning a tag", + start_mark, "did not find the expected '>'") + return false + } + + skip(parser) + } else { + // The tag has either the '!suffix' or the '!handle!suffix' form. + + // First, try to scan a handle. + if !yaml_parser_scan_tag_handle(parser, false, start_mark, &handle) { + return false + } + + // Check if it is, indeed, handle. + if handle[0] == '!' && len(handle) > 1 && handle[len(handle)-1] == '!' { + // Scan the suffix now. + if !yaml_parser_scan_tag_uri(parser, false, nil, start_mark, &suffix) { + return false + } + } else { + // It wasn't a handle after all. Scan the rest of the tag. + if !yaml_parser_scan_tag_uri(parser, false, handle, start_mark, &suffix) { + return false + } + + // Set the handle to '!'. + handle = []byte{'!'} + + // A special case: the '!' tag. Set the handle to '' and the + // suffix to '!'. + if len(suffix) == 0 { + handle, suffix = suffix, handle + } + } + } + + // Check the character which ends the tag. + if parser.unread < 1 && !yaml_parser_update_buffer(parser, 1) { + return false + } + if !is_blankz(parser.buffer, parser.buffer_pos) { + yaml_parser_set_scanner_error(parser, "while scanning a tag", + start_mark, "did not find expected whitespace or line break") + return false + } + + end_mark := parser.mark + + // Create a token. + *token = yaml_token_t{ + typ: yaml_TAG_TOKEN, + start_mark: start_mark, + end_mark: end_mark, + value: handle, + suffix: suffix, + } + return true +} + +// Scan a tag handle. +func yaml_parser_scan_tag_handle(parser *yaml_parser_t, directive bool, start_mark yaml_mark_t, handle *[]byte) bool { + // Check the initial '!' character. + if parser.unread < 1 && !yaml_parser_update_buffer(parser, 1) { + return false + } + if parser.buffer[parser.buffer_pos] != '!' { + yaml_parser_set_scanner_tag_error(parser, directive, + start_mark, "did not find expected '!'") + return false + } + + var s []byte + + // Copy the '!' character. + s = read(parser, s) + + // Copy all subsequent alphabetical and numerical characters. + if parser.unread < 1 && !yaml_parser_update_buffer(parser, 1) { + return false + } + for is_alpha(parser.buffer, parser.buffer_pos) { + s = read(parser, s) + if parser.unread < 1 && !yaml_parser_update_buffer(parser, 1) { + return false + } + } + + // Check if the trailing character is '!' and copy it. + if parser.buffer[parser.buffer_pos] == '!' { + s = read(parser, s) + } else { + // It's either the '!' tag or not really a tag handle. If it's a %TAG + // directive, it's an error. If it's a tag token, it must be a part of URI. + if directive && string(s) != "!" { + yaml_parser_set_scanner_tag_error(parser, directive, + start_mark, "did not find expected '!'") + return false + } + } + + *handle = s + return true +} + +// Scan a tag. +func yaml_parser_scan_tag_uri(parser *yaml_parser_t, directive bool, head []byte, start_mark yaml_mark_t, uri *[]byte) bool { + //size_t length = head ? strlen((char *)head) : 0 + var s []byte + hasTag := len(head) > 0 + + // Copy the head if needed. + // + // Note that we don't copy the leading '!' character. + if len(head) > 1 { + s = append(s, head[1:]...) + } + + // Scan the tag. + if parser.unread < 1 && !yaml_parser_update_buffer(parser, 1) { + return false + } + + // The set of characters that may appear in URI is as follows: + // + // '0'-'9', 'A'-'Z', 'a'-'z', '_', '-', ';', '/', '?', ':', '@', '&', + // '=', '+', '$', ',', '.', '!', '~', '*', '\'', '(', ')', '[', ']', + // '%'. + // [Go] Convert this into more reasonable logic. + for is_alpha(parser.buffer, parser.buffer_pos) || parser.buffer[parser.buffer_pos] == ';' || + parser.buffer[parser.buffer_pos] == '/' || parser.buffer[parser.buffer_pos] == '?' || + parser.buffer[parser.buffer_pos] == ':' || parser.buffer[parser.buffer_pos] == '@' || + parser.buffer[parser.buffer_pos] == '&' || parser.buffer[parser.buffer_pos] == '=' || + parser.buffer[parser.buffer_pos] == '+' || parser.buffer[parser.buffer_pos] == '$' || + parser.buffer[parser.buffer_pos] == ',' || parser.buffer[parser.buffer_pos] == '.' || + parser.buffer[parser.buffer_pos] == '!' || parser.buffer[parser.buffer_pos] == '~' || + parser.buffer[parser.buffer_pos] == '*' || parser.buffer[parser.buffer_pos] == '\'' || + parser.buffer[parser.buffer_pos] == '(' || parser.buffer[parser.buffer_pos] == ')' || + parser.buffer[parser.buffer_pos] == '[' || parser.buffer[parser.buffer_pos] == ']' || + parser.buffer[parser.buffer_pos] == '%' { + // Check if it is a URI-escape sequence. + if parser.buffer[parser.buffer_pos] == '%' { + if !yaml_parser_scan_uri_escapes(parser, directive, start_mark, &s) { + return false + } + } else { + s = read(parser, s) + } + if parser.unread < 1 && !yaml_parser_update_buffer(parser, 1) { + return false + } + hasTag = true + } + + if !hasTag { + yaml_parser_set_scanner_tag_error(parser, directive, + start_mark, "did not find expected tag URI") + return false + } + *uri = s + return true +} + +// Decode an URI-escape sequence corresponding to a single UTF-8 character. +func yaml_parser_scan_uri_escapes(parser *yaml_parser_t, directive bool, start_mark yaml_mark_t, s *[]byte) bool { + + // Decode the required number of characters. + w := 1024 + for w > 0 { + // Check for a URI-escaped octet. + if parser.unread < 3 && !yaml_parser_update_buffer(parser, 3) { + return false + } + + if !(parser.buffer[parser.buffer_pos] == '%' && + is_hex(parser.buffer, parser.buffer_pos+1) && + is_hex(parser.buffer, parser.buffer_pos+2)) { + return yaml_parser_set_scanner_tag_error(parser, directive, + start_mark, "did not find URI escaped octet") + } + + // Get the octet. + octet := byte((as_hex(parser.buffer, parser.buffer_pos+1) << 4) + as_hex(parser.buffer, parser.buffer_pos+2)) + + // If it is the leading octet, determine the length of the UTF-8 sequence. + if w == 1024 { + w = width(octet) + if w == 0 { + return yaml_parser_set_scanner_tag_error(parser, directive, + start_mark, "found an incorrect leading UTF-8 octet") + } + } else { + // Check if the trailing octet is correct. + if octet&0xC0 != 0x80 { + return yaml_parser_set_scanner_tag_error(parser, directive, + start_mark, "found an incorrect trailing UTF-8 octet") + } + } + + // Copy the octet and move the pointers. + *s = append(*s, octet) + skip(parser) + skip(parser) + skip(parser) + w-- + } + return true +} + +// Scan a block scalar. +func yaml_parser_scan_block_scalar(parser *yaml_parser_t, token *yaml_token_t, literal bool) bool { + // Eat the indicator '|' or '>'. + start_mark := parser.mark + skip(parser) + + // Scan the additional block scalar indicators. + if parser.unread < 1 && !yaml_parser_update_buffer(parser, 1) { + return false + } + + // Check for a chomping indicator. + var chomping, increment int + if parser.buffer[parser.buffer_pos] == '+' || parser.buffer[parser.buffer_pos] == '-' { + // Set the chomping method and eat the indicator. + if parser.buffer[parser.buffer_pos] == '+' { + chomping = +1 + } else { + chomping = -1 + } + skip(parser) + + // Check for an indentation indicator. + if parser.unread < 1 && !yaml_parser_update_buffer(parser, 1) { + return false + } + if is_digit(parser.buffer, parser.buffer_pos) { + // Check that the indentation is greater than 0. + if parser.buffer[parser.buffer_pos] == '0' { + yaml_parser_set_scanner_error(parser, "while scanning a block scalar", + start_mark, "found an indentation indicator equal to 0") + return false + } + + // Get the indentation level and eat the indicator. + increment = as_digit(parser.buffer, parser.buffer_pos) + skip(parser) + } + + } else if is_digit(parser.buffer, parser.buffer_pos) { + // Do the same as above, but in the opposite order. + + if parser.buffer[parser.buffer_pos] == '0' { + yaml_parser_set_scanner_error(parser, "while scanning a block scalar", + start_mark, "found an indentation indicator equal to 0") + return false + } + increment = as_digit(parser.buffer, parser.buffer_pos) + skip(parser) + + if parser.unread < 1 && !yaml_parser_update_buffer(parser, 1) { + return false + } + if parser.buffer[parser.buffer_pos] == '+' || parser.buffer[parser.buffer_pos] == '-' { + if parser.buffer[parser.buffer_pos] == '+' { + chomping = +1 + } else { + chomping = -1 + } + skip(parser) + } + } + + // Eat whitespaces and comments to the end of the line. + if parser.unread < 1 && !yaml_parser_update_buffer(parser, 1) { + return false + } + for is_blank(parser.buffer, parser.buffer_pos) { + skip(parser) + if parser.unread < 1 && !yaml_parser_update_buffer(parser, 1) { + return false + } + } + if parser.buffer[parser.buffer_pos] == '#' { + for !is_breakz(parser.buffer, parser.buffer_pos) { + skip(parser) + if parser.unread < 1 && !yaml_parser_update_buffer(parser, 1) { + return false + } + } + } + + // Check if we are at the end of the line. + if !is_breakz(parser.buffer, parser.buffer_pos) { + yaml_parser_set_scanner_error(parser, "while scanning a block scalar", + start_mark, "did not find expected comment or line break") + return false + } + + // Eat a line break. + if is_break(parser.buffer, parser.buffer_pos) { + if parser.unread < 2 && !yaml_parser_update_buffer(parser, 2) { + return false + } + skip_line(parser) + } + + end_mark := parser.mark + + // Set the indentation level if it was specified. + var indent int + if increment > 0 { + if parser.indent >= 0 { + indent = parser.indent + increment + } else { + indent = increment + } + } + + // Scan the leading line breaks and determine the indentation level if needed. + var s, leading_break, trailing_breaks []byte + if !yaml_parser_scan_block_scalar_breaks(parser, &indent, &trailing_breaks, start_mark, &end_mark) { + return false + } + + // Scan the block scalar content. + if parser.unread < 1 && !yaml_parser_update_buffer(parser, 1) { + return false + } + var leading_blank, trailing_blank bool + for parser.mark.column == indent && !is_z(parser.buffer, parser.buffer_pos) { + // We are at the beginning of a non-empty line. + + // Is it a trailing whitespace? + trailing_blank = is_blank(parser.buffer, parser.buffer_pos) + + // Check if we need to fold the leading line break. + if !literal && !leading_blank && !trailing_blank && len(leading_break) > 0 && leading_break[0] == '\n' { + // Do we need to join the lines by space? + if len(trailing_breaks) == 0 { + s = append(s, ' ') + } + } else { + s = append(s, leading_break...) + } + leading_break = leading_break[:0] + + // Append the remaining line breaks. + s = append(s, trailing_breaks...) + trailing_breaks = trailing_breaks[:0] + + // Is it a leading whitespace? + leading_blank = is_blank(parser.buffer, parser.buffer_pos) + + // Consume the current line. + for !is_breakz(parser.buffer, parser.buffer_pos) { + s = read(parser, s) + if parser.unread < 1 && !yaml_parser_update_buffer(parser, 1) { + return false + } + } + + // Consume the line break. + if parser.unread < 2 && !yaml_parser_update_buffer(parser, 2) { + return false + } + + leading_break = read_line(parser, leading_break) + + // Eat the following indentation spaces and line breaks. + if !yaml_parser_scan_block_scalar_breaks(parser, &indent, &trailing_breaks, start_mark, &end_mark) { + return false + } + } + + // Chomp the tail. + if chomping != -1 { + s = append(s, leading_break...) + } + if chomping == 1 { + s = append(s, trailing_breaks...) + } + + // Create a token. + *token = yaml_token_t{ + typ: yaml_SCALAR_TOKEN, + start_mark: start_mark, + end_mark: end_mark, + value: s, + style: yaml_LITERAL_SCALAR_STYLE, + } + if !literal { + token.style = yaml_FOLDED_SCALAR_STYLE + } + return true +} + +// Scan indentation spaces and line breaks for a block scalar. Determine the +// indentation level if needed. +func yaml_parser_scan_block_scalar_breaks(parser *yaml_parser_t, indent *int, breaks *[]byte, start_mark yaml_mark_t, end_mark *yaml_mark_t) bool { + *end_mark = parser.mark + + // Eat the indentation spaces and line breaks. + max_indent := 0 + for { + // Eat the indentation spaces. + if parser.unread < 1 && !yaml_parser_update_buffer(parser, 1) { + return false + } + for (*indent == 0 || parser.mark.column < *indent) && is_space(parser.buffer, parser.buffer_pos) { + skip(parser) + if parser.unread < 1 && !yaml_parser_update_buffer(parser, 1) { + return false + } + } + if parser.mark.column > max_indent { + max_indent = parser.mark.column + } + + // Check for a tab character messing the indentation. + if (*indent == 0 || parser.mark.column < *indent) && is_tab(parser.buffer, parser.buffer_pos) { + return yaml_parser_set_scanner_error(parser, "while scanning a block scalar", + start_mark, "found a tab character where an indentation space is expected") + } + + // Have we found a non-empty line? + if !is_break(parser.buffer, parser.buffer_pos) { + break + } + + // Consume the line break. + if parser.unread < 2 && !yaml_parser_update_buffer(parser, 2) { + return false + } + // [Go] Should really be returning breaks instead. + *breaks = read_line(parser, *breaks) + *end_mark = parser.mark + } + + // Determine the indentation level if needed. + if *indent == 0 { + *indent = max_indent + if *indent < parser.indent+1 { + *indent = parser.indent + 1 + } + if *indent < 1 { + *indent = 1 + } + } + return true +} + +// Scan a quoted scalar. +func yaml_parser_scan_flow_scalar(parser *yaml_parser_t, token *yaml_token_t, single bool) bool { + // Eat the left quote. + start_mark := parser.mark + skip(parser) + + // Consume the content of the quoted scalar. + var s, leading_break, trailing_breaks, whitespaces []byte + for { + // Check that there are no document indicators at the beginning of the line. + if parser.unread < 4 && !yaml_parser_update_buffer(parser, 4) { + return false + } + + if parser.mark.column == 0 && + ((parser.buffer[parser.buffer_pos+0] == '-' && + parser.buffer[parser.buffer_pos+1] == '-' && + parser.buffer[parser.buffer_pos+2] == '-') || + (parser.buffer[parser.buffer_pos+0] == '.' && + parser.buffer[parser.buffer_pos+1] == '.' && + parser.buffer[parser.buffer_pos+2] == '.')) && + is_blankz(parser.buffer, parser.buffer_pos+3) { + yaml_parser_set_scanner_error(parser, "while scanning a quoted scalar", + start_mark, "found unexpected document indicator") + return false + } + + // Check for EOF. + if is_z(parser.buffer, parser.buffer_pos) { + yaml_parser_set_scanner_error(parser, "while scanning a quoted scalar", + start_mark, "found unexpected end of stream") + return false + } + + // Consume non-blank characters. + leading_blanks := false + for !is_blankz(parser.buffer, parser.buffer_pos) { + if single && parser.buffer[parser.buffer_pos] == '\'' && parser.buffer[parser.buffer_pos+1] == '\'' { + // Is is an escaped single quote. + s = append(s, '\'') + skip(parser) + skip(parser) + + } else if single && parser.buffer[parser.buffer_pos] == '\'' { + // It is a right single quote. + break + } else if !single && parser.buffer[parser.buffer_pos] == '"' { + // It is a right double quote. + break + + } else if !single && parser.buffer[parser.buffer_pos] == '\\' && is_break(parser.buffer, parser.buffer_pos+1) { + // It is an escaped line break. + if parser.unread < 3 && !yaml_parser_update_buffer(parser, 3) { + return false + } + skip(parser) + skip_line(parser) + leading_blanks = true + break + + } else if !single && parser.buffer[parser.buffer_pos] == '\\' { + // It is an escape sequence. + code_length := 0 + + // Check the escape character. + switch parser.buffer[parser.buffer_pos+1] { + case '0': + s = append(s, 0) + case 'a': + s = append(s, '\x07') + case 'b': + s = append(s, '\x08') + case 't', '\t': + s = append(s, '\x09') + case 'n': + s = append(s, '\x0A') + case 'v': + s = append(s, '\x0B') + case 'f': + s = append(s, '\x0C') + case 'r': + s = append(s, '\x0D') + case 'e': + s = append(s, '\x1B') + case ' ': + s = append(s, '\x20') + case '"': + s = append(s, '"') + case '\'': + s = append(s, '\'') + case '\\': + s = append(s, '\\') + case 'N': // NEL (#x85) + s = append(s, '\xC2') + s = append(s, '\x85') + case '_': // #xA0 + s = append(s, '\xC2') + s = append(s, '\xA0') + case 'L': // LS (#x2028) + s = append(s, '\xE2') + s = append(s, '\x80') + s = append(s, '\xA8') + case 'P': // PS (#x2029) + s = append(s, '\xE2') + s = append(s, '\x80') + s = append(s, '\xA9') + case 'x': + code_length = 2 + case 'u': + code_length = 4 + case 'U': + code_length = 8 + default: + yaml_parser_set_scanner_error(parser, "while parsing a quoted scalar", + start_mark, "found unknown escape character") + return false + } + + skip(parser) + skip(parser) + + // Consume an arbitrary escape code. + if code_length > 0 { + var value int + + // Scan the character value. + if parser.unread < code_length && !yaml_parser_update_buffer(parser, code_length) { + return false + } + for k := 0; k < code_length; k++ { + if !is_hex(parser.buffer, parser.buffer_pos+k) { + yaml_parser_set_scanner_error(parser, "while parsing a quoted scalar", + start_mark, "did not find expected hexdecimal number") + return false + } + value = (value << 4) + as_hex(parser.buffer, parser.buffer_pos+k) + } + + // Check the value and write the character. + if (value >= 0xD800 && value <= 0xDFFF) || value > 0x10FFFF { + yaml_parser_set_scanner_error(parser, "while parsing a quoted scalar", + start_mark, "found invalid Unicode character escape code") + return false + } + if value <= 0x7F { + s = append(s, byte(value)) + } else if value <= 0x7FF { + s = append(s, byte(0xC0+(value>>6))) + s = append(s, byte(0x80+(value&0x3F))) + } else if value <= 0xFFFF { + s = append(s, byte(0xE0+(value>>12))) + s = append(s, byte(0x80+((value>>6)&0x3F))) + s = append(s, byte(0x80+(value&0x3F))) + } else { + s = append(s, byte(0xF0+(value>>18))) + s = append(s, byte(0x80+((value>>12)&0x3F))) + s = append(s, byte(0x80+((value>>6)&0x3F))) + s = append(s, byte(0x80+(value&0x3F))) + } + + // Advance the pointer. + for k := 0; k < code_length; k++ { + skip(parser) + } + } + } else { + // It is a non-escaped non-blank character. + s = read(parser, s) + } + if parser.unread < 2 && !yaml_parser_update_buffer(parser, 2) { + return false + } + } + + if parser.unread < 1 && !yaml_parser_update_buffer(parser, 1) { + return false + } + + // Check if we are at the end of the scalar. + if single { + if parser.buffer[parser.buffer_pos] == '\'' { + break + } + } else { + if parser.buffer[parser.buffer_pos] == '"' { + break + } + } + + // Consume blank characters. + for is_blank(parser.buffer, parser.buffer_pos) || is_break(parser.buffer, parser.buffer_pos) { + if is_blank(parser.buffer, parser.buffer_pos) { + // Consume a space or a tab character. + if !leading_blanks { + whitespaces = read(parser, whitespaces) + } else { + skip(parser) + } + } else { + if parser.unread < 2 && !yaml_parser_update_buffer(parser, 2) { + return false + } + + // Check if it is a first line break. + if !leading_blanks { + whitespaces = whitespaces[:0] + leading_break = read_line(parser, leading_break) + leading_blanks = true + } else { + trailing_breaks = read_line(parser, trailing_breaks) + } + } + if parser.unread < 1 && !yaml_parser_update_buffer(parser, 1) { + return false + } + } + + // Join the whitespaces or fold line breaks. + if leading_blanks { + // Do we need to fold line breaks? + if len(leading_break) > 0 && leading_break[0] == '\n' { + if len(trailing_breaks) == 0 { + s = append(s, ' ') + } else { + s = append(s, trailing_breaks...) + } + } else { + s = append(s, leading_break...) + s = append(s, trailing_breaks...) + } + trailing_breaks = trailing_breaks[:0] + leading_break = leading_break[:0] + } else { + s = append(s, whitespaces...) + whitespaces = whitespaces[:0] + } + } + + // Eat the right quote. + skip(parser) + end_mark := parser.mark + + // Create a token. + *token = yaml_token_t{ + typ: yaml_SCALAR_TOKEN, + start_mark: start_mark, + end_mark: end_mark, + value: s, + style: yaml_SINGLE_QUOTED_SCALAR_STYLE, + } + if !single { + token.style = yaml_DOUBLE_QUOTED_SCALAR_STYLE + } + return true +} + +// Scan a plain scalar. +func yaml_parser_scan_plain_scalar(parser *yaml_parser_t, token *yaml_token_t) bool { + + var s, leading_break, trailing_breaks, whitespaces []byte + var leading_blanks bool + var indent = parser.indent + 1 + + start_mark := parser.mark + end_mark := parser.mark + + // Consume the content of the plain scalar. + for { + // Check for a document indicator. + if parser.unread < 4 && !yaml_parser_update_buffer(parser, 4) { + return false + } + if parser.mark.column == 0 && + ((parser.buffer[parser.buffer_pos+0] == '-' && + parser.buffer[parser.buffer_pos+1] == '-' && + parser.buffer[parser.buffer_pos+2] == '-') || + (parser.buffer[parser.buffer_pos+0] == '.' && + parser.buffer[parser.buffer_pos+1] == '.' && + parser.buffer[parser.buffer_pos+2] == '.')) && + is_blankz(parser.buffer, parser.buffer_pos+3) { + break + } + + // Check for a comment. + if parser.buffer[parser.buffer_pos] == '#' { + break + } + + // Consume non-blank characters. + for !is_blankz(parser.buffer, parser.buffer_pos) { + + // Check for indicators that may end a plain scalar. + if (parser.buffer[parser.buffer_pos] == ':' && is_blankz(parser.buffer, parser.buffer_pos+1)) || + (parser.flow_level > 0 && + (parser.buffer[parser.buffer_pos] == ',' || + parser.buffer[parser.buffer_pos] == '?' || parser.buffer[parser.buffer_pos] == '[' || + parser.buffer[parser.buffer_pos] == ']' || parser.buffer[parser.buffer_pos] == '{' || + parser.buffer[parser.buffer_pos] == '}')) { + break + } + + // Check if we need to join whitespaces and breaks. + if leading_blanks || len(whitespaces) > 0 { + if leading_blanks { + // Do we need to fold line breaks? + if leading_break[0] == '\n' { + if len(trailing_breaks) == 0 { + s = append(s, ' ') + } else { + s = append(s, trailing_breaks...) + } + } else { + s = append(s, leading_break...) + s = append(s, trailing_breaks...) + } + trailing_breaks = trailing_breaks[:0] + leading_break = leading_break[:0] + leading_blanks = false + } else { + s = append(s, whitespaces...) + whitespaces = whitespaces[:0] + } + } + + // Copy the character. + s = read(parser, s) + + end_mark = parser.mark + if parser.unread < 2 && !yaml_parser_update_buffer(parser, 2) { + return false + } + } + + // Is it the end? + if !(is_blank(parser.buffer, parser.buffer_pos) || is_break(parser.buffer, parser.buffer_pos)) { + break + } + + // Consume blank characters. + if parser.unread < 1 && !yaml_parser_update_buffer(parser, 1) { + return false + } + + for is_blank(parser.buffer, parser.buffer_pos) || is_break(parser.buffer, parser.buffer_pos) { + if is_blank(parser.buffer, parser.buffer_pos) { + + // Check for tab characters that abuse indentation. + if leading_blanks && parser.mark.column < indent && is_tab(parser.buffer, parser.buffer_pos) { + yaml_parser_set_scanner_error(parser, "while scanning a plain scalar", + start_mark, "found a tab character that violates indentation") + return false + } + + // Consume a space or a tab character. + if !leading_blanks { + whitespaces = read(parser, whitespaces) + } else { + skip(parser) + } + } else { + if parser.unread < 2 && !yaml_parser_update_buffer(parser, 2) { + return false + } + + // Check if it is a first line break. + if !leading_blanks { + whitespaces = whitespaces[:0] + leading_break = read_line(parser, leading_break) + leading_blanks = true + } else { + trailing_breaks = read_line(parser, trailing_breaks) + } + } + if parser.unread < 1 && !yaml_parser_update_buffer(parser, 1) { + return false + } + } + + // Check indentation level. + if parser.flow_level == 0 && parser.mark.column < indent { + break + } + } + + // Create a token. + *token = yaml_token_t{ + typ: yaml_SCALAR_TOKEN, + start_mark: start_mark, + end_mark: end_mark, + value: s, + style: yaml_PLAIN_SCALAR_STYLE, + } + + // Note that we change the 'simple_key_allowed' flag. + if leading_blanks { + parser.simple_key_allowed = true + } + return true +} diff --git a/vendor/gopkg.in/yaml.v2/sorter.go b/vendor/gopkg.in/yaml.v2/sorter.go new file mode 100644 index 00000000..4c45e660 --- /dev/null +++ b/vendor/gopkg.in/yaml.v2/sorter.go @@ -0,0 +1,113 @@ +package yaml + +import ( + "reflect" + "unicode" +) + +type keyList []reflect.Value + +func (l keyList) Len() int { return len(l) } +func (l keyList) Swap(i, j int) { l[i], l[j] = l[j], l[i] } +func (l keyList) Less(i, j int) bool { + a := l[i] + b := l[j] + ak := a.Kind() + bk := b.Kind() + for (ak == reflect.Interface || ak == reflect.Ptr) && !a.IsNil() { + a = a.Elem() + ak = a.Kind() + } + for (bk == reflect.Interface || bk == reflect.Ptr) && !b.IsNil() { + b = b.Elem() + bk = b.Kind() + } + af, aok := keyFloat(a) + bf, bok := keyFloat(b) + if aok && bok { + if af != bf { + return af < bf + } + if ak != bk { + return ak < bk + } + return numLess(a, b) + } + if ak != reflect.String || bk != reflect.String { + return ak < bk + } + ar, br := []rune(a.String()), []rune(b.String()) + for i := 0; i < len(ar) && i < len(br); i++ { + if ar[i] == br[i] { + continue + } + al := unicode.IsLetter(ar[i]) + bl := unicode.IsLetter(br[i]) + if al && bl { + return ar[i] < br[i] + } + if al || bl { + return bl + } + var ai, bi int + var an, bn int64 + if ar[i] == '0' || br[i] == '0' { + for j := i-1; j >= 0 && unicode.IsDigit(ar[j]); j-- { + if ar[j] != '0' { + an = 1 + bn = 1 + break + } + } + } + for ai = i; ai < len(ar) && unicode.IsDigit(ar[ai]); ai++ { + an = an*10 + int64(ar[ai]-'0') + } + for bi = i; bi < len(br) && unicode.IsDigit(br[bi]); bi++ { + bn = bn*10 + int64(br[bi]-'0') + } + if an != bn { + return an < bn + } + if ai != bi { + return ai < bi + } + return ar[i] < br[i] + } + return len(ar) < len(br) +} + +// keyFloat returns a float value for v if it is a number/bool +// and whether it is a number/bool or not. +func keyFloat(v reflect.Value) (f float64, ok bool) { + switch v.Kind() { + case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64: + return float64(v.Int()), true + case reflect.Float32, reflect.Float64: + return v.Float(), true + case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64, reflect.Uintptr: + return float64(v.Uint()), true + case reflect.Bool: + if v.Bool() { + return 1, true + } + return 0, true + } + return 0, false +} + +// numLess returns whether a < b. +// a and b must necessarily have the same kind. +func numLess(a, b reflect.Value) bool { + switch a.Kind() { + case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64: + return a.Int() < b.Int() + case reflect.Float32, reflect.Float64: + return a.Float() < b.Float() + case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64, reflect.Uintptr: + return a.Uint() < b.Uint() + case reflect.Bool: + return !a.Bool() && b.Bool() + } + panic("not a number") +} diff --git a/vendor/gopkg.in/yaml.v2/writerc.go b/vendor/gopkg.in/yaml.v2/writerc.go new file mode 100644 index 00000000..a2dde608 --- /dev/null +++ b/vendor/gopkg.in/yaml.v2/writerc.go @@ -0,0 +1,26 @@ +package yaml + +// Set the writer error and return false. +func yaml_emitter_set_writer_error(emitter *yaml_emitter_t, problem string) bool { + emitter.error = yaml_WRITER_ERROR + emitter.problem = problem + return false +} + +// Flush the output buffer. +func yaml_emitter_flush(emitter *yaml_emitter_t) bool { + if emitter.write_handler == nil { + panic("write handler not set") + } + + // Check if the buffer is empty. + if emitter.buffer_pos == 0 { + return true + } + + if err := emitter.write_handler(emitter, emitter.buffer[:emitter.buffer_pos]); err != nil { + return yaml_emitter_set_writer_error(emitter, "write error: "+err.Error()) + } + emitter.buffer_pos = 0 + return true +} diff --git a/vendor/gopkg.in/yaml.v2/yaml.go b/vendor/gopkg.in/yaml.v2/yaml.go new file mode 100644 index 00000000..30813884 --- /dev/null +++ b/vendor/gopkg.in/yaml.v2/yaml.go @@ -0,0 +1,478 @@ +// Package yaml implements YAML support for the Go language. +// +// Source code and other details for the project are available at GitHub: +// +// https://github.com/go-yaml/yaml +// +package yaml + +import ( + "errors" + "fmt" + "io" + "reflect" + "strings" + "sync" +) + +// MapSlice encodes and decodes as a YAML map. +// The order of keys is preserved when encoding and decoding. +type MapSlice []MapItem + +// MapItem is an item in a MapSlice. +type MapItem struct { + Key, Value interface{} +} + +// The Unmarshaler interface may be implemented by types to customize their +// behavior when being unmarshaled from a YAML document. The UnmarshalYAML +// method receives a function that may be called to unmarshal the original +// YAML value into a field or variable. It is safe to call the unmarshal +// function parameter more than once if necessary. +type Unmarshaler interface { + UnmarshalYAML(unmarshal func(interface{}) error) error +} + +// The Marshaler interface may be implemented by types to customize their +// behavior when being marshaled into a YAML document. The returned value +// is marshaled in place of the original value implementing Marshaler. +// +// If an error is returned by MarshalYAML, the marshaling procedure stops +// and returns with the provided error. +type Marshaler interface { + MarshalYAML() (interface{}, error) +} + +// Unmarshal decodes the first document found within the in byte slice +// and assigns decoded values into the out value. +// +// Maps and pointers (to a struct, string, int, etc) are accepted as out +// values. If an internal pointer within a struct is not initialized, +// the yaml package will initialize it if necessary for unmarshalling +// the provided data. The out parameter must not be nil. +// +// The type of the decoded values should be compatible with the respective +// values in out. If one or more values cannot be decoded due to a type +// mismatches, decoding continues partially until the end of the YAML +// content, and a *yaml.TypeError is returned with details for all +// missed values. +// +// Struct fields are only unmarshalled if they are exported (have an +// upper case first letter), and are unmarshalled using the field name +// lowercased as the default key. Custom keys may be defined via the +// "yaml" name in the field tag: the content preceding the first comma +// is used as the key, and the following comma-separated options are +// used to tweak the marshalling process (see Marshal). +// Conflicting names result in a runtime error. +// +// For example: +// +// type T struct { +// F int `yaml:"a,omitempty"` +// B int +// } +// var t T +// yaml.Unmarshal([]byte("a: 1\nb: 2"), &t) +// +// See the documentation of Marshal for the format of tags and a list of +// supported tag options. +// +func Unmarshal(in []byte, out interface{}) (err error) { + return unmarshal(in, out, false) +} + +// UnmarshalStrict is like Unmarshal except that any fields that are found +// in the data that do not have corresponding struct members, or mapping +// keys that are duplicates, will result in +// an error. +func UnmarshalStrict(in []byte, out interface{}) (err error) { + return unmarshal(in, out, true) +} + +// A Decoder reads and decodes YAML values from an input stream. +type Decoder struct { + strict bool + parser *parser +} + +// NewDecoder returns a new decoder that reads from r. +// +// The decoder introduces its own buffering and may read +// data from r beyond the YAML values requested. +func NewDecoder(r io.Reader) *Decoder { + return &Decoder{ + parser: newParserFromReader(r), + } +} + +// SetStrict sets whether strict decoding behaviour is enabled when +// decoding items in the data (see UnmarshalStrict). By default, decoding is not strict. +func (dec *Decoder) SetStrict(strict bool) { + dec.strict = strict +} + +// Decode reads the next YAML-encoded value from its input +// and stores it in the value pointed to by v. +// +// See the documentation for Unmarshal for details about the +// conversion of YAML into a Go value. +func (dec *Decoder) Decode(v interface{}) (err error) { + d := newDecoder(dec.strict) + defer handleErr(&err) + node := dec.parser.parse() + if node == nil { + return io.EOF + } + out := reflect.ValueOf(v) + if out.Kind() == reflect.Ptr && !out.IsNil() { + out = out.Elem() + } + d.unmarshal(node, out) + if len(d.terrors) > 0 { + return &TypeError{d.terrors} + } + return nil +} + +func unmarshal(in []byte, out interface{}, strict bool) (err error) { + defer handleErr(&err) + d := newDecoder(strict) + p := newParser(in) + defer p.destroy() + node := p.parse() + if node != nil { + v := reflect.ValueOf(out) + if v.Kind() == reflect.Ptr && !v.IsNil() { + v = v.Elem() + } + d.unmarshal(node, v) + } + if len(d.terrors) > 0 { + return &TypeError{d.terrors} + } + return nil +} + +// Marshal serializes the value provided into a YAML document. The structure +// of the generated document will reflect the structure of the value itself. +// Maps and pointers (to struct, string, int, etc) are accepted as the in value. +// +// Struct fields are only marshalled if they are exported (have an upper case +// first letter), and are marshalled using the field name lowercased as the +// default key. Custom keys may be defined via the "yaml" name in the field +// tag: the content preceding the first comma is used as the key, and the +// following comma-separated options are used to tweak the marshalling process. +// Conflicting names result in a runtime error. +// +// The field tag format accepted is: +// +// `(...) yaml:"[][,[,]]" (...)` +// +// The following flags are currently supported: +// +// omitempty Only include the field if it's not set to the zero +// value for the type or to empty slices or maps. +// Zero valued structs will be omitted if all their public +// fields are zero, unless they implement an IsZero +// method (see the IsZeroer interface type), in which +// case the field will be excluded if IsZero returns true. +// +// flow Marshal using a flow style (useful for structs, +// sequences and maps). +// +// inline Inline the field, which must be a struct or a map, +// causing all of its fields or keys to be processed as if +// they were part of the outer struct. For maps, keys must +// not conflict with the yaml keys of other struct fields. +// +// In addition, if the key is "-", the field is ignored. +// +// For example: +// +// type T struct { +// F int `yaml:"a,omitempty"` +// B int +// } +// yaml.Marshal(&T{B: 2}) // Returns "b: 2\n" +// yaml.Marshal(&T{F: 1}} // Returns "a: 1\nb: 0\n" +// +func Marshal(in interface{}) (out []byte, err error) { + defer handleErr(&err) + e := newEncoder() + defer e.destroy() + e.marshalDoc("", reflect.ValueOf(in)) + e.finish() + out = e.out + return +} + +// An Encoder writes YAML values to an output stream. +type Encoder struct { + encoder *encoder +} + +// NewEncoder returns a new encoder that writes to w. +// The Encoder should be closed after use to flush all data +// to w. +func NewEncoder(w io.Writer) *Encoder { + return &Encoder{ + encoder: newEncoderWithWriter(w), + } +} + +// Encode writes the YAML encoding of v to the stream. +// If multiple items are encoded to the stream, the +// second and subsequent document will be preceded +// with a "---" document separator, but the first will not. +// +// See the documentation for Marshal for details about the conversion of Go +// values to YAML. +func (e *Encoder) Encode(v interface{}) (err error) { + defer handleErr(&err) + e.encoder.marshalDoc("", reflect.ValueOf(v)) + return nil +} + +// Close closes the encoder by writing any remaining data. +// It does not write a stream terminating string "...". +func (e *Encoder) Close() (err error) { + defer handleErr(&err) + e.encoder.finish() + return nil +} + +func handleErr(err *error) { + if v := recover(); v != nil { + if e, ok := v.(yamlError); ok { + *err = e.err + } else { + panic(v) + } + } +} + +type yamlError struct { + err error +} + +func fail(err error) { + panic(yamlError{err}) +} + +func failf(format string, args ...interface{}) { + panic(yamlError{fmt.Errorf("yaml: "+format, args...)}) +} + +// A TypeError is returned by Unmarshal when one or more fields in +// the YAML document cannot be properly decoded into the requested +// types. When this error is returned, the value is still +// unmarshaled partially. +type TypeError struct { + Errors []string +} + +func (e *TypeError) Error() string { + return fmt.Sprintf("yaml: unmarshal errors:\n %s", strings.Join(e.Errors, "\n ")) +} + +// -------------------------------------------------------------------------- +// Maintain a mapping of keys to structure field indexes + +// The code in this section was copied from mgo/bson. + +// structInfo holds details for the serialization of fields of +// a given struct. +type structInfo struct { + FieldsMap map[string]fieldInfo + FieldsList []fieldInfo + + // InlineMap is the number of the field in the struct that + // contains an ,inline map, or -1 if there's none. + InlineMap int +} + +type fieldInfo struct { + Key string + Num int + OmitEmpty bool + Flow bool + // Id holds the unique field identifier, so we can cheaply + // check for field duplicates without maintaining an extra map. + Id int + + // Inline holds the field index if the field is part of an inlined struct. + Inline []int +} + +var structMap = make(map[reflect.Type]*structInfo) +var fieldMapMutex sync.RWMutex + +func getStructInfo(st reflect.Type) (*structInfo, error) { + fieldMapMutex.RLock() + sinfo, found := structMap[st] + fieldMapMutex.RUnlock() + if found { + return sinfo, nil + } + + n := st.NumField() + fieldsMap := make(map[string]fieldInfo) + fieldsList := make([]fieldInfo, 0, n) + inlineMap := -1 + for i := 0; i != n; i++ { + field := st.Field(i) + if field.PkgPath != "" && !field.Anonymous { + continue // Private field + } + + info := fieldInfo{Num: i} + + tag := field.Tag.Get("yaml") + if tag == "" && strings.Index(string(field.Tag), ":") < 0 { + tag = string(field.Tag) + } + if tag == "-" { + continue + } + + inline := false + fields := strings.Split(tag, ",") + if len(fields) > 1 { + for _, flag := range fields[1:] { + switch flag { + case "omitempty": + info.OmitEmpty = true + case "flow": + info.Flow = true + case "inline": + inline = true + default: + return nil, errors.New(fmt.Sprintf("Unsupported flag %q in tag %q of type %s", flag, tag, st)) + } + } + tag = fields[0] + } + + if inline { + switch field.Type.Kind() { + case reflect.Map: + if inlineMap >= 0 { + return nil, errors.New("Multiple ,inline maps in struct " + st.String()) + } + if field.Type.Key() != reflect.TypeOf("") { + return nil, errors.New("Option ,inline needs a map with string keys in struct " + st.String()) + } + inlineMap = info.Num + case reflect.Struct: + sinfo, err := getStructInfo(field.Type) + if err != nil { + return nil, err + } + for _, finfo := range sinfo.FieldsList { + if _, found := fieldsMap[finfo.Key]; found { + msg := "Duplicated key '" + finfo.Key + "' in struct " + st.String() + return nil, errors.New(msg) + } + if finfo.Inline == nil { + finfo.Inline = []int{i, finfo.Num} + } else { + finfo.Inline = append([]int{i}, finfo.Inline...) + } + finfo.Id = len(fieldsList) + fieldsMap[finfo.Key] = finfo + fieldsList = append(fieldsList, finfo) + } + default: + //return nil, errors.New("Option ,inline needs a struct value or map field") + return nil, errors.New("Option ,inline needs a struct value field") + } + continue + } + + if tag != "" { + info.Key = tag + } else { + info.Key = strings.ToLower(field.Name) + } + + if _, found = fieldsMap[info.Key]; found { + msg := "Duplicated key '" + info.Key + "' in struct " + st.String() + return nil, errors.New(msg) + } + + info.Id = len(fieldsList) + fieldsList = append(fieldsList, info) + fieldsMap[info.Key] = info + } + + sinfo = &structInfo{ + FieldsMap: fieldsMap, + FieldsList: fieldsList, + InlineMap: inlineMap, + } + + fieldMapMutex.Lock() + structMap[st] = sinfo + fieldMapMutex.Unlock() + return sinfo, nil +} + +// IsZeroer is used to check whether an object is zero to +// determine whether it should be omitted when marshaling +// with the omitempty flag. One notable implementation +// is time.Time. +type IsZeroer interface { + IsZero() bool +} + +func isZero(v reflect.Value) bool { + kind := v.Kind() + if z, ok := v.Interface().(IsZeroer); ok { + if (kind == reflect.Ptr || kind == reflect.Interface) && v.IsNil() { + return true + } + return z.IsZero() + } + switch kind { + case reflect.String: + return len(v.String()) == 0 + case reflect.Interface, reflect.Ptr: + return v.IsNil() + case reflect.Slice: + return v.Len() == 0 + case reflect.Map: + return v.Len() == 0 + case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64: + return v.Int() == 0 + case reflect.Float32, reflect.Float64: + return v.Float() == 0 + case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64, reflect.Uintptr: + return v.Uint() == 0 + case reflect.Bool: + return !v.Bool() + case reflect.Struct: + vt := v.Type() + for i := v.NumField() - 1; i >= 0; i-- { + if vt.Field(i).PkgPath != "" { + continue // Private field + } + if !isZero(v.Field(i)) { + return false + } + } + return true + } + return false +} + +// FutureLineWrap globally disables line wrapping when encoding long strings. +// This is a temporary and thus deprecated method introduced to faciliate +// migration towards v3, which offers more control of line lengths on +// individual encodings, and has a default matching the behavior introduced +// by this function. +// +// The default formatting of v2 was erroneously changed in v2.3.0 and reverted +// in v2.4.0, at which point this function was introduced to help migration. +func FutureLineWrap() { + disableLineWrapping = true +} diff --git a/vendor/gopkg.in/yaml.v2/yamlh.go b/vendor/gopkg.in/yaml.v2/yamlh.go new file mode 100644 index 00000000..f6a9c8e3 --- /dev/null +++ b/vendor/gopkg.in/yaml.v2/yamlh.go @@ -0,0 +1,739 @@ +package yaml + +import ( + "fmt" + "io" +) + +// The version directive data. +type yaml_version_directive_t struct { + major int8 // The major version number. + minor int8 // The minor version number. +} + +// The tag directive data. +type yaml_tag_directive_t struct { + handle []byte // The tag handle. + prefix []byte // The tag prefix. +} + +type yaml_encoding_t int + +// The stream encoding. +const ( + // Let the parser choose the encoding. + yaml_ANY_ENCODING yaml_encoding_t = iota + + yaml_UTF8_ENCODING // The default UTF-8 encoding. + yaml_UTF16LE_ENCODING // The UTF-16-LE encoding with BOM. + yaml_UTF16BE_ENCODING // The UTF-16-BE encoding with BOM. +) + +type yaml_break_t int + +// Line break types. +const ( + // Let the parser choose the break type. + yaml_ANY_BREAK yaml_break_t = iota + + yaml_CR_BREAK // Use CR for line breaks (Mac style). + yaml_LN_BREAK // Use LN for line breaks (Unix style). + yaml_CRLN_BREAK // Use CR LN for line breaks (DOS style). +) + +type yaml_error_type_t int + +// Many bad things could happen with the parser and emitter. +const ( + // No error is produced. + yaml_NO_ERROR yaml_error_type_t = iota + + yaml_MEMORY_ERROR // Cannot allocate or reallocate a block of memory. + yaml_READER_ERROR // Cannot read or decode the input stream. + yaml_SCANNER_ERROR // Cannot scan the input stream. + yaml_PARSER_ERROR // Cannot parse the input stream. + yaml_COMPOSER_ERROR // Cannot compose a YAML document. + yaml_WRITER_ERROR // Cannot write to the output stream. + yaml_EMITTER_ERROR // Cannot emit a YAML stream. +) + +// The pointer position. +type yaml_mark_t struct { + index int // The position index. + line int // The position line. + column int // The position column. +} + +// Node Styles + +type yaml_style_t int8 + +type yaml_scalar_style_t yaml_style_t + +// Scalar styles. +const ( + // Let the emitter choose the style. + yaml_ANY_SCALAR_STYLE yaml_scalar_style_t = iota + + yaml_PLAIN_SCALAR_STYLE // The plain scalar style. + yaml_SINGLE_QUOTED_SCALAR_STYLE // The single-quoted scalar style. + yaml_DOUBLE_QUOTED_SCALAR_STYLE // The double-quoted scalar style. + yaml_LITERAL_SCALAR_STYLE // The literal scalar style. + yaml_FOLDED_SCALAR_STYLE // The folded scalar style. +) + +type yaml_sequence_style_t yaml_style_t + +// Sequence styles. +const ( + // Let the emitter choose the style. + yaml_ANY_SEQUENCE_STYLE yaml_sequence_style_t = iota + + yaml_BLOCK_SEQUENCE_STYLE // The block sequence style. + yaml_FLOW_SEQUENCE_STYLE // The flow sequence style. +) + +type yaml_mapping_style_t yaml_style_t + +// Mapping styles. +const ( + // Let the emitter choose the style. + yaml_ANY_MAPPING_STYLE yaml_mapping_style_t = iota + + yaml_BLOCK_MAPPING_STYLE // The block mapping style. + yaml_FLOW_MAPPING_STYLE // The flow mapping style. +) + +// Tokens + +type yaml_token_type_t int + +// Token types. +const ( + // An empty token. + yaml_NO_TOKEN yaml_token_type_t = iota + + yaml_STREAM_START_TOKEN // A STREAM-START token. + yaml_STREAM_END_TOKEN // A STREAM-END token. + + yaml_VERSION_DIRECTIVE_TOKEN // A VERSION-DIRECTIVE token. + yaml_TAG_DIRECTIVE_TOKEN // A TAG-DIRECTIVE token. + yaml_DOCUMENT_START_TOKEN // A DOCUMENT-START token. + yaml_DOCUMENT_END_TOKEN // A DOCUMENT-END token. + + yaml_BLOCK_SEQUENCE_START_TOKEN // A BLOCK-SEQUENCE-START token. + yaml_BLOCK_MAPPING_START_TOKEN // A BLOCK-SEQUENCE-END token. + yaml_BLOCK_END_TOKEN // A BLOCK-END token. + + yaml_FLOW_SEQUENCE_START_TOKEN // A FLOW-SEQUENCE-START token. + yaml_FLOW_SEQUENCE_END_TOKEN // A FLOW-SEQUENCE-END token. + yaml_FLOW_MAPPING_START_TOKEN // A FLOW-MAPPING-START token. + yaml_FLOW_MAPPING_END_TOKEN // A FLOW-MAPPING-END token. + + yaml_BLOCK_ENTRY_TOKEN // A BLOCK-ENTRY token. + yaml_FLOW_ENTRY_TOKEN // A FLOW-ENTRY token. + yaml_KEY_TOKEN // A KEY token. + yaml_VALUE_TOKEN // A VALUE token. + + yaml_ALIAS_TOKEN // An ALIAS token. + yaml_ANCHOR_TOKEN // An ANCHOR token. + yaml_TAG_TOKEN // A TAG token. + yaml_SCALAR_TOKEN // A SCALAR token. +) + +func (tt yaml_token_type_t) String() string { + switch tt { + case yaml_NO_TOKEN: + return "yaml_NO_TOKEN" + case yaml_STREAM_START_TOKEN: + return "yaml_STREAM_START_TOKEN" + case yaml_STREAM_END_TOKEN: + return "yaml_STREAM_END_TOKEN" + case yaml_VERSION_DIRECTIVE_TOKEN: + return "yaml_VERSION_DIRECTIVE_TOKEN" + case yaml_TAG_DIRECTIVE_TOKEN: + return "yaml_TAG_DIRECTIVE_TOKEN" + case yaml_DOCUMENT_START_TOKEN: + return "yaml_DOCUMENT_START_TOKEN" + case yaml_DOCUMENT_END_TOKEN: + return "yaml_DOCUMENT_END_TOKEN" + case yaml_BLOCK_SEQUENCE_START_TOKEN: + return "yaml_BLOCK_SEQUENCE_START_TOKEN" + case yaml_BLOCK_MAPPING_START_TOKEN: + return "yaml_BLOCK_MAPPING_START_TOKEN" + case yaml_BLOCK_END_TOKEN: + return "yaml_BLOCK_END_TOKEN" + case yaml_FLOW_SEQUENCE_START_TOKEN: + return "yaml_FLOW_SEQUENCE_START_TOKEN" + case yaml_FLOW_SEQUENCE_END_TOKEN: + return "yaml_FLOW_SEQUENCE_END_TOKEN" + case yaml_FLOW_MAPPING_START_TOKEN: + return "yaml_FLOW_MAPPING_START_TOKEN" + case yaml_FLOW_MAPPING_END_TOKEN: + return "yaml_FLOW_MAPPING_END_TOKEN" + case yaml_BLOCK_ENTRY_TOKEN: + return "yaml_BLOCK_ENTRY_TOKEN" + case yaml_FLOW_ENTRY_TOKEN: + return "yaml_FLOW_ENTRY_TOKEN" + case yaml_KEY_TOKEN: + return "yaml_KEY_TOKEN" + case yaml_VALUE_TOKEN: + return "yaml_VALUE_TOKEN" + case yaml_ALIAS_TOKEN: + return "yaml_ALIAS_TOKEN" + case yaml_ANCHOR_TOKEN: + return "yaml_ANCHOR_TOKEN" + case yaml_TAG_TOKEN: + return "yaml_TAG_TOKEN" + case yaml_SCALAR_TOKEN: + return "yaml_SCALAR_TOKEN" + } + return "" +} + +// The token structure. +type yaml_token_t struct { + // The token type. + typ yaml_token_type_t + + // The start/end of the token. + start_mark, end_mark yaml_mark_t + + // The stream encoding (for yaml_STREAM_START_TOKEN). + encoding yaml_encoding_t + + // The alias/anchor/scalar value or tag/tag directive handle + // (for yaml_ALIAS_TOKEN, yaml_ANCHOR_TOKEN, yaml_SCALAR_TOKEN, yaml_TAG_TOKEN, yaml_TAG_DIRECTIVE_TOKEN). + value []byte + + // The tag suffix (for yaml_TAG_TOKEN). + suffix []byte + + // The tag directive prefix (for yaml_TAG_DIRECTIVE_TOKEN). + prefix []byte + + // The scalar style (for yaml_SCALAR_TOKEN). + style yaml_scalar_style_t + + // The version directive major/minor (for yaml_VERSION_DIRECTIVE_TOKEN). + major, minor int8 +} + +// Events + +type yaml_event_type_t int8 + +// Event types. +const ( + // An empty event. + yaml_NO_EVENT yaml_event_type_t = iota + + yaml_STREAM_START_EVENT // A STREAM-START event. + yaml_STREAM_END_EVENT // A STREAM-END event. + yaml_DOCUMENT_START_EVENT // A DOCUMENT-START event. + yaml_DOCUMENT_END_EVENT // A DOCUMENT-END event. + yaml_ALIAS_EVENT // An ALIAS event. + yaml_SCALAR_EVENT // A SCALAR event. + yaml_SEQUENCE_START_EVENT // A SEQUENCE-START event. + yaml_SEQUENCE_END_EVENT // A SEQUENCE-END event. + yaml_MAPPING_START_EVENT // A MAPPING-START event. + yaml_MAPPING_END_EVENT // A MAPPING-END event. +) + +var eventStrings = []string{ + yaml_NO_EVENT: "none", + yaml_STREAM_START_EVENT: "stream start", + yaml_STREAM_END_EVENT: "stream end", + yaml_DOCUMENT_START_EVENT: "document start", + yaml_DOCUMENT_END_EVENT: "document end", + yaml_ALIAS_EVENT: "alias", + yaml_SCALAR_EVENT: "scalar", + yaml_SEQUENCE_START_EVENT: "sequence start", + yaml_SEQUENCE_END_EVENT: "sequence end", + yaml_MAPPING_START_EVENT: "mapping start", + yaml_MAPPING_END_EVENT: "mapping end", +} + +func (e yaml_event_type_t) String() string { + if e < 0 || int(e) >= len(eventStrings) { + return fmt.Sprintf("unknown event %d", e) + } + return eventStrings[e] +} + +// The event structure. +type yaml_event_t struct { + + // The event type. + typ yaml_event_type_t + + // The start and end of the event. + start_mark, end_mark yaml_mark_t + + // The document encoding (for yaml_STREAM_START_EVENT). + encoding yaml_encoding_t + + // The version directive (for yaml_DOCUMENT_START_EVENT). + version_directive *yaml_version_directive_t + + // The list of tag directives (for yaml_DOCUMENT_START_EVENT). + tag_directives []yaml_tag_directive_t + + // The anchor (for yaml_SCALAR_EVENT, yaml_SEQUENCE_START_EVENT, yaml_MAPPING_START_EVENT, yaml_ALIAS_EVENT). + anchor []byte + + // The tag (for yaml_SCALAR_EVENT, yaml_SEQUENCE_START_EVENT, yaml_MAPPING_START_EVENT). + tag []byte + + // The scalar value (for yaml_SCALAR_EVENT). + value []byte + + // Is the document start/end indicator implicit, or the tag optional? + // (for yaml_DOCUMENT_START_EVENT, yaml_DOCUMENT_END_EVENT, yaml_SEQUENCE_START_EVENT, yaml_MAPPING_START_EVENT, yaml_SCALAR_EVENT). + implicit bool + + // Is the tag optional for any non-plain style? (for yaml_SCALAR_EVENT). + quoted_implicit bool + + // The style (for yaml_SCALAR_EVENT, yaml_SEQUENCE_START_EVENT, yaml_MAPPING_START_EVENT). + style yaml_style_t +} + +func (e *yaml_event_t) scalar_style() yaml_scalar_style_t { return yaml_scalar_style_t(e.style) } +func (e *yaml_event_t) sequence_style() yaml_sequence_style_t { return yaml_sequence_style_t(e.style) } +func (e *yaml_event_t) mapping_style() yaml_mapping_style_t { return yaml_mapping_style_t(e.style) } + +// Nodes + +const ( + yaml_NULL_TAG = "tag:yaml.org,2002:null" // The tag !!null with the only possible value: null. + yaml_BOOL_TAG = "tag:yaml.org,2002:bool" // The tag !!bool with the values: true and false. + yaml_STR_TAG = "tag:yaml.org,2002:str" // The tag !!str for string values. + yaml_INT_TAG = "tag:yaml.org,2002:int" // The tag !!int for integer values. + yaml_FLOAT_TAG = "tag:yaml.org,2002:float" // The tag !!float for float values. + yaml_TIMESTAMP_TAG = "tag:yaml.org,2002:timestamp" // The tag !!timestamp for date and time values. + + yaml_SEQ_TAG = "tag:yaml.org,2002:seq" // The tag !!seq is used to denote sequences. + yaml_MAP_TAG = "tag:yaml.org,2002:map" // The tag !!map is used to denote mapping. + + // Not in original libyaml. + yaml_BINARY_TAG = "tag:yaml.org,2002:binary" + yaml_MERGE_TAG = "tag:yaml.org,2002:merge" + + yaml_DEFAULT_SCALAR_TAG = yaml_STR_TAG // The default scalar tag is !!str. + yaml_DEFAULT_SEQUENCE_TAG = yaml_SEQ_TAG // The default sequence tag is !!seq. + yaml_DEFAULT_MAPPING_TAG = yaml_MAP_TAG // The default mapping tag is !!map. +) + +type yaml_node_type_t int + +// Node types. +const ( + // An empty node. + yaml_NO_NODE yaml_node_type_t = iota + + yaml_SCALAR_NODE // A scalar node. + yaml_SEQUENCE_NODE // A sequence node. + yaml_MAPPING_NODE // A mapping node. +) + +// An element of a sequence node. +type yaml_node_item_t int + +// An element of a mapping node. +type yaml_node_pair_t struct { + key int // The key of the element. + value int // The value of the element. +} + +// The node structure. +type yaml_node_t struct { + typ yaml_node_type_t // The node type. + tag []byte // The node tag. + + // The node data. + + // The scalar parameters (for yaml_SCALAR_NODE). + scalar struct { + value []byte // The scalar value. + length int // The length of the scalar value. + style yaml_scalar_style_t // The scalar style. + } + + // The sequence parameters (for YAML_SEQUENCE_NODE). + sequence struct { + items_data []yaml_node_item_t // The stack of sequence items. + style yaml_sequence_style_t // The sequence style. + } + + // The mapping parameters (for yaml_MAPPING_NODE). + mapping struct { + pairs_data []yaml_node_pair_t // The stack of mapping pairs (key, value). + pairs_start *yaml_node_pair_t // The beginning of the stack. + pairs_end *yaml_node_pair_t // The end of the stack. + pairs_top *yaml_node_pair_t // The top of the stack. + style yaml_mapping_style_t // The mapping style. + } + + start_mark yaml_mark_t // The beginning of the node. + end_mark yaml_mark_t // The end of the node. + +} + +// The document structure. +type yaml_document_t struct { + + // The document nodes. + nodes []yaml_node_t + + // The version directive. + version_directive *yaml_version_directive_t + + // The list of tag directives. + tag_directives_data []yaml_tag_directive_t + tag_directives_start int // The beginning of the tag directives list. + tag_directives_end int // The end of the tag directives list. + + start_implicit int // Is the document start indicator implicit? + end_implicit int // Is the document end indicator implicit? + + // The start/end of the document. + start_mark, end_mark yaml_mark_t +} + +// The prototype of a read handler. +// +// The read handler is called when the parser needs to read more bytes from the +// source. The handler should write not more than size bytes to the buffer. +// The number of written bytes should be set to the size_read variable. +// +// [in,out] data A pointer to an application data specified by +// yaml_parser_set_input(). +// [out] buffer The buffer to write the data from the source. +// [in] size The size of the buffer. +// [out] size_read The actual number of bytes read from the source. +// +// On success, the handler should return 1. If the handler failed, +// the returned value should be 0. On EOF, the handler should set the +// size_read to 0 and return 1. +type yaml_read_handler_t func(parser *yaml_parser_t, buffer []byte) (n int, err error) + +// This structure holds information about a potential simple key. +type yaml_simple_key_t struct { + possible bool // Is a simple key possible? + required bool // Is a simple key required? + token_number int // The number of the token. + mark yaml_mark_t // The position mark. +} + +// The states of the parser. +type yaml_parser_state_t int + +const ( + yaml_PARSE_STREAM_START_STATE yaml_parser_state_t = iota + + yaml_PARSE_IMPLICIT_DOCUMENT_START_STATE // Expect the beginning of an implicit document. + yaml_PARSE_DOCUMENT_START_STATE // Expect DOCUMENT-START. + yaml_PARSE_DOCUMENT_CONTENT_STATE // Expect the content of a document. + yaml_PARSE_DOCUMENT_END_STATE // Expect DOCUMENT-END. + yaml_PARSE_BLOCK_NODE_STATE // Expect a block node. + yaml_PARSE_BLOCK_NODE_OR_INDENTLESS_SEQUENCE_STATE // Expect a block node or indentless sequence. + yaml_PARSE_FLOW_NODE_STATE // Expect a flow node. + yaml_PARSE_BLOCK_SEQUENCE_FIRST_ENTRY_STATE // Expect the first entry of a block sequence. + yaml_PARSE_BLOCK_SEQUENCE_ENTRY_STATE // Expect an entry of a block sequence. + yaml_PARSE_INDENTLESS_SEQUENCE_ENTRY_STATE // Expect an entry of an indentless sequence. + yaml_PARSE_BLOCK_MAPPING_FIRST_KEY_STATE // Expect the first key of a block mapping. + yaml_PARSE_BLOCK_MAPPING_KEY_STATE // Expect a block mapping key. + yaml_PARSE_BLOCK_MAPPING_VALUE_STATE // Expect a block mapping value. + yaml_PARSE_FLOW_SEQUENCE_FIRST_ENTRY_STATE // Expect the first entry of a flow sequence. + yaml_PARSE_FLOW_SEQUENCE_ENTRY_STATE // Expect an entry of a flow sequence. + yaml_PARSE_FLOW_SEQUENCE_ENTRY_MAPPING_KEY_STATE // Expect a key of an ordered mapping. + yaml_PARSE_FLOW_SEQUENCE_ENTRY_MAPPING_VALUE_STATE // Expect a value of an ordered mapping. + yaml_PARSE_FLOW_SEQUENCE_ENTRY_MAPPING_END_STATE // Expect the and of an ordered mapping entry. + yaml_PARSE_FLOW_MAPPING_FIRST_KEY_STATE // Expect the first key of a flow mapping. + yaml_PARSE_FLOW_MAPPING_KEY_STATE // Expect a key of a flow mapping. + yaml_PARSE_FLOW_MAPPING_VALUE_STATE // Expect a value of a flow mapping. + yaml_PARSE_FLOW_MAPPING_EMPTY_VALUE_STATE // Expect an empty value of a flow mapping. + yaml_PARSE_END_STATE // Expect nothing. +) + +func (ps yaml_parser_state_t) String() string { + switch ps { + case yaml_PARSE_STREAM_START_STATE: + return "yaml_PARSE_STREAM_START_STATE" + case yaml_PARSE_IMPLICIT_DOCUMENT_START_STATE: + return "yaml_PARSE_IMPLICIT_DOCUMENT_START_STATE" + case yaml_PARSE_DOCUMENT_START_STATE: + return "yaml_PARSE_DOCUMENT_START_STATE" + case yaml_PARSE_DOCUMENT_CONTENT_STATE: + return "yaml_PARSE_DOCUMENT_CONTENT_STATE" + case yaml_PARSE_DOCUMENT_END_STATE: + return "yaml_PARSE_DOCUMENT_END_STATE" + case yaml_PARSE_BLOCK_NODE_STATE: + return "yaml_PARSE_BLOCK_NODE_STATE" + case yaml_PARSE_BLOCK_NODE_OR_INDENTLESS_SEQUENCE_STATE: + return "yaml_PARSE_BLOCK_NODE_OR_INDENTLESS_SEQUENCE_STATE" + case yaml_PARSE_FLOW_NODE_STATE: + return "yaml_PARSE_FLOW_NODE_STATE" + case yaml_PARSE_BLOCK_SEQUENCE_FIRST_ENTRY_STATE: + return "yaml_PARSE_BLOCK_SEQUENCE_FIRST_ENTRY_STATE" + case yaml_PARSE_BLOCK_SEQUENCE_ENTRY_STATE: + return "yaml_PARSE_BLOCK_SEQUENCE_ENTRY_STATE" + case yaml_PARSE_INDENTLESS_SEQUENCE_ENTRY_STATE: + return "yaml_PARSE_INDENTLESS_SEQUENCE_ENTRY_STATE" + case yaml_PARSE_BLOCK_MAPPING_FIRST_KEY_STATE: + return "yaml_PARSE_BLOCK_MAPPING_FIRST_KEY_STATE" + case yaml_PARSE_BLOCK_MAPPING_KEY_STATE: + return "yaml_PARSE_BLOCK_MAPPING_KEY_STATE" + case yaml_PARSE_BLOCK_MAPPING_VALUE_STATE: + return "yaml_PARSE_BLOCK_MAPPING_VALUE_STATE" + case yaml_PARSE_FLOW_SEQUENCE_FIRST_ENTRY_STATE: + return "yaml_PARSE_FLOW_SEQUENCE_FIRST_ENTRY_STATE" + case yaml_PARSE_FLOW_SEQUENCE_ENTRY_STATE: + return "yaml_PARSE_FLOW_SEQUENCE_ENTRY_STATE" + case yaml_PARSE_FLOW_SEQUENCE_ENTRY_MAPPING_KEY_STATE: + return "yaml_PARSE_FLOW_SEQUENCE_ENTRY_MAPPING_KEY_STATE" + case yaml_PARSE_FLOW_SEQUENCE_ENTRY_MAPPING_VALUE_STATE: + return "yaml_PARSE_FLOW_SEQUENCE_ENTRY_MAPPING_VALUE_STATE" + case yaml_PARSE_FLOW_SEQUENCE_ENTRY_MAPPING_END_STATE: + return "yaml_PARSE_FLOW_SEQUENCE_ENTRY_MAPPING_END_STATE" + case yaml_PARSE_FLOW_MAPPING_FIRST_KEY_STATE: + return "yaml_PARSE_FLOW_MAPPING_FIRST_KEY_STATE" + case yaml_PARSE_FLOW_MAPPING_KEY_STATE: + return "yaml_PARSE_FLOW_MAPPING_KEY_STATE" + case yaml_PARSE_FLOW_MAPPING_VALUE_STATE: + return "yaml_PARSE_FLOW_MAPPING_VALUE_STATE" + case yaml_PARSE_FLOW_MAPPING_EMPTY_VALUE_STATE: + return "yaml_PARSE_FLOW_MAPPING_EMPTY_VALUE_STATE" + case yaml_PARSE_END_STATE: + return "yaml_PARSE_END_STATE" + } + return "" +} + +// This structure holds aliases data. +type yaml_alias_data_t struct { + anchor []byte // The anchor. + index int // The node id. + mark yaml_mark_t // The anchor mark. +} + +// The parser structure. +// +// All members are internal. Manage the structure using the +// yaml_parser_ family of functions. +type yaml_parser_t struct { + + // Error handling + + error yaml_error_type_t // Error type. + + problem string // Error description. + + // The byte about which the problem occurred. + problem_offset int + problem_value int + problem_mark yaml_mark_t + + // The error context. + context string + context_mark yaml_mark_t + + // Reader stuff + + read_handler yaml_read_handler_t // Read handler. + + input_reader io.Reader // File input data. + input []byte // String input data. + input_pos int + + eof bool // EOF flag + + buffer []byte // The working buffer. + buffer_pos int // The current position of the buffer. + + unread int // The number of unread characters in the buffer. + + raw_buffer []byte // The raw buffer. + raw_buffer_pos int // The current position of the buffer. + + encoding yaml_encoding_t // The input encoding. + + offset int // The offset of the current position (in bytes). + mark yaml_mark_t // The mark of the current position. + + // Scanner stuff + + stream_start_produced bool // Have we started to scan the input stream? + stream_end_produced bool // Have we reached the end of the input stream? + + flow_level int // The number of unclosed '[' and '{' indicators. + + tokens []yaml_token_t // The tokens queue. + tokens_head int // The head of the tokens queue. + tokens_parsed int // The number of tokens fetched from the queue. + token_available bool // Does the tokens queue contain a token ready for dequeueing. + + indent int // The current indentation level. + indents []int // The indentation levels stack. + + simple_key_allowed bool // May a simple key occur at the current position? + simple_keys []yaml_simple_key_t // The stack of simple keys. + simple_keys_by_tok map[int]int // possible simple_key indexes indexed by token_number + + // Parser stuff + + state yaml_parser_state_t // The current parser state. + states []yaml_parser_state_t // The parser states stack. + marks []yaml_mark_t // The stack of marks. + tag_directives []yaml_tag_directive_t // The list of TAG directives. + + // Dumper stuff + + aliases []yaml_alias_data_t // The alias data. + + document *yaml_document_t // The currently parsed document. +} + +// Emitter Definitions + +// The prototype of a write handler. +// +// The write handler is called when the emitter needs to flush the accumulated +// characters to the output. The handler should write @a size bytes of the +// @a buffer to the output. +// +// @param[in,out] data A pointer to an application data specified by +// yaml_emitter_set_output(). +// @param[in] buffer The buffer with bytes to be written. +// @param[in] size The size of the buffer. +// +// @returns On success, the handler should return @c 1. If the handler failed, +// the returned value should be @c 0. +// +type yaml_write_handler_t func(emitter *yaml_emitter_t, buffer []byte) error + +type yaml_emitter_state_t int + +// The emitter states. +const ( + // Expect STREAM-START. + yaml_EMIT_STREAM_START_STATE yaml_emitter_state_t = iota + + yaml_EMIT_FIRST_DOCUMENT_START_STATE // Expect the first DOCUMENT-START or STREAM-END. + yaml_EMIT_DOCUMENT_START_STATE // Expect DOCUMENT-START or STREAM-END. + yaml_EMIT_DOCUMENT_CONTENT_STATE // Expect the content of a document. + yaml_EMIT_DOCUMENT_END_STATE // Expect DOCUMENT-END. + yaml_EMIT_FLOW_SEQUENCE_FIRST_ITEM_STATE // Expect the first item of a flow sequence. + yaml_EMIT_FLOW_SEQUENCE_ITEM_STATE // Expect an item of a flow sequence. + yaml_EMIT_FLOW_MAPPING_FIRST_KEY_STATE // Expect the first key of a flow mapping. + yaml_EMIT_FLOW_MAPPING_KEY_STATE // Expect a key of a flow mapping. + yaml_EMIT_FLOW_MAPPING_SIMPLE_VALUE_STATE // Expect a value for a simple key of a flow mapping. + yaml_EMIT_FLOW_MAPPING_VALUE_STATE // Expect a value of a flow mapping. + yaml_EMIT_BLOCK_SEQUENCE_FIRST_ITEM_STATE // Expect the first item of a block sequence. + yaml_EMIT_BLOCK_SEQUENCE_ITEM_STATE // Expect an item of a block sequence. + yaml_EMIT_BLOCK_MAPPING_FIRST_KEY_STATE // Expect the first key of a block mapping. + yaml_EMIT_BLOCK_MAPPING_KEY_STATE // Expect the key of a block mapping. + yaml_EMIT_BLOCK_MAPPING_SIMPLE_VALUE_STATE // Expect a value for a simple key of a block mapping. + yaml_EMIT_BLOCK_MAPPING_VALUE_STATE // Expect a value of a block mapping. + yaml_EMIT_END_STATE // Expect nothing. +) + +// The emitter structure. +// +// All members are internal. Manage the structure using the @c yaml_emitter_ +// family of functions. +type yaml_emitter_t struct { + + // Error handling + + error yaml_error_type_t // Error type. + problem string // Error description. + + // Writer stuff + + write_handler yaml_write_handler_t // Write handler. + + output_buffer *[]byte // String output data. + output_writer io.Writer // File output data. + + buffer []byte // The working buffer. + buffer_pos int // The current position of the buffer. + + raw_buffer []byte // The raw buffer. + raw_buffer_pos int // The current position of the buffer. + + encoding yaml_encoding_t // The stream encoding. + + // Emitter stuff + + canonical bool // If the output is in the canonical style? + best_indent int // The number of indentation spaces. + best_width int // The preferred width of the output lines. + unicode bool // Allow unescaped non-ASCII characters? + line_break yaml_break_t // The preferred line break. + + state yaml_emitter_state_t // The current emitter state. + states []yaml_emitter_state_t // The stack of states. + + events []yaml_event_t // The event queue. + events_head int // The head of the event queue. + + indents []int // The stack of indentation levels. + + tag_directives []yaml_tag_directive_t // The list of tag directives. + + indent int // The current indentation level. + + flow_level int // The current flow level. + + root_context bool // Is it the document root context? + sequence_context bool // Is it a sequence context? + mapping_context bool // Is it a mapping context? + simple_key_context bool // Is it a simple mapping key context? + + line int // The current line. + column int // The current column. + whitespace bool // If the last character was a whitespace? + indention bool // If the last character was an indentation character (' ', '-', '?', ':')? + open_ended bool // If an explicit document end is required? + + // Anchor analysis. + anchor_data struct { + anchor []byte // The anchor value. + alias bool // Is it an alias? + } + + // Tag analysis. + tag_data struct { + handle []byte // The tag handle. + suffix []byte // The tag suffix. + } + + // Scalar analysis. + scalar_data struct { + value []byte // The scalar value. + multiline bool // Does the scalar contain line breaks? + flow_plain_allowed bool // Can the scalar be expessed in the flow plain style? + block_plain_allowed bool // Can the scalar be expressed in the block plain style? + single_quoted_allowed bool // Can the scalar be expressed in the single quoted style? + block_allowed bool // Can the scalar be expressed in the literal or folded styles? + style yaml_scalar_style_t // The output style. + } + + // Dumper stuff + + opened bool // If the stream was already opened? + closed bool // If the stream was already closed? + + // The information associated with the document nodes. + anchors *struct { + references int // The number of references. + anchor int // The anchor id. + serialized bool // If the node has been emitted? + } + + last_anchor_id int // The last assigned anchor id. + + document *yaml_document_t // The currently emitted document. +} diff --git a/vendor/gopkg.in/yaml.v2/yamlprivateh.go b/vendor/gopkg.in/yaml.v2/yamlprivateh.go new file mode 100644 index 00000000..8110ce3c --- /dev/null +++ b/vendor/gopkg.in/yaml.v2/yamlprivateh.go @@ -0,0 +1,173 @@ +package yaml + +const ( + // The size of the input raw buffer. + input_raw_buffer_size = 512 + + // The size of the input buffer. + // It should be possible to decode the whole raw buffer. + input_buffer_size = input_raw_buffer_size * 3 + + // The size of the output buffer. + output_buffer_size = 128 + + // The size of the output raw buffer. + // It should be possible to encode the whole output buffer. + output_raw_buffer_size = (output_buffer_size*2 + 2) + + // The size of other stacks and queues. + initial_stack_size = 16 + initial_queue_size = 16 + initial_string_size = 16 +) + +// Check if the character at the specified position is an alphabetical +// character, a digit, '_', or '-'. +func is_alpha(b []byte, i int) bool { + return b[i] >= '0' && b[i] <= '9' || b[i] >= 'A' && b[i] <= 'Z' || b[i] >= 'a' && b[i] <= 'z' || b[i] == '_' || b[i] == '-' +} + +// Check if the character at the specified position is a digit. +func is_digit(b []byte, i int) bool { + return b[i] >= '0' && b[i] <= '9' +} + +// Get the value of a digit. +func as_digit(b []byte, i int) int { + return int(b[i]) - '0' +} + +// Check if the character at the specified position is a hex-digit. +func is_hex(b []byte, i int) bool { + return b[i] >= '0' && b[i] <= '9' || b[i] >= 'A' && b[i] <= 'F' || b[i] >= 'a' && b[i] <= 'f' +} + +// Get the value of a hex-digit. +func as_hex(b []byte, i int) int { + bi := b[i] + if bi >= 'A' && bi <= 'F' { + return int(bi) - 'A' + 10 + } + if bi >= 'a' && bi <= 'f' { + return int(bi) - 'a' + 10 + } + return int(bi) - '0' +} + +// Check if the character is ASCII. +func is_ascii(b []byte, i int) bool { + return b[i] <= 0x7F +} + +// Check if the character at the start of the buffer can be printed unescaped. +func is_printable(b []byte, i int) bool { + return ((b[i] == 0x0A) || // . == #x0A + (b[i] >= 0x20 && b[i] <= 0x7E) || // #x20 <= . <= #x7E + (b[i] == 0xC2 && b[i+1] >= 0xA0) || // #0xA0 <= . <= #xD7FF + (b[i] > 0xC2 && b[i] < 0xED) || + (b[i] == 0xED && b[i+1] < 0xA0) || + (b[i] == 0xEE) || + (b[i] == 0xEF && // #xE000 <= . <= #xFFFD + !(b[i+1] == 0xBB && b[i+2] == 0xBF) && // && . != #xFEFF + !(b[i+1] == 0xBF && (b[i+2] == 0xBE || b[i+2] == 0xBF)))) +} + +// Check if the character at the specified position is NUL. +func is_z(b []byte, i int) bool { + return b[i] == 0x00 +} + +// Check if the beginning of the buffer is a BOM. +func is_bom(b []byte, i int) bool { + return b[0] == 0xEF && b[1] == 0xBB && b[2] == 0xBF +} + +// Check if the character at the specified position is space. +func is_space(b []byte, i int) bool { + return b[i] == ' ' +} + +// Check if the character at the specified position is tab. +func is_tab(b []byte, i int) bool { + return b[i] == '\t' +} + +// Check if the character at the specified position is blank (space or tab). +func is_blank(b []byte, i int) bool { + //return is_space(b, i) || is_tab(b, i) + return b[i] == ' ' || b[i] == '\t' +} + +// Check if the character at the specified position is a line break. +func is_break(b []byte, i int) bool { + return (b[i] == '\r' || // CR (#xD) + b[i] == '\n' || // LF (#xA) + b[i] == 0xC2 && b[i+1] == 0x85 || // NEL (#x85) + b[i] == 0xE2 && b[i+1] == 0x80 && b[i+2] == 0xA8 || // LS (#x2028) + b[i] == 0xE2 && b[i+1] == 0x80 && b[i+2] == 0xA9) // PS (#x2029) +} + +func is_crlf(b []byte, i int) bool { + return b[i] == '\r' && b[i+1] == '\n' +} + +// Check if the character is a line break or NUL. +func is_breakz(b []byte, i int) bool { + //return is_break(b, i) || is_z(b, i) + return ( // is_break: + b[i] == '\r' || // CR (#xD) + b[i] == '\n' || // LF (#xA) + b[i] == 0xC2 && b[i+1] == 0x85 || // NEL (#x85) + b[i] == 0xE2 && b[i+1] == 0x80 && b[i+2] == 0xA8 || // LS (#x2028) + b[i] == 0xE2 && b[i+1] == 0x80 && b[i+2] == 0xA9 || // PS (#x2029) + // is_z: + b[i] == 0) +} + +// Check if the character is a line break, space, or NUL. +func is_spacez(b []byte, i int) bool { + //return is_space(b, i) || is_breakz(b, i) + return ( // is_space: + b[i] == ' ' || + // is_breakz: + b[i] == '\r' || // CR (#xD) + b[i] == '\n' || // LF (#xA) + b[i] == 0xC2 && b[i+1] == 0x85 || // NEL (#x85) + b[i] == 0xE2 && b[i+1] == 0x80 && b[i+2] == 0xA8 || // LS (#x2028) + b[i] == 0xE2 && b[i+1] == 0x80 && b[i+2] == 0xA9 || // PS (#x2029) + b[i] == 0) +} + +// Check if the character is a line break, space, tab, or NUL. +func is_blankz(b []byte, i int) bool { + //return is_blank(b, i) || is_breakz(b, i) + return ( // is_blank: + b[i] == ' ' || b[i] == '\t' || + // is_breakz: + b[i] == '\r' || // CR (#xD) + b[i] == '\n' || // LF (#xA) + b[i] == 0xC2 && b[i+1] == 0x85 || // NEL (#x85) + b[i] == 0xE2 && b[i+1] == 0x80 && b[i+2] == 0xA8 || // LS (#x2028) + b[i] == 0xE2 && b[i+1] == 0x80 && b[i+2] == 0xA9 || // PS (#x2029) + b[i] == 0) +} + +// Determine the width of the character. +func width(b byte) int { + // Don't replace these by a switch without first + // confirming that it is being inlined. + if b&0x80 == 0x00 { + return 1 + } + if b&0xE0 == 0xC0 { + return 2 + } + if b&0xF0 == 0xE0 { + return 3 + } + if b&0xF8 == 0xF0 { + return 4 + } + return 0 + +} diff --git a/vendor/gopkg.in/yaml.v3/LICENSE b/vendor/gopkg.in/yaml.v3/LICENSE new file mode 100644 index 00000000..2683e4bb --- /dev/null +++ b/vendor/gopkg.in/yaml.v3/LICENSE @@ -0,0 +1,50 @@ + +This project is covered by two different licenses: MIT and Apache. + +#### MIT License #### + +The following files were ported to Go from C files of libyaml, and thus +are still covered by their original MIT license, with the additional +copyright staring in 2011 when the project was ported over: + + apic.go emitterc.go parserc.go readerc.go scannerc.go + writerc.go yamlh.go yamlprivateh.go + +Copyright (c) 2006-2010 Kirill Simonov +Copyright (c) 2006-2011 Kirill Simonov + +Permission is hereby granted, free of charge, to any person obtaining a copy of +this software and associated documentation files (the "Software"), to deal in +the Software without restriction, including without limitation the rights to +use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies +of the Software, and to permit persons to whom the Software is furnished to do +so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in all +copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE +SOFTWARE. + +### Apache License ### + +All the remaining project files are covered by the Apache license: + +Copyright (c) 2011-2019 Canonical Ltd + +Licensed under the Apache License, Version 2.0 (the "License"); +you may not use this file except in compliance with the License. +You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + +Unless required by applicable law or agreed to in writing, software +distributed under the License is distributed on an "AS IS" BASIS, +WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +See the License for the specific language governing permissions and +limitations under the License. diff --git a/vendor/gopkg.in/yaml.v3/NOTICE b/vendor/gopkg.in/yaml.v3/NOTICE new file mode 100644 index 00000000..866d74a7 --- /dev/null +++ b/vendor/gopkg.in/yaml.v3/NOTICE @@ -0,0 +1,13 @@ +Copyright 2011-2016 Canonical Ltd. + +Licensed under the Apache License, Version 2.0 (the "License"); +you may not use this file except in compliance with the License. +You may obtain a copy of the License at + + http://www.apache.org/licenses/LICENSE-2.0 + +Unless required by applicable law or agreed to in writing, software +distributed under the License is distributed on an "AS IS" BASIS, +WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +See the License for the specific language governing permissions and +limitations under the License. diff --git a/vendor/gopkg.in/yaml.v3/README.md b/vendor/gopkg.in/yaml.v3/README.md new file mode 100644 index 00000000..08eb1bab --- /dev/null +++ b/vendor/gopkg.in/yaml.v3/README.md @@ -0,0 +1,150 @@ +# YAML support for the Go language + +Introduction +------------ + +The yaml package enables Go programs to comfortably encode and decode YAML +values. It was developed within [Canonical](https://www.canonical.com) as +part of the [juju](https://juju.ubuntu.com) project, and is based on a +pure Go port of the well-known [libyaml](http://pyyaml.org/wiki/LibYAML) +C library to parse and generate YAML data quickly and reliably. + +Compatibility +------------- + +The yaml package supports most of YAML 1.2, but preserves some behavior +from 1.1 for backwards compatibility. + +Specifically, as of v3 of the yaml package: + + - YAML 1.1 bools (_yes/no, on/off_) are supported as long as they are being + decoded into a typed bool value. Otherwise they behave as a string. Booleans + in YAML 1.2 are _true/false_ only. + - Octals encode and decode as _0777_ per YAML 1.1, rather than _0o777_ + as specified in YAML 1.2, because most parsers still use the old format. + Octals in the _0o777_ format are supported though, so new files work. + - Does not support base-60 floats. These are gone from YAML 1.2, and were + actually never supported by this package as it's clearly a poor choice. + +and offers backwards +compatibility with YAML 1.1 in some cases. +1.2, including support for +anchors, tags, map merging, etc. Multi-document unmarshalling is not yet +implemented, and base-60 floats from YAML 1.1 are purposefully not +supported since they're a poor design and are gone in YAML 1.2. + +Installation and usage +---------------------- + +The import path for the package is *gopkg.in/yaml.v3*. + +To install it, run: + + go get gopkg.in/yaml.v3 + +API documentation +----------------- + +If opened in a browser, the import path itself leads to the API documentation: + + - [https://gopkg.in/yaml.v3](https://gopkg.in/yaml.v3) + +API stability +------------- + +The package API for yaml v3 will remain stable as described in [gopkg.in](https://gopkg.in). + + +License +------- + +The yaml package is licensed under the MIT and Apache License 2.0 licenses. +Please see the LICENSE file for details. + + +Example +------- + +```Go +package main + +import ( + "fmt" + "log" + + "gopkg.in/yaml.v3" +) + +var data = ` +a: Easy! +b: + c: 2 + d: [3, 4] +` + +// Note: struct fields must be public in order for unmarshal to +// correctly populate the data. +type T struct { + A string + B struct { + RenamedC int `yaml:"c"` + D []int `yaml:",flow"` + } +} + +func main() { + t := T{} + + err := yaml.Unmarshal([]byte(data), &t) + if err != nil { + log.Fatalf("error: %v", err) + } + fmt.Printf("--- t:\n%v\n\n", t) + + d, err := yaml.Marshal(&t) + if err != nil { + log.Fatalf("error: %v", err) + } + fmt.Printf("--- t dump:\n%s\n\n", string(d)) + + m := make(map[interface{}]interface{}) + + err = yaml.Unmarshal([]byte(data), &m) + if err != nil { + log.Fatalf("error: %v", err) + } + fmt.Printf("--- m:\n%v\n\n", m) + + d, err = yaml.Marshal(&m) + if err != nil { + log.Fatalf("error: %v", err) + } + fmt.Printf("--- m dump:\n%s\n\n", string(d)) +} +``` + +This example will generate the following output: + +``` +--- t: +{Easy! {2 [3 4]}} + +--- t dump: +a: Easy! +b: + c: 2 + d: [3, 4] + + +--- m: +map[a:Easy! b:map[c:2 d:[3 4]]] + +--- m dump: +a: Easy! +b: + c: 2 + d: + - 3 + - 4 +``` + diff --git a/vendor/gopkg.in/yaml.v3/apic.go b/vendor/gopkg.in/yaml.v3/apic.go new file mode 100644 index 00000000..ae7d049f --- /dev/null +++ b/vendor/gopkg.in/yaml.v3/apic.go @@ -0,0 +1,747 @@ +// +// Copyright (c) 2011-2019 Canonical Ltd +// Copyright (c) 2006-2010 Kirill Simonov +// +// Permission is hereby granted, free of charge, to any person obtaining a copy of +// this software and associated documentation files (the "Software"), to deal in +// the Software without restriction, including without limitation the rights to +// use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies +// of the Software, and to permit persons to whom the Software is furnished to do +// so, subject to the following conditions: +// +// The above copyright notice and this permission notice shall be included in all +// copies or substantial portions of the Software. +// +// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE +// SOFTWARE. + +package yaml + +import ( + "io" +) + +func yaml_insert_token(parser *yaml_parser_t, pos int, token *yaml_token_t) { + //fmt.Println("yaml_insert_token", "pos:", pos, "typ:", token.typ, "head:", parser.tokens_head, "len:", len(parser.tokens)) + + // Check if we can move the queue at the beginning of the buffer. + if parser.tokens_head > 0 && len(parser.tokens) == cap(parser.tokens) { + if parser.tokens_head != len(parser.tokens) { + copy(parser.tokens, parser.tokens[parser.tokens_head:]) + } + parser.tokens = parser.tokens[:len(parser.tokens)-parser.tokens_head] + parser.tokens_head = 0 + } + parser.tokens = append(parser.tokens, *token) + if pos < 0 { + return + } + copy(parser.tokens[parser.tokens_head+pos+1:], parser.tokens[parser.tokens_head+pos:]) + parser.tokens[parser.tokens_head+pos] = *token +} + +// Create a new parser object. +func yaml_parser_initialize(parser *yaml_parser_t) bool { + *parser = yaml_parser_t{ + raw_buffer: make([]byte, 0, input_raw_buffer_size), + buffer: make([]byte, 0, input_buffer_size), + } + return true +} + +// Destroy a parser object. +func yaml_parser_delete(parser *yaml_parser_t) { + *parser = yaml_parser_t{} +} + +// String read handler. +func yaml_string_read_handler(parser *yaml_parser_t, buffer []byte) (n int, err error) { + if parser.input_pos == len(parser.input) { + return 0, io.EOF + } + n = copy(buffer, parser.input[parser.input_pos:]) + parser.input_pos += n + return n, nil +} + +// Reader read handler. +func yaml_reader_read_handler(parser *yaml_parser_t, buffer []byte) (n int, err error) { + return parser.input_reader.Read(buffer) +} + +// Set a string input. +func yaml_parser_set_input_string(parser *yaml_parser_t, input []byte) { + if parser.read_handler != nil { + panic("must set the input source only once") + } + parser.read_handler = yaml_string_read_handler + parser.input = input + parser.input_pos = 0 +} + +// Set a file input. +func yaml_parser_set_input_reader(parser *yaml_parser_t, r io.Reader) { + if parser.read_handler != nil { + panic("must set the input source only once") + } + parser.read_handler = yaml_reader_read_handler + parser.input_reader = r +} + +// Set the source encoding. +func yaml_parser_set_encoding(parser *yaml_parser_t, encoding yaml_encoding_t) { + if parser.encoding != yaml_ANY_ENCODING { + panic("must set the encoding only once") + } + parser.encoding = encoding +} + +// Create a new emitter object. +func yaml_emitter_initialize(emitter *yaml_emitter_t) { + *emitter = yaml_emitter_t{ + buffer: make([]byte, output_buffer_size), + raw_buffer: make([]byte, 0, output_raw_buffer_size), + states: make([]yaml_emitter_state_t, 0, initial_stack_size), + events: make([]yaml_event_t, 0, initial_queue_size), + best_width: -1, + } +} + +// Destroy an emitter object. +func yaml_emitter_delete(emitter *yaml_emitter_t) { + *emitter = yaml_emitter_t{} +} + +// String write handler. +func yaml_string_write_handler(emitter *yaml_emitter_t, buffer []byte) error { + *emitter.output_buffer = append(*emitter.output_buffer, buffer...) + return nil +} + +// yaml_writer_write_handler uses emitter.output_writer to write the +// emitted text. +func yaml_writer_write_handler(emitter *yaml_emitter_t, buffer []byte) error { + _, err := emitter.output_writer.Write(buffer) + return err +} + +// Set a string output. +func yaml_emitter_set_output_string(emitter *yaml_emitter_t, output_buffer *[]byte) { + if emitter.write_handler != nil { + panic("must set the output target only once") + } + emitter.write_handler = yaml_string_write_handler + emitter.output_buffer = output_buffer +} + +// Set a file output. +func yaml_emitter_set_output_writer(emitter *yaml_emitter_t, w io.Writer) { + if emitter.write_handler != nil { + panic("must set the output target only once") + } + emitter.write_handler = yaml_writer_write_handler + emitter.output_writer = w +} + +// Set the output encoding. +func yaml_emitter_set_encoding(emitter *yaml_emitter_t, encoding yaml_encoding_t) { + if emitter.encoding != yaml_ANY_ENCODING { + panic("must set the output encoding only once") + } + emitter.encoding = encoding +} + +// Set the canonical output style. +func yaml_emitter_set_canonical(emitter *yaml_emitter_t, canonical bool) { + emitter.canonical = canonical +} + +// Set the indentation increment. +func yaml_emitter_set_indent(emitter *yaml_emitter_t, indent int) { + if indent < 2 || indent > 9 { + indent = 2 + } + emitter.best_indent = indent +} + +// Set the preferred line width. +func yaml_emitter_set_width(emitter *yaml_emitter_t, width int) { + if width < 0 { + width = -1 + } + emitter.best_width = width +} + +// Set if unescaped non-ASCII characters are allowed. +func yaml_emitter_set_unicode(emitter *yaml_emitter_t, unicode bool) { + emitter.unicode = unicode +} + +// Set the preferred line break character. +func yaml_emitter_set_break(emitter *yaml_emitter_t, line_break yaml_break_t) { + emitter.line_break = line_break +} + +///* +// * Destroy a token object. +// */ +// +//YAML_DECLARE(void) +//yaml_token_delete(yaml_token_t *token) +//{ +// assert(token); // Non-NULL token object expected. +// +// switch (token.type) +// { +// case YAML_TAG_DIRECTIVE_TOKEN: +// yaml_free(token.data.tag_directive.handle); +// yaml_free(token.data.tag_directive.prefix); +// break; +// +// case YAML_ALIAS_TOKEN: +// yaml_free(token.data.alias.value); +// break; +// +// case YAML_ANCHOR_TOKEN: +// yaml_free(token.data.anchor.value); +// break; +// +// case YAML_TAG_TOKEN: +// yaml_free(token.data.tag.handle); +// yaml_free(token.data.tag.suffix); +// break; +// +// case YAML_SCALAR_TOKEN: +// yaml_free(token.data.scalar.value); +// break; +// +// default: +// break; +// } +// +// memset(token, 0, sizeof(yaml_token_t)); +//} +// +///* +// * Check if a string is a valid UTF-8 sequence. +// * +// * Check 'reader.c' for more details on UTF-8 encoding. +// */ +// +//static int +//yaml_check_utf8(yaml_char_t *start, size_t length) +//{ +// yaml_char_t *end = start+length; +// yaml_char_t *pointer = start; +// +// while (pointer < end) { +// unsigned char octet; +// unsigned int width; +// unsigned int value; +// size_t k; +// +// octet = pointer[0]; +// width = (octet & 0x80) == 0x00 ? 1 : +// (octet & 0xE0) == 0xC0 ? 2 : +// (octet & 0xF0) == 0xE0 ? 3 : +// (octet & 0xF8) == 0xF0 ? 4 : 0; +// value = (octet & 0x80) == 0x00 ? octet & 0x7F : +// (octet & 0xE0) == 0xC0 ? octet & 0x1F : +// (octet & 0xF0) == 0xE0 ? octet & 0x0F : +// (octet & 0xF8) == 0xF0 ? octet & 0x07 : 0; +// if (!width) return 0; +// if (pointer+width > end) return 0; +// for (k = 1; k < width; k ++) { +// octet = pointer[k]; +// if ((octet & 0xC0) != 0x80) return 0; +// value = (value << 6) + (octet & 0x3F); +// } +// if (!((width == 1) || +// (width == 2 && value >= 0x80) || +// (width == 3 && value >= 0x800) || +// (width == 4 && value >= 0x10000))) return 0; +// +// pointer += width; +// } +// +// return 1; +//} +// + +// Create STREAM-START. +func yaml_stream_start_event_initialize(event *yaml_event_t, encoding yaml_encoding_t) { + *event = yaml_event_t{ + typ: yaml_STREAM_START_EVENT, + encoding: encoding, + } +} + +// Create STREAM-END. +func yaml_stream_end_event_initialize(event *yaml_event_t) { + *event = yaml_event_t{ + typ: yaml_STREAM_END_EVENT, + } +} + +// Create DOCUMENT-START. +func yaml_document_start_event_initialize( + event *yaml_event_t, + version_directive *yaml_version_directive_t, + tag_directives []yaml_tag_directive_t, + implicit bool, +) { + *event = yaml_event_t{ + typ: yaml_DOCUMENT_START_EVENT, + version_directive: version_directive, + tag_directives: tag_directives, + implicit: implicit, + } +} + +// Create DOCUMENT-END. +func yaml_document_end_event_initialize(event *yaml_event_t, implicit bool) { + *event = yaml_event_t{ + typ: yaml_DOCUMENT_END_EVENT, + implicit: implicit, + } +} + +// Create ALIAS. +func yaml_alias_event_initialize(event *yaml_event_t, anchor []byte) bool { + *event = yaml_event_t{ + typ: yaml_ALIAS_EVENT, + anchor: anchor, + } + return true +} + +// Create SCALAR. +func yaml_scalar_event_initialize(event *yaml_event_t, anchor, tag, value []byte, plain_implicit, quoted_implicit bool, style yaml_scalar_style_t) bool { + *event = yaml_event_t{ + typ: yaml_SCALAR_EVENT, + anchor: anchor, + tag: tag, + value: value, + implicit: plain_implicit, + quoted_implicit: quoted_implicit, + style: yaml_style_t(style), + } + return true +} + +// Create SEQUENCE-START. +func yaml_sequence_start_event_initialize(event *yaml_event_t, anchor, tag []byte, implicit bool, style yaml_sequence_style_t) bool { + *event = yaml_event_t{ + typ: yaml_SEQUENCE_START_EVENT, + anchor: anchor, + tag: tag, + implicit: implicit, + style: yaml_style_t(style), + } + return true +} + +// Create SEQUENCE-END. +func yaml_sequence_end_event_initialize(event *yaml_event_t) bool { + *event = yaml_event_t{ + typ: yaml_SEQUENCE_END_EVENT, + } + return true +} + +// Create MAPPING-START. +func yaml_mapping_start_event_initialize(event *yaml_event_t, anchor, tag []byte, implicit bool, style yaml_mapping_style_t) { + *event = yaml_event_t{ + typ: yaml_MAPPING_START_EVENT, + anchor: anchor, + tag: tag, + implicit: implicit, + style: yaml_style_t(style), + } +} + +// Create MAPPING-END. +func yaml_mapping_end_event_initialize(event *yaml_event_t) { + *event = yaml_event_t{ + typ: yaml_MAPPING_END_EVENT, + } +} + +// Destroy an event object. +func yaml_event_delete(event *yaml_event_t) { + *event = yaml_event_t{} +} + +///* +// * Create a document object. +// */ +// +//YAML_DECLARE(int) +//yaml_document_initialize(document *yaml_document_t, +// version_directive *yaml_version_directive_t, +// tag_directives_start *yaml_tag_directive_t, +// tag_directives_end *yaml_tag_directive_t, +// start_implicit int, end_implicit int) +//{ +// struct { +// error yaml_error_type_t +// } context +// struct { +// start *yaml_node_t +// end *yaml_node_t +// top *yaml_node_t +// } nodes = { NULL, NULL, NULL } +// version_directive_copy *yaml_version_directive_t = NULL +// struct { +// start *yaml_tag_directive_t +// end *yaml_tag_directive_t +// top *yaml_tag_directive_t +// } tag_directives_copy = { NULL, NULL, NULL } +// value yaml_tag_directive_t = { NULL, NULL } +// mark yaml_mark_t = { 0, 0, 0 } +// +// assert(document) // Non-NULL document object is expected. +// assert((tag_directives_start && tag_directives_end) || +// (tag_directives_start == tag_directives_end)) +// // Valid tag directives are expected. +// +// if (!STACK_INIT(&context, nodes, INITIAL_STACK_SIZE)) goto error +// +// if (version_directive) { +// version_directive_copy = yaml_malloc(sizeof(yaml_version_directive_t)) +// if (!version_directive_copy) goto error +// version_directive_copy.major = version_directive.major +// version_directive_copy.minor = version_directive.minor +// } +// +// if (tag_directives_start != tag_directives_end) { +// tag_directive *yaml_tag_directive_t +// if (!STACK_INIT(&context, tag_directives_copy, INITIAL_STACK_SIZE)) +// goto error +// for (tag_directive = tag_directives_start +// tag_directive != tag_directives_end; tag_directive ++) { +// assert(tag_directive.handle) +// assert(tag_directive.prefix) +// if (!yaml_check_utf8(tag_directive.handle, +// strlen((char *)tag_directive.handle))) +// goto error +// if (!yaml_check_utf8(tag_directive.prefix, +// strlen((char *)tag_directive.prefix))) +// goto error +// value.handle = yaml_strdup(tag_directive.handle) +// value.prefix = yaml_strdup(tag_directive.prefix) +// if (!value.handle || !value.prefix) goto error +// if (!PUSH(&context, tag_directives_copy, value)) +// goto error +// value.handle = NULL +// value.prefix = NULL +// } +// } +// +// DOCUMENT_INIT(*document, nodes.start, nodes.end, version_directive_copy, +// tag_directives_copy.start, tag_directives_copy.top, +// start_implicit, end_implicit, mark, mark) +// +// return 1 +// +//error: +// STACK_DEL(&context, nodes) +// yaml_free(version_directive_copy) +// while (!STACK_EMPTY(&context, tag_directives_copy)) { +// value yaml_tag_directive_t = POP(&context, tag_directives_copy) +// yaml_free(value.handle) +// yaml_free(value.prefix) +// } +// STACK_DEL(&context, tag_directives_copy) +// yaml_free(value.handle) +// yaml_free(value.prefix) +// +// return 0 +//} +// +///* +// * Destroy a document object. +// */ +// +//YAML_DECLARE(void) +//yaml_document_delete(document *yaml_document_t) +//{ +// struct { +// error yaml_error_type_t +// } context +// tag_directive *yaml_tag_directive_t +// +// context.error = YAML_NO_ERROR // Eliminate a compiler warning. +// +// assert(document) // Non-NULL document object is expected. +// +// while (!STACK_EMPTY(&context, document.nodes)) { +// node yaml_node_t = POP(&context, document.nodes) +// yaml_free(node.tag) +// switch (node.type) { +// case YAML_SCALAR_NODE: +// yaml_free(node.data.scalar.value) +// break +// case YAML_SEQUENCE_NODE: +// STACK_DEL(&context, node.data.sequence.items) +// break +// case YAML_MAPPING_NODE: +// STACK_DEL(&context, node.data.mapping.pairs) +// break +// default: +// assert(0) // Should not happen. +// } +// } +// STACK_DEL(&context, document.nodes) +// +// yaml_free(document.version_directive) +// for (tag_directive = document.tag_directives.start +// tag_directive != document.tag_directives.end +// tag_directive++) { +// yaml_free(tag_directive.handle) +// yaml_free(tag_directive.prefix) +// } +// yaml_free(document.tag_directives.start) +// +// memset(document, 0, sizeof(yaml_document_t)) +//} +// +///** +// * Get a document node. +// */ +// +//YAML_DECLARE(yaml_node_t *) +//yaml_document_get_node(document *yaml_document_t, index int) +//{ +// assert(document) // Non-NULL document object is expected. +// +// if (index > 0 && document.nodes.start + index <= document.nodes.top) { +// return document.nodes.start + index - 1 +// } +// return NULL +//} +// +///** +// * Get the root object. +// */ +// +//YAML_DECLARE(yaml_node_t *) +//yaml_document_get_root_node(document *yaml_document_t) +//{ +// assert(document) // Non-NULL document object is expected. +// +// if (document.nodes.top != document.nodes.start) { +// return document.nodes.start +// } +// return NULL +//} +// +///* +// * Add a scalar node to a document. +// */ +// +//YAML_DECLARE(int) +//yaml_document_add_scalar(document *yaml_document_t, +// tag *yaml_char_t, value *yaml_char_t, length int, +// style yaml_scalar_style_t) +//{ +// struct { +// error yaml_error_type_t +// } context +// mark yaml_mark_t = { 0, 0, 0 } +// tag_copy *yaml_char_t = NULL +// value_copy *yaml_char_t = NULL +// node yaml_node_t +// +// assert(document) // Non-NULL document object is expected. +// assert(value) // Non-NULL value is expected. +// +// if (!tag) { +// tag = (yaml_char_t *)YAML_DEFAULT_SCALAR_TAG +// } +// +// if (!yaml_check_utf8(tag, strlen((char *)tag))) goto error +// tag_copy = yaml_strdup(tag) +// if (!tag_copy) goto error +// +// if (length < 0) { +// length = strlen((char *)value) +// } +// +// if (!yaml_check_utf8(value, length)) goto error +// value_copy = yaml_malloc(length+1) +// if (!value_copy) goto error +// memcpy(value_copy, value, length) +// value_copy[length] = '\0' +// +// SCALAR_NODE_INIT(node, tag_copy, value_copy, length, style, mark, mark) +// if (!PUSH(&context, document.nodes, node)) goto error +// +// return document.nodes.top - document.nodes.start +// +//error: +// yaml_free(tag_copy) +// yaml_free(value_copy) +// +// return 0 +//} +// +///* +// * Add a sequence node to a document. +// */ +// +//YAML_DECLARE(int) +//yaml_document_add_sequence(document *yaml_document_t, +// tag *yaml_char_t, style yaml_sequence_style_t) +//{ +// struct { +// error yaml_error_type_t +// } context +// mark yaml_mark_t = { 0, 0, 0 } +// tag_copy *yaml_char_t = NULL +// struct { +// start *yaml_node_item_t +// end *yaml_node_item_t +// top *yaml_node_item_t +// } items = { NULL, NULL, NULL } +// node yaml_node_t +// +// assert(document) // Non-NULL document object is expected. +// +// if (!tag) { +// tag = (yaml_char_t *)YAML_DEFAULT_SEQUENCE_TAG +// } +// +// if (!yaml_check_utf8(tag, strlen((char *)tag))) goto error +// tag_copy = yaml_strdup(tag) +// if (!tag_copy) goto error +// +// if (!STACK_INIT(&context, items, INITIAL_STACK_SIZE)) goto error +// +// SEQUENCE_NODE_INIT(node, tag_copy, items.start, items.end, +// style, mark, mark) +// if (!PUSH(&context, document.nodes, node)) goto error +// +// return document.nodes.top - document.nodes.start +// +//error: +// STACK_DEL(&context, items) +// yaml_free(tag_copy) +// +// return 0 +//} +// +///* +// * Add a mapping node to a document. +// */ +// +//YAML_DECLARE(int) +//yaml_document_add_mapping(document *yaml_document_t, +// tag *yaml_char_t, style yaml_mapping_style_t) +//{ +// struct { +// error yaml_error_type_t +// } context +// mark yaml_mark_t = { 0, 0, 0 } +// tag_copy *yaml_char_t = NULL +// struct { +// start *yaml_node_pair_t +// end *yaml_node_pair_t +// top *yaml_node_pair_t +// } pairs = { NULL, NULL, NULL } +// node yaml_node_t +// +// assert(document) // Non-NULL document object is expected. +// +// if (!tag) { +// tag = (yaml_char_t *)YAML_DEFAULT_MAPPING_TAG +// } +// +// if (!yaml_check_utf8(tag, strlen((char *)tag))) goto error +// tag_copy = yaml_strdup(tag) +// if (!tag_copy) goto error +// +// if (!STACK_INIT(&context, pairs, INITIAL_STACK_SIZE)) goto error +// +// MAPPING_NODE_INIT(node, tag_copy, pairs.start, pairs.end, +// style, mark, mark) +// if (!PUSH(&context, document.nodes, node)) goto error +// +// return document.nodes.top - document.nodes.start +// +//error: +// STACK_DEL(&context, pairs) +// yaml_free(tag_copy) +// +// return 0 +//} +// +///* +// * Append an item to a sequence node. +// */ +// +//YAML_DECLARE(int) +//yaml_document_append_sequence_item(document *yaml_document_t, +// sequence int, item int) +//{ +// struct { +// error yaml_error_type_t +// } context +// +// assert(document) // Non-NULL document is required. +// assert(sequence > 0 +// && document.nodes.start + sequence <= document.nodes.top) +// // Valid sequence id is required. +// assert(document.nodes.start[sequence-1].type == YAML_SEQUENCE_NODE) +// // A sequence node is required. +// assert(item > 0 && document.nodes.start + item <= document.nodes.top) +// // Valid item id is required. +// +// if (!PUSH(&context, +// document.nodes.start[sequence-1].data.sequence.items, item)) +// return 0 +// +// return 1 +//} +// +///* +// * Append a pair of a key and a value to a mapping node. +// */ +// +//YAML_DECLARE(int) +//yaml_document_append_mapping_pair(document *yaml_document_t, +// mapping int, key int, value int) +//{ +// struct { +// error yaml_error_type_t +// } context +// +// pair yaml_node_pair_t +// +// assert(document) // Non-NULL document is required. +// assert(mapping > 0 +// && document.nodes.start + mapping <= document.nodes.top) +// // Valid mapping id is required. +// assert(document.nodes.start[mapping-1].type == YAML_MAPPING_NODE) +// // A mapping node is required. +// assert(key > 0 && document.nodes.start + key <= document.nodes.top) +// // Valid key id is required. +// assert(value > 0 && document.nodes.start + value <= document.nodes.top) +// // Valid value id is required. +// +// pair.key = key +// pair.value = value +// +// if (!PUSH(&context, +// document.nodes.start[mapping-1].data.mapping.pairs, pair)) +// return 0 +// +// return 1 +//} +// +// diff --git a/vendor/gopkg.in/yaml.v3/decode.go b/vendor/gopkg.in/yaml.v3/decode.go new file mode 100644 index 00000000..df36e3a3 --- /dev/null +++ b/vendor/gopkg.in/yaml.v3/decode.go @@ -0,0 +1,950 @@ +// +// Copyright (c) 2011-2019 Canonical Ltd +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. + +package yaml + +import ( + "encoding" + "encoding/base64" + "fmt" + "io" + "math" + "reflect" + "strconv" + "time" +) + +// ---------------------------------------------------------------------------- +// Parser, produces a node tree out of a libyaml event stream. + +type parser struct { + parser yaml_parser_t + event yaml_event_t + doc *Node + anchors map[string]*Node + doneInit bool + textless bool +} + +func newParser(b []byte) *parser { + p := parser{} + if !yaml_parser_initialize(&p.parser) { + panic("failed to initialize YAML emitter") + } + if len(b) == 0 { + b = []byte{'\n'} + } + yaml_parser_set_input_string(&p.parser, b) + return &p +} + +func newParserFromReader(r io.Reader) *parser { + p := parser{} + if !yaml_parser_initialize(&p.parser) { + panic("failed to initialize YAML emitter") + } + yaml_parser_set_input_reader(&p.parser, r) + return &p +} + +func (p *parser) init() { + if p.doneInit { + return + } + p.anchors = make(map[string]*Node) + p.expect(yaml_STREAM_START_EVENT) + p.doneInit = true +} + +func (p *parser) destroy() { + if p.event.typ != yaml_NO_EVENT { + yaml_event_delete(&p.event) + } + yaml_parser_delete(&p.parser) +} + +// expect consumes an event from the event stream and +// checks that it's of the expected type. +func (p *parser) expect(e yaml_event_type_t) { + if p.event.typ == yaml_NO_EVENT { + if !yaml_parser_parse(&p.parser, &p.event) { + p.fail() + } + } + if p.event.typ == yaml_STREAM_END_EVENT { + failf("attempted to go past the end of stream; corrupted value?") + } + if p.event.typ != e { + p.parser.problem = fmt.Sprintf("expected %s event but got %s", e, p.event.typ) + p.fail() + } + yaml_event_delete(&p.event) + p.event.typ = yaml_NO_EVENT +} + +// peek peeks at the next event in the event stream, +// puts the results into p.event and returns the event type. +func (p *parser) peek() yaml_event_type_t { + if p.event.typ != yaml_NO_EVENT { + return p.event.typ + } + if !yaml_parser_parse(&p.parser, &p.event) { + p.fail() + } + return p.event.typ +} + +func (p *parser) fail() { + var where string + var line int + if p.parser.context_mark.line != 0 { + line = p.parser.context_mark.line + // Scanner errors don't iterate line before returning error + if p.parser.error == yaml_SCANNER_ERROR { + line++ + } + } else if p.parser.problem_mark.line != 0 { + line = p.parser.problem_mark.line + // Scanner errors don't iterate line before returning error + if p.parser.error == yaml_SCANNER_ERROR { + line++ + } + } + if line != 0 { + where = "line " + strconv.Itoa(line) + ": " + } + var msg string + if len(p.parser.problem) > 0 { + msg = p.parser.problem + } else { + msg = "unknown problem parsing YAML content" + } + failf("%s%s", where, msg) +} + +func (p *parser) anchor(n *Node, anchor []byte) { + if anchor != nil { + n.Anchor = string(anchor) + p.anchors[n.Anchor] = n + } +} + +func (p *parser) parse() *Node { + p.init() + switch p.peek() { + case yaml_SCALAR_EVENT: + return p.scalar() + case yaml_ALIAS_EVENT: + return p.alias() + case yaml_MAPPING_START_EVENT: + return p.mapping() + case yaml_SEQUENCE_START_EVENT: + return p.sequence() + case yaml_DOCUMENT_START_EVENT: + return p.document() + case yaml_STREAM_END_EVENT: + // Happens when attempting to decode an empty buffer. + return nil + case yaml_TAIL_COMMENT_EVENT: + panic("internal error: unexpected tail comment event (please report)") + default: + panic("internal error: attempted to parse unknown event (please report): " + p.event.typ.String()) + } +} + +func (p *parser) node(kind Kind, defaultTag, tag, value string) *Node { + var style Style + if tag != "" && tag != "!" { + tag = shortTag(tag) + style = TaggedStyle + } else if defaultTag != "" { + tag = defaultTag + } else if kind == ScalarNode { + tag, _ = resolve("", value) + } + n := &Node{ + Kind: kind, + Tag: tag, + Value: value, + Style: style, + } + if !p.textless { + n.Line = p.event.start_mark.line + 1 + n.Column = p.event.start_mark.column + 1 + n.HeadComment = string(p.event.head_comment) + n.LineComment = string(p.event.line_comment) + n.FootComment = string(p.event.foot_comment) + } + return n +} + +func (p *parser) parseChild(parent *Node) *Node { + child := p.parse() + parent.Content = append(parent.Content, child) + return child +} + +func (p *parser) document() *Node { + n := p.node(DocumentNode, "", "", "") + p.doc = n + p.expect(yaml_DOCUMENT_START_EVENT) + p.parseChild(n) + if p.peek() == yaml_DOCUMENT_END_EVENT { + n.FootComment = string(p.event.foot_comment) + } + p.expect(yaml_DOCUMENT_END_EVENT) + return n +} + +func (p *parser) alias() *Node { + n := p.node(AliasNode, "", "", string(p.event.anchor)) + n.Alias = p.anchors[n.Value] + if n.Alias == nil { + failf("unknown anchor '%s' referenced", n.Value) + } + p.expect(yaml_ALIAS_EVENT) + return n +} + +func (p *parser) scalar() *Node { + var parsedStyle = p.event.scalar_style() + var nodeStyle Style + switch { + case parsedStyle&yaml_DOUBLE_QUOTED_SCALAR_STYLE != 0: + nodeStyle = DoubleQuotedStyle + case parsedStyle&yaml_SINGLE_QUOTED_SCALAR_STYLE != 0: + nodeStyle = SingleQuotedStyle + case parsedStyle&yaml_LITERAL_SCALAR_STYLE != 0: + nodeStyle = LiteralStyle + case parsedStyle&yaml_FOLDED_SCALAR_STYLE != 0: + nodeStyle = FoldedStyle + } + var nodeValue = string(p.event.value) + var nodeTag = string(p.event.tag) + var defaultTag string + if nodeStyle == 0 { + if nodeValue == "<<" { + defaultTag = mergeTag + } + } else { + defaultTag = strTag + } + n := p.node(ScalarNode, defaultTag, nodeTag, nodeValue) + n.Style |= nodeStyle + p.anchor(n, p.event.anchor) + p.expect(yaml_SCALAR_EVENT) + return n +} + +func (p *parser) sequence() *Node { + n := p.node(SequenceNode, seqTag, string(p.event.tag), "") + if p.event.sequence_style()&yaml_FLOW_SEQUENCE_STYLE != 0 { + n.Style |= FlowStyle + } + p.anchor(n, p.event.anchor) + p.expect(yaml_SEQUENCE_START_EVENT) + for p.peek() != yaml_SEQUENCE_END_EVENT { + p.parseChild(n) + } + n.LineComment = string(p.event.line_comment) + n.FootComment = string(p.event.foot_comment) + p.expect(yaml_SEQUENCE_END_EVENT) + return n +} + +func (p *parser) mapping() *Node { + n := p.node(MappingNode, mapTag, string(p.event.tag), "") + block := true + if p.event.mapping_style()&yaml_FLOW_MAPPING_STYLE != 0 { + block = false + n.Style |= FlowStyle + } + p.anchor(n, p.event.anchor) + p.expect(yaml_MAPPING_START_EVENT) + for p.peek() != yaml_MAPPING_END_EVENT { + k := p.parseChild(n) + if block && k.FootComment != "" { + // Must be a foot comment for the prior value when being dedented. + if len(n.Content) > 2 { + n.Content[len(n.Content)-3].FootComment = k.FootComment + k.FootComment = "" + } + } + v := p.parseChild(n) + if k.FootComment == "" && v.FootComment != "" { + k.FootComment = v.FootComment + v.FootComment = "" + } + if p.peek() == yaml_TAIL_COMMENT_EVENT { + if k.FootComment == "" { + k.FootComment = string(p.event.foot_comment) + } + p.expect(yaml_TAIL_COMMENT_EVENT) + } + } + n.LineComment = string(p.event.line_comment) + n.FootComment = string(p.event.foot_comment) + if n.Style&FlowStyle == 0 && n.FootComment != "" && len(n.Content) > 1 { + n.Content[len(n.Content)-2].FootComment = n.FootComment + n.FootComment = "" + } + p.expect(yaml_MAPPING_END_EVENT) + return n +} + +// ---------------------------------------------------------------------------- +// Decoder, unmarshals a node into a provided value. + +type decoder struct { + doc *Node + aliases map[*Node]bool + terrors []string + + stringMapType reflect.Type + generalMapType reflect.Type + + knownFields bool + uniqueKeys bool + decodeCount int + aliasCount int + aliasDepth int +} + +var ( + nodeType = reflect.TypeOf(Node{}) + durationType = reflect.TypeOf(time.Duration(0)) + stringMapType = reflect.TypeOf(map[string]interface{}{}) + generalMapType = reflect.TypeOf(map[interface{}]interface{}{}) + ifaceType = generalMapType.Elem() + timeType = reflect.TypeOf(time.Time{}) + ptrTimeType = reflect.TypeOf(&time.Time{}) +) + +func newDecoder() *decoder { + d := &decoder{ + stringMapType: stringMapType, + generalMapType: generalMapType, + uniqueKeys: true, + } + d.aliases = make(map[*Node]bool) + return d +} + +func (d *decoder) terror(n *Node, tag string, out reflect.Value) { + if n.Tag != "" { + tag = n.Tag + } + value := n.Value + if tag != seqTag && tag != mapTag { + if len(value) > 10 { + value = " `" + value[:7] + "...`" + } else { + value = " `" + value + "`" + } + } + d.terrors = append(d.terrors, fmt.Sprintf("line %d: cannot unmarshal %s%s into %s", n.Line, shortTag(tag), value, out.Type())) +} + +func (d *decoder) callUnmarshaler(n *Node, u Unmarshaler) (good bool) { + err := u.UnmarshalYAML(n) + if e, ok := err.(*TypeError); ok { + d.terrors = append(d.terrors, e.Errors...) + return false + } + if err != nil { + fail(err) + } + return true +} + +func (d *decoder) callObsoleteUnmarshaler(n *Node, u obsoleteUnmarshaler) (good bool) { + terrlen := len(d.terrors) + err := u.UnmarshalYAML(func(v interface{}) (err error) { + defer handleErr(&err) + d.unmarshal(n, reflect.ValueOf(v)) + if len(d.terrors) > terrlen { + issues := d.terrors[terrlen:] + d.terrors = d.terrors[:terrlen] + return &TypeError{issues} + } + return nil + }) + if e, ok := err.(*TypeError); ok { + d.terrors = append(d.terrors, e.Errors...) + return false + } + if err != nil { + fail(err) + } + return true +} + +// d.prepare initializes and dereferences pointers and calls UnmarshalYAML +// if a value is found to implement it. +// It returns the initialized and dereferenced out value, whether +// unmarshalling was already done by UnmarshalYAML, and if so whether +// its types unmarshalled appropriately. +// +// If n holds a null value, prepare returns before doing anything. +func (d *decoder) prepare(n *Node, out reflect.Value) (newout reflect.Value, unmarshaled, good bool) { + if n.ShortTag() == nullTag { + return out, false, false + } + again := true + for again { + again = false + if out.Kind() == reflect.Ptr { + if out.IsNil() { + out.Set(reflect.New(out.Type().Elem())) + } + out = out.Elem() + again = true + } + if out.CanAddr() { + outi := out.Addr().Interface() + if u, ok := outi.(Unmarshaler); ok { + good = d.callUnmarshaler(n, u) + return out, true, good + } + if u, ok := outi.(obsoleteUnmarshaler); ok { + good = d.callObsoleteUnmarshaler(n, u) + return out, true, good + } + } + } + return out, false, false +} + +func (d *decoder) fieldByIndex(n *Node, v reflect.Value, index []int) (field reflect.Value) { + if n.ShortTag() == nullTag { + return reflect.Value{} + } + for _, num := range index { + for { + if v.Kind() == reflect.Ptr { + if v.IsNil() { + v.Set(reflect.New(v.Type().Elem())) + } + v = v.Elem() + continue + } + break + } + v = v.Field(num) + } + return v +} + +const ( + // 400,000 decode operations is ~500kb of dense object declarations, or + // ~5kb of dense object declarations with 10000% alias expansion + alias_ratio_range_low = 400000 + + // 4,000,000 decode operations is ~5MB of dense object declarations, or + // ~4.5MB of dense object declarations with 10% alias expansion + alias_ratio_range_high = 4000000 + + // alias_ratio_range is the range over which we scale allowed alias ratios + alias_ratio_range = float64(alias_ratio_range_high - alias_ratio_range_low) +) + +func allowedAliasRatio(decodeCount int) float64 { + switch { + case decodeCount <= alias_ratio_range_low: + // allow 99% to come from alias expansion for small-to-medium documents + return 0.99 + case decodeCount >= alias_ratio_range_high: + // allow 10% to come from alias expansion for very large documents + return 0.10 + default: + // scale smoothly from 99% down to 10% over the range. + // this maps to 396,000 - 400,000 allowed alias-driven decodes over the range. + // 400,000 decode operations is ~100MB of allocations in worst-case scenarios (single-item maps). + return 0.99 - 0.89*(float64(decodeCount-alias_ratio_range_low)/alias_ratio_range) + } +} + +func (d *decoder) unmarshal(n *Node, out reflect.Value) (good bool) { + d.decodeCount++ + if d.aliasDepth > 0 { + d.aliasCount++ + } + if d.aliasCount > 100 && d.decodeCount > 1000 && float64(d.aliasCount)/float64(d.decodeCount) > allowedAliasRatio(d.decodeCount) { + failf("document contains excessive aliasing") + } + if out.Type() == nodeType { + out.Set(reflect.ValueOf(n).Elem()) + return true + } + switch n.Kind { + case DocumentNode: + return d.document(n, out) + case AliasNode: + return d.alias(n, out) + } + out, unmarshaled, good := d.prepare(n, out) + if unmarshaled { + return good + } + switch n.Kind { + case ScalarNode: + good = d.scalar(n, out) + case MappingNode: + good = d.mapping(n, out) + case SequenceNode: + good = d.sequence(n, out) + case 0: + if n.IsZero() { + return d.null(out) + } + fallthrough + default: + failf("cannot decode node with unknown kind %d", n.Kind) + } + return good +} + +func (d *decoder) document(n *Node, out reflect.Value) (good bool) { + if len(n.Content) == 1 { + d.doc = n + d.unmarshal(n.Content[0], out) + return true + } + return false +} + +func (d *decoder) alias(n *Node, out reflect.Value) (good bool) { + if d.aliases[n] { + // TODO this could actually be allowed in some circumstances. + failf("anchor '%s' value contains itself", n.Value) + } + d.aliases[n] = true + d.aliasDepth++ + good = d.unmarshal(n.Alias, out) + d.aliasDepth-- + delete(d.aliases, n) + return good +} + +var zeroValue reflect.Value + +func resetMap(out reflect.Value) { + for _, k := range out.MapKeys() { + out.SetMapIndex(k, zeroValue) + } +} + +func (d *decoder) null(out reflect.Value) bool { + if out.CanAddr() { + switch out.Kind() { + case reflect.Interface, reflect.Ptr, reflect.Map, reflect.Slice: + out.Set(reflect.Zero(out.Type())) + return true + } + } + return false +} + +func (d *decoder) scalar(n *Node, out reflect.Value) bool { + var tag string + var resolved interface{} + if n.indicatedString() { + tag = strTag + resolved = n.Value + } else { + tag, resolved = resolve(n.Tag, n.Value) + if tag == binaryTag { + data, err := base64.StdEncoding.DecodeString(resolved.(string)) + if err != nil { + failf("!!binary value contains invalid base64 data") + } + resolved = string(data) + } + } + if resolved == nil { + return d.null(out) + } + if resolvedv := reflect.ValueOf(resolved); out.Type() == resolvedv.Type() { + // We've resolved to exactly the type we want, so use that. + out.Set(resolvedv) + return true + } + // Perhaps we can use the value as a TextUnmarshaler to + // set its value. + if out.CanAddr() { + u, ok := out.Addr().Interface().(encoding.TextUnmarshaler) + if ok { + var text []byte + if tag == binaryTag { + text = []byte(resolved.(string)) + } else { + // We let any value be unmarshaled into TextUnmarshaler. + // That might be more lax than we'd like, but the + // TextUnmarshaler itself should bowl out any dubious values. + text = []byte(n.Value) + } + err := u.UnmarshalText(text) + if err != nil { + fail(err) + } + return true + } + } + switch out.Kind() { + case reflect.String: + if tag == binaryTag { + out.SetString(resolved.(string)) + return true + } + out.SetString(n.Value) + return true + case reflect.Interface: + out.Set(reflect.ValueOf(resolved)) + return true + case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64: + // This used to work in v2, but it's very unfriendly. + isDuration := out.Type() == durationType + + switch resolved := resolved.(type) { + case int: + if !isDuration && !out.OverflowInt(int64(resolved)) { + out.SetInt(int64(resolved)) + return true + } + case int64: + if !isDuration && !out.OverflowInt(resolved) { + out.SetInt(resolved) + return true + } + case uint64: + if !isDuration && resolved <= math.MaxInt64 && !out.OverflowInt(int64(resolved)) { + out.SetInt(int64(resolved)) + return true + } + case float64: + if !isDuration && resolved <= math.MaxInt64 && !out.OverflowInt(int64(resolved)) { + out.SetInt(int64(resolved)) + return true + } + case string: + if out.Type() == durationType { + d, err := time.ParseDuration(resolved) + if err == nil { + out.SetInt(int64(d)) + return true + } + } + } + case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64, reflect.Uintptr: + switch resolved := resolved.(type) { + case int: + if resolved >= 0 && !out.OverflowUint(uint64(resolved)) { + out.SetUint(uint64(resolved)) + return true + } + case int64: + if resolved >= 0 && !out.OverflowUint(uint64(resolved)) { + out.SetUint(uint64(resolved)) + return true + } + case uint64: + if !out.OverflowUint(uint64(resolved)) { + out.SetUint(uint64(resolved)) + return true + } + case float64: + if resolved <= math.MaxUint64 && !out.OverflowUint(uint64(resolved)) { + out.SetUint(uint64(resolved)) + return true + } + } + case reflect.Bool: + switch resolved := resolved.(type) { + case bool: + out.SetBool(resolved) + return true + case string: + // This offers some compatibility with the 1.1 spec (https://yaml.org/type/bool.html). + // It only works if explicitly attempting to unmarshal into a typed bool value. + switch resolved { + case "y", "Y", "yes", "Yes", "YES", "on", "On", "ON": + out.SetBool(true) + return true + case "n", "N", "no", "No", "NO", "off", "Off", "OFF": + out.SetBool(false) + return true + } + } + case reflect.Float32, reflect.Float64: + switch resolved := resolved.(type) { + case int: + out.SetFloat(float64(resolved)) + return true + case int64: + out.SetFloat(float64(resolved)) + return true + case uint64: + out.SetFloat(float64(resolved)) + return true + case float64: + out.SetFloat(resolved) + return true + } + case reflect.Struct: + if resolvedv := reflect.ValueOf(resolved); out.Type() == resolvedv.Type() { + out.Set(resolvedv) + return true + } + case reflect.Ptr: + panic("yaml internal error: please report the issue") + } + d.terror(n, tag, out) + return false +} + +func settableValueOf(i interface{}) reflect.Value { + v := reflect.ValueOf(i) + sv := reflect.New(v.Type()).Elem() + sv.Set(v) + return sv +} + +func (d *decoder) sequence(n *Node, out reflect.Value) (good bool) { + l := len(n.Content) + + var iface reflect.Value + switch out.Kind() { + case reflect.Slice: + out.Set(reflect.MakeSlice(out.Type(), l, l)) + case reflect.Array: + if l != out.Len() { + failf("invalid array: want %d elements but got %d", out.Len(), l) + } + case reflect.Interface: + // No type hints. Will have to use a generic sequence. + iface = out + out = settableValueOf(make([]interface{}, l)) + default: + d.terror(n, seqTag, out) + return false + } + et := out.Type().Elem() + + j := 0 + for i := 0; i < l; i++ { + e := reflect.New(et).Elem() + if ok := d.unmarshal(n.Content[i], e); ok { + out.Index(j).Set(e) + j++ + } + } + if out.Kind() != reflect.Array { + out.Set(out.Slice(0, j)) + } + if iface.IsValid() { + iface.Set(out) + } + return true +} + +func (d *decoder) mapping(n *Node, out reflect.Value) (good bool) { + l := len(n.Content) + if d.uniqueKeys { + nerrs := len(d.terrors) + for i := 0; i < l; i += 2 { + ni := n.Content[i] + for j := i + 2; j < l; j += 2 { + nj := n.Content[j] + if ni.Kind == nj.Kind && ni.Value == nj.Value { + d.terrors = append(d.terrors, fmt.Sprintf("line %d: mapping key %#v already defined at line %d", nj.Line, nj.Value, ni.Line)) + } + } + } + if len(d.terrors) > nerrs { + return false + } + } + switch out.Kind() { + case reflect.Struct: + return d.mappingStruct(n, out) + case reflect.Map: + // okay + case reflect.Interface: + iface := out + if isStringMap(n) { + out = reflect.MakeMap(d.stringMapType) + } else { + out = reflect.MakeMap(d.generalMapType) + } + iface.Set(out) + default: + d.terror(n, mapTag, out) + return false + } + + outt := out.Type() + kt := outt.Key() + et := outt.Elem() + + stringMapType := d.stringMapType + generalMapType := d.generalMapType + if outt.Elem() == ifaceType { + if outt.Key().Kind() == reflect.String { + d.stringMapType = outt + } else if outt.Key() == ifaceType { + d.generalMapType = outt + } + } + + mapIsNew := false + if out.IsNil() { + out.Set(reflect.MakeMap(outt)) + mapIsNew = true + } + for i := 0; i < l; i += 2 { + if isMerge(n.Content[i]) { + d.merge(n.Content[i+1], out) + continue + } + k := reflect.New(kt).Elem() + if d.unmarshal(n.Content[i], k) { + kkind := k.Kind() + if kkind == reflect.Interface { + kkind = k.Elem().Kind() + } + if kkind == reflect.Map || kkind == reflect.Slice { + failf("invalid map key: %#v", k.Interface()) + } + e := reflect.New(et).Elem() + if d.unmarshal(n.Content[i+1], e) || n.Content[i+1].ShortTag() == nullTag && (mapIsNew || !out.MapIndex(k).IsValid()) { + out.SetMapIndex(k, e) + } + } + } + d.stringMapType = stringMapType + d.generalMapType = generalMapType + return true +} + +func isStringMap(n *Node) bool { + if n.Kind != MappingNode { + return false + } + l := len(n.Content) + for i := 0; i < l; i += 2 { + if n.Content[i].ShortTag() != strTag { + return false + } + } + return true +} + +func (d *decoder) mappingStruct(n *Node, out reflect.Value) (good bool) { + sinfo, err := getStructInfo(out.Type()) + if err != nil { + panic(err) + } + + var inlineMap reflect.Value + var elemType reflect.Type + if sinfo.InlineMap != -1 { + inlineMap = out.Field(sinfo.InlineMap) + inlineMap.Set(reflect.New(inlineMap.Type()).Elem()) + elemType = inlineMap.Type().Elem() + } + + for _, index := range sinfo.InlineUnmarshalers { + field := d.fieldByIndex(n, out, index) + d.prepare(n, field) + } + + var doneFields []bool + if d.uniqueKeys { + doneFields = make([]bool, len(sinfo.FieldsList)) + } + name := settableValueOf("") + l := len(n.Content) + for i := 0; i < l; i += 2 { + ni := n.Content[i] + if isMerge(ni) { + d.merge(n.Content[i+1], out) + continue + } + if !d.unmarshal(ni, name) { + continue + } + if info, ok := sinfo.FieldsMap[name.String()]; ok { + if d.uniqueKeys { + if doneFields[info.Id] { + d.terrors = append(d.terrors, fmt.Sprintf("line %d: field %s already set in type %s", ni.Line, name.String(), out.Type())) + continue + } + doneFields[info.Id] = true + } + var field reflect.Value + if info.Inline == nil { + field = out.Field(info.Num) + } else { + field = d.fieldByIndex(n, out, info.Inline) + } + d.unmarshal(n.Content[i+1], field) + } else if sinfo.InlineMap != -1 { + if inlineMap.IsNil() { + inlineMap.Set(reflect.MakeMap(inlineMap.Type())) + } + value := reflect.New(elemType).Elem() + d.unmarshal(n.Content[i+1], value) + inlineMap.SetMapIndex(name, value) + } else if d.knownFields { + d.terrors = append(d.terrors, fmt.Sprintf("line %d: field %s not found in type %s", ni.Line, name.String(), out.Type())) + } + } + return true +} + +func failWantMap() { + failf("map merge requires map or sequence of maps as the value") +} + +func (d *decoder) merge(n *Node, out reflect.Value) { + switch n.Kind { + case MappingNode: + d.unmarshal(n, out) + case AliasNode: + if n.Alias != nil && n.Alias.Kind != MappingNode { + failWantMap() + } + d.unmarshal(n, out) + case SequenceNode: + // Step backwards as earlier nodes take precedence. + for i := len(n.Content) - 1; i >= 0; i-- { + ni := n.Content[i] + if ni.Kind == AliasNode { + if ni.Alias != nil && ni.Alias.Kind != MappingNode { + failWantMap() + } + } else if ni.Kind != MappingNode { + failWantMap() + } + d.unmarshal(ni, out) + } + default: + failWantMap() + } +} + +func isMerge(n *Node) bool { + return n.Kind == ScalarNode && n.Value == "<<" && (n.Tag == "" || n.Tag == "!" || shortTag(n.Tag) == mergeTag) +} diff --git a/vendor/gopkg.in/yaml.v3/emitterc.go b/vendor/gopkg.in/yaml.v3/emitterc.go new file mode 100644 index 00000000..0f47c9ca --- /dev/null +++ b/vendor/gopkg.in/yaml.v3/emitterc.go @@ -0,0 +1,2020 @@ +// +// Copyright (c) 2011-2019 Canonical Ltd +// Copyright (c) 2006-2010 Kirill Simonov +// +// Permission is hereby granted, free of charge, to any person obtaining a copy of +// this software and associated documentation files (the "Software"), to deal in +// the Software without restriction, including without limitation the rights to +// use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies +// of the Software, and to permit persons to whom the Software is furnished to do +// so, subject to the following conditions: +// +// The above copyright notice and this permission notice shall be included in all +// copies or substantial portions of the Software. +// +// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE +// SOFTWARE. + +package yaml + +import ( + "bytes" + "fmt" +) + +// Flush the buffer if needed. +func flush(emitter *yaml_emitter_t) bool { + if emitter.buffer_pos+5 >= len(emitter.buffer) { + return yaml_emitter_flush(emitter) + } + return true +} + +// Put a character to the output buffer. +func put(emitter *yaml_emitter_t, value byte) bool { + if emitter.buffer_pos+5 >= len(emitter.buffer) && !yaml_emitter_flush(emitter) { + return false + } + emitter.buffer[emitter.buffer_pos] = value + emitter.buffer_pos++ + emitter.column++ + return true +} + +// Put a line break to the output buffer. +func put_break(emitter *yaml_emitter_t) bool { + if emitter.buffer_pos+5 >= len(emitter.buffer) && !yaml_emitter_flush(emitter) { + return false + } + switch emitter.line_break { + case yaml_CR_BREAK: + emitter.buffer[emitter.buffer_pos] = '\r' + emitter.buffer_pos += 1 + case yaml_LN_BREAK: + emitter.buffer[emitter.buffer_pos] = '\n' + emitter.buffer_pos += 1 + case yaml_CRLN_BREAK: + emitter.buffer[emitter.buffer_pos+0] = '\r' + emitter.buffer[emitter.buffer_pos+1] = '\n' + emitter.buffer_pos += 2 + default: + panic("unknown line break setting") + } + if emitter.column == 0 { + emitter.space_above = true + } + emitter.column = 0 + emitter.line++ + // [Go] Do this here and below and drop from everywhere else (see commented lines). + emitter.indention = true + return true +} + +// Copy a character from a string into buffer. +func write(emitter *yaml_emitter_t, s []byte, i *int) bool { + if emitter.buffer_pos+5 >= len(emitter.buffer) && !yaml_emitter_flush(emitter) { + return false + } + p := emitter.buffer_pos + w := width(s[*i]) + switch w { + case 4: + emitter.buffer[p+3] = s[*i+3] + fallthrough + case 3: + emitter.buffer[p+2] = s[*i+2] + fallthrough + case 2: + emitter.buffer[p+1] = s[*i+1] + fallthrough + case 1: + emitter.buffer[p+0] = s[*i+0] + default: + panic("unknown character width") + } + emitter.column++ + emitter.buffer_pos += w + *i += w + return true +} + +// Write a whole string into buffer. +func write_all(emitter *yaml_emitter_t, s []byte) bool { + for i := 0; i < len(s); { + if !write(emitter, s, &i) { + return false + } + } + return true +} + +// Copy a line break character from a string into buffer. +func write_break(emitter *yaml_emitter_t, s []byte, i *int) bool { + if s[*i] == '\n' { + if !put_break(emitter) { + return false + } + *i++ + } else { + if !write(emitter, s, i) { + return false + } + if emitter.column == 0 { + emitter.space_above = true + } + emitter.column = 0 + emitter.line++ + // [Go] Do this here and above and drop from everywhere else (see commented lines). + emitter.indention = true + } + return true +} + +// Set an emitter error and return false. +func yaml_emitter_set_emitter_error(emitter *yaml_emitter_t, problem string) bool { + emitter.error = yaml_EMITTER_ERROR + emitter.problem = problem + return false +} + +// Emit an event. +func yaml_emitter_emit(emitter *yaml_emitter_t, event *yaml_event_t) bool { + emitter.events = append(emitter.events, *event) + for !yaml_emitter_need_more_events(emitter) { + event := &emitter.events[emitter.events_head] + if !yaml_emitter_analyze_event(emitter, event) { + return false + } + if !yaml_emitter_state_machine(emitter, event) { + return false + } + yaml_event_delete(event) + emitter.events_head++ + } + return true +} + +// Check if we need to accumulate more events before emitting. +// +// We accumulate extra +// - 1 event for DOCUMENT-START +// - 2 events for SEQUENCE-START +// - 3 events for MAPPING-START +// +func yaml_emitter_need_more_events(emitter *yaml_emitter_t) bool { + if emitter.events_head == len(emitter.events) { + return true + } + var accumulate int + switch emitter.events[emitter.events_head].typ { + case yaml_DOCUMENT_START_EVENT: + accumulate = 1 + break + case yaml_SEQUENCE_START_EVENT: + accumulate = 2 + break + case yaml_MAPPING_START_EVENT: + accumulate = 3 + break + default: + return false + } + if len(emitter.events)-emitter.events_head > accumulate { + return false + } + var level int + for i := emitter.events_head; i < len(emitter.events); i++ { + switch emitter.events[i].typ { + case yaml_STREAM_START_EVENT, yaml_DOCUMENT_START_EVENT, yaml_SEQUENCE_START_EVENT, yaml_MAPPING_START_EVENT: + level++ + case yaml_STREAM_END_EVENT, yaml_DOCUMENT_END_EVENT, yaml_SEQUENCE_END_EVENT, yaml_MAPPING_END_EVENT: + level-- + } + if level == 0 { + return false + } + } + return true +} + +// Append a directive to the directives stack. +func yaml_emitter_append_tag_directive(emitter *yaml_emitter_t, value *yaml_tag_directive_t, allow_duplicates bool) bool { + for i := 0; i < len(emitter.tag_directives); i++ { + if bytes.Equal(value.handle, emitter.tag_directives[i].handle) { + if allow_duplicates { + return true + } + return yaml_emitter_set_emitter_error(emitter, "duplicate %TAG directive") + } + } + + // [Go] Do we actually need to copy this given garbage collection + // and the lack of deallocating destructors? + tag_copy := yaml_tag_directive_t{ + handle: make([]byte, len(value.handle)), + prefix: make([]byte, len(value.prefix)), + } + copy(tag_copy.handle, value.handle) + copy(tag_copy.prefix, value.prefix) + emitter.tag_directives = append(emitter.tag_directives, tag_copy) + return true +} + +// Increase the indentation level. +func yaml_emitter_increase_indent(emitter *yaml_emitter_t, flow, indentless bool) bool { + emitter.indents = append(emitter.indents, emitter.indent) + if emitter.indent < 0 { + if flow { + emitter.indent = emitter.best_indent + } else { + emitter.indent = 0 + } + } else if !indentless { + // [Go] This was changed so that indentations are more regular. + if emitter.states[len(emitter.states)-1] == yaml_EMIT_BLOCK_SEQUENCE_ITEM_STATE { + // The first indent inside a sequence will just skip the "- " indicator. + emitter.indent += 2 + } else { + // Everything else aligns to the chosen indentation. + emitter.indent = emitter.best_indent*((emitter.indent+emitter.best_indent)/emitter.best_indent) + } + } + return true +} + +// State dispatcher. +func yaml_emitter_state_machine(emitter *yaml_emitter_t, event *yaml_event_t) bool { + switch emitter.state { + default: + case yaml_EMIT_STREAM_START_STATE: + return yaml_emitter_emit_stream_start(emitter, event) + + case yaml_EMIT_FIRST_DOCUMENT_START_STATE: + return yaml_emitter_emit_document_start(emitter, event, true) + + case yaml_EMIT_DOCUMENT_START_STATE: + return yaml_emitter_emit_document_start(emitter, event, false) + + case yaml_EMIT_DOCUMENT_CONTENT_STATE: + return yaml_emitter_emit_document_content(emitter, event) + + case yaml_EMIT_DOCUMENT_END_STATE: + return yaml_emitter_emit_document_end(emitter, event) + + case yaml_EMIT_FLOW_SEQUENCE_FIRST_ITEM_STATE: + return yaml_emitter_emit_flow_sequence_item(emitter, event, true, false) + + case yaml_EMIT_FLOW_SEQUENCE_TRAIL_ITEM_STATE: + return yaml_emitter_emit_flow_sequence_item(emitter, event, false, true) + + case yaml_EMIT_FLOW_SEQUENCE_ITEM_STATE: + return yaml_emitter_emit_flow_sequence_item(emitter, event, false, false) + + case yaml_EMIT_FLOW_MAPPING_FIRST_KEY_STATE: + return yaml_emitter_emit_flow_mapping_key(emitter, event, true, false) + + case yaml_EMIT_FLOW_MAPPING_TRAIL_KEY_STATE: + return yaml_emitter_emit_flow_mapping_key(emitter, event, false, true) + + case yaml_EMIT_FLOW_MAPPING_KEY_STATE: + return yaml_emitter_emit_flow_mapping_key(emitter, event, false, false) + + case yaml_EMIT_FLOW_MAPPING_SIMPLE_VALUE_STATE: + return yaml_emitter_emit_flow_mapping_value(emitter, event, true) + + case yaml_EMIT_FLOW_MAPPING_VALUE_STATE: + return yaml_emitter_emit_flow_mapping_value(emitter, event, false) + + case yaml_EMIT_BLOCK_SEQUENCE_FIRST_ITEM_STATE: + return yaml_emitter_emit_block_sequence_item(emitter, event, true) + + case yaml_EMIT_BLOCK_SEQUENCE_ITEM_STATE: + return yaml_emitter_emit_block_sequence_item(emitter, event, false) + + case yaml_EMIT_BLOCK_MAPPING_FIRST_KEY_STATE: + return yaml_emitter_emit_block_mapping_key(emitter, event, true) + + case yaml_EMIT_BLOCK_MAPPING_KEY_STATE: + return yaml_emitter_emit_block_mapping_key(emitter, event, false) + + case yaml_EMIT_BLOCK_MAPPING_SIMPLE_VALUE_STATE: + return yaml_emitter_emit_block_mapping_value(emitter, event, true) + + case yaml_EMIT_BLOCK_MAPPING_VALUE_STATE: + return yaml_emitter_emit_block_mapping_value(emitter, event, false) + + case yaml_EMIT_END_STATE: + return yaml_emitter_set_emitter_error(emitter, "expected nothing after STREAM-END") + } + panic("invalid emitter state") +} + +// Expect STREAM-START. +func yaml_emitter_emit_stream_start(emitter *yaml_emitter_t, event *yaml_event_t) bool { + if event.typ != yaml_STREAM_START_EVENT { + return yaml_emitter_set_emitter_error(emitter, "expected STREAM-START") + } + if emitter.encoding == yaml_ANY_ENCODING { + emitter.encoding = event.encoding + if emitter.encoding == yaml_ANY_ENCODING { + emitter.encoding = yaml_UTF8_ENCODING + } + } + if emitter.best_indent < 2 || emitter.best_indent > 9 { + emitter.best_indent = 2 + } + if emitter.best_width >= 0 && emitter.best_width <= emitter.best_indent*2 { + emitter.best_width = 80 + } + if emitter.best_width < 0 { + emitter.best_width = 1<<31 - 1 + } + if emitter.line_break == yaml_ANY_BREAK { + emitter.line_break = yaml_LN_BREAK + } + + emitter.indent = -1 + emitter.line = 0 + emitter.column = 0 + emitter.whitespace = true + emitter.indention = true + emitter.space_above = true + emitter.foot_indent = -1 + + if emitter.encoding != yaml_UTF8_ENCODING { + if !yaml_emitter_write_bom(emitter) { + return false + } + } + emitter.state = yaml_EMIT_FIRST_DOCUMENT_START_STATE + return true +} + +// Expect DOCUMENT-START or STREAM-END. +func yaml_emitter_emit_document_start(emitter *yaml_emitter_t, event *yaml_event_t, first bool) bool { + + if event.typ == yaml_DOCUMENT_START_EVENT { + + if event.version_directive != nil { + if !yaml_emitter_analyze_version_directive(emitter, event.version_directive) { + return false + } + } + + for i := 0; i < len(event.tag_directives); i++ { + tag_directive := &event.tag_directives[i] + if !yaml_emitter_analyze_tag_directive(emitter, tag_directive) { + return false + } + if !yaml_emitter_append_tag_directive(emitter, tag_directive, false) { + return false + } + } + + for i := 0; i < len(default_tag_directives); i++ { + tag_directive := &default_tag_directives[i] + if !yaml_emitter_append_tag_directive(emitter, tag_directive, true) { + return false + } + } + + implicit := event.implicit + if !first || emitter.canonical { + implicit = false + } + + if emitter.open_ended && (event.version_directive != nil || len(event.tag_directives) > 0) { + if !yaml_emitter_write_indicator(emitter, []byte("..."), true, false, false) { + return false + } + if !yaml_emitter_write_indent(emitter) { + return false + } + } + + if event.version_directive != nil { + implicit = false + if !yaml_emitter_write_indicator(emitter, []byte("%YAML"), true, false, false) { + return false + } + if !yaml_emitter_write_indicator(emitter, []byte("1.1"), true, false, false) { + return false + } + if !yaml_emitter_write_indent(emitter) { + return false + } + } + + if len(event.tag_directives) > 0 { + implicit = false + for i := 0; i < len(event.tag_directives); i++ { + tag_directive := &event.tag_directives[i] + if !yaml_emitter_write_indicator(emitter, []byte("%TAG"), true, false, false) { + return false + } + if !yaml_emitter_write_tag_handle(emitter, tag_directive.handle) { + return false + } + if !yaml_emitter_write_tag_content(emitter, tag_directive.prefix, true) { + return false + } + if !yaml_emitter_write_indent(emitter) { + return false + } + } + } + + if yaml_emitter_check_empty_document(emitter) { + implicit = false + } + if !implicit { + if !yaml_emitter_write_indent(emitter) { + return false + } + if !yaml_emitter_write_indicator(emitter, []byte("---"), true, false, false) { + return false + } + if emitter.canonical || true { + if !yaml_emitter_write_indent(emitter) { + return false + } + } + } + + if len(emitter.head_comment) > 0 { + if !yaml_emitter_process_head_comment(emitter) { + return false + } + if !put_break(emitter) { + return false + } + } + + emitter.state = yaml_EMIT_DOCUMENT_CONTENT_STATE + return true + } + + if event.typ == yaml_STREAM_END_EVENT { + if emitter.open_ended { + if !yaml_emitter_write_indicator(emitter, []byte("..."), true, false, false) { + return false + } + if !yaml_emitter_write_indent(emitter) { + return false + } + } + if !yaml_emitter_flush(emitter) { + return false + } + emitter.state = yaml_EMIT_END_STATE + return true + } + + return yaml_emitter_set_emitter_error(emitter, "expected DOCUMENT-START or STREAM-END") +} + +// Expect the root node. +func yaml_emitter_emit_document_content(emitter *yaml_emitter_t, event *yaml_event_t) bool { + emitter.states = append(emitter.states, yaml_EMIT_DOCUMENT_END_STATE) + + if !yaml_emitter_process_head_comment(emitter) { + return false + } + if !yaml_emitter_emit_node(emitter, event, true, false, false, false) { + return false + } + if !yaml_emitter_process_line_comment(emitter) { + return false + } + if !yaml_emitter_process_foot_comment(emitter) { + return false + } + return true +} + +// Expect DOCUMENT-END. +func yaml_emitter_emit_document_end(emitter *yaml_emitter_t, event *yaml_event_t) bool { + if event.typ != yaml_DOCUMENT_END_EVENT { + return yaml_emitter_set_emitter_error(emitter, "expected DOCUMENT-END") + } + // [Go] Force document foot separation. + emitter.foot_indent = 0 + if !yaml_emitter_process_foot_comment(emitter) { + return false + } + emitter.foot_indent = -1 + if !yaml_emitter_write_indent(emitter) { + return false + } + if !event.implicit { + // [Go] Allocate the slice elsewhere. + if !yaml_emitter_write_indicator(emitter, []byte("..."), true, false, false) { + return false + } + if !yaml_emitter_write_indent(emitter) { + return false + } + } + if !yaml_emitter_flush(emitter) { + return false + } + emitter.state = yaml_EMIT_DOCUMENT_START_STATE + emitter.tag_directives = emitter.tag_directives[:0] + return true +} + +// Expect a flow item node. +func yaml_emitter_emit_flow_sequence_item(emitter *yaml_emitter_t, event *yaml_event_t, first, trail bool) bool { + if first { + if !yaml_emitter_write_indicator(emitter, []byte{'['}, true, true, false) { + return false + } + if !yaml_emitter_increase_indent(emitter, true, false) { + return false + } + emitter.flow_level++ + } + + if event.typ == yaml_SEQUENCE_END_EVENT { + if emitter.canonical && !first && !trail { + if !yaml_emitter_write_indicator(emitter, []byte{','}, false, false, false) { + return false + } + } + emitter.flow_level-- + emitter.indent = emitter.indents[len(emitter.indents)-1] + emitter.indents = emitter.indents[:len(emitter.indents)-1] + if emitter.column == 0 || emitter.canonical && !first { + if !yaml_emitter_write_indent(emitter) { + return false + } + } + if !yaml_emitter_write_indicator(emitter, []byte{']'}, false, false, false) { + return false + } + if !yaml_emitter_process_line_comment(emitter) { + return false + } + if !yaml_emitter_process_foot_comment(emitter) { + return false + } + emitter.state = emitter.states[len(emitter.states)-1] + emitter.states = emitter.states[:len(emitter.states)-1] + + return true + } + + if !first && !trail { + if !yaml_emitter_write_indicator(emitter, []byte{','}, false, false, false) { + return false + } + } + + if !yaml_emitter_process_head_comment(emitter) { + return false + } + if emitter.column == 0 { + if !yaml_emitter_write_indent(emitter) { + return false + } + } + + if emitter.canonical || emitter.column > emitter.best_width { + if !yaml_emitter_write_indent(emitter) { + return false + } + } + if len(emitter.line_comment)+len(emitter.foot_comment)+len(emitter.tail_comment) > 0 { + emitter.states = append(emitter.states, yaml_EMIT_FLOW_SEQUENCE_TRAIL_ITEM_STATE) + } else { + emitter.states = append(emitter.states, yaml_EMIT_FLOW_SEQUENCE_ITEM_STATE) + } + if !yaml_emitter_emit_node(emitter, event, false, true, false, false) { + return false + } + if len(emitter.line_comment)+len(emitter.foot_comment)+len(emitter.tail_comment) > 0 { + if !yaml_emitter_write_indicator(emitter, []byte{','}, false, false, false) { + return false + } + } + if !yaml_emitter_process_line_comment(emitter) { + return false + } + if !yaml_emitter_process_foot_comment(emitter) { + return false + } + return true +} + +// Expect a flow key node. +func yaml_emitter_emit_flow_mapping_key(emitter *yaml_emitter_t, event *yaml_event_t, first, trail bool) bool { + if first { + if !yaml_emitter_write_indicator(emitter, []byte{'{'}, true, true, false) { + return false + } + if !yaml_emitter_increase_indent(emitter, true, false) { + return false + } + emitter.flow_level++ + } + + if event.typ == yaml_MAPPING_END_EVENT { + if (emitter.canonical || len(emitter.head_comment)+len(emitter.foot_comment)+len(emitter.tail_comment) > 0) && !first && !trail { + if !yaml_emitter_write_indicator(emitter, []byte{','}, false, false, false) { + return false + } + } + if !yaml_emitter_process_head_comment(emitter) { + return false + } + emitter.flow_level-- + emitter.indent = emitter.indents[len(emitter.indents)-1] + emitter.indents = emitter.indents[:len(emitter.indents)-1] + if emitter.canonical && !first { + if !yaml_emitter_write_indent(emitter) { + return false + } + } + if !yaml_emitter_write_indicator(emitter, []byte{'}'}, false, false, false) { + return false + } + if !yaml_emitter_process_line_comment(emitter) { + return false + } + if !yaml_emitter_process_foot_comment(emitter) { + return false + } + emitter.state = emitter.states[len(emitter.states)-1] + emitter.states = emitter.states[:len(emitter.states)-1] + return true + } + + if !first && !trail { + if !yaml_emitter_write_indicator(emitter, []byte{','}, false, false, false) { + return false + } + } + + if !yaml_emitter_process_head_comment(emitter) { + return false + } + + if emitter.column == 0 { + if !yaml_emitter_write_indent(emitter) { + return false + } + } + + if emitter.canonical || emitter.column > emitter.best_width { + if !yaml_emitter_write_indent(emitter) { + return false + } + } + + if !emitter.canonical && yaml_emitter_check_simple_key(emitter) { + emitter.states = append(emitter.states, yaml_EMIT_FLOW_MAPPING_SIMPLE_VALUE_STATE) + return yaml_emitter_emit_node(emitter, event, false, false, true, true) + } + if !yaml_emitter_write_indicator(emitter, []byte{'?'}, true, false, false) { + return false + } + emitter.states = append(emitter.states, yaml_EMIT_FLOW_MAPPING_VALUE_STATE) + return yaml_emitter_emit_node(emitter, event, false, false, true, false) +} + +// Expect a flow value node. +func yaml_emitter_emit_flow_mapping_value(emitter *yaml_emitter_t, event *yaml_event_t, simple bool) bool { + if simple { + if !yaml_emitter_write_indicator(emitter, []byte{':'}, false, false, false) { + return false + } + } else { + if emitter.canonical || emitter.column > emitter.best_width { + if !yaml_emitter_write_indent(emitter) { + return false + } + } + if !yaml_emitter_write_indicator(emitter, []byte{':'}, true, false, false) { + return false + } + } + if len(emitter.line_comment)+len(emitter.foot_comment)+len(emitter.tail_comment) > 0 { + emitter.states = append(emitter.states, yaml_EMIT_FLOW_MAPPING_TRAIL_KEY_STATE) + } else { + emitter.states = append(emitter.states, yaml_EMIT_FLOW_MAPPING_KEY_STATE) + } + if !yaml_emitter_emit_node(emitter, event, false, false, true, false) { + return false + } + if len(emitter.line_comment)+len(emitter.foot_comment)+len(emitter.tail_comment) > 0 { + if !yaml_emitter_write_indicator(emitter, []byte{','}, false, false, false) { + return false + } + } + if !yaml_emitter_process_line_comment(emitter) { + return false + } + if !yaml_emitter_process_foot_comment(emitter) { + return false + } + return true +} + +// Expect a block item node. +func yaml_emitter_emit_block_sequence_item(emitter *yaml_emitter_t, event *yaml_event_t, first bool) bool { + if first { + if !yaml_emitter_increase_indent(emitter, false, false) { + return false + } + } + if event.typ == yaml_SEQUENCE_END_EVENT { + emitter.indent = emitter.indents[len(emitter.indents)-1] + emitter.indents = emitter.indents[:len(emitter.indents)-1] + emitter.state = emitter.states[len(emitter.states)-1] + emitter.states = emitter.states[:len(emitter.states)-1] + return true + } + if !yaml_emitter_process_head_comment(emitter) { + return false + } + if !yaml_emitter_write_indent(emitter) { + return false + } + if !yaml_emitter_write_indicator(emitter, []byte{'-'}, true, false, true) { + return false + } + emitter.states = append(emitter.states, yaml_EMIT_BLOCK_SEQUENCE_ITEM_STATE) + if !yaml_emitter_emit_node(emitter, event, false, true, false, false) { + return false + } + if !yaml_emitter_process_line_comment(emitter) { + return false + } + if !yaml_emitter_process_foot_comment(emitter) { + return false + } + return true +} + +// Expect a block key node. +func yaml_emitter_emit_block_mapping_key(emitter *yaml_emitter_t, event *yaml_event_t, first bool) bool { + if first { + if !yaml_emitter_increase_indent(emitter, false, false) { + return false + } + } + if !yaml_emitter_process_head_comment(emitter) { + return false + } + if event.typ == yaml_MAPPING_END_EVENT { + emitter.indent = emitter.indents[len(emitter.indents)-1] + emitter.indents = emitter.indents[:len(emitter.indents)-1] + emitter.state = emitter.states[len(emitter.states)-1] + emitter.states = emitter.states[:len(emitter.states)-1] + return true + } + if !yaml_emitter_write_indent(emitter) { + return false + } + if len(emitter.line_comment) > 0 { + // [Go] A line comment was provided for the key. That's unusual as the + // scanner associates line comments with the value. Either way, + // save the line comment and render it appropriately later. + emitter.key_line_comment = emitter.line_comment + emitter.line_comment = nil + } + if yaml_emitter_check_simple_key(emitter) { + emitter.states = append(emitter.states, yaml_EMIT_BLOCK_MAPPING_SIMPLE_VALUE_STATE) + return yaml_emitter_emit_node(emitter, event, false, false, true, true) + } + if !yaml_emitter_write_indicator(emitter, []byte{'?'}, true, false, true) { + return false + } + emitter.states = append(emitter.states, yaml_EMIT_BLOCK_MAPPING_VALUE_STATE) + return yaml_emitter_emit_node(emitter, event, false, false, true, false) +} + +// Expect a block value node. +func yaml_emitter_emit_block_mapping_value(emitter *yaml_emitter_t, event *yaml_event_t, simple bool) bool { + if simple { + if !yaml_emitter_write_indicator(emitter, []byte{':'}, false, false, false) { + return false + } + } else { + if !yaml_emitter_write_indent(emitter) { + return false + } + if !yaml_emitter_write_indicator(emitter, []byte{':'}, true, false, true) { + return false + } + } + if len(emitter.key_line_comment) > 0 { + // [Go] Line comments are generally associated with the value, but when there's + // no value on the same line as a mapping key they end up attached to the + // key itself. + if event.typ == yaml_SCALAR_EVENT { + if len(emitter.line_comment) == 0 { + // A scalar is coming and it has no line comments by itself yet, + // so just let it handle the line comment as usual. If it has a + // line comment, we can't have both so the one from the key is lost. + emitter.line_comment = emitter.key_line_comment + emitter.key_line_comment = nil + } + } else if event.sequence_style() != yaml_FLOW_SEQUENCE_STYLE && (event.typ == yaml_MAPPING_START_EVENT || event.typ == yaml_SEQUENCE_START_EVENT) { + // An indented block follows, so write the comment right now. + emitter.line_comment, emitter.key_line_comment = emitter.key_line_comment, emitter.line_comment + if !yaml_emitter_process_line_comment(emitter) { + return false + } + emitter.line_comment, emitter.key_line_comment = emitter.key_line_comment, emitter.line_comment + } + } + emitter.states = append(emitter.states, yaml_EMIT_BLOCK_MAPPING_KEY_STATE) + if !yaml_emitter_emit_node(emitter, event, false, false, true, false) { + return false + } + if !yaml_emitter_process_line_comment(emitter) { + return false + } + if !yaml_emitter_process_foot_comment(emitter) { + return false + } + return true +} + +func yaml_emitter_silent_nil_event(emitter *yaml_emitter_t, event *yaml_event_t) bool { + return event.typ == yaml_SCALAR_EVENT && event.implicit && !emitter.canonical && len(emitter.scalar_data.value) == 0 +} + +// Expect a node. +func yaml_emitter_emit_node(emitter *yaml_emitter_t, event *yaml_event_t, + root bool, sequence bool, mapping bool, simple_key bool) bool { + + emitter.root_context = root + emitter.sequence_context = sequence + emitter.mapping_context = mapping + emitter.simple_key_context = simple_key + + switch event.typ { + case yaml_ALIAS_EVENT: + return yaml_emitter_emit_alias(emitter, event) + case yaml_SCALAR_EVENT: + return yaml_emitter_emit_scalar(emitter, event) + case yaml_SEQUENCE_START_EVENT: + return yaml_emitter_emit_sequence_start(emitter, event) + case yaml_MAPPING_START_EVENT: + return yaml_emitter_emit_mapping_start(emitter, event) + default: + return yaml_emitter_set_emitter_error(emitter, + fmt.Sprintf("expected SCALAR, SEQUENCE-START, MAPPING-START, or ALIAS, but got %v", event.typ)) + } +} + +// Expect ALIAS. +func yaml_emitter_emit_alias(emitter *yaml_emitter_t, event *yaml_event_t) bool { + if !yaml_emitter_process_anchor(emitter) { + return false + } + emitter.state = emitter.states[len(emitter.states)-1] + emitter.states = emitter.states[:len(emitter.states)-1] + return true +} + +// Expect SCALAR. +func yaml_emitter_emit_scalar(emitter *yaml_emitter_t, event *yaml_event_t) bool { + if !yaml_emitter_select_scalar_style(emitter, event) { + return false + } + if !yaml_emitter_process_anchor(emitter) { + return false + } + if !yaml_emitter_process_tag(emitter) { + return false + } + if !yaml_emitter_increase_indent(emitter, true, false) { + return false + } + if !yaml_emitter_process_scalar(emitter) { + return false + } + emitter.indent = emitter.indents[len(emitter.indents)-1] + emitter.indents = emitter.indents[:len(emitter.indents)-1] + emitter.state = emitter.states[len(emitter.states)-1] + emitter.states = emitter.states[:len(emitter.states)-1] + return true +} + +// Expect SEQUENCE-START. +func yaml_emitter_emit_sequence_start(emitter *yaml_emitter_t, event *yaml_event_t) bool { + if !yaml_emitter_process_anchor(emitter) { + return false + } + if !yaml_emitter_process_tag(emitter) { + return false + } + if emitter.flow_level > 0 || emitter.canonical || event.sequence_style() == yaml_FLOW_SEQUENCE_STYLE || + yaml_emitter_check_empty_sequence(emitter) { + emitter.state = yaml_EMIT_FLOW_SEQUENCE_FIRST_ITEM_STATE + } else { + emitter.state = yaml_EMIT_BLOCK_SEQUENCE_FIRST_ITEM_STATE + } + return true +} + +// Expect MAPPING-START. +func yaml_emitter_emit_mapping_start(emitter *yaml_emitter_t, event *yaml_event_t) bool { + if !yaml_emitter_process_anchor(emitter) { + return false + } + if !yaml_emitter_process_tag(emitter) { + return false + } + if emitter.flow_level > 0 || emitter.canonical || event.mapping_style() == yaml_FLOW_MAPPING_STYLE || + yaml_emitter_check_empty_mapping(emitter) { + emitter.state = yaml_EMIT_FLOW_MAPPING_FIRST_KEY_STATE + } else { + emitter.state = yaml_EMIT_BLOCK_MAPPING_FIRST_KEY_STATE + } + return true +} + +// Check if the document content is an empty scalar. +func yaml_emitter_check_empty_document(emitter *yaml_emitter_t) bool { + return false // [Go] Huh? +} + +// Check if the next events represent an empty sequence. +func yaml_emitter_check_empty_sequence(emitter *yaml_emitter_t) bool { + if len(emitter.events)-emitter.events_head < 2 { + return false + } + return emitter.events[emitter.events_head].typ == yaml_SEQUENCE_START_EVENT && + emitter.events[emitter.events_head+1].typ == yaml_SEQUENCE_END_EVENT +} + +// Check if the next events represent an empty mapping. +func yaml_emitter_check_empty_mapping(emitter *yaml_emitter_t) bool { + if len(emitter.events)-emitter.events_head < 2 { + return false + } + return emitter.events[emitter.events_head].typ == yaml_MAPPING_START_EVENT && + emitter.events[emitter.events_head+1].typ == yaml_MAPPING_END_EVENT +} + +// Check if the next node can be expressed as a simple key. +func yaml_emitter_check_simple_key(emitter *yaml_emitter_t) bool { + length := 0 + switch emitter.events[emitter.events_head].typ { + case yaml_ALIAS_EVENT: + length += len(emitter.anchor_data.anchor) + case yaml_SCALAR_EVENT: + if emitter.scalar_data.multiline { + return false + } + length += len(emitter.anchor_data.anchor) + + len(emitter.tag_data.handle) + + len(emitter.tag_data.suffix) + + len(emitter.scalar_data.value) + case yaml_SEQUENCE_START_EVENT: + if !yaml_emitter_check_empty_sequence(emitter) { + return false + } + length += len(emitter.anchor_data.anchor) + + len(emitter.tag_data.handle) + + len(emitter.tag_data.suffix) + case yaml_MAPPING_START_EVENT: + if !yaml_emitter_check_empty_mapping(emitter) { + return false + } + length += len(emitter.anchor_data.anchor) + + len(emitter.tag_data.handle) + + len(emitter.tag_data.suffix) + default: + return false + } + return length <= 128 +} + +// Determine an acceptable scalar style. +func yaml_emitter_select_scalar_style(emitter *yaml_emitter_t, event *yaml_event_t) bool { + + no_tag := len(emitter.tag_data.handle) == 0 && len(emitter.tag_data.suffix) == 0 + if no_tag && !event.implicit && !event.quoted_implicit { + return yaml_emitter_set_emitter_error(emitter, "neither tag nor implicit flags are specified") + } + + style := event.scalar_style() + if style == yaml_ANY_SCALAR_STYLE { + style = yaml_PLAIN_SCALAR_STYLE + } + if emitter.canonical { + style = yaml_DOUBLE_QUOTED_SCALAR_STYLE + } + if emitter.simple_key_context && emitter.scalar_data.multiline { + style = yaml_DOUBLE_QUOTED_SCALAR_STYLE + } + + if style == yaml_PLAIN_SCALAR_STYLE { + if emitter.flow_level > 0 && !emitter.scalar_data.flow_plain_allowed || + emitter.flow_level == 0 && !emitter.scalar_data.block_plain_allowed { + style = yaml_SINGLE_QUOTED_SCALAR_STYLE + } + if len(emitter.scalar_data.value) == 0 && (emitter.flow_level > 0 || emitter.simple_key_context) { + style = yaml_SINGLE_QUOTED_SCALAR_STYLE + } + if no_tag && !event.implicit { + style = yaml_SINGLE_QUOTED_SCALAR_STYLE + } + } + if style == yaml_SINGLE_QUOTED_SCALAR_STYLE { + if !emitter.scalar_data.single_quoted_allowed { + style = yaml_DOUBLE_QUOTED_SCALAR_STYLE + } + } + if style == yaml_LITERAL_SCALAR_STYLE || style == yaml_FOLDED_SCALAR_STYLE { + if !emitter.scalar_data.block_allowed || emitter.flow_level > 0 || emitter.simple_key_context { + style = yaml_DOUBLE_QUOTED_SCALAR_STYLE + } + } + + if no_tag && !event.quoted_implicit && style != yaml_PLAIN_SCALAR_STYLE { + emitter.tag_data.handle = []byte{'!'} + } + emitter.scalar_data.style = style + return true +} + +// Write an anchor. +func yaml_emitter_process_anchor(emitter *yaml_emitter_t) bool { + if emitter.anchor_data.anchor == nil { + return true + } + c := []byte{'&'} + if emitter.anchor_data.alias { + c[0] = '*' + } + if !yaml_emitter_write_indicator(emitter, c, true, false, false) { + return false + } + return yaml_emitter_write_anchor(emitter, emitter.anchor_data.anchor) +} + +// Write a tag. +func yaml_emitter_process_tag(emitter *yaml_emitter_t) bool { + if len(emitter.tag_data.handle) == 0 && len(emitter.tag_data.suffix) == 0 { + return true + } + if len(emitter.tag_data.handle) > 0 { + if !yaml_emitter_write_tag_handle(emitter, emitter.tag_data.handle) { + return false + } + if len(emitter.tag_data.suffix) > 0 { + if !yaml_emitter_write_tag_content(emitter, emitter.tag_data.suffix, false) { + return false + } + } + } else { + // [Go] Allocate these slices elsewhere. + if !yaml_emitter_write_indicator(emitter, []byte("!<"), true, false, false) { + return false + } + if !yaml_emitter_write_tag_content(emitter, emitter.tag_data.suffix, false) { + return false + } + if !yaml_emitter_write_indicator(emitter, []byte{'>'}, false, false, false) { + return false + } + } + return true +} + +// Write a scalar. +func yaml_emitter_process_scalar(emitter *yaml_emitter_t) bool { + switch emitter.scalar_data.style { + case yaml_PLAIN_SCALAR_STYLE: + return yaml_emitter_write_plain_scalar(emitter, emitter.scalar_data.value, !emitter.simple_key_context) + + case yaml_SINGLE_QUOTED_SCALAR_STYLE: + return yaml_emitter_write_single_quoted_scalar(emitter, emitter.scalar_data.value, !emitter.simple_key_context) + + case yaml_DOUBLE_QUOTED_SCALAR_STYLE: + return yaml_emitter_write_double_quoted_scalar(emitter, emitter.scalar_data.value, !emitter.simple_key_context) + + case yaml_LITERAL_SCALAR_STYLE: + return yaml_emitter_write_literal_scalar(emitter, emitter.scalar_data.value) + + case yaml_FOLDED_SCALAR_STYLE: + return yaml_emitter_write_folded_scalar(emitter, emitter.scalar_data.value) + } + panic("unknown scalar style") +} + +// Write a head comment. +func yaml_emitter_process_head_comment(emitter *yaml_emitter_t) bool { + if len(emitter.tail_comment) > 0 { + if !yaml_emitter_write_indent(emitter) { + return false + } + if !yaml_emitter_write_comment(emitter, emitter.tail_comment) { + return false + } + emitter.tail_comment = emitter.tail_comment[:0] + emitter.foot_indent = emitter.indent + if emitter.foot_indent < 0 { + emitter.foot_indent = 0 + } + } + + if len(emitter.head_comment) == 0 { + return true + } + if !yaml_emitter_write_indent(emitter) { + return false + } + if !yaml_emitter_write_comment(emitter, emitter.head_comment) { + return false + } + emitter.head_comment = emitter.head_comment[:0] + return true +} + +// Write an line comment. +func yaml_emitter_process_line_comment(emitter *yaml_emitter_t) bool { + if len(emitter.line_comment) == 0 { + return true + } + if !emitter.whitespace { + if !put(emitter, ' ') { + return false + } + } + if !yaml_emitter_write_comment(emitter, emitter.line_comment) { + return false + } + emitter.line_comment = emitter.line_comment[:0] + return true +} + +// Write a foot comment. +func yaml_emitter_process_foot_comment(emitter *yaml_emitter_t) bool { + if len(emitter.foot_comment) == 0 { + return true + } + if !yaml_emitter_write_indent(emitter) { + return false + } + if !yaml_emitter_write_comment(emitter, emitter.foot_comment) { + return false + } + emitter.foot_comment = emitter.foot_comment[:0] + emitter.foot_indent = emitter.indent + if emitter.foot_indent < 0 { + emitter.foot_indent = 0 + } + return true +} + +// Check if a %YAML directive is valid. +func yaml_emitter_analyze_version_directive(emitter *yaml_emitter_t, version_directive *yaml_version_directive_t) bool { + if version_directive.major != 1 || version_directive.minor != 1 { + return yaml_emitter_set_emitter_error(emitter, "incompatible %YAML directive") + } + return true +} + +// Check if a %TAG directive is valid. +func yaml_emitter_analyze_tag_directive(emitter *yaml_emitter_t, tag_directive *yaml_tag_directive_t) bool { + handle := tag_directive.handle + prefix := tag_directive.prefix + if len(handle) == 0 { + return yaml_emitter_set_emitter_error(emitter, "tag handle must not be empty") + } + if handle[0] != '!' { + return yaml_emitter_set_emitter_error(emitter, "tag handle must start with '!'") + } + if handle[len(handle)-1] != '!' { + return yaml_emitter_set_emitter_error(emitter, "tag handle must end with '!'") + } + for i := 1; i < len(handle)-1; i += width(handle[i]) { + if !is_alpha(handle, i) { + return yaml_emitter_set_emitter_error(emitter, "tag handle must contain alphanumerical characters only") + } + } + if len(prefix) == 0 { + return yaml_emitter_set_emitter_error(emitter, "tag prefix must not be empty") + } + return true +} + +// Check if an anchor is valid. +func yaml_emitter_analyze_anchor(emitter *yaml_emitter_t, anchor []byte, alias bool) bool { + if len(anchor) == 0 { + problem := "anchor value must not be empty" + if alias { + problem = "alias value must not be empty" + } + return yaml_emitter_set_emitter_error(emitter, problem) + } + for i := 0; i < len(anchor); i += width(anchor[i]) { + if !is_alpha(anchor, i) { + problem := "anchor value must contain alphanumerical characters only" + if alias { + problem = "alias value must contain alphanumerical characters only" + } + return yaml_emitter_set_emitter_error(emitter, problem) + } + } + emitter.anchor_data.anchor = anchor + emitter.anchor_data.alias = alias + return true +} + +// Check if a tag is valid. +func yaml_emitter_analyze_tag(emitter *yaml_emitter_t, tag []byte) bool { + if len(tag) == 0 { + return yaml_emitter_set_emitter_error(emitter, "tag value must not be empty") + } + for i := 0; i < len(emitter.tag_directives); i++ { + tag_directive := &emitter.tag_directives[i] + if bytes.HasPrefix(tag, tag_directive.prefix) { + emitter.tag_data.handle = tag_directive.handle + emitter.tag_data.suffix = tag[len(tag_directive.prefix):] + return true + } + } + emitter.tag_data.suffix = tag + return true +} + +// Check if a scalar is valid. +func yaml_emitter_analyze_scalar(emitter *yaml_emitter_t, value []byte) bool { + var ( + block_indicators = false + flow_indicators = false + line_breaks = false + special_characters = false + tab_characters = false + + leading_space = false + leading_break = false + trailing_space = false + trailing_break = false + break_space = false + space_break = false + + preceded_by_whitespace = false + followed_by_whitespace = false + previous_space = false + previous_break = false + ) + + emitter.scalar_data.value = value + + if len(value) == 0 { + emitter.scalar_data.multiline = false + emitter.scalar_data.flow_plain_allowed = false + emitter.scalar_data.block_plain_allowed = true + emitter.scalar_data.single_quoted_allowed = true + emitter.scalar_data.block_allowed = false + return true + } + + if len(value) >= 3 && ((value[0] == '-' && value[1] == '-' && value[2] == '-') || (value[0] == '.' && value[1] == '.' && value[2] == '.')) { + block_indicators = true + flow_indicators = true + } + + preceded_by_whitespace = true + for i, w := 0, 0; i < len(value); i += w { + w = width(value[i]) + followed_by_whitespace = i+w >= len(value) || is_blank(value, i+w) + + if i == 0 { + switch value[i] { + case '#', ',', '[', ']', '{', '}', '&', '*', '!', '|', '>', '\'', '"', '%', '@', '`': + flow_indicators = true + block_indicators = true + case '?', ':': + flow_indicators = true + if followed_by_whitespace { + block_indicators = true + } + case '-': + if followed_by_whitespace { + flow_indicators = true + block_indicators = true + } + } + } else { + switch value[i] { + case ',', '?', '[', ']', '{', '}': + flow_indicators = true + case ':': + flow_indicators = true + if followed_by_whitespace { + block_indicators = true + } + case '#': + if preceded_by_whitespace { + flow_indicators = true + block_indicators = true + } + } + } + + if value[i] == '\t' { + tab_characters = true + } else if !is_printable(value, i) || !is_ascii(value, i) && !emitter.unicode { + special_characters = true + } + if is_space(value, i) { + if i == 0 { + leading_space = true + } + if i+width(value[i]) == len(value) { + trailing_space = true + } + if previous_break { + break_space = true + } + previous_space = true + previous_break = false + } else if is_break(value, i) { + line_breaks = true + if i == 0 { + leading_break = true + } + if i+width(value[i]) == len(value) { + trailing_break = true + } + if previous_space { + space_break = true + } + previous_space = false + previous_break = true + } else { + previous_space = false + previous_break = false + } + + // [Go]: Why 'z'? Couldn't be the end of the string as that's the loop condition. + preceded_by_whitespace = is_blankz(value, i) + } + + emitter.scalar_data.multiline = line_breaks + emitter.scalar_data.flow_plain_allowed = true + emitter.scalar_data.block_plain_allowed = true + emitter.scalar_data.single_quoted_allowed = true + emitter.scalar_data.block_allowed = true + + if leading_space || leading_break || trailing_space || trailing_break { + emitter.scalar_data.flow_plain_allowed = false + emitter.scalar_data.block_plain_allowed = false + } + if trailing_space { + emitter.scalar_data.block_allowed = false + } + if break_space { + emitter.scalar_data.flow_plain_allowed = false + emitter.scalar_data.block_plain_allowed = false + emitter.scalar_data.single_quoted_allowed = false + } + if space_break || tab_characters || special_characters { + emitter.scalar_data.flow_plain_allowed = false + emitter.scalar_data.block_plain_allowed = false + emitter.scalar_data.single_quoted_allowed = false + } + if space_break || special_characters { + emitter.scalar_data.block_allowed = false + } + if line_breaks { + emitter.scalar_data.flow_plain_allowed = false + emitter.scalar_data.block_plain_allowed = false + } + if flow_indicators { + emitter.scalar_data.flow_plain_allowed = false + } + if block_indicators { + emitter.scalar_data.block_plain_allowed = false + } + return true +} + +// Check if the event data is valid. +func yaml_emitter_analyze_event(emitter *yaml_emitter_t, event *yaml_event_t) bool { + + emitter.anchor_data.anchor = nil + emitter.tag_data.handle = nil + emitter.tag_data.suffix = nil + emitter.scalar_data.value = nil + + if len(event.head_comment) > 0 { + emitter.head_comment = event.head_comment + } + if len(event.line_comment) > 0 { + emitter.line_comment = event.line_comment + } + if len(event.foot_comment) > 0 { + emitter.foot_comment = event.foot_comment + } + if len(event.tail_comment) > 0 { + emitter.tail_comment = event.tail_comment + } + + switch event.typ { + case yaml_ALIAS_EVENT: + if !yaml_emitter_analyze_anchor(emitter, event.anchor, true) { + return false + } + + case yaml_SCALAR_EVENT: + if len(event.anchor) > 0 { + if !yaml_emitter_analyze_anchor(emitter, event.anchor, false) { + return false + } + } + if len(event.tag) > 0 && (emitter.canonical || (!event.implicit && !event.quoted_implicit)) { + if !yaml_emitter_analyze_tag(emitter, event.tag) { + return false + } + } + if !yaml_emitter_analyze_scalar(emitter, event.value) { + return false + } + + case yaml_SEQUENCE_START_EVENT: + if len(event.anchor) > 0 { + if !yaml_emitter_analyze_anchor(emitter, event.anchor, false) { + return false + } + } + if len(event.tag) > 0 && (emitter.canonical || !event.implicit) { + if !yaml_emitter_analyze_tag(emitter, event.tag) { + return false + } + } + + case yaml_MAPPING_START_EVENT: + if len(event.anchor) > 0 { + if !yaml_emitter_analyze_anchor(emitter, event.anchor, false) { + return false + } + } + if len(event.tag) > 0 && (emitter.canonical || !event.implicit) { + if !yaml_emitter_analyze_tag(emitter, event.tag) { + return false + } + } + } + return true +} + +// Write the BOM character. +func yaml_emitter_write_bom(emitter *yaml_emitter_t) bool { + if !flush(emitter) { + return false + } + pos := emitter.buffer_pos + emitter.buffer[pos+0] = '\xEF' + emitter.buffer[pos+1] = '\xBB' + emitter.buffer[pos+2] = '\xBF' + emitter.buffer_pos += 3 + return true +} + +func yaml_emitter_write_indent(emitter *yaml_emitter_t) bool { + indent := emitter.indent + if indent < 0 { + indent = 0 + } + if !emitter.indention || emitter.column > indent || (emitter.column == indent && !emitter.whitespace) { + if !put_break(emitter) { + return false + } + } + if emitter.foot_indent == indent { + if !put_break(emitter) { + return false + } + } + for emitter.column < indent { + if !put(emitter, ' ') { + return false + } + } + emitter.whitespace = true + //emitter.indention = true + emitter.space_above = false + emitter.foot_indent = -1 + return true +} + +func yaml_emitter_write_indicator(emitter *yaml_emitter_t, indicator []byte, need_whitespace, is_whitespace, is_indention bool) bool { + if need_whitespace && !emitter.whitespace { + if !put(emitter, ' ') { + return false + } + } + if !write_all(emitter, indicator) { + return false + } + emitter.whitespace = is_whitespace + emitter.indention = (emitter.indention && is_indention) + emitter.open_ended = false + return true +} + +func yaml_emitter_write_anchor(emitter *yaml_emitter_t, value []byte) bool { + if !write_all(emitter, value) { + return false + } + emitter.whitespace = false + emitter.indention = false + return true +} + +func yaml_emitter_write_tag_handle(emitter *yaml_emitter_t, value []byte) bool { + if !emitter.whitespace { + if !put(emitter, ' ') { + return false + } + } + if !write_all(emitter, value) { + return false + } + emitter.whitespace = false + emitter.indention = false + return true +} + +func yaml_emitter_write_tag_content(emitter *yaml_emitter_t, value []byte, need_whitespace bool) bool { + if need_whitespace && !emitter.whitespace { + if !put(emitter, ' ') { + return false + } + } + for i := 0; i < len(value); { + var must_write bool + switch value[i] { + case ';', '/', '?', ':', '@', '&', '=', '+', '$', ',', '_', '.', '~', '*', '\'', '(', ')', '[', ']': + must_write = true + default: + must_write = is_alpha(value, i) + } + if must_write { + if !write(emitter, value, &i) { + return false + } + } else { + w := width(value[i]) + for k := 0; k < w; k++ { + octet := value[i] + i++ + if !put(emitter, '%') { + return false + } + + c := octet >> 4 + if c < 10 { + c += '0' + } else { + c += 'A' - 10 + } + if !put(emitter, c) { + return false + } + + c = octet & 0x0f + if c < 10 { + c += '0' + } else { + c += 'A' - 10 + } + if !put(emitter, c) { + return false + } + } + } + } + emitter.whitespace = false + emitter.indention = false + return true +} + +func yaml_emitter_write_plain_scalar(emitter *yaml_emitter_t, value []byte, allow_breaks bool) bool { + if len(value) > 0 && !emitter.whitespace { + if !put(emitter, ' ') { + return false + } + } + + spaces := false + breaks := false + for i := 0; i < len(value); { + if is_space(value, i) { + if allow_breaks && !spaces && emitter.column > emitter.best_width && !is_space(value, i+1) { + if !yaml_emitter_write_indent(emitter) { + return false + } + i += width(value[i]) + } else { + if !write(emitter, value, &i) { + return false + } + } + spaces = true + } else if is_break(value, i) { + if !breaks && value[i] == '\n' { + if !put_break(emitter) { + return false + } + } + if !write_break(emitter, value, &i) { + return false + } + //emitter.indention = true + breaks = true + } else { + if breaks { + if !yaml_emitter_write_indent(emitter) { + return false + } + } + if !write(emitter, value, &i) { + return false + } + emitter.indention = false + spaces = false + breaks = false + } + } + + if len(value) > 0 { + emitter.whitespace = false + } + emitter.indention = false + if emitter.root_context { + emitter.open_ended = true + } + + return true +} + +func yaml_emitter_write_single_quoted_scalar(emitter *yaml_emitter_t, value []byte, allow_breaks bool) bool { + + if !yaml_emitter_write_indicator(emitter, []byte{'\''}, true, false, false) { + return false + } + + spaces := false + breaks := false + for i := 0; i < len(value); { + if is_space(value, i) { + if allow_breaks && !spaces && emitter.column > emitter.best_width && i > 0 && i < len(value)-1 && !is_space(value, i+1) { + if !yaml_emitter_write_indent(emitter) { + return false + } + i += width(value[i]) + } else { + if !write(emitter, value, &i) { + return false + } + } + spaces = true + } else if is_break(value, i) { + if !breaks && value[i] == '\n' { + if !put_break(emitter) { + return false + } + } + if !write_break(emitter, value, &i) { + return false + } + //emitter.indention = true + breaks = true + } else { + if breaks { + if !yaml_emitter_write_indent(emitter) { + return false + } + } + if value[i] == '\'' { + if !put(emitter, '\'') { + return false + } + } + if !write(emitter, value, &i) { + return false + } + emitter.indention = false + spaces = false + breaks = false + } + } + if !yaml_emitter_write_indicator(emitter, []byte{'\''}, false, false, false) { + return false + } + emitter.whitespace = false + emitter.indention = false + return true +} + +func yaml_emitter_write_double_quoted_scalar(emitter *yaml_emitter_t, value []byte, allow_breaks bool) bool { + spaces := false + if !yaml_emitter_write_indicator(emitter, []byte{'"'}, true, false, false) { + return false + } + + for i := 0; i < len(value); { + if !is_printable(value, i) || (!emitter.unicode && !is_ascii(value, i)) || + is_bom(value, i) || is_break(value, i) || + value[i] == '"' || value[i] == '\\' { + + octet := value[i] + + var w int + var v rune + switch { + case octet&0x80 == 0x00: + w, v = 1, rune(octet&0x7F) + case octet&0xE0 == 0xC0: + w, v = 2, rune(octet&0x1F) + case octet&0xF0 == 0xE0: + w, v = 3, rune(octet&0x0F) + case octet&0xF8 == 0xF0: + w, v = 4, rune(octet&0x07) + } + for k := 1; k < w; k++ { + octet = value[i+k] + v = (v << 6) + (rune(octet) & 0x3F) + } + i += w + + if !put(emitter, '\\') { + return false + } + + var ok bool + switch v { + case 0x00: + ok = put(emitter, '0') + case 0x07: + ok = put(emitter, 'a') + case 0x08: + ok = put(emitter, 'b') + case 0x09: + ok = put(emitter, 't') + case 0x0A: + ok = put(emitter, 'n') + case 0x0b: + ok = put(emitter, 'v') + case 0x0c: + ok = put(emitter, 'f') + case 0x0d: + ok = put(emitter, 'r') + case 0x1b: + ok = put(emitter, 'e') + case 0x22: + ok = put(emitter, '"') + case 0x5c: + ok = put(emitter, '\\') + case 0x85: + ok = put(emitter, 'N') + case 0xA0: + ok = put(emitter, '_') + case 0x2028: + ok = put(emitter, 'L') + case 0x2029: + ok = put(emitter, 'P') + default: + if v <= 0xFF { + ok = put(emitter, 'x') + w = 2 + } else if v <= 0xFFFF { + ok = put(emitter, 'u') + w = 4 + } else { + ok = put(emitter, 'U') + w = 8 + } + for k := (w - 1) * 4; ok && k >= 0; k -= 4 { + digit := byte((v >> uint(k)) & 0x0F) + if digit < 10 { + ok = put(emitter, digit+'0') + } else { + ok = put(emitter, digit+'A'-10) + } + } + } + if !ok { + return false + } + spaces = false + } else if is_space(value, i) { + if allow_breaks && !spaces && emitter.column > emitter.best_width && i > 0 && i < len(value)-1 { + if !yaml_emitter_write_indent(emitter) { + return false + } + if is_space(value, i+1) { + if !put(emitter, '\\') { + return false + } + } + i += width(value[i]) + } else if !write(emitter, value, &i) { + return false + } + spaces = true + } else { + if !write(emitter, value, &i) { + return false + } + spaces = false + } + } + if !yaml_emitter_write_indicator(emitter, []byte{'"'}, false, false, false) { + return false + } + emitter.whitespace = false + emitter.indention = false + return true +} + +func yaml_emitter_write_block_scalar_hints(emitter *yaml_emitter_t, value []byte) bool { + if is_space(value, 0) || is_break(value, 0) { + indent_hint := []byte{'0' + byte(emitter.best_indent)} + if !yaml_emitter_write_indicator(emitter, indent_hint, false, false, false) { + return false + } + } + + emitter.open_ended = false + + var chomp_hint [1]byte + if len(value) == 0 { + chomp_hint[0] = '-' + } else { + i := len(value) - 1 + for value[i]&0xC0 == 0x80 { + i-- + } + if !is_break(value, i) { + chomp_hint[0] = '-' + } else if i == 0 { + chomp_hint[0] = '+' + emitter.open_ended = true + } else { + i-- + for value[i]&0xC0 == 0x80 { + i-- + } + if is_break(value, i) { + chomp_hint[0] = '+' + emitter.open_ended = true + } + } + } + if chomp_hint[0] != 0 { + if !yaml_emitter_write_indicator(emitter, chomp_hint[:], false, false, false) { + return false + } + } + return true +} + +func yaml_emitter_write_literal_scalar(emitter *yaml_emitter_t, value []byte) bool { + if !yaml_emitter_write_indicator(emitter, []byte{'|'}, true, false, false) { + return false + } + if !yaml_emitter_write_block_scalar_hints(emitter, value) { + return false + } + if !yaml_emitter_process_line_comment(emitter) { + return false + } + //emitter.indention = true + emitter.whitespace = true + breaks := true + for i := 0; i < len(value); { + if is_break(value, i) { + if !write_break(emitter, value, &i) { + return false + } + //emitter.indention = true + breaks = true + } else { + if breaks { + if !yaml_emitter_write_indent(emitter) { + return false + } + } + if !write(emitter, value, &i) { + return false + } + emitter.indention = false + breaks = false + } + } + + return true +} + +func yaml_emitter_write_folded_scalar(emitter *yaml_emitter_t, value []byte) bool { + if !yaml_emitter_write_indicator(emitter, []byte{'>'}, true, false, false) { + return false + } + if !yaml_emitter_write_block_scalar_hints(emitter, value) { + return false + } + if !yaml_emitter_process_line_comment(emitter) { + return false + } + + //emitter.indention = true + emitter.whitespace = true + + breaks := true + leading_spaces := true + for i := 0; i < len(value); { + if is_break(value, i) { + if !breaks && !leading_spaces && value[i] == '\n' { + k := 0 + for is_break(value, k) { + k += width(value[k]) + } + if !is_blankz(value, k) { + if !put_break(emitter) { + return false + } + } + } + if !write_break(emitter, value, &i) { + return false + } + //emitter.indention = true + breaks = true + } else { + if breaks { + if !yaml_emitter_write_indent(emitter) { + return false + } + leading_spaces = is_blank(value, i) + } + if !breaks && is_space(value, i) && !is_space(value, i+1) && emitter.column > emitter.best_width { + if !yaml_emitter_write_indent(emitter) { + return false + } + i += width(value[i]) + } else { + if !write(emitter, value, &i) { + return false + } + } + emitter.indention = false + breaks = false + } + } + return true +} + +func yaml_emitter_write_comment(emitter *yaml_emitter_t, comment []byte) bool { + breaks := false + pound := false + for i := 0; i < len(comment); { + if is_break(comment, i) { + if !write_break(emitter, comment, &i) { + return false + } + //emitter.indention = true + breaks = true + pound = false + } else { + if breaks && !yaml_emitter_write_indent(emitter) { + return false + } + if !pound { + if comment[i] != '#' && (!put(emitter, '#') || !put(emitter, ' ')) { + return false + } + pound = true + } + if !write(emitter, comment, &i) { + return false + } + emitter.indention = false + breaks = false + } + } + if !breaks && !put_break(emitter) { + return false + } + + emitter.whitespace = true + //emitter.indention = true + return true +} diff --git a/vendor/gopkg.in/yaml.v3/encode.go b/vendor/gopkg.in/yaml.v3/encode.go new file mode 100644 index 00000000..de9e72a3 --- /dev/null +++ b/vendor/gopkg.in/yaml.v3/encode.go @@ -0,0 +1,577 @@ +// +// Copyright (c) 2011-2019 Canonical Ltd +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. + +package yaml + +import ( + "encoding" + "fmt" + "io" + "reflect" + "regexp" + "sort" + "strconv" + "strings" + "time" + "unicode/utf8" +) + +type encoder struct { + emitter yaml_emitter_t + event yaml_event_t + out []byte + flow bool + indent int + doneInit bool +} + +func newEncoder() *encoder { + e := &encoder{} + yaml_emitter_initialize(&e.emitter) + yaml_emitter_set_output_string(&e.emitter, &e.out) + yaml_emitter_set_unicode(&e.emitter, true) + return e +} + +func newEncoderWithWriter(w io.Writer) *encoder { + e := &encoder{} + yaml_emitter_initialize(&e.emitter) + yaml_emitter_set_output_writer(&e.emitter, w) + yaml_emitter_set_unicode(&e.emitter, true) + return e +} + +func (e *encoder) init() { + if e.doneInit { + return + } + if e.indent == 0 { + e.indent = 4 + } + e.emitter.best_indent = e.indent + yaml_stream_start_event_initialize(&e.event, yaml_UTF8_ENCODING) + e.emit() + e.doneInit = true +} + +func (e *encoder) finish() { + e.emitter.open_ended = false + yaml_stream_end_event_initialize(&e.event) + e.emit() +} + +func (e *encoder) destroy() { + yaml_emitter_delete(&e.emitter) +} + +func (e *encoder) emit() { + // This will internally delete the e.event value. + e.must(yaml_emitter_emit(&e.emitter, &e.event)) +} + +func (e *encoder) must(ok bool) { + if !ok { + msg := e.emitter.problem + if msg == "" { + msg = "unknown problem generating YAML content" + } + failf("%s", msg) + } +} + +func (e *encoder) marshalDoc(tag string, in reflect.Value) { + e.init() + var node *Node + if in.IsValid() { + node, _ = in.Interface().(*Node) + } + if node != nil && node.Kind == DocumentNode { + e.nodev(in) + } else { + yaml_document_start_event_initialize(&e.event, nil, nil, true) + e.emit() + e.marshal(tag, in) + yaml_document_end_event_initialize(&e.event, true) + e.emit() + } +} + +func (e *encoder) marshal(tag string, in reflect.Value) { + tag = shortTag(tag) + if !in.IsValid() || in.Kind() == reflect.Ptr && in.IsNil() { + e.nilv() + return + } + iface := in.Interface() + switch value := iface.(type) { + case *Node: + e.nodev(in) + return + case Node: + if !in.CanAddr() { + var n = reflect.New(in.Type()).Elem() + n.Set(in) + in = n + } + e.nodev(in.Addr()) + return + case time.Time: + e.timev(tag, in) + return + case *time.Time: + e.timev(tag, in.Elem()) + return + case time.Duration: + e.stringv(tag, reflect.ValueOf(value.String())) + return + case Marshaler: + v, err := value.MarshalYAML() + if err != nil { + fail(err) + } + if v == nil { + e.nilv() + return + } + e.marshal(tag, reflect.ValueOf(v)) + return + case encoding.TextMarshaler: + text, err := value.MarshalText() + if err != nil { + fail(err) + } + in = reflect.ValueOf(string(text)) + case nil: + e.nilv() + return + } + switch in.Kind() { + case reflect.Interface: + e.marshal(tag, in.Elem()) + case reflect.Map: + e.mapv(tag, in) + case reflect.Ptr: + e.marshal(tag, in.Elem()) + case reflect.Struct: + e.structv(tag, in) + case reflect.Slice, reflect.Array: + e.slicev(tag, in) + case reflect.String: + e.stringv(tag, in) + case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64: + e.intv(tag, in) + case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64, reflect.Uintptr: + e.uintv(tag, in) + case reflect.Float32, reflect.Float64: + e.floatv(tag, in) + case reflect.Bool: + e.boolv(tag, in) + default: + panic("cannot marshal type: " + in.Type().String()) + } +} + +func (e *encoder) mapv(tag string, in reflect.Value) { + e.mappingv(tag, func() { + keys := keyList(in.MapKeys()) + sort.Sort(keys) + for _, k := range keys { + e.marshal("", k) + e.marshal("", in.MapIndex(k)) + } + }) +} + +func (e *encoder) fieldByIndex(v reflect.Value, index []int) (field reflect.Value) { + for _, num := range index { + for { + if v.Kind() == reflect.Ptr { + if v.IsNil() { + return reflect.Value{} + } + v = v.Elem() + continue + } + break + } + v = v.Field(num) + } + return v +} + +func (e *encoder) structv(tag string, in reflect.Value) { + sinfo, err := getStructInfo(in.Type()) + if err != nil { + panic(err) + } + e.mappingv(tag, func() { + for _, info := range sinfo.FieldsList { + var value reflect.Value + if info.Inline == nil { + value = in.Field(info.Num) + } else { + value = e.fieldByIndex(in, info.Inline) + if !value.IsValid() { + continue + } + } + if info.OmitEmpty && isZero(value) { + continue + } + e.marshal("", reflect.ValueOf(info.Key)) + e.flow = info.Flow + e.marshal("", value) + } + if sinfo.InlineMap >= 0 { + m := in.Field(sinfo.InlineMap) + if m.Len() > 0 { + e.flow = false + keys := keyList(m.MapKeys()) + sort.Sort(keys) + for _, k := range keys { + if _, found := sinfo.FieldsMap[k.String()]; found { + panic(fmt.Sprintf("cannot have key %q in inlined map: conflicts with struct field", k.String())) + } + e.marshal("", k) + e.flow = false + e.marshal("", m.MapIndex(k)) + } + } + } + }) +} + +func (e *encoder) mappingv(tag string, f func()) { + implicit := tag == "" + style := yaml_BLOCK_MAPPING_STYLE + if e.flow { + e.flow = false + style = yaml_FLOW_MAPPING_STYLE + } + yaml_mapping_start_event_initialize(&e.event, nil, []byte(tag), implicit, style) + e.emit() + f() + yaml_mapping_end_event_initialize(&e.event) + e.emit() +} + +func (e *encoder) slicev(tag string, in reflect.Value) { + implicit := tag == "" + style := yaml_BLOCK_SEQUENCE_STYLE + if e.flow { + e.flow = false + style = yaml_FLOW_SEQUENCE_STYLE + } + e.must(yaml_sequence_start_event_initialize(&e.event, nil, []byte(tag), implicit, style)) + e.emit() + n := in.Len() + for i := 0; i < n; i++ { + e.marshal("", in.Index(i)) + } + e.must(yaml_sequence_end_event_initialize(&e.event)) + e.emit() +} + +// isBase60 returns whether s is in base 60 notation as defined in YAML 1.1. +// +// The base 60 float notation in YAML 1.1 is a terrible idea and is unsupported +// in YAML 1.2 and by this package, but these should be marshalled quoted for +// the time being for compatibility with other parsers. +func isBase60Float(s string) (result bool) { + // Fast path. + if s == "" { + return false + } + c := s[0] + if !(c == '+' || c == '-' || c >= '0' && c <= '9') || strings.IndexByte(s, ':') < 0 { + return false + } + // Do the full match. + return base60float.MatchString(s) +} + +// From http://yaml.org/type/float.html, except the regular expression there +// is bogus. In practice parsers do not enforce the "\.[0-9_]*" suffix. +var base60float = regexp.MustCompile(`^[-+]?[0-9][0-9_]*(?::[0-5]?[0-9])+(?:\.[0-9_]*)?$`) + +// isOldBool returns whether s is bool notation as defined in YAML 1.1. +// +// We continue to force strings that YAML 1.1 would interpret as booleans to be +// rendered as quotes strings so that the marshalled output valid for YAML 1.1 +// parsing. +func isOldBool(s string) (result bool) { + switch s { + case "y", "Y", "yes", "Yes", "YES", "on", "On", "ON", + "n", "N", "no", "No", "NO", "off", "Off", "OFF": + return true + default: + return false + } +} + +func (e *encoder) stringv(tag string, in reflect.Value) { + var style yaml_scalar_style_t + s := in.String() + canUsePlain := true + switch { + case !utf8.ValidString(s): + if tag == binaryTag { + failf("explicitly tagged !!binary data must be base64-encoded") + } + if tag != "" { + failf("cannot marshal invalid UTF-8 data as %s", shortTag(tag)) + } + // It can't be encoded directly as YAML so use a binary tag + // and encode it as base64. + tag = binaryTag + s = encodeBase64(s) + case tag == "": + // Check to see if it would resolve to a specific + // tag when encoded unquoted. If it doesn't, + // there's no need to quote it. + rtag, _ := resolve("", s) + canUsePlain = rtag == strTag && !(isBase60Float(s) || isOldBool(s)) + } + // Note: it's possible for user code to emit invalid YAML + // if they explicitly specify a tag and a string containing + // text that's incompatible with that tag. + switch { + case strings.Contains(s, "\n"): + if e.flow { + style = yaml_DOUBLE_QUOTED_SCALAR_STYLE + } else { + style = yaml_LITERAL_SCALAR_STYLE + } + case canUsePlain: + style = yaml_PLAIN_SCALAR_STYLE + default: + style = yaml_DOUBLE_QUOTED_SCALAR_STYLE + } + e.emitScalar(s, "", tag, style, nil, nil, nil, nil) +} + +func (e *encoder) boolv(tag string, in reflect.Value) { + var s string + if in.Bool() { + s = "true" + } else { + s = "false" + } + e.emitScalar(s, "", tag, yaml_PLAIN_SCALAR_STYLE, nil, nil, nil, nil) +} + +func (e *encoder) intv(tag string, in reflect.Value) { + s := strconv.FormatInt(in.Int(), 10) + e.emitScalar(s, "", tag, yaml_PLAIN_SCALAR_STYLE, nil, nil, nil, nil) +} + +func (e *encoder) uintv(tag string, in reflect.Value) { + s := strconv.FormatUint(in.Uint(), 10) + e.emitScalar(s, "", tag, yaml_PLAIN_SCALAR_STYLE, nil, nil, nil, nil) +} + +func (e *encoder) timev(tag string, in reflect.Value) { + t := in.Interface().(time.Time) + s := t.Format(time.RFC3339Nano) + e.emitScalar(s, "", tag, yaml_PLAIN_SCALAR_STYLE, nil, nil, nil, nil) +} + +func (e *encoder) floatv(tag string, in reflect.Value) { + // Issue #352: When formatting, use the precision of the underlying value + precision := 64 + if in.Kind() == reflect.Float32 { + precision = 32 + } + + s := strconv.FormatFloat(in.Float(), 'g', -1, precision) + switch s { + case "+Inf": + s = ".inf" + case "-Inf": + s = "-.inf" + case "NaN": + s = ".nan" + } + e.emitScalar(s, "", tag, yaml_PLAIN_SCALAR_STYLE, nil, nil, nil, nil) +} + +func (e *encoder) nilv() { + e.emitScalar("null", "", "", yaml_PLAIN_SCALAR_STYLE, nil, nil, nil, nil) +} + +func (e *encoder) emitScalar(value, anchor, tag string, style yaml_scalar_style_t, head, line, foot, tail []byte) { + // TODO Kill this function. Replace all initialize calls by their underlining Go literals. + implicit := tag == "" + if !implicit { + tag = longTag(tag) + } + e.must(yaml_scalar_event_initialize(&e.event, []byte(anchor), []byte(tag), []byte(value), implicit, implicit, style)) + e.event.head_comment = head + e.event.line_comment = line + e.event.foot_comment = foot + e.event.tail_comment = tail + e.emit() +} + +func (e *encoder) nodev(in reflect.Value) { + e.node(in.Interface().(*Node), "") +} + +func (e *encoder) node(node *Node, tail string) { + // Zero nodes behave as nil. + if node.Kind == 0 && node.IsZero() { + e.nilv() + return + } + + // If the tag was not explicitly requested, and dropping it won't change the + // implicit tag of the value, don't include it in the presentation. + var tag = node.Tag + var stag = shortTag(tag) + var forceQuoting bool + if tag != "" && node.Style&TaggedStyle == 0 { + if node.Kind == ScalarNode { + if stag == strTag && node.Style&(SingleQuotedStyle|DoubleQuotedStyle|LiteralStyle|FoldedStyle) != 0 { + tag = "" + } else { + rtag, _ := resolve("", node.Value) + if rtag == stag { + tag = "" + } else if stag == strTag { + tag = "" + forceQuoting = true + } + } + } else { + var rtag string + switch node.Kind { + case MappingNode: + rtag = mapTag + case SequenceNode: + rtag = seqTag + } + if rtag == stag { + tag = "" + } + } + } + + switch node.Kind { + case DocumentNode: + yaml_document_start_event_initialize(&e.event, nil, nil, true) + e.event.head_comment = []byte(node.HeadComment) + e.emit() + for _, node := range node.Content { + e.node(node, "") + } + yaml_document_end_event_initialize(&e.event, true) + e.event.foot_comment = []byte(node.FootComment) + e.emit() + + case SequenceNode: + style := yaml_BLOCK_SEQUENCE_STYLE + if node.Style&FlowStyle != 0 { + style = yaml_FLOW_SEQUENCE_STYLE + } + e.must(yaml_sequence_start_event_initialize(&e.event, []byte(node.Anchor), []byte(longTag(tag)), tag == "", style)) + e.event.head_comment = []byte(node.HeadComment) + e.emit() + for _, node := range node.Content { + e.node(node, "") + } + e.must(yaml_sequence_end_event_initialize(&e.event)) + e.event.line_comment = []byte(node.LineComment) + e.event.foot_comment = []byte(node.FootComment) + e.emit() + + case MappingNode: + style := yaml_BLOCK_MAPPING_STYLE + if node.Style&FlowStyle != 0 { + style = yaml_FLOW_MAPPING_STYLE + } + yaml_mapping_start_event_initialize(&e.event, []byte(node.Anchor), []byte(longTag(tag)), tag == "", style) + e.event.tail_comment = []byte(tail) + e.event.head_comment = []byte(node.HeadComment) + e.emit() + + // The tail logic below moves the foot comment of prior keys to the following key, + // since the value for each key may be a nested structure and the foot needs to be + // processed only the entirety of the value is streamed. The last tail is processed + // with the mapping end event. + var tail string + for i := 0; i+1 < len(node.Content); i += 2 { + k := node.Content[i] + foot := k.FootComment + if foot != "" { + kopy := *k + kopy.FootComment = "" + k = &kopy + } + e.node(k, tail) + tail = foot + + v := node.Content[i+1] + e.node(v, "") + } + + yaml_mapping_end_event_initialize(&e.event) + e.event.tail_comment = []byte(tail) + e.event.line_comment = []byte(node.LineComment) + e.event.foot_comment = []byte(node.FootComment) + e.emit() + + case AliasNode: + yaml_alias_event_initialize(&e.event, []byte(node.Value)) + e.event.head_comment = []byte(node.HeadComment) + e.event.line_comment = []byte(node.LineComment) + e.event.foot_comment = []byte(node.FootComment) + e.emit() + + case ScalarNode: + value := node.Value + if !utf8.ValidString(value) { + if stag == binaryTag { + failf("explicitly tagged !!binary data must be base64-encoded") + } + if stag != "" { + failf("cannot marshal invalid UTF-8 data as %s", stag) + } + // It can't be encoded directly as YAML so use a binary tag + // and encode it as base64. + tag = binaryTag + value = encodeBase64(value) + } + + style := yaml_PLAIN_SCALAR_STYLE + switch { + case node.Style&DoubleQuotedStyle != 0: + style = yaml_DOUBLE_QUOTED_SCALAR_STYLE + case node.Style&SingleQuotedStyle != 0: + style = yaml_SINGLE_QUOTED_SCALAR_STYLE + case node.Style&LiteralStyle != 0: + style = yaml_LITERAL_SCALAR_STYLE + case node.Style&FoldedStyle != 0: + style = yaml_FOLDED_SCALAR_STYLE + case strings.Contains(value, "\n"): + style = yaml_LITERAL_SCALAR_STYLE + case forceQuoting: + style = yaml_DOUBLE_QUOTED_SCALAR_STYLE + } + + e.emitScalar(value, node.Anchor, tag, style, []byte(node.HeadComment), []byte(node.LineComment), []byte(node.FootComment), []byte(tail)) + default: + failf("cannot encode node with unknown kind %d", node.Kind) + } +} diff --git a/vendor/gopkg.in/yaml.v3/parserc.go b/vendor/gopkg.in/yaml.v3/parserc.go new file mode 100644 index 00000000..ac66fccc --- /dev/null +++ b/vendor/gopkg.in/yaml.v3/parserc.go @@ -0,0 +1,1249 @@ +// +// Copyright (c) 2011-2019 Canonical Ltd +// Copyright (c) 2006-2010 Kirill Simonov +// +// Permission is hereby granted, free of charge, to any person obtaining a copy of +// this software and associated documentation files (the "Software"), to deal in +// the Software without restriction, including without limitation the rights to +// use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies +// of the Software, and to permit persons to whom the Software is furnished to do +// so, subject to the following conditions: +// +// The above copyright notice and this permission notice shall be included in all +// copies or substantial portions of the Software. +// +// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE +// SOFTWARE. + +package yaml + +import ( + "bytes" +) + +// The parser implements the following grammar: +// +// stream ::= STREAM-START implicit_document? explicit_document* STREAM-END +// implicit_document ::= block_node DOCUMENT-END* +// explicit_document ::= DIRECTIVE* DOCUMENT-START block_node? DOCUMENT-END* +// block_node_or_indentless_sequence ::= +// ALIAS +// | properties (block_content | indentless_block_sequence)? +// | block_content +// | indentless_block_sequence +// block_node ::= ALIAS +// | properties block_content? +// | block_content +// flow_node ::= ALIAS +// | properties flow_content? +// | flow_content +// properties ::= TAG ANCHOR? | ANCHOR TAG? +// block_content ::= block_collection | flow_collection | SCALAR +// flow_content ::= flow_collection | SCALAR +// block_collection ::= block_sequence | block_mapping +// flow_collection ::= flow_sequence | flow_mapping +// block_sequence ::= BLOCK-SEQUENCE-START (BLOCK-ENTRY block_node?)* BLOCK-END +// indentless_sequence ::= (BLOCK-ENTRY block_node?)+ +// block_mapping ::= BLOCK-MAPPING_START +// ((KEY block_node_or_indentless_sequence?)? +// (VALUE block_node_or_indentless_sequence?)?)* +// BLOCK-END +// flow_sequence ::= FLOW-SEQUENCE-START +// (flow_sequence_entry FLOW-ENTRY)* +// flow_sequence_entry? +// FLOW-SEQUENCE-END +// flow_sequence_entry ::= flow_node | KEY flow_node? (VALUE flow_node?)? +// flow_mapping ::= FLOW-MAPPING-START +// (flow_mapping_entry FLOW-ENTRY)* +// flow_mapping_entry? +// FLOW-MAPPING-END +// flow_mapping_entry ::= flow_node | KEY flow_node? (VALUE flow_node?)? + +// Peek the next token in the token queue. +func peek_token(parser *yaml_parser_t) *yaml_token_t { + if parser.token_available || yaml_parser_fetch_more_tokens(parser) { + token := &parser.tokens[parser.tokens_head] + yaml_parser_unfold_comments(parser, token) + return token + } + return nil +} + +// yaml_parser_unfold_comments walks through the comments queue and joins all +// comments behind the position of the provided token into the respective +// top-level comment slices in the parser. +func yaml_parser_unfold_comments(parser *yaml_parser_t, token *yaml_token_t) { + for parser.comments_head < len(parser.comments) && token.start_mark.index >= parser.comments[parser.comments_head].token_mark.index { + comment := &parser.comments[parser.comments_head] + if len(comment.head) > 0 { + if token.typ == yaml_BLOCK_END_TOKEN { + // No heads on ends, so keep comment.head for a follow up token. + break + } + if len(parser.head_comment) > 0 { + parser.head_comment = append(parser.head_comment, '\n') + } + parser.head_comment = append(parser.head_comment, comment.head...) + } + if len(comment.foot) > 0 { + if len(parser.foot_comment) > 0 { + parser.foot_comment = append(parser.foot_comment, '\n') + } + parser.foot_comment = append(parser.foot_comment, comment.foot...) + } + if len(comment.line) > 0 { + if len(parser.line_comment) > 0 { + parser.line_comment = append(parser.line_comment, '\n') + } + parser.line_comment = append(parser.line_comment, comment.line...) + } + *comment = yaml_comment_t{} + parser.comments_head++ + } +} + +// Remove the next token from the queue (must be called after peek_token). +func skip_token(parser *yaml_parser_t) { + parser.token_available = false + parser.tokens_parsed++ + parser.stream_end_produced = parser.tokens[parser.tokens_head].typ == yaml_STREAM_END_TOKEN + parser.tokens_head++ +} + +// Get the next event. +func yaml_parser_parse(parser *yaml_parser_t, event *yaml_event_t) bool { + // Erase the event object. + *event = yaml_event_t{} + + // No events after the end of the stream or error. + if parser.stream_end_produced || parser.error != yaml_NO_ERROR || parser.state == yaml_PARSE_END_STATE { + return true + } + + // Generate the next event. + return yaml_parser_state_machine(parser, event) +} + +// Set parser error. +func yaml_parser_set_parser_error(parser *yaml_parser_t, problem string, problem_mark yaml_mark_t) bool { + parser.error = yaml_PARSER_ERROR + parser.problem = problem + parser.problem_mark = problem_mark + return false +} + +func yaml_parser_set_parser_error_context(parser *yaml_parser_t, context string, context_mark yaml_mark_t, problem string, problem_mark yaml_mark_t) bool { + parser.error = yaml_PARSER_ERROR + parser.context = context + parser.context_mark = context_mark + parser.problem = problem + parser.problem_mark = problem_mark + return false +} + +// State dispatcher. +func yaml_parser_state_machine(parser *yaml_parser_t, event *yaml_event_t) bool { + //trace("yaml_parser_state_machine", "state:", parser.state.String()) + + switch parser.state { + case yaml_PARSE_STREAM_START_STATE: + return yaml_parser_parse_stream_start(parser, event) + + case yaml_PARSE_IMPLICIT_DOCUMENT_START_STATE: + return yaml_parser_parse_document_start(parser, event, true) + + case yaml_PARSE_DOCUMENT_START_STATE: + return yaml_parser_parse_document_start(parser, event, false) + + case yaml_PARSE_DOCUMENT_CONTENT_STATE: + return yaml_parser_parse_document_content(parser, event) + + case yaml_PARSE_DOCUMENT_END_STATE: + return yaml_parser_parse_document_end(parser, event) + + case yaml_PARSE_BLOCK_NODE_STATE: + return yaml_parser_parse_node(parser, event, true, false) + + case yaml_PARSE_BLOCK_NODE_OR_INDENTLESS_SEQUENCE_STATE: + return yaml_parser_parse_node(parser, event, true, true) + + case yaml_PARSE_FLOW_NODE_STATE: + return yaml_parser_parse_node(parser, event, false, false) + + case yaml_PARSE_BLOCK_SEQUENCE_FIRST_ENTRY_STATE: + return yaml_parser_parse_block_sequence_entry(parser, event, true) + + case yaml_PARSE_BLOCK_SEQUENCE_ENTRY_STATE: + return yaml_parser_parse_block_sequence_entry(parser, event, false) + + case yaml_PARSE_INDENTLESS_SEQUENCE_ENTRY_STATE: + return yaml_parser_parse_indentless_sequence_entry(parser, event) + + case yaml_PARSE_BLOCK_MAPPING_FIRST_KEY_STATE: + return yaml_parser_parse_block_mapping_key(parser, event, true) + + case yaml_PARSE_BLOCK_MAPPING_KEY_STATE: + return yaml_parser_parse_block_mapping_key(parser, event, false) + + case yaml_PARSE_BLOCK_MAPPING_VALUE_STATE: + return yaml_parser_parse_block_mapping_value(parser, event) + + case yaml_PARSE_FLOW_SEQUENCE_FIRST_ENTRY_STATE: + return yaml_parser_parse_flow_sequence_entry(parser, event, true) + + case yaml_PARSE_FLOW_SEQUENCE_ENTRY_STATE: + return yaml_parser_parse_flow_sequence_entry(parser, event, false) + + case yaml_PARSE_FLOW_SEQUENCE_ENTRY_MAPPING_KEY_STATE: + return yaml_parser_parse_flow_sequence_entry_mapping_key(parser, event) + + case yaml_PARSE_FLOW_SEQUENCE_ENTRY_MAPPING_VALUE_STATE: + return yaml_parser_parse_flow_sequence_entry_mapping_value(parser, event) + + case yaml_PARSE_FLOW_SEQUENCE_ENTRY_MAPPING_END_STATE: + return yaml_parser_parse_flow_sequence_entry_mapping_end(parser, event) + + case yaml_PARSE_FLOW_MAPPING_FIRST_KEY_STATE: + return yaml_parser_parse_flow_mapping_key(parser, event, true) + + case yaml_PARSE_FLOW_MAPPING_KEY_STATE: + return yaml_parser_parse_flow_mapping_key(parser, event, false) + + case yaml_PARSE_FLOW_MAPPING_VALUE_STATE: + return yaml_parser_parse_flow_mapping_value(parser, event, false) + + case yaml_PARSE_FLOW_MAPPING_EMPTY_VALUE_STATE: + return yaml_parser_parse_flow_mapping_value(parser, event, true) + + default: + panic("invalid parser state") + } +} + +// Parse the production: +// stream ::= STREAM-START implicit_document? explicit_document* STREAM-END +// ************ +func yaml_parser_parse_stream_start(parser *yaml_parser_t, event *yaml_event_t) bool { + token := peek_token(parser) + if token == nil { + return false + } + if token.typ != yaml_STREAM_START_TOKEN { + return yaml_parser_set_parser_error(parser, "did not find expected ", token.start_mark) + } + parser.state = yaml_PARSE_IMPLICIT_DOCUMENT_START_STATE + *event = yaml_event_t{ + typ: yaml_STREAM_START_EVENT, + start_mark: token.start_mark, + end_mark: token.end_mark, + encoding: token.encoding, + } + skip_token(parser) + return true +} + +// Parse the productions: +// implicit_document ::= block_node DOCUMENT-END* +// * +// explicit_document ::= DIRECTIVE* DOCUMENT-START block_node? DOCUMENT-END* +// ************************* +func yaml_parser_parse_document_start(parser *yaml_parser_t, event *yaml_event_t, implicit bool) bool { + + token := peek_token(parser) + if token == nil { + return false + } + + // Parse extra document end indicators. + if !implicit { + for token.typ == yaml_DOCUMENT_END_TOKEN { + skip_token(parser) + token = peek_token(parser) + if token == nil { + return false + } + } + } + + if implicit && token.typ != yaml_VERSION_DIRECTIVE_TOKEN && + token.typ != yaml_TAG_DIRECTIVE_TOKEN && + token.typ != yaml_DOCUMENT_START_TOKEN && + token.typ != yaml_STREAM_END_TOKEN { + // Parse an implicit document. + if !yaml_parser_process_directives(parser, nil, nil) { + return false + } + parser.states = append(parser.states, yaml_PARSE_DOCUMENT_END_STATE) + parser.state = yaml_PARSE_BLOCK_NODE_STATE + + var head_comment []byte + if len(parser.head_comment) > 0 { + // [Go] Scan the header comment backwards, and if an empty line is found, break + // the header so the part before the last empty line goes into the + // document header, while the bottom of it goes into a follow up event. + for i := len(parser.head_comment) - 1; i > 0; i-- { + if parser.head_comment[i] == '\n' { + if i == len(parser.head_comment)-1 { + head_comment = parser.head_comment[:i] + parser.head_comment = parser.head_comment[i+1:] + break + } else if parser.head_comment[i-1] == '\n' { + head_comment = parser.head_comment[:i-1] + parser.head_comment = parser.head_comment[i+1:] + break + } + } + } + } + + *event = yaml_event_t{ + typ: yaml_DOCUMENT_START_EVENT, + start_mark: token.start_mark, + end_mark: token.end_mark, + + head_comment: head_comment, + } + + } else if token.typ != yaml_STREAM_END_TOKEN { + // Parse an explicit document. + var version_directive *yaml_version_directive_t + var tag_directives []yaml_tag_directive_t + start_mark := token.start_mark + if !yaml_parser_process_directives(parser, &version_directive, &tag_directives) { + return false + } + token = peek_token(parser) + if token == nil { + return false + } + if token.typ != yaml_DOCUMENT_START_TOKEN { + yaml_parser_set_parser_error(parser, + "did not find expected ", token.start_mark) + return false + } + parser.states = append(parser.states, yaml_PARSE_DOCUMENT_END_STATE) + parser.state = yaml_PARSE_DOCUMENT_CONTENT_STATE + end_mark := token.end_mark + + *event = yaml_event_t{ + typ: yaml_DOCUMENT_START_EVENT, + start_mark: start_mark, + end_mark: end_mark, + version_directive: version_directive, + tag_directives: tag_directives, + implicit: false, + } + skip_token(parser) + + } else { + // Parse the stream end. + parser.state = yaml_PARSE_END_STATE + *event = yaml_event_t{ + typ: yaml_STREAM_END_EVENT, + start_mark: token.start_mark, + end_mark: token.end_mark, + } + skip_token(parser) + } + + return true +} + +// Parse the productions: +// explicit_document ::= DIRECTIVE* DOCUMENT-START block_node? DOCUMENT-END* +// *********** +// +func yaml_parser_parse_document_content(parser *yaml_parser_t, event *yaml_event_t) bool { + token := peek_token(parser) + if token == nil { + return false + } + + if token.typ == yaml_VERSION_DIRECTIVE_TOKEN || + token.typ == yaml_TAG_DIRECTIVE_TOKEN || + token.typ == yaml_DOCUMENT_START_TOKEN || + token.typ == yaml_DOCUMENT_END_TOKEN || + token.typ == yaml_STREAM_END_TOKEN { + parser.state = parser.states[len(parser.states)-1] + parser.states = parser.states[:len(parser.states)-1] + return yaml_parser_process_empty_scalar(parser, event, + token.start_mark) + } + return yaml_parser_parse_node(parser, event, true, false) +} + +// Parse the productions: +// implicit_document ::= block_node DOCUMENT-END* +// ************* +// explicit_document ::= DIRECTIVE* DOCUMENT-START block_node? DOCUMENT-END* +// +func yaml_parser_parse_document_end(parser *yaml_parser_t, event *yaml_event_t) bool { + token := peek_token(parser) + if token == nil { + return false + } + + start_mark := token.start_mark + end_mark := token.start_mark + + implicit := true + if token.typ == yaml_DOCUMENT_END_TOKEN { + end_mark = token.end_mark + skip_token(parser) + implicit = false + } + + parser.tag_directives = parser.tag_directives[:0] + + parser.state = yaml_PARSE_DOCUMENT_START_STATE + *event = yaml_event_t{ + typ: yaml_DOCUMENT_END_EVENT, + start_mark: start_mark, + end_mark: end_mark, + implicit: implicit, + } + yaml_parser_set_event_comments(parser, event) + if len(event.head_comment) > 0 && len(event.foot_comment) == 0 { + event.foot_comment = event.head_comment + event.head_comment = nil + } + return true +} + +func yaml_parser_set_event_comments(parser *yaml_parser_t, event *yaml_event_t) { + event.head_comment = parser.head_comment + event.line_comment = parser.line_comment + event.foot_comment = parser.foot_comment + parser.head_comment = nil + parser.line_comment = nil + parser.foot_comment = nil + parser.tail_comment = nil + parser.stem_comment = nil +} + +// Parse the productions: +// block_node_or_indentless_sequence ::= +// ALIAS +// ***** +// | properties (block_content | indentless_block_sequence)? +// ********** * +// | block_content | indentless_block_sequence +// * +// block_node ::= ALIAS +// ***** +// | properties block_content? +// ********** * +// | block_content +// * +// flow_node ::= ALIAS +// ***** +// | properties flow_content? +// ********** * +// | flow_content +// * +// properties ::= TAG ANCHOR? | ANCHOR TAG? +// ************************* +// block_content ::= block_collection | flow_collection | SCALAR +// ****** +// flow_content ::= flow_collection | SCALAR +// ****** +func yaml_parser_parse_node(parser *yaml_parser_t, event *yaml_event_t, block, indentless_sequence bool) bool { + //defer trace("yaml_parser_parse_node", "block:", block, "indentless_sequence:", indentless_sequence)() + + token := peek_token(parser) + if token == nil { + return false + } + + if token.typ == yaml_ALIAS_TOKEN { + parser.state = parser.states[len(parser.states)-1] + parser.states = parser.states[:len(parser.states)-1] + *event = yaml_event_t{ + typ: yaml_ALIAS_EVENT, + start_mark: token.start_mark, + end_mark: token.end_mark, + anchor: token.value, + } + yaml_parser_set_event_comments(parser, event) + skip_token(parser) + return true + } + + start_mark := token.start_mark + end_mark := token.start_mark + + var tag_token bool + var tag_handle, tag_suffix, anchor []byte + var tag_mark yaml_mark_t + if token.typ == yaml_ANCHOR_TOKEN { + anchor = token.value + start_mark = token.start_mark + end_mark = token.end_mark + skip_token(parser) + token = peek_token(parser) + if token == nil { + return false + } + if token.typ == yaml_TAG_TOKEN { + tag_token = true + tag_handle = token.value + tag_suffix = token.suffix + tag_mark = token.start_mark + end_mark = token.end_mark + skip_token(parser) + token = peek_token(parser) + if token == nil { + return false + } + } + } else if token.typ == yaml_TAG_TOKEN { + tag_token = true + tag_handle = token.value + tag_suffix = token.suffix + start_mark = token.start_mark + tag_mark = token.start_mark + end_mark = token.end_mark + skip_token(parser) + token = peek_token(parser) + if token == nil { + return false + } + if token.typ == yaml_ANCHOR_TOKEN { + anchor = token.value + end_mark = token.end_mark + skip_token(parser) + token = peek_token(parser) + if token == nil { + return false + } + } + } + + var tag []byte + if tag_token { + if len(tag_handle) == 0 { + tag = tag_suffix + tag_suffix = nil + } else { + for i := range parser.tag_directives { + if bytes.Equal(parser.tag_directives[i].handle, tag_handle) { + tag = append([]byte(nil), parser.tag_directives[i].prefix...) + tag = append(tag, tag_suffix...) + break + } + } + if len(tag) == 0 { + yaml_parser_set_parser_error_context(parser, + "while parsing a node", start_mark, + "found undefined tag handle", tag_mark) + return false + } + } + } + + implicit := len(tag) == 0 + if indentless_sequence && token.typ == yaml_BLOCK_ENTRY_TOKEN { + end_mark = token.end_mark + parser.state = yaml_PARSE_INDENTLESS_SEQUENCE_ENTRY_STATE + *event = yaml_event_t{ + typ: yaml_SEQUENCE_START_EVENT, + start_mark: start_mark, + end_mark: end_mark, + anchor: anchor, + tag: tag, + implicit: implicit, + style: yaml_style_t(yaml_BLOCK_SEQUENCE_STYLE), + } + return true + } + if token.typ == yaml_SCALAR_TOKEN { + var plain_implicit, quoted_implicit bool + end_mark = token.end_mark + if (len(tag) == 0 && token.style == yaml_PLAIN_SCALAR_STYLE) || (len(tag) == 1 && tag[0] == '!') { + plain_implicit = true + } else if len(tag) == 0 { + quoted_implicit = true + } + parser.state = parser.states[len(parser.states)-1] + parser.states = parser.states[:len(parser.states)-1] + + *event = yaml_event_t{ + typ: yaml_SCALAR_EVENT, + start_mark: start_mark, + end_mark: end_mark, + anchor: anchor, + tag: tag, + value: token.value, + implicit: plain_implicit, + quoted_implicit: quoted_implicit, + style: yaml_style_t(token.style), + } + yaml_parser_set_event_comments(parser, event) + skip_token(parser) + return true + } + if token.typ == yaml_FLOW_SEQUENCE_START_TOKEN { + // [Go] Some of the events below can be merged as they differ only on style. + end_mark = token.end_mark + parser.state = yaml_PARSE_FLOW_SEQUENCE_FIRST_ENTRY_STATE + *event = yaml_event_t{ + typ: yaml_SEQUENCE_START_EVENT, + start_mark: start_mark, + end_mark: end_mark, + anchor: anchor, + tag: tag, + implicit: implicit, + style: yaml_style_t(yaml_FLOW_SEQUENCE_STYLE), + } + yaml_parser_set_event_comments(parser, event) + return true + } + if token.typ == yaml_FLOW_MAPPING_START_TOKEN { + end_mark = token.end_mark + parser.state = yaml_PARSE_FLOW_MAPPING_FIRST_KEY_STATE + *event = yaml_event_t{ + typ: yaml_MAPPING_START_EVENT, + start_mark: start_mark, + end_mark: end_mark, + anchor: anchor, + tag: tag, + implicit: implicit, + style: yaml_style_t(yaml_FLOW_MAPPING_STYLE), + } + yaml_parser_set_event_comments(parser, event) + return true + } + if block && token.typ == yaml_BLOCK_SEQUENCE_START_TOKEN { + end_mark = token.end_mark + parser.state = yaml_PARSE_BLOCK_SEQUENCE_FIRST_ENTRY_STATE + *event = yaml_event_t{ + typ: yaml_SEQUENCE_START_EVENT, + start_mark: start_mark, + end_mark: end_mark, + anchor: anchor, + tag: tag, + implicit: implicit, + style: yaml_style_t(yaml_BLOCK_SEQUENCE_STYLE), + } + if parser.stem_comment != nil { + event.head_comment = parser.stem_comment + parser.stem_comment = nil + } + return true + } + if block && token.typ == yaml_BLOCK_MAPPING_START_TOKEN { + end_mark = token.end_mark + parser.state = yaml_PARSE_BLOCK_MAPPING_FIRST_KEY_STATE + *event = yaml_event_t{ + typ: yaml_MAPPING_START_EVENT, + start_mark: start_mark, + end_mark: end_mark, + anchor: anchor, + tag: tag, + implicit: implicit, + style: yaml_style_t(yaml_BLOCK_MAPPING_STYLE), + } + if parser.stem_comment != nil { + event.head_comment = parser.stem_comment + parser.stem_comment = nil + } + return true + } + if len(anchor) > 0 || len(tag) > 0 { + parser.state = parser.states[len(parser.states)-1] + parser.states = parser.states[:len(parser.states)-1] + + *event = yaml_event_t{ + typ: yaml_SCALAR_EVENT, + start_mark: start_mark, + end_mark: end_mark, + anchor: anchor, + tag: tag, + implicit: implicit, + quoted_implicit: false, + style: yaml_style_t(yaml_PLAIN_SCALAR_STYLE), + } + return true + } + + context := "while parsing a flow node" + if block { + context = "while parsing a block node" + } + yaml_parser_set_parser_error_context(parser, context, start_mark, + "did not find expected node content", token.start_mark) + return false +} + +// Parse the productions: +// block_sequence ::= BLOCK-SEQUENCE-START (BLOCK-ENTRY block_node?)* BLOCK-END +// ******************** *********** * ********* +// +func yaml_parser_parse_block_sequence_entry(parser *yaml_parser_t, event *yaml_event_t, first bool) bool { + if first { + token := peek_token(parser) + parser.marks = append(parser.marks, token.start_mark) + skip_token(parser) + } + + token := peek_token(parser) + if token == nil { + return false + } + + if token.typ == yaml_BLOCK_ENTRY_TOKEN { + mark := token.end_mark + prior_head_len := len(parser.head_comment) + skip_token(parser) + yaml_parser_split_stem_comment(parser, prior_head_len) + token = peek_token(parser) + if token == nil { + return false + } + if token.typ != yaml_BLOCK_ENTRY_TOKEN && token.typ != yaml_BLOCK_END_TOKEN { + parser.states = append(parser.states, yaml_PARSE_BLOCK_SEQUENCE_ENTRY_STATE) + return yaml_parser_parse_node(parser, event, true, false) + } else { + parser.state = yaml_PARSE_BLOCK_SEQUENCE_ENTRY_STATE + return yaml_parser_process_empty_scalar(parser, event, mark) + } + } + if token.typ == yaml_BLOCK_END_TOKEN { + parser.state = parser.states[len(parser.states)-1] + parser.states = parser.states[:len(parser.states)-1] + parser.marks = parser.marks[:len(parser.marks)-1] + + *event = yaml_event_t{ + typ: yaml_SEQUENCE_END_EVENT, + start_mark: token.start_mark, + end_mark: token.end_mark, + } + + skip_token(parser) + return true + } + + context_mark := parser.marks[len(parser.marks)-1] + parser.marks = parser.marks[:len(parser.marks)-1] + return yaml_parser_set_parser_error_context(parser, + "while parsing a block collection", context_mark, + "did not find expected '-' indicator", token.start_mark) +} + +// Parse the productions: +// indentless_sequence ::= (BLOCK-ENTRY block_node?)+ +// *********** * +func yaml_parser_parse_indentless_sequence_entry(parser *yaml_parser_t, event *yaml_event_t) bool { + token := peek_token(parser) + if token == nil { + return false + } + + if token.typ == yaml_BLOCK_ENTRY_TOKEN { + mark := token.end_mark + prior_head_len := len(parser.head_comment) + skip_token(parser) + yaml_parser_split_stem_comment(parser, prior_head_len) + token = peek_token(parser) + if token == nil { + return false + } + if token.typ != yaml_BLOCK_ENTRY_TOKEN && + token.typ != yaml_KEY_TOKEN && + token.typ != yaml_VALUE_TOKEN && + token.typ != yaml_BLOCK_END_TOKEN { + parser.states = append(parser.states, yaml_PARSE_INDENTLESS_SEQUENCE_ENTRY_STATE) + return yaml_parser_parse_node(parser, event, true, false) + } + parser.state = yaml_PARSE_INDENTLESS_SEQUENCE_ENTRY_STATE + return yaml_parser_process_empty_scalar(parser, event, mark) + } + parser.state = parser.states[len(parser.states)-1] + parser.states = parser.states[:len(parser.states)-1] + + *event = yaml_event_t{ + typ: yaml_SEQUENCE_END_EVENT, + start_mark: token.start_mark, + end_mark: token.start_mark, // [Go] Shouldn't this be token.end_mark? + } + return true +} + +// Split stem comment from head comment. +// +// When a sequence or map is found under a sequence entry, the former head comment +// is assigned to the underlying sequence or map as a whole, not the individual +// sequence or map entry as would be expected otherwise. To handle this case the +// previous head comment is moved aside as the stem comment. +func yaml_parser_split_stem_comment(parser *yaml_parser_t, stem_len int) { + if stem_len == 0 { + return + } + + token := peek_token(parser) + if token.typ != yaml_BLOCK_SEQUENCE_START_TOKEN && token.typ != yaml_BLOCK_MAPPING_START_TOKEN { + return + } + + parser.stem_comment = parser.head_comment[:stem_len] + if len(parser.head_comment) == stem_len { + parser.head_comment = nil + } else { + // Copy suffix to prevent very strange bugs if someone ever appends + // further bytes to the prefix in the stem_comment slice above. + parser.head_comment = append([]byte(nil), parser.head_comment[stem_len+1:]...) + } +} + +// Parse the productions: +// block_mapping ::= BLOCK-MAPPING_START +// ******************* +// ((KEY block_node_or_indentless_sequence?)? +// *** * +// (VALUE block_node_or_indentless_sequence?)?)* +// +// BLOCK-END +// ********* +// +func yaml_parser_parse_block_mapping_key(parser *yaml_parser_t, event *yaml_event_t, first bool) bool { + if first { + token := peek_token(parser) + parser.marks = append(parser.marks, token.start_mark) + skip_token(parser) + } + + token := peek_token(parser) + if token == nil { + return false + } + + // [Go] A tail comment was left from the prior mapping value processed. Emit an event + // as it needs to be processed with that value and not the following key. + if len(parser.tail_comment) > 0 { + *event = yaml_event_t{ + typ: yaml_TAIL_COMMENT_EVENT, + start_mark: token.start_mark, + end_mark: token.end_mark, + foot_comment: parser.tail_comment, + } + parser.tail_comment = nil + return true + } + + if token.typ == yaml_KEY_TOKEN { + mark := token.end_mark + skip_token(parser) + token = peek_token(parser) + if token == nil { + return false + } + if token.typ != yaml_KEY_TOKEN && + token.typ != yaml_VALUE_TOKEN && + token.typ != yaml_BLOCK_END_TOKEN { + parser.states = append(parser.states, yaml_PARSE_BLOCK_MAPPING_VALUE_STATE) + return yaml_parser_parse_node(parser, event, true, true) + } else { + parser.state = yaml_PARSE_BLOCK_MAPPING_VALUE_STATE + return yaml_parser_process_empty_scalar(parser, event, mark) + } + } else if token.typ == yaml_BLOCK_END_TOKEN { + parser.state = parser.states[len(parser.states)-1] + parser.states = parser.states[:len(parser.states)-1] + parser.marks = parser.marks[:len(parser.marks)-1] + *event = yaml_event_t{ + typ: yaml_MAPPING_END_EVENT, + start_mark: token.start_mark, + end_mark: token.end_mark, + } + yaml_parser_set_event_comments(parser, event) + skip_token(parser) + return true + } + + context_mark := parser.marks[len(parser.marks)-1] + parser.marks = parser.marks[:len(parser.marks)-1] + return yaml_parser_set_parser_error_context(parser, + "while parsing a block mapping", context_mark, + "did not find expected key", token.start_mark) +} + +// Parse the productions: +// block_mapping ::= BLOCK-MAPPING_START +// +// ((KEY block_node_or_indentless_sequence?)? +// +// (VALUE block_node_or_indentless_sequence?)?)* +// ***** * +// BLOCK-END +// +// +func yaml_parser_parse_block_mapping_value(parser *yaml_parser_t, event *yaml_event_t) bool { + token := peek_token(parser) + if token == nil { + return false + } + if token.typ == yaml_VALUE_TOKEN { + mark := token.end_mark + skip_token(parser) + token = peek_token(parser) + if token == nil { + return false + } + if token.typ != yaml_KEY_TOKEN && + token.typ != yaml_VALUE_TOKEN && + token.typ != yaml_BLOCK_END_TOKEN { + parser.states = append(parser.states, yaml_PARSE_BLOCK_MAPPING_KEY_STATE) + return yaml_parser_parse_node(parser, event, true, true) + } + parser.state = yaml_PARSE_BLOCK_MAPPING_KEY_STATE + return yaml_parser_process_empty_scalar(parser, event, mark) + } + parser.state = yaml_PARSE_BLOCK_MAPPING_KEY_STATE + return yaml_parser_process_empty_scalar(parser, event, token.start_mark) +} + +// Parse the productions: +// flow_sequence ::= FLOW-SEQUENCE-START +// ******************* +// (flow_sequence_entry FLOW-ENTRY)* +// * ********** +// flow_sequence_entry? +// * +// FLOW-SEQUENCE-END +// ***************** +// flow_sequence_entry ::= flow_node | KEY flow_node? (VALUE flow_node?)? +// * +// +func yaml_parser_parse_flow_sequence_entry(parser *yaml_parser_t, event *yaml_event_t, first bool) bool { + if first { + token := peek_token(parser) + parser.marks = append(parser.marks, token.start_mark) + skip_token(parser) + } + token := peek_token(parser) + if token == nil { + return false + } + if token.typ != yaml_FLOW_SEQUENCE_END_TOKEN { + if !first { + if token.typ == yaml_FLOW_ENTRY_TOKEN { + skip_token(parser) + token = peek_token(parser) + if token == nil { + return false + } + } else { + context_mark := parser.marks[len(parser.marks)-1] + parser.marks = parser.marks[:len(parser.marks)-1] + return yaml_parser_set_parser_error_context(parser, + "while parsing a flow sequence", context_mark, + "did not find expected ',' or ']'", token.start_mark) + } + } + + if token.typ == yaml_KEY_TOKEN { + parser.state = yaml_PARSE_FLOW_SEQUENCE_ENTRY_MAPPING_KEY_STATE + *event = yaml_event_t{ + typ: yaml_MAPPING_START_EVENT, + start_mark: token.start_mark, + end_mark: token.end_mark, + implicit: true, + style: yaml_style_t(yaml_FLOW_MAPPING_STYLE), + } + skip_token(parser) + return true + } else if token.typ != yaml_FLOW_SEQUENCE_END_TOKEN { + parser.states = append(parser.states, yaml_PARSE_FLOW_SEQUENCE_ENTRY_STATE) + return yaml_parser_parse_node(parser, event, false, false) + } + } + + parser.state = parser.states[len(parser.states)-1] + parser.states = parser.states[:len(parser.states)-1] + parser.marks = parser.marks[:len(parser.marks)-1] + + *event = yaml_event_t{ + typ: yaml_SEQUENCE_END_EVENT, + start_mark: token.start_mark, + end_mark: token.end_mark, + } + yaml_parser_set_event_comments(parser, event) + + skip_token(parser) + return true +} + +// +// Parse the productions: +// flow_sequence_entry ::= flow_node | KEY flow_node? (VALUE flow_node?)? +// *** * +// +func yaml_parser_parse_flow_sequence_entry_mapping_key(parser *yaml_parser_t, event *yaml_event_t) bool { + token := peek_token(parser) + if token == nil { + return false + } + if token.typ != yaml_VALUE_TOKEN && + token.typ != yaml_FLOW_ENTRY_TOKEN && + token.typ != yaml_FLOW_SEQUENCE_END_TOKEN { + parser.states = append(parser.states, yaml_PARSE_FLOW_SEQUENCE_ENTRY_MAPPING_VALUE_STATE) + return yaml_parser_parse_node(parser, event, false, false) + } + mark := token.end_mark + skip_token(parser) + parser.state = yaml_PARSE_FLOW_SEQUENCE_ENTRY_MAPPING_VALUE_STATE + return yaml_parser_process_empty_scalar(parser, event, mark) +} + +// Parse the productions: +// flow_sequence_entry ::= flow_node | KEY flow_node? (VALUE flow_node?)? +// ***** * +// +func yaml_parser_parse_flow_sequence_entry_mapping_value(parser *yaml_parser_t, event *yaml_event_t) bool { + token := peek_token(parser) + if token == nil { + return false + } + if token.typ == yaml_VALUE_TOKEN { + skip_token(parser) + token := peek_token(parser) + if token == nil { + return false + } + if token.typ != yaml_FLOW_ENTRY_TOKEN && token.typ != yaml_FLOW_SEQUENCE_END_TOKEN { + parser.states = append(parser.states, yaml_PARSE_FLOW_SEQUENCE_ENTRY_MAPPING_END_STATE) + return yaml_parser_parse_node(parser, event, false, false) + } + } + parser.state = yaml_PARSE_FLOW_SEQUENCE_ENTRY_MAPPING_END_STATE + return yaml_parser_process_empty_scalar(parser, event, token.start_mark) +} + +// Parse the productions: +// flow_sequence_entry ::= flow_node | KEY flow_node? (VALUE flow_node?)? +// * +// +func yaml_parser_parse_flow_sequence_entry_mapping_end(parser *yaml_parser_t, event *yaml_event_t) bool { + token := peek_token(parser) + if token == nil { + return false + } + parser.state = yaml_PARSE_FLOW_SEQUENCE_ENTRY_STATE + *event = yaml_event_t{ + typ: yaml_MAPPING_END_EVENT, + start_mark: token.start_mark, + end_mark: token.start_mark, // [Go] Shouldn't this be end_mark? + } + return true +} + +// Parse the productions: +// flow_mapping ::= FLOW-MAPPING-START +// ****************** +// (flow_mapping_entry FLOW-ENTRY)* +// * ********** +// flow_mapping_entry? +// ****************** +// FLOW-MAPPING-END +// **************** +// flow_mapping_entry ::= flow_node | KEY flow_node? (VALUE flow_node?)? +// * *** * +// +func yaml_parser_parse_flow_mapping_key(parser *yaml_parser_t, event *yaml_event_t, first bool) bool { + if first { + token := peek_token(parser) + parser.marks = append(parser.marks, token.start_mark) + skip_token(parser) + } + + token := peek_token(parser) + if token == nil { + return false + } + + if token.typ != yaml_FLOW_MAPPING_END_TOKEN { + if !first { + if token.typ == yaml_FLOW_ENTRY_TOKEN { + skip_token(parser) + token = peek_token(parser) + if token == nil { + return false + } + } else { + context_mark := parser.marks[len(parser.marks)-1] + parser.marks = parser.marks[:len(parser.marks)-1] + return yaml_parser_set_parser_error_context(parser, + "while parsing a flow mapping", context_mark, + "did not find expected ',' or '}'", token.start_mark) + } + } + + if token.typ == yaml_KEY_TOKEN { + skip_token(parser) + token = peek_token(parser) + if token == nil { + return false + } + if token.typ != yaml_VALUE_TOKEN && + token.typ != yaml_FLOW_ENTRY_TOKEN && + token.typ != yaml_FLOW_MAPPING_END_TOKEN { + parser.states = append(parser.states, yaml_PARSE_FLOW_MAPPING_VALUE_STATE) + return yaml_parser_parse_node(parser, event, false, false) + } else { + parser.state = yaml_PARSE_FLOW_MAPPING_VALUE_STATE + return yaml_parser_process_empty_scalar(parser, event, token.start_mark) + } + } else if token.typ != yaml_FLOW_MAPPING_END_TOKEN { + parser.states = append(parser.states, yaml_PARSE_FLOW_MAPPING_EMPTY_VALUE_STATE) + return yaml_parser_parse_node(parser, event, false, false) + } + } + + parser.state = parser.states[len(parser.states)-1] + parser.states = parser.states[:len(parser.states)-1] + parser.marks = parser.marks[:len(parser.marks)-1] + *event = yaml_event_t{ + typ: yaml_MAPPING_END_EVENT, + start_mark: token.start_mark, + end_mark: token.end_mark, + } + yaml_parser_set_event_comments(parser, event) + skip_token(parser) + return true +} + +// Parse the productions: +// flow_mapping_entry ::= flow_node | KEY flow_node? (VALUE flow_node?)? +// * ***** * +// +func yaml_parser_parse_flow_mapping_value(parser *yaml_parser_t, event *yaml_event_t, empty bool) bool { + token := peek_token(parser) + if token == nil { + return false + } + if empty { + parser.state = yaml_PARSE_FLOW_MAPPING_KEY_STATE + return yaml_parser_process_empty_scalar(parser, event, token.start_mark) + } + if token.typ == yaml_VALUE_TOKEN { + skip_token(parser) + token = peek_token(parser) + if token == nil { + return false + } + if token.typ != yaml_FLOW_ENTRY_TOKEN && token.typ != yaml_FLOW_MAPPING_END_TOKEN { + parser.states = append(parser.states, yaml_PARSE_FLOW_MAPPING_KEY_STATE) + return yaml_parser_parse_node(parser, event, false, false) + } + } + parser.state = yaml_PARSE_FLOW_MAPPING_KEY_STATE + return yaml_parser_process_empty_scalar(parser, event, token.start_mark) +} + +// Generate an empty scalar event. +func yaml_parser_process_empty_scalar(parser *yaml_parser_t, event *yaml_event_t, mark yaml_mark_t) bool { + *event = yaml_event_t{ + typ: yaml_SCALAR_EVENT, + start_mark: mark, + end_mark: mark, + value: nil, // Empty + implicit: true, + style: yaml_style_t(yaml_PLAIN_SCALAR_STYLE), + } + return true +} + +var default_tag_directives = []yaml_tag_directive_t{ + {[]byte("!"), []byte("!")}, + {[]byte("!!"), []byte("tag:yaml.org,2002:")}, +} + +// Parse directives. +func yaml_parser_process_directives(parser *yaml_parser_t, + version_directive_ref **yaml_version_directive_t, + tag_directives_ref *[]yaml_tag_directive_t) bool { + + var version_directive *yaml_version_directive_t + var tag_directives []yaml_tag_directive_t + + token := peek_token(parser) + if token == nil { + return false + } + + for token.typ == yaml_VERSION_DIRECTIVE_TOKEN || token.typ == yaml_TAG_DIRECTIVE_TOKEN { + if token.typ == yaml_VERSION_DIRECTIVE_TOKEN { + if version_directive != nil { + yaml_parser_set_parser_error(parser, + "found duplicate %YAML directive", token.start_mark) + return false + } + if token.major != 1 || token.minor != 1 { + yaml_parser_set_parser_error(parser, + "found incompatible YAML document", token.start_mark) + return false + } + version_directive = &yaml_version_directive_t{ + major: token.major, + minor: token.minor, + } + } else if token.typ == yaml_TAG_DIRECTIVE_TOKEN { + value := yaml_tag_directive_t{ + handle: token.value, + prefix: token.prefix, + } + if !yaml_parser_append_tag_directive(parser, value, false, token.start_mark) { + return false + } + tag_directives = append(tag_directives, value) + } + + skip_token(parser) + token = peek_token(parser) + if token == nil { + return false + } + } + + for i := range default_tag_directives { + if !yaml_parser_append_tag_directive(parser, default_tag_directives[i], true, token.start_mark) { + return false + } + } + + if version_directive_ref != nil { + *version_directive_ref = version_directive + } + if tag_directives_ref != nil { + *tag_directives_ref = tag_directives + } + return true +} + +// Append a tag directive to the directives stack. +func yaml_parser_append_tag_directive(parser *yaml_parser_t, value yaml_tag_directive_t, allow_duplicates bool, mark yaml_mark_t) bool { + for i := range parser.tag_directives { + if bytes.Equal(value.handle, parser.tag_directives[i].handle) { + if allow_duplicates { + return true + } + return yaml_parser_set_parser_error(parser, "found duplicate %TAG directive", mark) + } + } + + // [Go] I suspect the copy is unnecessary. This was likely done + // because there was no way to track ownership of the data. + value_copy := yaml_tag_directive_t{ + handle: make([]byte, len(value.handle)), + prefix: make([]byte, len(value.prefix)), + } + copy(value_copy.handle, value.handle) + copy(value_copy.prefix, value.prefix) + parser.tag_directives = append(parser.tag_directives, value_copy) + return true +} diff --git a/vendor/gopkg.in/yaml.v3/readerc.go b/vendor/gopkg.in/yaml.v3/readerc.go new file mode 100644 index 00000000..b7de0a89 --- /dev/null +++ b/vendor/gopkg.in/yaml.v3/readerc.go @@ -0,0 +1,434 @@ +// +// Copyright (c) 2011-2019 Canonical Ltd +// Copyright (c) 2006-2010 Kirill Simonov +// +// Permission is hereby granted, free of charge, to any person obtaining a copy of +// this software and associated documentation files (the "Software"), to deal in +// the Software without restriction, including without limitation the rights to +// use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies +// of the Software, and to permit persons to whom the Software is furnished to do +// so, subject to the following conditions: +// +// The above copyright notice and this permission notice shall be included in all +// copies or substantial portions of the Software. +// +// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE +// SOFTWARE. + +package yaml + +import ( + "io" +) + +// Set the reader error and return 0. +func yaml_parser_set_reader_error(parser *yaml_parser_t, problem string, offset int, value int) bool { + parser.error = yaml_READER_ERROR + parser.problem = problem + parser.problem_offset = offset + parser.problem_value = value + return false +} + +// Byte order marks. +const ( + bom_UTF8 = "\xef\xbb\xbf" + bom_UTF16LE = "\xff\xfe" + bom_UTF16BE = "\xfe\xff" +) + +// Determine the input stream encoding by checking the BOM symbol. If no BOM is +// found, the UTF-8 encoding is assumed. Return 1 on success, 0 on failure. +func yaml_parser_determine_encoding(parser *yaml_parser_t) bool { + // Ensure that we had enough bytes in the raw buffer. + for !parser.eof && len(parser.raw_buffer)-parser.raw_buffer_pos < 3 { + if !yaml_parser_update_raw_buffer(parser) { + return false + } + } + + // Determine the encoding. + buf := parser.raw_buffer + pos := parser.raw_buffer_pos + avail := len(buf) - pos + if avail >= 2 && buf[pos] == bom_UTF16LE[0] && buf[pos+1] == bom_UTF16LE[1] { + parser.encoding = yaml_UTF16LE_ENCODING + parser.raw_buffer_pos += 2 + parser.offset += 2 + } else if avail >= 2 && buf[pos] == bom_UTF16BE[0] && buf[pos+1] == bom_UTF16BE[1] { + parser.encoding = yaml_UTF16BE_ENCODING + parser.raw_buffer_pos += 2 + parser.offset += 2 + } else if avail >= 3 && buf[pos] == bom_UTF8[0] && buf[pos+1] == bom_UTF8[1] && buf[pos+2] == bom_UTF8[2] { + parser.encoding = yaml_UTF8_ENCODING + parser.raw_buffer_pos += 3 + parser.offset += 3 + } else { + parser.encoding = yaml_UTF8_ENCODING + } + return true +} + +// Update the raw buffer. +func yaml_parser_update_raw_buffer(parser *yaml_parser_t) bool { + size_read := 0 + + // Return if the raw buffer is full. + if parser.raw_buffer_pos == 0 && len(parser.raw_buffer) == cap(parser.raw_buffer) { + return true + } + + // Return on EOF. + if parser.eof { + return true + } + + // Move the remaining bytes in the raw buffer to the beginning. + if parser.raw_buffer_pos > 0 && parser.raw_buffer_pos < len(parser.raw_buffer) { + copy(parser.raw_buffer, parser.raw_buffer[parser.raw_buffer_pos:]) + } + parser.raw_buffer = parser.raw_buffer[:len(parser.raw_buffer)-parser.raw_buffer_pos] + parser.raw_buffer_pos = 0 + + // Call the read handler to fill the buffer. + size_read, err := parser.read_handler(parser, parser.raw_buffer[len(parser.raw_buffer):cap(parser.raw_buffer)]) + parser.raw_buffer = parser.raw_buffer[:len(parser.raw_buffer)+size_read] + if err == io.EOF { + parser.eof = true + } else if err != nil { + return yaml_parser_set_reader_error(parser, "input error: "+err.Error(), parser.offset, -1) + } + return true +} + +// Ensure that the buffer contains at least `length` characters. +// Return true on success, false on failure. +// +// The length is supposed to be significantly less that the buffer size. +func yaml_parser_update_buffer(parser *yaml_parser_t, length int) bool { + if parser.read_handler == nil { + panic("read handler must be set") + } + + // [Go] This function was changed to guarantee the requested length size at EOF. + // The fact we need to do this is pretty awful, but the description above implies + // for that to be the case, and there are tests + + // If the EOF flag is set and the raw buffer is empty, do nothing. + if parser.eof && parser.raw_buffer_pos == len(parser.raw_buffer) { + // [Go] ACTUALLY! Read the documentation of this function above. + // This is just broken. To return true, we need to have the + // given length in the buffer. Not doing that means every single + // check that calls this function to make sure the buffer has a + // given length is Go) panicking; or C) accessing invalid memory. + //return true + } + + // Return if the buffer contains enough characters. + if parser.unread >= length { + return true + } + + // Determine the input encoding if it is not known yet. + if parser.encoding == yaml_ANY_ENCODING { + if !yaml_parser_determine_encoding(parser) { + return false + } + } + + // Move the unread characters to the beginning of the buffer. + buffer_len := len(parser.buffer) + if parser.buffer_pos > 0 && parser.buffer_pos < buffer_len { + copy(parser.buffer, parser.buffer[parser.buffer_pos:]) + buffer_len -= parser.buffer_pos + parser.buffer_pos = 0 + } else if parser.buffer_pos == buffer_len { + buffer_len = 0 + parser.buffer_pos = 0 + } + + // Open the whole buffer for writing, and cut it before returning. + parser.buffer = parser.buffer[:cap(parser.buffer)] + + // Fill the buffer until it has enough characters. + first := true + for parser.unread < length { + + // Fill the raw buffer if necessary. + if !first || parser.raw_buffer_pos == len(parser.raw_buffer) { + if !yaml_parser_update_raw_buffer(parser) { + parser.buffer = parser.buffer[:buffer_len] + return false + } + } + first = false + + // Decode the raw buffer. + inner: + for parser.raw_buffer_pos != len(parser.raw_buffer) { + var value rune + var width int + + raw_unread := len(parser.raw_buffer) - parser.raw_buffer_pos + + // Decode the next character. + switch parser.encoding { + case yaml_UTF8_ENCODING: + // Decode a UTF-8 character. Check RFC 3629 + // (http://www.ietf.org/rfc/rfc3629.txt) for more details. + // + // The following table (taken from the RFC) is used for + // decoding. + // + // Char. number range | UTF-8 octet sequence + // (hexadecimal) | (binary) + // --------------------+------------------------------------ + // 0000 0000-0000 007F | 0xxxxxxx + // 0000 0080-0000 07FF | 110xxxxx 10xxxxxx + // 0000 0800-0000 FFFF | 1110xxxx 10xxxxxx 10xxxxxx + // 0001 0000-0010 FFFF | 11110xxx 10xxxxxx 10xxxxxx 10xxxxxx + // + // Additionally, the characters in the range 0xD800-0xDFFF + // are prohibited as they are reserved for use with UTF-16 + // surrogate pairs. + + // Determine the length of the UTF-8 sequence. + octet := parser.raw_buffer[parser.raw_buffer_pos] + switch { + case octet&0x80 == 0x00: + width = 1 + case octet&0xE0 == 0xC0: + width = 2 + case octet&0xF0 == 0xE0: + width = 3 + case octet&0xF8 == 0xF0: + width = 4 + default: + // The leading octet is invalid. + return yaml_parser_set_reader_error(parser, + "invalid leading UTF-8 octet", + parser.offset, int(octet)) + } + + // Check if the raw buffer contains an incomplete character. + if width > raw_unread { + if parser.eof { + return yaml_parser_set_reader_error(parser, + "incomplete UTF-8 octet sequence", + parser.offset, -1) + } + break inner + } + + // Decode the leading octet. + switch { + case octet&0x80 == 0x00: + value = rune(octet & 0x7F) + case octet&0xE0 == 0xC0: + value = rune(octet & 0x1F) + case octet&0xF0 == 0xE0: + value = rune(octet & 0x0F) + case octet&0xF8 == 0xF0: + value = rune(octet & 0x07) + default: + value = 0 + } + + // Check and decode the trailing octets. + for k := 1; k < width; k++ { + octet = parser.raw_buffer[parser.raw_buffer_pos+k] + + // Check if the octet is valid. + if (octet & 0xC0) != 0x80 { + return yaml_parser_set_reader_error(parser, + "invalid trailing UTF-8 octet", + parser.offset+k, int(octet)) + } + + // Decode the octet. + value = (value << 6) + rune(octet&0x3F) + } + + // Check the length of the sequence against the value. + switch { + case width == 1: + case width == 2 && value >= 0x80: + case width == 3 && value >= 0x800: + case width == 4 && value >= 0x10000: + default: + return yaml_parser_set_reader_error(parser, + "invalid length of a UTF-8 sequence", + parser.offset, -1) + } + + // Check the range of the value. + if value >= 0xD800 && value <= 0xDFFF || value > 0x10FFFF { + return yaml_parser_set_reader_error(parser, + "invalid Unicode character", + parser.offset, int(value)) + } + + case yaml_UTF16LE_ENCODING, yaml_UTF16BE_ENCODING: + var low, high int + if parser.encoding == yaml_UTF16LE_ENCODING { + low, high = 0, 1 + } else { + low, high = 1, 0 + } + + // The UTF-16 encoding is not as simple as one might + // naively think. Check RFC 2781 + // (http://www.ietf.org/rfc/rfc2781.txt). + // + // Normally, two subsequent bytes describe a Unicode + // character. However a special technique (called a + // surrogate pair) is used for specifying character + // values larger than 0xFFFF. + // + // A surrogate pair consists of two pseudo-characters: + // high surrogate area (0xD800-0xDBFF) + // low surrogate area (0xDC00-0xDFFF) + // + // The following formulas are used for decoding + // and encoding characters using surrogate pairs: + // + // U = U' + 0x10000 (0x01 00 00 <= U <= 0x10 FF FF) + // U' = yyyyyyyyyyxxxxxxxxxx (0 <= U' <= 0x0F FF FF) + // W1 = 110110yyyyyyyyyy + // W2 = 110111xxxxxxxxxx + // + // where U is the character value, W1 is the high surrogate + // area, W2 is the low surrogate area. + + // Check for incomplete UTF-16 character. + if raw_unread < 2 { + if parser.eof { + return yaml_parser_set_reader_error(parser, + "incomplete UTF-16 character", + parser.offset, -1) + } + break inner + } + + // Get the character. + value = rune(parser.raw_buffer[parser.raw_buffer_pos+low]) + + (rune(parser.raw_buffer[parser.raw_buffer_pos+high]) << 8) + + // Check for unexpected low surrogate area. + if value&0xFC00 == 0xDC00 { + return yaml_parser_set_reader_error(parser, + "unexpected low surrogate area", + parser.offset, int(value)) + } + + // Check for a high surrogate area. + if value&0xFC00 == 0xD800 { + width = 4 + + // Check for incomplete surrogate pair. + if raw_unread < 4 { + if parser.eof { + return yaml_parser_set_reader_error(parser, + "incomplete UTF-16 surrogate pair", + parser.offset, -1) + } + break inner + } + + // Get the next character. + value2 := rune(parser.raw_buffer[parser.raw_buffer_pos+low+2]) + + (rune(parser.raw_buffer[parser.raw_buffer_pos+high+2]) << 8) + + // Check for a low surrogate area. + if value2&0xFC00 != 0xDC00 { + return yaml_parser_set_reader_error(parser, + "expected low surrogate area", + parser.offset+2, int(value2)) + } + + // Generate the value of the surrogate pair. + value = 0x10000 + ((value & 0x3FF) << 10) + (value2 & 0x3FF) + } else { + width = 2 + } + + default: + panic("impossible") + } + + // Check if the character is in the allowed range: + // #x9 | #xA | #xD | [#x20-#x7E] (8 bit) + // | #x85 | [#xA0-#xD7FF] | [#xE000-#xFFFD] (16 bit) + // | [#x10000-#x10FFFF] (32 bit) + switch { + case value == 0x09: + case value == 0x0A: + case value == 0x0D: + case value >= 0x20 && value <= 0x7E: + case value == 0x85: + case value >= 0xA0 && value <= 0xD7FF: + case value >= 0xE000 && value <= 0xFFFD: + case value >= 0x10000 && value <= 0x10FFFF: + default: + return yaml_parser_set_reader_error(parser, + "control characters are not allowed", + parser.offset, int(value)) + } + + // Move the raw pointers. + parser.raw_buffer_pos += width + parser.offset += width + + // Finally put the character into the buffer. + if value <= 0x7F { + // 0000 0000-0000 007F . 0xxxxxxx + parser.buffer[buffer_len+0] = byte(value) + buffer_len += 1 + } else if value <= 0x7FF { + // 0000 0080-0000 07FF . 110xxxxx 10xxxxxx + parser.buffer[buffer_len+0] = byte(0xC0 + (value >> 6)) + parser.buffer[buffer_len+1] = byte(0x80 + (value & 0x3F)) + buffer_len += 2 + } else if value <= 0xFFFF { + // 0000 0800-0000 FFFF . 1110xxxx 10xxxxxx 10xxxxxx + parser.buffer[buffer_len+0] = byte(0xE0 + (value >> 12)) + parser.buffer[buffer_len+1] = byte(0x80 + ((value >> 6) & 0x3F)) + parser.buffer[buffer_len+2] = byte(0x80 + (value & 0x3F)) + buffer_len += 3 + } else { + // 0001 0000-0010 FFFF . 11110xxx 10xxxxxx 10xxxxxx 10xxxxxx + parser.buffer[buffer_len+0] = byte(0xF0 + (value >> 18)) + parser.buffer[buffer_len+1] = byte(0x80 + ((value >> 12) & 0x3F)) + parser.buffer[buffer_len+2] = byte(0x80 + ((value >> 6) & 0x3F)) + parser.buffer[buffer_len+3] = byte(0x80 + (value & 0x3F)) + buffer_len += 4 + } + + parser.unread++ + } + + // On EOF, put NUL into the buffer and return. + if parser.eof { + parser.buffer[buffer_len] = 0 + buffer_len++ + parser.unread++ + break + } + } + // [Go] Read the documentation of this function above. To return true, + // we need to have the given length in the buffer. Not doing that means + // every single check that calls this function to make sure the buffer + // has a given length is Go) panicking; or C) accessing invalid memory. + // This happens here due to the EOF above breaking early. + for buffer_len < length { + parser.buffer[buffer_len] = 0 + buffer_len++ + } + parser.buffer = parser.buffer[:buffer_len] + return true +} diff --git a/vendor/gopkg.in/yaml.v3/resolve.go b/vendor/gopkg.in/yaml.v3/resolve.go new file mode 100644 index 00000000..64ae8880 --- /dev/null +++ b/vendor/gopkg.in/yaml.v3/resolve.go @@ -0,0 +1,326 @@ +// +// Copyright (c) 2011-2019 Canonical Ltd +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. + +package yaml + +import ( + "encoding/base64" + "math" + "regexp" + "strconv" + "strings" + "time" +) + +type resolveMapItem struct { + value interface{} + tag string +} + +var resolveTable = make([]byte, 256) +var resolveMap = make(map[string]resolveMapItem) + +func init() { + t := resolveTable + t[int('+')] = 'S' // Sign + t[int('-')] = 'S' + for _, c := range "0123456789" { + t[int(c)] = 'D' // Digit + } + for _, c := range "yYnNtTfFoO~" { + t[int(c)] = 'M' // In map + } + t[int('.')] = '.' // Float (potentially in map) + + var resolveMapList = []struct { + v interface{} + tag string + l []string + }{ + {true, boolTag, []string{"true", "True", "TRUE"}}, + {false, boolTag, []string{"false", "False", "FALSE"}}, + {nil, nullTag, []string{"", "~", "null", "Null", "NULL"}}, + {math.NaN(), floatTag, []string{".nan", ".NaN", ".NAN"}}, + {math.Inf(+1), floatTag, []string{".inf", ".Inf", ".INF"}}, + {math.Inf(+1), floatTag, []string{"+.inf", "+.Inf", "+.INF"}}, + {math.Inf(-1), floatTag, []string{"-.inf", "-.Inf", "-.INF"}}, + {"<<", mergeTag, []string{"<<"}}, + } + + m := resolveMap + for _, item := range resolveMapList { + for _, s := range item.l { + m[s] = resolveMapItem{item.v, item.tag} + } + } +} + +const ( + nullTag = "!!null" + boolTag = "!!bool" + strTag = "!!str" + intTag = "!!int" + floatTag = "!!float" + timestampTag = "!!timestamp" + seqTag = "!!seq" + mapTag = "!!map" + binaryTag = "!!binary" + mergeTag = "!!merge" +) + +var longTags = make(map[string]string) +var shortTags = make(map[string]string) + +func init() { + for _, stag := range []string{nullTag, boolTag, strTag, intTag, floatTag, timestampTag, seqTag, mapTag, binaryTag, mergeTag} { + ltag := longTag(stag) + longTags[stag] = ltag + shortTags[ltag] = stag + } +} + +const longTagPrefix = "tag:yaml.org,2002:" + +func shortTag(tag string) string { + if strings.HasPrefix(tag, longTagPrefix) { + if stag, ok := shortTags[tag]; ok { + return stag + } + return "!!" + tag[len(longTagPrefix):] + } + return tag +} + +func longTag(tag string) string { + if strings.HasPrefix(tag, "!!") { + if ltag, ok := longTags[tag]; ok { + return ltag + } + return longTagPrefix + tag[2:] + } + return tag +} + +func resolvableTag(tag string) bool { + switch tag { + case "", strTag, boolTag, intTag, floatTag, nullTag, timestampTag: + return true + } + return false +} + +var yamlStyleFloat = regexp.MustCompile(`^[-+]?(\.[0-9]+|[0-9]+(\.[0-9]*)?)([eE][-+]?[0-9]+)?$`) + +func resolve(tag string, in string) (rtag string, out interface{}) { + tag = shortTag(tag) + if !resolvableTag(tag) { + return tag, in + } + + defer func() { + switch tag { + case "", rtag, strTag, binaryTag: + return + case floatTag: + if rtag == intTag { + switch v := out.(type) { + case int64: + rtag = floatTag + out = float64(v) + return + case int: + rtag = floatTag + out = float64(v) + return + } + } + } + failf("cannot decode %s `%s` as a %s", shortTag(rtag), in, shortTag(tag)) + }() + + // Any data is accepted as a !!str or !!binary. + // Otherwise, the prefix is enough of a hint about what it might be. + hint := byte('N') + if in != "" { + hint = resolveTable[in[0]] + } + if hint != 0 && tag != strTag && tag != binaryTag { + // Handle things we can lookup in a map. + if item, ok := resolveMap[in]; ok { + return item.tag, item.value + } + + // Base 60 floats are a bad idea, were dropped in YAML 1.2, and + // are purposefully unsupported here. They're still quoted on + // the way out for compatibility with other parser, though. + + switch hint { + case 'M': + // We've already checked the map above. + + case '.': + // Not in the map, so maybe a normal float. + floatv, err := strconv.ParseFloat(in, 64) + if err == nil { + return floatTag, floatv + } + + case 'D', 'S': + // Int, float, or timestamp. + // Only try values as a timestamp if the value is unquoted or there's an explicit + // !!timestamp tag. + if tag == "" || tag == timestampTag { + t, ok := parseTimestamp(in) + if ok { + return timestampTag, t + } + } + + plain := strings.Replace(in, "_", "", -1) + intv, err := strconv.ParseInt(plain, 0, 64) + if err == nil { + if intv == int64(int(intv)) { + return intTag, int(intv) + } else { + return intTag, intv + } + } + uintv, err := strconv.ParseUint(plain, 0, 64) + if err == nil { + return intTag, uintv + } + if yamlStyleFloat.MatchString(plain) { + floatv, err := strconv.ParseFloat(plain, 64) + if err == nil { + return floatTag, floatv + } + } + if strings.HasPrefix(plain, "0b") { + intv, err := strconv.ParseInt(plain[2:], 2, 64) + if err == nil { + if intv == int64(int(intv)) { + return intTag, int(intv) + } else { + return intTag, intv + } + } + uintv, err := strconv.ParseUint(plain[2:], 2, 64) + if err == nil { + return intTag, uintv + } + } else if strings.HasPrefix(plain, "-0b") { + intv, err := strconv.ParseInt("-"+plain[3:], 2, 64) + if err == nil { + if true || intv == int64(int(intv)) { + return intTag, int(intv) + } else { + return intTag, intv + } + } + } + // Octals as introduced in version 1.2 of the spec. + // Octals from the 1.1 spec, spelled as 0777, are still + // decoded by default in v3 as well for compatibility. + // May be dropped in v4 depending on how usage evolves. + if strings.HasPrefix(plain, "0o") { + intv, err := strconv.ParseInt(plain[2:], 8, 64) + if err == nil { + if intv == int64(int(intv)) { + return intTag, int(intv) + } else { + return intTag, intv + } + } + uintv, err := strconv.ParseUint(plain[2:], 8, 64) + if err == nil { + return intTag, uintv + } + } else if strings.HasPrefix(plain, "-0o") { + intv, err := strconv.ParseInt("-"+plain[3:], 8, 64) + if err == nil { + if true || intv == int64(int(intv)) { + return intTag, int(intv) + } else { + return intTag, intv + } + } + } + default: + panic("internal error: missing handler for resolver table: " + string(rune(hint)) + " (with " + in + ")") + } + } + return strTag, in +} + +// encodeBase64 encodes s as base64 that is broken up into multiple lines +// as appropriate for the resulting length. +func encodeBase64(s string) string { + const lineLen = 70 + encLen := base64.StdEncoding.EncodedLen(len(s)) + lines := encLen/lineLen + 1 + buf := make([]byte, encLen*2+lines) + in := buf[0:encLen] + out := buf[encLen:] + base64.StdEncoding.Encode(in, []byte(s)) + k := 0 + for i := 0; i < len(in); i += lineLen { + j := i + lineLen + if j > len(in) { + j = len(in) + } + k += copy(out[k:], in[i:j]) + if lines > 1 { + out[k] = '\n' + k++ + } + } + return string(out[:k]) +} + +// This is a subset of the formats allowed by the regular expression +// defined at http://yaml.org/type/timestamp.html. +var allowedTimestampFormats = []string{ + "2006-1-2T15:4:5.999999999Z07:00", // RCF3339Nano with short date fields. + "2006-1-2t15:4:5.999999999Z07:00", // RFC3339Nano with short date fields and lower-case "t". + "2006-1-2 15:4:5.999999999", // space separated with no time zone + "2006-1-2", // date only + // Notable exception: time.Parse cannot handle: "2001-12-14 21:59:43.10 -5" + // from the set of examples. +} + +// parseTimestamp parses s as a timestamp string and +// returns the timestamp and reports whether it succeeded. +// Timestamp formats are defined at http://yaml.org/type/timestamp.html +func parseTimestamp(s string) (time.Time, bool) { + // TODO write code to check all the formats supported by + // http://yaml.org/type/timestamp.html instead of using time.Parse. + + // Quick check: all date formats start with YYYY-. + i := 0 + for ; i < len(s); i++ { + if c := s[i]; c < '0' || c > '9' { + break + } + } + if i != 4 || i == len(s) || s[i] != '-' { + return time.Time{}, false + } + for _, format := range allowedTimestampFormats { + if t, err := time.Parse(format, s); err == nil { + return t, true + } + } + return time.Time{}, false +} diff --git a/vendor/gopkg.in/yaml.v3/scannerc.go b/vendor/gopkg.in/yaml.v3/scannerc.go new file mode 100644 index 00000000..ca007010 --- /dev/null +++ b/vendor/gopkg.in/yaml.v3/scannerc.go @@ -0,0 +1,3038 @@ +// +// Copyright (c) 2011-2019 Canonical Ltd +// Copyright (c) 2006-2010 Kirill Simonov +// +// Permission is hereby granted, free of charge, to any person obtaining a copy of +// this software and associated documentation files (the "Software"), to deal in +// the Software without restriction, including without limitation the rights to +// use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies +// of the Software, and to permit persons to whom the Software is furnished to do +// so, subject to the following conditions: +// +// The above copyright notice and this permission notice shall be included in all +// copies or substantial portions of the Software. +// +// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE +// SOFTWARE. + +package yaml + +import ( + "bytes" + "fmt" +) + +// Introduction +// ************ +// +// The following notes assume that you are familiar with the YAML specification +// (http://yaml.org/spec/1.2/spec.html). We mostly follow it, although in +// some cases we are less restrictive that it requires. +// +// The process of transforming a YAML stream into a sequence of events is +// divided on two steps: Scanning and Parsing. +// +// The Scanner transforms the input stream into a sequence of tokens, while the +// parser transform the sequence of tokens produced by the Scanner into a +// sequence of parsing events. +// +// The Scanner is rather clever and complicated. The Parser, on the contrary, +// is a straightforward implementation of a recursive-descendant parser (or, +// LL(1) parser, as it is usually called). +// +// Actually there are two issues of Scanning that might be called "clever", the +// rest is quite straightforward. The issues are "block collection start" and +// "simple keys". Both issues are explained below in details. +// +// Here the Scanning step is explained and implemented. We start with the list +// of all the tokens produced by the Scanner together with short descriptions. +// +// Now, tokens: +// +// STREAM-START(encoding) # The stream start. +// STREAM-END # The stream end. +// VERSION-DIRECTIVE(major,minor) # The '%YAML' directive. +// TAG-DIRECTIVE(handle,prefix) # The '%TAG' directive. +// DOCUMENT-START # '---' +// DOCUMENT-END # '...' +// BLOCK-SEQUENCE-START # Indentation increase denoting a block +// BLOCK-MAPPING-START # sequence or a block mapping. +// BLOCK-END # Indentation decrease. +// FLOW-SEQUENCE-START # '[' +// FLOW-SEQUENCE-END # ']' +// BLOCK-SEQUENCE-START # '{' +// BLOCK-SEQUENCE-END # '}' +// BLOCK-ENTRY # '-' +// FLOW-ENTRY # ',' +// KEY # '?' or nothing (simple keys). +// VALUE # ':' +// ALIAS(anchor) # '*anchor' +// ANCHOR(anchor) # '&anchor' +// TAG(handle,suffix) # '!handle!suffix' +// SCALAR(value,style) # A scalar. +// +// The following two tokens are "virtual" tokens denoting the beginning and the +// end of the stream: +// +// STREAM-START(encoding) +// STREAM-END +// +// We pass the information about the input stream encoding with the +// STREAM-START token. +// +// The next two tokens are responsible for tags: +// +// VERSION-DIRECTIVE(major,minor) +// TAG-DIRECTIVE(handle,prefix) +// +// Example: +// +// %YAML 1.1 +// %TAG ! !foo +// %TAG !yaml! tag:yaml.org,2002: +// --- +// +// The correspoding sequence of tokens: +// +// STREAM-START(utf-8) +// VERSION-DIRECTIVE(1,1) +// TAG-DIRECTIVE("!","!foo") +// TAG-DIRECTIVE("!yaml","tag:yaml.org,2002:") +// DOCUMENT-START +// STREAM-END +// +// Note that the VERSION-DIRECTIVE and TAG-DIRECTIVE tokens occupy a whole +// line. +// +// The document start and end indicators are represented by: +// +// DOCUMENT-START +// DOCUMENT-END +// +// Note that if a YAML stream contains an implicit document (without '---' +// and '...' indicators), no DOCUMENT-START and DOCUMENT-END tokens will be +// produced. +// +// In the following examples, we present whole documents together with the +// produced tokens. +// +// 1. An implicit document: +// +// 'a scalar' +// +// Tokens: +// +// STREAM-START(utf-8) +// SCALAR("a scalar",single-quoted) +// STREAM-END +// +// 2. An explicit document: +// +// --- +// 'a scalar' +// ... +// +// Tokens: +// +// STREAM-START(utf-8) +// DOCUMENT-START +// SCALAR("a scalar",single-quoted) +// DOCUMENT-END +// STREAM-END +// +// 3. Several documents in a stream: +// +// 'a scalar' +// --- +// 'another scalar' +// --- +// 'yet another scalar' +// +// Tokens: +// +// STREAM-START(utf-8) +// SCALAR("a scalar",single-quoted) +// DOCUMENT-START +// SCALAR("another scalar",single-quoted) +// DOCUMENT-START +// SCALAR("yet another scalar",single-quoted) +// STREAM-END +// +// We have already introduced the SCALAR token above. The following tokens are +// used to describe aliases, anchors, tag, and scalars: +// +// ALIAS(anchor) +// ANCHOR(anchor) +// TAG(handle,suffix) +// SCALAR(value,style) +// +// The following series of examples illustrate the usage of these tokens: +// +// 1. A recursive sequence: +// +// &A [ *A ] +// +// Tokens: +// +// STREAM-START(utf-8) +// ANCHOR("A") +// FLOW-SEQUENCE-START +// ALIAS("A") +// FLOW-SEQUENCE-END +// STREAM-END +// +// 2. A tagged scalar: +// +// !!float "3.14" # A good approximation. +// +// Tokens: +// +// STREAM-START(utf-8) +// TAG("!!","float") +// SCALAR("3.14",double-quoted) +// STREAM-END +// +// 3. Various scalar styles: +// +// --- # Implicit empty plain scalars do not produce tokens. +// --- a plain scalar +// --- 'a single-quoted scalar' +// --- "a double-quoted scalar" +// --- |- +// a literal scalar +// --- >- +// a folded +// scalar +// +// Tokens: +// +// STREAM-START(utf-8) +// DOCUMENT-START +// DOCUMENT-START +// SCALAR("a plain scalar",plain) +// DOCUMENT-START +// SCALAR("a single-quoted scalar",single-quoted) +// DOCUMENT-START +// SCALAR("a double-quoted scalar",double-quoted) +// DOCUMENT-START +// SCALAR("a literal scalar",literal) +// DOCUMENT-START +// SCALAR("a folded scalar",folded) +// STREAM-END +// +// Now it's time to review collection-related tokens. We will start with +// flow collections: +// +// FLOW-SEQUENCE-START +// FLOW-SEQUENCE-END +// FLOW-MAPPING-START +// FLOW-MAPPING-END +// FLOW-ENTRY +// KEY +// VALUE +// +// The tokens FLOW-SEQUENCE-START, FLOW-SEQUENCE-END, FLOW-MAPPING-START, and +// FLOW-MAPPING-END represent the indicators '[', ']', '{', and '}' +// correspondingly. FLOW-ENTRY represent the ',' indicator. Finally the +// indicators '?' and ':', which are used for denoting mapping keys and values, +// are represented by the KEY and VALUE tokens. +// +// The following examples show flow collections: +// +// 1. A flow sequence: +// +// [item 1, item 2, item 3] +// +// Tokens: +// +// STREAM-START(utf-8) +// FLOW-SEQUENCE-START +// SCALAR("item 1",plain) +// FLOW-ENTRY +// SCALAR("item 2",plain) +// FLOW-ENTRY +// SCALAR("item 3",plain) +// FLOW-SEQUENCE-END +// STREAM-END +// +// 2. A flow mapping: +// +// { +// a simple key: a value, # Note that the KEY token is produced. +// ? a complex key: another value, +// } +// +// Tokens: +// +// STREAM-START(utf-8) +// FLOW-MAPPING-START +// KEY +// SCALAR("a simple key",plain) +// VALUE +// SCALAR("a value",plain) +// FLOW-ENTRY +// KEY +// SCALAR("a complex key",plain) +// VALUE +// SCALAR("another value",plain) +// FLOW-ENTRY +// FLOW-MAPPING-END +// STREAM-END +// +// A simple key is a key which is not denoted by the '?' indicator. Note that +// the Scanner still produce the KEY token whenever it encounters a simple key. +// +// For scanning block collections, the following tokens are used (note that we +// repeat KEY and VALUE here): +// +// BLOCK-SEQUENCE-START +// BLOCK-MAPPING-START +// BLOCK-END +// BLOCK-ENTRY +// KEY +// VALUE +// +// The tokens BLOCK-SEQUENCE-START and BLOCK-MAPPING-START denote indentation +// increase that precedes a block collection (cf. the INDENT token in Python). +// The token BLOCK-END denote indentation decrease that ends a block collection +// (cf. the DEDENT token in Python). However YAML has some syntax pecularities +// that makes detections of these tokens more complex. +// +// The tokens BLOCK-ENTRY, KEY, and VALUE are used to represent the indicators +// '-', '?', and ':' correspondingly. +// +// The following examples show how the tokens BLOCK-SEQUENCE-START, +// BLOCK-MAPPING-START, and BLOCK-END are emitted by the Scanner: +// +// 1. Block sequences: +// +// - item 1 +// - item 2 +// - +// - item 3.1 +// - item 3.2 +// - +// key 1: value 1 +// key 2: value 2 +// +// Tokens: +// +// STREAM-START(utf-8) +// BLOCK-SEQUENCE-START +// BLOCK-ENTRY +// SCALAR("item 1",plain) +// BLOCK-ENTRY +// SCALAR("item 2",plain) +// BLOCK-ENTRY +// BLOCK-SEQUENCE-START +// BLOCK-ENTRY +// SCALAR("item 3.1",plain) +// BLOCK-ENTRY +// SCALAR("item 3.2",plain) +// BLOCK-END +// BLOCK-ENTRY +// BLOCK-MAPPING-START +// KEY +// SCALAR("key 1",plain) +// VALUE +// SCALAR("value 1",plain) +// KEY +// SCALAR("key 2",plain) +// VALUE +// SCALAR("value 2",plain) +// BLOCK-END +// BLOCK-END +// STREAM-END +// +// 2. Block mappings: +// +// a simple key: a value # The KEY token is produced here. +// ? a complex key +// : another value +// a mapping: +// key 1: value 1 +// key 2: value 2 +// a sequence: +// - item 1 +// - item 2 +// +// Tokens: +// +// STREAM-START(utf-8) +// BLOCK-MAPPING-START +// KEY +// SCALAR("a simple key",plain) +// VALUE +// SCALAR("a value",plain) +// KEY +// SCALAR("a complex key",plain) +// VALUE +// SCALAR("another value",plain) +// KEY +// SCALAR("a mapping",plain) +// BLOCK-MAPPING-START +// KEY +// SCALAR("key 1",plain) +// VALUE +// SCALAR("value 1",plain) +// KEY +// SCALAR("key 2",plain) +// VALUE +// SCALAR("value 2",plain) +// BLOCK-END +// KEY +// SCALAR("a sequence",plain) +// VALUE +// BLOCK-SEQUENCE-START +// BLOCK-ENTRY +// SCALAR("item 1",plain) +// BLOCK-ENTRY +// SCALAR("item 2",plain) +// BLOCK-END +// BLOCK-END +// STREAM-END +// +// YAML does not always require to start a new block collection from a new +// line. If the current line contains only '-', '?', and ':' indicators, a new +// block collection may start at the current line. The following examples +// illustrate this case: +// +// 1. Collections in a sequence: +// +// - - item 1 +// - item 2 +// - key 1: value 1 +// key 2: value 2 +// - ? complex key +// : complex value +// +// Tokens: +// +// STREAM-START(utf-8) +// BLOCK-SEQUENCE-START +// BLOCK-ENTRY +// BLOCK-SEQUENCE-START +// BLOCK-ENTRY +// SCALAR("item 1",plain) +// BLOCK-ENTRY +// SCALAR("item 2",plain) +// BLOCK-END +// BLOCK-ENTRY +// BLOCK-MAPPING-START +// KEY +// SCALAR("key 1",plain) +// VALUE +// SCALAR("value 1",plain) +// KEY +// SCALAR("key 2",plain) +// VALUE +// SCALAR("value 2",plain) +// BLOCK-END +// BLOCK-ENTRY +// BLOCK-MAPPING-START +// KEY +// SCALAR("complex key") +// VALUE +// SCALAR("complex value") +// BLOCK-END +// BLOCK-END +// STREAM-END +// +// 2. Collections in a mapping: +// +// ? a sequence +// : - item 1 +// - item 2 +// ? a mapping +// : key 1: value 1 +// key 2: value 2 +// +// Tokens: +// +// STREAM-START(utf-8) +// BLOCK-MAPPING-START +// KEY +// SCALAR("a sequence",plain) +// VALUE +// BLOCK-SEQUENCE-START +// BLOCK-ENTRY +// SCALAR("item 1",plain) +// BLOCK-ENTRY +// SCALAR("item 2",plain) +// BLOCK-END +// KEY +// SCALAR("a mapping",plain) +// VALUE +// BLOCK-MAPPING-START +// KEY +// SCALAR("key 1",plain) +// VALUE +// SCALAR("value 1",plain) +// KEY +// SCALAR("key 2",plain) +// VALUE +// SCALAR("value 2",plain) +// BLOCK-END +// BLOCK-END +// STREAM-END +// +// YAML also permits non-indented sequences if they are included into a block +// mapping. In this case, the token BLOCK-SEQUENCE-START is not produced: +// +// key: +// - item 1 # BLOCK-SEQUENCE-START is NOT produced here. +// - item 2 +// +// Tokens: +// +// STREAM-START(utf-8) +// BLOCK-MAPPING-START +// KEY +// SCALAR("key",plain) +// VALUE +// BLOCK-ENTRY +// SCALAR("item 1",plain) +// BLOCK-ENTRY +// SCALAR("item 2",plain) +// BLOCK-END +// + +// Ensure that the buffer contains the required number of characters. +// Return true on success, false on failure (reader error or memory error). +func cache(parser *yaml_parser_t, length int) bool { + // [Go] This was inlined: !cache(A, B) -> unread < B && !update(A, B) + return parser.unread >= length || yaml_parser_update_buffer(parser, length) +} + +// Advance the buffer pointer. +func skip(parser *yaml_parser_t) { + if !is_blank(parser.buffer, parser.buffer_pos) { + parser.newlines = 0 + } + parser.mark.index++ + parser.mark.column++ + parser.unread-- + parser.buffer_pos += width(parser.buffer[parser.buffer_pos]) +} + +func skip_line(parser *yaml_parser_t) { + if is_crlf(parser.buffer, parser.buffer_pos) { + parser.mark.index += 2 + parser.mark.column = 0 + parser.mark.line++ + parser.unread -= 2 + parser.buffer_pos += 2 + parser.newlines++ + } else if is_break(parser.buffer, parser.buffer_pos) { + parser.mark.index++ + parser.mark.column = 0 + parser.mark.line++ + parser.unread-- + parser.buffer_pos += width(parser.buffer[parser.buffer_pos]) + parser.newlines++ + } +} + +// Copy a character to a string buffer and advance pointers. +func read(parser *yaml_parser_t, s []byte) []byte { + if !is_blank(parser.buffer, parser.buffer_pos) { + parser.newlines = 0 + } + w := width(parser.buffer[parser.buffer_pos]) + if w == 0 { + panic("invalid character sequence") + } + if len(s) == 0 { + s = make([]byte, 0, 32) + } + if w == 1 && len(s)+w <= cap(s) { + s = s[:len(s)+1] + s[len(s)-1] = parser.buffer[parser.buffer_pos] + parser.buffer_pos++ + } else { + s = append(s, parser.buffer[parser.buffer_pos:parser.buffer_pos+w]...) + parser.buffer_pos += w + } + parser.mark.index++ + parser.mark.column++ + parser.unread-- + return s +} + +// Copy a line break character to a string buffer and advance pointers. +func read_line(parser *yaml_parser_t, s []byte) []byte { + buf := parser.buffer + pos := parser.buffer_pos + switch { + case buf[pos] == '\r' && buf[pos+1] == '\n': + // CR LF . LF + s = append(s, '\n') + parser.buffer_pos += 2 + parser.mark.index++ + parser.unread-- + case buf[pos] == '\r' || buf[pos] == '\n': + // CR|LF . LF + s = append(s, '\n') + parser.buffer_pos += 1 + case buf[pos] == '\xC2' && buf[pos+1] == '\x85': + // NEL . LF + s = append(s, '\n') + parser.buffer_pos += 2 + case buf[pos] == '\xE2' && buf[pos+1] == '\x80' && (buf[pos+2] == '\xA8' || buf[pos+2] == '\xA9'): + // LS|PS . LS|PS + s = append(s, buf[parser.buffer_pos:pos+3]...) + parser.buffer_pos += 3 + default: + return s + } + parser.mark.index++ + parser.mark.column = 0 + parser.mark.line++ + parser.unread-- + parser.newlines++ + return s +} + +// Get the next token. +func yaml_parser_scan(parser *yaml_parser_t, token *yaml_token_t) bool { + // Erase the token object. + *token = yaml_token_t{} // [Go] Is this necessary? + + // No tokens after STREAM-END or error. + if parser.stream_end_produced || parser.error != yaml_NO_ERROR { + return true + } + + // Ensure that the tokens queue contains enough tokens. + if !parser.token_available { + if !yaml_parser_fetch_more_tokens(parser) { + return false + } + } + + // Fetch the next token from the queue. + *token = parser.tokens[parser.tokens_head] + parser.tokens_head++ + parser.tokens_parsed++ + parser.token_available = false + + if token.typ == yaml_STREAM_END_TOKEN { + parser.stream_end_produced = true + } + return true +} + +// Set the scanner error and return false. +func yaml_parser_set_scanner_error(parser *yaml_parser_t, context string, context_mark yaml_mark_t, problem string) bool { + parser.error = yaml_SCANNER_ERROR + parser.context = context + parser.context_mark = context_mark + parser.problem = problem + parser.problem_mark = parser.mark + return false +} + +func yaml_parser_set_scanner_tag_error(parser *yaml_parser_t, directive bool, context_mark yaml_mark_t, problem string) bool { + context := "while parsing a tag" + if directive { + context = "while parsing a %TAG directive" + } + return yaml_parser_set_scanner_error(parser, context, context_mark, problem) +} + +func trace(args ...interface{}) func() { + pargs := append([]interface{}{"+++"}, args...) + fmt.Println(pargs...) + pargs = append([]interface{}{"---"}, args...) + return func() { fmt.Println(pargs...) } +} + +// Ensure that the tokens queue contains at least one token which can be +// returned to the Parser. +func yaml_parser_fetch_more_tokens(parser *yaml_parser_t) bool { + // While we need more tokens to fetch, do it. + for { + // [Go] The comment parsing logic requires a lookahead of two tokens + // so that foot comments may be parsed in time of associating them + // with the tokens that are parsed before them, and also for line + // comments to be transformed into head comments in some edge cases. + if parser.tokens_head < len(parser.tokens)-2 { + // If a potential simple key is at the head position, we need to fetch + // the next token to disambiguate it. + head_tok_idx, ok := parser.simple_keys_by_tok[parser.tokens_parsed] + if !ok { + break + } else if valid, ok := yaml_simple_key_is_valid(parser, &parser.simple_keys[head_tok_idx]); !ok { + return false + } else if !valid { + break + } + } + // Fetch the next token. + if !yaml_parser_fetch_next_token(parser) { + return false + } + } + + parser.token_available = true + return true +} + +// The dispatcher for token fetchers. +func yaml_parser_fetch_next_token(parser *yaml_parser_t) (ok bool) { + // Ensure that the buffer is initialized. + if parser.unread < 1 && !yaml_parser_update_buffer(parser, 1) { + return false + } + + // Check if we just started scanning. Fetch STREAM-START then. + if !parser.stream_start_produced { + return yaml_parser_fetch_stream_start(parser) + } + + scan_mark := parser.mark + + // Eat whitespaces and comments until we reach the next token. + if !yaml_parser_scan_to_next_token(parser) { + return false + } + + // [Go] While unrolling indents, transform the head comments of prior + // indentation levels observed after scan_start into foot comments at + // the respective indexes. + + // Check the indentation level against the current column. + if !yaml_parser_unroll_indent(parser, parser.mark.column, scan_mark) { + return false + } + + // Ensure that the buffer contains at least 4 characters. 4 is the length + // of the longest indicators ('--- ' and '... '). + if parser.unread < 4 && !yaml_parser_update_buffer(parser, 4) { + return false + } + + // Is it the end of the stream? + if is_z(parser.buffer, parser.buffer_pos) { + return yaml_parser_fetch_stream_end(parser) + } + + // Is it a directive? + if parser.mark.column == 0 && parser.buffer[parser.buffer_pos] == '%' { + return yaml_parser_fetch_directive(parser) + } + + buf := parser.buffer + pos := parser.buffer_pos + + // Is it the document start indicator? + if parser.mark.column == 0 && buf[pos] == '-' && buf[pos+1] == '-' && buf[pos+2] == '-' && is_blankz(buf, pos+3) { + return yaml_parser_fetch_document_indicator(parser, yaml_DOCUMENT_START_TOKEN) + } + + // Is it the document end indicator? + if parser.mark.column == 0 && buf[pos] == '.' && buf[pos+1] == '.' && buf[pos+2] == '.' && is_blankz(buf, pos+3) { + return yaml_parser_fetch_document_indicator(parser, yaml_DOCUMENT_END_TOKEN) + } + + comment_mark := parser.mark + if len(parser.tokens) > 0 && (parser.flow_level == 0 && buf[pos] == ':' || parser.flow_level > 0 && buf[pos] == ',') { + // Associate any following comments with the prior token. + comment_mark = parser.tokens[len(parser.tokens)-1].start_mark + } + defer func() { + if !ok { + return + } + if len(parser.tokens) > 0 && parser.tokens[len(parser.tokens)-1].typ == yaml_BLOCK_ENTRY_TOKEN { + // Sequence indicators alone have no line comments. It becomes + // a head comment for whatever follows. + return + } + if !yaml_parser_scan_line_comment(parser, comment_mark) { + ok = false + return + } + }() + + // Is it the flow sequence start indicator? + if buf[pos] == '[' { + return yaml_parser_fetch_flow_collection_start(parser, yaml_FLOW_SEQUENCE_START_TOKEN) + } + + // Is it the flow mapping start indicator? + if parser.buffer[parser.buffer_pos] == '{' { + return yaml_parser_fetch_flow_collection_start(parser, yaml_FLOW_MAPPING_START_TOKEN) + } + + // Is it the flow sequence end indicator? + if parser.buffer[parser.buffer_pos] == ']' { + return yaml_parser_fetch_flow_collection_end(parser, + yaml_FLOW_SEQUENCE_END_TOKEN) + } + + // Is it the flow mapping end indicator? + if parser.buffer[parser.buffer_pos] == '}' { + return yaml_parser_fetch_flow_collection_end(parser, + yaml_FLOW_MAPPING_END_TOKEN) + } + + // Is it the flow entry indicator? + if parser.buffer[parser.buffer_pos] == ',' { + return yaml_parser_fetch_flow_entry(parser) + } + + // Is it the block entry indicator? + if parser.buffer[parser.buffer_pos] == '-' && is_blankz(parser.buffer, parser.buffer_pos+1) { + return yaml_parser_fetch_block_entry(parser) + } + + // Is it the key indicator? + if parser.buffer[parser.buffer_pos] == '?' && (parser.flow_level > 0 || is_blankz(parser.buffer, parser.buffer_pos+1)) { + return yaml_parser_fetch_key(parser) + } + + // Is it the value indicator? + if parser.buffer[parser.buffer_pos] == ':' && (parser.flow_level > 0 || is_blankz(parser.buffer, parser.buffer_pos+1)) { + return yaml_parser_fetch_value(parser) + } + + // Is it an alias? + if parser.buffer[parser.buffer_pos] == '*' { + return yaml_parser_fetch_anchor(parser, yaml_ALIAS_TOKEN) + } + + // Is it an anchor? + if parser.buffer[parser.buffer_pos] == '&' { + return yaml_parser_fetch_anchor(parser, yaml_ANCHOR_TOKEN) + } + + // Is it a tag? + if parser.buffer[parser.buffer_pos] == '!' { + return yaml_parser_fetch_tag(parser) + } + + // Is it a literal scalar? + if parser.buffer[parser.buffer_pos] == '|' && parser.flow_level == 0 { + return yaml_parser_fetch_block_scalar(parser, true) + } + + // Is it a folded scalar? + if parser.buffer[parser.buffer_pos] == '>' && parser.flow_level == 0 { + return yaml_parser_fetch_block_scalar(parser, false) + } + + // Is it a single-quoted scalar? + if parser.buffer[parser.buffer_pos] == '\'' { + return yaml_parser_fetch_flow_scalar(parser, true) + } + + // Is it a double-quoted scalar? + if parser.buffer[parser.buffer_pos] == '"' { + return yaml_parser_fetch_flow_scalar(parser, false) + } + + // Is it a plain scalar? + // + // A plain scalar may start with any non-blank characters except + // + // '-', '?', ':', ',', '[', ']', '{', '}', + // '#', '&', '*', '!', '|', '>', '\'', '\"', + // '%', '@', '`'. + // + // In the block context (and, for the '-' indicator, in the flow context + // too), it may also start with the characters + // + // '-', '?', ':' + // + // if it is followed by a non-space character. + // + // The last rule is more restrictive than the specification requires. + // [Go] TODO Make this logic more reasonable. + //switch parser.buffer[parser.buffer_pos] { + //case '-', '?', ':', ',', '?', '-', ',', ':', ']', '[', '}', '{', '&', '#', '!', '*', '>', '|', '"', '\'', '@', '%', '-', '`': + //} + if !(is_blankz(parser.buffer, parser.buffer_pos) || parser.buffer[parser.buffer_pos] == '-' || + parser.buffer[parser.buffer_pos] == '?' || parser.buffer[parser.buffer_pos] == ':' || + parser.buffer[parser.buffer_pos] == ',' || parser.buffer[parser.buffer_pos] == '[' || + parser.buffer[parser.buffer_pos] == ']' || parser.buffer[parser.buffer_pos] == '{' || + parser.buffer[parser.buffer_pos] == '}' || parser.buffer[parser.buffer_pos] == '#' || + parser.buffer[parser.buffer_pos] == '&' || parser.buffer[parser.buffer_pos] == '*' || + parser.buffer[parser.buffer_pos] == '!' || parser.buffer[parser.buffer_pos] == '|' || + parser.buffer[parser.buffer_pos] == '>' || parser.buffer[parser.buffer_pos] == '\'' || + parser.buffer[parser.buffer_pos] == '"' || parser.buffer[parser.buffer_pos] == '%' || + parser.buffer[parser.buffer_pos] == '@' || parser.buffer[parser.buffer_pos] == '`') || + (parser.buffer[parser.buffer_pos] == '-' && !is_blank(parser.buffer, parser.buffer_pos+1)) || + (parser.flow_level == 0 && + (parser.buffer[parser.buffer_pos] == '?' || parser.buffer[parser.buffer_pos] == ':') && + !is_blankz(parser.buffer, parser.buffer_pos+1)) { + return yaml_parser_fetch_plain_scalar(parser) + } + + // If we don't determine the token type so far, it is an error. + return yaml_parser_set_scanner_error(parser, + "while scanning for the next token", parser.mark, + "found character that cannot start any token") +} + +func yaml_simple_key_is_valid(parser *yaml_parser_t, simple_key *yaml_simple_key_t) (valid, ok bool) { + if !simple_key.possible { + return false, true + } + + // The 1.2 specification says: + // + // "If the ? indicator is omitted, parsing needs to see past the + // implicit key to recognize it as such. To limit the amount of + // lookahead required, the “:” indicator must appear at most 1024 + // Unicode characters beyond the start of the key. In addition, the key + // is restricted to a single line." + // + if simple_key.mark.line < parser.mark.line || simple_key.mark.index+1024 < parser.mark.index { + // Check if the potential simple key to be removed is required. + if simple_key.required { + return false, yaml_parser_set_scanner_error(parser, + "while scanning a simple key", simple_key.mark, + "could not find expected ':'") + } + simple_key.possible = false + return false, true + } + return true, true +} + +// Check if a simple key may start at the current position and add it if +// needed. +func yaml_parser_save_simple_key(parser *yaml_parser_t) bool { + // A simple key is required at the current position if the scanner is in + // the block context and the current column coincides with the indentation + // level. + + required := parser.flow_level == 0 && parser.indent == parser.mark.column + + // + // If the current position may start a simple key, save it. + // + if parser.simple_key_allowed { + simple_key := yaml_simple_key_t{ + possible: true, + required: required, + token_number: parser.tokens_parsed + (len(parser.tokens) - parser.tokens_head), + mark: parser.mark, + } + + if !yaml_parser_remove_simple_key(parser) { + return false + } + parser.simple_keys[len(parser.simple_keys)-1] = simple_key + parser.simple_keys_by_tok[simple_key.token_number] = len(parser.simple_keys) - 1 + } + return true +} + +// Remove a potential simple key at the current flow level. +func yaml_parser_remove_simple_key(parser *yaml_parser_t) bool { + i := len(parser.simple_keys) - 1 + if parser.simple_keys[i].possible { + // If the key is required, it is an error. + if parser.simple_keys[i].required { + return yaml_parser_set_scanner_error(parser, + "while scanning a simple key", parser.simple_keys[i].mark, + "could not find expected ':'") + } + // Remove the key from the stack. + parser.simple_keys[i].possible = false + delete(parser.simple_keys_by_tok, parser.simple_keys[i].token_number) + } + return true +} + +// max_flow_level limits the flow_level +const max_flow_level = 10000 + +// Increase the flow level and resize the simple key list if needed. +func yaml_parser_increase_flow_level(parser *yaml_parser_t) bool { + // Reset the simple key on the next level. + parser.simple_keys = append(parser.simple_keys, yaml_simple_key_t{ + possible: false, + required: false, + token_number: parser.tokens_parsed + (len(parser.tokens) - parser.tokens_head), + mark: parser.mark, + }) + + // Increase the flow level. + parser.flow_level++ + if parser.flow_level > max_flow_level { + return yaml_parser_set_scanner_error(parser, + "while increasing flow level", parser.simple_keys[len(parser.simple_keys)-1].mark, + fmt.Sprintf("exceeded max depth of %d", max_flow_level)) + } + return true +} + +// Decrease the flow level. +func yaml_parser_decrease_flow_level(parser *yaml_parser_t) bool { + if parser.flow_level > 0 { + parser.flow_level-- + last := len(parser.simple_keys) - 1 + delete(parser.simple_keys_by_tok, parser.simple_keys[last].token_number) + parser.simple_keys = parser.simple_keys[:last] + } + return true +} + +// max_indents limits the indents stack size +const max_indents = 10000 + +// Push the current indentation level to the stack and set the new level +// the current column is greater than the indentation level. In this case, +// append or insert the specified token into the token queue. +func yaml_parser_roll_indent(parser *yaml_parser_t, column, number int, typ yaml_token_type_t, mark yaml_mark_t) bool { + // In the flow context, do nothing. + if parser.flow_level > 0 { + return true + } + + if parser.indent < column { + // Push the current indentation level to the stack and set the new + // indentation level. + parser.indents = append(parser.indents, parser.indent) + parser.indent = column + if len(parser.indents) > max_indents { + return yaml_parser_set_scanner_error(parser, + "while increasing indent level", parser.simple_keys[len(parser.simple_keys)-1].mark, + fmt.Sprintf("exceeded max depth of %d", max_indents)) + } + + // Create a token and insert it into the queue. + token := yaml_token_t{ + typ: typ, + start_mark: mark, + end_mark: mark, + } + if number > -1 { + number -= parser.tokens_parsed + } + yaml_insert_token(parser, number, &token) + } + return true +} + +// Pop indentation levels from the indents stack until the current level +// becomes less or equal to the column. For each indentation level, append +// the BLOCK-END token. +func yaml_parser_unroll_indent(parser *yaml_parser_t, column int, scan_mark yaml_mark_t) bool { + // In the flow context, do nothing. + if parser.flow_level > 0 { + return true + } + + block_mark := scan_mark + block_mark.index-- + + // Loop through the indentation levels in the stack. + for parser.indent > column { + + // [Go] Reposition the end token before potential following + // foot comments of parent blocks. For that, search + // backwards for recent comments that were at the same + // indent as the block that is ending now. + stop_index := block_mark.index + for i := len(parser.comments) - 1; i >= 0; i-- { + comment := &parser.comments[i] + + if comment.end_mark.index < stop_index { + // Don't go back beyond the start of the comment/whitespace scan, unless column < 0. + // If requested indent column is < 0, then the document is over and everything else + // is a foot anyway. + break + } + if comment.start_mark.column == parser.indent+1 { + // This is a good match. But maybe there's a former comment + // at that same indent level, so keep searching. + block_mark = comment.start_mark + } + + // While the end of the former comment matches with + // the start of the following one, we know there's + // nothing in between and scanning is still safe. + stop_index = comment.scan_mark.index + } + + // Create a token and append it to the queue. + token := yaml_token_t{ + typ: yaml_BLOCK_END_TOKEN, + start_mark: block_mark, + end_mark: block_mark, + } + yaml_insert_token(parser, -1, &token) + + // Pop the indentation level. + parser.indent = parser.indents[len(parser.indents)-1] + parser.indents = parser.indents[:len(parser.indents)-1] + } + return true +} + +// Initialize the scanner and produce the STREAM-START token. +func yaml_parser_fetch_stream_start(parser *yaml_parser_t) bool { + + // Set the initial indentation. + parser.indent = -1 + + // Initialize the simple key stack. + parser.simple_keys = append(parser.simple_keys, yaml_simple_key_t{}) + + parser.simple_keys_by_tok = make(map[int]int) + + // A simple key is allowed at the beginning of the stream. + parser.simple_key_allowed = true + + // We have started. + parser.stream_start_produced = true + + // Create the STREAM-START token and append it to the queue. + token := yaml_token_t{ + typ: yaml_STREAM_START_TOKEN, + start_mark: parser.mark, + end_mark: parser.mark, + encoding: parser.encoding, + } + yaml_insert_token(parser, -1, &token) + return true +} + +// Produce the STREAM-END token and shut down the scanner. +func yaml_parser_fetch_stream_end(parser *yaml_parser_t) bool { + + // Force new line. + if parser.mark.column != 0 { + parser.mark.column = 0 + parser.mark.line++ + } + + // Reset the indentation level. + if !yaml_parser_unroll_indent(parser, -1, parser.mark) { + return false + } + + // Reset simple keys. + if !yaml_parser_remove_simple_key(parser) { + return false + } + + parser.simple_key_allowed = false + + // Create the STREAM-END token and append it to the queue. + token := yaml_token_t{ + typ: yaml_STREAM_END_TOKEN, + start_mark: parser.mark, + end_mark: parser.mark, + } + yaml_insert_token(parser, -1, &token) + return true +} + +// Produce a VERSION-DIRECTIVE or TAG-DIRECTIVE token. +func yaml_parser_fetch_directive(parser *yaml_parser_t) bool { + // Reset the indentation level. + if !yaml_parser_unroll_indent(parser, -1, parser.mark) { + return false + } + + // Reset simple keys. + if !yaml_parser_remove_simple_key(parser) { + return false + } + + parser.simple_key_allowed = false + + // Create the YAML-DIRECTIVE or TAG-DIRECTIVE token. + token := yaml_token_t{} + if !yaml_parser_scan_directive(parser, &token) { + return false + } + // Append the token to the queue. + yaml_insert_token(parser, -1, &token) + return true +} + +// Produce the DOCUMENT-START or DOCUMENT-END token. +func yaml_parser_fetch_document_indicator(parser *yaml_parser_t, typ yaml_token_type_t) bool { + // Reset the indentation level. + if !yaml_parser_unroll_indent(parser, -1, parser.mark) { + return false + } + + // Reset simple keys. + if !yaml_parser_remove_simple_key(parser) { + return false + } + + parser.simple_key_allowed = false + + // Consume the token. + start_mark := parser.mark + + skip(parser) + skip(parser) + skip(parser) + + end_mark := parser.mark + + // Create the DOCUMENT-START or DOCUMENT-END token. + token := yaml_token_t{ + typ: typ, + start_mark: start_mark, + end_mark: end_mark, + } + // Append the token to the queue. + yaml_insert_token(parser, -1, &token) + return true +} + +// Produce the FLOW-SEQUENCE-START or FLOW-MAPPING-START token. +func yaml_parser_fetch_flow_collection_start(parser *yaml_parser_t, typ yaml_token_type_t) bool { + + // The indicators '[' and '{' may start a simple key. + if !yaml_parser_save_simple_key(parser) { + return false + } + + // Increase the flow level. + if !yaml_parser_increase_flow_level(parser) { + return false + } + + // A simple key may follow the indicators '[' and '{'. + parser.simple_key_allowed = true + + // Consume the token. + start_mark := parser.mark + skip(parser) + end_mark := parser.mark + + // Create the FLOW-SEQUENCE-START of FLOW-MAPPING-START token. + token := yaml_token_t{ + typ: typ, + start_mark: start_mark, + end_mark: end_mark, + } + // Append the token to the queue. + yaml_insert_token(parser, -1, &token) + return true +} + +// Produce the FLOW-SEQUENCE-END or FLOW-MAPPING-END token. +func yaml_parser_fetch_flow_collection_end(parser *yaml_parser_t, typ yaml_token_type_t) bool { + // Reset any potential simple key on the current flow level. + if !yaml_parser_remove_simple_key(parser) { + return false + } + + // Decrease the flow level. + if !yaml_parser_decrease_flow_level(parser) { + return false + } + + // No simple keys after the indicators ']' and '}'. + parser.simple_key_allowed = false + + // Consume the token. + + start_mark := parser.mark + skip(parser) + end_mark := parser.mark + + // Create the FLOW-SEQUENCE-END of FLOW-MAPPING-END token. + token := yaml_token_t{ + typ: typ, + start_mark: start_mark, + end_mark: end_mark, + } + // Append the token to the queue. + yaml_insert_token(parser, -1, &token) + return true +} + +// Produce the FLOW-ENTRY token. +func yaml_parser_fetch_flow_entry(parser *yaml_parser_t) bool { + // Reset any potential simple keys on the current flow level. + if !yaml_parser_remove_simple_key(parser) { + return false + } + + // Simple keys are allowed after ','. + parser.simple_key_allowed = true + + // Consume the token. + start_mark := parser.mark + skip(parser) + end_mark := parser.mark + + // Create the FLOW-ENTRY token and append it to the queue. + token := yaml_token_t{ + typ: yaml_FLOW_ENTRY_TOKEN, + start_mark: start_mark, + end_mark: end_mark, + } + yaml_insert_token(parser, -1, &token) + return true +} + +// Produce the BLOCK-ENTRY token. +func yaml_parser_fetch_block_entry(parser *yaml_parser_t) bool { + // Check if the scanner is in the block context. + if parser.flow_level == 0 { + // Check if we are allowed to start a new entry. + if !parser.simple_key_allowed { + return yaml_parser_set_scanner_error(parser, "", parser.mark, + "block sequence entries are not allowed in this context") + } + // Add the BLOCK-SEQUENCE-START token if needed. + if !yaml_parser_roll_indent(parser, parser.mark.column, -1, yaml_BLOCK_SEQUENCE_START_TOKEN, parser.mark) { + return false + } + } else { + // It is an error for the '-' indicator to occur in the flow context, + // but we let the Parser detect and report about it because the Parser + // is able to point to the context. + } + + // Reset any potential simple keys on the current flow level. + if !yaml_parser_remove_simple_key(parser) { + return false + } + + // Simple keys are allowed after '-'. + parser.simple_key_allowed = true + + // Consume the token. + start_mark := parser.mark + skip(parser) + end_mark := parser.mark + + // Create the BLOCK-ENTRY token and append it to the queue. + token := yaml_token_t{ + typ: yaml_BLOCK_ENTRY_TOKEN, + start_mark: start_mark, + end_mark: end_mark, + } + yaml_insert_token(parser, -1, &token) + return true +} + +// Produce the KEY token. +func yaml_parser_fetch_key(parser *yaml_parser_t) bool { + + // In the block context, additional checks are required. + if parser.flow_level == 0 { + // Check if we are allowed to start a new key (not nessesary simple). + if !parser.simple_key_allowed { + return yaml_parser_set_scanner_error(parser, "", parser.mark, + "mapping keys are not allowed in this context") + } + // Add the BLOCK-MAPPING-START token if needed. + if !yaml_parser_roll_indent(parser, parser.mark.column, -1, yaml_BLOCK_MAPPING_START_TOKEN, parser.mark) { + return false + } + } + + // Reset any potential simple keys on the current flow level. + if !yaml_parser_remove_simple_key(parser) { + return false + } + + // Simple keys are allowed after '?' in the block context. + parser.simple_key_allowed = parser.flow_level == 0 + + // Consume the token. + start_mark := parser.mark + skip(parser) + end_mark := parser.mark + + // Create the KEY token and append it to the queue. + token := yaml_token_t{ + typ: yaml_KEY_TOKEN, + start_mark: start_mark, + end_mark: end_mark, + } + yaml_insert_token(parser, -1, &token) + return true +} + +// Produce the VALUE token. +func yaml_parser_fetch_value(parser *yaml_parser_t) bool { + + simple_key := &parser.simple_keys[len(parser.simple_keys)-1] + + // Have we found a simple key? + if valid, ok := yaml_simple_key_is_valid(parser, simple_key); !ok { + return false + + } else if valid { + + // Create the KEY token and insert it into the queue. + token := yaml_token_t{ + typ: yaml_KEY_TOKEN, + start_mark: simple_key.mark, + end_mark: simple_key.mark, + } + yaml_insert_token(parser, simple_key.token_number-parser.tokens_parsed, &token) + + // In the block context, we may need to add the BLOCK-MAPPING-START token. + if !yaml_parser_roll_indent(parser, simple_key.mark.column, + simple_key.token_number, + yaml_BLOCK_MAPPING_START_TOKEN, simple_key.mark) { + return false + } + + // Remove the simple key. + simple_key.possible = false + delete(parser.simple_keys_by_tok, simple_key.token_number) + + // A simple key cannot follow another simple key. + parser.simple_key_allowed = false + + } else { + // The ':' indicator follows a complex key. + + // In the block context, extra checks are required. + if parser.flow_level == 0 { + + // Check if we are allowed to start a complex value. + if !parser.simple_key_allowed { + return yaml_parser_set_scanner_error(parser, "", parser.mark, + "mapping values are not allowed in this context") + } + + // Add the BLOCK-MAPPING-START token if needed. + if !yaml_parser_roll_indent(parser, parser.mark.column, -1, yaml_BLOCK_MAPPING_START_TOKEN, parser.mark) { + return false + } + } + + // Simple keys after ':' are allowed in the block context. + parser.simple_key_allowed = parser.flow_level == 0 + } + + // Consume the token. + start_mark := parser.mark + skip(parser) + end_mark := parser.mark + + // Create the VALUE token and append it to the queue. + token := yaml_token_t{ + typ: yaml_VALUE_TOKEN, + start_mark: start_mark, + end_mark: end_mark, + } + yaml_insert_token(parser, -1, &token) + return true +} + +// Produce the ALIAS or ANCHOR token. +func yaml_parser_fetch_anchor(parser *yaml_parser_t, typ yaml_token_type_t) bool { + // An anchor or an alias could be a simple key. + if !yaml_parser_save_simple_key(parser) { + return false + } + + // A simple key cannot follow an anchor or an alias. + parser.simple_key_allowed = false + + // Create the ALIAS or ANCHOR token and append it to the queue. + var token yaml_token_t + if !yaml_parser_scan_anchor(parser, &token, typ) { + return false + } + yaml_insert_token(parser, -1, &token) + return true +} + +// Produce the TAG token. +func yaml_parser_fetch_tag(parser *yaml_parser_t) bool { + // A tag could be a simple key. + if !yaml_parser_save_simple_key(parser) { + return false + } + + // A simple key cannot follow a tag. + parser.simple_key_allowed = false + + // Create the TAG token and append it to the queue. + var token yaml_token_t + if !yaml_parser_scan_tag(parser, &token) { + return false + } + yaml_insert_token(parser, -1, &token) + return true +} + +// Produce the SCALAR(...,literal) or SCALAR(...,folded) tokens. +func yaml_parser_fetch_block_scalar(parser *yaml_parser_t, literal bool) bool { + // Remove any potential simple keys. + if !yaml_parser_remove_simple_key(parser) { + return false + } + + // A simple key may follow a block scalar. + parser.simple_key_allowed = true + + // Create the SCALAR token and append it to the queue. + var token yaml_token_t + if !yaml_parser_scan_block_scalar(parser, &token, literal) { + return false + } + yaml_insert_token(parser, -1, &token) + return true +} + +// Produce the SCALAR(...,single-quoted) or SCALAR(...,double-quoted) tokens. +func yaml_parser_fetch_flow_scalar(parser *yaml_parser_t, single bool) bool { + // A plain scalar could be a simple key. + if !yaml_parser_save_simple_key(parser) { + return false + } + + // A simple key cannot follow a flow scalar. + parser.simple_key_allowed = false + + // Create the SCALAR token and append it to the queue. + var token yaml_token_t + if !yaml_parser_scan_flow_scalar(parser, &token, single) { + return false + } + yaml_insert_token(parser, -1, &token) + return true +} + +// Produce the SCALAR(...,plain) token. +func yaml_parser_fetch_plain_scalar(parser *yaml_parser_t) bool { + // A plain scalar could be a simple key. + if !yaml_parser_save_simple_key(parser) { + return false + } + + // A simple key cannot follow a flow scalar. + parser.simple_key_allowed = false + + // Create the SCALAR token and append it to the queue. + var token yaml_token_t + if !yaml_parser_scan_plain_scalar(parser, &token) { + return false + } + yaml_insert_token(parser, -1, &token) + return true +} + +// Eat whitespaces and comments until the next token is found. +func yaml_parser_scan_to_next_token(parser *yaml_parser_t) bool { + + scan_mark := parser.mark + + // Until the next token is not found. + for { + // Allow the BOM mark to start a line. + if parser.unread < 1 && !yaml_parser_update_buffer(parser, 1) { + return false + } + if parser.mark.column == 0 && is_bom(parser.buffer, parser.buffer_pos) { + skip(parser) + } + + // Eat whitespaces. + // Tabs are allowed: + // - in the flow context + // - in the block context, but not at the beginning of the line or + // after '-', '?', or ':' (complex value). + if parser.unread < 1 && !yaml_parser_update_buffer(parser, 1) { + return false + } + + for parser.buffer[parser.buffer_pos] == ' ' || ((parser.flow_level > 0 || !parser.simple_key_allowed) && parser.buffer[parser.buffer_pos] == '\t') { + skip(parser) + if parser.unread < 1 && !yaml_parser_update_buffer(parser, 1) { + return false + } + } + + // Check if we just had a line comment under a sequence entry that + // looks more like a header to the following content. Similar to this: + // + // - # The comment + // - Some data + // + // If so, transform the line comment to a head comment and reposition. + if len(parser.comments) > 0 && len(parser.tokens) > 1 { + tokenA := parser.tokens[len(parser.tokens)-2] + tokenB := parser.tokens[len(parser.tokens)-1] + comment := &parser.comments[len(parser.comments)-1] + if tokenA.typ == yaml_BLOCK_SEQUENCE_START_TOKEN && tokenB.typ == yaml_BLOCK_ENTRY_TOKEN && len(comment.line) > 0 && !is_break(parser.buffer, parser.buffer_pos) { + // If it was in the prior line, reposition so it becomes a + // header of the follow up token. Otherwise, keep it in place + // so it becomes a header of the former. + comment.head = comment.line + comment.line = nil + if comment.start_mark.line == parser.mark.line-1 { + comment.token_mark = parser.mark + } + } + } + + // Eat a comment until a line break. + if parser.buffer[parser.buffer_pos] == '#' { + if !yaml_parser_scan_comments(parser, scan_mark) { + return false + } + } + + // If it is a line break, eat it. + if is_break(parser.buffer, parser.buffer_pos) { + if parser.unread < 2 && !yaml_parser_update_buffer(parser, 2) { + return false + } + skip_line(parser) + + // In the block context, a new line may start a simple key. + if parser.flow_level == 0 { + parser.simple_key_allowed = true + } + } else { + break // We have found a token. + } + } + + return true +} + +// Scan a YAML-DIRECTIVE or TAG-DIRECTIVE token. +// +// Scope: +// %YAML 1.1 # a comment \n +// ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ +// %TAG !yaml! tag:yaml.org,2002: \n +// ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ +// +func yaml_parser_scan_directive(parser *yaml_parser_t, token *yaml_token_t) bool { + // Eat '%'. + start_mark := parser.mark + skip(parser) + + // Scan the directive name. + var name []byte + if !yaml_parser_scan_directive_name(parser, start_mark, &name) { + return false + } + + // Is it a YAML directive? + if bytes.Equal(name, []byte("YAML")) { + // Scan the VERSION directive value. + var major, minor int8 + if !yaml_parser_scan_version_directive_value(parser, start_mark, &major, &minor) { + return false + } + end_mark := parser.mark + + // Create a VERSION-DIRECTIVE token. + *token = yaml_token_t{ + typ: yaml_VERSION_DIRECTIVE_TOKEN, + start_mark: start_mark, + end_mark: end_mark, + major: major, + minor: minor, + } + + // Is it a TAG directive? + } else if bytes.Equal(name, []byte("TAG")) { + // Scan the TAG directive value. + var handle, prefix []byte + if !yaml_parser_scan_tag_directive_value(parser, start_mark, &handle, &prefix) { + return false + } + end_mark := parser.mark + + // Create a TAG-DIRECTIVE token. + *token = yaml_token_t{ + typ: yaml_TAG_DIRECTIVE_TOKEN, + start_mark: start_mark, + end_mark: end_mark, + value: handle, + prefix: prefix, + } + + // Unknown directive. + } else { + yaml_parser_set_scanner_error(parser, "while scanning a directive", + start_mark, "found unknown directive name") + return false + } + + // Eat the rest of the line including any comments. + if parser.unread < 1 && !yaml_parser_update_buffer(parser, 1) { + return false + } + + for is_blank(parser.buffer, parser.buffer_pos) { + skip(parser) + if parser.unread < 1 && !yaml_parser_update_buffer(parser, 1) { + return false + } + } + + if parser.buffer[parser.buffer_pos] == '#' { + // [Go] Discard this inline comment for the time being. + //if !yaml_parser_scan_line_comment(parser, start_mark) { + // return false + //} + for !is_breakz(parser.buffer, parser.buffer_pos) { + skip(parser) + if parser.unread < 1 && !yaml_parser_update_buffer(parser, 1) { + return false + } + } + } + + // Check if we are at the end of the line. + if !is_breakz(parser.buffer, parser.buffer_pos) { + yaml_parser_set_scanner_error(parser, "while scanning a directive", + start_mark, "did not find expected comment or line break") + return false + } + + // Eat a line break. + if is_break(parser.buffer, parser.buffer_pos) { + if parser.unread < 2 && !yaml_parser_update_buffer(parser, 2) { + return false + } + skip_line(parser) + } + + return true +} + +// Scan the directive name. +// +// Scope: +// %YAML 1.1 # a comment \n +// ^^^^ +// %TAG !yaml! tag:yaml.org,2002: \n +// ^^^ +// +func yaml_parser_scan_directive_name(parser *yaml_parser_t, start_mark yaml_mark_t, name *[]byte) bool { + // Consume the directive name. + if parser.unread < 1 && !yaml_parser_update_buffer(parser, 1) { + return false + } + + var s []byte + for is_alpha(parser.buffer, parser.buffer_pos) { + s = read(parser, s) + if parser.unread < 1 && !yaml_parser_update_buffer(parser, 1) { + return false + } + } + + // Check if the name is empty. + if len(s) == 0 { + yaml_parser_set_scanner_error(parser, "while scanning a directive", + start_mark, "could not find expected directive name") + return false + } + + // Check for an blank character after the name. + if !is_blankz(parser.buffer, parser.buffer_pos) { + yaml_parser_set_scanner_error(parser, "while scanning a directive", + start_mark, "found unexpected non-alphabetical character") + return false + } + *name = s + return true +} + +// Scan the value of VERSION-DIRECTIVE. +// +// Scope: +// %YAML 1.1 # a comment \n +// ^^^^^^ +func yaml_parser_scan_version_directive_value(parser *yaml_parser_t, start_mark yaml_mark_t, major, minor *int8) bool { + // Eat whitespaces. + if parser.unread < 1 && !yaml_parser_update_buffer(parser, 1) { + return false + } + for is_blank(parser.buffer, parser.buffer_pos) { + skip(parser) + if parser.unread < 1 && !yaml_parser_update_buffer(parser, 1) { + return false + } + } + + // Consume the major version number. + if !yaml_parser_scan_version_directive_number(parser, start_mark, major) { + return false + } + + // Eat '.'. + if parser.buffer[parser.buffer_pos] != '.' { + return yaml_parser_set_scanner_error(parser, "while scanning a %YAML directive", + start_mark, "did not find expected digit or '.' character") + } + + skip(parser) + + // Consume the minor version number. + if !yaml_parser_scan_version_directive_number(parser, start_mark, minor) { + return false + } + return true +} + +const max_number_length = 2 + +// Scan the version number of VERSION-DIRECTIVE. +// +// Scope: +// %YAML 1.1 # a comment \n +// ^ +// %YAML 1.1 # a comment \n +// ^ +func yaml_parser_scan_version_directive_number(parser *yaml_parser_t, start_mark yaml_mark_t, number *int8) bool { + + // Repeat while the next character is digit. + if parser.unread < 1 && !yaml_parser_update_buffer(parser, 1) { + return false + } + var value, length int8 + for is_digit(parser.buffer, parser.buffer_pos) { + // Check if the number is too long. + length++ + if length > max_number_length { + return yaml_parser_set_scanner_error(parser, "while scanning a %YAML directive", + start_mark, "found extremely long version number") + } + value = value*10 + int8(as_digit(parser.buffer, parser.buffer_pos)) + skip(parser) + if parser.unread < 1 && !yaml_parser_update_buffer(parser, 1) { + return false + } + } + + // Check if the number was present. + if length == 0 { + return yaml_parser_set_scanner_error(parser, "while scanning a %YAML directive", + start_mark, "did not find expected version number") + } + *number = value + return true +} + +// Scan the value of a TAG-DIRECTIVE token. +// +// Scope: +// %TAG !yaml! tag:yaml.org,2002: \n +// ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^ +// +func yaml_parser_scan_tag_directive_value(parser *yaml_parser_t, start_mark yaml_mark_t, handle, prefix *[]byte) bool { + var handle_value, prefix_value []byte + + // Eat whitespaces. + if parser.unread < 1 && !yaml_parser_update_buffer(parser, 1) { + return false + } + + for is_blank(parser.buffer, parser.buffer_pos) { + skip(parser) + if parser.unread < 1 && !yaml_parser_update_buffer(parser, 1) { + return false + } + } + + // Scan a handle. + if !yaml_parser_scan_tag_handle(parser, true, start_mark, &handle_value) { + return false + } + + // Expect a whitespace. + if parser.unread < 1 && !yaml_parser_update_buffer(parser, 1) { + return false + } + if !is_blank(parser.buffer, parser.buffer_pos) { + yaml_parser_set_scanner_error(parser, "while scanning a %TAG directive", + start_mark, "did not find expected whitespace") + return false + } + + // Eat whitespaces. + for is_blank(parser.buffer, parser.buffer_pos) { + skip(parser) + if parser.unread < 1 && !yaml_parser_update_buffer(parser, 1) { + return false + } + } + + // Scan a prefix. + if !yaml_parser_scan_tag_uri(parser, true, nil, start_mark, &prefix_value) { + return false + } + + // Expect a whitespace or line break. + if parser.unread < 1 && !yaml_parser_update_buffer(parser, 1) { + return false + } + if !is_blankz(parser.buffer, parser.buffer_pos) { + yaml_parser_set_scanner_error(parser, "while scanning a %TAG directive", + start_mark, "did not find expected whitespace or line break") + return false + } + + *handle = handle_value + *prefix = prefix_value + return true +} + +func yaml_parser_scan_anchor(parser *yaml_parser_t, token *yaml_token_t, typ yaml_token_type_t) bool { + var s []byte + + // Eat the indicator character. + start_mark := parser.mark + skip(parser) + + // Consume the value. + if parser.unread < 1 && !yaml_parser_update_buffer(parser, 1) { + return false + } + + for is_alpha(parser.buffer, parser.buffer_pos) { + s = read(parser, s) + if parser.unread < 1 && !yaml_parser_update_buffer(parser, 1) { + return false + } + } + + end_mark := parser.mark + + /* + * Check if length of the anchor is greater than 0 and it is followed by + * a whitespace character or one of the indicators: + * + * '?', ':', ',', ']', '}', '%', '@', '`'. + */ + + if len(s) == 0 || + !(is_blankz(parser.buffer, parser.buffer_pos) || parser.buffer[parser.buffer_pos] == '?' || + parser.buffer[parser.buffer_pos] == ':' || parser.buffer[parser.buffer_pos] == ',' || + parser.buffer[parser.buffer_pos] == ']' || parser.buffer[parser.buffer_pos] == '}' || + parser.buffer[parser.buffer_pos] == '%' || parser.buffer[parser.buffer_pos] == '@' || + parser.buffer[parser.buffer_pos] == '`') { + context := "while scanning an alias" + if typ == yaml_ANCHOR_TOKEN { + context = "while scanning an anchor" + } + yaml_parser_set_scanner_error(parser, context, start_mark, + "did not find expected alphabetic or numeric character") + return false + } + + // Create a token. + *token = yaml_token_t{ + typ: typ, + start_mark: start_mark, + end_mark: end_mark, + value: s, + } + + return true +} + +/* + * Scan a TAG token. + */ + +func yaml_parser_scan_tag(parser *yaml_parser_t, token *yaml_token_t) bool { + var handle, suffix []byte + + start_mark := parser.mark + + // Check if the tag is in the canonical form. + if parser.unread < 2 && !yaml_parser_update_buffer(parser, 2) { + return false + } + + if parser.buffer[parser.buffer_pos+1] == '<' { + // Keep the handle as '' + + // Eat '!<' + skip(parser) + skip(parser) + + // Consume the tag value. + if !yaml_parser_scan_tag_uri(parser, false, nil, start_mark, &suffix) { + return false + } + + // Check for '>' and eat it. + if parser.buffer[parser.buffer_pos] != '>' { + yaml_parser_set_scanner_error(parser, "while scanning a tag", + start_mark, "did not find the expected '>'") + return false + } + + skip(parser) + } else { + // The tag has either the '!suffix' or the '!handle!suffix' form. + + // First, try to scan a handle. + if !yaml_parser_scan_tag_handle(parser, false, start_mark, &handle) { + return false + } + + // Check if it is, indeed, handle. + if handle[0] == '!' && len(handle) > 1 && handle[len(handle)-1] == '!' { + // Scan the suffix now. + if !yaml_parser_scan_tag_uri(parser, false, nil, start_mark, &suffix) { + return false + } + } else { + // It wasn't a handle after all. Scan the rest of the tag. + if !yaml_parser_scan_tag_uri(parser, false, handle, start_mark, &suffix) { + return false + } + + // Set the handle to '!'. + handle = []byte{'!'} + + // A special case: the '!' tag. Set the handle to '' and the + // suffix to '!'. + if len(suffix) == 0 { + handle, suffix = suffix, handle + } + } + } + + // Check the character which ends the tag. + if parser.unread < 1 && !yaml_parser_update_buffer(parser, 1) { + return false + } + if !is_blankz(parser.buffer, parser.buffer_pos) { + yaml_parser_set_scanner_error(parser, "while scanning a tag", + start_mark, "did not find expected whitespace or line break") + return false + } + + end_mark := parser.mark + + // Create a token. + *token = yaml_token_t{ + typ: yaml_TAG_TOKEN, + start_mark: start_mark, + end_mark: end_mark, + value: handle, + suffix: suffix, + } + return true +} + +// Scan a tag handle. +func yaml_parser_scan_tag_handle(parser *yaml_parser_t, directive bool, start_mark yaml_mark_t, handle *[]byte) bool { + // Check the initial '!' character. + if parser.unread < 1 && !yaml_parser_update_buffer(parser, 1) { + return false + } + if parser.buffer[parser.buffer_pos] != '!' { + yaml_parser_set_scanner_tag_error(parser, directive, + start_mark, "did not find expected '!'") + return false + } + + var s []byte + + // Copy the '!' character. + s = read(parser, s) + + // Copy all subsequent alphabetical and numerical characters. + if parser.unread < 1 && !yaml_parser_update_buffer(parser, 1) { + return false + } + for is_alpha(parser.buffer, parser.buffer_pos) { + s = read(parser, s) + if parser.unread < 1 && !yaml_parser_update_buffer(parser, 1) { + return false + } + } + + // Check if the trailing character is '!' and copy it. + if parser.buffer[parser.buffer_pos] == '!' { + s = read(parser, s) + } else { + // It's either the '!' tag or not really a tag handle. If it's a %TAG + // directive, it's an error. If it's a tag token, it must be a part of URI. + if directive && string(s) != "!" { + yaml_parser_set_scanner_tag_error(parser, directive, + start_mark, "did not find expected '!'") + return false + } + } + + *handle = s + return true +} + +// Scan a tag. +func yaml_parser_scan_tag_uri(parser *yaml_parser_t, directive bool, head []byte, start_mark yaml_mark_t, uri *[]byte) bool { + //size_t length = head ? strlen((char *)head) : 0 + var s []byte + hasTag := len(head) > 0 + + // Copy the head if needed. + // + // Note that we don't copy the leading '!' character. + if len(head) > 1 { + s = append(s, head[1:]...) + } + + // Scan the tag. + if parser.unread < 1 && !yaml_parser_update_buffer(parser, 1) { + return false + } + + // The set of characters that may appear in URI is as follows: + // + // '0'-'9', 'A'-'Z', 'a'-'z', '_', '-', ';', '/', '?', ':', '@', '&', + // '=', '+', '$', ',', '.', '!', '~', '*', '\'', '(', ')', '[', ']', + // '%'. + // [Go] TODO Convert this into more reasonable logic. + for is_alpha(parser.buffer, parser.buffer_pos) || parser.buffer[parser.buffer_pos] == ';' || + parser.buffer[parser.buffer_pos] == '/' || parser.buffer[parser.buffer_pos] == '?' || + parser.buffer[parser.buffer_pos] == ':' || parser.buffer[parser.buffer_pos] == '@' || + parser.buffer[parser.buffer_pos] == '&' || parser.buffer[parser.buffer_pos] == '=' || + parser.buffer[parser.buffer_pos] == '+' || parser.buffer[parser.buffer_pos] == '$' || + parser.buffer[parser.buffer_pos] == ',' || parser.buffer[parser.buffer_pos] == '.' || + parser.buffer[parser.buffer_pos] == '!' || parser.buffer[parser.buffer_pos] == '~' || + parser.buffer[parser.buffer_pos] == '*' || parser.buffer[parser.buffer_pos] == '\'' || + parser.buffer[parser.buffer_pos] == '(' || parser.buffer[parser.buffer_pos] == ')' || + parser.buffer[parser.buffer_pos] == '[' || parser.buffer[parser.buffer_pos] == ']' || + parser.buffer[parser.buffer_pos] == '%' { + // Check if it is a URI-escape sequence. + if parser.buffer[parser.buffer_pos] == '%' { + if !yaml_parser_scan_uri_escapes(parser, directive, start_mark, &s) { + return false + } + } else { + s = read(parser, s) + } + if parser.unread < 1 && !yaml_parser_update_buffer(parser, 1) { + return false + } + hasTag = true + } + + if !hasTag { + yaml_parser_set_scanner_tag_error(parser, directive, + start_mark, "did not find expected tag URI") + return false + } + *uri = s + return true +} + +// Decode an URI-escape sequence corresponding to a single UTF-8 character. +func yaml_parser_scan_uri_escapes(parser *yaml_parser_t, directive bool, start_mark yaml_mark_t, s *[]byte) bool { + + // Decode the required number of characters. + w := 1024 + for w > 0 { + // Check for a URI-escaped octet. + if parser.unread < 3 && !yaml_parser_update_buffer(parser, 3) { + return false + } + + if !(parser.buffer[parser.buffer_pos] == '%' && + is_hex(parser.buffer, parser.buffer_pos+1) && + is_hex(parser.buffer, parser.buffer_pos+2)) { + return yaml_parser_set_scanner_tag_error(parser, directive, + start_mark, "did not find URI escaped octet") + } + + // Get the octet. + octet := byte((as_hex(parser.buffer, parser.buffer_pos+1) << 4) + as_hex(parser.buffer, parser.buffer_pos+2)) + + // If it is the leading octet, determine the length of the UTF-8 sequence. + if w == 1024 { + w = width(octet) + if w == 0 { + return yaml_parser_set_scanner_tag_error(parser, directive, + start_mark, "found an incorrect leading UTF-8 octet") + } + } else { + // Check if the trailing octet is correct. + if octet&0xC0 != 0x80 { + return yaml_parser_set_scanner_tag_error(parser, directive, + start_mark, "found an incorrect trailing UTF-8 octet") + } + } + + // Copy the octet and move the pointers. + *s = append(*s, octet) + skip(parser) + skip(parser) + skip(parser) + w-- + } + return true +} + +// Scan a block scalar. +func yaml_parser_scan_block_scalar(parser *yaml_parser_t, token *yaml_token_t, literal bool) bool { + // Eat the indicator '|' or '>'. + start_mark := parser.mark + skip(parser) + + // Scan the additional block scalar indicators. + if parser.unread < 1 && !yaml_parser_update_buffer(parser, 1) { + return false + } + + // Check for a chomping indicator. + var chomping, increment int + if parser.buffer[parser.buffer_pos] == '+' || parser.buffer[parser.buffer_pos] == '-' { + // Set the chomping method and eat the indicator. + if parser.buffer[parser.buffer_pos] == '+' { + chomping = +1 + } else { + chomping = -1 + } + skip(parser) + + // Check for an indentation indicator. + if parser.unread < 1 && !yaml_parser_update_buffer(parser, 1) { + return false + } + if is_digit(parser.buffer, parser.buffer_pos) { + // Check that the indentation is greater than 0. + if parser.buffer[parser.buffer_pos] == '0' { + yaml_parser_set_scanner_error(parser, "while scanning a block scalar", + start_mark, "found an indentation indicator equal to 0") + return false + } + + // Get the indentation level and eat the indicator. + increment = as_digit(parser.buffer, parser.buffer_pos) + skip(parser) + } + + } else if is_digit(parser.buffer, parser.buffer_pos) { + // Do the same as above, but in the opposite order. + + if parser.buffer[parser.buffer_pos] == '0' { + yaml_parser_set_scanner_error(parser, "while scanning a block scalar", + start_mark, "found an indentation indicator equal to 0") + return false + } + increment = as_digit(parser.buffer, parser.buffer_pos) + skip(parser) + + if parser.unread < 1 && !yaml_parser_update_buffer(parser, 1) { + return false + } + if parser.buffer[parser.buffer_pos] == '+' || parser.buffer[parser.buffer_pos] == '-' { + if parser.buffer[parser.buffer_pos] == '+' { + chomping = +1 + } else { + chomping = -1 + } + skip(parser) + } + } + + // Eat whitespaces and comments to the end of the line. + if parser.unread < 1 && !yaml_parser_update_buffer(parser, 1) { + return false + } + for is_blank(parser.buffer, parser.buffer_pos) { + skip(parser) + if parser.unread < 1 && !yaml_parser_update_buffer(parser, 1) { + return false + } + } + if parser.buffer[parser.buffer_pos] == '#' { + if !yaml_parser_scan_line_comment(parser, start_mark) { + return false + } + for !is_breakz(parser.buffer, parser.buffer_pos) { + skip(parser) + if parser.unread < 1 && !yaml_parser_update_buffer(parser, 1) { + return false + } + } + } + + // Check if we are at the end of the line. + if !is_breakz(parser.buffer, parser.buffer_pos) { + yaml_parser_set_scanner_error(parser, "while scanning a block scalar", + start_mark, "did not find expected comment or line break") + return false + } + + // Eat a line break. + if is_break(parser.buffer, parser.buffer_pos) { + if parser.unread < 2 && !yaml_parser_update_buffer(parser, 2) { + return false + } + skip_line(parser) + } + + end_mark := parser.mark + + // Set the indentation level if it was specified. + var indent int + if increment > 0 { + if parser.indent >= 0 { + indent = parser.indent + increment + } else { + indent = increment + } + } + + // Scan the leading line breaks and determine the indentation level if needed. + var s, leading_break, trailing_breaks []byte + if !yaml_parser_scan_block_scalar_breaks(parser, &indent, &trailing_breaks, start_mark, &end_mark) { + return false + } + + // Scan the block scalar content. + if parser.unread < 1 && !yaml_parser_update_buffer(parser, 1) { + return false + } + var leading_blank, trailing_blank bool + for parser.mark.column == indent && !is_z(parser.buffer, parser.buffer_pos) { + // We are at the beginning of a non-empty line. + + // Is it a trailing whitespace? + trailing_blank = is_blank(parser.buffer, parser.buffer_pos) + + // Check if we need to fold the leading line break. + if !literal && !leading_blank && !trailing_blank && len(leading_break) > 0 && leading_break[0] == '\n' { + // Do we need to join the lines by space? + if len(trailing_breaks) == 0 { + s = append(s, ' ') + } + } else { + s = append(s, leading_break...) + } + leading_break = leading_break[:0] + + // Append the remaining line breaks. + s = append(s, trailing_breaks...) + trailing_breaks = trailing_breaks[:0] + + // Is it a leading whitespace? + leading_blank = is_blank(parser.buffer, parser.buffer_pos) + + // Consume the current line. + for !is_breakz(parser.buffer, parser.buffer_pos) { + s = read(parser, s) + if parser.unread < 1 && !yaml_parser_update_buffer(parser, 1) { + return false + } + } + + // Consume the line break. + if parser.unread < 2 && !yaml_parser_update_buffer(parser, 2) { + return false + } + + leading_break = read_line(parser, leading_break) + + // Eat the following indentation spaces and line breaks. + if !yaml_parser_scan_block_scalar_breaks(parser, &indent, &trailing_breaks, start_mark, &end_mark) { + return false + } + } + + // Chomp the tail. + if chomping != -1 { + s = append(s, leading_break...) + } + if chomping == 1 { + s = append(s, trailing_breaks...) + } + + // Create a token. + *token = yaml_token_t{ + typ: yaml_SCALAR_TOKEN, + start_mark: start_mark, + end_mark: end_mark, + value: s, + style: yaml_LITERAL_SCALAR_STYLE, + } + if !literal { + token.style = yaml_FOLDED_SCALAR_STYLE + } + return true +} + +// Scan indentation spaces and line breaks for a block scalar. Determine the +// indentation level if needed. +func yaml_parser_scan_block_scalar_breaks(parser *yaml_parser_t, indent *int, breaks *[]byte, start_mark yaml_mark_t, end_mark *yaml_mark_t) bool { + *end_mark = parser.mark + + // Eat the indentation spaces and line breaks. + max_indent := 0 + for { + // Eat the indentation spaces. + if parser.unread < 1 && !yaml_parser_update_buffer(parser, 1) { + return false + } + for (*indent == 0 || parser.mark.column < *indent) && is_space(parser.buffer, parser.buffer_pos) { + skip(parser) + if parser.unread < 1 && !yaml_parser_update_buffer(parser, 1) { + return false + } + } + if parser.mark.column > max_indent { + max_indent = parser.mark.column + } + + // Check for a tab character messing the indentation. + if (*indent == 0 || parser.mark.column < *indent) && is_tab(parser.buffer, parser.buffer_pos) { + return yaml_parser_set_scanner_error(parser, "while scanning a block scalar", + start_mark, "found a tab character where an indentation space is expected") + } + + // Have we found a non-empty line? + if !is_break(parser.buffer, parser.buffer_pos) { + break + } + + // Consume the line break. + if parser.unread < 2 && !yaml_parser_update_buffer(parser, 2) { + return false + } + // [Go] Should really be returning breaks instead. + *breaks = read_line(parser, *breaks) + *end_mark = parser.mark + } + + // Determine the indentation level if needed. + if *indent == 0 { + *indent = max_indent + if *indent < parser.indent+1 { + *indent = parser.indent + 1 + } + if *indent < 1 { + *indent = 1 + } + } + return true +} + +// Scan a quoted scalar. +func yaml_parser_scan_flow_scalar(parser *yaml_parser_t, token *yaml_token_t, single bool) bool { + // Eat the left quote. + start_mark := parser.mark + skip(parser) + + // Consume the content of the quoted scalar. + var s, leading_break, trailing_breaks, whitespaces []byte + for { + // Check that there are no document indicators at the beginning of the line. + if parser.unread < 4 && !yaml_parser_update_buffer(parser, 4) { + return false + } + + if parser.mark.column == 0 && + ((parser.buffer[parser.buffer_pos+0] == '-' && + parser.buffer[parser.buffer_pos+1] == '-' && + parser.buffer[parser.buffer_pos+2] == '-') || + (parser.buffer[parser.buffer_pos+0] == '.' && + parser.buffer[parser.buffer_pos+1] == '.' && + parser.buffer[parser.buffer_pos+2] == '.')) && + is_blankz(parser.buffer, parser.buffer_pos+3) { + yaml_parser_set_scanner_error(parser, "while scanning a quoted scalar", + start_mark, "found unexpected document indicator") + return false + } + + // Check for EOF. + if is_z(parser.buffer, parser.buffer_pos) { + yaml_parser_set_scanner_error(parser, "while scanning a quoted scalar", + start_mark, "found unexpected end of stream") + return false + } + + // Consume non-blank characters. + leading_blanks := false + for !is_blankz(parser.buffer, parser.buffer_pos) { + if single && parser.buffer[parser.buffer_pos] == '\'' && parser.buffer[parser.buffer_pos+1] == '\'' { + // Is is an escaped single quote. + s = append(s, '\'') + skip(parser) + skip(parser) + + } else if single && parser.buffer[parser.buffer_pos] == '\'' { + // It is a right single quote. + break + } else if !single && parser.buffer[parser.buffer_pos] == '"' { + // It is a right double quote. + break + + } else if !single && parser.buffer[parser.buffer_pos] == '\\' && is_break(parser.buffer, parser.buffer_pos+1) { + // It is an escaped line break. + if parser.unread < 3 && !yaml_parser_update_buffer(parser, 3) { + return false + } + skip(parser) + skip_line(parser) + leading_blanks = true + break + + } else if !single && parser.buffer[parser.buffer_pos] == '\\' { + // It is an escape sequence. + code_length := 0 + + // Check the escape character. + switch parser.buffer[parser.buffer_pos+1] { + case '0': + s = append(s, 0) + case 'a': + s = append(s, '\x07') + case 'b': + s = append(s, '\x08') + case 't', '\t': + s = append(s, '\x09') + case 'n': + s = append(s, '\x0A') + case 'v': + s = append(s, '\x0B') + case 'f': + s = append(s, '\x0C') + case 'r': + s = append(s, '\x0D') + case 'e': + s = append(s, '\x1B') + case ' ': + s = append(s, '\x20') + case '"': + s = append(s, '"') + case '\'': + s = append(s, '\'') + case '\\': + s = append(s, '\\') + case 'N': // NEL (#x85) + s = append(s, '\xC2') + s = append(s, '\x85') + case '_': // #xA0 + s = append(s, '\xC2') + s = append(s, '\xA0') + case 'L': // LS (#x2028) + s = append(s, '\xE2') + s = append(s, '\x80') + s = append(s, '\xA8') + case 'P': // PS (#x2029) + s = append(s, '\xE2') + s = append(s, '\x80') + s = append(s, '\xA9') + case 'x': + code_length = 2 + case 'u': + code_length = 4 + case 'U': + code_length = 8 + default: + yaml_parser_set_scanner_error(parser, "while parsing a quoted scalar", + start_mark, "found unknown escape character") + return false + } + + skip(parser) + skip(parser) + + // Consume an arbitrary escape code. + if code_length > 0 { + var value int + + // Scan the character value. + if parser.unread < code_length && !yaml_parser_update_buffer(parser, code_length) { + return false + } + for k := 0; k < code_length; k++ { + if !is_hex(parser.buffer, parser.buffer_pos+k) { + yaml_parser_set_scanner_error(parser, "while parsing a quoted scalar", + start_mark, "did not find expected hexdecimal number") + return false + } + value = (value << 4) + as_hex(parser.buffer, parser.buffer_pos+k) + } + + // Check the value and write the character. + if (value >= 0xD800 && value <= 0xDFFF) || value > 0x10FFFF { + yaml_parser_set_scanner_error(parser, "while parsing a quoted scalar", + start_mark, "found invalid Unicode character escape code") + return false + } + if value <= 0x7F { + s = append(s, byte(value)) + } else if value <= 0x7FF { + s = append(s, byte(0xC0+(value>>6))) + s = append(s, byte(0x80+(value&0x3F))) + } else if value <= 0xFFFF { + s = append(s, byte(0xE0+(value>>12))) + s = append(s, byte(0x80+((value>>6)&0x3F))) + s = append(s, byte(0x80+(value&0x3F))) + } else { + s = append(s, byte(0xF0+(value>>18))) + s = append(s, byte(0x80+((value>>12)&0x3F))) + s = append(s, byte(0x80+((value>>6)&0x3F))) + s = append(s, byte(0x80+(value&0x3F))) + } + + // Advance the pointer. + for k := 0; k < code_length; k++ { + skip(parser) + } + } + } else { + // It is a non-escaped non-blank character. + s = read(parser, s) + } + if parser.unread < 2 && !yaml_parser_update_buffer(parser, 2) { + return false + } + } + + if parser.unread < 1 && !yaml_parser_update_buffer(parser, 1) { + return false + } + + // Check if we are at the end of the scalar. + if single { + if parser.buffer[parser.buffer_pos] == '\'' { + break + } + } else { + if parser.buffer[parser.buffer_pos] == '"' { + break + } + } + + // Consume blank characters. + for is_blank(parser.buffer, parser.buffer_pos) || is_break(parser.buffer, parser.buffer_pos) { + if is_blank(parser.buffer, parser.buffer_pos) { + // Consume a space or a tab character. + if !leading_blanks { + whitespaces = read(parser, whitespaces) + } else { + skip(parser) + } + } else { + if parser.unread < 2 && !yaml_parser_update_buffer(parser, 2) { + return false + } + + // Check if it is a first line break. + if !leading_blanks { + whitespaces = whitespaces[:0] + leading_break = read_line(parser, leading_break) + leading_blanks = true + } else { + trailing_breaks = read_line(parser, trailing_breaks) + } + } + if parser.unread < 1 && !yaml_parser_update_buffer(parser, 1) { + return false + } + } + + // Join the whitespaces or fold line breaks. + if leading_blanks { + // Do we need to fold line breaks? + if len(leading_break) > 0 && leading_break[0] == '\n' { + if len(trailing_breaks) == 0 { + s = append(s, ' ') + } else { + s = append(s, trailing_breaks...) + } + } else { + s = append(s, leading_break...) + s = append(s, trailing_breaks...) + } + trailing_breaks = trailing_breaks[:0] + leading_break = leading_break[:0] + } else { + s = append(s, whitespaces...) + whitespaces = whitespaces[:0] + } + } + + // Eat the right quote. + skip(parser) + end_mark := parser.mark + + // Create a token. + *token = yaml_token_t{ + typ: yaml_SCALAR_TOKEN, + start_mark: start_mark, + end_mark: end_mark, + value: s, + style: yaml_SINGLE_QUOTED_SCALAR_STYLE, + } + if !single { + token.style = yaml_DOUBLE_QUOTED_SCALAR_STYLE + } + return true +} + +// Scan a plain scalar. +func yaml_parser_scan_plain_scalar(parser *yaml_parser_t, token *yaml_token_t) bool { + + var s, leading_break, trailing_breaks, whitespaces []byte + var leading_blanks bool + var indent = parser.indent + 1 + + start_mark := parser.mark + end_mark := parser.mark + + // Consume the content of the plain scalar. + for { + // Check for a document indicator. + if parser.unread < 4 && !yaml_parser_update_buffer(parser, 4) { + return false + } + if parser.mark.column == 0 && + ((parser.buffer[parser.buffer_pos+0] == '-' && + parser.buffer[parser.buffer_pos+1] == '-' && + parser.buffer[parser.buffer_pos+2] == '-') || + (parser.buffer[parser.buffer_pos+0] == '.' && + parser.buffer[parser.buffer_pos+1] == '.' && + parser.buffer[parser.buffer_pos+2] == '.')) && + is_blankz(parser.buffer, parser.buffer_pos+3) { + break + } + + // Check for a comment. + if parser.buffer[parser.buffer_pos] == '#' { + break + } + + // Consume non-blank characters. + for !is_blankz(parser.buffer, parser.buffer_pos) { + + // Check for indicators that may end a plain scalar. + if (parser.buffer[parser.buffer_pos] == ':' && is_blankz(parser.buffer, parser.buffer_pos+1)) || + (parser.flow_level > 0 && + (parser.buffer[parser.buffer_pos] == ',' || + parser.buffer[parser.buffer_pos] == '?' || parser.buffer[parser.buffer_pos] == '[' || + parser.buffer[parser.buffer_pos] == ']' || parser.buffer[parser.buffer_pos] == '{' || + parser.buffer[parser.buffer_pos] == '}')) { + break + } + + // Check if we need to join whitespaces and breaks. + if leading_blanks || len(whitespaces) > 0 { + if leading_blanks { + // Do we need to fold line breaks? + if leading_break[0] == '\n' { + if len(trailing_breaks) == 0 { + s = append(s, ' ') + } else { + s = append(s, trailing_breaks...) + } + } else { + s = append(s, leading_break...) + s = append(s, trailing_breaks...) + } + trailing_breaks = trailing_breaks[:0] + leading_break = leading_break[:0] + leading_blanks = false + } else { + s = append(s, whitespaces...) + whitespaces = whitespaces[:0] + } + } + + // Copy the character. + s = read(parser, s) + + end_mark = parser.mark + if parser.unread < 2 && !yaml_parser_update_buffer(parser, 2) { + return false + } + } + + // Is it the end? + if !(is_blank(parser.buffer, parser.buffer_pos) || is_break(parser.buffer, parser.buffer_pos)) { + break + } + + // Consume blank characters. + if parser.unread < 1 && !yaml_parser_update_buffer(parser, 1) { + return false + } + + for is_blank(parser.buffer, parser.buffer_pos) || is_break(parser.buffer, parser.buffer_pos) { + if is_blank(parser.buffer, parser.buffer_pos) { + + // Check for tab characters that abuse indentation. + if leading_blanks && parser.mark.column < indent && is_tab(parser.buffer, parser.buffer_pos) { + yaml_parser_set_scanner_error(parser, "while scanning a plain scalar", + start_mark, "found a tab character that violates indentation") + return false + } + + // Consume a space or a tab character. + if !leading_blanks { + whitespaces = read(parser, whitespaces) + } else { + skip(parser) + } + } else { + if parser.unread < 2 && !yaml_parser_update_buffer(parser, 2) { + return false + } + + // Check if it is a first line break. + if !leading_blanks { + whitespaces = whitespaces[:0] + leading_break = read_line(parser, leading_break) + leading_blanks = true + } else { + trailing_breaks = read_line(parser, trailing_breaks) + } + } + if parser.unread < 1 && !yaml_parser_update_buffer(parser, 1) { + return false + } + } + + // Check indentation level. + if parser.flow_level == 0 && parser.mark.column < indent { + break + } + } + + // Create a token. + *token = yaml_token_t{ + typ: yaml_SCALAR_TOKEN, + start_mark: start_mark, + end_mark: end_mark, + value: s, + style: yaml_PLAIN_SCALAR_STYLE, + } + + // Note that we change the 'simple_key_allowed' flag. + if leading_blanks { + parser.simple_key_allowed = true + } + return true +} + +func yaml_parser_scan_line_comment(parser *yaml_parser_t, token_mark yaml_mark_t) bool { + if parser.newlines > 0 { + return true + } + + var start_mark yaml_mark_t + var text []byte + + for peek := 0; peek < 512; peek++ { + if parser.unread < peek+1 && !yaml_parser_update_buffer(parser, peek+1) { + break + } + if is_blank(parser.buffer, parser.buffer_pos+peek) { + continue + } + if parser.buffer[parser.buffer_pos+peek] == '#' { + seen := parser.mark.index+peek + for { + if parser.unread < 1 && !yaml_parser_update_buffer(parser, 1) { + return false + } + if is_breakz(parser.buffer, parser.buffer_pos) { + if parser.mark.index >= seen { + break + } + if parser.unread < 2 && !yaml_parser_update_buffer(parser, 2) { + return false + } + skip_line(parser) + } else if parser.mark.index >= seen { + if len(text) == 0 { + start_mark = parser.mark + } + text = read(parser, text) + } else { + skip(parser) + } + } + } + break + } + if len(text) > 0 { + parser.comments = append(parser.comments, yaml_comment_t{ + token_mark: token_mark, + start_mark: start_mark, + line: text, + }) + } + return true +} + +func yaml_parser_scan_comments(parser *yaml_parser_t, scan_mark yaml_mark_t) bool { + token := parser.tokens[len(parser.tokens)-1] + + if token.typ == yaml_FLOW_ENTRY_TOKEN && len(parser.tokens) > 1 { + token = parser.tokens[len(parser.tokens)-2] + } + + var token_mark = token.start_mark + var start_mark yaml_mark_t + var next_indent = parser.indent + if next_indent < 0 { + next_indent = 0 + } + + var recent_empty = false + var first_empty = parser.newlines <= 1 + + var line = parser.mark.line + var column = parser.mark.column + + var text []byte + + // The foot line is the place where a comment must start to + // still be considered as a foot of the prior content. + // If there's some content in the currently parsed line, then + // the foot is the line below it. + var foot_line = -1 + if scan_mark.line > 0 { + foot_line = parser.mark.line-parser.newlines+1 + if parser.newlines == 0 && parser.mark.column > 1 { + foot_line++ + } + } + + var peek = 0 + for ; peek < 512; peek++ { + if parser.unread < peek+1 && !yaml_parser_update_buffer(parser, peek+1) { + break + } + column++ + if is_blank(parser.buffer, parser.buffer_pos+peek) { + continue + } + c := parser.buffer[parser.buffer_pos+peek] + var close_flow = parser.flow_level > 0 && (c == ']' || c == '}') + if close_flow || is_breakz(parser.buffer, parser.buffer_pos+peek) { + // Got line break or terminator. + if close_flow || !recent_empty { + if close_flow || first_empty && (start_mark.line == foot_line && token.typ != yaml_VALUE_TOKEN || start_mark.column-1 < next_indent) { + // This is the first empty line and there were no empty lines before, + // so this initial part of the comment is a foot of the prior token + // instead of being a head for the following one. Split it up. + // Alternatively, this might also be the last comment inside a flow + // scope, so it must be a footer. + if len(text) > 0 { + if start_mark.column-1 < next_indent { + // If dedented it's unrelated to the prior token. + token_mark = start_mark + } + parser.comments = append(parser.comments, yaml_comment_t{ + scan_mark: scan_mark, + token_mark: token_mark, + start_mark: start_mark, + end_mark: yaml_mark_t{parser.mark.index + peek, line, column}, + foot: text, + }) + scan_mark = yaml_mark_t{parser.mark.index + peek, line, column} + token_mark = scan_mark + text = nil + } + } else { + if len(text) > 0 && parser.buffer[parser.buffer_pos+peek] != 0 { + text = append(text, '\n') + } + } + } + if !is_break(parser.buffer, parser.buffer_pos+peek) { + break + } + first_empty = false + recent_empty = true + column = 0 + line++ + continue + } + + if len(text) > 0 && (close_flow || column-1 < next_indent && column != start_mark.column) { + // The comment at the different indentation is a foot of the + // preceding data rather than a head of the upcoming one. + parser.comments = append(parser.comments, yaml_comment_t{ + scan_mark: scan_mark, + token_mark: token_mark, + start_mark: start_mark, + end_mark: yaml_mark_t{parser.mark.index + peek, line, column}, + foot: text, + }) + scan_mark = yaml_mark_t{parser.mark.index + peek, line, column} + token_mark = scan_mark + text = nil + } + + if parser.buffer[parser.buffer_pos+peek] != '#' { + break + } + + if len(text) == 0 { + start_mark = yaml_mark_t{parser.mark.index + peek, line, column} + } else { + text = append(text, '\n') + } + + recent_empty = false + + // Consume until after the consumed comment line. + seen := parser.mark.index+peek + for { + if parser.unread < 1 && !yaml_parser_update_buffer(parser, 1) { + return false + } + if is_breakz(parser.buffer, parser.buffer_pos) { + if parser.mark.index >= seen { + break + } + if parser.unread < 2 && !yaml_parser_update_buffer(parser, 2) { + return false + } + skip_line(parser) + } else if parser.mark.index >= seen { + text = read(parser, text) + } else { + skip(parser) + } + } + + peek = 0 + column = 0 + line = parser.mark.line + next_indent = parser.indent + if next_indent < 0 { + next_indent = 0 + } + } + + if len(text) > 0 { + parser.comments = append(parser.comments, yaml_comment_t{ + scan_mark: scan_mark, + token_mark: start_mark, + start_mark: start_mark, + end_mark: yaml_mark_t{parser.mark.index + peek - 1, line, column}, + head: text, + }) + } + return true +} diff --git a/vendor/gopkg.in/yaml.v3/sorter.go b/vendor/gopkg.in/yaml.v3/sorter.go new file mode 100644 index 00000000..9210ece7 --- /dev/null +++ b/vendor/gopkg.in/yaml.v3/sorter.go @@ -0,0 +1,134 @@ +// +// Copyright (c) 2011-2019 Canonical Ltd +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. + +package yaml + +import ( + "reflect" + "unicode" +) + +type keyList []reflect.Value + +func (l keyList) Len() int { return len(l) } +func (l keyList) Swap(i, j int) { l[i], l[j] = l[j], l[i] } +func (l keyList) Less(i, j int) bool { + a := l[i] + b := l[j] + ak := a.Kind() + bk := b.Kind() + for (ak == reflect.Interface || ak == reflect.Ptr) && !a.IsNil() { + a = a.Elem() + ak = a.Kind() + } + for (bk == reflect.Interface || bk == reflect.Ptr) && !b.IsNil() { + b = b.Elem() + bk = b.Kind() + } + af, aok := keyFloat(a) + bf, bok := keyFloat(b) + if aok && bok { + if af != bf { + return af < bf + } + if ak != bk { + return ak < bk + } + return numLess(a, b) + } + if ak != reflect.String || bk != reflect.String { + return ak < bk + } + ar, br := []rune(a.String()), []rune(b.String()) + digits := false + for i := 0; i < len(ar) && i < len(br); i++ { + if ar[i] == br[i] { + digits = unicode.IsDigit(ar[i]) + continue + } + al := unicode.IsLetter(ar[i]) + bl := unicode.IsLetter(br[i]) + if al && bl { + return ar[i] < br[i] + } + if al || bl { + if digits { + return al + } else { + return bl + } + } + var ai, bi int + var an, bn int64 + if ar[i] == '0' || br[i] == '0' { + for j := i - 1; j >= 0 && unicode.IsDigit(ar[j]); j-- { + if ar[j] != '0' { + an = 1 + bn = 1 + break + } + } + } + for ai = i; ai < len(ar) && unicode.IsDigit(ar[ai]); ai++ { + an = an*10 + int64(ar[ai]-'0') + } + for bi = i; bi < len(br) && unicode.IsDigit(br[bi]); bi++ { + bn = bn*10 + int64(br[bi]-'0') + } + if an != bn { + return an < bn + } + if ai != bi { + return ai < bi + } + return ar[i] < br[i] + } + return len(ar) < len(br) +} + +// keyFloat returns a float value for v if it is a number/bool +// and whether it is a number/bool or not. +func keyFloat(v reflect.Value) (f float64, ok bool) { + switch v.Kind() { + case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64: + return float64(v.Int()), true + case reflect.Float32, reflect.Float64: + return v.Float(), true + case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64, reflect.Uintptr: + return float64(v.Uint()), true + case reflect.Bool: + if v.Bool() { + return 1, true + } + return 0, true + } + return 0, false +} + +// numLess returns whether a < b. +// a and b must necessarily have the same kind. +func numLess(a, b reflect.Value) bool { + switch a.Kind() { + case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64: + return a.Int() < b.Int() + case reflect.Float32, reflect.Float64: + return a.Float() < b.Float() + case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64, reflect.Uintptr: + return a.Uint() < b.Uint() + case reflect.Bool: + return !a.Bool() && b.Bool() + } + panic("not a number") +} diff --git a/vendor/gopkg.in/yaml.v3/writerc.go b/vendor/gopkg.in/yaml.v3/writerc.go new file mode 100644 index 00000000..b8a116bf --- /dev/null +++ b/vendor/gopkg.in/yaml.v3/writerc.go @@ -0,0 +1,48 @@ +// +// Copyright (c) 2011-2019 Canonical Ltd +// Copyright (c) 2006-2010 Kirill Simonov +// +// Permission is hereby granted, free of charge, to any person obtaining a copy of +// this software and associated documentation files (the "Software"), to deal in +// the Software without restriction, including without limitation the rights to +// use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies +// of the Software, and to permit persons to whom the Software is furnished to do +// so, subject to the following conditions: +// +// The above copyright notice and this permission notice shall be included in all +// copies or substantial portions of the Software. +// +// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE +// SOFTWARE. + +package yaml + +// Set the writer error and return false. +func yaml_emitter_set_writer_error(emitter *yaml_emitter_t, problem string) bool { + emitter.error = yaml_WRITER_ERROR + emitter.problem = problem + return false +} + +// Flush the output buffer. +func yaml_emitter_flush(emitter *yaml_emitter_t) bool { + if emitter.write_handler == nil { + panic("write handler not set") + } + + // Check if the buffer is empty. + if emitter.buffer_pos == 0 { + return true + } + + if err := emitter.write_handler(emitter, emitter.buffer[:emitter.buffer_pos]); err != nil { + return yaml_emitter_set_writer_error(emitter, "write error: "+err.Error()) + } + emitter.buffer_pos = 0 + return true +} diff --git a/vendor/gopkg.in/yaml.v3/yaml.go b/vendor/gopkg.in/yaml.v3/yaml.go new file mode 100644 index 00000000..8cec6da4 --- /dev/null +++ b/vendor/gopkg.in/yaml.v3/yaml.go @@ -0,0 +1,698 @@ +// +// Copyright (c) 2011-2019 Canonical Ltd +// +// Licensed under the Apache License, Version 2.0 (the "License"); +// you may not use this file except in compliance with the License. +// You may obtain a copy of the License at +// +// http://www.apache.org/licenses/LICENSE-2.0 +// +// Unless required by applicable law or agreed to in writing, software +// distributed under the License is distributed on an "AS IS" BASIS, +// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. +// See the License for the specific language governing permissions and +// limitations under the License. + +// Package yaml implements YAML support for the Go language. +// +// Source code and other details for the project are available at GitHub: +// +// https://github.com/go-yaml/yaml +// +package yaml + +import ( + "errors" + "fmt" + "io" + "reflect" + "strings" + "sync" + "unicode/utf8" +) + +// The Unmarshaler interface may be implemented by types to customize their +// behavior when being unmarshaled from a YAML document. +type Unmarshaler interface { + UnmarshalYAML(value *Node) error +} + +type obsoleteUnmarshaler interface { + UnmarshalYAML(unmarshal func(interface{}) error) error +} + +// The Marshaler interface may be implemented by types to customize their +// behavior when being marshaled into a YAML document. The returned value +// is marshaled in place of the original value implementing Marshaler. +// +// If an error is returned by MarshalYAML, the marshaling procedure stops +// and returns with the provided error. +type Marshaler interface { + MarshalYAML() (interface{}, error) +} + +// Unmarshal decodes the first document found within the in byte slice +// and assigns decoded values into the out value. +// +// Maps and pointers (to a struct, string, int, etc) are accepted as out +// values. If an internal pointer within a struct is not initialized, +// the yaml package will initialize it if necessary for unmarshalling +// the provided data. The out parameter must not be nil. +// +// The type of the decoded values should be compatible with the respective +// values in out. If one or more values cannot be decoded due to a type +// mismatches, decoding continues partially until the end of the YAML +// content, and a *yaml.TypeError is returned with details for all +// missed values. +// +// Struct fields are only unmarshalled if they are exported (have an +// upper case first letter), and are unmarshalled using the field name +// lowercased as the default key. Custom keys may be defined via the +// "yaml" name in the field tag: the content preceding the first comma +// is used as the key, and the following comma-separated options are +// used to tweak the marshalling process (see Marshal). +// Conflicting names result in a runtime error. +// +// For example: +// +// type T struct { +// F int `yaml:"a,omitempty"` +// B int +// } +// var t T +// yaml.Unmarshal([]byte("a: 1\nb: 2"), &t) +// +// See the documentation of Marshal for the format of tags and a list of +// supported tag options. +// +func Unmarshal(in []byte, out interface{}) (err error) { + return unmarshal(in, out, false) +} + +// A Decoder reads and decodes YAML values from an input stream. +type Decoder struct { + parser *parser + knownFields bool +} + +// NewDecoder returns a new decoder that reads from r. +// +// The decoder introduces its own buffering and may read +// data from r beyond the YAML values requested. +func NewDecoder(r io.Reader) *Decoder { + return &Decoder{ + parser: newParserFromReader(r), + } +} + +// KnownFields ensures that the keys in decoded mappings to +// exist as fields in the struct being decoded into. +func (dec *Decoder) KnownFields(enable bool) { + dec.knownFields = enable +} + +// Decode reads the next YAML-encoded value from its input +// and stores it in the value pointed to by v. +// +// See the documentation for Unmarshal for details about the +// conversion of YAML into a Go value. +func (dec *Decoder) Decode(v interface{}) (err error) { + d := newDecoder() + d.knownFields = dec.knownFields + defer handleErr(&err) + node := dec.parser.parse() + if node == nil { + return io.EOF + } + out := reflect.ValueOf(v) + if out.Kind() == reflect.Ptr && !out.IsNil() { + out = out.Elem() + } + d.unmarshal(node, out) + if len(d.terrors) > 0 { + return &TypeError{d.terrors} + } + return nil +} + +// Decode decodes the node and stores its data into the value pointed to by v. +// +// See the documentation for Unmarshal for details about the +// conversion of YAML into a Go value. +func (n *Node) Decode(v interface{}) (err error) { + d := newDecoder() + defer handleErr(&err) + out := reflect.ValueOf(v) + if out.Kind() == reflect.Ptr && !out.IsNil() { + out = out.Elem() + } + d.unmarshal(n, out) + if len(d.terrors) > 0 { + return &TypeError{d.terrors} + } + return nil +} + +func unmarshal(in []byte, out interface{}, strict bool) (err error) { + defer handleErr(&err) + d := newDecoder() + p := newParser(in) + defer p.destroy() + node := p.parse() + if node != nil { + v := reflect.ValueOf(out) + if v.Kind() == reflect.Ptr && !v.IsNil() { + v = v.Elem() + } + d.unmarshal(node, v) + } + if len(d.terrors) > 0 { + return &TypeError{d.terrors} + } + return nil +} + +// Marshal serializes the value provided into a YAML document. The structure +// of the generated document will reflect the structure of the value itself. +// Maps and pointers (to struct, string, int, etc) are accepted as the in value. +// +// Struct fields are only marshalled if they are exported (have an upper case +// first letter), and are marshalled using the field name lowercased as the +// default key. Custom keys may be defined via the "yaml" name in the field +// tag: the content preceding the first comma is used as the key, and the +// following comma-separated options are used to tweak the marshalling process. +// Conflicting names result in a runtime error. +// +// The field tag format accepted is: +// +// `(...) yaml:"[][,[,]]" (...)` +// +// The following flags are currently supported: +// +// omitempty Only include the field if it's not set to the zero +// value for the type or to empty slices or maps. +// Zero valued structs will be omitted if all their public +// fields are zero, unless they implement an IsZero +// method (see the IsZeroer interface type), in which +// case the field will be excluded if IsZero returns true. +// +// flow Marshal using a flow style (useful for structs, +// sequences and maps). +// +// inline Inline the field, which must be a struct or a map, +// causing all of its fields or keys to be processed as if +// they were part of the outer struct. For maps, keys must +// not conflict with the yaml keys of other struct fields. +// +// In addition, if the key is "-", the field is ignored. +// +// For example: +// +// type T struct { +// F int `yaml:"a,omitempty"` +// B int +// } +// yaml.Marshal(&T{B: 2}) // Returns "b: 2\n" +// yaml.Marshal(&T{F: 1}} // Returns "a: 1\nb: 0\n" +// +func Marshal(in interface{}) (out []byte, err error) { + defer handleErr(&err) + e := newEncoder() + defer e.destroy() + e.marshalDoc("", reflect.ValueOf(in)) + e.finish() + out = e.out + return +} + +// An Encoder writes YAML values to an output stream. +type Encoder struct { + encoder *encoder +} + +// NewEncoder returns a new encoder that writes to w. +// The Encoder should be closed after use to flush all data +// to w. +func NewEncoder(w io.Writer) *Encoder { + return &Encoder{ + encoder: newEncoderWithWriter(w), + } +} + +// Encode writes the YAML encoding of v to the stream. +// If multiple items are encoded to the stream, the +// second and subsequent document will be preceded +// with a "---" document separator, but the first will not. +// +// See the documentation for Marshal for details about the conversion of Go +// values to YAML. +func (e *Encoder) Encode(v interface{}) (err error) { + defer handleErr(&err) + e.encoder.marshalDoc("", reflect.ValueOf(v)) + return nil +} + +// Encode encodes value v and stores its representation in n. +// +// See the documentation for Marshal for details about the +// conversion of Go values into YAML. +func (n *Node) Encode(v interface{}) (err error) { + defer handleErr(&err) + e := newEncoder() + defer e.destroy() + e.marshalDoc("", reflect.ValueOf(v)) + e.finish() + p := newParser(e.out) + p.textless = true + defer p.destroy() + doc := p.parse() + *n = *doc.Content[0] + return nil +} + +// SetIndent changes the used indentation used when encoding. +func (e *Encoder) SetIndent(spaces int) { + if spaces < 0 { + panic("yaml: cannot indent to a negative number of spaces") + } + e.encoder.indent = spaces +} + +// Close closes the encoder by writing any remaining data. +// It does not write a stream terminating string "...". +func (e *Encoder) Close() (err error) { + defer handleErr(&err) + e.encoder.finish() + return nil +} + +func handleErr(err *error) { + if v := recover(); v != nil { + if e, ok := v.(yamlError); ok { + *err = e.err + } else { + panic(v) + } + } +} + +type yamlError struct { + err error +} + +func fail(err error) { + panic(yamlError{err}) +} + +func failf(format string, args ...interface{}) { + panic(yamlError{fmt.Errorf("yaml: "+format, args...)}) +} + +// A TypeError is returned by Unmarshal when one or more fields in +// the YAML document cannot be properly decoded into the requested +// types. When this error is returned, the value is still +// unmarshaled partially. +type TypeError struct { + Errors []string +} + +func (e *TypeError) Error() string { + return fmt.Sprintf("yaml: unmarshal errors:\n %s", strings.Join(e.Errors, "\n ")) +} + +type Kind uint32 + +const ( + DocumentNode Kind = 1 << iota + SequenceNode + MappingNode + ScalarNode + AliasNode +) + +type Style uint32 + +const ( + TaggedStyle Style = 1 << iota + DoubleQuotedStyle + SingleQuotedStyle + LiteralStyle + FoldedStyle + FlowStyle +) + +// Node represents an element in the YAML document hierarchy. While documents +// are typically encoded and decoded into higher level types, such as structs +// and maps, Node is an intermediate representation that allows detailed +// control over the content being decoded or encoded. +// +// It's worth noting that although Node offers access into details such as +// line numbers, colums, and comments, the content when re-encoded will not +// have its original textual representation preserved. An effort is made to +// render the data plesantly, and to preserve comments near the data they +// describe, though. +// +// Values that make use of the Node type interact with the yaml package in the +// same way any other type would do, by encoding and decoding yaml data +// directly or indirectly into them. +// +// For example: +// +// var person struct { +// Name string +// Address yaml.Node +// } +// err := yaml.Unmarshal(data, &person) +// +// Or by itself: +// +// var person Node +// err := yaml.Unmarshal(data, &person) +// +type Node struct { + // Kind defines whether the node is a document, a mapping, a sequence, + // a scalar value, or an alias to another node. The specific data type of + // scalar nodes may be obtained via the ShortTag and LongTag methods. + Kind Kind + + // Style allows customizing the apperance of the node in the tree. + Style Style + + // Tag holds the YAML tag defining the data type for the value. + // When decoding, this field will always be set to the resolved tag, + // even when it wasn't explicitly provided in the YAML content. + // When encoding, if this field is unset the value type will be + // implied from the node properties, and if it is set, it will only + // be serialized into the representation if TaggedStyle is used or + // the implicit tag diverges from the provided one. + Tag string + + // Value holds the unescaped and unquoted represenation of the value. + Value string + + // Anchor holds the anchor name for this node, which allows aliases to point to it. + Anchor string + + // Alias holds the node that this alias points to. Only valid when Kind is AliasNode. + Alias *Node + + // Content holds contained nodes for documents, mappings, and sequences. + Content []*Node + + // HeadComment holds any comments in the lines preceding the node and + // not separated by an empty line. + HeadComment string + + // LineComment holds any comments at the end of the line where the node is in. + LineComment string + + // FootComment holds any comments following the node and before empty lines. + FootComment string + + // Line and Column hold the node position in the decoded YAML text. + // These fields are not respected when encoding the node. + Line int + Column int +} + +// IsZero returns whether the node has all of its fields unset. +func (n *Node) IsZero() bool { + return n.Kind == 0 && n.Style == 0 && n.Tag == "" && n.Value == "" && n.Anchor == "" && n.Alias == nil && n.Content == nil && + n.HeadComment == "" && n.LineComment == "" && n.FootComment == "" && n.Line == 0 && n.Column == 0 +} + + +// LongTag returns the long form of the tag that indicates the data type for +// the node. If the Tag field isn't explicitly defined, one will be computed +// based on the node properties. +func (n *Node) LongTag() string { + return longTag(n.ShortTag()) +} + +// ShortTag returns the short form of the YAML tag that indicates data type for +// the node. If the Tag field isn't explicitly defined, one will be computed +// based on the node properties. +func (n *Node) ShortTag() string { + if n.indicatedString() { + return strTag + } + if n.Tag == "" || n.Tag == "!" { + switch n.Kind { + case MappingNode: + return mapTag + case SequenceNode: + return seqTag + case AliasNode: + if n.Alias != nil { + return n.Alias.ShortTag() + } + case ScalarNode: + tag, _ := resolve("", n.Value) + return tag + case 0: + // Special case to make the zero value convenient. + if n.IsZero() { + return nullTag + } + } + return "" + } + return shortTag(n.Tag) +} + +func (n *Node) indicatedString() bool { + return n.Kind == ScalarNode && + (shortTag(n.Tag) == strTag || + (n.Tag == "" || n.Tag == "!") && n.Style&(SingleQuotedStyle|DoubleQuotedStyle|LiteralStyle|FoldedStyle) != 0) +} + +// SetString is a convenience function that sets the node to a string value +// and defines its style in a pleasant way depending on its content. +func (n *Node) SetString(s string) { + n.Kind = ScalarNode + if utf8.ValidString(s) { + n.Value = s + n.Tag = strTag + } else { + n.Value = encodeBase64(s) + n.Tag = binaryTag + } + if strings.Contains(n.Value, "\n") { + n.Style = LiteralStyle + } +} + +// -------------------------------------------------------------------------- +// Maintain a mapping of keys to structure field indexes + +// The code in this section was copied from mgo/bson. + +// structInfo holds details for the serialization of fields of +// a given struct. +type structInfo struct { + FieldsMap map[string]fieldInfo + FieldsList []fieldInfo + + // InlineMap is the number of the field in the struct that + // contains an ,inline map, or -1 if there's none. + InlineMap int + + // InlineUnmarshalers holds indexes to inlined fields that + // contain unmarshaler values. + InlineUnmarshalers [][]int +} + +type fieldInfo struct { + Key string + Num int + OmitEmpty bool + Flow bool + // Id holds the unique field identifier, so we can cheaply + // check for field duplicates without maintaining an extra map. + Id int + + // Inline holds the field index if the field is part of an inlined struct. + Inline []int +} + +var structMap = make(map[reflect.Type]*structInfo) +var fieldMapMutex sync.RWMutex +var unmarshalerType reflect.Type + +func init() { + var v Unmarshaler + unmarshalerType = reflect.ValueOf(&v).Elem().Type() +} + +func getStructInfo(st reflect.Type) (*structInfo, error) { + fieldMapMutex.RLock() + sinfo, found := structMap[st] + fieldMapMutex.RUnlock() + if found { + return sinfo, nil + } + + n := st.NumField() + fieldsMap := make(map[string]fieldInfo) + fieldsList := make([]fieldInfo, 0, n) + inlineMap := -1 + inlineUnmarshalers := [][]int(nil) + for i := 0; i != n; i++ { + field := st.Field(i) + if field.PkgPath != "" && !field.Anonymous { + continue // Private field + } + + info := fieldInfo{Num: i} + + tag := field.Tag.Get("yaml") + if tag == "" && strings.Index(string(field.Tag), ":") < 0 { + tag = string(field.Tag) + } + if tag == "-" { + continue + } + + inline := false + fields := strings.Split(tag, ",") + if len(fields) > 1 { + for _, flag := range fields[1:] { + switch flag { + case "omitempty": + info.OmitEmpty = true + case "flow": + info.Flow = true + case "inline": + inline = true + default: + return nil, errors.New(fmt.Sprintf("unsupported flag %q in tag %q of type %s", flag, tag, st)) + } + } + tag = fields[0] + } + + if inline { + switch field.Type.Kind() { + case reflect.Map: + if inlineMap >= 0 { + return nil, errors.New("multiple ,inline maps in struct " + st.String()) + } + if field.Type.Key() != reflect.TypeOf("") { + return nil, errors.New("option ,inline needs a map with string keys in struct " + st.String()) + } + inlineMap = info.Num + case reflect.Struct, reflect.Ptr: + ftype := field.Type + for ftype.Kind() == reflect.Ptr { + ftype = ftype.Elem() + } + if ftype.Kind() != reflect.Struct { + return nil, errors.New("option ,inline may only be used on a struct or map field") + } + if reflect.PtrTo(ftype).Implements(unmarshalerType) { + inlineUnmarshalers = append(inlineUnmarshalers, []int{i}) + } else { + sinfo, err := getStructInfo(ftype) + if err != nil { + return nil, err + } + for _, index := range sinfo.InlineUnmarshalers { + inlineUnmarshalers = append(inlineUnmarshalers, append([]int{i}, index...)) + } + for _, finfo := range sinfo.FieldsList { + if _, found := fieldsMap[finfo.Key]; found { + msg := "duplicated key '" + finfo.Key + "' in struct " + st.String() + return nil, errors.New(msg) + } + if finfo.Inline == nil { + finfo.Inline = []int{i, finfo.Num} + } else { + finfo.Inline = append([]int{i}, finfo.Inline...) + } + finfo.Id = len(fieldsList) + fieldsMap[finfo.Key] = finfo + fieldsList = append(fieldsList, finfo) + } + } + default: + return nil, errors.New("option ,inline may only be used on a struct or map field") + } + continue + } + + if tag != "" { + info.Key = tag + } else { + info.Key = strings.ToLower(field.Name) + } + + if _, found = fieldsMap[info.Key]; found { + msg := "duplicated key '" + info.Key + "' in struct " + st.String() + return nil, errors.New(msg) + } + + info.Id = len(fieldsList) + fieldsList = append(fieldsList, info) + fieldsMap[info.Key] = info + } + + sinfo = &structInfo{ + FieldsMap: fieldsMap, + FieldsList: fieldsList, + InlineMap: inlineMap, + InlineUnmarshalers: inlineUnmarshalers, + } + + fieldMapMutex.Lock() + structMap[st] = sinfo + fieldMapMutex.Unlock() + return sinfo, nil +} + +// IsZeroer is used to check whether an object is zero to +// determine whether it should be omitted when marshaling +// with the omitempty flag. One notable implementation +// is time.Time. +type IsZeroer interface { + IsZero() bool +} + +func isZero(v reflect.Value) bool { + kind := v.Kind() + if z, ok := v.Interface().(IsZeroer); ok { + if (kind == reflect.Ptr || kind == reflect.Interface) && v.IsNil() { + return true + } + return z.IsZero() + } + switch kind { + case reflect.String: + return len(v.String()) == 0 + case reflect.Interface, reflect.Ptr: + return v.IsNil() + case reflect.Slice: + return v.Len() == 0 + case reflect.Map: + return v.Len() == 0 + case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64: + return v.Int() == 0 + case reflect.Float32, reflect.Float64: + return v.Float() == 0 + case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64, reflect.Uintptr: + return v.Uint() == 0 + case reflect.Bool: + return !v.Bool() + case reflect.Struct: + vt := v.Type() + for i := v.NumField() - 1; i >= 0; i-- { + if vt.Field(i).PkgPath != "" { + continue // Private field + } + if !isZero(v.Field(i)) { + return false + } + } + return true + } + return false +} diff --git a/vendor/gopkg.in/yaml.v3/yamlh.go b/vendor/gopkg.in/yaml.v3/yamlh.go new file mode 100644 index 00000000..7c6d0077 --- /dev/null +++ b/vendor/gopkg.in/yaml.v3/yamlh.go @@ -0,0 +1,807 @@ +// +// Copyright (c) 2011-2019 Canonical Ltd +// Copyright (c) 2006-2010 Kirill Simonov +// +// Permission is hereby granted, free of charge, to any person obtaining a copy of +// this software and associated documentation files (the "Software"), to deal in +// the Software without restriction, including without limitation the rights to +// use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies +// of the Software, and to permit persons to whom the Software is furnished to do +// so, subject to the following conditions: +// +// The above copyright notice and this permission notice shall be included in all +// copies or substantial portions of the Software. +// +// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE +// SOFTWARE. + +package yaml + +import ( + "fmt" + "io" +) + +// The version directive data. +type yaml_version_directive_t struct { + major int8 // The major version number. + minor int8 // The minor version number. +} + +// The tag directive data. +type yaml_tag_directive_t struct { + handle []byte // The tag handle. + prefix []byte // The tag prefix. +} + +type yaml_encoding_t int + +// The stream encoding. +const ( + // Let the parser choose the encoding. + yaml_ANY_ENCODING yaml_encoding_t = iota + + yaml_UTF8_ENCODING // The default UTF-8 encoding. + yaml_UTF16LE_ENCODING // The UTF-16-LE encoding with BOM. + yaml_UTF16BE_ENCODING // The UTF-16-BE encoding with BOM. +) + +type yaml_break_t int + +// Line break types. +const ( + // Let the parser choose the break type. + yaml_ANY_BREAK yaml_break_t = iota + + yaml_CR_BREAK // Use CR for line breaks (Mac style). + yaml_LN_BREAK // Use LN for line breaks (Unix style). + yaml_CRLN_BREAK // Use CR LN for line breaks (DOS style). +) + +type yaml_error_type_t int + +// Many bad things could happen with the parser and emitter. +const ( + // No error is produced. + yaml_NO_ERROR yaml_error_type_t = iota + + yaml_MEMORY_ERROR // Cannot allocate or reallocate a block of memory. + yaml_READER_ERROR // Cannot read or decode the input stream. + yaml_SCANNER_ERROR // Cannot scan the input stream. + yaml_PARSER_ERROR // Cannot parse the input stream. + yaml_COMPOSER_ERROR // Cannot compose a YAML document. + yaml_WRITER_ERROR // Cannot write to the output stream. + yaml_EMITTER_ERROR // Cannot emit a YAML stream. +) + +// The pointer position. +type yaml_mark_t struct { + index int // The position index. + line int // The position line. + column int // The position column. +} + +// Node Styles + +type yaml_style_t int8 + +type yaml_scalar_style_t yaml_style_t + +// Scalar styles. +const ( + // Let the emitter choose the style. + yaml_ANY_SCALAR_STYLE yaml_scalar_style_t = 0 + + yaml_PLAIN_SCALAR_STYLE yaml_scalar_style_t = 1 << iota // The plain scalar style. + yaml_SINGLE_QUOTED_SCALAR_STYLE // The single-quoted scalar style. + yaml_DOUBLE_QUOTED_SCALAR_STYLE // The double-quoted scalar style. + yaml_LITERAL_SCALAR_STYLE // The literal scalar style. + yaml_FOLDED_SCALAR_STYLE // The folded scalar style. +) + +type yaml_sequence_style_t yaml_style_t + +// Sequence styles. +const ( + // Let the emitter choose the style. + yaml_ANY_SEQUENCE_STYLE yaml_sequence_style_t = iota + + yaml_BLOCK_SEQUENCE_STYLE // The block sequence style. + yaml_FLOW_SEQUENCE_STYLE // The flow sequence style. +) + +type yaml_mapping_style_t yaml_style_t + +// Mapping styles. +const ( + // Let the emitter choose the style. + yaml_ANY_MAPPING_STYLE yaml_mapping_style_t = iota + + yaml_BLOCK_MAPPING_STYLE // The block mapping style. + yaml_FLOW_MAPPING_STYLE // The flow mapping style. +) + +// Tokens + +type yaml_token_type_t int + +// Token types. +const ( + // An empty token. + yaml_NO_TOKEN yaml_token_type_t = iota + + yaml_STREAM_START_TOKEN // A STREAM-START token. + yaml_STREAM_END_TOKEN // A STREAM-END token. + + yaml_VERSION_DIRECTIVE_TOKEN // A VERSION-DIRECTIVE token. + yaml_TAG_DIRECTIVE_TOKEN // A TAG-DIRECTIVE token. + yaml_DOCUMENT_START_TOKEN // A DOCUMENT-START token. + yaml_DOCUMENT_END_TOKEN // A DOCUMENT-END token. + + yaml_BLOCK_SEQUENCE_START_TOKEN // A BLOCK-SEQUENCE-START token. + yaml_BLOCK_MAPPING_START_TOKEN // A BLOCK-SEQUENCE-END token. + yaml_BLOCK_END_TOKEN // A BLOCK-END token. + + yaml_FLOW_SEQUENCE_START_TOKEN // A FLOW-SEQUENCE-START token. + yaml_FLOW_SEQUENCE_END_TOKEN // A FLOW-SEQUENCE-END token. + yaml_FLOW_MAPPING_START_TOKEN // A FLOW-MAPPING-START token. + yaml_FLOW_MAPPING_END_TOKEN // A FLOW-MAPPING-END token. + + yaml_BLOCK_ENTRY_TOKEN // A BLOCK-ENTRY token. + yaml_FLOW_ENTRY_TOKEN // A FLOW-ENTRY token. + yaml_KEY_TOKEN // A KEY token. + yaml_VALUE_TOKEN // A VALUE token. + + yaml_ALIAS_TOKEN // An ALIAS token. + yaml_ANCHOR_TOKEN // An ANCHOR token. + yaml_TAG_TOKEN // A TAG token. + yaml_SCALAR_TOKEN // A SCALAR token. +) + +func (tt yaml_token_type_t) String() string { + switch tt { + case yaml_NO_TOKEN: + return "yaml_NO_TOKEN" + case yaml_STREAM_START_TOKEN: + return "yaml_STREAM_START_TOKEN" + case yaml_STREAM_END_TOKEN: + return "yaml_STREAM_END_TOKEN" + case yaml_VERSION_DIRECTIVE_TOKEN: + return "yaml_VERSION_DIRECTIVE_TOKEN" + case yaml_TAG_DIRECTIVE_TOKEN: + return "yaml_TAG_DIRECTIVE_TOKEN" + case yaml_DOCUMENT_START_TOKEN: + return "yaml_DOCUMENT_START_TOKEN" + case yaml_DOCUMENT_END_TOKEN: + return "yaml_DOCUMENT_END_TOKEN" + case yaml_BLOCK_SEQUENCE_START_TOKEN: + return "yaml_BLOCK_SEQUENCE_START_TOKEN" + case yaml_BLOCK_MAPPING_START_TOKEN: + return "yaml_BLOCK_MAPPING_START_TOKEN" + case yaml_BLOCK_END_TOKEN: + return "yaml_BLOCK_END_TOKEN" + case yaml_FLOW_SEQUENCE_START_TOKEN: + return "yaml_FLOW_SEQUENCE_START_TOKEN" + case yaml_FLOW_SEQUENCE_END_TOKEN: + return "yaml_FLOW_SEQUENCE_END_TOKEN" + case yaml_FLOW_MAPPING_START_TOKEN: + return "yaml_FLOW_MAPPING_START_TOKEN" + case yaml_FLOW_MAPPING_END_TOKEN: + return "yaml_FLOW_MAPPING_END_TOKEN" + case yaml_BLOCK_ENTRY_TOKEN: + return "yaml_BLOCK_ENTRY_TOKEN" + case yaml_FLOW_ENTRY_TOKEN: + return "yaml_FLOW_ENTRY_TOKEN" + case yaml_KEY_TOKEN: + return "yaml_KEY_TOKEN" + case yaml_VALUE_TOKEN: + return "yaml_VALUE_TOKEN" + case yaml_ALIAS_TOKEN: + return "yaml_ALIAS_TOKEN" + case yaml_ANCHOR_TOKEN: + return "yaml_ANCHOR_TOKEN" + case yaml_TAG_TOKEN: + return "yaml_TAG_TOKEN" + case yaml_SCALAR_TOKEN: + return "yaml_SCALAR_TOKEN" + } + return "" +} + +// The token structure. +type yaml_token_t struct { + // The token type. + typ yaml_token_type_t + + // The start/end of the token. + start_mark, end_mark yaml_mark_t + + // The stream encoding (for yaml_STREAM_START_TOKEN). + encoding yaml_encoding_t + + // The alias/anchor/scalar value or tag/tag directive handle + // (for yaml_ALIAS_TOKEN, yaml_ANCHOR_TOKEN, yaml_SCALAR_TOKEN, yaml_TAG_TOKEN, yaml_TAG_DIRECTIVE_TOKEN). + value []byte + + // The tag suffix (for yaml_TAG_TOKEN). + suffix []byte + + // The tag directive prefix (for yaml_TAG_DIRECTIVE_TOKEN). + prefix []byte + + // The scalar style (for yaml_SCALAR_TOKEN). + style yaml_scalar_style_t + + // The version directive major/minor (for yaml_VERSION_DIRECTIVE_TOKEN). + major, minor int8 +} + +// Events + +type yaml_event_type_t int8 + +// Event types. +const ( + // An empty event. + yaml_NO_EVENT yaml_event_type_t = iota + + yaml_STREAM_START_EVENT // A STREAM-START event. + yaml_STREAM_END_EVENT // A STREAM-END event. + yaml_DOCUMENT_START_EVENT // A DOCUMENT-START event. + yaml_DOCUMENT_END_EVENT // A DOCUMENT-END event. + yaml_ALIAS_EVENT // An ALIAS event. + yaml_SCALAR_EVENT // A SCALAR event. + yaml_SEQUENCE_START_EVENT // A SEQUENCE-START event. + yaml_SEQUENCE_END_EVENT // A SEQUENCE-END event. + yaml_MAPPING_START_EVENT // A MAPPING-START event. + yaml_MAPPING_END_EVENT // A MAPPING-END event. + yaml_TAIL_COMMENT_EVENT +) + +var eventStrings = []string{ + yaml_NO_EVENT: "none", + yaml_STREAM_START_EVENT: "stream start", + yaml_STREAM_END_EVENT: "stream end", + yaml_DOCUMENT_START_EVENT: "document start", + yaml_DOCUMENT_END_EVENT: "document end", + yaml_ALIAS_EVENT: "alias", + yaml_SCALAR_EVENT: "scalar", + yaml_SEQUENCE_START_EVENT: "sequence start", + yaml_SEQUENCE_END_EVENT: "sequence end", + yaml_MAPPING_START_EVENT: "mapping start", + yaml_MAPPING_END_EVENT: "mapping end", + yaml_TAIL_COMMENT_EVENT: "tail comment", +} + +func (e yaml_event_type_t) String() string { + if e < 0 || int(e) >= len(eventStrings) { + return fmt.Sprintf("unknown event %d", e) + } + return eventStrings[e] +} + +// The event structure. +type yaml_event_t struct { + + // The event type. + typ yaml_event_type_t + + // The start and end of the event. + start_mark, end_mark yaml_mark_t + + // The document encoding (for yaml_STREAM_START_EVENT). + encoding yaml_encoding_t + + // The version directive (for yaml_DOCUMENT_START_EVENT). + version_directive *yaml_version_directive_t + + // The list of tag directives (for yaml_DOCUMENT_START_EVENT). + tag_directives []yaml_tag_directive_t + + // The comments + head_comment []byte + line_comment []byte + foot_comment []byte + tail_comment []byte + + // The anchor (for yaml_SCALAR_EVENT, yaml_SEQUENCE_START_EVENT, yaml_MAPPING_START_EVENT, yaml_ALIAS_EVENT). + anchor []byte + + // The tag (for yaml_SCALAR_EVENT, yaml_SEQUENCE_START_EVENT, yaml_MAPPING_START_EVENT). + tag []byte + + // The scalar value (for yaml_SCALAR_EVENT). + value []byte + + // Is the document start/end indicator implicit, or the tag optional? + // (for yaml_DOCUMENT_START_EVENT, yaml_DOCUMENT_END_EVENT, yaml_SEQUENCE_START_EVENT, yaml_MAPPING_START_EVENT, yaml_SCALAR_EVENT). + implicit bool + + // Is the tag optional for any non-plain style? (for yaml_SCALAR_EVENT). + quoted_implicit bool + + // The style (for yaml_SCALAR_EVENT, yaml_SEQUENCE_START_EVENT, yaml_MAPPING_START_EVENT). + style yaml_style_t +} + +func (e *yaml_event_t) scalar_style() yaml_scalar_style_t { return yaml_scalar_style_t(e.style) } +func (e *yaml_event_t) sequence_style() yaml_sequence_style_t { return yaml_sequence_style_t(e.style) } +func (e *yaml_event_t) mapping_style() yaml_mapping_style_t { return yaml_mapping_style_t(e.style) } + +// Nodes + +const ( + yaml_NULL_TAG = "tag:yaml.org,2002:null" // The tag !!null with the only possible value: null. + yaml_BOOL_TAG = "tag:yaml.org,2002:bool" // The tag !!bool with the values: true and false. + yaml_STR_TAG = "tag:yaml.org,2002:str" // The tag !!str for string values. + yaml_INT_TAG = "tag:yaml.org,2002:int" // The tag !!int for integer values. + yaml_FLOAT_TAG = "tag:yaml.org,2002:float" // The tag !!float for float values. + yaml_TIMESTAMP_TAG = "tag:yaml.org,2002:timestamp" // The tag !!timestamp for date and time values. + + yaml_SEQ_TAG = "tag:yaml.org,2002:seq" // The tag !!seq is used to denote sequences. + yaml_MAP_TAG = "tag:yaml.org,2002:map" // The tag !!map is used to denote mapping. + + // Not in original libyaml. + yaml_BINARY_TAG = "tag:yaml.org,2002:binary" + yaml_MERGE_TAG = "tag:yaml.org,2002:merge" + + yaml_DEFAULT_SCALAR_TAG = yaml_STR_TAG // The default scalar tag is !!str. + yaml_DEFAULT_SEQUENCE_TAG = yaml_SEQ_TAG // The default sequence tag is !!seq. + yaml_DEFAULT_MAPPING_TAG = yaml_MAP_TAG // The default mapping tag is !!map. +) + +type yaml_node_type_t int + +// Node types. +const ( + // An empty node. + yaml_NO_NODE yaml_node_type_t = iota + + yaml_SCALAR_NODE // A scalar node. + yaml_SEQUENCE_NODE // A sequence node. + yaml_MAPPING_NODE // A mapping node. +) + +// An element of a sequence node. +type yaml_node_item_t int + +// An element of a mapping node. +type yaml_node_pair_t struct { + key int // The key of the element. + value int // The value of the element. +} + +// The node structure. +type yaml_node_t struct { + typ yaml_node_type_t // The node type. + tag []byte // The node tag. + + // The node data. + + // The scalar parameters (for yaml_SCALAR_NODE). + scalar struct { + value []byte // The scalar value. + length int // The length of the scalar value. + style yaml_scalar_style_t // The scalar style. + } + + // The sequence parameters (for YAML_SEQUENCE_NODE). + sequence struct { + items_data []yaml_node_item_t // The stack of sequence items. + style yaml_sequence_style_t // The sequence style. + } + + // The mapping parameters (for yaml_MAPPING_NODE). + mapping struct { + pairs_data []yaml_node_pair_t // The stack of mapping pairs (key, value). + pairs_start *yaml_node_pair_t // The beginning of the stack. + pairs_end *yaml_node_pair_t // The end of the stack. + pairs_top *yaml_node_pair_t // The top of the stack. + style yaml_mapping_style_t // The mapping style. + } + + start_mark yaml_mark_t // The beginning of the node. + end_mark yaml_mark_t // The end of the node. + +} + +// The document structure. +type yaml_document_t struct { + + // The document nodes. + nodes []yaml_node_t + + // The version directive. + version_directive *yaml_version_directive_t + + // The list of tag directives. + tag_directives_data []yaml_tag_directive_t + tag_directives_start int // The beginning of the tag directives list. + tag_directives_end int // The end of the tag directives list. + + start_implicit int // Is the document start indicator implicit? + end_implicit int // Is the document end indicator implicit? + + // The start/end of the document. + start_mark, end_mark yaml_mark_t +} + +// The prototype of a read handler. +// +// The read handler is called when the parser needs to read more bytes from the +// source. The handler should write not more than size bytes to the buffer. +// The number of written bytes should be set to the size_read variable. +// +// [in,out] data A pointer to an application data specified by +// yaml_parser_set_input(). +// [out] buffer The buffer to write the data from the source. +// [in] size The size of the buffer. +// [out] size_read The actual number of bytes read from the source. +// +// On success, the handler should return 1. If the handler failed, +// the returned value should be 0. On EOF, the handler should set the +// size_read to 0 and return 1. +type yaml_read_handler_t func(parser *yaml_parser_t, buffer []byte) (n int, err error) + +// This structure holds information about a potential simple key. +type yaml_simple_key_t struct { + possible bool // Is a simple key possible? + required bool // Is a simple key required? + token_number int // The number of the token. + mark yaml_mark_t // The position mark. +} + +// The states of the parser. +type yaml_parser_state_t int + +const ( + yaml_PARSE_STREAM_START_STATE yaml_parser_state_t = iota + + yaml_PARSE_IMPLICIT_DOCUMENT_START_STATE // Expect the beginning of an implicit document. + yaml_PARSE_DOCUMENT_START_STATE // Expect DOCUMENT-START. + yaml_PARSE_DOCUMENT_CONTENT_STATE // Expect the content of a document. + yaml_PARSE_DOCUMENT_END_STATE // Expect DOCUMENT-END. + yaml_PARSE_BLOCK_NODE_STATE // Expect a block node. + yaml_PARSE_BLOCK_NODE_OR_INDENTLESS_SEQUENCE_STATE // Expect a block node or indentless sequence. + yaml_PARSE_FLOW_NODE_STATE // Expect a flow node. + yaml_PARSE_BLOCK_SEQUENCE_FIRST_ENTRY_STATE // Expect the first entry of a block sequence. + yaml_PARSE_BLOCK_SEQUENCE_ENTRY_STATE // Expect an entry of a block sequence. + yaml_PARSE_INDENTLESS_SEQUENCE_ENTRY_STATE // Expect an entry of an indentless sequence. + yaml_PARSE_BLOCK_MAPPING_FIRST_KEY_STATE // Expect the first key of a block mapping. + yaml_PARSE_BLOCK_MAPPING_KEY_STATE // Expect a block mapping key. + yaml_PARSE_BLOCK_MAPPING_VALUE_STATE // Expect a block mapping value. + yaml_PARSE_FLOW_SEQUENCE_FIRST_ENTRY_STATE // Expect the first entry of a flow sequence. + yaml_PARSE_FLOW_SEQUENCE_ENTRY_STATE // Expect an entry of a flow sequence. + yaml_PARSE_FLOW_SEQUENCE_ENTRY_MAPPING_KEY_STATE // Expect a key of an ordered mapping. + yaml_PARSE_FLOW_SEQUENCE_ENTRY_MAPPING_VALUE_STATE // Expect a value of an ordered mapping. + yaml_PARSE_FLOW_SEQUENCE_ENTRY_MAPPING_END_STATE // Expect the and of an ordered mapping entry. + yaml_PARSE_FLOW_MAPPING_FIRST_KEY_STATE // Expect the first key of a flow mapping. + yaml_PARSE_FLOW_MAPPING_KEY_STATE // Expect a key of a flow mapping. + yaml_PARSE_FLOW_MAPPING_VALUE_STATE // Expect a value of a flow mapping. + yaml_PARSE_FLOW_MAPPING_EMPTY_VALUE_STATE // Expect an empty value of a flow mapping. + yaml_PARSE_END_STATE // Expect nothing. +) + +func (ps yaml_parser_state_t) String() string { + switch ps { + case yaml_PARSE_STREAM_START_STATE: + return "yaml_PARSE_STREAM_START_STATE" + case yaml_PARSE_IMPLICIT_DOCUMENT_START_STATE: + return "yaml_PARSE_IMPLICIT_DOCUMENT_START_STATE" + case yaml_PARSE_DOCUMENT_START_STATE: + return "yaml_PARSE_DOCUMENT_START_STATE" + case yaml_PARSE_DOCUMENT_CONTENT_STATE: + return "yaml_PARSE_DOCUMENT_CONTENT_STATE" + case yaml_PARSE_DOCUMENT_END_STATE: + return "yaml_PARSE_DOCUMENT_END_STATE" + case yaml_PARSE_BLOCK_NODE_STATE: + return "yaml_PARSE_BLOCK_NODE_STATE" + case yaml_PARSE_BLOCK_NODE_OR_INDENTLESS_SEQUENCE_STATE: + return "yaml_PARSE_BLOCK_NODE_OR_INDENTLESS_SEQUENCE_STATE" + case yaml_PARSE_FLOW_NODE_STATE: + return "yaml_PARSE_FLOW_NODE_STATE" + case yaml_PARSE_BLOCK_SEQUENCE_FIRST_ENTRY_STATE: + return "yaml_PARSE_BLOCK_SEQUENCE_FIRST_ENTRY_STATE" + case yaml_PARSE_BLOCK_SEQUENCE_ENTRY_STATE: + return "yaml_PARSE_BLOCK_SEQUENCE_ENTRY_STATE" + case yaml_PARSE_INDENTLESS_SEQUENCE_ENTRY_STATE: + return "yaml_PARSE_INDENTLESS_SEQUENCE_ENTRY_STATE" + case yaml_PARSE_BLOCK_MAPPING_FIRST_KEY_STATE: + return "yaml_PARSE_BLOCK_MAPPING_FIRST_KEY_STATE" + case yaml_PARSE_BLOCK_MAPPING_KEY_STATE: + return "yaml_PARSE_BLOCK_MAPPING_KEY_STATE" + case yaml_PARSE_BLOCK_MAPPING_VALUE_STATE: + return "yaml_PARSE_BLOCK_MAPPING_VALUE_STATE" + case yaml_PARSE_FLOW_SEQUENCE_FIRST_ENTRY_STATE: + return "yaml_PARSE_FLOW_SEQUENCE_FIRST_ENTRY_STATE" + case yaml_PARSE_FLOW_SEQUENCE_ENTRY_STATE: + return "yaml_PARSE_FLOW_SEQUENCE_ENTRY_STATE" + case yaml_PARSE_FLOW_SEQUENCE_ENTRY_MAPPING_KEY_STATE: + return "yaml_PARSE_FLOW_SEQUENCE_ENTRY_MAPPING_KEY_STATE" + case yaml_PARSE_FLOW_SEQUENCE_ENTRY_MAPPING_VALUE_STATE: + return "yaml_PARSE_FLOW_SEQUENCE_ENTRY_MAPPING_VALUE_STATE" + case yaml_PARSE_FLOW_SEQUENCE_ENTRY_MAPPING_END_STATE: + return "yaml_PARSE_FLOW_SEQUENCE_ENTRY_MAPPING_END_STATE" + case yaml_PARSE_FLOW_MAPPING_FIRST_KEY_STATE: + return "yaml_PARSE_FLOW_MAPPING_FIRST_KEY_STATE" + case yaml_PARSE_FLOW_MAPPING_KEY_STATE: + return "yaml_PARSE_FLOW_MAPPING_KEY_STATE" + case yaml_PARSE_FLOW_MAPPING_VALUE_STATE: + return "yaml_PARSE_FLOW_MAPPING_VALUE_STATE" + case yaml_PARSE_FLOW_MAPPING_EMPTY_VALUE_STATE: + return "yaml_PARSE_FLOW_MAPPING_EMPTY_VALUE_STATE" + case yaml_PARSE_END_STATE: + return "yaml_PARSE_END_STATE" + } + return "" +} + +// This structure holds aliases data. +type yaml_alias_data_t struct { + anchor []byte // The anchor. + index int // The node id. + mark yaml_mark_t // The anchor mark. +} + +// The parser structure. +// +// All members are internal. Manage the structure using the +// yaml_parser_ family of functions. +type yaml_parser_t struct { + + // Error handling + + error yaml_error_type_t // Error type. + + problem string // Error description. + + // The byte about which the problem occurred. + problem_offset int + problem_value int + problem_mark yaml_mark_t + + // The error context. + context string + context_mark yaml_mark_t + + // Reader stuff + + read_handler yaml_read_handler_t // Read handler. + + input_reader io.Reader // File input data. + input []byte // String input data. + input_pos int + + eof bool // EOF flag + + buffer []byte // The working buffer. + buffer_pos int // The current position of the buffer. + + unread int // The number of unread characters in the buffer. + + newlines int // The number of line breaks since last non-break/non-blank character + + raw_buffer []byte // The raw buffer. + raw_buffer_pos int // The current position of the buffer. + + encoding yaml_encoding_t // The input encoding. + + offset int // The offset of the current position (in bytes). + mark yaml_mark_t // The mark of the current position. + + // Comments + + head_comment []byte // The current head comments + line_comment []byte // The current line comments + foot_comment []byte // The current foot comments + tail_comment []byte // Foot comment that happens at the end of a block. + stem_comment []byte // Comment in item preceding a nested structure (list inside list item, etc) + + comments []yaml_comment_t // The folded comments for all parsed tokens + comments_head int + + // Scanner stuff + + stream_start_produced bool // Have we started to scan the input stream? + stream_end_produced bool // Have we reached the end of the input stream? + + flow_level int // The number of unclosed '[' and '{' indicators. + + tokens []yaml_token_t // The tokens queue. + tokens_head int // The head of the tokens queue. + tokens_parsed int // The number of tokens fetched from the queue. + token_available bool // Does the tokens queue contain a token ready for dequeueing. + + indent int // The current indentation level. + indents []int // The indentation levels stack. + + simple_key_allowed bool // May a simple key occur at the current position? + simple_keys []yaml_simple_key_t // The stack of simple keys. + simple_keys_by_tok map[int]int // possible simple_key indexes indexed by token_number + + // Parser stuff + + state yaml_parser_state_t // The current parser state. + states []yaml_parser_state_t // The parser states stack. + marks []yaml_mark_t // The stack of marks. + tag_directives []yaml_tag_directive_t // The list of TAG directives. + + // Dumper stuff + + aliases []yaml_alias_data_t // The alias data. + + document *yaml_document_t // The currently parsed document. +} + +type yaml_comment_t struct { + + scan_mark yaml_mark_t // Position where scanning for comments started + token_mark yaml_mark_t // Position after which tokens will be associated with this comment + start_mark yaml_mark_t // Position of '#' comment mark + end_mark yaml_mark_t // Position where comment terminated + + head []byte + line []byte + foot []byte +} + +// Emitter Definitions + +// The prototype of a write handler. +// +// The write handler is called when the emitter needs to flush the accumulated +// characters to the output. The handler should write @a size bytes of the +// @a buffer to the output. +// +// @param[in,out] data A pointer to an application data specified by +// yaml_emitter_set_output(). +// @param[in] buffer The buffer with bytes to be written. +// @param[in] size The size of the buffer. +// +// @returns On success, the handler should return @c 1. If the handler failed, +// the returned value should be @c 0. +// +type yaml_write_handler_t func(emitter *yaml_emitter_t, buffer []byte) error + +type yaml_emitter_state_t int + +// The emitter states. +const ( + // Expect STREAM-START. + yaml_EMIT_STREAM_START_STATE yaml_emitter_state_t = iota + + yaml_EMIT_FIRST_DOCUMENT_START_STATE // Expect the first DOCUMENT-START or STREAM-END. + yaml_EMIT_DOCUMENT_START_STATE // Expect DOCUMENT-START or STREAM-END. + yaml_EMIT_DOCUMENT_CONTENT_STATE // Expect the content of a document. + yaml_EMIT_DOCUMENT_END_STATE // Expect DOCUMENT-END. + yaml_EMIT_FLOW_SEQUENCE_FIRST_ITEM_STATE // Expect the first item of a flow sequence. + yaml_EMIT_FLOW_SEQUENCE_TRAIL_ITEM_STATE // Expect the next item of a flow sequence, with the comma already written out + yaml_EMIT_FLOW_SEQUENCE_ITEM_STATE // Expect an item of a flow sequence. + yaml_EMIT_FLOW_MAPPING_FIRST_KEY_STATE // Expect the first key of a flow mapping. + yaml_EMIT_FLOW_MAPPING_TRAIL_KEY_STATE // Expect the next key of a flow mapping, with the comma already written out + yaml_EMIT_FLOW_MAPPING_KEY_STATE // Expect a key of a flow mapping. + yaml_EMIT_FLOW_MAPPING_SIMPLE_VALUE_STATE // Expect a value for a simple key of a flow mapping. + yaml_EMIT_FLOW_MAPPING_VALUE_STATE // Expect a value of a flow mapping. + yaml_EMIT_BLOCK_SEQUENCE_FIRST_ITEM_STATE // Expect the first item of a block sequence. + yaml_EMIT_BLOCK_SEQUENCE_ITEM_STATE // Expect an item of a block sequence. + yaml_EMIT_BLOCK_MAPPING_FIRST_KEY_STATE // Expect the first key of a block mapping. + yaml_EMIT_BLOCK_MAPPING_KEY_STATE // Expect the key of a block mapping. + yaml_EMIT_BLOCK_MAPPING_SIMPLE_VALUE_STATE // Expect a value for a simple key of a block mapping. + yaml_EMIT_BLOCK_MAPPING_VALUE_STATE // Expect a value of a block mapping. + yaml_EMIT_END_STATE // Expect nothing. +) + +// The emitter structure. +// +// All members are internal. Manage the structure using the @c yaml_emitter_ +// family of functions. +type yaml_emitter_t struct { + + // Error handling + + error yaml_error_type_t // Error type. + problem string // Error description. + + // Writer stuff + + write_handler yaml_write_handler_t // Write handler. + + output_buffer *[]byte // String output data. + output_writer io.Writer // File output data. + + buffer []byte // The working buffer. + buffer_pos int // The current position of the buffer. + + raw_buffer []byte // The raw buffer. + raw_buffer_pos int // The current position of the buffer. + + encoding yaml_encoding_t // The stream encoding. + + // Emitter stuff + + canonical bool // If the output is in the canonical style? + best_indent int // The number of indentation spaces. + best_width int // The preferred width of the output lines. + unicode bool // Allow unescaped non-ASCII characters? + line_break yaml_break_t // The preferred line break. + + state yaml_emitter_state_t // The current emitter state. + states []yaml_emitter_state_t // The stack of states. + + events []yaml_event_t // The event queue. + events_head int // The head of the event queue. + + indents []int // The stack of indentation levels. + + tag_directives []yaml_tag_directive_t // The list of tag directives. + + indent int // The current indentation level. + + flow_level int // The current flow level. + + root_context bool // Is it the document root context? + sequence_context bool // Is it a sequence context? + mapping_context bool // Is it a mapping context? + simple_key_context bool // Is it a simple mapping key context? + + line int // The current line. + column int // The current column. + whitespace bool // If the last character was a whitespace? + indention bool // If the last character was an indentation character (' ', '-', '?', ':')? + open_ended bool // If an explicit document end is required? + + space_above bool // Is there's an empty line above? + foot_indent int // The indent used to write the foot comment above, or -1 if none. + + // Anchor analysis. + anchor_data struct { + anchor []byte // The anchor value. + alias bool // Is it an alias? + } + + // Tag analysis. + tag_data struct { + handle []byte // The tag handle. + suffix []byte // The tag suffix. + } + + // Scalar analysis. + scalar_data struct { + value []byte // The scalar value. + multiline bool // Does the scalar contain line breaks? + flow_plain_allowed bool // Can the scalar be expessed in the flow plain style? + block_plain_allowed bool // Can the scalar be expressed in the block plain style? + single_quoted_allowed bool // Can the scalar be expressed in the single quoted style? + block_allowed bool // Can the scalar be expressed in the literal or folded styles? + style yaml_scalar_style_t // The output style. + } + + // Comments + head_comment []byte + line_comment []byte + foot_comment []byte + tail_comment []byte + + key_line_comment []byte + + // Dumper stuff + + opened bool // If the stream was already opened? + closed bool // If the stream was already closed? + + // The information associated with the document nodes. + anchors *struct { + references int // The number of references. + anchor int // The anchor id. + serialized bool // If the node has been emitted? + } + + last_anchor_id int // The last assigned anchor id. + + document *yaml_document_t // The currently emitted document. +} diff --git a/vendor/gopkg.in/yaml.v3/yamlprivateh.go b/vendor/gopkg.in/yaml.v3/yamlprivateh.go new file mode 100644 index 00000000..e88f9c54 --- /dev/null +++ b/vendor/gopkg.in/yaml.v3/yamlprivateh.go @@ -0,0 +1,198 @@ +// +// Copyright (c) 2011-2019 Canonical Ltd +// Copyright (c) 2006-2010 Kirill Simonov +// +// Permission is hereby granted, free of charge, to any person obtaining a copy of +// this software and associated documentation files (the "Software"), to deal in +// the Software without restriction, including without limitation the rights to +// use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies +// of the Software, and to permit persons to whom the Software is furnished to do +// so, subject to the following conditions: +// +// The above copyright notice and this permission notice shall be included in all +// copies or substantial portions of the Software. +// +// THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +// IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +// FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +// AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +// LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +// OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE +// SOFTWARE. + +package yaml + +const ( + // The size of the input raw buffer. + input_raw_buffer_size = 512 + + // The size of the input buffer. + // It should be possible to decode the whole raw buffer. + input_buffer_size = input_raw_buffer_size * 3 + + // The size of the output buffer. + output_buffer_size = 128 + + // The size of the output raw buffer. + // It should be possible to encode the whole output buffer. + output_raw_buffer_size = (output_buffer_size*2 + 2) + + // The size of other stacks and queues. + initial_stack_size = 16 + initial_queue_size = 16 + initial_string_size = 16 +) + +// Check if the character at the specified position is an alphabetical +// character, a digit, '_', or '-'. +func is_alpha(b []byte, i int) bool { + return b[i] >= '0' && b[i] <= '9' || b[i] >= 'A' && b[i] <= 'Z' || b[i] >= 'a' && b[i] <= 'z' || b[i] == '_' || b[i] == '-' +} + +// Check if the character at the specified position is a digit. +func is_digit(b []byte, i int) bool { + return b[i] >= '0' && b[i] <= '9' +} + +// Get the value of a digit. +func as_digit(b []byte, i int) int { + return int(b[i]) - '0' +} + +// Check if the character at the specified position is a hex-digit. +func is_hex(b []byte, i int) bool { + return b[i] >= '0' && b[i] <= '9' || b[i] >= 'A' && b[i] <= 'F' || b[i] >= 'a' && b[i] <= 'f' +} + +// Get the value of a hex-digit. +func as_hex(b []byte, i int) int { + bi := b[i] + if bi >= 'A' && bi <= 'F' { + return int(bi) - 'A' + 10 + } + if bi >= 'a' && bi <= 'f' { + return int(bi) - 'a' + 10 + } + return int(bi) - '0' +} + +// Check if the character is ASCII. +func is_ascii(b []byte, i int) bool { + return b[i] <= 0x7F +} + +// Check if the character at the start of the buffer can be printed unescaped. +func is_printable(b []byte, i int) bool { + return ((b[i] == 0x0A) || // . == #x0A + (b[i] >= 0x20 && b[i] <= 0x7E) || // #x20 <= . <= #x7E + (b[i] == 0xC2 && b[i+1] >= 0xA0) || // #0xA0 <= . <= #xD7FF + (b[i] > 0xC2 && b[i] < 0xED) || + (b[i] == 0xED && b[i+1] < 0xA0) || + (b[i] == 0xEE) || + (b[i] == 0xEF && // #xE000 <= . <= #xFFFD + !(b[i+1] == 0xBB && b[i+2] == 0xBF) && // && . != #xFEFF + !(b[i+1] == 0xBF && (b[i+2] == 0xBE || b[i+2] == 0xBF)))) +} + +// Check if the character at the specified position is NUL. +func is_z(b []byte, i int) bool { + return b[i] == 0x00 +} + +// Check if the beginning of the buffer is a BOM. +func is_bom(b []byte, i int) bool { + return b[0] == 0xEF && b[1] == 0xBB && b[2] == 0xBF +} + +// Check if the character at the specified position is space. +func is_space(b []byte, i int) bool { + return b[i] == ' ' +} + +// Check if the character at the specified position is tab. +func is_tab(b []byte, i int) bool { + return b[i] == '\t' +} + +// Check if the character at the specified position is blank (space or tab). +func is_blank(b []byte, i int) bool { + //return is_space(b, i) || is_tab(b, i) + return b[i] == ' ' || b[i] == '\t' +} + +// Check if the character at the specified position is a line break. +func is_break(b []byte, i int) bool { + return (b[i] == '\r' || // CR (#xD) + b[i] == '\n' || // LF (#xA) + b[i] == 0xC2 && b[i+1] == 0x85 || // NEL (#x85) + b[i] == 0xE2 && b[i+1] == 0x80 && b[i+2] == 0xA8 || // LS (#x2028) + b[i] == 0xE2 && b[i+1] == 0x80 && b[i+2] == 0xA9) // PS (#x2029) +} + +func is_crlf(b []byte, i int) bool { + return b[i] == '\r' && b[i+1] == '\n' +} + +// Check if the character is a line break or NUL. +func is_breakz(b []byte, i int) bool { + //return is_break(b, i) || is_z(b, i) + return ( + // is_break: + b[i] == '\r' || // CR (#xD) + b[i] == '\n' || // LF (#xA) + b[i] == 0xC2 && b[i+1] == 0x85 || // NEL (#x85) + b[i] == 0xE2 && b[i+1] == 0x80 && b[i+2] == 0xA8 || // LS (#x2028) + b[i] == 0xE2 && b[i+1] == 0x80 && b[i+2] == 0xA9 || // PS (#x2029) + // is_z: + b[i] == 0) +} + +// Check if the character is a line break, space, or NUL. +func is_spacez(b []byte, i int) bool { + //return is_space(b, i) || is_breakz(b, i) + return ( + // is_space: + b[i] == ' ' || + // is_breakz: + b[i] == '\r' || // CR (#xD) + b[i] == '\n' || // LF (#xA) + b[i] == 0xC2 && b[i+1] == 0x85 || // NEL (#x85) + b[i] == 0xE2 && b[i+1] == 0x80 && b[i+2] == 0xA8 || // LS (#x2028) + b[i] == 0xE2 && b[i+1] == 0x80 && b[i+2] == 0xA9 || // PS (#x2029) + b[i] == 0) +} + +// Check if the character is a line break, space, tab, or NUL. +func is_blankz(b []byte, i int) bool { + //return is_blank(b, i) || is_breakz(b, i) + return ( + // is_blank: + b[i] == ' ' || b[i] == '\t' || + // is_breakz: + b[i] == '\r' || // CR (#xD) + b[i] == '\n' || // LF (#xA) + b[i] == 0xC2 && b[i+1] == 0x85 || // NEL (#x85) + b[i] == 0xE2 && b[i+1] == 0x80 && b[i+2] == 0xA8 || // LS (#x2028) + b[i] == 0xE2 && b[i+1] == 0x80 && b[i+2] == 0xA9 || // PS (#x2029) + b[i] == 0) +} + +// Determine the width of the character. +func width(b byte) int { + // Don't replace these by a switch without first + // confirming that it is being inlined. + if b&0x80 == 0x00 { + return 1 + } + if b&0xE0 == 0xC0 { + return 2 + } + if b&0xF0 == 0xE0 { + return 3 + } + if b&0xF8 == 0xF0 { + return 4 + } + return 0 + +} diff --git a/vendor/modules.txt b/vendor/modules.txt index ecf0c3b1..4dc19682 100644 --- a/vendor/modules.txt +++ b/vendor/modules.txt @@ -1,3 +1,6 @@ +# github.com/OneOfOne/xxhash v1.2.8 +## explicit; go 1.11 +github.com/OneOfOne/xxhash # github.com/beorn7/perks v1.0.1 ## explicit; go 1.11 github.com/beorn7/perks/quantile @@ -10,6 +13,19 @@ github.com/davecgh/go-spew/spew # github.com/docker/go-units v0.5.0 ## explicit github.com/docker/go-units +# github.com/ghodss/yaml v1.0.0 +## explicit +github.com/ghodss/yaml +# github.com/gobwas/glob v0.2.3 +## explicit +github.com/gobwas/glob +github.com/gobwas/glob/compiler +github.com/gobwas/glob/match +github.com/gobwas/glob/syntax +github.com/gobwas/glob/syntax/ast +github.com/gobwas/glob/syntax/lexer +github.com/gobwas/glob/util/runes +github.com/gobwas/glob/util/strings # github.com/golang/protobuf v1.5.2 ## explicit; go 1.9 github.com/golang/protobuf/proto @@ -17,9 +33,76 @@ github.com/golang/protobuf/ptypes/timestamp # github.com/matttproud/golang_protobuf_extensions v1.0.2 ## explicit; go 1.9 github.com/matttproud/golang_protobuf_extensions/pbutil +# github.com/open-policy-agent/opa v0.36.1 +## explicit; go 1.16 +github.com/open-policy-agent/opa/ast +github.com/open-policy-agent/opa/ast/internal/scanner +github.com/open-policy-agent/opa/ast/internal/tokens +github.com/open-policy-agent/opa/ast/location +github.com/open-policy-agent/opa/bundle +github.com/open-policy-agent/opa/format +github.com/open-policy-agent/opa/internal/bundle +github.com/open-policy-agent/opa/internal/cidr/merge +github.com/open-policy-agent/opa/internal/compiler/wasm +github.com/open-policy-agent/opa/internal/compiler/wasm/opa +github.com/open-policy-agent/opa/internal/debug +github.com/open-policy-agent/opa/internal/deepcopy +github.com/open-policy-agent/opa/internal/file/archive +github.com/open-policy-agent/opa/internal/file/url +github.com/open-policy-agent/opa/internal/future +github.com/open-policy-agent/opa/internal/gojsonschema +github.com/open-policy-agent/opa/internal/ir +github.com/open-policy-agent/opa/internal/jwx/buffer +github.com/open-policy-agent/opa/internal/jwx/jwa +github.com/open-policy-agent/opa/internal/jwx/jwk +github.com/open-policy-agent/opa/internal/jwx/jws +github.com/open-policy-agent/opa/internal/jwx/jws/sign +github.com/open-policy-agent/opa/internal/jwx/jws/verify +github.com/open-policy-agent/opa/internal/lcss +github.com/open-policy-agent/opa/internal/leb128 +github.com/open-policy-agent/opa/internal/merge +github.com/open-policy-agent/opa/internal/planner +github.com/open-policy-agent/opa/internal/rego/opa +github.com/open-policy-agent/opa/internal/semver +github.com/open-policy-agent/opa/internal/strings +github.com/open-policy-agent/opa/internal/uuid +github.com/open-policy-agent/opa/internal/version +github.com/open-policy-agent/opa/internal/wasm/constant +github.com/open-policy-agent/opa/internal/wasm/encoding +github.com/open-policy-agent/opa/internal/wasm/instruction +github.com/open-policy-agent/opa/internal/wasm/module +github.com/open-policy-agent/opa/internal/wasm/opcode +github.com/open-policy-agent/opa/internal/wasm/sdk/opa/capabilities +github.com/open-policy-agent/opa/internal/wasm/types +github.com/open-policy-agent/opa/internal/wasm/util +github.com/open-policy-agent/opa/keys +github.com/open-policy-agent/opa/loader +github.com/open-policy-agent/opa/metrics +github.com/open-policy-agent/opa/rego +github.com/open-policy-agent/opa/resolver +github.com/open-policy-agent/opa/resolver/wasm +github.com/open-policy-agent/opa/storage +github.com/open-policy-agent/opa/storage/inmem +github.com/open-policy-agent/opa/storage/internal/errors +github.com/open-policy-agent/opa/storage/internal/ptr +github.com/open-policy-agent/opa/topdown +github.com/open-policy-agent/opa/topdown/builtins +github.com/open-policy-agent/opa/topdown/cache +github.com/open-policy-agent/opa/topdown/copypropagation +github.com/open-policy-agent/opa/topdown/print +github.com/open-policy-agent/opa/tracing +github.com/open-policy-agent/opa/types +github.com/open-policy-agent/opa/util +github.com/open-policy-agent/opa/version # github.com/openfaas/faas-middleware v1.2.2 ## explicit; go 1.18 github.com/openfaas/faas-middleware/concurrency-limiter +# github.com/pkg/errors v0.9.1 +## explicit +github.com/pkg/errors +# github.com/pmezard/go-difflib v1.0.0 +## explicit +github.com/pmezard/go-difflib/difflib # github.com/prometheus/client_golang v1.13.0 ## explicit; go 1.17 github.com/prometheus/client_golang/prometheus @@ -41,6 +124,22 @@ github.com/prometheus/common/model github.com/prometheus/procfs github.com/prometheus/procfs/internal/fs github.com/prometheus/procfs/internal/util +# github.com/rcrowley/go-metrics v0.0.0-20200313005456-10cdbea86bc0 +## explicit +github.com/rcrowley/go-metrics +# github.com/stretchr/testify v1.7.0 +## explicit; go 1.13 +github.com/stretchr/testify/assert +github.com/stretchr/testify/require +# github.com/xeipuuv/gojsonpointer v0.0.0-20190905194746-02993c407bfb +## explicit +github.com/xeipuuv/gojsonpointer +# github.com/xeipuuv/gojsonreference v0.0.0-20180127040603-bd5ef7bd5415 +## explicit +github.com/xeipuuv/gojsonreference +# github.com/yashtewari/glob-intersection v0.0.0-20180916065949-5c77d914dd0b +## explicit +github.com/yashtewari/glob-intersection # golang.org/x/sys v0.0.0-20221013171732-95e765b1cc43 ## explicit; go 1.17 golang.org/x/sys/internal/unsafeheader @@ -76,3 +175,9 @@ google.golang.org/protobuf/runtime/protoiface google.golang.org/protobuf/runtime/protoimpl google.golang.org/protobuf/types/descriptorpb google.golang.org/protobuf/types/known/timestamppb +# gopkg.in/yaml.v2 v2.4.0 +## explicit; go 1.15 +gopkg.in/yaml.v2 +# gopkg.in/yaml.v3 v3.0.0-20210107192922-496545a6307b +## explicit +gopkg.in/yaml.v3 From 2d7a7de231bba2c31328f7ec18084832579a0507 Mon Sep 17 00:00:00 2001 From: Lucas Roesler Date: Sun, 23 Oct 2022 15:01:43 +0200 Subject: [PATCH 2/5] feat: add ability to skip paths when auth is enabled Add `OPA_SKIP_PATHS` configuration and update the implementation to skip these paths. chore: add HMAC auth example in the tests Signed-off-by: Lucas Roesler --- auth/local.go | 45 + auth/local_test.go | 35 + auth/middleware.go | 14 + auth/testdata/basic_auth.rego | 15 +- auth/testdata/hmac_auth.rego | 26 + go.mod | 39 +- go.sum | 417 +- .../agnivade/levenshtein/.gitignore | 5 + .../agnivade/levenshtein/.travis.yml | 23 + .../agnivade/levenshtein/License.txt | 21 + .../github.com/agnivade/levenshtein/Makefile | 15 + .../github.com/agnivade/levenshtein/README.md | 80 + .../agnivade/levenshtein/levenshtein.go | 89 + .../open-policy-agent/opa/ast/annotations.go | 786 +++ .../open-policy-agent/opa/ast/builtins.go | 2198 +++++---- .../open-policy-agent/opa/ast/capabilities.go | 55 +- .../open-policy-agent/opa/ast/check.go | 221 +- .../open-policy-agent/opa/ast/compare.go | 5 + .../open-policy-agent/opa/ast/compile.go | 1208 ++++- .../opa/ast/compilehelper.go | 3 +- .../open-policy-agent/opa/ast/errors.go | 9 - .../open-policy-agent/opa/ast/fuzz.go | 1 + .../opa/ast/internal/scanner/scanner.go | 20 + .../opa/ast/internal/tokens/tokens.go | 13 + .../opa/ast/location/location.go | 2 +- .../open-policy-agent/opa/ast/parser.go | 533 +- .../open-policy-agent/opa/ast/parser_ext.go | 115 +- .../open-policy-agent/opa/ast/policy.go | 277 +- .../open-policy-agent/opa/ast/schema.go | 2 +- .../open-policy-agent/opa/ast/term.go | 275 +- .../open-policy-agent/opa/ast/transform.go | 20 + .../open-policy-agent/opa/ast/varset.go | 2 +- .../open-policy-agent/opa/ast/visit.go | 72 +- .../open-policy-agent/opa/bundle/bundle.go | 408 +- .../open-policy-agent/opa/bundle/file.go | 161 +- .../open-policy-agent/opa/bundle/filefs.go | 42 +- .../open-policy-agent/opa/bundle/store.go | 453 +- .../open-policy-agent/opa/bundle/verify.go | 6 +- .../opa/capabilities/capabilities.go | 18 + .../opa/capabilities/v0.17.0.json | 2392 +++++++++ .../opa/capabilities/v0.17.1.json | 2392 +++++++++ .../opa/capabilities/v0.17.2.json | 2525 ++++++++++ .../opa/capabilities/v0.17.3.json | 2525 ++++++++++ .../opa/capabilities/v0.18.0.json | 2685 ++++++++++ .../opa/capabilities/v0.19.0-rc1.json | 2835 +++++++++++ .../opa/capabilities/v0.19.0.json | 2858 +++++++++++ .../opa/capabilities/v0.19.1.json | 2858 +++++++++++ .../opa/capabilities/v0.19.2.json | 2858 +++++++++++ .../opa/capabilities/v0.20.0.json | 3064 ++++++++++++ .../opa/capabilities/v0.20.1.json | 3064 ++++++++++++ .../opa/capabilities/v0.20.2.json | 3064 ++++++++++++ .../opa/capabilities/v0.20.3.json | 3064 ++++++++++++ .../opa/capabilities/v0.20.4.json | 3064 ++++++++++++ .../opa/capabilities/v0.20.5.json | 3064 ++++++++++++ .../opa/capabilities/v0.21.0.json | 3086 ++++++++++++ .../opa/capabilities/v0.21.1.json | 3086 ++++++++++++ .../opa/capabilities/v0.22.0.json | 3137 ++++++++++++ .../opa/capabilities/v0.23.0.json | 3168 ++++++++++++ .../opa/capabilities/v0.23.1.json | 3168 ++++++++++++ .../opa/capabilities/v0.23.2.json | 3168 ++++++++++++ .../opa/capabilities/v0.24.0.json | 3243 +++++++++++++ .../opa/capabilities/v0.25.0-rc1.json | 3271 +++++++++++++ .../opa/capabilities/v0.25.0-rc2.json | 3313 +++++++++++++ .../opa/capabilities/v0.25.0-rc3.json | 3313 +++++++++++++ .../opa/capabilities/v0.25.0-rc4.json | 3313 +++++++++++++ .../opa/capabilities/v0.25.0.json | 3355 +++++++++++++ .../opa/capabilities/v0.25.1.json | 3355 +++++++++++++ .../opa/capabilities/v0.25.2.json | 3355 +++++++++++++ .../opa/capabilities/v0.26.0.json | 3383 +++++++++++++ .../opa/capabilities/v0.27.0.json | 3389 +++++++++++++ .../opa/capabilities/v0.27.1.json | 3389 +++++++++++++ .../opa/capabilities/v0.28.0.json | 3458 +++++++++++++ .../opa/capabilities/v0.29.0.json | 3458 +++++++++++++ .../opa/capabilities/v0.29.1.json | 3458 +++++++++++++ .../opa/capabilities/v0.29.2.json | 3458 +++++++++++++ .../opa/capabilities/v0.29.3.json | 3458 +++++++++++++ .../opa/capabilities/v0.29.4.json | 3458 +++++++++++++ .../opa/capabilities/v0.30.0.json | 3458 +++++++++++++ .../opa/capabilities/v0.30.1.json | 3458 +++++++++++++ .../opa/capabilities/v0.30.2.json | 3458 +++++++++++++ .../opa/capabilities/v0.31.0.json | 3512 ++++++++++++++ .../opa/capabilities/v0.32.0.json | 3512 ++++++++++++++ .../opa/capabilities/v0.32.1.json | 3512 ++++++++++++++ .../opa/capabilities/v0.33.0.json | 3534 ++++++++++++++ .../opa/capabilities/v0.33.1.json | 3534 ++++++++++++++ .../opa/capabilities/v0.34.0.json | 3602 ++++++++++++++ .../opa/capabilities/v0.34.1.json | 3602 ++++++++++++++ .../opa/capabilities/v0.34.2.json | 3602 ++++++++++++++ .../opa/capabilities/v0.35.0.json | 3619 ++++++++++++++ .../opa/capabilities/v0.36.0.json | 3721 ++++++++++++++ .../opa/capabilities/v0.36.1.json | 3721 ++++++++++++++ .../opa/capabilities/v0.37.0.json | 3825 +++++++++++++++ .../opa/capabilities/v0.37.1.json | 3825 +++++++++++++++ .../opa/capabilities/v0.37.2.json | 3825 +++++++++++++++ .../opa/capabilities/v0.38.0.json | 3826 +++++++++++++++ .../opa/capabilities/v0.38.1.json | 3826 +++++++++++++++ .../opa/capabilities/v0.39.0.json | 3826 +++++++++++++++ .../opa/capabilities/v0.40.0.json | 3847 +++++++++++++++ .../opa/capabilities/v0.41.0.json | 4007 +++++++++++++++ .../opa/capabilities/v0.42.0.json | 4076 ++++++++++++++++ .../opa/capabilities/v0.42.1.json | 4076 ++++++++++++++++ .../opa/capabilities/v0.42.2.json | 4076 ++++++++++++++++ .../opa/capabilities/v0.43.0.json | 4079 ++++++++++++++++ .../opa/capabilities/v0.43.1.json | 4079 ++++++++++++++++ .../opa/capabilities/v0.44.0.json | 4190 ++++++++++++++++ .../opa/capabilities/v0.45.0.json | 4306 +++++++++++++++++ .../open-policy-agent/opa/format/format.go | 222 +- .../internal/compiler/wasm/opa/callgraph.csv | 33 +- .../opa/internal/compiler/wasm/opa/opa.go | 46 +- .../opa/internal/compiler/wasm/opa/opa.wasm | Bin 421950 -> 423668 bytes .../internal/compiler/wasm/optimizations.go | 5 +- .../opa/internal/compiler/wasm/wasm.go | 52 +- .../opa/internal/deepcopy/deepcopy.go | 14 +- .../opa/internal/future/filter_imports.go | 17 + .../opa/internal/gojsonschema/jsonLoader.go | 6 + .../opa/internal/gojsonschema/schema.go | 15 +- .../opa/internal/gqlparser/LICENSE | 19 + .../opa/internal/gqlparser/ast/argmap.go | 37 + .../opa/internal/gqlparser/ast/collections.go | 148 + .../opa/internal/gqlparser/ast/decode.go | 216 + .../opa/internal/gqlparser/ast/definition.go | 94 + .../opa/internal/gqlparser/ast/directive.go | 43 + .../opa/internal/gqlparser/ast/document.go | 79 + .../opa/internal/gqlparser/ast/dumper.go | 159 + .../opa/internal/gqlparser/ast/fragment.go | 38 + .../opa/internal/gqlparser/ast/operation.go | 30 + .../opa/internal/gqlparser/ast/path.go | 67 + .../opa/internal/gqlparser/ast/selection.go | 39 + .../opa/internal/gqlparser/ast/source.go | 19 + .../opa/internal/gqlparser/ast/type.go | 68 + .../opa/internal/gqlparser/ast/value.go | 120 + .../opa/internal/gqlparser/gqlerror/error.go | 147 + .../internal/gqlparser/lexer/blockstring.go | 58 + .../opa/internal/gqlparser/lexer/lexer.go | 517 ++ .../internal/gqlparser/lexer/lexer_test.yml | 692 +++ .../opa/internal/gqlparser/lexer/token.go | 148 + .../opa/internal/gqlparser/parser/parser.go | 136 + .../opa/internal/gqlparser/parser/query.go | 349 ++ .../internal/gqlparser/parser/query_test.yml | 544 +++ .../opa/internal/gqlparser/parser/schema.go | 535 ++ .../internal/gqlparser/parser/schema_test.yml | 646 +++ .../opa/internal/gqlparser/validator/error.go | 55 + .../internal/gqlparser/validator/messaging.go | 39 + .../internal/gqlparser/validator/prelude.go | 16 + .../gqlparser/validator/prelude.graphql | 121 + .../validator/rules/fields_on_correct_type.go | 97 + .../rules/fragments_on_composite_types.go | 41 + .../validator/rules/known_argument_names.go | 59 + .../validator/rules/known_directives.go | 49 + .../validator/rules/known_fragment_names.go | 21 + .../validator/rules/known_root_type.go | 37 + .../validator/rules/known_type_names.go | 61 + .../rules/lone_anonymous_operation.go | 21 + .../validator/rules/no_fragment_cycles.go | 95 + .../validator/rules/no_undefined_variables.go | 30 + .../validator/rules/no_unused_fragments.go | 32 + .../validator/rules/no_unused_variables.go | 32 + .../rules/overlapping_fields_can_be_merged.go | 562 +++ .../rules/possible_fragment_spreads.go | 70 + .../rules/provided_required_arguments.go | 64 + .../gqlparser/validator/rules/scalar_leafs.go | 38 + .../rules/single_field_subscriptions.go | 88 + .../validator/rules/unique_argument_names.go | 35 + .../rules/unique_directives_per_location.go | 26 + .../validator/rules/unique_fragment_names.go | 24 + .../rules/unique_input_field_names.go | 29 + .../validator/rules/unique_operation_names.go | 24 + .../validator/rules/unique_variable_names.go | 26 + .../validator/rules/values_of_correct_type.go | 170 + .../rules/variables_are_input_types.go | 30 + .../rules/variables_in_allowed_position.go | 40 + .../internal/gqlparser/validator/schema.go | 513 ++ .../gqlparser/validator/schema_test.yml | 678 +++ .../gqlparser/validator/suggestionList.go | 69 + .../internal/gqlparser/validator/validator.go | 45 + .../opa/internal/gqlparser/validator/vars.go | 258 + .../opa/internal/gqlparser/validator/walk.go | 292 ++ .../opa/internal/json/patch/patch.go | 45 + .../opa/internal/jwx/buffer/buffer.go | 11 +- .../opa/internal/jwx/jwa/key_type.go | 12 +- .../opa/internal/jwx/jwa/signature.go | 14 +- .../opa/internal/jwx/jwk/ecdsa.go | 18 +- .../opa/internal/jwx/jwk/headers.go | 20 +- .../opa/internal/jwx/jwk/jwk.go | 27 +- .../opa/internal/jwx/jwk/rsa.go | 18 +- .../opa/internal/jwx/jwk/symmetric.go | 4 +- .../opa/internal/jwx/jws/headers.go | 20 +- .../opa/internal/jwx/jws/jws.go | 36 +- .../opa/internal/jwx/jws/sign/ecdsa.go | 12 +- .../opa/internal/jwx/jws/verify/hmac.go | 10 +- .../opa/internal/planner/planner.go | 485 +- .../opa/internal/planner/rules.go | 46 + .../opa/internal/planner/varstack.go | 15 +- .../open-policy-agent/opa/internal/ref/ref.go | 39 + .../opa/internal/rego/opa/options.go | 2 + .../opa/internal/strings/strings.go | 2 +- .../opa/internal/wasm/encoding/reader.go | 36 +- .../opa/{internal => }/ir/ir.go | 264 +- .../open-policy-agent/opa/ir/marshal.go | 137 + .../opa/{internal => }/ir/pretty.go | 0 .../opa/{internal => }/ir/walk.go | 0 .../opa/loader/filter/filter.go | 5 + .../loader/internal/embedtest/bar/bar.rego | 3 + .../loader/internal/embedtest/bar/bar.yaml | 1 + .../internal/embedtest/baz/qux/qux.json | 1 + .../opa/loader/internal/embedtest/foo.json | 1 + .../open-policy-agent/opa/loader/loader.go | 111 +- .../open-policy-agent/opa/metrics/metrics.go | 2 +- .../open-policy-agent/opa/rego/rego.go | 82 +- .../open-policy-agent/opa/storage/errors.go | 2 +- .../opa/storage/inmem/inmem.go | 196 +- .../opa/storage/inmem/opts.go | 25 + .../opa/storage/inmem/txn.go | 12 +- .../opa/storage/interface.go | 71 +- .../opa/storage/internal/errors/errors.go | 7 + .../opa/storage/internal/ptr/ptr.go | 28 +- .../open-policy-agent/opa/storage/storage.go | 14 +- .../opa/topdown/aggregates.go | 2 +- .../open-policy-agent/opa/topdown/builtins.go | 1 + .../opa/topdown/builtins/builtins.go | 111 +- .../open-policy-agent/opa/topdown/cache.go | 54 + .../opa/topdown/cache/cache.go | 2 +- .../open-policy-agent/opa/topdown/cidr.go | 38 +- .../copypropagation/copypropagation.go | 60 +- .../open-policy-agent/opa/topdown/encoding.go | 10 +- .../open-policy-agent/opa/topdown/eval.go | 362 +- .../open-policy-agent/opa/topdown/glob.go | 35 +- .../open-policy-agent/opa/topdown/graphql.go | 462 ++ .../open-policy-agent/opa/topdown/http.go | 58 +- .../opa/topdown/http_fixup.go | 8 + .../opa/topdown/http_fixup_darwin.go | 13 + .../open-policy-agent/opa/topdown/json.go | 6 + .../open-policy-agent/opa/topdown/net.go | 3 +- .../open-policy-agent/opa/topdown/object.go | 87 +- .../open-policy-agent/opa/topdown/parse.go | 12 + .../opa/topdown/parse_bytes.go | 18 +- .../opa/topdown/parse_units.go | 125 + .../open-policy-agent/opa/topdown/query.go | 19 + .../opa/topdown/reachable.go | 101 +- .../open-policy-agent/opa/topdown/regex.go | 27 + .../open-policy-agent/opa/topdown/runtime.go | 109 +- .../open-policy-agent/opa/topdown/save.go | 8 +- .../open-policy-agent/opa/topdown/semver.go | 3 +- .../open-policy-agent/opa/topdown/sets.go | 12 +- .../open-policy-agent/opa/topdown/strings.go | 171 +- .../open-policy-agent/opa/topdown/subset.go | 263 + .../open-policy-agent/opa/topdown/tokens.go | 166 +- .../open-policy-agent/opa/topdown/trace.go | 32 +- .../open-policy-agent/opa/topdown/type.go | 16 +- .../open-policy-agent/opa/types/types.go | 119 +- .../open-policy-agent/opa/util/backoff.go | 9 + .../open-policy-agent/opa/util/compare.go | 26 +- .../open-policy-agent/opa/util/json.go | 3 + .../open-policy-agent/opa/version/version.go | 5 +- .../opa/version/version_go1.18.go | 33 + .../prometheus/client_model/go/metrics.pb.go | 333 +- .../rcrowley/go-metrics/.travis.yml | 1 + .../testify/assert/assertion_compare.go | 76 +- .../assert/assertion_compare_can_convert.go | 16 + .../assert/assertion_compare_legacy.go | 16 + .../testify/assert/assertion_format.go | 22 + .../testify/assert/assertion_forward.go | 44 + .../testify/assert/assertion_order.go | 8 +- .../stretchr/testify/assert/assertions.go | 190 +- .../stretchr/testify/require/require.go | 56 + .../testify/require/require_forward.go | 44 + .../github.com/tchap/go-patricia/v2/AUTHORS | 3 + .../github.com/tchap/go-patricia/v2/LICENSE | 20 + .../tchap/go-patricia/v2/patricia/children.go | 363 ++ .../tchap/go-patricia/v2/patricia/patricia.go | 606 +++ .../yashtewari/glob-intersection/match.go | 2 +- .../yashtewari/glob-intersection/tokenize.go | 31 +- vendor/golang.org/x/crypto/AUTHORS | 3 + vendor/golang.org/x/crypto/CONTRIBUTORS | 3 + vendor/golang.org/x/crypto/LICENSE | 27 + vendor/golang.org/x/crypto/PATENTS | 22 + vendor/golang.org/x/crypto/bcrypt/base64.go | 35 + vendor/golang.org/x/crypto/bcrypt/bcrypt.go | 295 ++ vendor/golang.org/x/crypto/blowfish/block.go | 159 + vendor/golang.org/x/crypto/blowfish/cipher.go | 99 + vendor/golang.org/x/crypto/blowfish/const.go | 199 + vendor/gopkg.in/yaml.v3/decode.go | 78 +- vendor/gopkg.in/yaml.v3/parserc.go | 11 +- vendor/modules.txt | 50 +- 284 files changed, 248956 insertions(+), 3339 deletions(-) create mode 100644 auth/testdata/hmac_auth.rego create mode 100644 vendor/github.com/agnivade/levenshtein/.gitignore create mode 100644 vendor/github.com/agnivade/levenshtein/.travis.yml create mode 100644 vendor/github.com/agnivade/levenshtein/License.txt create mode 100644 vendor/github.com/agnivade/levenshtein/Makefile create mode 100644 vendor/github.com/agnivade/levenshtein/README.md create mode 100644 vendor/github.com/agnivade/levenshtein/levenshtein.go create mode 100644 vendor/github.com/open-policy-agent/opa/ast/annotations.go create mode 100644 vendor/github.com/open-policy-agent/opa/capabilities/capabilities.go create mode 100644 vendor/github.com/open-policy-agent/opa/capabilities/v0.17.0.json create mode 100644 vendor/github.com/open-policy-agent/opa/capabilities/v0.17.1.json create mode 100644 vendor/github.com/open-policy-agent/opa/capabilities/v0.17.2.json create mode 100644 vendor/github.com/open-policy-agent/opa/capabilities/v0.17.3.json create mode 100644 vendor/github.com/open-policy-agent/opa/capabilities/v0.18.0.json create mode 100644 vendor/github.com/open-policy-agent/opa/capabilities/v0.19.0-rc1.json create mode 100644 vendor/github.com/open-policy-agent/opa/capabilities/v0.19.0.json create mode 100644 vendor/github.com/open-policy-agent/opa/capabilities/v0.19.1.json create mode 100644 vendor/github.com/open-policy-agent/opa/capabilities/v0.19.2.json create mode 100644 vendor/github.com/open-policy-agent/opa/capabilities/v0.20.0.json create mode 100644 vendor/github.com/open-policy-agent/opa/capabilities/v0.20.1.json create mode 100644 vendor/github.com/open-policy-agent/opa/capabilities/v0.20.2.json create mode 100644 vendor/github.com/open-policy-agent/opa/capabilities/v0.20.3.json create mode 100644 vendor/github.com/open-policy-agent/opa/capabilities/v0.20.4.json create mode 100644 vendor/github.com/open-policy-agent/opa/capabilities/v0.20.5.json create mode 100644 vendor/github.com/open-policy-agent/opa/capabilities/v0.21.0.json create mode 100644 vendor/github.com/open-policy-agent/opa/capabilities/v0.21.1.json create mode 100644 vendor/github.com/open-policy-agent/opa/capabilities/v0.22.0.json create mode 100644 vendor/github.com/open-policy-agent/opa/capabilities/v0.23.0.json create mode 100644 vendor/github.com/open-policy-agent/opa/capabilities/v0.23.1.json create mode 100644 vendor/github.com/open-policy-agent/opa/capabilities/v0.23.2.json create mode 100644 vendor/github.com/open-policy-agent/opa/capabilities/v0.24.0.json create mode 100644 vendor/github.com/open-policy-agent/opa/capabilities/v0.25.0-rc1.json create mode 100644 vendor/github.com/open-policy-agent/opa/capabilities/v0.25.0-rc2.json create mode 100644 vendor/github.com/open-policy-agent/opa/capabilities/v0.25.0-rc3.json create mode 100644 vendor/github.com/open-policy-agent/opa/capabilities/v0.25.0-rc4.json create mode 100644 vendor/github.com/open-policy-agent/opa/capabilities/v0.25.0.json create mode 100644 vendor/github.com/open-policy-agent/opa/capabilities/v0.25.1.json create mode 100644 vendor/github.com/open-policy-agent/opa/capabilities/v0.25.2.json create mode 100644 vendor/github.com/open-policy-agent/opa/capabilities/v0.26.0.json create mode 100644 vendor/github.com/open-policy-agent/opa/capabilities/v0.27.0.json create mode 100644 vendor/github.com/open-policy-agent/opa/capabilities/v0.27.1.json create mode 100644 vendor/github.com/open-policy-agent/opa/capabilities/v0.28.0.json create mode 100644 vendor/github.com/open-policy-agent/opa/capabilities/v0.29.0.json create mode 100644 vendor/github.com/open-policy-agent/opa/capabilities/v0.29.1.json create mode 100644 vendor/github.com/open-policy-agent/opa/capabilities/v0.29.2.json create mode 100644 vendor/github.com/open-policy-agent/opa/capabilities/v0.29.3.json create mode 100644 vendor/github.com/open-policy-agent/opa/capabilities/v0.29.4.json create mode 100644 vendor/github.com/open-policy-agent/opa/capabilities/v0.30.0.json create mode 100644 vendor/github.com/open-policy-agent/opa/capabilities/v0.30.1.json create mode 100644 vendor/github.com/open-policy-agent/opa/capabilities/v0.30.2.json create mode 100644 vendor/github.com/open-policy-agent/opa/capabilities/v0.31.0.json create mode 100644 vendor/github.com/open-policy-agent/opa/capabilities/v0.32.0.json create mode 100644 vendor/github.com/open-policy-agent/opa/capabilities/v0.32.1.json create mode 100644 vendor/github.com/open-policy-agent/opa/capabilities/v0.33.0.json create mode 100644 vendor/github.com/open-policy-agent/opa/capabilities/v0.33.1.json create mode 100644 vendor/github.com/open-policy-agent/opa/capabilities/v0.34.0.json create mode 100644 vendor/github.com/open-policy-agent/opa/capabilities/v0.34.1.json create mode 100644 vendor/github.com/open-policy-agent/opa/capabilities/v0.34.2.json create mode 100644 vendor/github.com/open-policy-agent/opa/capabilities/v0.35.0.json create mode 100644 vendor/github.com/open-policy-agent/opa/capabilities/v0.36.0.json create mode 100644 vendor/github.com/open-policy-agent/opa/capabilities/v0.36.1.json create mode 100644 vendor/github.com/open-policy-agent/opa/capabilities/v0.37.0.json create mode 100644 vendor/github.com/open-policy-agent/opa/capabilities/v0.37.1.json create mode 100644 vendor/github.com/open-policy-agent/opa/capabilities/v0.37.2.json create mode 100644 vendor/github.com/open-policy-agent/opa/capabilities/v0.38.0.json create mode 100644 vendor/github.com/open-policy-agent/opa/capabilities/v0.38.1.json create mode 100644 vendor/github.com/open-policy-agent/opa/capabilities/v0.39.0.json create mode 100644 vendor/github.com/open-policy-agent/opa/capabilities/v0.40.0.json create mode 100644 vendor/github.com/open-policy-agent/opa/capabilities/v0.41.0.json create mode 100644 vendor/github.com/open-policy-agent/opa/capabilities/v0.42.0.json create mode 100644 vendor/github.com/open-policy-agent/opa/capabilities/v0.42.1.json create mode 100644 vendor/github.com/open-policy-agent/opa/capabilities/v0.42.2.json create mode 100644 vendor/github.com/open-policy-agent/opa/capabilities/v0.43.0.json create mode 100644 vendor/github.com/open-policy-agent/opa/capabilities/v0.43.1.json create mode 100644 vendor/github.com/open-policy-agent/opa/capabilities/v0.44.0.json create mode 100644 vendor/github.com/open-policy-agent/opa/capabilities/v0.45.0.json create mode 100644 vendor/github.com/open-policy-agent/opa/internal/gqlparser/LICENSE create mode 100644 vendor/github.com/open-policy-agent/opa/internal/gqlparser/ast/argmap.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/gqlparser/ast/collections.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/gqlparser/ast/decode.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/gqlparser/ast/definition.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/gqlparser/ast/directive.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/gqlparser/ast/document.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/gqlparser/ast/dumper.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/gqlparser/ast/fragment.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/gqlparser/ast/operation.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/gqlparser/ast/path.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/gqlparser/ast/selection.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/gqlparser/ast/source.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/gqlparser/ast/type.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/gqlparser/ast/value.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/gqlparser/gqlerror/error.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/gqlparser/lexer/blockstring.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/gqlparser/lexer/lexer.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/gqlparser/lexer/lexer_test.yml create mode 100644 vendor/github.com/open-policy-agent/opa/internal/gqlparser/lexer/token.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/gqlparser/parser/parser.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/gqlparser/parser/query.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/gqlparser/parser/query_test.yml create mode 100644 vendor/github.com/open-policy-agent/opa/internal/gqlparser/parser/schema.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/gqlparser/parser/schema_test.yml create mode 100644 vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/error.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/messaging.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/prelude.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/prelude.graphql create mode 100644 vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/fields_on_correct_type.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/fragments_on_composite_types.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/known_argument_names.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/known_directives.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/known_fragment_names.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/known_root_type.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/known_type_names.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/lone_anonymous_operation.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/no_fragment_cycles.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/no_undefined_variables.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/no_unused_fragments.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/no_unused_variables.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/overlapping_fields_can_be_merged.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/possible_fragment_spreads.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/provided_required_arguments.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/scalar_leafs.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/single_field_subscriptions.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/unique_argument_names.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/unique_directives_per_location.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/unique_fragment_names.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/unique_input_field_names.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/unique_operation_names.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/unique_variable_names.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/values_of_correct_type.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/variables_are_input_types.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/variables_in_allowed_position.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/schema.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/schema_test.yml create mode 100644 vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/suggestionList.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/validator.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/vars.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/walk.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/json/patch/patch.go create mode 100644 vendor/github.com/open-policy-agent/opa/internal/ref/ref.go rename vendor/github.com/open-policy-agent/opa/{internal => }/ir/ir.go (71%) create mode 100644 vendor/github.com/open-policy-agent/opa/ir/marshal.go rename vendor/github.com/open-policy-agent/opa/{internal => }/ir/pretty.go (100%) rename vendor/github.com/open-policy-agent/opa/{internal => }/ir/walk.go (100%) create mode 100644 vendor/github.com/open-policy-agent/opa/loader/filter/filter.go create mode 100644 vendor/github.com/open-policy-agent/opa/loader/internal/embedtest/bar/bar.rego create mode 100644 vendor/github.com/open-policy-agent/opa/loader/internal/embedtest/bar/bar.yaml create mode 100644 vendor/github.com/open-policy-agent/opa/loader/internal/embedtest/baz/qux/qux.json create mode 100644 vendor/github.com/open-policy-agent/opa/loader/internal/embedtest/foo.json create mode 100644 vendor/github.com/open-policy-agent/opa/storage/inmem/opts.go create mode 100644 vendor/github.com/open-policy-agent/opa/topdown/graphql.go create mode 100644 vendor/github.com/open-policy-agent/opa/topdown/http_fixup.go create mode 100644 vendor/github.com/open-policy-agent/opa/topdown/http_fixup_darwin.go create mode 100644 vendor/github.com/open-policy-agent/opa/topdown/parse_units.go create mode 100644 vendor/github.com/open-policy-agent/opa/topdown/subset.go create mode 100644 vendor/github.com/open-policy-agent/opa/version/version_go1.18.go create mode 100644 vendor/github.com/stretchr/testify/assert/assertion_compare_can_convert.go create mode 100644 vendor/github.com/stretchr/testify/assert/assertion_compare_legacy.go create mode 100644 vendor/github.com/tchap/go-patricia/v2/AUTHORS create mode 100644 vendor/github.com/tchap/go-patricia/v2/LICENSE create mode 100644 vendor/github.com/tchap/go-patricia/v2/patricia/children.go create mode 100644 vendor/github.com/tchap/go-patricia/v2/patricia/patricia.go create mode 100644 vendor/golang.org/x/crypto/AUTHORS create mode 100644 vendor/golang.org/x/crypto/CONTRIBUTORS create mode 100644 vendor/golang.org/x/crypto/LICENSE create mode 100644 vendor/golang.org/x/crypto/PATENTS create mode 100644 vendor/golang.org/x/crypto/bcrypt/base64.go create mode 100644 vendor/golang.org/x/crypto/bcrypt/bcrypt.go create mode 100644 vendor/golang.org/x/crypto/blowfish/block.go create mode 100644 vendor/golang.org/x/crypto/blowfish/cipher.go create mode 100644 vendor/golang.org/x/crypto/blowfish/const.go diff --git a/auth/local.go b/auth/local.go index 6dd2ca0a..23275309 100644 --- a/auth/local.go +++ b/auth/local.go @@ -2,15 +2,21 @@ package auth import ( "context" + "crypto/subtle" "encoding/json" + "errors" "fmt" "log" "os" "path/filepath" "strings" + "golang.org/x/crypto/bcrypt" + + "github.com/open-policy-agent/opa/ast" "github.com/open-policy-agent/opa/rego" "github.com/open-policy-agent/opa/topdown" + "github.com/open-policy-agent/opa/types" ) // Policy is the OPA policy configuration method, which is returned from @@ -59,6 +65,45 @@ func NewLocalAuthorizer(policy Policy, cfg OPAConfig) (_ Authorizer, err error) r := rego.New( rego.Query(cfg.Query), policy, + rego.Function2( + ®o.Function{ + Name: "bcrypt_eq", + Decl: types.NewFunction(types.Args(types.S, types.S), types.B), + }, + func(_ rego.BuiltinContext, hash *ast.Term, pwd *ast.Term) (*ast.Term, error) { + hashStr, ok := hash.Value.(ast.String) + if !ok { + return nil, errors.New("Hash must be a string") + } + + pwdStr, ok := pwd.Value.(ast.String) + if !ok { + return nil, errors.New("Password must be a string") + } + + err := bcrypt.CompareHashAndPassword([]byte(hashStr), []byte(pwdStr)) + return ast.BooleanTerm(err == nil), nil + }, + ), + rego.Function2( + ®o.Function{ + Name: "constant_compare", + Decl: types.NewFunction(types.Args(types.S, types.S), types.B), + }, + func(_ rego.BuiltinContext, value1Term *ast.Term, value2Term *ast.Term) (*ast.Term, error) { + value1, ok := value1Term.Value.(ast.String) + if !ok { + return nil, errors.New("Value 1 must be a string") + } + + value2, ok := value2Term.Value.(ast.String) + if !ok { + return nil, errors.New("Value 2 must be a string") + } + + return ast.BooleanTerm(subtle.ConstantTimeCompare([]byte(value1), []byte(value2)) == 1), nil + }, + ), rego.EnablePrintStatements(cfg.Debug), rego.PrintHook(topdown.NewPrintHook(log.Writer())), ) diff --git a/auth/local_test.go b/auth/local_test.go index 8ed7b636..cd2f376a 100644 --- a/auth/local_test.go +++ b/auth/local_test.go @@ -2,8 +2,11 @@ package auth import ( "context" + "crypto/hmac" + "crypto/sha256" "encoding/base64" "encoding/json" + "fmt" "net/http" "testing" "time" @@ -147,6 +150,28 @@ func TestOPAAuthorizer(t *testing.T) { }, expected: true, }, + { + name: "can apply HMAC auth policy", + cfg: OPAConfig{ + Debug: true, + Query: "data.api.authz.hmac.allow", + }, + policy: "testdata/hmac_auth.rego", + input: Input{ + Method: http.MethodPost, + Path: "/api/private", + RawBody: `{"message": "hello world"}`, + Authorization: generateHMAC("secretvalue", `ts=2022-01-01T00:00:00Z,path=/api/private,method=POST,body={"message": "hello world"}`), + Headers: http.Header{ + "X-Auth-Timestamp": []string{"2022-01-01T00:00:00Z"}, + }, + // we load the key from a secret named "secretKey" + Data: map[string]string{ + "secretKey": "secretvalue", + }, + }, + expected: true, + }, } for _, tc := range cases { @@ -162,3 +187,13 @@ func TestOPAAuthorizer(t *testing.T) { }) } } + +func generateHMAC(key, data string) string { + h := hmac.New(sha256.New, []byte(key)) + h.Write([]byte(data)) + digest := string(h.Sum(nil)) + out := fmt.Sprintf("%x", digest) + fmt.Printf("data: %s\nhmac: %s\n---\n", data, out) + + return out +} diff --git a/auth/middleware.go b/auth/middleware.go index 548798e7..544eb876 100644 --- a/auth/middleware.go +++ b/auth/middleware.go @@ -34,6 +34,7 @@ type InputConfig struct { // authorizer rejects the request. ErrorContentType string AdditionalData map[string]string + SkipPaths map[string]struct{} } type Input struct { @@ -56,6 +57,14 @@ func InputConfigFromEnv() (cfg InputConfig, err error) { cfg.IncludeRawBody = truthy("OPA_INCLUDE_RAW_BODY", "false") cfg.IncludeHeaders = truthy("OPA_INCLUDE_HEADERS", "false") + cfg.SkipPaths = make(map[string]struct{}) + paths := os.Getenv("OPA_SKIP_PATHS") + for _, path := range strings.Split(paths, ",") { + cfg.SkipPaths[path] = struct{}{} + } + + //TODO: add SkipPattern and use a radix tree to support skip patterns? see go-chi and https://github.com/armon/go-radix + env := authEnviron() cfg.AdditionalData, err = loadAdditionalData(env) @@ -72,6 +81,11 @@ func New(impl Authorizer, cfg InputConfig) Middleware { return func(next http.Handler) http.Handler { return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) { + if _, ok := cfg.SkipPaths[r.URL.Path]; ok { + next.ServeHTTP(w, r) + return + } + input := Input{ Method: r.Method, Path: r.URL.Path, diff --git a/auth/testdata/basic_auth.rego b/auth/testdata/basic_auth.rego index fc607c4e..84bfd442 100644 --- a/auth/testdata/basic_auth.rego +++ b/auth/testdata/basic_auth.rego @@ -11,16 +11,17 @@ valid_basic_auth { value := trim_prefix(input.authorization, "Basic ") decoded := base64.decode(value) - print(decoded) - colon := indexof(decoded, ":") username := substring(decoded, 0, colon) password := substring(decoded, colon + 1, -1) - print(username) - print(password) - - credentials[username] == password + bcrypt_eq(credentials[username].hash, password) } -credentials := {"bob": "secretvalue"} +# generally don't store the plain value, it is only included here +# for demontration and ease of testing. +# this value would normally be loaded from a secret. +credentials := {"bob": { + "hash": "$2y$10$/7tG7c7RQKnLki5OveBiyejtOvdR5.I3wtoAUHBUO5azpg00bbUvy", + "plain": "secretvalue", +}} diff --git a/auth/testdata/hmac_auth.rego b/auth/testdata/hmac_auth.rego new file mode 100644 index 00000000..15143e95 --- /dev/null +++ b/auth/testdata/hmac_auth.rego @@ -0,0 +1,26 @@ +package api.authz.hmac + +default allow = false + +allow { + input.path = "/api/private" + valid_signature +} + +valid_signature { + authTimestamp = input.headers["X-Auth-Timestamp"][0] + data := sprintf( + "ts=%s,path=%s,method=%s,body=%s", + [ + authTimestamp, + input.path, + input.method, + input.rawBody, + ], + ) + + # we assume secretKey is read read from a single secret + generated := crypto.hmac.sha256(data, input.data.secretKey) + + constant_compare(generated, input.authorization) +} diff --git a/go.mod b/go.mod index 7f8f5daf..324fd4bd 100644 --- a/go.mod +++ b/go.mod @@ -4,35 +4,40 @@ go 1.18 require ( github.com/docker/go-units v0.5.0 + github.com/open-policy-agent/opa v0.45.0 github.com/openfaas/faas-middleware v1.2.2 github.com/prometheus/client_golang v1.13.0 + github.com/stretchr/testify v1.8.0 + golang.org/x/crypto v0.0.0-20220525230936-793ad666bf5e ) require ( github.com/OneOfOne/xxhash v1.2.8 // indirect - github.com/ghodss/yaml v1.0.0 // indirect - github.com/gobwas/glob v0.2.3 // indirect - github.com/pkg/errors v0.9.1 // indirect - github.com/pmezard/go-difflib v1.0.0 // indirect - github.com/rcrowley/go-metrics v0.0.0-20200313005456-10cdbea86bc0 // indirect - github.com/xeipuuv/gojsonpointer v0.0.0-20190905194746-02993c407bfb // indirect - github.com/xeipuuv/gojsonreference v0.0.0-20180127040603-bd5ef7bd5415 // indirect - github.com/yashtewari/glob-intersection v0.0.0-20180916065949-5c77d914dd0b // indirect - gopkg.in/yaml.v2 v2.4.0 // indirect - gopkg.in/yaml.v3 v3.0.0-20210107192922-496545a6307b // indirect -) - -require ( + github.com/agnivade/levenshtein v1.1.1 // indirect github.com/beorn7/perks v1.0.1 // indirect github.com/cespare/xxhash/v2 v2.1.2 // indirect github.com/davecgh/go-spew v1.1.1 // indirect + github.com/dgraph-io/badger/v3 v3.2103.3 // indirect + github.com/ghodss/yaml v1.0.0 // indirect + github.com/gobwas/glob v0.2.3 // indirect github.com/golang/protobuf v1.5.2 // indirect + github.com/google/flatbuffers v22.9.29+incompatible // indirect + github.com/google/go-cmp v0.5.9 // indirect + github.com/klauspost/compress v1.15.11 // indirect github.com/matttproud/golang_protobuf_extensions v1.0.2 // indirect - github.com/open-policy-agent/opa v0.36.1 - github.com/prometheus/client_model v0.2.0 // indirect + github.com/pkg/errors v0.9.1 // indirect + github.com/pmezard/go-difflib v1.0.0 // indirect + github.com/prometheus/client_model v0.3.0 // indirect github.com/prometheus/common v0.37.0 // indirect github.com/prometheus/procfs v0.8.0 // indirect - github.com/stretchr/testify v1.7.0 - golang.org/x/sys v0.0.0-20221013171732-95e765b1cc43 // indirect + github.com/rcrowley/go-metrics v0.0.0-20201227073835-cf1acfcdf475 // indirect + github.com/tchap/go-patricia/v2 v2.3.1 // indirect + github.com/xeipuuv/gojsonpointer v0.0.0-20190905194746-02993c407bfb // indirect + github.com/xeipuuv/gojsonreference v0.0.0-20180127040603-bd5ef7bd5415 // indirect + github.com/yashtewari/glob-intersection v0.1.0 // indirect + golang.org/x/net v0.1.0 // indirect + golang.org/x/sys v0.1.0 // indirect google.golang.org/protobuf v1.28.1 // indirect + gopkg.in/yaml.v2 v2.4.0 // indirect + gopkg.in/yaml.v3 v3.0.1 // indirect ) diff --git a/go.sum b/go.sum index ed61694e..414b3829 100644 --- a/go.sum +++ b/go.sum @@ -13,20 +13,6 @@ cloud.google.com/go v0.56.0/go.mod h1:jr7tqZxxKOVYizybht9+26Z/gUq7tiRzu+ACVAMbKV cloud.google.com/go v0.57.0/go.mod h1:oXiQ6Rzq3RAkkY7N6t3TcE6jE+CIBBbA36lwQ1JyzZs= cloud.google.com/go v0.62.0/go.mod h1:jmCYTdRCQuc1PHIIJ/maLInMho30T/Y0M4hTdTShOYc= cloud.google.com/go v0.65.0/go.mod h1:O5N8zS7uWy9vkA9vayVHs65eM1ubvY4h553ofrNHObY= -cloud.google.com/go v0.72.0/go.mod h1:M+5Vjvlc2wnp6tjzE102Dw08nGShTscUx2nZMufOKPI= -cloud.google.com/go v0.74.0/go.mod h1:VV1xSbzvo+9QJOxLDaJfTjx5e+MePCpCWwvftOeQmWk= -cloud.google.com/go v0.78.0/go.mod h1:QjdrLG0uq+YwhjoVOLsS1t7TW8fs36kLs4XO5R5ECHg= -cloud.google.com/go v0.79.0/go.mod h1:3bzgcEeQlzbuEAYu4mrWhKqWjmpprinYgKJLgKHnbb8= -cloud.google.com/go v0.81.0/go.mod h1:mk/AM35KwGk/Nm2YSeZbxXdrNK3KZOYHmLkOqC2V6E0= -cloud.google.com/go v0.83.0/go.mod h1:Z7MJUsANfY0pYPdw0lbnivPx4/vhy/e2FEkSkF7vAVY= -cloud.google.com/go v0.84.0/go.mod h1:RazrYuxIK6Kb7YrzzhPoLmCVzl7Sup4NrbKPg8KHSUM= -cloud.google.com/go v0.87.0/go.mod h1:TpDYlFy7vuLzZMMZ+B6iRiELaY7z/gJPaqbMx6mlWcY= -cloud.google.com/go v0.90.0/go.mod h1:kRX0mNRHe0e2rC6oNakvwQqzyDmg57xJ+SZU1eT2aDQ= -cloud.google.com/go v0.93.3/go.mod h1:8utlLll2EF5XMAV15woO4lSbWQlk8rer9aLOfLh7+YI= -cloud.google.com/go v0.94.1/go.mod h1:qAlAugsXlC+JWO+Bke5vCtc9ONxjQT3drlTTnAplMW4= -cloud.google.com/go v0.97.0/go.mod h1:GF7l59pYBVlXQIBLx3a761cZ41F9bBH3JUlihCt2Udc= -cloud.google.com/go v0.98.0/go.mod h1:ua6Ush4NALrHk5QXDWnjvZHN93OuF0HfuEPq9I1X0cM= -cloud.google.com/go v0.99.0/go.mod h1:w0Xx2nLzqWJPuozYQX+hFfCSI8WioryfRDzkoI/Y2ZA= cloud.google.com/go/bigquery v1.0.1/go.mod h1:i/xbL2UlR5RvWAURpBYZTtm/cXjCha9lbfbpx4poX+o= cloud.google.com/go/bigquery v1.3.0/go.mod h1:PjpwJnslEMmckchkHFfq+HTD2DmtT67aNFKH1/VBDHE= cloud.google.com/go/bigquery v1.4.0/go.mod h1:S8dzgnTigyfTmLBfrtrhyYhwRxG72rYxvftPBK2Dvzc= @@ -35,7 +21,6 @@ cloud.google.com/go/bigquery v1.7.0/go.mod h1://okPTzCYNXSlb24MZs83e2Do+h+VXtc4g cloud.google.com/go/bigquery v1.8.0/go.mod h1:J5hqkt3O0uAFnINi6JXValWIb1v0goeZM77hZzJN/fQ= cloud.google.com/go/datastore v1.0.0/go.mod h1:LXYbyblFSglQ5pkeyhO+Qmw7ukd3C+pD7TKLgZqpHYE= cloud.google.com/go/datastore v1.1.0/go.mod h1:umbIZjpQpHh4hmRpGhH4tLFup+FVzqBi1b3c64qFpCk= -cloud.google.com/go/firestore v1.6.1/go.mod h1:asNXNOzBdyVQmEU+ggO8UPodTkEVFW5Qx+rwHnAz+EY= cloud.google.com/go/pubsub v1.0.1/go.mod h1:R0Gpsv3s54REJCy4fxDixWD93lHJMoZTyQ2kNxGRt3I= cloud.google.com/go/pubsub v1.1.0/go.mod h1:EwwdRX2sKPjnvnqCa270oGRyludottCI76h+R3AArQw= cloud.google.com/go/pubsub v1.2.0/go.mod h1:jhfEVHT8odbXTkndysNHCcx0awwzvfOlguIAii9o8iA= @@ -48,32 +33,25 @@ cloud.google.com/go/storage v1.10.0/go.mod h1:FLPqc6j+Ki4BU591ie1oL6qBQGu2Bl/tZ9 dmitri.shuralyov.com/gpu/mtl v0.0.0-20190408044501-666a987793e9/go.mod h1:H6x//7gZCb22OMCxBHrMx7a5I7Hp++hsVxbQ4BYO7hU= github.com/BurntSushi/toml v0.3.1/go.mod h1:xHWCNGjB5oqiDr8zfno3MHue2Ht5sIBksp03qcyfWMU= github.com/BurntSushi/xgb v0.0.0-20160522181843-27f122750802/go.mod h1:IVnqGOEym/WlBOVXweHU+Q+/VP0lqqI8lqeDx9IjBqo= -github.com/DataDog/datadog-go v3.2.0+incompatible/go.mod h1:LButxg5PwREeZtORoXG3tL4fMGNddJ+vMq1mwgfaqoQ= github.com/OneOfOne/xxhash v1.2.2/go.mod h1:HSdplMjZKSmBqAxg5vPj2TmRDmfkzw+cTzAElWljhcU= github.com/OneOfOne/xxhash v1.2.8 h1:31czK/TI9sNkxIKfaUfGlU47BAxQ0ztGgd9vPyqimf8= github.com/OneOfOne/xxhash v1.2.8/go.mod h1:eZbhyaAYD41SGSSsnmcpxVoRiQ/MPUTjUdIIOT9Um7Q= +github.com/agnivade/levenshtein v1.1.1 h1:QY8M92nrzkmr798gCo3kmMyqXFzdQVpxLlGPRBij0P8= +github.com/agnivade/levenshtein v1.1.1/go.mod h1:veldBMzWxcCG2ZvUTKD2kJNRdCk5hVbJomOvKkmgYbo= github.com/alecthomas/template v0.0.0-20160405071501-a0175ee3bccc/go.mod h1:LOuyumcjzFXgccqObfd/Ljyb9UuFJ6TxHnclSeseNhc= github.com/alecthomas/template v0.0.0-20190718012654-fb15b899a751/go.mod h1:LOuyumcjzFXgccqObfd/Ljyb9UuFJ6TxHnclSeseNhc= github.com/alecthomas/units v0.0.0-20151022065526-2efee857e7cf/go.mod h1:ybxpYRFXyAe+OPACYpWeL0wqObRcbAqCMya13uyzqw0= github.com/alecthomas/units v0.0.0-20190717042225-c3de453c63f4/go.mod h1:ybxpYRFXyAe+OPACYpWeL0wqObRcbAqCMya13uyzqw0= github.com/alecthomas/units v0.0.0-20190924025748-f65c72e2690d/go.mod h1:rBZYJk541a8SKzHPHnH3zbiI+7dagKZ0cgpgrD7Fyho= -github.com/antihax/optional v1.0.0/go.mod h1:uupD/76wgC+ih3iEmQUL+0Ugr19nfwCT1kdvxnR2qWY= -github.com/armon/circbuf v0.0.0-20150827004946-bbbad097214e/go.mod h1:3U/XgcO3hCbHZ8TKRvWD2dDTCfh9M9ya+I9JpbB7O8o= +github.com/arbovm/levenshtein v0.0.0-20160628152529-48b4e1c0c4d0 h1:jfIu9sQUG6Ig+0+Ap1h4unLjW6YQJpKZVmUzxsD4E/Q= +github.com/arbovm/levenshtein v0.0.0-20160628152529-48b4e1c0c4d0/go.mod h1:t2tdKJDJF9BV14lnkjHmOQgcvEKgtqs5a1N3LNdJhGE= github.com/armon/consul-api v0.0.0-20180202201655-eb2c6b5be1b6/go.mod h1:grANhF5doyWs3UAsr3K4I6qtAmlQcZDesFNEHPZAzj8= -github.com/armon/go-metrics v0.0.0-20180917152333-f0300d1749da/go.mod h1:Q73ZrmVTwzkszR9V5SSuryQ31EELlFMUz1kKyl939pY= -github.com/armon/go-metrics v0.3.10/go.mod h1:4O98XIr/9W0sxpJ8UaYkvjk10Iff7SnFrb4QAOwNTFc= -github.com/armon/go-radix v0.0.0-20180808171621-7fddfc383310/go.mod h1:ufUuZ+zHj4x4TnLV4JWEpy2hxWSpsRywHrMgIH9cCH8= -github.com/armon/go-radix v1.0.0/go.mod h1:ufUuZ+zHj4x4TnLV4JWEpy2hxWSpsRywHrMgIH9cCH8= github.com/beorn7/perks v0.0.0-20180321164747-3a771d992973/go.mod h1:Dwedo/Wpr24TaqPxmxbtue+5NUziq4I4S80YR8gNf3Q= github.com/beorn7/perks v1.0.0/go.mod h1:KWe93zE9D1o94FZ5RNwFwVgaQK1VOXiVxmqh+CedLV8= github.com/beorn7/perks v1.0.1 h1:VlbKKnNfV8bJzeqoa4cOKqO6bYr3WgKZxO8Z16+hsOM= github.com/beorn7/perks v1.0.1/go.mod h1:G2ZrVWU2WbWT9wwq4/hrbKbnv/1ERSJQ0ibhJ6rlkpw= -github.com/bgentry/speakeasy v0.1.0/go.mod h1:+zsyZBPWlz7T6j88CTgSN5bM796AkVf0kBD4zp0CCIs= -github.com/bytecodealliance/wasmtime-go v0.32.0 h1:/GsrnJz2bfULAIZygN4vUElLYliQrx/o/1opP9X7Gck= -github.com/bytecodealliance/wasmtime-go v0.32.0/go.mod h1:q320gUxqyI8yB+ZqRuaJOEnGkAnHh6WtJjMaT2CW4wI= -github.com/cenkalti/backoff/v4 v4.1.2/go.mod h1:scbssz8iZGpm3xbr14ovlUdkxfGXNInqkPWOWmG2CLw= +github.com/bytecodealliance/wasmtime-go v1.0.0 h1:9u9gqaUiaJeN5IoD1L7egD8atOnTGyJcNp8BhkL9cUU= github.com/census-instrumentation/opencensus-proto v0.2.1/go.mod h1:f6KPmirojxKA12rnyqOA5BBL4O983OfeGPqjHWSTneU= -github.com/census-instrumentation/opencensus-proto v0.3.0/go.mod h1:f6KPmirojxKA12rnyqOA5BBL4O983OfeGPqjHWSTneU= github.com/cespare/xxhash v1.1.0 h1:a6HrQnmkObjyL+Gs60czilIUGqrzKutQD6XZog3p+ko= github.com/cespare/xxhash v1.1.0/go.mod h1:XrSqR1VqqWfGrhpAt58auRo0WTKS1nRRg3ghfAqPWnc= github.com/cespare/xxhash/v2 v2.1.1/go.mod h1:VGX0DQ3Q6kWi7AoAeZDth3/j3BFtOZR5XLFGgcrjCOs= @@ -82,36 +60,22 @@ github.com/cespare/xxhash/v2 v2.1.2/go.mod h1:VGX0DQ3Q6kWi7AoAeZDth3/j3BFtOZR5XL github.com/chzyer/logex v1.1.10/go.mod h1:+Ywpsq7O8HXn0nuIou7OrIPyXbp3wmkHB+jjWRnGsAI= github.com/chzyer/readline v0.0.0-20180603132655-2972be24d48e/go.mod h1:nSuG5e5PlCu98SY8svDHJxuZscDgtXS6KTTbou5AhLI= github.com/chzyer/test v0.0.0-20180213035817-a1ea475d72b1/go.mod h1:Q3SI9o4m/ZMnBNeIyt5eFwwo7qiLfzFZmjNmxjkiQlU= -github.com/circonus-labs/circonus-gometrics v2.3.1+incompatible/go.mod h1:nmEj6Dob7S7YxXgwXpfOuvO54S+tGdZdw9fuRZt25Ag= -github.com/circonus-labs/circonusllhist v0.1.3/go.mod h1:kMXHVDlOchFAehlya5ePtbp5jckzBHf4XRpQvBOLI+I= github.com/client9/misspell v0.3.4/go.mod h1:qj6jICC3Q7zFZvVWo7KLAzC3yx5G7kyvSDkc90ppPyw= github.com/cncf/udpa/go v0.0.0-20191209042840-269d4d468f6f/go.mod h1:M8M6+tZqaGXZJjfX53e64911xZQV5JYwmTeXPW+k8Sc= -github.com/cncf/udpa/go v0.0.0-20200629203442-efcf912fb354/go.mod h1:WmhPx2Nbnhtbo57+VJT5O0JRkEi1Wbu0z5j0R8u5Hbk= -github.com/cncf/udpa/go v0.0.0-20201120205902-5459f2c99403/go.mod h1:WmhPx2Nbnhtbo57+VJT5O0JRkEi1Wbu0z5j0R8u5Hbk= -github.com/cncf/udpa/go v0.0.0-20210930031921-04548b0d99d4/go.mod h1:6pvJx4me5XPnfI9Z40ddWsdw2W/uZgQLFXToKeRcDiI= -github.com/cncf/xds/go v0.0.0-20210312221358-fbca930ec8ed/go.mod h1:eXthEFrGJvWHgFFCl3hGmgk+/aYT6PnTQLykKQRLhEs= -github.com/cncf/xds/go v0.0.0-20210805033703-aa0b78936158/go.mod h1:eXthEFrGJvWHgFFCl3hGmgk+/aYT6PnTQLykKQRLhEs= -github.com/cncf/xds/go v0.0.0-20210922020428-25de7278fc84/go.mod h1:eXthEFrGJvWHgFFCl3hGmgk+/aYT6PnTQLykKQRLhEs= -github.com/cncf/xds/go v0.0.0-20211001041855-01bcc9b48dfe/go.mod h1:eXthEFrGJvWHgFFCl3hGmgk+/aYT6PnTQLykKQRLhEs= -github.com/cncf/xds/go v0.0.0-20211011173535-cb28da3451f1/go.mod h1:eXthEFrGJvWHgFFCl3hGmgk+/aYT6PnTQLykKQRLhEs= -github.com/cncf/xds/go v0.0.0-20211130200136-a8f946100490/go.mod h1:eXthEFrGJvWHgFFCl3hGmgk+/aYT6PnTQLykKQRLhEs= github.com/coreos/etcd v3.3.10+incompatible/go.mod h1:uF7uidLiAD3TWHmW31ZFd/JWoc32PjwdhPthX9715RE= github.com/coreos/go-etcd v2.0.0+incompatible/go.mod h1:Jez6KQU2B/sWsbdaef3ED8NzMklzPG4d5KIOhIy30Tk= github.com/coreos/go-semver v0.2.0/go.mod h1:nnelYz7RCh+5ahJtPPxZlU+153eP4D4r3EedlOD2RNk= -github.com/coreos/go-semver v0.3.0/go.mod h1:nnelYz7RCh+5ahJtPPxZlU+153eP4D4r3EedlOD2RNk= -github.com/coreos/go-systemd/v22 v22.3.2/go.mod h1:Y58oyj3AT4RCenI/lSvhwexgC+NSVTIJ3seZv2GcEnc= github.com/cpuguy83/go-md2man v1.0.10/go.mod h1:SmD6nW6nTyfqj6ABTjUi3V3JVMnlJmwcJI5acqYI6dE= -github.com/cpuguy83/go-md2man/v2 v2.0.1/go.mod h1:tgQtvFlXSQOSOSIRvRPT7W67SCa46tRHOmNcaadrF8o= -github.com/creack/pty v1.1.9/go.mod h1:oKZEueFk5CKHvIhNR5MUki03XCEU+Q6VDXinZuGJ33E= github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c= github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= -github.com/dgraph-io/badger/v3 v3.2103.2 h1:dpyM5eCJAtQCBcMCZcT4UBZchuTJgCywerHHgmxfxM8= -github.com/dgraph-io/badger/v3 v3.2103.2/go.mod h1:RHo4/GmYcKKh5Lxu63wLEMHJ70Pac2JqZRYGhlyAo2M= -github.com/dgraph-io/ristretto v0.1.0 h1:Jv3CGQHp9OjuMBSne1485aDpUkTKEcUqF+jm/LuerPI= -github.com/dgraph-io/ristretto v0.1.0/go.mod h1:fux0lOrBhrVCJd3lcTHsIJhq1T2rokOu6v9Vcb3Q9ug= +github.com/dgraph-io/badger/v3 v3.2103.3 h1:s63J1pisDhKpzWslXFe+ChuthuZptpwTE6qEKoczPb4= +github.com/dgraph-io/badger/v3 v3.2103.3/go.mod h1:4MPiseMeDQ3FNCYwRbbcBOGJLf5jsE0PPFzRiKjtcdw= +github.com/dgraph-io/ristretto v0.1.1 h1:6CWw5tJNgpegArSHpNHJKldNeq03FQCwYvfMVWajOK8= +github.com/dgraph-io/ristretto v0.1.1/go.mod h1:S1GPSBCYCIhmVNfcth17y2zZtQT6wzkzgwUve0VDWWA= github.com/dgryski/go-farm v0.0.0-20190423205320-6a90982ecee2/go.mod h1:SqUrOPUnsFjfmXRMNPybcSiG0BgUW2AuFH8PAnS2iTw= -github.com/dgryski/go-farm v0.0.0-20200201041132-a6ae2369ad13/go.mod h1:SqUrOPUnsFjfmXRMNPybcSiG0BgUW2AuFH8PAnS2iTw= +github.com/dgryski/trifles v0.0.0-20200323201526-dd97f9abfb48 h1:fRzb/w+pyskVMQ+UbP35JkH8yB7MYb4q/qhBarqZE6g= +github.com/dgryski/trifles v0.0.0-20200323201526-dd97f9abfb48/go.mod h1:if7Fbed8SFyPtHLHbg49SI7NAdJiC5WIA09pe59rfAA= github.com/docker/go-units v0.5.0 h1:69rxXcBk27SvSaaxTtLh/8llcHD8vYHT7WSdRZ/jvr4= github.com/docker/go-units v0.5.0/go.mod h1:fgPhTUdO+D/Jk86RDLlptpiXQzgHJF7gydDDbaIK4Dk= github.com/dustin/go-humanize v1.0.0 h1:VSnTsYCnlFHaM2/igO1h6X3HA71jcobQuxemgkq4zYo= @@ -119,30 +83,15 @@ github.com/dustin/go-humanize v1.0.0/go.mod h1:HtrtbFcZ19U5GC7JDqmcUSB87Iq5E25Kn github.com/envoyproxy/go-control-plane v0.9.0/go.mod h1:YTl/9mNaCwkRvm6d1a2C3ymFceY/DCBVvsKhRF0iEA4= github.com/envoyproxy/go-control-plane v0.9.1-0.20191026205805-5f8ba28d4473/go.mod h1:YTl/9mNaCwkRvm6d1a2C3ymFceY/DCBVvsKhRF0iEA4= github.com/envoyproxy/go-control-plane v0.9.4/go.mod h1:6rpuAdCZL397s3pYoYcLgu1mIlRU8Am5FuJP05cCM98= -github.com/envoyproxy/go-control-plane v0.9.7/go.mod h1:cwu0lG7PUMfa9snN8LXBig5ynNVH9qI8YYLbd1fK2po= -github.com/envoyproxy/go-control-plane v0.9.9-0.20201210154907-fd9021fe5dad/go.mod h1:cXg6YxExXjJnVBQHBLXeUAgxn2UodCpnH306RInaBQk= -github.com/envoyproxy/go-control-plane v0.9.9-0.20210217033140-668b12f5399d/go.mod h1:cXg6YxExXjJnVBQHBLXeUAgxn2UodCpnH306RInaBQk= -github.com/envoyproxy/go-control-plane v0.9.9-0.20210512163311-63b5d3c536b0/go.mod h1:hliV/p42l8fGbc6Y9bQ70uLwIvmJyVE5k4iMKlh8wCQ= -github.com/envoyproxy/go-control-plane v0.9.10-0.20210907150352-cf90f659a021/go.mod h1:AFq3mo9L8Lqqiid3OhADV3RfLJnjiw63cSpi+fDTRC0= -github.com/envoyproxy/go-control-plane v0.10.1/go.mod h1:AY7fTTXNdv/aJ2O5jwpxAPOWUZ7hQAEvzN5Pf27BkQQ= github.com/envoyproxy/protoc-gen-validate v0.1.0/go.mod h1:iSmxcyjqTsJpI2R4NaDN7+kN2VEUnK/pcBlmesArF7c= -github.com/envoyproxy/protoc-gen-validate v0.6.2/go.mod h1:2t7qjJNvHPx8IjnBOzl9E9/baC+qXE/TeeyBRzgJDws= -github.com/fatih/color v1.7.0/go.mod h1:Zm6kSWBoL9eyXnKyktHP6abPY2pDugNf5KwzbycvMj4= -github.com/fatih/color v1.9.0/go.mod h1:eQcE1qtQxscV5RaZvpXrrb8Drkc3/DdQ+uUYCNjL+zU= -github.com/fatih/color v1.13.0/go.mod h1:kLAiJbzzSOZDVNGyDpeOxJ47H46qBXwg5ILebYFFOfk= -github.com/felixge/httpsnoop v1.0.2/go.mod h1:m8KPJKqk1gH5J9DgRY2ASl2lWCfGKXixSwevea8zH2U= github.com/fortytw2/leaktest v1.3.0 h1:u8491cBMTQ8ft8aeV+adlcytMZylmA5nnwwkRZjI8vw= -github.com/fortytw2/leaktest v1.3.0/go.mod h1:jDsjWgpAGjm2CA7WthBh/CdZYEPF31XHquHwclZch5g= github.com/foxcpp/go-mockdns v0.0.0-20210729171921-fb145fc6f897 h1:E52jfcE64UG42SwLmrW0QByONfGynWuzBvm86BoB9z8= -github.com/foxcpp/go-mockdns v0.0.0-20210729171921-fb145fc6f897/go.mod h1:lgRN6+KxQBawyIghpnl5CezHFGS9VLzvtVlwxvzXTQ4= github.com/fsnotify/fsnotify v1.4.7/go.mod h1:jwhsz4b93w/PPRr/qN1Yymfu8t87LnFCMoQvtojpjFo= -github.com/fsnotify/fsnotify v1.5.1/go.mod h1:T3375wBYaZdLLcVNkcVbzGHY7f1l/uK5T5Ai1i3InKU= github.com/ghodss/yaml v1.0.0 h1:wQHKEahhL6wmXdzwWG11gIVCkOv05bNOh+Rxn0yngAk= github.com/ghodss/yaml v1.0.0/go.mod h1:4dBDuWmgqj2HViK6kFavaiC9ZROes6MMH2rRYeMEF04= github.com/go-gl/glfw v0.0.0-20190409004039-e6da0acd62b1/go.mod h1:vR7hzQXu2zJy9AVAgeJqvqgH9Q5CA+iKCZ2gyEVpxRU= github.com/go-gl/glfw/v3.3/glfw v0.0.0-20191125211704-12ad95a8df72/go.mod h1:tQ2UAYgL5IevRw8kRxooKSPJfGvJ9fJQFa0TUsXzTg8= github.com/go-gl/glfw/v3.3/glfw v0.0.0-20200222043503-6f7a984d4dc4/go.mod h1:tQ2UAYgL5IevRw8kRxooKSPJfGvJ9fJQFa0TUsXzTg8= -github.com/go-ini/ini v1.66.2/go.mod h1:ByCAeIL28uOIIG0E3PJtZPDL8WnHpFKFOtgjp+3Ies8= github.com/go-kit/kit v0.8.0/go.mod h1:xBxKIO96dXMWWy0MnWVtmwkA9/13aqxPnvrjFYMA2as= github.com/go-kit/kit v0.9.0/go.mod h1:xBxKIO96dXMWWy0MnWVtmwkA9/13aqxPnvrjFYMA2as= github.com/go-kit/log v0.1.0/go.mod h1:zbhenjAZHb184qTLMA9ZjW7ThYL0H2mk7Q6pNt4vbaY= @@ -151,25 +100,18 @@ github.com/go-logfmt/logfmt v0.3.0/go.mod h1:Qt1PoO58o5twSAckw1HlFXLmHsOX5/0LbT9 github.com/go-logfmt/logfmt v0.4.0/go.mod h1:3RMwSq7FuexP4Kalkev3ejPJsZTpXXBr9+V4qmtdjCk= github.com/go-logfmt/logfmt v0.5.0/go.mod h1:wCYkCAKZfumFQihp8CzCvQ3paCTfi41vtzG1KdI/P7A= github.com/go-logfmt/logfmt v0.5.1/go.mod h1:WYhtIu8zTZfxdn5+rREduYbwxfcBr/Vr6KEVveWlfTs= -github.com/go-logr/logr v1.2.0/go.mod h1:jdQByPbusPIv2/zmleS9BjJVeZ6kBagPoEUsqbVz/1A= -github.com/go-logr/logr v1.2.1/go.mod h1:jdQByPbusPIv2/zmleS9BjJVeZ6kBagPoEUsqbVz/1A= -github.com/go-logr/logr v1.2.2/go.mod h1:jdQByPbusPIv2/zmleS9BjJVeZ6kBagPoEUsqbVz/1A= -github.com/go-logr/stdr v1.2.0/go.mod h1:YkVgnZu1ZjjL7xTxrfm/LLZBfkhTqSR1ydtm6jTKKwI= github.com/go-stack/stack v1.8.0/go.mod h1:v0f6uXyyMGvRgIKkXu+yp6POWl0qKG85gN/melR3HDY= github.com/gobwas/glob v0.2.3 h1:A4xDbljILXROh+kObIiy5kIaPYD8e96x1tgBhUI5J+Y= github.com/gobwas/glob v0.2.3/go.mod h1:d3Ez4x06l9bZtSvzIay5+Yzi0fmZzPgnTbPcKjJAkT8= -github.com/godbus/dbus/v5 v5.0.4/go.mod h1:xhWf0FNVPg57R7Z0UbKHbJfkEywrmjJnf7w5xrFpKfA= github.com/gogo/protobuf v1.1.1/go.mod h1:r8qH/GZQm5c6nD/R0oafs1akxWv10x8SbQlK7atdtwQ= github.com/gogo/protobuf v1.3.2 h1:Ov1cvc58UF3b5XjBnZv7+opcTcQFZebYjWzi34vdm4Q= github.com/gogo/protobuf v1.3.2/go.mod h1:P1XiOD3dCwIKUDQYPy72D8LYyHL2YPYrpS2s69NZV8Q= github.com/golang/glog v0.0.0-20160126235308-23def4e6c14b/go.mod h1:SBH7ygxi8pfUlaOkMMuAQtPIUF8ecWP5IEl/CR7VP2Q= github.com/golang/glog v1.0.0 h1:nfP3RFugxnNRyKgeWd4oI1nYvXpxrx8ck8ZrcizshdQ= -github.com/golang/glog v1.0.0/go.mod h1:EWib/APOK0SL3dFbYqvxE3UYd8E6s1ouQ7iEp/0LWV4= github.com/golang/groupcache v0.0.0-20190702054246-869f871628b6/go.mod h1:cIg4eruTrX1D+g88fzRXU5OdNfaM+9IcxsU14FzY7Hc= github.com/golang/groupcache v0.0.0-20191227052852-215e87163ea7/go.mod h1:cIg4eruTrX1D+g88fzRXU5OdNfaM+9IcxsU14FzY7Hc= github.com/golang/groupcache v0.0.0-20200121045136-8c9f03a8e57e/go.mod h1:cIg4eruTrX1D+g88fzRXU5OdNfaM+9IcxsU14FzY7Hc= github.com/golang/groupcache v0.0.0-20210331224755-41bb18bfe9da h1:oI5xCqsCo564l8iNU+DwB5epxmsaqB+rhGL0m5jtYqE= -github.com/golang/groupcache v0.0.0-20210331224755-41bb18bfe9da/go.mod h1:cIg4eruTrX1D+g88fzRXU5OdNfaM+9IcxsU14FzY7Hc= github.com/golang/mock v1.1.1/go.mod h1:oTYuIxOrZwtPieC+H1uAHpcLFnEyAGVDL/k47Jfbm0A= github.com/golang/mock v1.2.0/go.mod h1:oTYuIxOrZwtPieC+H1uAHpcLFnEyAGVDL/k47Jfbm0A= github.com/golang/mock v1.3.1/go.mod h1:sBzyDLLjw3U8JLTeZvSv8jJB+tU5PVekmnlKIyFUx0Y= @@ -177,8 +119,6 @@ github.com/golang/mock v1.4.0/go.mod h1:UOMv5ysSaYNkG+OFQykRIcU/QvvxJf3p21QfJ2Bt github.com/golang/mock v1.4.1/go.mod h1:UOMv5ysSaYNkG+OFQykRIcU/QvvxJf3p21QfJ2Bt3cw= github.com/golang/mock v1.4.3/go.mod h1:UOMv5ysSaYNkG+OFQykRIcU/QvvxJf3p21QfJ2Bt3cw= github.com/golang/mock v1.4.4/go.mod h1:l3mdAwkq5BuhzHwde/uurv3sEJeZMXNpwsxVWU71h+4= -github.com/golang/mock v1.5.0/go.mod h1:CWnOUgYIOo4TcNZ0wHX3YZCqsaM1I1Jvs6v3mP3KVu8= -github.com/golang/mock v1.6.0/go.mod h1:p6yTPP+5HYm5mzsMV8JkE6ZKdX+/wYM6Hr+LicevLPs= github.com/golang/protobuf v1.2.0/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U= github.com/golang/protobuf v1.3.1/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U= github.com/golang/protobuf v1.3.2/go.mod h1:6lQm79b+lXiMfvg/cZm0SGofjICqVBUtrP5yJMmIC1U= @@ -194,16 +134,15 @@ github.com/golang/protobuf v1.4.1/go.mod h1:U8fpvMrcmy5pZrNK1lt4xCsGvpyWQ/VVv6QD github.com/golang/protobuf v1.4.2/go.mod h1:oDoupMAO8OvCJWAcko0GGGIgR6R6ocIYbsSw735rRwI= github.com/golang/protobuf v1.4.3/go.mod h1:oDoupMAO8OvCJWAcko0GGGIgR6R6ocIYbsSw735rRwI= github.com/golang/protobuf v1.5.0/go.mod h1:FsONVRAS9T7sI+LIUmWTfcYkHO4aIWwzhcaSAoJOfIk= -github.com/golang/protobuf v1.5.1/go.mod h1:DopwsBzvsk0Fs44TXzsVbJyPhcCPeIwnvohx4u74HPM= github.com/golang/protobuf v1.5.2 h1:ROPKBNFfQgOUMifHyP+KYbvpjbdoFNs+aK7DXlji0Tw= github.com/golang/protobuf v1.5.2/go.mod h1:XVQd3VNwM+JqD3oG2Ue2ip4fOMUkwXdXDdiuN0vRsmY= github.com/golang/snappy v0.0.3/go.mod h1:/XxbfmMg8lxefKM7IXC3fBNl/7bRcc72aCRzEWrmP2Q= github.com/golang/snappy v0.0.4 h1:yAGX7huGHXlcLOEtBnF4w7FQwA26wojNCwOYAEhLjQM= -github.com/golang/snappy v0.0.4/go.mod h1:/XxbfmMg8lxefKM7IXC3fBNl/7bRcc72aCRzEWrmP2Q= github.com/google/btree v0.0.0-20180813153112-4030bb1f1f0c/go.mod h1:lNA+9X1NB3Zf8V7Ke586lFgjr2dZNuvo3lPJSGZ5JPQ= github.com/google/btree v1.0.0/go.mod h1:lNA+9X1NB3Zf8V7Ke586lFgjr2dZNuvo3lPJSGZ5JPQ= -github.com/google/flatbuffers v1.12.1 h1:MVlul7pQNoDzWRLTw5imwYsl+usrS1TXG2H4jg6ImGw= github.com/google/flatbuffers v1.12.1/go.mod h1:1AeVuKshWv4vARoZatz6mlQ0JxURH0Kv5+zNeJKJCa8= +github.com/google/flatbuffers v22.9.29+incompatible h1:3UBb679lq3V/O9rgzoJmnkP1jJzmC9OdFzITUBkLU/A= +github.com/google/flatbuffers v22.9.29+incompatible/go.mod h1:1AeVuKshWv4vARoZatz6mlQ0JxURH0Kv5+zNeJKJCa8= github.com/google/go-cmp v0.2.0/go.mod h1:oXzfMopK8JAjlY9xF4vHSVASa0yLyX7SntLO5aqRK0M= github.com/google/go-cmp v0.3.0/go.mod h1:8QqcDgzrUqlUb/G2PQTWiueGozuR1884gddMywk6iLU= github.com/google/go-cmp v0.3.1/go.mod h1:8QqcDgzrUqlUb/G2PQTWiueGozuR1884gddMywk6iLU= @@ -211,17 +150,13 @@ github.com/google/go-cmp v0.4.0/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/ github.com/google/go-cmp v0.4.1/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= github.com/google/go-cmp v0.5.0/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= github.com/google/go-cmp v0.5.1/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= -github.com/google/go-cmp v0.5.2/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= -github.com/google/go-cmp v0.5.3/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= github.com/google/go-cmp v0.5.4/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= github.com/google/go-cmp v0.5.5/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= -github.com/google/go-cmp v0.5.6/go.mod h1:v8dTdLbMG2kIc/vJvl+f65V22dbkXbowE6jgT/gNBxE= -github.com/google/go-cmp v0.5.8 h1:e6P7q2lk1O+qJJb4BtCQXlK8vWEO8V1ZeuEdJNOqZyg= +github.com/google/go-cmp v0.5.9 h1:O2Tfq5qg4qc4AmwVlvv0oLiVAGB7enBSJ2x2DqQFi38= +github.com/google/go-cmp v0.5.9/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeNGIjoY= github.com/google/gofuzz v1.0.0/go.mod h1:dBl0BpW6vV/+mYPU4Po3pmUjxk6FQPldtuIdl/M65Eg= github.com/google/martian v2.1.0+incompatible/go.mod h1:9I4somxYTbIHy5NJKHRl3wXiIaQGbYVAs8BPL6v8lEs= github.com/google/martian/v3 v3.0.0/go.mod h1:y5Zk1BBys9G+gd6Jrk0W3cC1+ELVxBWuIGO+w/tUAp0= -github.com/google/martian/v3 v3.1.0/go.mod h1:y5Zk1BBys9G+gd6Jrk0W3cC1+ELVxBWuIGO+w/tUAp0= -github.com/google/martian/v3 v3.2.1/go.mod h1:oBOf6HBosgwRXnUGWUB05QECsc6uvmMiJ3+6W4l/CUk= github.com/google/pprof v0.0.0-20181206194817-3ea8567a2e57/go.mod h1:zfwlbNMJ+OItoe0UupaVj+oy1omPYYDuagoSzA8v9mc= github.com/google/pprof v0.0.0-20190515194954-54271f7e092f/go.mod h1:zfwlbNMJ+OItoe0UupaVj+oy1omPYYDuagoSzA8v9mc= github.com/google/pprof v0.0.0-20191218002539-d4f498aebedc/go.mod h1:ZgVRPoUq/hfqzAqh7sHMqb3I9Rq5C59dIz2SbBwJ4eM= @@ -229,58 +164,16 @@ github.com/google/pprof v0.0.0-20200212024743-f11f1df84d12/go.mod h1:ZgVRPoUq/hf github.com/google/pprof v0.0.0-20200229191704-1ebb73c60ed3/go.mod h1:ZgVRPoUq/hfqzAqh7sHMqb3I9Rq5C59dIz2SbBwJ4eM= github.com/google/pprof v0.0.0-20200430221834-fc25d7d30c6d/go.mod h1:ZgVRPoUq/hfqzAqh7sHMqb3I9Rq5C59dIz2SbBwJ4eM= github.com/google/pprof v0.0.0-20200708004538-1a94d8640e99/go.mod h1:ZgVRPoUq/hfqzAqh7sHMqb3I9Rq5C59dIz2SbBwJ4eM= -github.com/google/pprof v0.0.0-20201023163331-3e6fc7fc9c4c/go.mod h1:kpwsk12EmLew5upagYY7GY0pfYCcupk39gWOCRROcvE= -github.com/google/pprof v0.0.0-20201203190320-1bf35d6f28c2/go.mod h1:kpwsk12EmLew5upagYY7GY0pfYCcupk39gWOCRROcvE= -github.com/google/pprof v0.0.0-20210122040257-d980be63207e/go.mod h1:kpwsk12EmLew5upagYY7GY0pfYCcupk39gWOCRROcvE= -github.com/google/pprof v0.0.0-20210226084205-cbba55b83ad5/go.mod h1:kpwsk12EmLew5upagYY7GY0pfYCcupk39gWOCRROcvE= -github.com/google/pprof v0.0.0-20210601050228-01bbb1931b22/go.mod h1:kpwsk12EmLew5upagYY7GY0pfYCcupk39gWOCRROcvE= -github.com/google/pprof v0.0.0-20210609004039-a478d1d731e9/go.mod h1:kpwsk12EmLew5upagYY7GY0pfYCcupk39gWOCRROcvE= -github.com/google/pprof v0.0.0-20210720184732-4bb14d4b1be1/go.mod h1:kpwsk12EmLew5upagYY7GY0pfYCcupk39gWOCRROcvE= github.com/google/renameio v0.1.0/go.mod h1:KWCgfxg9yswjAJkECMjeO8J8rahYeXnNhOm40UhjYkI= -github.com/google/uuid v1.1.2/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= github.com/googleapis/gax-go/v2 v2.0.4/go.mod h1:0Wqv26UfaUD9n4G6kQubkQ+KchISgw+vpHVxEJEs9eg= github.com/googleapis/gax-go/v2 v2.0.5/go.mod h1:DWXyrwAJ9X0FpwwEdw+IPEYBICEFu5mhpdKc/us6bOk= -github.com/googleapis/gax-go/v2 v2.1.0/go.mod h1:Q3nei7sK6ybPYH7twZdmQpAd1MKb7pfu6SK+H1/DsU0= -github.com/googleapis/gax-go/v2 v2.1.1/go.mod h1:hddJymUZASv3XPyGkUpKj8pPO47Rmb0eJc8R6ouapiM= -github.com/gorilla/mux v1.8.0/go.mod h1:DVbg23sWSpFRCP0SfiEN6jmj59UnW/n46BH5rLB71So= -github.com/grpc-ecosystem/grpc-gateway v1.16.0/go.mod h1:BDjrQk3hbvj6Nolgz8mAMFbcEtjT1g+wF4CSlocrBnw= -github.com/hashicorp/consul/api v1.11.0/go.mod h1:XjsvQN+RJGWI2TWy1/kqaE16HrR2J/FWgkYjdZQsX9M= -github.com/hashicorp/consul/sdk v0.8.0/go.mod h1:GBvyrGALthsZObzUGsfgHZQDXjg4lOjagTIwIR1vPms= -github.com/hashicorp/errwrap v1.0.0/go.mod h1:YH+1FKiLXxHSkmPseP+kNlulaMuP3n2brvKWEqk/Jc4= -github.com/hashicorp/go-cleanhttp v0.5.0/go.mod h1:JpRdi6/HCYpAwUzNwuwqhbovhLtngrth3wmdIIUrZ80= -github.com/hashicorp/go-cleanhttp v0.5.1/go.mod h1:JpRdi6/HCYpAwUzNwuwqhbovhLtngrth3wmdIIUrZ80= -github.com/hashicorp/go-cleanhttp v0.5.2/go.mod h1:kO/YDlP8L1346E6Sodw+PrpBSV4/SoxCXGY6BqNFT48= -github.com/hashicorp/go-hclog v0.12.0/go.mod h1:whpDNt7SSdeAju8AWKIWsul05p54N/39EeqMAyrmvFQ= -github.com/hashicorp/go-hclog v1.0.0/go.mod h1:whpDNt7SSdeAju8AWKIWsul05p54N/39EeqMAyrmvFQ= -github.com/hashicorp/go-immutable-radix v1.0.0/go.mod h1:0y9vanUI8NX6FsYoO3zeMjhV/C5i9g4Q3DwcSNZ4P60= -github.com/hashicorp/go-immutable-radix v1.3.1/go.mod h1:0y9vanUI8NX6FsYoO3zeMjhV/C5i9g4Q3DwcSNZ4P60= -github.com/hashicorp/go-msgpack v0.5.3/go.mod h1:ahLV/dePpqEmjfWmKiqvPkv/twdG7iPBM1vqhUKIvfM= -github.com/hashicorp/go-multierror v1.0.0/go.mod h1:dHtQlpGsu+cZNNAkkCN/P3hoUDHhCYQXV3UM06sGGrk= -github.com/hashicorp/go-multierror v1.1.0/go.mod h1:spPvp8C1qA32ftKqdAHm4hHTbPw+vmowP0z+KUhOZdA= -github.com/hashicorp/go-retryablehttp v0.5.3/go.mod h1:9B5zBasrRhHXnJnui7y6sL7es7NDiJgTc6Er0maI1Xs= -github.com/hashicorp/go-rootcerts v1.0.2/go.mod h1:pqUvnprVnM5bf7AOirdbb01K4ccR319Vf4pU3K5EGc8= -github.com/hashicorp/go-sockaddr v1.0.0/go.mod h1:7Xibr9yA9JjQq1JpNB2Vw7kxv8xerXegt+ozgdvDeDU= -github.com/hashicorp/go-syslog v1.0.0/go.mod h1:qPfqrKkXGihmCqbJM2mZgkZGvKG1dFdvsLplgctolz4= -github.com/hashicorp/go-uuid v1.0.0/go.mod h1:6SBZvOh/SIDV7/2o3Jml5SYk/TvGqwFJ/bN7x4byOro= -github.com/hashicorp/go-uuid v1.0.1/go.mod h1:6SBZvOh/SIDV7/2o3Jml5SYk/TvGqwFJ/bN7x4byOro= github.com/hashicorp/golang-lru v0.5.0/go.mod h1:/m3WP610KZHVQ1SGc6re/UDhFvYD7pJ4Ao+sR/qLZy8= github.com/hashicorp/golang-lru v0.5.1/go.mod h1:/m3WP610KZHVQ1SGc6re/UDhFvYD7pJ4Ao+sR/qLZy8= -github.com/hashicorp/golang-lru v0.5.4/go.mod h1:iADmTwqILo4mZ8BN3D2Q6+9jd8WM5uGBxy+E8yxSoD4= github.com/hashicorp/hcl v1.0.0/go.mod h1:E5yfLk+7swimpb2L/Alb/PJmXilQ/rhwaUYs4T20WEQ= -github.com/hashicorp/logutils v1.0.0/go.mod h1:QIAnNjmIWmVIIkWDTG1z5v++HQmx9WQRO+LraFDTW64= -github.com/hashicorp/mdns v1.0.1/go.mod h1:4gW7WsVCke5TE7EPeYliwHlRUyBtfCwuFwuMg2DmyNY= -github.com/hashicorp/mdns v1.0.4/go.mod h1:mtBihi+LeNXGtG8L9dX59gAEa12BDtBQSp4v/YAJqrc= -github.com/hashicorp/memberlist v0.2.2/go.mod h1:MS2lj3INKhZjWNqd3N0m3J+Jxf3DAOnAH9VT3Sh9MUE= -github.com/hashicorp/memberlist v0.3.0/go.mod h1:MS2lj3INKhZjWNqd3N0m3J+Jxf3DAOnAH9VT3Sh9MUE= -github.com/hashicorp/serf v0.9.5/go.mod h1:UWDWwZeL5cuWDJdl0C6wrvrUwEqtQ4ZKBKKENpqIUyk= -github.com/hashicorp/serf v0.9.6/go.mod h1:TXZNMjZQijwlDvp+r0b63xZ45H7JmCmgg4gpTwn9UV4= -github.com/iancoleman/strcase v0.2.0/go.mod h1:iwCmte+B7n89clKwxIoIXy/HfoL7AsD47ZCWhYzw7ho= github.com/ianlancetaylor/demangle v0.0.0-20181102032728-5e5cf60278f6/go.mod h1:aSSvb/t6k1mPoxDqO4vJh6VOCGPwU4O0C2/Eqndh1Sc= -github.com/ianlancetaylor/demangle v0.0.0-20200824232613-28f6c0f3b639/go.mod h1:aSSvb/t6k1mPoxDqO4vJh6VOCGPwU4O0C2/Eqndh1Sc= github.com/inconshreveable/mousetrap v1.0.0/go.mod h1:PxqpIevigyE2G7u3NXJIT2ANytuPF1OarO4DADm73n8= github.com/jpillora/backoff v1.0.0/go.mod h1:J/6gKK9jxlEcS3zixgDgUAsiuZ7yrSoa/FX5e0EB2j4= github.com/json-iterator/go v1.1.6/go.mod h1:+SdeFBvtyEkXs7REEP0seUULqWtbJapLOCVDaaPEHmU= -github.com/json-iterator/go v1.1.9/go.mod h1:KdQUCv79m/52Kvf8AW2vK1V8akMuk1QjK/uOdHXbAo4= github.com/json-iterator/go v1.1.10/go.mod h1:KdQUCv79m/52Kvf8AW2vK1V8akMuk1QjK/uOdHXbAo4= github.com/json-iterator/go v1.1.11/go.mod h1:KdQUCv79m/52Kvf8AW2vK1V8akMuk1QjK/uOdHXbAo4= github.com/json-iterator/go v1.1.12/go.mod h1:e30LSqwooZae/UwlEbR2852Gd8hjQvJoHmT4TnhNGBo= @@ -291,48 +184,23 @@ github.com/julienschmidt/httprouter v1.3.0/go.mod h1:JR6WtHb+2LUe8TCKY3cZOxFyyO8 github.com/kisielk/errcheck v1.5.0/go.mod h1:pFxgyoBC7bSaBwPgfKdkLd5X25qrDl4LWUI2bnpBCr8= github.com/kisielk/gotool v1.0.0/go.mod h1:XhKaO+MFFWcvkIS/tQcRk01m1F5IRFswLeQ+oQHNcck= github.com/klauspost/compress v1.12.3/go.mod h1:8dP1Hq4DHOhN9w426knH3Rhby4rFm6D8eO+e+Dq5Gzg= -github.com/klauspost/compress v1.13.5 h1:9O69jUPDcsT9fEm74W92rZL9FQY7rCdaXVneq+yyzl4= -github.com/klauspost/compress v1.13.5/go.mod h1:/3/Vjq9QcHkK5uEr5lBEmyoZ1iFhe47etQ6QUkpK6sk= +github.com/klauspost/compress v1.15.11 h1:Lcadnb3RKGin4FYM/orgq0qde+nc15E5Cbqg4B9Sx9c= +github.com/klauspost/compress v1.15.11/go.mod h1:QPwzmACJjUTFsnSHH934V6woptycfrDDJnH7hvFVbGM= github.com/konsorten/go-windows-terminal-sequences v1.0.1/go.mod h1:T0+1ngSBFLxvqU3pZ+m/2kptfBszLMUkC4ZK/EgS/cQ= github.com/konsorten/go-windows-terminal-sequences v1.0.3/go.mod h1:T0+1ngSBFLxvqU3pZ+m/2kptfBszLMUkC4ZK/EgS/cQ= -github.com/kr/fs v0.1.0/go.mod h1:FFnZGqtBN9Gxj7eW1uZ42v5BccTP0vu6NEaFoC2HwRg= github.com/kr/logfmt v0.0.0-20140226030751-b84e30acd515/go.mod h1:+0opPa2QZZtGFBFZlji/RkVcI2GknAs/DXo4wKdlNEc= +github.com/kr/pretty v0.1.0 h1:L/CwN0zerZDmRFUapSPitk6f+Q3+0za1rQkzVuMiMFI= github.com/kr/pretty v0.1.0/go.mod h1:dAy3ld7l9f0ibDNOQOHHMYYIIbhfbHSm3C4ZsoJORNo= -github.com/kr/pretty v0.2.0/go.mod h1:ipq/a2n7PKx3OHsz4KJII5eveXtPO4qwEXGdVfWzfnI= github.com/kr/pty v1.1.1/go.mod h1:pFQYn66WHrOpPYNljwOMqo10TkYh1fy3cYio2l3bCsQ= github.com/kr/text v0.1.0/go.mod h1:4Jbv+DJW3UT/LiOwJeYQe1efqtUx/iVham/4vfdArNI= github.com/kr/text v0.2.0 h1:5Nx0Ya0ZqY2ygV366QzturHI13Jq95ApcVaJBhpS+AY= -github.com/kr/text v0.2.0/go.mod h1:eLer722TekiGuMkidMxC/pM04lWEeraHUUmBw8l2grE= -github.com/lyft/protoc-gen-star v0.5.3/go.mod h1:V0xaHgaf5oCCqmcxYcWiDfTiKsZsRc87/1qhoTACD8w= github.com/magiconair/properties v1.8.0/go.mod h1:PppfXfuXeibc/6YijjN8zIbojt8czPbwD3XqdrwzmxQ= -github.com/magiconair/properties v1.8.5/go.mod h1:y3VJvCyxH9uVvJTWEGAELF3aiYNyPKd5NZ3oSwXrF60= -github.com/mattn/go-colorable v0.0.9/go.mod h1:9vuHe8Xs5qXnSaW/c/ABM9alt+Vo+STaOChaDxuIBZU= -github.com/mattn/go-colorable v0.1.4/go.mod h1:U0ppj6V5qS13XJ6of8GYAs25YV2eR4EVcfRqFIhoBtE= -github.com/mattn/go-colorable v0.1.6/go.mod h1:u6P/XSegPjTcexA+o6vUJrdnUu04hMope9wVRipJSqc= -github.com/mattn/go-colorable v0.1.9/go.mod h1:u6P/XSegPjTcexA+o6vUJrdnUu04hMope9wVRipJSqc= -github.com/mattn/go-colorable v0.1.12/go.mod h1:u5H1YNBxpqRaxsYJYSkiCWKzEfiAb1Gb520KVy5xxl4= -github.com/mattn/go-isatty v0.0.3/go.mod h1:M+lRXTBqGeGNdLjl/ufCoiOlB5xdOkqRJdNxMWT7Zi4= -github.com/mattn/go-isatty v0.0.8/go.mod h1:Iq45c/XA43vh69/j3iqttzPXn0bhXyGjM0Hdxcsrc5s= -github.com/mattn/go-isatty v0.0.10/go.mod h1:qgIWMr58cqv1PHHyhnkY9lrL7etaEgOFcMEpPG5Rm84= -github.com/mattn/go-isatty v0.0.11/go.mod h1:PhnuNfih5lzO57/f3n+odYbM4JtupLOxQOAqxQCu2WE= -github.com/mattn/go-isatty v0.0.12/go.mod h1:cbi8OIDigv2wuxKPP5vlRcQ1OAZbq2CE4Kysco4FUpU= -github.com/mattn/go-isatty v0.0.14/go.mod h1:7GGIvUiUoEMVVmxf/4nioHXj79iQHKdU27kJ6hsGG94= -github.com/mattn/go-runewidth v0.0.9/go.mod h1:H031xJmbD/WCDINGzjvQ9THkh0rPKHF+m2gUSrubnMI= github.com/matttproud/golang_protobuf_extensions v1.0.1/go.mod h1:D8He9yQNgCq6Z5Ld7szi9bcBfOoFv/3dc6xSMkL2PC0= github.com/matttproud/golang_protobuf_extensions v1.0.2 h1:hAHbPm5IJGijwng3PWk09JkG9WeqChjprR5s9bBZ+OM= github.com/matttproud/golang_protobuf_extensions v1.0.2/go.mod h1:BSXmuO+STAnVfrANrmjBb36TMTDstsz7MSK+HVaYKv4= -github.com/miekg/dns v1.0.14/go.mod h1:W1PPwlIAgtquWBMBEV9nkV9Cazfe8ScdGz/Lj7v3Nrg= -github.com/miekg/dns v1.1.25/go.mod h1:bPDLeHnStXmXAq1m/Ch/hvfNHr14JKNPMBo3VZKjuso= -github.com/miekg/dns v1.1.26/go.mod h1:bPDLeHnStXmXAq1m/Ch/hvfNHr14JKNPMBo3VZKjuso= -github.com/miekg/dns v1.1.41/go.mod h1:p6aan82bvRIyn+zDIv9xYNUpwa73JcSh9BKwknJysuI= github.com/miekg/dns v1.1.43 h1:JKfpVSCB84vrAmHzyrsxB5NAr5kLoMXZArPSw7Qlgyg= -github.com/miekg/dns v1.1.43/go.mod h1:+evo5L0630/F6ca/Z9+GAqzhjGyn8/c+TBaOyfEl0V4= -github.com/mitchellh/cli v1.1.0/go.mod h1:xcISNoH86gajksDmfB23e/pu+B+GeFRMYmoHXxx3xhI= github.com/mitchellh/go-homedir v1.1.0/go.mod h1:SfyaCUpYCn1Vlf4IUYiD9fPX4A5wJrkLzIz1N1q0pr0= -github.com/mitchellh/go-testing-interface v1.0.0/go.mod h1:kRemZodwjscx+RGhAo8eIhFbs2+BFgRtFPeD/KE+zxI= -github.com/mitchellh/mapstructure v0.0.0-20160808181253-ca63d7c062ee/go.mod h1:FVVH3fgwuzCH5S8UJGiWEs2h04kUh9fWfEaFds41c1Y= github.com/mitchellh/mapstructure v1.1.2/go.mod h1:FVVH3fgwuzCH5S8UJGiWEs2h04kUh9fWfEaFds41c1Y= -github.com/mitchellh/mapstructure v1.4.3/go.mod h1:bFUtVrKA4DC2yAKiSyO/QUcy7e+RRV2QTWOzhPopBRo= github.com/modern-go/concurrent v0.0.0-20180228061459-e0a39a4cb421/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q= github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q= github.com/modern-go/reflect2 v0.0.0-20180701023420-4b7aa43c6742/go.mod h1:bx2lNnkwVCuqBIxFjflWJWanXIb3RllmbCylyMrvgv0= @@ -340,30 +208,19 @@ github.com/modern-go/reflect2 v1.0.1/go.mod h1:bx2lNnkwVCuqBIxFjflWJWanXIb3Rllmb github.com/modern-go/reflect2 v1.0.2/go.mod h1:yWuevngMOJpCy52FWWMvUC8ws7m/LJsjYzDa0/r8luk= github.com/mwitkow/go-conntrack v0.0.0-20161129095857-cc309e4a2223/go.mod h1:qRWi+5nqEBWmkhHvq77mSJWrCKwh8bxhgT7d/eI7P4U= github.com/mwitkow/go-conntrack v0.0.0-20190716064945-2f068394615f/go.mod h1:qRWi+5nqEBWmkhHvq77mSJWrCKwh8bxhgT7d/eI7P4U= -github.com/niemeyer/pretty v0.0.0-20200227124842-a10e7caefd8e h1:fD57ERR4JtEqsWbfPhv4DMiApHyliiK5xCTNVSPiaAs= -github.com/niemeyer/pretty v0.0.0-20200227124842-a10e7caefd8e/go.mod h1:zD1mROLANZcx1PVRCS0qkT7pwLkGfwJo4zjcN/Tysno= -github.com/olekukonko/tablewriter v0.0.5/go.mod h1:hPp6KlRPjbx+hW8ykQs1w3UBbZlj6HuIJcUGPhkA7kY= -github.com/open-policy-agent/opa v0.36.1 h1:FkopbI3Rlor1nAvu78hgbdOLZyRd4vPepeMY4Tcm1Hc= -github.com/open-policy-agent/opa v0.36.1/go.mod h1:/ZcbCeVlsRFp+n7aAvcTfElCL24b53p9v9QGo2y+0RM= +github.com/open-policy-agent/opa v0.45.0 h1:P5nuhVRtR+e58fk3CMMbiqr6ZFyWQPNOC3otsorGsFs= +github.com/open-policy-agent/opa v0.45.0/go.mod h1:/OnsYljNEWJ6DXeFOOnoGn8CvwZGMUS4iRqzYdJvmBI= github.com/openfaas/faas-middleware v1.2.2 h1:0aFr55OSODCR1t6FQ5LoOetgL3RzCZTz5doHI1mAdwo= github.com/openfaas/faas-middleware v1.2.2/go.mod h1:RgkVC/llBh+Eqb4bKxcFneB4OMnYsjUrEs7TWrRf51s= -github.com/pascaldekloe/goe v0.0.0-20180627143212-57f6aae5913c/go.mod h1:lzWF7FIEvWOWxwDKqyGYQf6ZUaNfKdP144TG7ZOy1lc= -github.com/pascaldekloe/goe v0.1.0/go.mod h1:lzWF7FIEvWOWxwDKqyGYQf6ZUaNfKdP144TG7ZOy1lc= github.com/pelletier/go-toml v1.2.0/go.mod h1:5z9KED0ma1S8pY6P1sdut58dfprrGBbd/94hg7ilaic= -github.com/pelletier/go-toml v1.9.4/go.mod h1:u1nR/EPcESfeI/szUZKdtJ0xRNbUoANCkoOuaOx1Y+c= -github.com/peterh/liner v0.0.0-20170211195444-bf27d3ba8e1d/go.mod h1:xIteQHvHuaLYG9IFj6mSxM0fCKrs34IrEQUhOYuGPHc= github.com/pkg/errors v0.8.0/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0= github.com/pkg/errors v0.8.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0= github.com/pkg/errors v0.9.1 h1:FEBLx1zS214owpjy7qsBeixbURkuhQAwrK5UwLGTwt4= github.com/pkg/errors v0.9.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0= -github.com/pkg/sftp v1.10.1/go.mod h1:lYOWFsE0bwd1+KfKJaKeuokY15vzFx25BLbzYYoAxZI= github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM= github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4= -github.com/posener/complete v1.1.1/go.mod h1:em0nMJCgc9GFtwrmVmEMR/ZL6WyhyjMBndrE9hABlRI= -github.com/posener/complete v1.2.3/go.mod h1:WZIdtGGp+qx0sLrYKtIRAruyNpv6hFCicSgv7Sy7s/s= github.com/prometheus/client_golang v0.9.1/go.mod h1:7SWBe2y4D6OKWSNQJUaRYU/AaXPKyh/dDVn+NZz0KFw= github.com/prometheus/client_golang v1.0.0/go.mod h1:db9x61etRT2tGnBNRi70OPL5FsnadC4Ky3P0J6CfImo= -github.com/prometheus/client_golang v1.4.0/go.mod h1:e9GMxYsXl05ICDXkRhurwBS4Q3OK1iX/F2sw+iXX5zU= github.com/prometheus/client_golang v1.7.1/go.mod h1:PY5Wy2awLA44sXw4AOSfFBetzPP4j5+D6mVACh+pe2M= github.com/prometheus/client_golang v1.11.0/go.mod h1:Z6t4BnS23TR94PD6BsDNk8yVqroYurpAkEiz0P2BEV0= github.com/prometheus/client_golang v1.12.1/go.mod h1:3Z9XVyYiZYEO+YQWt3RD2R3jrbd179Rt297l4aS6nDY= @@ -372,79 +229,61 @@ github.com/prometheus/client_golang v1.13.0/go.mod h1:vTeo+zgvILHsnnj/39Ou/1fPN5 github.com/prometheus/client_model v0.0.0-20180712105110-5c3871d89910/go.mod h1:MbSGuTsp3dbXC40dX6PRTWyKYBIrTGTE9sqQNg2J8bo= github.com/prometheus/client_model v0.0.0-20190129233127-fd36f4220a90/go.mod h1:xMI15A0UPsDsEKsMN9yxemIoYk6Tm2C1GtYGdfGttqA= github.com/prometheus/client_model v0.0.0-20190812154241-14fe0d1b01d4/go.mod h1:xMI15A0UPsDsEKsMN9yxemIoYk6Tm2C1GtYGdfGttqA= -github.com/prometheus/client_model v0.2.0 h1:uq5h0d+GuxiXLJLNABMgp2qUWDPiLvgCzz2dUR+/W/M= github.com/prometheus/client_model v0.2.0/go.mod h1:xMI15A0UPsDsEKsMN9yxemIoYk6Tm2C1GtYGdfGttqA= +github.com/prometheus/client_model v0.3.0 h1:UBgGFHqYdG/TPFD1B1ogZywDqEkwp3fBMvqdiQ7Xew4= +github.com/prometheus/client_model v0.3.0/go.mod h1:LDGWKZIo7rky3hgvBe+caln+Dr3dPggB5dvjtD7w9+w= github.com/prometheus/common v0.4.1/go.mod h1:TNfzLD0ON7rHzMJeJkieUDPYmFC7Snx/y86RQel1bk4= -github.com/prometheus/common v0.9.1/go.mod h1:yhUN8i9wzaXS3w1O07YhxHEBxD+W35wd8bs7vj7HSQ4= github.com/prometheus/common v0.10.0/go.mod h1:Tlit/dnDKsSWFlCLTWaA1cyBgKHSMdTB80sz/V91rCo= github.com/prometheus/common v0.26.0/go.mod h1:M7rCNAaPfAosfx8veZJCuw84e35h3Cfd9VFqTh1DIvc= -github.com/prometheus/common v0.29.0/go.mod h1:vu+V0TpY+O6vW9J44gczi3Ap/oXXR10b+M/gUGO4Hls= github.com/prometheus/common v0.32.1/go.mod h1:vu+V0TpY+O6vW9J44gczi3Ap/oXXR10b+M/gUGO4Hls= github.com/prometheus/common v0.37.0 h1:ccBbHCgIiT9uSoFY0vX8H3zsNR5eLt17/RQLUvn8pXE= github.com/prometheus/common v0.37.0/go.mod h1:phzohg0JFMnBEFGxTDbfu3QyL5GI8gTQJFhYO5B3mfA= github.com/prometheus/procfs v0.0.0-20181005140218-185b4288413d/go.mod h1:c3At6R/oaqEKCNdg8wHV1ftS6bRYblBhIjjI8uT2IGk= github.com/prometheus/procfs v0.0.2/go.mod h1:TjEm7ze935MbeOT/UhFTIMYKhuLP4wbCsTZCD3I8kEA= -github.com/prometheus/procfs v0.0.8/go.mod h1:7Qr8sr6344vo1JqZ6HhLceV9o3AJ1Ff+GxbHq6oeK9A= github.com/prometheus/procfs v0.1.3/go.mod h1:lV6e/gmhEcM9IjHGsFOCxxuZ+z1YqCvr4OA4YeYWdaU= github.com/prometheus/procfs v0.6.0/go.mod h1:cz+aTbrPOrUb4q7XlbU9ygM+/jj0fzG6c1xBZuNvfVA= github.com/prometheus/procfs v0.7.3/go.mod h1:cz+aTbrPOrUb4q7XlbU9ygM+/jj0fzG6c1xBZuNvfVA= github.com/prometheus/procfs v0.8.0 h1:ODq8ZFEaYeCaZOJlZZdJA2AbQR98dSHSM1KW/You5mo= github.com/prometheus/procfs v0.8.0/go.mod h1:z7EfXMXOkbkqb9IINtpCn86r/to3BnA0uaxHdg830/4= -github.com/rcrowley/go-metrics v0.0.0-20200313005456-10cdbea86bc0 h1:MkV+77GLUNo5oJ0jf870itWm3D0Sjh7+Za9gazKc5LQ= -github.com/rcrowley/go-metrics v0.0.0-20200313005456-10cdbea86bc0/go.mod h1:bCqnVzQkZxMG4s8nGwiZ5l3QUCyqpo9Y+/ZMZ9VjZe4= -github.com/rogpeppe/fastuuid v1.2.0/go.mod h1:jVj6XXZzXRy/MSR5jhDC/2q6DgLz+nrA6LYCDYWNEvQ= +github.com/rcrowley/go-metrics v0.0.0-20201227073835-cf1acfcdf475 h1:N/ElC8H3+5XpJzTSTfLsJV/mx9Q9g7kxmchpfZyxgzM= +github.com/rcrowley/go-metrics v0.0.0-20201227073835-cf1acfcdf475/go.mod h1:bCqnVzQkZxMG4s8nGwiZ5l3QUCyqpo9Y+/ZMZ9VjZe4= github.com/rogpeppe/go-internal v1.3.0/go.mod h1:M8bDsm7K2OlrFYOpmOWEs/qY81heoFRclV5y23lUDJ4= github.com/russross/blackfriday v1.5.2/go.mod h1:JO/DiYxRf+HjHt06OyowR9PTA263kcR/rfWxYHBV53g= -github.com/russross/blackfriday/v2 v2.1.0/go.mod h1:+Rmxgy9KzJVeS9/2gXHxylqXiyQDYRxCVz55jmeOWTM= -github.com/ryanuber/columnize v0.0.0-20160712163229-9b3edd62028f/go.mod h1:sm1tb6uqfes/u+d4ooFouqFdy9/2g9QGwK3SQygK0Ts= -github.com/sagikazarmark/crypt v0.3.0/go.mod h1:uD/D+6UF4SrIR1uGEv7bBNkNqLGqUr43MRiaGWX1Nig= -github.com/sean-/seed v0.0.0-20170313163322-e2103e2c3529/go.mod h1:DxrIzT+xaE7yg65j358z/aeFdxmN0P9QXhEzd20vsDc= github.com/sirupsen/logrus v1.2.0/go.mod h1:LxeOpSwHxABJmUn/MG1IvRgCAasNZTLOkJPxbbu5VWo= github.com/sirupsen/logrus v1.4.2/go.mod h1:tLMulIdttU9McNUspp0xgXVQah82FyeX6MwdIuYE2rE= github.com/sirupsen/logrus v1.6.0/go.mod h1:7uNnSEd1DgxDLC74fIahvMZmmYsHGZGEOFrfsX/uA88= -github.com/sirupsen/logrus v1.8.1/go.mod h1:yWOB1SBYBC5VeMP7gHvWumXLIWorT60ONWic61uBYv0= +github.com/sirupsen/logrus v1.9.0 h1:trlNQbNUG3OdDrDil03MCb1H2o9nJ1x4/5LYw7byDE0= github.com/spaolacci/murmur3 v0.0.0-20180118202830-f09979ecbc72/go.mod h1:JwIasOWyU6f++ZhiEuf87xNszmSA2myDM2Kzu9HwQUA= github.com/spaolacci/murmur3 v1.1.0/go.mod h1:JwIasOWyU6f++ZhiEuf87xNszmSA2myDM2Kzu9HwQUA= github.com/spf13/afero v1.1.2/go.mod h1:j4pytiNVoe2o6bmDsKpLACNPDBIoEAkihy7loJ1B0CQ= -github.com/spf13/afero v1.3.3/go.mod h1:5KUK8ByomD5Ti5Artl0RtHeI5pTF7MIDuXL3yY520V4= -github.com/spf13/afero v1.6.0/go.mod h1:Ai8FlHk4v/PARR026UzYexafAt9roJ7LcLMAmO6Z93I= github.com/spf13/cast v1.3.0/go.mod h1:Qx5cxh0v+4UWYiBimWS+eyWzqEqokIECu5etghLkUJE= -github.com/spf13/cast v1.4.1/go.mod h1:Qx5cxh0v+4UWYiBimWS+eyWzqEqokIECu5etghLkUJE= github.com/spf13/cobra v0.0.5/go.mod h1:3K3wKZymM7VvHMDS9+Akkh4K60UwM26emMESw8tLCHU= -github.com/spf13/cobra v1.3.0/go.mod h1:BrRVncBjOJa/eUcVVm9CE+oC6as8k+VYr4NY7WCi9V4= github.com/spf13/jwalterweatherman v1.0.0/go.mod h1:cQK4TGJAtQXfYWX+Ddv3mKDzgVb68N+wFjFa4jdeBTo= -github.com/spf13/jwalterweatherman v1.1.0/go.mod h1:aNWZUN0dPAAO/Ljvb5BEdw96iTZ0EXowPYD95IqWIGo= github.com/spf13/pflag v1.0.3/go.mod h1:DYY7MBk1bdzusC3SYhjObp+wFpr4gzcvqqNjLnInEg4= -github.com/spf13/pflag v1.0.5/go.mod h1:McXfInJRrz4CZXVZOBLb0bTZqETkiAhM9Iw0y3An2Bg= github.com/spf13/viper v1.3.2/go.mod h1:ZiWeW+zYFKm7srdB9IoDzzZXaJaI5eL9QjNiN/DMA2s= -github.com/spf13/viper v1.10.0/go.mod h1:SoyBPwAtKDzypXNDFKN5kzH7ppppbGZtls1UpIy5AsM= github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME= github.com/stretchr/objx v0.1.1/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME= +github.com/stretchr/objx v0.4.0/go.mod h1:YvHI0jy2hoMjB+UWwv71VJQ9isScKT/TqJzVSSt89Yw= github.com/stretchr/testify v1.2.2/go.mod h1:a8OnRcib4nhh0OaRAV+Yts87kKdq0PP7pXfy6kDkUVs= github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI= github.com/stretchr/testify v1.4.0/go.mod h1:j7eGeouHqKxXV5pUuKE4zz7dFj8WfuZ+81PSLYec5m4= -github.com/stretchr/testify v1.5.1/go.mod h1:5W2xD1RspED5o8YsWQXVCued0rvSQ+mT+I5cxcmMvtA= -github.com/stretchr/testify v1.6.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= -github.com/stretchr/testify v1.7.0 h1:nwc3DEeHmmLAfoZucVR881uASk0Mfjw8xYJ99tb5CcY= -github.com/stretchr/testify v1.7.0/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= -github.com/subosito/gotenv v1.2.0/go.mod h1:N0PQaV/YGNqwC0u51sEeR/aUtSLEXKX9iv69rRypqCw= -github.com/tv42/httpunix v0.0.0-20150427012821-b75d8614f926/go.mod h1:9ESjWnEqriFuLhtthL60Sar/7RFoluCcXsuvEwTV5KM= +github.com/stretchr/testify v1.7.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= +github.com/stretchr/testify v1.8.0 h1:pSgiaMZlXftHpm5L7V1+rVB+AZJydKsMxsQBIJw4PKk= +github.com/stretchr/testify v1.8.0/go.mod h1:yNjHg4UonilssWZ8iaSj1OCr/vHnekPRkoO+kdMU+MU= +github.com/tchap/go-patricia/v2 v2.3.1 h1:6rQp39lgIYZ+MHmdEq4xzuk1t7OdC35z/xm0BGhTkes= +github.com/tchap/go-patricia/v2 v2.3.1/go.mod h1:VZRHKAb53DLaG+nA9EaYYiaEx6YztwDlLElMsnSHD4k= github.com/ugorji/go/codec v0.0.0-20181204163529-d75b2dcb6bc8/go.mod h1:VFNgLljTbGfSG7qAOspJ7OScBnGdDN/yBr0sguwnwf0= github.com/xeipuuv/gojsonpointer v0.0.0-20190905194746-02993c407bfb h1:zGWFAtiMcyryUHoUjUJX0/lt1H2+i2Ka2n+D3DImSNo= github.com/xeipuuv/gojsonpointer v0.0.0-20190905194746-02993c407bfb/go.mod h1:N2zxlSyiKSe5eX1tZViRH5QA0qijqEDrYZiPEAiq3wU= github.com/xeipuuv/gojsonreference v0.0.0-20180127040603-bd5ef7bd5415 h1:EzJWgHovont7NscjpAxXsDA8S8BMYve8Y5+7cuRE7R0= github.com/xeipuuv/gojsonreference v0.0.0-20180127040603-bd5ef7bd5415/go.mod h1:GwrjFmJcFw6At/Gs6z4yjiIwzuJ1/+UwLxMQDVQXShQ= github.com/xordataexchange/crypt v0.0.3-0.20170626215501-b2862e3d0a77/go.mod h1:aYKd//L2LvnjZzWKhF00oedf4jCCReLcmhLdhm1A27Q= -github.com/yashtewari/glob-intersection v0.0.0-20180916065949-5c77d914dd0b h1:vVRagRXf67ESqAb72hG2C/ZwI8NtJF2u2V76EsuOHGY= -github.com/yashtewari/glob-intersection v0.0.0-20180916065949-5c77d914dd0b/go.mod h1:HptNXiXVDcJjXe9SqMd0v2FsL9f8dz4GnXgltU6q/co= +github.com/yashtewari/glob-intersection v0.1.0 h1:6gJvMYQlTDOL3dMsPF6J0+26vwX9MB8/1q3uAdhmTrg= +github.com/yashtewari/glob-intersection v0.1.0/go.mod h1:LK7pIC3piUjovexikBbJ26Yml7g8xa5bsjfx2v1fwok= github.com/yuin/goldmark v1.1.25/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74= github.com/yuin/goldmark v1.1.27/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74= github.com/yuin/goldmark v1.1.32/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74= github.com/yuin/goldmark v1.2.1/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74= -github.com/yuin/goldmark v1.3.5/go.mod h1:mwnBkeHKe2W/ZEtQ+71ViKU8L12m81fl3OWwC1Zlc8k= -go.etcd.io/etcd/api/v3 v3.5.1/go.mod h1:cbVKeC6lCfl7j/8jBhAK6aIYO9XOjdptoxU/nLQcPvs= -go.etcd.io/etcd/client/pkg/v3 v3.5.1/go.mod h1:IJHfcCEKxYu1Os13ZdwCwIUTUVGYTSAM3YSwc9/Ac1g= -go.etcd.io/etcd/client/v2 v2.305.1/go.mod h1:pMEacxZW7o8pg4CrFE7pquyCJJzZvkvdD2RibOCCCGs= go.opencensus.io v0.21.0/go.mod h1:mSImk1erAIZhrmZN+AvHh14ztQfjbGwt4TtuofqLduU= go.opencensus.io v0.22.0/go.mod h1:+kGneAE2xo2IficOXnaByMWTGM9T73dGwxeWcUqIpI8= go.opencensus.io v0.22.2/go.mod h1:yxeiOL68Rb0Xd1ddK5vPZ/oVn4vY4Ynel7k9FzqtOIw= @@ -452,34 +291,15 @@ go.opencensus.io v0.22.3/go.mod h1:yxeiOL68Rb0Xd1ddK5vPZ/oVn4vY4Ynel7k9FzqtOIw= go.opencensus.io v0.22.4/go.mod h1:yxeiOL68Rb0Xd1ddK5vPZ/oVn4vY4Ynel7k9FzqtOIw= go.opencensus.io v0.22.5/go.mod h1:5pWMHQbX5EPX2/62yrJeAkowc+lfs/XD7Uxpq3pI6kk= go.opencensus.io v0.23.0 h1:gqCw0LfLxScz8irSi8exQc7fyQ0fKQU/qnC/X8+V/1M= -go.opencensus.io v0.23.0/go.mod h1:XItmlyltB5F7CS4xOC1DcqMoFqwtC6OG2xF7mCv7P7E= -go.opentelemetry.io/contrib/instrumentation/net/http/otelhttp v0.28.0/go.mod h1:Ihno+mNBfZlT0Qot3XyRTdZ/9U/Cg2Pfgj75DTdIfq4= -go.opentelemetry.io/otel v1.3.0/go.mod h1:PWIKzi6JCp7sM0k9yZ43VX+T345uNbAkDKwHVjb2PTs= -go.opentelemetry.io/otel/exporters/otlp/internal/retry v1.3.0/go.mod h1:VpP4/RMn8bv8gNo9uK7/IMY4mtWLELsS+JIP0inH0h4= -go.opentelemetry.io/otel/exporters/otlp/otlptrace v1.3.0/go.mod h1:hO1KLR7jcKaDDKDkvI9dP/FIhpmna5lkqPUQdEjFAM8= -go.opentelemetry.io/otel/exporters/otlp/otlptrace/otlptracegrpc v1.3.0/go.mod h1:keUU7UfnwWTWpJ+FWnyqmogPa82nuU5VUANFq49hlMY= -go.opentelemetry.io/otel/internal/metric v0.26.0/go.mod h1:CbBP6AxKynRs3QCbhklyLUtpfzbqCLiafV9oY2Zj1Jk= -go.opentelemetry.io/otel/metric v0.26.0/go.mod h1:c6YL0fhRo4YVoNs6GoByzUgBp36hBL523rECoZA5UWg= -go.opentelemetry.io/otel/sdk v1.3.0/go.mod h1:rIo4suHNhQwBIPg9axF8V9CA72Wz2mKF1teNrup8yzs= -go.opentelemetry.io/otel/trace v1.3.0/go.mod h1:c/VDhno8888bvQYmbYLqe41/Ldmr/KKunbvWM4/fEjk= -go.opentelemetry.io/proto/otlp v0.7.0/go.mod h1:PqfVotwruBrMGOCsRd/89rSnXhoiJIqeYNgFYFoEGnI= -go.opentelemetry.io/proto/otlp v0.11.0/go.mod h1:QpEjXPrNQzrFDZgoTo49dgHR9RYRSrg3NAKnUGl9YpQ= -go.uber.org/atomic v1.7.0/go.mod h1:fEN4uk6kAWBTFdckzkM89CLk9XfWZrxpCo0nPH17wJc= -go.uber.org/automaxprocs v1.4.0/go.mod h1:/mTEdr7LvHhs0v7mjdxDreTz1OG5zdZGqgOnhWiR/+Q= -go.uber.org/goleak v1.1.12/go.mod h1:cwTWslyiVhfpKIDGSZEM2HlOvcqm+tG4zioyIeLoqMQ= -go.uber.org/multierr v1.6.0/go.mod h1:cdWPpRnG4AhwMwsgIHip0KRBQjJy5kYEpYjJxpXp9iU= -go.uber.org/zap v1.17.0/go.mod h1:MXVU+bhUf/A7Xi2HNOnopQOrmycQ5Ih87HtOu4q5SSo= golang.org/x/crypto v0.0.0-20180904163835-0709b304e793/go.mod h1:6SG95UA2DQfeDnfUPMdvaQW0Q7yPrPDi9nlGo2tz2b4= -golang.org/x/crypto v0.0.0-20181029021203-45a5f77698d3/go.mod h1:6SG95UA2DQfeDnfUPMdvaQW0Q7yPrPDi9nlGo2tz2b4= golang.org/x/crypto v0.0.0-20181203042331-505ab145d0a9/go.mod h1:6SG95UA2DQfeDnfUPMdvaQW0Q7yPrPDi9nlGo2tz2b4= golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w= golang.org/x/crypto v0.0.0-20190510104115-cbcb75029529/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= golang.org/x/crypto v0.0.0-20190605123033-f99c8df09eb5/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= -golang.org/x/crypto v0.0.0-20190820162420-60c769a6c586/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= -golang.org/x/crypto v0.0.0-20190923035154-9ee001bba392/go.mod h1:/lpIB1dKB+9EgE3H3cr1v9wB50oz8l4C4h62xy7jSTY= golang.org/x/crypto v0.0.0-20191011191535-87dc89f01550/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= golang.org/x/crypto v0.0.0-20200622213623-75b288015ac9/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto= -golang.org/x/crypto v0.0.0-20210817164053-32db794688a5/go.mod h1:GvvjBRRGRdwPK5ydBHafDWAxML/pGHZbMvKqRZ5+Abc= +golang.org/x/crypto v0.0.0-20220525230936-793ad666bf5e h1:T8NU3HyQ8ClP4SEE+KbFlg6n0NhuTsN4MyznaarGsZM= +golang.org/x/crypto v0.0.0-20220525230936-793ad666bf5e/go.mod h1:IxCIyHEi3zRg3s0A5j5BB6A9Jmi73HwBIUl50j+osU4= golang.org/x/exp v0.0.0-20190121172915-509febef88a4/go.mod h1:CJ0aWSM057203Lf6IL+f9T1iT9GByDxfZKAQTCR3kQA= golang.org/x/exp v0.0.0-20190306152737-a1d7652674e8/go.mod h1:CJ0aWSM057203Lf6IL+f9T1iT9GByDxfZKAQTCR3kQA= golang.org/x/exp v0.0.0-20190510132918-efd6b22b2522/go.mod h1:ZjyILWgesfNpC6sMxTJOJm9Kp84zZh5NQWvqDGG3Qr8= @@ -502,8 +322,6 @@ golang.org/x/lint v0.0.0-20190930215403-16217165b5de/go.mod h1:6SW0HCj/g11FgYtHl golang.org/x/lint v0.0.0-20191125180803-fdd1cda4f05f/go.mod h1:5qLYkcX4OjUUV8bRuDixDT3tpyyb+LUpUlRWLxfhWrs= golang.org/x/lint v0.0.0-20200130185559-910be7a94367/go.mod h1:3xt1FjdF8hUf6vQPIChWIBhFzV8gjjsPE/fR3IyQdNY= golang.org/x/lint v0.0.0-20200302205851-738671d3881b/go.mod h1:3xt1FjdF8hUf6vQPIChWIBhFzV8gjjsPE/fR3IyQdNY= -golang.org/x/lint v0.0.0-20201208152925-83fdc39ff7b5/go.mod h1:3xt1FjdF8hUf6vQPIChWIBhFzV8gjjsPE/fR3IyQdNY= -golang.org/x/lint v0.0.0-20210508222113-6edffad5e616/go.mod h1:3xt1FjdF8hUf6vQPIChWIBhFzV8gjjsPE/fR3IyQdNY= golang.org/x/mobile v0.0.0-20190312151609-d3739f865fa6/go.mod h1:z+o9i4GpDbdi3rU15maQ/Ox0txvL9dWGYEHz965HBQE= golang.org/x/mobile v0.0.0-20190719004257-d2bd2a29d028/go.mod h1:E/iHnbuqvinMTCcRqshq8CkpyQDoeVncDDYHnLhea+o= golang.org/x/mod v0.0.0-20190513183733-4bf6d317e70e/go.mod h1:mXi4GBBbnImb6dmsKGUJ2LatrhH/nqhxcFungHvyanc= @@ -512,13 +330,8 @@ golang.org/x/mod v0.1.1-0.20191105210325-c90efee705ee/go.mod h1:QqPTAvyqsEbceGzB golang.org/x/mod v0.1.1-0.20191107180719-034126e5016b/go.mod h1:QqPTAvyqsEbceGzBzNggFXnrqF1CaUcvgkdR5Ot7KZg= golang.org/x/mod v0.2.0/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA= golang.org/x/mod v0.3.0/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA= -golang.org/x/mod v0.4.0/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA= -golang.org/x/mod v0.4.1/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA= -golang.org/x/mod v0.4.2/go.mod h1:s0Qsj1ACt9ePp/hMypM3fl4fZqREWJwdYDEqhRiZZUA= -golang.org/x/mod v0.5.0/go.mod h1:5OXOZSfqPIIbmVBIIKWRFfZjPR0E5r58TLhUjH0a2Ro= golang.org/x/net v0.0.0-20180724234803-3673e40ba225/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= golang.org/x/net v0.0.0-20180826012351-8a410e7b638d/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= -golang.org/x/net v0.0.0-20181023162649-9b4f9f5ad519/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= golang.org/x/net v0.0.0-20181114220301-adae6a3d119a/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= golang.org/x/net v0.0.0-20190108225652-1e06a53dbb7e/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= golang.org/x/net v0.0.0-20190213061140-3a22650c66bd/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= @@ -531,7 +344,6 @@ golang.org/x/net v0.0.0-20190613194153-d28f0bde5980/go.mod h1:z5CRVTTTmAJ677TzLL golang.org/x/net v0.0.0-20190620200207-3b0461eec859/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= golang.org/x/net v0.0.0-20190628185345-da137c7871d7/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= golang.org/x/net v0.0.0-20190724013045-ca1201d0de80/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= -golang.org/x/net v0.0.0-20190923162816-aa69164e4478/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= golang.org/x/net v0.0.0-20191209160850-c0dbc17a3553/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= golang.org/x/net v0.0.0-20200114155413-6afb5195e5aa/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= golang.org/x/net v0.0.0-20200202094626-16171245cfb2/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= @@ -547,38 +359,17 @@ golang.org/x/net v0.0.0-20200625001655-4c5254603344/go.mod h1:/O7V0waA8r7cgGh81R golang.org/x/net v0.0.0-20200707034311-ab3426394381/go.mod h1:/O7V0waA8r7cgGh81Ro3o1hOxt32SMVPicZroKQ2sZA= golang.org/x/net v0.0.0-20200822124328-c89045814202/go.mod h1:/O7V0waA8r7cgGh81Ro3o1hOxt32SMVPicZroKQ2sZA= golang.org/x/net v0.0.0-20201021035429-f5854403a974/go.mod h1:sp8m0HH+o8qH0wwXwYZr8TS3Oi6o0r6Gce1SSxlDquU= -golang.org/x/net v0.0.0-20201031054903-ff519b6c9102/go.mod h1:sp8m0HH+o8qH0wwXwYZr8TS3Oi6o0r6Gce1SSxlDquU= -golang.org/x/net v0.0.0-20201110031124-69a78807bb2b/go.mod h1:sp8m0HH+o8qH0wwXwYZr8TS3Oi6o0r6Gce1SSxlDquU= -golang.org/x/net v0.0.0-20201209123823-ac852fbbde11/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg= -golang.org/x/net v0.0.0-20210119194325-5f4716e94777/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg= -golang.org/x/net v0.0.0-20210226172049-e18ecbb05110/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg= -golang.org/x/net v0.0.0-20210316092652-d523dce5a7f4/go.mod h1:RBQZq4jEuRlivfhVLdyRGr576XBO4/greRjx4P4O3yc= -golang.org/x/net v0.0.0-20210405180319-a5a99cb37ef4/go.mod h1:p54w0d4576C0XHj96bSt6lcn1PtDYWL6XObtHCRCNQM= -golang.org/x/net v0.0.0-20210410081132-afb366fc7cd1/go.mod h1:9tjilg8BloeKEkVJvy7fQ90B1CfIiPueXVOjqfkSzI8= -golang.org/x/net v0.0.0-20210503060351-7fd8e65b6420/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y= golang.org/x/net v0.0.0-20210525063256-abc453219eb5/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y= -golang.org/x/net v0.0.0-20210813160813-60bc85c4be6d/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y= -golang.org/x/net v0.0.0-20211111083644-e5c967477495/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y= golang.org/x/net v0.0.0-20220127200216-cd36cc0744dd/go.mod h1:CfG3xpIq0wQ8r1q4Su4UZFWDARRcnwPjda9FqA0JpMk= -golang.org/x/net v0.0.0-20220225172249-27dd8689420f h1:oA4XRj0qtSt8Yo1Zms0CUlsT3KG69V2UGQWPBxujDmc= golang.org/x/net v0.0.0-20220225172249-27dd8689420f/go.mod h1:CfG3xpIq0wQ8r1q4Su4UZFWDARRcnwPjda9FqA0JpMk= +golang.org/x/net v0.1.0 h1:hZ/3BUoy5aId7sCpA/Tc5lt8DkFgdVS2onTpJsZ/fl0= +golang.org/x/net v0.1.0/go.mod h1:Cx3nUiGt4eDBEyega/BKRp+/AlGL8hYe7U9odMt2Cco= golang.org/x/oauth2 v0.0.0-20180821212333-d2e6202438be/go.mod h1:N/0e6XlmueqKjAGxoOufVs8QHGRruUQn6yWY3a++T0U= golang.org/x/oauth2 v0.0.0-20190226205417-e64efc72b421/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw= golang.org/x/oauth2 v0.0.0-20190604053449-0f29369cfe45/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw= golang.org/x/oauth2 v0.0.0-20191202225959-858c2ad4c8b6/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw= golang.org/x/oauth2 v0.0.0-20200107190931-bf48bf16ab8d/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw= -golang.org/x/oauth2 v0.0.0-20200902213428-5d25da1a8d43/go.mod h1:KelEdhl1UZF7XfJ4dDtk6s++YSgaE7mD/BuKKDLBl4A= -golang.org/x/oauth2 v0.0.0-20201109201403-9fd604954f58/go.mod h1:KelEdhl1UZF7XfJ4dDtk6s++YSgaE7mD/BuKKDLBl4A= -golang.org/x/oauth2 v0.0.0-20201208152858-08078c50e5b5/go.mod h1:KelEdhl1UZF7XfJ4dDtk6s++YSgaE7mD/BuKKDLBl4A= -golang.org/x/oauth2 v0.0.0-20210218202405-ba52d332ba99/go.mod h1:KelEdhl1UZF7XfJ4dDtk6s++YSgaE7mD/BuKKDLBl4A= -golang.org/x/oauth2 v0.0.0-20210220000619-9bb904979d93/go.mod h1:KelEdhl1UZF7XfJ4dDtk6s++YSgaE7mD/BuKKDLBl4A= -golang.org/x/oauth2 v0.0.0-20210313182246-cd4f82c27b84/go.mod h1:KelEdhl1UZF7XfJ4dDtk6s++YSgaE7mD/BuKKDLBl4A= golang.org/x/oauth2 v0.0.0-20210514164344-f6687ab2804c/go.mod h1:KelEdhl1UZF7XfJ4dDtk6s++YSgaE7mD/BuKKDLBl4A= -golang.org/x/oauth2 v0.0.0-20210628180205-a41e5a781914/go.mod h1:KelEdhl1UZF7XfJ4dDtk6s++YSgaE7mD/BuKKDLBl4A= -golang.org/x/oauth2 v0.0.0-20210805134026-6f1e6394065a/go.mod h1:KelEdhl1UZF7XfJ4dDtk6s++YSgaE7mD/BuKKDLBl4A= -golang.org/x/oauth2 v0.0.0-20210819190943-2bc19b11175f/go.mod h1:KelEdhl1UZF7XfJ4dDtk6s++YSgaE7mD/BuKKDLBl4A= -golang.org/x/oauth2 v0.0.0-20211005180243-6b3c2da341f1/go.mod h1:KelEdhl1UZF7XfJ4dDtk6s++YSgaE7mD/BuKKDLBl4A= -golang.org/x/oauth2 v0.0.0-20211104180415-d3ed0bb246c8/go.mod h1:KelEdhl1UZF7XfJ4dDtk6s++YSgaE7mD/BuKKDLBl4A= golang.org/x/oauth2 v0.0.0-20220223155221-ee480838109b/go.mod h1:DAh4E804XQdzx2j+YRIaUnCqCV2RuMz24cGBJ5QYIrc= golang.org/x/sync v0.0.0-20180314180146-1d60e4601c6f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= golang.org/x/sync v0.0.0-20181108010431-42b317875d0f/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= @@ -590,15 +381,11 @@ golang.org/x/sync v0.0.0-20200317015054-43a5402ce75a/go.mod h1:RxMgew5VJxzue5/jJ golang.org/x/sync v0.0.0-20200625203802-6e8e738ad208/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= golang.org/x/sync v0.0.0-20201020160332-67f06af15bc9/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= golang.org/x/sync v0.0.0-20201207232520-09787c993a3a/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= -golang.org/x/sync v0.0.0-20210220032951-036812b2e83c/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= -golang.org/x/sys v0.0.0-20180823144017-11551d06cbcc/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= golang.org/x/sys v0.0.0-20180830151530-49385e6e1522/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= golang.org/x/sys v0.0.0-20180905080454-ebe1bf3edb33/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= -golang.org/x/sys v0.0.0-20181026203630-95b1ffbd15a5/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= golang.org/x/sys v0.0.0-20181116152217-5ac8a444bdc5/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= golang.org/x/sys v0.0.0-20181205085412-a5c9d58dba9a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= -golang.org/x/sys v0.0.0-20190222072716-a9d3bda3a223/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= golang.org/x/sys v0.0.0-20190312061237-fead79001313/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20190412213103-97732733099d/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20190422165155-953cdadca894/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= @@ -607,18 +394,12 @@ golang.org/x/sys v0.0.0-20190507160741-ecd444e8653b/go.mod h1:h1NjWce9XRLGQEsW7w golang.org/x/sys v0.0.0-20190606165138-5da285871e9c/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20190624142023-c5567b49c5d0/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20190726091711-fc99dfbffb4e/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20190922100055-0a153f010e69/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20190924154521-2837fb4f24fe/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20191001151750-bb3f8db39f24/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20191008105621-543471e840be/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20191026070338-33540a1f6037/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20191204072324-ce4227a45e2e/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20191228213918-04cbcbbfeed8/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20200106162015-b016eb3dc98e/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20200113162924-86b910548bc1/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20200116001909-b77594299b42/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20200122134326-e047566fdf82/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20200124204421-9fbb57f87de9/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20200202164722-d101bd2416d5/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20200212091648-12a6c2dcc1e4/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20200223170610-d5e6a3e2c0ae/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= @@ -632,41 +413,17 @@ golang.org/x/sys v0.0.0-20200523222454-059865788121/go.mod h1:h1NjWce9XRLGQEsW7w golang.org/x/sys v0.0.0-20200615200032-f1bc736245b1/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20200625212154-ddb9806d33ae/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20200803210538-64077c9b5642/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20200905004654-be1d3432aa8f/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20200930185726-fdedc70b468f/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20201119102817-f84b799fce68/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20201201145000-ef89a241ccb3/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20210104204734-6f8348627aad/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20210119212857-b64e53b001e4/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20210124154548-22da62e12c0c/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20210220050731-9a76102bfb43/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20210303074136-134d130e1a04/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20210305230114-8fe3ee5dd75b/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20210315160823-c6e025ad8005/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20210320140829-1e4c9ba3b0c4/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20210330210617-4fbd30eecc44/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20210403161142-5e06dd20ab57/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20210423082822-04245dca01da/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20210423185535-09eb48e85fd7/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= -golang.org/x/sys v0.0.0-20210510120138-977fb7262007/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= -golang.org/x/sys v0.0.0-20210514084401-e8d321eab015/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20210603081109-ebe580a85c40/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= -golang.org/x/sys v0.0.0-20210603125802-9665404d3644/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20210615035016-665e8c7367d1/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= -golang.org/x/sys v0.0.0-20210616094352-59db8d763f22/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= -golang.org/x/sys v0.0.0-20210630005230-0f9fa26af87c/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= -golang.org/x/sys v0.0.0-20210806184541-e5e7981a1069/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= -golang.org/x/sys v0.0.0-20210816183151-1e6c022a8912/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= -golang.org/x/sys v0.0.0-20210823070655-63515b42dcdf/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= -golang.org/x/sys v0.0.0-20210908233432-aa78b53d3365/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= -golang.org/x/sys v0.0.0-20210927094055-39ccf1dd6fa6/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= -golang.org/x/sys v0.0.0-20211007075335-d3039528d8ac/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= -golang.org/x/sys v0.0.0-20211124211545-fe61309f8881/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= -golang.org/x/sys v0.0.0-20211205182925-97ca703d548d/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20211216021012-1d35b9e2eb4e/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20220114195835-da31bd327af9/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= -golang.org/x/sys v0.0.0-20221013171732-95e765b1cc43 h1:OK7RB6t2WQX54srQQYSXMW8dF5C6/8+oA/s5QBmmto4= -golang.org/x/sys v0.0.0-20221013171732-95e765b1cc43/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.0.0-20221010170243-090e33056c14/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.1.0 h1:kunALQeHf1/185U1i0GOB/fy1IPRDDpuoOOqRReG57U= +golang.org/x/sys v0.1.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo= golang.org/x/term v0.0.0-20210927222741-03fcf44c2211/go.mod h1:jbD1KX2456YbFQfuXm/mYQcufACuNUgVhRMnK/tPxf8= golang.org/x/text v0.0.0-20170915032832-14c0d48ead0c/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= @@ -674,14 +431,11 @@ golang.org/x/text v0.3.0/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= golang.org/x/text v0.3.1-0.20180807135948-17ff2d5776d2/go.mod h1:NqM8EUOU14njkJ3fqMW+pc6Ldnwhi/IjpwHt7yyuwOQ= golang.org/x/text v0.3.2/go.mod h1:bEr9sfX3Q8Zfm5fL9x+3itogRgK3+ptLWKqgva+5dAk= golang.org/x/text v0.3.3/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= -golang.org/x/text v0.3.4/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= -golang.org/x/text v0.3.5/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= golang.org/x/text v0.3.6/go.mod h1:5Zoc/QRtKVWzQhOtBMvqHzDpF6irO9z98xDceosuGiQ= golang.org/x/text v0.3.7/go.mod h1:u+2+/6zg+i71rQMx5EYifcz6MCKuco9NR6JIITiCfzQ= golang.org/x/time v0.0.0-20181108054448-85acf8d2951c/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ= golang.org/x/time v0.0.0-20190308202827-9d24e82272b4/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ= golang.org/x/time v0.0.0-20191024005414-555d28b269f0/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ= -golang.org/x/time v0.0.0-20210220033141-f8bda1e9f3ba/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ= golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= golang.org/x/tools v0.0.0-20190114222345-bf090417da8b/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= golang.org/x/tools v0.0.0-20190226205152-f727befe758c/go.mod h1:9Yl7xja0Znq3iFh3HoIrodX9oNMXvdceNzlUR8zjMvY= @@ -695,7 +449,6 @@ golang.org/x/tools v0.0.0-20190606124116-d0a3d012864b/go.mod h1:/rFqwRUd4F7ZHNgw golang.org/x/tools v0.0.0-20190621195816-6e04913cbbac/go.mod h1:/rFqwRUd4F7ZHNgwSSTFct+R/Kf4OFW1sUzUTQQTgfc= golang.org/x/tools v0.0.0-20190628153133-6cdbf07be9d0/go.mod h1:/rFqwRUd4F7ZHNgwSSTFct+R/Kf4OFW1sUzUTQQTgfc= golang.org/x/tools v0.0.0-20190816200558-6889da9d5479/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= -golang.org/x/tools v0.0.0-20190907020128-2ca718005c18/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= golang.org/x/tools v0.0.0-20190911174233-4f2ddba30aff/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= golang.org/x/tools v0.0.0-20191012152004-8de300cfc20a/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= golang.org/x/tools v0.0.0-20191113191852-77e3bb0ad9e7/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= @@ -724,18 +477,7 @@ golang.org/x/tools v0.0.0-20200619180055-7c47624df98f/go.mod h1:EkVYQZoAsY45+roY golang.org/x/tools v0.0.0-20200729194436-6467de6f59a7/go.mod h1:njjCfa9FT2d7l9Bc6FUM5FLjQPp3cFF28FI3qnDFljA= golang.org/x/tools v0.0.0-20200804011535-6c149bb5ef0d/go.mod h1:njjCfa9FT2d7l9Bc6FUM5FLjQPp3cFF28FI3qnDFljA= golang.org/x/tools v0.0.0-20200825202427-b303f430e36d/go.mod h1:njjCfa9FT2d7l9Bc6FUM5FLjQPp3cFF28FI3qnDFljA= -golang.org/x/tools v0.0.0-20200904185747-39188db58858/go.mod h1:Cj7w3i3Rnn0Xh82ur9kSqwfTHTeVxaDqrfMjpcNT6bE= -golang.org/x/tools v0.0.0-20201110124207-079ba7bd75cd/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA= -golang.org/x/tools v0.0.0-20201201161351-ac6f37ff4c2a/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA= -golang.org/x/tools v0.0.0-20201208233053-a543418bbed2/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA= -golang.org/x/tools v0.0.0-20210105154028-b0ab187a4818/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA= golang.org/x/tools v0.0.0-20210106214847-113979e3529a/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA= -golang.org/x/tools v0.1.0/go.mod h1:xkSsbof2nBLbhDlRMhhhyNLN/zl3eTqcnHD5viDpcZ0= -golang.org/x/tools v0.1.1/go.mod h1:o0xws9oXOQQZyjljx8fwUC0k7L1pTE6eaCbjGeHmOkk= -golang.org/x/tools v0.1.2/go.mod h1:o0xws9oXOQQZyjljx8fwUC0k7L1pTE6eaCbjGeHmOkk= -golang.org/x/tools v0.1.3/go.mod h1:o0xws9oXOQQZyjljx8fwUC0k7L1pTE6eaCbjGeHmOkk= -golang.org/x/tools v0.1.4/go.mod h1:o0xws9oXOQQZyjljx8fwUC0k7L1pTE6eaCbjGeHmOkk= -golang.org/x/tools v0.1.5/go.mod h1:o0xws9oXOQQZyjljx8fwUC0k7L1pTE6eaCbjGeHmOkk= golang.org/x/xerrors v0.0.0-20190717185122-a985d3407aa7/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= golang.org/x/xerrors v0.0.0-20191011141410-1b5146add898/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= golang.org/x/xerrors v0.0.0-20191204190536-9bdfabe68543/go.mod h1:I/5z698sn9Ka8TeJc9MKroUUfqBBauWjQqLJ2OPfmY0= @@ -756,29 +498,12 @@ google.golang.org/api v0.24.0/go.mod h1:lIXQywCXRcnZPGlsd8NbLnOjtAoL6em04bJ9+z0M google.golang.org/api v0.28.0/go.mod h1:lIXQywCXRcnZPGlsd8NbLnOjtAoL6em04bJ9+z0MncE= google.golang.org/api v0.29.0/go.mod h1:Lcubydp8VUV7KeIHD9z2Bys/sm/vGKnG1UHuDBSrHWM= google.golang.org/api v0.30.0/go.mod h1:QGmEvQ87FHZNiUVJkT14jQNYJ4ZJjdRF23ZXz5138Fc= -google.golang.org/api v0.35.0/go.mod h1:/XrVsuzM0rZmrsbjJutiuftIzeuTQcEeaYcSk/mQ1dg= -google.golang.org/api v0.36.0/go.mod h1:+z5ficQTmoYpPn8LCUNVpK5I7hwkpjbcgqA7I34qYtE= -google.golang.org/api v0.40.0/go.mod h1:fYKFpnQN0DsDSKRVRcQSDQNtqWPfM9i+zNPxepjRCQ8= -google.golang.org/api v0.41.0/go.mod h1:RkxM5lITDfTzmyKFPt+wGrCJbVfniCr2ool8kTBzRTU= -google.golang.org/api v0.43.0/go.mod h1:nQsDGjRXMo4lvh5hP0TKqF244gqhGcr/YSIykhUk/94= -google.golang.org/api v0.47.0/go.mod h1:Wbvgpq1HddcWVtzsVLyfLp8lDg6AA241LmgIL59tHXo= -google.golang.org/api v0.48.0/go.mod h1:71Pr1vy+TAZRPkPs/xlCf5SsU8WjuAWv1Pfjbtukyy4= -google.golang.org/api v0.50.0/go.mod h1:4bNT5pAuq5ji4SRZm+5QIkjny9JAyVD/3gaSihNefaw= -google.golang.org/api v0.51.0/go.mod h1:t4HdrdoNgyN5cbEfm7Lum0lcLDLiise1F8qDKX00sOU= -google.golang.org/api v0.54.0/go.mod h1:7C4bFFOvVDGXjfDTAsgGwDgAxRDeQ4X8NvUedIt6z3k= -google.golang.org/api v0.55.0/go.mod h1:38yMfeP1kfjsl8isn0tliTjIb1rJXcQi4UXlbqivdVE= -google.golang.org/api v0.56.0/go.mod h1:38yMfeP1kfjsl8isn0tliTjIb1rJXcQi4UXlbqivdVE= -google.golang.org/api v0.57.0/go.mod h1:dVPlbZyBo2/OjBpmvNdpn2GRm6rPy75jyU7bmhdrMgI= -google.golang.org/api v0.59.0/go.mod h1:sT2boj7M9YJxZzgeZqXogmhfmRWDtPzT31xkieUbuZU= -google.golang.org/api v0.61.0/go.mod h1:xQRti5UdCmoCEqFxcz93fTl338AVqDgyaDRuOZ3hg9I= -google.golang.org/api v0.62.0/go.mod h1:dKmwPCydfsad4qCH08MSdgWjfHOyfpd4VtDGgRFdavw= google.golang.org/appengine v1.1.0/go.mod h1:EbEs0AVv82hx2wNQdGPgUI5lhzA/G0D9YwlJXL52JkM= google.golang.org/appengine v1.4.0/go.mod h1:xpcJRLb0r/rnEns0DIKYYv+WjYCduHsrkT7/EB5XEv4= google.golang.org/appengine v1.5.0/go.mod h1:xpcJRLb0r/rnEns0DIKYYv+WjYCduHsrkT7/EB5XEv4= google.golang.org/appengine v1.6.1/go.mod h1:i06prIuMbXzDqacNJfV5OdTW448YApPu5ww/cMBSeb0= google.golang.org/appengine v1.6.5/go.mod h1:8WjMMxjGQR8xUklV/ARdw2HLXBOI7O7uCIDZVag1xfc= google.golang.org/appengine v1.6.6/go.mod h1:8WjMMxjGQR8xUklV/ARdw2HLXBOI7O7uCIDZVag1xfc= -google.golang.org/appengine v1.6.7/go.mod h1:8WjMMxjGQR8xUklV/ARdw2HLXBOI7O7uCIDZVag1xfc= google.golang.org/genproto v0.0.0-20180817151627-c66870c02cf8/go.mod h1:JiN7NxoALGmiZfu7CAH4rXhgtRTLTxftemlI0sWmxmc= google.golang.org/genproto v0.0.0-20190307195333-5fe7a883aa19/go.mod h1:VzzqZJRnGkLBvHegQrXjBqPurQTc5/KpmUdxsrq26oE= google.golang.org/genproto v0.0.0-20190418145605-e7d98fc518a7/go.mod h1:VzzqZJRnGkLBvHegQrXjBqPurQTc5/KpmUdxsrq26oE= @@ -802,46 +527,12 @@ google.golang.org/genproto v0.0.0-20200312145019-da6875a35672/go.mod h1:55QSHmfG google.golang.org/genproto v0.0.0-20200331122359-1ee6d9798940/go.mod h1:55QSHmfGQM9UVYDPBsyGGes0y52j32PQ3BqQfXhyH3c= google.golang.org/genproto v0.0.0-20200430143042-b979b6f78d84/go.mod h1:55QSHmfGQM9UVYDPBsyGGes0y52j32PQ3BqQfXhyH3c= google.golang.org/genproto v0.0.0-20200511104702-f5ebc3bea380/go.mod h1:55QSHmfGQM9UVYDPBsyGGes0y52j32PQ3BqQfXhyH3c= -google.golang.org/genproto v0.0.0-20200513103714-09dca8ec2884/go.mod h1:55QSHmfGQM9UVYDPBsyGGes0y52j32PQ3BqQfXhyH3c= google.golang.org/genproto v0.0.0-20200515170657-fc4c6c6a6587/go.mod h1:YsZOwe1myG/8QRHRsmBRE1LrgQY60beZKjly0O1fX9U= google.golang.org/genproto v0.0.0-20200526211855-cb27e3aa2013/go.mod h1:NbSheEEYHJ7i3ixzK3sjbqSGDJWnxyFXZblF3eUsNvo= google.golang.org/genproto v0.0.0-20200618031413-b414f8b61790/go.mod h1:jDfRM7FcilCzHH/e9qn6dsT145K34l5v+OpcnNgKAAA= google.golang.org/genproto v0.0.0-20200729003335-053ba62fc06f/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no= google.golang.org/genproto v0.0.0-20200804131852-c06518451d9c/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no= google.golang.org/genproto v0.0.0-20200825200019-8632dd797987/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no= -google.golang.org/genproto v0.0.0-20200904004341-0bd0a958aa1d/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no= -google.golang.org/genproto v0.0.0-20201109203340-2640f1f9cdfb/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no= -google.golang.org/genproto v0.0.0-20201201144952-b05cb90ed32e/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no= -google.golang.org/genproto v0.0.0-20201210142538-e3217bee35cc/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no= -google.golang.org/genproto v0.0.0-20201214200347-8c77b98c765d/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no= -google.golang.org/genproto v0.0.0-20210222152913-aa3ee6e6a81c/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no= -google.golang.org/genproto v0.0.0-20210303154014-9728d6b83eeb/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no= -google.golang.org/genproto v0.0.0-20210310155132-4ce2db91004e/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no= -google.golang.org/genproto v0.0.0-20210319143718-93e7006c17a6/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no= -google.golang.org/genproto v0.0.0-20210402141018-6c239bbf2bb1/go.mod h1:9lPAdzaEmUacj36I+k7YKbEc5CXzPIeORRgDAUOu28A= -google.golang.org/genproto v0.0.0-20210513213006-bf773b8c8384/go.mod h1:P3QM42oQyzQSnHPnZ/vqoCdDmzH28fzWByN9asMeM8A= -google.golang.org/genproto v0.0.0-20210602131652-f16073e35f0c/go.mod h1:UODoCrxHCcBojKKwX1terBiRUaqAsFqJiF615XL43r0= -google.golang.org/genproto v0.0.0-20210604141403-392c879c8b08/go.mod h1:UODoCrxHCcBojKKwX1terBiRUaqAsFqJiF615XL43r0= -google.golang.org/genproto v0.0.0-20210608205507-b6d2f5bf0d7d/go.mod h1:UODoCrxHCcBojKKwX1terBiRUaqAsFqJiF615XL43r0= -google.golang.org/genproto v0.0.0-20210624195500-8bfb893ecb84/go.mod h1:SzzZ/N+nwJDaO1kznhnlzqS8ocJICar6hYhVyhi++24= -google.golang.org/genproto v0.0.0-20210713002101-d411969a0d9a/go.mod h1:AxrInvYm1dci+enl5hChSFPOmmUF1+uAa/UsgNRWd7k= -google.golang.org/genproto v0.0.0-20210716133855-ce7ef5c701ea/go.mod h1:AxrInvYm1dci+enl5hChSFPOmmUF1+uAa/UsgNRWd7k= -google.golang.org/genproto v0.0.0-20210728212813-7823e685a01f/go.mod h1:ob2IJxKrgPT52GcgX759i1sleT07tiKowYBGbczaW48= -google.golang.org/genproto v0.0.0-20210805201207-89edb61ffb67/go.mod h1:ob2IJxKrgPT52GcgX759i1sleT07tiKowYBGbczaW48= -google.golang.org/genproto v0.0.0-20210813162853-db860fec028c/go.mod h1:cFeNkxwySK631ADgubI+/XFU/xp8FD5KIVV4rj8UC5w= -google.golang.org/genproto v0.0.0-20210821163610-241b8fcbd6c8/go.mod h1:eFjDcFEctNawg4eG61bRv87N7iHBWyVhJu7u1kqDUXY= -google.golang.org/genproto v0.0.0-20210828152312-66f60bf46e71/go.mod h1:eFjDcFEctNawg4eG61bRv87N7iHBWyVhJu7u1kqDUXY= -google.golang.org/genproto v0.0.0-20210831024726-fe130286e0e2/go.mod h1:eFjDcFEctNawg4eG61bRv87N7iHBWyVhJu7u1kqDUXY= -google.golang.org/genproto v0.0.0-20210903162649-d08c68adba83/go.mod h1:eFjDcFEctNawg4eG61bRv87N7iHBWyVhJu7u1kqDUXY= -google.golang.org/genproto v0.0.0-20210909211513-a8c4777a87af/go.mod h1:eFjDcFEctNawg4eG61bRv87N7iHBWyVhJu7u1kqDUXY= -google.golang.org/genproto v0.0.0-20210924002016-3dee208752a0/go.mod h1:5CzLGKJ67TSI2B9POpiiyGha0AjJvZIUgRMt1dSmuhc= -google.golang.org/genproto v0.0.0-20211008145708-270636b82663/go.mod h1:5CzLGKJ67TSI2B9POpiiyGha0AjJvZIUgRMt1dSmuhc= -google.golang.org/genproto v0.0.0-20211028162531-8db9c33dc351/go.mod h1:5CzLGKJ67TSI2B9POpiiyGha0AjJvZIUgRMt1dSmuhc= -google.golang.org/genproto v0.0.0-20211118181313-81c1377c94b1/go.mod h1:5CzLGKJ67TSI2B9POpiiyGha0AjJvZIUgRMt1dSmuhc= -google.golang.org/genproto v0.0.0-20211129164237-f09f9a12af12/go.mod h1:5CzLGKJ67TSI2B9POpiiyGha0AjJvZIUgRMt1dSmuhc= -google.golang.org/genproto v0.0.0-20211203200212-54befc351ae9/go.mod h1:5CzLGKJ67TSI2B9POpiiyGha0AjJvZIUgRMt1dSmuhc= -google.golang.org/genproto v0.0.0-20211206160659-862468c7d6e0/go.mod h1:5CzLGKJ67TSI2B9POpiiyGha0AjJvZIUgRMt1dSmuhc= -google.golang.org/genproto v0.0.0-20211208223120-3a66f561d7aa/go.mod h1:5CzLGKJ67TSI2B9POpiiyGha0AjJvZIUgRMt1dSmuhc= google.golang.org/grpc v1.19.0/go.mod h1:mqu4LbDTu4XGKhr4mRzUsmM4RtVoemTSY81AxZiDr8c= google.golang.org/grpc v1.20.1/go.mod h1:10oTOabMzJvdu6/UiuZezV6QK5dSlG84ov/aaiqXj38= google.golang.org/grpc v1.21.1/go.mod h1:oYelfM1adQP15Ek0mdvEgi9Df8B9CZIaU1084ijfRaM= @@ -854,23 +545,6 @@ google.golang.org/grpc v1.28.0/go.mod h1:rpkK4SK4GF4Ach/+MFLZUBavHOvF2JJB5uozKKa google.golang.org/grpc v1.29.1/go.mod h1:itym6AZVZYACWQqET3MqgPpjcuV5QH3BxFS3IjizoKk= google.golang.org/grpc v1.30.0/go.mod h1:N36X2cJ7JwdamYAgDz+s+rVMFjt3numwzf/HckM8pak= google.golang.org/grpc v1.31.0/go.mod h1:N36X2cJ7JwdamYAgDz+s+rVMFjt3numwzf/HckM8pak= -google.golang.org/grpc v1.31.1/go.mod h1:N36X2cJ7JwdamYAgDz+s+rVMFjt3numwzf/HckM8pak= -google.golang.org/grpc v1.33.1/go.mod h1:fr5YgcSWrqhRRxogOsw7RzIpsmvOZ6IcH4kBYTpR3n0= -google.golang.org/grpc v1.33.2/go.mod h1:JMHMWHQWaTccqQQlmk3MJZS+GWXOdAesneDmEnv2fbc= -google.golang.org/grpc v1.34.0/go.mod h1:WotjhfgOW/POjDeRt8vscBtXq+2VjORFy659qA51WJ8= -google.golang.org/grpc v1.35.0/go.mod h1:qjiiYl8FncCW8feJPdyg3v6XW24KsRHe+dy9BAGRRjU= -google.golang.org/grpc v1.36.0/go.mod h1:qjiiYl8FncCW8feJPdyg3v6XW24KsRHe+dy9BAGRRjU= -google.golang.org/grpc v1.36.1/go.mod h1:qjiiYl8FncCW8feJPdyg3v6XW24KsRHe+dy9BAGRRjU= -google.golang.org/grpc v1.37.0/go.mod h1:NREThFqKR1f3iQ6oBuvc5LadQuXVGo9rkm5ZGrQdJfM= -google.golang.org/grpc v1.37.1/go.mod h1:NREThFqKR1f3iQ6oBuvc5LadQuXVGo9rkm5ZGrQdJfM= -google.golang.org/grpc v1.38.0/go.mod h1:NREThFqKR1f3iQ6oBuvc5LadQuXVGo9rkm5ZGrQdJfM= -google.golang.org/grpc v1.39.0/go.mod h1:PImNr+rS9TWYb2O4/emRugxiyHZ5JyHW5F+RPnDzfrE= -google.golang.org/grpc v1.39.1/go.mod h1:PImNr+rS9TWYb2O4/emRugxiyHZ5JyHW5F+RPnDzfrE= -google.golang.org/grpc v1.40.0/go.mod h1:ogyxbiOoUXAkP+4+xa6PZSE9DZgIHtSpzjDTB9KAK34= -google.golang.org/grpc v1.40.1/go.mod h1:ogyxbiOoUXAkP+4+xa6PZSE9DZgIHtSpzjDTB9KAK34= -google.golang.org/grpc v1.42.0/go.mod h1:k+4IHHFw41K8+bbowsex27ge2rCb65oeWqe4jJ590SU= -google.golang.org/grpc v1.43.0/go.mod h1:k+4IHHFw41K8+bbowsex27ge2rCb65oeWqe4jJ590SU= -google.golang.org/grpc/cmd/protoc-gen-go-grpc v1.1.0/go.mod h1:6Kw0yEErY5E/yWrBtf03jp27GLLJujG4z/JK95pnjjw= google.golang.org/protobuf v0.0.0-20200109180630-ec00e32a8dfd/go.mod h1:DFci5gLYBciE7Vtevhsrf46CRTquxDuWsQurQQe4oz8= google.golang.org/protobuf v0.0.0-20200221191635-4d8936d0db64/go.mod h1:kwYJMbMJ01Woi6D6+Kah6886xMZcty6N08ah7+eCXa0= google.golang.org/protobuf v0.0.0-20200228230310-ab0ca4ff8a60/go.mod h1:cfTl7dwQJ+fmap5saPgwCLgHXTUD7jkjRqWcaiX5VyM= @@ -883,29 +557,24 @@ google.golang.org/protobuf v1.24.0/go.mod h1:r/3tXBNzIEhYS9I1OUVjXDlt8tc493IdKGj google.golang.org/protobuf v1.25.0/go.mod h1:9JNX74DMeImyA3h4bdi1ymwjUzf21/xIlbajtzgsN7c= google.golang.org/protobuf v1.26.0-rc.1/go.mod h1:jlhhOSvTdKEhbULTjvd4ARK9grFBp09yW+WbY/TyQbw= google.golang.org/protobuf v1.26.0/go.mod h1:9q0QmTI4eRPtz6boOQmLYwt+qCgq0jsYwAQnmE0givc= -google.golang.org/protobuf v1.27.1/go.mod h1:9q0QmTI4eRPtz6boOQmLYwt+qCgq0jsYwAQnmE0givc= google.golang.org/protobuf v1.28.1 h1:d0NfwRgPtno5B1Wa6L2DAG+KivqkdutMf1UhdNx175w= google.golang.org/protobuf v1.28.1/go.mod h1:HV8QOd/L58Z+nl8r43ehVNZIU/HEI6OcFqwMG9pJV4I= gopkg.in/alecthomas/kingpin.v2 v2.2.6/go.mod h1:FMv+mEhP44yOT+4EoQTLFTRgOQ1FBLkstjWtayDeSgw= gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= gopkg.in/check.v1 v1.0.0-20180628173108-788fd7840127/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= gopkg.in/check.v1 v1.0.0-20190902080502-41f04d3bba15/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= -gopkg.in/check.v1 v1.0.0-20200227125254-8fa46927fb4f h1:BLraFXnmrev5lT+xlilqcH8XK9/i0At2xKjWk4p6zsU= -gopkg.in/check.v1 v1.0.0-20200227125254-8fa46927fb4f/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= +gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c h1:Hei/4ADfdWqJk1ZMxUNpqntNwaWcugrBjAiHlqqRiVk= gopkg.in/errgo.v2 v2.1.0/go.mod h1:hNsd1EY+bozCKY1Ytp96fpM3vjJbqLJn88ws8XvfDNI= -gopkg.in/ini.v1 v1.66.2/go.mod h1:pNLf8WUiyNEtQjuu5G5vTm06TEv9tsIgeAvK8hOrP4k= gopkg.in/yaml.v2 v2.2.1/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= gopkg.in/yaml.v2 v2.2.2/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= -gopkg.in/yaml.v2 v2.2.3/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= gopkg.in/yaml.v2 v2.2.4/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= gopkg.in/yaml.v2 v2.2.5/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= -gopkg.in/yaml.v2 v2.2.8/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= gopkg.in/yaml.v2 v2.3.0/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= gopkg.in/yaml.v2 v2.4.0 h1:D8xgwECY7CYvx+Y2n4sBz93Jn9JRvxdiyyo8CTfuKaY= gopkg.in/yaml.v2 v2.4.0/go.mod h1:RDklbk79AGWmwhnvt/jBztapEOGDOx6ZbXqjP6csGnQ= gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= -gopkg.in/yaml.v3 v3.0.0-20210107192922-496545a6307b h1:h8qDotaEPuJATrMmW04NCwg7v22aHH28wwpauUhK9Oo= -gopkg.in/yaml.v3 v3.0.0-20210107192922-496545a6307b/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= +gopkg.in/yaml.v3 v3.0.1 h1:fxVm/GzAzEWqLHuvctI91KS9hhNmmWOoWu0XTYJS7CA= +gopkg.in/yaml.v3 v3.0.1/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= honnef.co/go/tools v0.0.0-20190102054323-c2f93a96b099/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4= honnef.co/go/tools v0.0.0-20190106161140-3f1c8253044a/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4= honnef.co/go/tools v0.0.0-20190418001031-e561f6794a2a/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4= diff --git a/vendor/github.com/agnivade/levenshtein/.gitignore b/vendor/github.com/agnivade/levenshtein/.gitignore new file mode 100644 index 00000000..345780a4 --- /dev/null +++ b/vendor/github.com/agnivade/levenshtein/.gitignore @@ -0,0 +1,5 @@ +coverage.txt +fuzz/fuzz-fuzz.zip +fuzz/corpus/corpus/* +fuzz/corpus/suppressions/* +fuzz/corpus/crashes/* diff --git a/vendor/github.com/agnivade/levenshtein/.travis.yml b/vendor/github.com/agnivade/levenshtein/.travis.yml new file mode 100644 index 00000000..0873fa98 --- /dev/null +++ b/vendor/github.com/agnivade/levenshtein/.travis.yml @@ -0,0 +1,23 @@ +language: go + +# See https://travis-ci.community/t/goos-js-goarch-wasm-go-run-fails-panic-newosproc-not-implemented/1651 +#addons: +# chrome: stable + +before_install: +- export GO111MODULE=on + +#install: +#- go get github.com/agnivade/wasmbrowsertest +#- mv $GOPATH/bin/wasmbrowsertest $GOPATH/bin/go_js_wasm_exec +#- export PATH=$GOPATH/bin:$PATH + +go: +- 1.13.x +- 1.14.x +- 1.15.x +- tip + +script: +#- GOOS=js GOARCH=wasm go test -v +- go test -v diff --git a/vendor/github.com/agnivade/levenshtein/License.txt b/vendor/github.com/agnivade/levenshtein/License.txt new file mode 100644 index 00000000..54b51f49 --- /dev/null +++ b/vendor/github.com/agnivade/levenshtein/License.txt @@ -0,0 +1,21 @@ +The MIT License (MIT) + +Copyright (c) 2015 Agniva De Sarker + +Permission is hereby granted, free of charge, to any person obtaining a copy +of this software and associated documentation files (the "Software"), to deal +in the Software without restriction, including without limitation the rights +to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the Software is +furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in all +copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE +SOFTWARE. diff --git a/vendor/github.com/agnivade/levenshtein/Makefile b/vendor/github.com/agnivade/levenshtein/Makefile new file mode 100644 index 00000000..5f6890d6 --- /dev/null +++ b/vendor/github.com/agnivade/levenshtein/Makefile @@ -0,0 +1,15 @@ +all: test install + +install: + go install + +lint: + gofmt -l -s -w . && go vet . && golint -set_exit_status=1 . + +test: # The first 2 go gets are to support older Go versions + go get github.com/arbovm/levenshtein + go get github.com/dgryski/trifles/leven + GO111MODULE=on go test -race -v -coverprofile=coverage.txt -covermode=atomic + +bench: + go test -run=XXX -bench=. -benchmem -count=5 diff --git a/vendor/github.com/agnivade/levenshtein/README.md b/vendor/github.com/agnivade/levenshtein/README.md new file mode 100644 index 00000000..13c52a21 --- /dev/null +++ b/vendor/github.com/agnivade/levenshtein/README.md @@ -0,0 +1,80 @@ +levenshtein [![Build Status](https://travis-ci.org/agnivade/levenshtein.svg?branch=master)](https://travis-ci.org/agnivade/levenshtein) [![Go Report Card](https://goreportcard.com/badge/github.com/agnivade/levenshtein)](https://goreportcard.com/report/github.com/agnivade/levenshtein) [![PkgGoDev](https://pkg.go.dev/badge/github.com/agnivade/levenshtein)](https://pkg.go.dev/github.com/agnivade/levenshtein) +=========== + +[Go](http://golang.org) package to calculate the [Levenshtein Distance](http://en.wikipedia.org/wiki/Levenshtein_distance) + +The library is fully capable of working with non-ascii strings. But the strings are not normalized. That is left as a user-dependant use case. Please normalize the strings before passing it to the library if you have such a requirement. +- https://blog.golang.org/normalization + +#### Limitation + +As a performance optimization, the library can handle strings only up to 65536 characters (runes). If you need to handle strings larger than that, please pin to version 1.0.3. + +Install +------- + + go get github.com/agnivade/levenshtein + +Example +------- + +```go +package main + +import ( + "fmt" + "github.com/agnivade/levenshtein" +) + +func main() { + s1 := "kitten" + s2 := "sitting" + distance := levenshtein.ComputeDistance(s1, s2) + fmt.Printf("The distance between %s and %s is %d.\n", s1, s2, distance) + // Output: + // The distance between kitten and sitting is 3. +} + +``` + +Benchmarks +---------- + +``` +name time/op +Simple/ASCII-4 330ns ± 2% +Simple/French-4 617ns ± 2% +Simple/Nordic-4 1.16µs ± 4% +Simple/Tibetan-4 1.05µs ± 1% + +name alloc/op +Simple/ASCII-4 96.0B ± 0% +Simple/French-4 128B ± 0% +Simple/Nordic-4 192B ± 0% +Simple/Tibetan-4 144B ± 0% + +name allocs/op +Simple/ASCII-4 1.00 ± 0% +Simple/French-4 1.00 ± 0% +Simple/Nordic-4 1.00 ± 0% +Simple/Tibetan-4 1.00 ± 0% +``` + +Comparisons with other libraries +-------------------------------- + +``` +name time/op +Leven/ASCII/agniva-4 353ns ± 1% +Leven/ASCII/arbovm-4 485ns ± 1% +Leven/ASCII/dgryski-4 395ns ± 0% +Leven/French/agniva-4 648ns ± 1% +Leven/French/arbovm-4 791ns ± 0% +Leven/French/dgryski-4 682ns ± 0% +Leven/Nordic/agniva-4 1.28µs ± 1% +Leven/Nordic/arbovm-4 1.52µs ± 1% +Leven/Nordic/dgryski-4 1.32µs ± 1% +Leven/Tibetan/agniva-4 1.12µs ± 1% +Leven/Tibetan/arbovm-4 1.31µs ± 0% +Leven/Tibetan/dgryski-4 1.16µs ± 0% +``` diff --git a/vendor/github.com/agnivade/levenshtein/levenshtein.go b/vendor/github.com/agnivade/levenshtein/levenshtein.go new file mode 100644 index 00000000..f727a66f --- /dev/null +++ b/vendor/github.com/agnivade/levenshtein/levenshtein.go @@ -0,0 +1,89 @@ +// Package levenshtein is a Go implementation to calculate Levenshtein Distance. +// +// Implementation taken from +// https://gist.github.com/andrei-m/982927#gistcomment-1931258 +package levenshtein + +import "unicode/utf8" + +// minLengthThreshold is the length of the string beyond which +// an allocation will be made. Strings smaller than this will be +// zero alloc. +const minLengthThreshold = 32 + +// ComputeDistance computes the levenshtein distance between the two +// strings passed as an argument. The return value is the levenshtein distance +// +// Works on runes (Unicode code points) but does not normalize +// the input strings. See https://blog.golang.org/normalization +// and the golang.org/x/text/unicode/norm package. +func ComputeDistance(a, b string) int { + if len(a) == 0 { + return utf8.RuneCountInString(b) + } + + if len(b) == 0 { + return utf8.RuneCountInString(a) + } + + if a == b { + return 0 + } + + // We need to convert to []rune if the strings are non-ASCII. + // This could be avoided by using utf8.RuneCountInString + // and then doing some juggling with rune indices, + // but leads to far more bounds checks. It is a reasonable trade-off. + s1 := []rune(a) + s2 := []rune(b) + + // swap to save some memory O(min(a,b)) instead of O(a) + if len(s1) > len(s2) { + s1, s2 = s2, s1 + } + lenS1 := len(s1) + lenS2 := len(s2) + + // Init the row. + var x []uint16 + if lenS1+1 > minLengthThreshold { + x = make([]uint16, lenS1+1) + } else { + // We make a small optimization here for small strings. + // Because a slice of constant length is effectively an array, + // it does not allocate. So we can re-slice it to the right length + // as long as it is below a desired threshold. + x = make([]uint16, minLengthThreshold) + x = x[:lenS1+1] + } + + // we start from 1 because index 0 is already 0. + for i := 1; i < len(x); i++ { + x[i] = uint16(i) + } + + // make a dummy bounds check to prevent the 2 bounds check down below. + // The one inside the loop is particularly costly. + _ = x[lenS1] + // fill in the rest + for i := 1; i <= lenS2; i++ { + prev := uint16(i) + for j := 1; j <= lenS1; j++ { + current := x[j-1] // match + if s2[i-1] != s1[j-1] { + current = min(min(x[j-1]+1, prev+1), x[j]+1) + } + x[j-1] = prev + prev = current + } + x[lenS1] = prev + } + return int(x[lenS1]) +} + +func min(a, b uint16) uint16 { + if a < b { + return a + } + return b +} diff --git a/vendor/github.com/open-policy-agent/opa/ast/annotations.go b/vendor/github.com/open-policy-agent/opa/ast/annotations.go new file mode 100644 index 00000000..d255d2f0 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/ast/annotations.go @@ -0,0 +1,786 @@ +// Copyright 2022 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package ast + +import ( + "encoding/json" + "fmt" + "net/url" + "sort" + "strings" + + "github.com/open-policy-agent/opa/internal/deepcopy" + "github.com/open-policy-agent/opa/util" +) + +const ( + annotationScopePackage = "package" + annotationScopeImport = "import" + annotationScopeRule = "rule" + annotationScopeDocument = "document" + annotationScopeSubpackages = "subpackages" +) + +type ( + // Annotations represents metadata attached to other AST nodes such as rules. + Annotations struct { + Location *Location `json:"-"` + Scope string `json:"scope"` + Title string `json:"title,omitempty"` + Description string `json:"description,omitempty"` + Organizations []string `json:"organizations,omitempty"` + RelatedResources []*RelatedResourceAnnotation `json:"related_resources,omitempty"` + Authors []*AuthorAnnotation `json:"authors,omitempty"` + Schemas []*SchemaAnnotation `json:"schemas,omitempty"` + Custom map[string]interface{} `json:"custom,omitempty"` + node Node + } + + // SchemaAnnotation contains a schema declaration for the document identified by the path. + SchemaAnnotation struct { + Path Ref `json:"path"` + Schema Ref `json:"schema,omitempty"` + Definition *interface{} `json:"definition,omitempty"` + } + + AuthorAnnotation struct { + Name string `json:"name"` + Email string `json:"email,omitempty"` + } + + RelatedResourceAnnotation struct { + Ref url.URL `json:"ref"` + Description string `json:"description,omitempty"` + } + + AnnotationSet struct { + byRule map[*Rule][]*Annotations + byPackage map[*Package]*Annotations + byPath *annotationTreeNode + modules []*Module // Modules this set was constructed from + } + + annotationTreeNode struct { + Value *Annotations + Children map[Value]*annotationTreeNode // we assume key elements are hashable (vars and strings only!) + } + + AnnotationsRef struct { + Location *Location `json:"location"` // The location of the node the annotations are applied to + Path Ref `json:"path"` // The path of the node the annotations are applied to + Annotations *Annotations `json:"annotations,omitempty"` + node Node // The node the annotations are applied to + } +) + +func (a *Annotations) String() string { + bs, _ := json.Marshal(a) + return string(bs) +} + +// Loc returns the location of this annotation. +func (a *Annotations) Loc() *Location { + return a.Location +} + +// SetLoc updates the location of this annotation. +func (a *Annotations) SetLoc(l *Location) { + a.Location = l +} + +// Compare returns an integer indicating if a is less than, equal to, or greater +// than other. +func (a *Annotations) Compare(other *Annotations) int { + + if a == nil && other == nil { + return 0 + } + + if a == nil { + return -1 + } + + if other == nil { + return 1 + } + + if cmp := scopeCompare(a.Scope, other.Scope); cmp != 0 { + return cmp + } + + if cmp := strings.Compare(a.Title, other.Title); cmp != 0 { + return cmp + } + + if cmp := strings.Compare(a.Description, other.Description); cmp != 0 { + return cmp + } + + if cmp := compareStringLists(a.Organizations, other.Organizations); cmp != 0 { + return cmp + } + + if cmp := compareRelatedResources(a.RelatedResources, other.RelatedResources); cmp != 0 { + return cmp + } + + if cmp := compareAuthors(a.Authors, other.Authors); cmp != 0 { + return cmp + } + + if cmp := compareSchemas(a.Schemas, other.Schemas); cmp != 0 { + return cmp + } + + if cmp := util.Compare(a.Custom, other.Custom); cmp != 0 { + return cmp + } + + return 0 +} + +// GetTargetPath returns the path of the node these Annotations are applied to (the target) +func (a *Annotations) GetTargetPath() Ref { + switch n := a.node.(type) { + case *Package: + return n.Path + case *Rule: + return n.Path() + default: + return nil + } +} + +func NewAnnotationsRef(a *Annotations) *AnnotationsRef { + return &AnnotationsRef{ + Location: a.node.Loc(), + Path: a.GetTargetPath(), + Annotations: a, + node: a.node, + } +} + +func (ar *AnnotationsRef) GetPackage() *Package { + switch n := ar.node.(type) { + case *Package: + return n + case *Rule: + return n.Module.Package + default: + return nil + } +} + +func (ar *AnnotationsRef) GetRule() *Rule { + switch n := ar.node.(type) { + case *Rule: + return n + default: + return nil + } +} + +func scopeCompare(s1, s2 string) int { + + o1 := scopeOrder(s1) + o2 := scopeOrder(s2) + + if o2 < o1 { + return 1 + } else if o2 > o1 { + return -1 + } + + if s1 < s2 { + return -1 + } else if s2 < s1 { + return 1 + } + + return 0 +} + +func scopeOrder(s string) int { + switch s { + case annotationScopeRule: + return 1 + } + return 0 +} + +func compareAuthors(a, b []*AuthorAnnotation) int { + if len(a) > len(b) { + return 1 + } else if len(a) < len(b) { + return -1 + } + + for i := 0; i < len(a); i++ { + if cmp := a[i].Compare(b[i]); cmp != 0 { + return cmp + } + } + + return 0 +} + +func compareRelatedResources(a, b []*RelatedResourceAnnotation) int { + if len(a) > len(b) { + return 1 + } else if len(a) < len(b) { + return -1 + } + + for i := 0; i < len(a); i++ { + if cmp := strings.Compare(a[i].String(), b[i].String()); cmp != 0 { + return cmp + } + } + + return 0 +} + +func compareSchemas(a, b []*SchemaAnnotation) int { + max := len(a) + if len(b) < max { + max = len(b) + } + + for i := 0; i < max; i++ { + if cmp := a[i].Compare(b[i]); cmp != 0 { + return cmp + } + } + + if len(a) > len(b) { + return 1 + } else if len(a) < len(b) { + return -1 + } + + return 0 +} + +func compareStringLists(a, b []string) int { + if len(a) > len(b) { + return 1 + } else if len(a) < len(b) { + return -1 + } + + for i := 0; i < len(a); i++ { + if cmp := strings.Compare(a[i], b[i]); cmp != 0 { + return cmp + } + } + + return 0 +} + +// Copy returns a deep copy of s. +func (a *Annotations) Copy(node Node) *Annotations { + cpy := *a + + cpy.Organizations = make([]string, len(a.Organizations)) + copy(cpy.Organizations, a.Organizations) + + cpy.RelatedResources = make([]*RelatedResourceAnnotation, len(a.RelatedResources)) + for i := range a.RelatedResources { + cpy.RelatedResources[i] = a.RelatedResources[i].Copy() + } + + cpy.Authors = make([]*AuthorAnnotation, len(a.Authors)) + for i := range a.Authors { + cpy.Authors[i] = a.Authors[i].Copy() + } + + cpy.Schemas = make([]*SchemaAnnotation, len(a.Schemas)) + for i := range a.Schemas { + cpy.Schemas[i] = a.Schemas[i].Copy() + } + + cpy.Custom = deepcopy.Map(a.Custom) + + cpy.node = node + + return &cpy +} + +// toObject constructs an AST Object from a. +func (a *Annotations) toObject() (*Object, *Error) { + obj := NewObject() + + if a == nil { + return &obj, nil + } + + if len(a.Scope) > 0 { + obj.Insert(StringTerm("scope"), StringTerm(a.Scope)) + } + + if len(a.Title) > 0 { + obj.Insert(StringTerm("title"), StringTerm(a.Title)) + } + + if len(a.Description) > 0 { + obj.Insert(StringTerm("description"), StringTerm(a.Description)) + } + + if len(a.Organizations) > 0 { + orgs := make([]*Term, 0, len(a.Organizations)) + for _, org := range a.Organizations { + orgs = append(orgs, StringTerm(org)) + } + obj.Insert(StringTerm("organizations"), ArrayTerm(orgs...)) + } + + if len(a.RelatedResources) > 0 { + rrs := make([]*Term, 0, len(a.RelatedResources)) + for _, rr := range a.RelatedResources { + rrObj := NewObject(Item(StringTerm("ref"), StringTerm(rr.Ref.String()))) + if len(rr.Description) > 0 { + rrObj.Insert(StringTerm("description"), StringTerm(rr.Description)) + } + rrs = append(rrs, NewTerm(rrObj)) + } + obj.Insert(StringTerm("related_resources"), ArrayTerm(rrs...)) + } + + if len(a.Authors) > 0 { + as := make([]*Term, 0, len(a.Authors)) + for _, author := range a.Authors { + aObj := NewObject() + if len(author.Name) > 0 { + aObj.Insert(StringTerm("name"), StringTerm(author.Name)) + } + if len(author.Email) > 0 { + aObj.Insert(StringTerm("email"), StringTerm(author.Email)) + } + as = append(as, NewTerm(aObj)) + } + obj.Insert(StringTerm("authors"), ArrayTerm(as...)) + } + + if len(a.Schemas) > 0 { + ss := make([]*Term, 0, len(a.Schemas)) + for _, s := range a.Schemas { + sObj := NewObject() + if len(s.Path) > 0 { + sObj.Insert(StringTerm("path"), NewTerm(s.Path.toArray())) + } + if len(s.Schema) > 0 { + sObj.Insert(StringTerm("schema"), NewTerm(s.Schema.toArray())) + } + if s.Definition != nil { + def, err := InterfaceToValue(s.Definition) + if err != nil { + return nil, NewError(CompileErr, a.Location, "invalid definition in schema annotation: %s", err.Error()) + } + sObj.Insert(StringTerm("definition"), NewTerm(def)) + } + ss = append(ss, NewTerm(sObj)) + } + obj.Insert(StringTerm("schemas"), ArrayTerm(ss...)) + } + + if len(a.Custom) > 0 { + c, err := InterfaceToValue(a.Custom) + if err != nil { + return nil, NewError(CompileErr, a.Location, "invalid custom annotation %s", err.Error()) + } + obj.Insert(StringTerm("custom"), NewTerm(c)) + } + + return &obj, nil +} + +func attachAnnotationsNodes(mod *Module) Errors { + var errs Errors + + // Find first non-annotation statement following each annotation and attach + // the annotation to that statement. + for _, a := range mod.Annotations { + for _, stmt := range mod.stmts { + _, ok := stmt.(*Annotations) + if !ok { + if stmt.Loc().Row > a.Location.Row { + a.node = stmt + break + } + } + } + + if a.Scope == "" { + switch a.node.(type) { + case *Rule: + a.Scope = annotationScopeRule + case *Package: + a.Scope = annotationScopePackage + case *Import: + a.Scope = annotationScopeImport + } + } + + if err := validateAnnotationScopeAttachment(a); err != nil { + errs = append(errs, err) + } + } + + return errs +} + +func validateAnnotationScopeAttachment(a *Annotations) *Error { + + switch a.Scope { + case annotationScopeRule, annotationScopeDocument: + if _, ok := a.node.(*Rule); ok { + return nil + } + return newScopeAttachmentErr(a, "rule") + case annotationScopePackage, annotationScopeSubpackages: + if _, ok := a.node.(*Package); ok { + return nil + } + return newScopeAttachmentErr(a, "package") + } + + return NewError(ParseErr, a.Loc(), "invalid annotation scope '%v'", a.Scope) +} + +// Copy returns a deep copy of a. +func (a *AuthorAnnotation) Copy() *AuthorAnnotation { + cpy := *a + return &cpy +} + +// Compare returns an integer indicating if s is less than, equal to, or greater +// than other. +func (a *AuthorAnnotation) Compare(other *AuthorAnnotation) int { + if cmp := strings.Compare(a.Name, other.Name); cmp != 0 { + return cmp + } + + if cmp := strings.Compare(a.Email, other.Email); cmp != 0 { + return cmp + } + + return 0 +} + +func (a *AuthorAnnotation) String() string { + if len(a.Email) == 0 { + return a.Name + } else if len(a.Name) == 0 { + return fmt.Sprintf("<%s>", a.Email) + } else { + return fmt.Sprintf("%s <%s>", a.Name, a.Email) + } +} + +// Copy returns a deep copy of rr. +func (rr *RelatedResourceAnnotation) Copy() *RelatedResourceAnnotation { + cpy := *rr + return &cpy +} + +// Compare returns an integer indicating if s is less than, equal to, or greater +// than other. +func (rr *RelatedResourceAnnotation) Compare(other *RelatedResourceAnnotation) int { + if cmp := strings.Compare(rr.Description, other.Description); cmp != 0 { + return cmp + } + + if cmp := strings.Compare(rr.Ref.String(), other.Ref.String()); cmp != 0 { + return cmp + } + + return 0 +} + +func (rr *RelatedResourceAnnotation) String() string { + bs, _ := json.Marshal(rr) + return string(bs) +} + +func (rr *RelatedResourceAnnotation) MarshalJSON() ([]byte, error) { + d := map[string]interface{}{ + "ref": rr.Ref.String(), + } + + if len(rr.Description) > 0 { + d["description"] = rr.Description + } + + return json.Marshal(d) +} + +// Copy returns a deep copy of s. +func (s *SchemaAnnotation) Copy() *SchemaAnnotation { + cpy := *s + return &cpy +} + +// Compare returns an integer indicating if s is less than, equal to, or greater +// than other. +func (s *SchemaAnnotation) Compare(other *SchemaAnnotation) int { + + if cmp := s.Path.Compare(other.Path); cmp != 0 { + return cmp + } + + if cmp := s.Schema.Compare(other.Schema); cmp != 0 { + return cmp + } + + if s.Definition != nil && other.Definition == nil { + return -1 + } else if s.Definition == nil && other.Definition != nil { + return 1 + } else if s.Definition != nil && other.Definition != nil { + return util.Compare(*s.Definition, *other.Definition) + } + + return 0 +} + +func (s *SchemaAnnotation) String() string { + bs, _ := json.Marshal(s) + return string(bs) +} + +func newAnnotationSet() *AnnotationSet { + return &AnnotationSet{ + byRule: map[*Rule][]*Annotations{}, + byPackage: map[*Package]*Annotations{}, + byPath: newAnnotationTree(), + } +} + +func BuildAnnotationSet(modules []*Module) (*AnnotationSet, Errors) { + as := newAnnotationSet() + var errs Errors + for _, m := range modules { + for _, a := range m.Annotations { + if err := as.add(a); err != nil { + errs = append(errs, err) + } + } + } + if len(errs) > 0 { + return nil, errs + } + as.modules = modules + return as, nil +} + +func (as *AnnotationSet) add(a *Annotations) *Error { + switch a.Scope { + case annotationScopeRule: + rule := a.node.(*Rule) + as.byRule[rule] = append(as.byRule[rule], a) + case annotationScopePackage: + pkg := a.node.(*Package) + if exist, ok := as.byPackage[pkg]; ok { + return errAnnotationRedeclared(a, exist.Location) + } + as.byPackage[pkg] = a + case annotationScopeDocument: + rule := a.node.(*Rule) + path := rule.Path() + x := as.byPath.get(path) + if x != nil { + return errAnnotationRedeclared(a, x.Value.Location) + } + as.byPath.insert(path, a) + case annotationScopeSubpackages: + pkg := a.node.(*Package) + x := as.byPath.get(pkg.Path) + if x != nil && x.Value != nil { + return errAnnotationRedeclared(a, x.Value.Location) + } + as.byPath.insert(pkg.Path, a) + } + return nil +} + +func (as *AnnotationSet) GetRuleScope(r *Rule) []*Annotations { + if as == nil { + return nil + } + return as.byRule[r] +} + +func (as *AnnotationSet) GetSubpackagesScope(path Ref) []*Annotations { + if as == nil { + return nil + } + return as.byPath.ancestors(path) +} + +func (as *AnnotationSet) GetDocumentScope(path Ref) *Annotations { + if as == nil { + return nil + } + if node := as.byPath.get(path); node != nil { + return node.Value + } + return nil +} + +func (as *AnnotationSet) GetPackageScope(pkg *Package) *Annotations { + if as == nil { + return nil + } + return as.byPackage[pkg] +} + +// Flatten returns a flattened list view of this AnnotationSet. +// The returned slice is sorted, first by the annotations' target path, then by their target location +func (as *AnnotationSet) Flatten() []*AnnotationsRef { + // This preallocation often won't be optimal, but it's superior to starting with a nil slice. + refs := make([]*AnnotationsRef, 0, len(as.byPath.Children)+len(as.byRule)+len(as.byPackage)) + + refs = as.byPath.flatten(refs) + + for _, a := range as.byPackage { + refs = append(refs, NewAnnotationsRef(a)) + } + + for _, as := range as.byRule { + for _, a := range as { + refs = append(refs, NewAnnotationsRef(a)) + } + } + + // Sort by path, then annotation location, for stable output + sort.SliceStable(refs, func(i, j int) bool { + if refs[i].Path.Compare(refs[j].Path) < 0 { + return true + } + if refs[i].Annotations.Location.Compare(refs[j].Annotations.Location) < 0 { + return true + } + return false + }) + + return refs +} + +// Chain returns the chain of annotations leading up to the given rule. +// The returned slice is ordered as follows +// 0. Entries for the given rule, ordered from the METADATA block declared immediately above the rule, to the block declared farthest away (always at least one entry) +// 1. The 'document' scope entry, if any +// 2. The 'package' scope entry, if any +// 3. Entries for the 'subpackages' scope, if any; ordered from the closest package path to the fartest. E.g.: 'do.re.mi', 'do.re', 'do' +// The returned slice is guaranteed to always contain at least one entry, corresponding to the given rule. +func (as *AnnotationSet) Chain(rule *Rule) []*AnnotationsRef { + var refs []*AnnotationsRef + + ruleAnnots := as.GetRuleScope(rule) + + if len(ruleAnnots) >= 1 { + for _, a := range ruleAnnots { + refs = append(refs, NewAnnotationsRef(a)) + } + } else { + // Make sure there is always a leading entry representing the passed rule, even if it has no annotations + refs = append(refs, &AnnotationsRef{ + Location: rule.Location, + Path: rule.Path(), + node: rule, + }) + } + + if len(refs) > 1 { + // Sort by annotation location; chain must start with annotations declared closest to rule, then going outward + sort.SliceStable(refs, func(i, j int) bool { + return refs[i].Annotations.Location.Compare(refs[j].Annotations.Location) > 0 + }) + } + + docAnnots := as.GetDocumentScope(rule.Path()) + if docAnnots != nil { + refs = append(refs, NewAnnotationsRef(docAnnots)) + } + + pkg := rule.Module.Package + pkgAnnots := as.GetPackageScope(pkg) + if pkgAnnots != nil { + refs = append(refs, NewAnnotationsRef(pkgAnnots)) + } + + subPkgAnnots := as.GetSubpackagesScope(pkg.Path) + // We need to reverse the order, as subPkgAnnots ordering will start at the root, + // whereas we want to end at the root. + for i := len(subPkgAnnots) - 1; i >= 0; i-- { + refs = append(refs, NewAnnotationsRef(subPkgAnnots[i])) + } + + return refs +} + +func newAnnotationTree() *annotationTreeNode { + return &annotationTreeNode{ + Value: nil, + Children: map[Value]*annotationTreeNode{}, + } +} + +func (t *annotationTreeNode) insert(path Ref, value *Annotations) { + node := t + for _, k := range path { + child, ok := node.Children[k.Value] + if !ok { + child = newAnnotationTree() + node.Children[k.Value] = child + } + node = child + } + node.Value = value +} + +func (t *annotationTreeNode) get(path Ref) *annotationTreeNode { + node := t + for _, k := range path { + if node == nil { + return nil + } + child, ok := node.Children[k.Value] + if !ok { + return nil + } + node = child + } + return node +} + +// ancestors returns a slice of annotations in ascending order, starting with the root of ref; e.g.: 'root', 'root.foo', 'root.foo.bar'. +func (t *annotationTreeNode) ancestors(path Ref) (result []*Annotations) { + node := t + for _, k := range path { + if node == nil { + return result + } + child, ok := node.Children[k.Value] + if !ok { + return result + } + if child.Value != nil { + result = append(result, child.Value) + } + node = child + } + return result +} + +func (t *annotationTreeNode) flatten(refs []*AnnotationsRef) []*AnnotationsRef { + if a := t.Value; a != nil { + refs = append(refs, NewAnnotationsRef(a)) + } + for _, c := range t.Children { + refs = c.flatten(refs) + } + return refs +} diff --git a/vendor/github.com/open-policy-agent/opa/ast/builtins.go b/vendor/github.com/open-policy-agent/opa/ast/builtins.go index 4afa98f3..e04ec33f 100644 --- a/vendor/github.com/open-policy-agent/opa/ast/builtins.go +++ b/vendor/github.com/open-policy-agent/opa/ast/builtins.go @@ -102,6 +102,7 @@ var DefaultBuiltins = [...]*Builtin{ RegexTemplateMatch, RegexFind, RegexFindAllStringSubmatch, + RegexReplace, // Sets SetDiff, @@ -109,9 +110,12 @@ var DefaultBuiltins = [...]*Builtin{ Union, // Strings + AnyPrefixMatch, + AnySuffixMatch, Concat, FormatInt, IndexOf, + IndexOfN, Substring, Lower, Upper, @@ -156,9 +160,11 @@ var DefaultBuiltins = [...]*Builtin{ // Object Manipulation ObjectUnion, + ObjectUnionN, ObjectRemove, ObjectFilter, ObjectGet, + ObjectSubset, // JSON Object Manipulation JSONFilter, @@ -210,6 +216,7 @@ var DefaultBuiltins = [...]*Builtin{ // Graphs WalkBuiltin, ReachableBuiltin, + ReachablePathsBuiltin, // Sort Sort, @@ -227,8 +234,17 @@ var DefaultBuiltins = [...]*Builtin{ // HTTP HTTPSend, + // GraphQL + GraphQLParse, + GraphQLParseAndVerify, + GraphQLParseQuery, + GraphQLParseSchema, + GraphQLIsValid, + // Rego RegoParseModule, + RegoMetadataChain, + RegoMetadataRule, // OPA OPARuntime, @@ -250,12 +266,13 @@ var DefaultBuiltins = [...]*Builtin{ GlobQuoteMeta, // Units + UnitsParse, UnitsParseBytes, // UUIDs UUIDRFC4122, - //SemVers + // SemVers SemVerIsValid, SemVerCompare, @@ -268,14 +285,18 @@ var DefaultBuiltins = [...]*Builtin{ // built-in definitions. var BuiltinMap map[string]*Builtin -// IgnoreDuringPartialEval is a set of built-in functions that should not be -// evaluated during partial evaluation. These functions are not partially -// evaluated because they are not pure. +// Deprecated: Builtins can now be directly annotated with the +// Nondeterministic property, and when set to true, will be ignored +// for partial evaluation. var IgnoreDuringPartialEval = []*Builtin{ + RandIntn, + UUIDRFC4122, + JWTDecodeVerify, + JWTEncodeSignRaw, + JWTEncodeSign, NowNanos, HTTPSend, - UUIDRFC4122, - RandIntn, + OPARuntime, NetLookupIPAddr, } @@ -338,219 +359,283 @@ var MemberWithKey = &Builtin{ /** * Comparisons */ +var comparison = category("comparison") -// GreaterThan represents the ">" comparison operator. var GreaterThan = &Builtin{ - Name: "gt", - Infix: ">", + Name: "gt", + Infix: ">", + Categories: comparison, Decl: types.NewFunction( - types.Args(types.A, types.A), - types.B, + types.Args( + types.Named("x", types.A), + types.Named("y", types.A), + ), + types.Named("result", types.B).Description("true if `x` is greater than `y`; false otherwise"), ), } -// GreaterThanEq represents the ">=" comparison operator. var GreaterThanEq = &Builtin{ - Name: "gte", - Infix: ">=", + Name: "gte", + Infix: ">=", + Categories: comparison, Decl: types.NewFunction( - types.Args(types.A, types.A), - types.B, + types.Args( + types.Named("x", types.A), + types.Named("y", types.A), + ), + types.Named("result", types.B).Description("true if `x` is greater or equal to `y`; false otherwise"), ), } // LessThan represents the "<" comparison operator. var LessThan = &Builtin{ - Name: "lt", - Infix: "<", + Name: "lt", + Infix: "<", + Categories: comparison, Decl: types.NewFunction( - types.Args(types.A, types.A), - types.B, + types.Args( + types.Named("x", types.A), + types.Named("y", types.A), + ), + types.Named("result", types.B).Description("true if `x` is less than `y`; false otherwise"), ), } -// LessThanEq represents the "<=" comparison operator. var LessThanEq = &Builtin{ - Name: "lte", - Infix: "<=", + Name: "lte", + Infix: "<=", + Categories: comparison, Decl: types.NewFunction( - types.Args(types.A, types.A), - types.B, + types.Args( + types.Named("x", types.A), + types.Named("y", types.A), + ), + types.Named("result", types.B).Description("true if `x` is less than or equal to `y`; false otherwise"), ), } -// NotEqual represents the "!=" comparison operator. var NotEqual = &Builtin{ - Name: "neq", - Infix: "!=", + Name: "neq", + Infix: "!=", + Categories: comparison, Decl: types.NewFunction( - types.Args(types.A, types.A), - types.B, + types.Args( + types.Named("x", types.A), + types.Named("y", types.A), + ), + types.Named("result", types.B).Description("true if `x` is not equal to `y`; false otherwise"), ), } // Equal represents the "==" comparison operator. var Equal = &Builtin{ - Name: "equal", - Infix: "==", + Name: "equal", + Infix: "==", + Categories: comparison, Decl: types.NewFunction( - types.Args(types.A, types.A), - types.B, + types.Args( + types.Named("x", types.A), + types.Named("y", types.A), + ), + types.Named("result", types.B).Description("true if `x` is equal to `y`; false otherwise"), ), } /** * Arithmetic */ +var number = category("numbers") -// Plus adds two numbers together. var Plus = &Builtin{ - Name: "plus", - Infix: "+", + Name: "plus", + Infix: "+", + Description: "Plus adds two numbers together.", Decl: types.NewFunction( - types.Args(types.N, types.N), - types.N, + types.Args( + types.Named("x", types.N), + types.Named("y", types.N), + ), + types.Named("z", types.N).Description("the sum of `x` and `y`"), ), + Categories: number, } -// Minus subtracts the second number from the first number or computes the diff -// between two sets. var Minus = &Builtin{ - Name: "minus", - Infix: "-", + Name: "minus", + Infix: "-", + Description: "Minus subtracts the second number from the first number or computes the difference between two sets.", Decl: types.NewFunction( types.Args( - types.NewAny(types.N, types.NewSet(types.A)), - types.NewAny(types.N, types.NewSet(types.A)), + types.Named("x", types.NewAny(types.N, types.NewSet(types.A))), + types.Named("y", types.NewAny(types.N, types.NewSet(types.A))), ), - types.NewAny(types.N, types.NewSet(types.A)), + types.Named("z", types.NewAny(types.N, types.NewSet(types.A))).Description("the difference of `x` and `y`"), ), + Categories: category("sets", "numbers"), } -// Multiply multiplies two numbers together. var Multiply = &Builtin{ - Name: "mul", - Infix: "*", + Name: "mul", + Infix: "*", + Description: "Multiplies two numbers.", Decl: types.NewFunction( - types.Args(types.N, types.N), - types.N, + types.Args( + types.Named("x", types.N), + types.Named("y", types.N), + ), + types.Named("z", types.N).Description("the product of `x` and `y`"), ), + Categories: number, } -// Divide divides the first number by the second number. var Divide = &Builtin{ - Name: "div", - Infix: "/", + Name: "div", + Infix: "/", + Description: "Divides the first number by the second number.", Decl: types.NewFunction( - types.Args(types.N, types.N), - types.N, + types.Args( + types.Named("x", types.N).Description("the dividend"), + types.Named("y", types.N).Description("the divisor"), + ), + types.Named("z", types.N).Description("the result of `x` divided by `y`"), ), + Categories: number, } -// Round rounds the number to the nearest integer. var Round = &Builtin{ - Name: "round", + Name: "round", + Description: "Rounds the number to the nearest integer.", Decl: types.NewFunction( - types.Args(types.N), - types.N, + types.Args( + types.Named("x", types.N).Description("the number to round"), + ), + types.Named("y", types.N).Description("the result of rounding `x`"), ), + Categories: number, } -// Ceil rounds the number up to the nearest integer. var Ceil = &Builtin{ - Name: "ceil", + Name: "ceil", + Description: "Rounds the number _up_ to the nearest integer.", Decl: types.NewFunction( - types.Args(types.N), - types.N, + types.Args( + types.Named("x", types.N).Description("the number to round"), + ), + types.Named("y", types.N).Description("the result of rounding `x` _up_"), ), + Categories: number, } -// Floor rounds the number down to the nearest integer. var Floor = &Builtin{ - Name: "floor", + Name: "floor", + Description: "Rounds the number _down_ to the nearest integer.", Decl: types.NewFunction( - types.Args(types.N), - types.N, + types.Args( + types.Named("x", types.N).Description("the number to round"), + ), + types.Named("y", types.N).Description("the result of rounding `x` _down_"), ), + Categories: number, } -// Abs returns the number without its sign. var Abs = &Builtin{ - Name: "abs", + Name: "abs", + Description: "Returns the number without its sign.", Decl: types.NewFunction( - types.Args(types.N), - types.N, + types.Args( + types.Named("x", types.N), + ), + types.Named("y", types.N).Description("the absolute value of `x`"), ), + Categories: number, } -// Rem returns the remainder for x%y for y != 0. var Rem = &Builtin{ - Name: "rem", - Infix: "%", + Name: "rem", + Infix: "%", + Description: "Returns the remainder for of `x` divided by `y`, for `y != 0`.", Decl: types.NewFunction( - types.Args(types.N, types.N), - types.N, + types.Args( + types.Named("x", types.N), + types.Named("y", types.N), + ), + types.Named("z", types.N).Description("the remainder"), ), + Categories: number, } /** * Bitwise */ -// BitsOr returns the bitwise "or" of two integers. var BitsOr = &Builtin{ - Name: "bits.or", + Name: "bits.or", + Description: "Returns the bitwise \"OR\" of two integers.", Decl: types.NewFunction( - types.Args(types.N, types.N), - types.N, + types.Args( + types.Named("x", types.N), + types.Named("y", types.N), + ), + types.Named("z", types.N), ), } -// BitsAnd returns the bitwise "and" of two integers. var BitsAnd = &Builtin{ - Name: "bits.and", + Name: "bits.and", + Description: "Returns the bitwise \"AND\" of two integers.", Decl: types.NewFunction( - types.Args(types.N, types.N), - types.N, + types.Args( + types.Named("x", types.N), + types.Named("y", types.N), + ), + types.Named("z", types.N), ), } -// BitsNegate returns the bitwise "negation" of an integer (i.e. flips each -// bit). var BitsNegate = &Builtin{ - Name: "bits.negate", + Name: "bits.negate", + Description: "Returns the bitwise negation (flip) of an integer.", Decl: types.NewFunction( - types.Args(types.N), - types.N, + types.Args( + types.Named("x", types.N), + ), + types.Named("z", types.N), ), } -// BitsXOr returns the bitwise "exclusive-or" of two integers. var BitsXOr = &Builtin{ - Name: "bits.xor", + Name: "bits.xor", + Description: "Returns the bitwise \"XOR\" (exclusive-or) of two integers.", Decl: types.NewFunction( - types.Args(types.N, types.N), - types.N, + types.Args( + types.Named("x", types.N), + types.Named("y", types.N), + ), + types.Named("z", types.N), ), } -// BitsShiftLeft returns a new integer with its bits shifted some value to the -// left. var BitsShiftLeft = &Builtin{ - Name: "bits.lsh", + Name: "bits.lsh", + Description: "Returns a new integer with its bits shifted `s` bits to the left.", Decl: types.NewFunction( - types.Args(types.N, types.N), - types.N, + types.Args( + types.Named("x", types.N), + types.Named("s", types.N), + ), + types.Named("z", types.N), ), } -// BitsShiftRight returns a new integer with its bits shifted some value to the -// right. var BitsShiftRight = &Builtin{ - Name: "bits.rsh", + Name: "bits.rsh", + Description: "Returns a new integer with its bits shifted `s` bits to the right.", Decl: types.NewFunction( - types.Args(types.N, types.N), - types.N, + types.Args( + types.Named("x", types.N), + types.Named("s", types.N), + ), + types.Named("z", types.N), ), } @@ -558,550 +643,662 @@ var BitsShiftRight = &Builtin{ * Sets */ -// And performs an intersection operation on sets. +var sets = category("sets") + var And = &Builtin{ - Name: "and", - Infix: "&", + Name: "and", + Infix: "&", + Description: "Returns the intersection of two sets.", Decl: types.NewFunction( types.Args( - types.NewSet(types.A), - types.NewSet(types.A), + types.Named("x", types.NewSet(types.A)), + types.Named("y", types.NewSet(types.A)), ), - types.NewSet(types.A), + types.Named("z", types.NewSet(types.A)).Description("the intersection of `x` and `y`"), ), + Categories: sets, } // Or performs a union operation on sets. var Or = &Builtin{ - Name: "or", - Infix: "|", + Name: "or", + Infix: "|", + Description: "Returns the union of two sets.", Decl: types.NewFunction( types.Args( - types.NewSet(types.A), - types.NewSet(types.A), + types.Named("x", types.NewSet(types.A)), + types.Named("y", types.NewSet(types.A)), ), - types.NewSet(types.A), + types.Named("z", types.NewSet(types.A)).Description("the union of `x` and `y`"), + ), + Categories: sets, +} + +var Intersection = &Builtin{ + Name: "intersection", + Description: "Returns the intersection of the given input sets.", + Decl: types.NewFunction( + types.Args( + types.Named("xs", types.NewSet(types.NewSet(types.A))).Description("set of sets to intersect"), + ), + types.Named("y", types.NewSet(types.A)).Description("the intersection of all `xs` sets"), + ), + Categories: sets, +} + +var Union = &Builtin{ + Name: "union", + Description: "Returns the union of the given input sets.", + Decl: types.NewFunction( + types.Args( + types.Named("xs", types.NewSet(types.NewSet(types.A))).Description("set of sets to merge"), + ), + types.Named("y", types.NewSet(types.A)).Description("the union of all `xs` sets"), ), + Categories: sets, } /** * Aggregates */ -// Count takes a collection or string and counts the number of elements in it. +var aggregates = category("aggregates") + var Count = &Builtin{ - Name: "count", + Name: "count", + Description: " Count takes a collection or string and returns the number of elements (or characters) in it.", Decl: types.NewFunction( types.Args( - types.NewAny( + types.Named("collection", types.NewAny( types.NewSet(types.A), types.NewArray(nil, types.A), types.NewObject(nil, types.NewDynamicProperty(types.A, types.A)), types.S, - ), + )).Description("the set/array/object/string to be counted"), ), - types.N, + types.Named("n", types.N).Description("the count of elements, key/val pairs, or characters, respectively."), ), + Categories: aggregates, } -// Sum takes an array or set of numbers and sums them. var Sum = &Builtin{ - Name: "sum", + Name: "sum", + Description: "Sums elements of an array or set of numbers.", Decl: types.NewFunction( types.Args( - types.NewAny( + types.Named("collection", types.NewAny( types.NewSet(types.N), types.NewArray(nil, types.N), - ), + )), ), - types.N, + types.Named("n", types.N).Description("the sum of all elements"), ), + Categories: aggregates, } -// Product takes an array or set of numbers and multiplies them. var Product = &Builtin{ - Name: "product", + Name: "product", + Description: "Muliplies elements of an array or set of numbers", Decl: types.NewFunction( types.Args( - types.NewAny( + types.Named("collection", types.NewAny( types.NewSet(types.N), types.NewArray(nil, types.N), - ), + )), ), - types.N, + types.Named("n", types.N).Description("the product of all elements"), ), + Categories: aggregates, } -// Max returns the maximum value in a collection. var Max = &Builtin{ - Name: "max", + Name: "max", + Description: "Returns the maximum value in a collection.", Decl: types.NewFunction( types.Args( - types.NewAny( + types.Named("collection", types.NewAny( types.NewSet(types.A), types.NewArray(nil, types.A), - ), + )), ), - types.A, + types.Named("n", types.A).Description("the maximum of all elements"), ), + Categories: aggregates, } -// Min returns the minimum value in a collection. var Min = &Builtin{ - Name: "min", + Name: "min", + Description: "Returns the minimum value in a collection.", Decl: types.NewFunction( types.Args( - types.NewAny( + types.Named("collection", types.NewAny( types.NewSet(types.A), types.NewArray(nil, types.A), - ), + )), ), - types.A, + types.Named("n", types.A).Description("the minimum of all elements"), ), + Categories: aggregates, } -// All takes a list and returns true if all of the items -// are true. A collection of length 0 returns true. -var All = &Builtin{ - Name: "all", - Decl: types.NewFunction( - types.Args( - types.NewAny( - types.NewSet(types.A), - types.NewArray(nil, types.A), - ), - ), - types.B, - ), -} +/** + * Sorting + */ -// Any takes a collection and returns true if any of the items -// is true. A collection of length 0 returns false. -var Any = &Builtin{ - Name: "any", +var Sort = &Builtin{ + Name: "sort", + Description: "Returns a sorted array.", Decl: types.NewFunction( types.Args( - types.NewAny( - types.NewSet(types.A), + types.Named("collection", types.NewAny( types.NewArray(nil, types.A), - ), + types.NewSet(types.A), + )).Description("the array or set to be sorted"), ), - types.B, + types.Named("n", types.NewArray(nil, types.A)).Description("the sorted array"), ), + Categories: aggregates, } /** * Arrays */ -// ArrayConcat returns the result of concatenating two arrays together. var ArrayConcat = &Builtin{ - Name: "array.concat", + Name: "array.concat", + Description: "Concatenates two arrays.", Decl: types.NewFunction( types.Args( - types.NewArray(nil, types.A), - types.NewArray(nil, types.A), + types.Named("x", types.NewArray(nil, types.A)), + types.Named("y", types.NewArray(nil, types.A)), ), - types.NewArray(nil, types.A), + types.Named("z", types.NewArray(nil, types.A)).Description("the concatenation of `x` and `y`"), ), } -// ArraySlice returns a slice of a given array var ArraySlice = &Builtin{ - Name: "array.slice", + Name: "array.slice", + Description: "Returns a slice of a given array. If `start` is greater or equal than `stop`, `slice` is `[]`.", Decl: types.NewFunction( types.Args( - types.NewArray(nil, types.A), - types.NewNumber(), - types.NewNumber(), + types.Named("arr", types.NewArray(nil, types.A)).Description("the array to be sliced"), + types.Named("start", types.NewNumber()).Description("the start index of the returned slice; if less than zero, it's clamped to 0"), + types.Named("stop", types.NewNumber()).Description("the stop index of the returned slice; if larger than `count(arr)`, it's clamped to `count(arr)`"), ), - types.NewArray(nil, types.A), + types.Named("slice", types.NewArray(nil, types.A)).Description("the subslice of `array`, from `start` to `end`, including `arr[start]`, but excluding `arr[end]`"), ), -} +} // NOTE(sr): this function really needs examples -// ArrayReverse returns a given array, reversed var ArrayReverse = &Builtin{ - Name: "array.reverse", + Name: "array.reverse", + Description: "Returns the reverse of a given array.", Decl: types.NewFunction( types.Args( - types.NewArray(nil, types.A), + types.Named("arr", types.NewArray(nil, types.A)).Description("the array to be reversed"), ), - types.NewArray(nil, types.A), + types.Named("rev", types.NewArray(nil, types.A)).Description("an array containing the elements of `arr` in reverse order"), ), } /** * Conversions */ +var conversions = category("conversions") -// ToNumber takes a string, bool, or number value and converts it to a number. -// Strings are converted to numbers using strconv.Atoi. -// Boolean false is converted to 0 and boolean true is converted to 1. var ToNumber = &Builtin{ - Name: "to_number", + Name: "to_number", + Description: "Converts a string, bool, or number value to a number: Strings are converted to numbers using `strconv.Atoi`, Boolean `false` is converted to 0 and `true` is converted to 1.", Decl: types.NewFunction( types.Args( - types.NewAny( + types.Named("x", types.NewAny( types.N, types.S, types.B, types.NewNull(), - ), + )), ), - types.N, + types.Named("num", types.N), ), + Categories: conversions, } /** * Regular Expressions */ -// RegexMatch takes two strings and evaluates to true if the string in the second -// position matches the pattern in the first position. var RegexMatch = &Builtin{ - Name: "regex.match", + Name: "regex.match", + Description: "Matches a string against a regular expression.", Decl: types.NewFunction( types.Args( - types.S, - types.S, + types.Named("pattern", types.S).Description("regular expression"), + types.Named("value", types.S).Description("value to match against `pattern`"), ), - types.B, + types.Named("result", types.B), ), } -// RegexIsValid returns true if the regex pattern string is valid, otherwise false. var RegexIsValid = &Builtin{ - Name: "regex.is_valid", + Name: "regex.is_valid", + Description: "Checks if a string is a valid regular expression: the detailed syntax for patterns is defined by https://github.com/google/re2/wiki/Syntax.", Decl: types.NewFunction( types.Args( - types.S, + types.Named("pattern", types.S).Description("regular expression"), ), - types.B, + types.Named("result", types.B), ), } -// RegexFindAllStringSubmatch returns an array of all successive matches of the expression. -// It takes two strings and a number, the pattern, the value and number of matches to -// return, -1 means all matches. var RegexFindAllStringSubmatch = &Builtin{ - Name: "regex.find_all_string_submatch_n", + Name: "regex.find_all_string_submatch_n", + Description: "Returns all successive matches of the expression.", Decl: types.NewFunction( types.Args( - types.S, - types.S, - types.N, + types.Named("pattern", types.S).Description("regular expression"), + types.Named("value", types.S).Description("string to match"), + types.Named("number", types.N).Description("number of matches to return; `-1` means all matches"), ), - types.NewArray(nil, types.NewArray(nil, types.S)), + types.Named("output", types.NewArray(nil, types.NewArray(nil, types.S))), ), } -// RegexTemplateMatch takes two strings and evaluates to true if the string in the second -// position matches the pattern in the first position. var RegexTemplateMatch = &Builtin{ - Name: "regex.template_match", + Name: "regex.template_match", + Description: "Matches a string against a pattern, where there pattern may be glob-like", Decl: types.NewFunction( types.Args( - types.S, - types.S, - types.S, - types.S, + types.Named("template", types.S).Description("template expression containing `0..n` regular expressions"), + types.Named("value", types.S).Description("string to match"), + types.Named("delimiter_start", types.S).Description("start delimiter of the regular expression in `template`"), + types.Named("delimiter_end", types.S).Description("end delimiter of the regular expression in `template`"), ), - types.B, + types.Named("result", types.B), ), -} +} // TODO(sr): example:`regex.template_match("urn:foo:{.*}", "urn:foo:bar:baz", "{", "}")`` returns ``true``. -// RegexSplit splits the input string by the occurrences of the given pattern. var RegexSplit = &Builtin{ - Name: "regex.split", + Name: "regex.split", + Description: "Splits the input string by the occurrences of the given pattern.", Decl: types.NewFunction( types.Args( - types.S, - types.S, + types.Named("pattern", types.S).Description("regular expression"), + types.Named("value", types.S).Description("string to match"), ), - types.NewArray(nil, types.S), + types.Named("output", types.NewArray(nil, types.S)).Description("the parts obtained by splitting `value`"), ), } // RegexFind takes two strings and a number, the pattern, the value and number of match values to // return, -1 means all match values. var RegexFind = &Builtin{ - Name: "regex.find_n", + Name: "regex.find_n", + Description: "Returns the specified number of matches when matching the input against the pattern.", Decl: types.NewFunction( types.Args( - types.S, - types.S, - types.N, + types.Named("pattern", types.S).Description("regular expression"), + types.Named("value", types.S).Description("string to match"), + types.Named("number", types.N).Description("number of matches to return, if `-1`, returns all matches"), ), - types.NewArray(nil, types.S), + types.Named("output", types.NewArray(nil, types.S)).Description("collected matches"), ), } // GlobsMatch takes two strings regexp-style strings and evaluates to true if their // intersection matches a non-empty set of non-empty strings. // Examples: -// - "a.a." and ".b.b" -> true. -// - "[a-z]*" and [0-9]+" -> not true. +// - "a.a." and ".b.b" -> true. +// - "[a-z]*" and [0-9]+" -> not true. var GlobsMatch = &Builtin{ Name: "regex.globs_match", + Description: `Checks if the intersection of two glob-style regular expressions matches a non-empty set of non-empty strings. +The set of regex symbols is limited for this builtin: only ` + "`.`, `*`, `+`, `[`, `-`, `]` and `\\` are treated as special symbols.", Decl: types.NewFunction( types.Args( - types.S, - types.S, + types.Named("glob1", types.S), + types.Named("glob2", types.S), ), - types.B, + types.Named("result", types.B), ), } /** * Strings */ +var stringsCat = category("strings") + +var AnyPrefixMatch = &Builtin{ + Name: "strings.any_prefix_match", + Description: "Returns true if any of the search strings begins with any of the base strings.", + Decl: types.NewFunction( + types.Args( + types.Named("search", types.NewAny( + types.S, + types.NewSet(types.S), + types.NewArray(nil, types.S), + )).Description("search string(s)"), + types.Named("base", types.NewAny( + types.S, + types.NewSet(types.S), + types.NewArray(nil, types.S), + )).Description("base string(s)"), + ), + types.Named("result", types.B).Description("result of the prefix check"), + ), + Categories: stringsCat, +} + +var AnySuffixMatch = &Builtin{ + Name: "strings.any_suffix_match", + Description: "Returns true if any of the search strings ends with any of the base strings.", + Decl: types.NewFunction( + types.Args( + types.Named("search", types.NewAny( + types.S, + types.NewSet(types.S), + types.NewArray(nil, types.S), + )).Description("search string(s)"), + types.Named("base", types.NewAny( + types.S, + types.NewSet(types.S), + types.NewArray(nil, types.S), + )).Description("base string(s)"), + ), + types.Named("result", types.B).Description("result of the suffix check"), + ), + Categories: stringsCat, +} -// Concat joins an array of strings with an input string. var Concat = &Builtin{ - Name: "concat", + Name: "concat", + Description: "Joins a set or array of strings with a delimiter.", Decl: types.NewFunction( types.Args( - types.S, - types.NewAny( + types.Named("delimiter", types.S), + types.Named("collection", types.NewAny( types.NewSet(types.S), types.NewArray(nil, types.S), - ), + )).Description("strings to join"), ), - types.S, + types.Named("output", types.S), ), + Categories: stringsCat, } -// FormatInt returns the string representation of the number in the given base after converting it to an integer value. var FormatInt = &Builtin{ - Name: "format_int", + Name: "format_int", + Description: "Returns the string representation of the number in the given base after rounding it down to an integer value.", Decl: types.NewFunction( types.Args( - types.N, - types.N, + types.Named("number", types.N).Description("number to format"), + types.Named("base", types.N).Description("base of number representation to use"), ), - types.S, + types.Named("output", types.S).Description("formatted number"), ), + Categories: stringsCat, } -// IndexOf returns the index of a substring contained inside a string var IndexOf = &Builtin{ - Name: "indexof", + Name: "indexof", + Description: "Returns the index of a substring contained inside a string.", Decl: types.NewFunction( types.Args( - types.S, - types.S, + types.Named("haystack", types.S).Description("string to search in"), + types.Named("needle", types.S).Description("substring to look for"), + ), + types.Named("output", types.N).Description("index of first occurrence, `-1` if not found"), + ), + Categories: stringsCat, +} + +var IndexOfN = &Builtin{ + Name: "indexof_n", + Description: "Returns a list of all the indexes of a substring contained inside a string.", + Decl: types.NewFunction( + types.Args( + types.Named("haystack", types.S).Description("string to search in"), + types.Named("needle", types.S).Description("substring to look for"), ), - types.N, + types.Named("output", types.NewArray(nil, types.N)).Description("all indices at which `needle` occurs in `haystack`, may be empty"), ), + Categories: stringsCat, } -// Substring returns the portion of a string for a given start index and a length. -// If the length is less than zero, then substring returns the remainder of the string. var Substring = &Builtin{ - Name: "substring", + Name: "substring", + Description: "Returns the portion of a string for a given `offset` and a `length`. If `length < 0`, `output` is the remainder of the string.", Decl: types.NewFunction( types.Args( - types.S, - types.N, - types.N, + types.Named("value", types.S), + types.Named("offset", types.N).Description("offset, must be positive"), + types.Named("length", types.N).Description("length of the substring starting from `offset`"), ), - types.S, + types.Named("output", types.S).Description("substring of `value` from `offset`, of length `length`"), ), + Categories: stringsCat, } -// Contains returns true if the search string is included in the base string var Contains = &Builtin{ - Name: "contains", + Name: "contains", + Description: "Returns `true` if the search string is included in the base string", Decl: types.NewFunction( types.Args( - types.S, - types.S, + types.Named("haystack", types.S).Description("string to search in"), + types.Named("needle", types.S).Description("substring to look for"), ), - types.B, + types.Named("result", types.B).Description("result of the containment check"), ), + Categories: stringsCat, } -// StartsWith returns true if the search string begins with the base string var StartsWith = &Builtin{ - Name: "startswith", + Name: "startswith", + Description: "Returns true if the search string begins with the base string.", Decl: types.NewFunction( types.Args( - types.S, - types.S, + types.Named("search", types.S).Description("search string"), + types.Named("base", types.S).Description("base string"), ), - types.B, + types.Named("result", types.B).Description("result of the prefix check"), ), + Categories: stringsCat, } -// EndsWith returns true if the search string begins with the base string var EndsWith = &Builtin{ - Name: "endswith", + Name: "endswith", + Description: "Returns true if the search string ends with the base string.", Decl: types.NewFunction( types.Args( - types.S, - types.S, + types.Named("search", types.S).Description("search string"), + types.Named("base", types.S).Description("base string"), ), - types.B, + types.Named("result", types.B).Description("result of the suffix check"), ), + Categories: stringsCat, } -// Lower returns the input string but with all characters in lower-case var Lower = &Builtin{ - Name: "lower", + Name: "lower", + Description: "Returns the input string but with all characters in lower-case.", Decl: types.NewFunction( - types.Args(types.S), - types.S, + types.Args( + types.Named("x", types.S).Description("string that is converted to lower-case"), + ), + types.Named("y", types.S).Description("lower-case of x"), ), + Categories: stringsCat, } -// Upper returns the input string but with all characters in upper-case var Upper = &Builtin{ - Name: "upper", + Name: "upper", + Description: "Returns the input string but with all characters in upper-case.", Decl: types.NewFunction( - types.Args(types.S), - types.S, + types.Args( + types.Named("x", types.S).Description("string that is converted to upper-case"), + ), + types.Named("y", types.S).Description("upper-case of x"), ), + Categories: stringsCat, } -// Split returns an array containing elements of the input string split on a delimiter. var Split = &Builtin{ - Name: "split", + Name: "split", + Description: "Split returns an array containing elements of the input string split on a delimiter.", Decl: types.NewFunction( types.Args( - types.S, - types.S, + types.Named("x", types.S).Description("string that is split"), + types.Named("delimiter", types.S).Description("delimiter used for splitting"), ), - types.NewArray(nil, types.S), + types.Named("ys", types.NewArray(nil, types.S)).Description("splitted parts"), ), + Categories: stringsCat, } -// Replace returns the given string with all instances of the second argument replaced -// by the third. var Replace = &Builtin{ - Name: "replace", + Name: "replace", + Description: "Replace replaces all instances of a sub-string.", Decl: types.NewFunction( types.Args( - types.S, - types.S, - types.S, + types.Named("x", types.S).Description("string being processed"), + types.Named("old", types.S).Description("substring to replace"), + types.Named("new", types.S).Description("string to replace `old` with"), ), - types.S, + types.Named("y", types.S).Description("string with replaced substrings"), ), + Categories: stringsCat, } -// ReplaceN replaces a string from a list of old, new string pairs. -// Replacements are performed in the order they appear in the target string, without overlapping matches. -// The old string comparisons are done in argument order. var ReplaceN = &Builtin{ Name: "strings.replace_n", + Description: `Replaces a string from a list of old, new string pairs. +Replacements are performed in the order they appear in the target string, without overlapping matches. +The old string comparisons are done in argument order.`, Decl: types.NewFunction( types.Args( - types.NewObject( + types.Named("patterns", types.NewObject( nil, types.NewDynamicProperty( types.S, types.S)), - types.S, + ).Description("replacement pairs"), + types.Named("value", types.S).Description("string to replace substring matches in"), ), - types.S, + types.Named("output", types.S), + ), +} + +var RegexReplace = &Builtin{ + Name: "regex.replace", + Description: `Find and replaces the text using the regular expression pattern.`, + Decl: types.NewFunction( + types.Args( + types.Named("s", types.S).Description("string being processed"), + types.Named("pattern", types.S).Description("regex pattern to be applied"), + types.Named("value", types.S).Description("regex value"), + ), + types.Named("output", types.S), ), } -// Trim returns the given string with all leading or trailing instances of the second -// argument removed. var Trim = &Builtin{ - Name: "trim", + Name: "trim", + Description: "Returns `value` with all leading or trailing instances of the `cutset` characters removed.", Decl: types.NewFunction( types.Args( - types.S, - types.S, + types.Named("value", types.S).Description("string to trim"), + types.Named("cutset", types.S).Description("string of characters that are cut off"), ), - types.S, + types.Named("output", types.S).Description("string trimmed of `cutset` characters"), ), + Categories: stringsCat, } -// TrimLeft returns the given string with all leading instances of second argument removed. var TrimLeft = &Builtin{ - Name: "trim_left", + Name: "trim_left", + Description: "Returns `value` with all leading instances of the `cutset` chartacters removed.", Decl: types.NewFunction( types.Args( - types.S, - types.S, + types.Named("value", types.S).Description("string to trim"), + types.Named("cutset", types.S).Description("string of characters that are cut off on the left"), ), - types.S, + types.Named("output", types.S).Description("string left-trimmed of `cutset` characters"), ), + Categories: stringsCat, } -// TrimPrefix returns the given string without the second argument prefix string. -// If the given string doesn't start with prefix, it is returned unchanged. var TrimPrefix = &Builtin{ - Name: "trim_prefix", + Name: "trim_prefix", + Description: "Returns `value` without the prefix. If `value` doesn't start with `prefix`, it is returned unchanged.", Decl: types.NewFunction( types.Args( - types.S, - types.S, + types.Named("value", types.S).Description("string to trim"), + types.Named("prefix", types.S).Description("prefix to cut off"), ), - types.S, + types.Named("output", types.S).Description("string with `prefix` cut off"), ), + Categories: stringsCat, } -// TrimRight returns the given string with all trailing instances of second argument removed. var TrimRight = &Builtin{ - Name: "trim_right", + Name: "trim_right", + Description: "Returns `value` with all trailing instances of the `cutset` chartacters removed.", Decl: types.NewFunction( types.Args( - types.S, - types.S, + types.Named("value", types.S).Description("string to trim"), + types.Named("cutset", types.S).Description("string of characters that are cut off on the right"), ), - types.S, + types.Named("output", types.S).Description("string right-trimmed of `cutset` characters"), ), + Categories: stringsCat, } -// TrimSuffix returns the given string without the second argument suffix string. -// If the given string doesn't end with suffix, it is returned unchanged. var TrimSuffix = &Builtin{ - Name: "trim_suffix", + Name: "trim_suffix", + Description: "Returns `value` without the suffix. If `value` doesn't end with `suffix`, it is returned unchanged.", Decl: types.NewFunction( types.Args( - types.S, - types.S, + types.Named("value", types.S).Description("string to trim"), + types.Named("suffix", types.S).Description("suffix to cut off"), ), - types.S, + types.Named("output", types.S).Description("string with `suffix` cut off"), ), + Categories: stringsCat, } -// TrimSpace return the given string with all leading and trailing white space removed. var TrimSpace = &Builtin{ - Name: "trim_space", + Name: "trim_space", + Description: "Return the given string with all leading and trailing white space removed.", Decl: types.NewFunction( types.Args( - types.S, + types.Named("value", types.S).Description("string to trim"), ), - types.S, + types.Named("output", types.S).Description("string leading and trailing white space cut off"), ), + Categories: stringsCat, } -// Sprintf returns the given string, formatted. var Sprintf = &Builtin{ - Name: "sprintf", + Name: "sprintf", + Description: "Returns the given string, formatted.", Decl: types.NewFunction( types.Args( - types.S, - types.NewArray(nil, types.A), + types.Named("format", types.S).Description("string with formatting verbs"), + types.Named("values", types.NewArray(nil, types.A)).Description("arguments to format into formatting verbs"), ), - types.S, + types.Named("output", types.S).Description("`format` formatted by the values in `values`"), ), + Categories: stringsCat, } -// StringReverse returns the given string, reversed. var StringReverse = &Builtin{ - Name: "strings.reverse", + Name: "strings.reverse", + Description: "Reverses a given string.", Decl: types.NewFunction( types.Args( - types.S, + types.Named("x", types.S), ), - types.S, + types.Named("y", types.S), ), + Categories: stringsCat, } /** @@ -1109,26 +1306,30 @@ var StringReverse = &Builtin{ */ // RandIntn returns a random number 0 - n +// Marked non-deterministic because it relies on RNG internally. var RandIntn = &Builtin{ - Name: "rand.intn", + Name: "rand.intn", + Description: "Returns a random integer between `0` and `n` (`n` exlusive). If `n` is `0`, then `y` is always `0`. For any given argument pair (`str`, `n`), the output will be consistent throughout a query evaluation.", Decl: types.NewFunction( types.Args( - types.S, - types.N, + types.Named("str", types.S), + types.Named("n", types.N), ), - types.N, + types.Named("y", types.N).Description("random integer in the range `[0, abs(n))`"), ), + Categories: number, + Nondeterministic: true, } -// NumbersRange returns an array of numbers in the given inclusive range. var NumbersRange = &Builtin{ - Name: "numbers.range", + Name: "numbers.range", + Description: "Returns an array of numbers in the given (inclusive) range. If `a==b`, then `range == [a]`; if `a > b`, then `range` is in descending order.", Decl: types.NewFunction( types.Args( - types.N, - types.N, + types.Named("a", types.N), + types.Named("b", types.N), ), - types.NewArray(nil, types.N), + types.Named("range", types.NewArray(nil, types.N)).Description("the range between `a` and `b`"), ), } @@ -1136,15 +1337,33 @@ var NumbersRange = &Builtin{ * Units */ -// UnitsParseBytes converts strings like 10GB, 5K, 4mb, and the like into an -// integer number of bytes. +var UnitsParse = &Builtin{ + Name: "units.parse", + Description: `Converts strings like "10G", "5K", "4M", "1500m" and the like into a number. +This number can be a non-integer, such as 1.5, 0.22, etc. Supports standard metric decimal and +binary SI units (e.g., K, Ki, M, Mi, G, Gi etc.) m, K, M, G, T, P, and E are treated as decimal +units and Ki, Mi, Gi, Ti, Pi, and Ei are treated as binary units. + +Note that 'm' and 'M' are case-sensitive, to allow distinguishing between "milli" and "mega" units respectively. Other units are case-insensitive.`, + Decl: types.NewFunction( + types.Args( + types.Named("x", types.S).Description("the unit to parse"), + ), + types.Named("y", types.N).Description("the parsed number"), + ), +} + var UnitsParseBytes = &Builtin{ Name: "units.parse_bytes", + Description: `Converts strings like "10GB", "5K", "4mb" into an integer number of bytes. +Supports standard byte units (e.g., KB, KiB, etc.) KB, MB, GB, and TB are treated as decimal +units and KiB, MiB, GiB, and TiB are treated as binary units. The bytes symbol (b/B) in the +unit is optional and omitting it wil give the same result (e.g. Mi and MiB).`, Decl: types.NewFunction( types.Args( - types.S, + types.Named("x", types.S).Description("the byte unit to parse"), ), - types.N, + types.Named("y", types.N).Description("the parsed number"), ), } @@ -1154,55 +1373,37 @@ var UnitsParseBytes = &Builtin{ */ // UUIDRFC4122 returns a version 4 UUID string. +// Marked non-deterministic because it relies on RNG internally. var UUIDRFC4122 = &Builtin{ - Name: "uuid.rfc4122", + Name: "uuid.rfc4122", + Description: "Returns a new UUIDv4.", Decl: types.NewFunction( - types.Args(types.S), - types.S, + types.Args( + types.Named("k", types.S), + ), + types.Named("output", types.S).Description("a version 4 UUID; for any given `k`, the output will be consistent throughout a query evaluation"), ), + Nondeterministic: true, } /** * JSON */ -// JSONMarshal serializes the input term. -var JSONMarshal = &Builtin{ - Name: "json.marshal", - Decl: types.NewFunction( - types.Args(types.A), - types.S, - ), -} - -// JSONUnmarshal deserializes the input string. -var JSONUnmarshal = &Builtin{ - Name: "json.unmarshal", - Decl: types.NewFunction( - types.Args(types.S), - types.A, - ), -} - -// JSONIsValid verifies the input string is a valid JSON document. -var JSONIsValid = &Builtin{ - Name: "json.is_valid", - Decl: types.NewFunction( - types.Args(types.S), - types.B, - ), -} +var objectCat = category("object") -// JSONFilter filters the JSON object var JSONFilter = &Builtin{ Name: "json.filter", + Description: "Filters the object. " + + "For example: `json.filter({\"a\": {\"b\": \"x\", \"c\": \"y\"}}, [\"a/b\"])` will result in `{\"a\": {\"b\": \"x\"}}`). " + + "Paths are not filtered in-order and are deduplicated before being evaluated.", Decl: types.NewFunction( types.Args( - types.NewObject( + types.Named("object", types.NewObject( nil, types.NewDynamicProperty(types.A, types.A), - ), - types.NewAny( + )), + types.Named("paths", types.NewAny( types.NewArray( nil, types.NewAny( @@ -1222,22 +1423,25 @@ var JSONFilter = &Builtin{ ), ), ), - ), + )).Description("JSON string paths"), ), - types.A, + types.Named("filtered", types.A).Description("remaining data from `object` with only keys specified in `paths`"), ), + Categories: objectCat, } -// JSONRemove removes paths in the JSON object var JSONRemove = &Builtin{ Name: "json.remove", + Description: "Removes paths from an object. " + + "For example: `json.remove({\"a\": {\"b\": \"x\", \"c\": \"y\"}}, [\"a/b\"])` will result in `{\"a\": {\"c\": \"y\"}}`. " + + "Paths are not removed in-order and are deduplicated before being evaluated.", Decl: types.NewFunction( types.Args( - types.NewObject( + types.Named("object", types.NewObject( nil, types.NewDynamicProperty(types.A, types.A), - ), - types.NewAny( + )), + types.Named("paths", types.NewAny( types.NewArray( nil, types.NewAny( @@ -1257,19 +1461,21 @@ var JSONRemove = &Builtin{ ), ), ), - ), + )).Description("JSON string paths"), ), - types.A, + types.Named("output", types.A).Description("result of removing all keys specified in `paths`"), ), + Categories: objectCat, } -// JSONPatch patches a JSON object according to RFC6902 var JSONPatch = &Builtin{ Name: "json.patch", + Description: "Patches an object according to RFC6902. " + + "For example: `json.patch({\"a\": {\"foo\": 1}}, [{\"op\": \"add\", \"path\": \"/a/bar\", \"value\": 2}])` results in `{\"a\": {\"foo\": 1, \"bar\": 2}`. The patches are applied atomically: if any of them fails, the result will be undefined.", Decl: types.NewFunction( types.Args( - types.A, - types.NewArray( + types.Named("object", types.A), // TODO(sr): types.A? + types.Named("patches", types.NewArray( nil, types.NewObject( []*types.StaticProperty{ @@ -1278,547 +1484,718 @@ var JSONPatch = &Builtin{ }, types.NewDynamicProperty(types.A, types.A), ), - ), + )), ), - types.A, + types.Named("output", types.A).Description("result obtained after consecutively applying all patch operations in `patches`"), ), + Categories: objectCat, } -// ObjectGet returns takes an object and returns a value under its key if -// present, otherwise it returns the default. -var ObjectGet = &Builtin{ - Name: "object.get", +var ObjectSubset = &Builtin{ + Name: "object.subset", + Description: "Determines if an object `sub` is a subset of another object `super`." + + "Object `sub` is a subset of object `super` if and only if every key in `sub` is also in `super`, " + + "**and** for all keys which `sub` and `super` share, they have the same value. " + + "This function works with objects, sets, arrays and a set of array and set." + + "If both arguments are objects, then the operation is recursive, e.g. " + + "`{\"c\": {\"x\": {10, 15, 20}}` is a subset of `{\"a\": \"b\", \"c\": {\"x\": {10, 15, 20, 25}, \"y\": \"z\"}`. " + + "If both arguments are sets, then this function checks if every element of `sub` is a member of `super`, " + + "but does not attempt to recurse. If both arguments are arrays, " + + "then this function checks if `sub` appears contiguously in order within `super`, " + + "and also does not attempt to recurse. If `super` is array and `sub` is set, " + + "then this function checks if `super` contains every element of `sub` with no consideration of ordering, " + + "and also does not attempt to recurse.", Decl: types.NewFunction( types.Args( - types.NewObject(nil, types.NewDynamicProperty(types.A, types.A)), - types.A, - types.A, + types.Named("super", types.NewAny(types.NewObject( + nil, + types.NewDynamicProperty(types.A, types.A), + ), + types.NewSet(types.A), + types.NewArray(nil, types.A), + )).Description("object to test if sub is a subset of"), + types.Named("sub", types.NewAny(types.NewObject( + nil, + types.NewDynamicProperty(types.A, types.A), + ), + types.NewSet(types.A), + types.NewArray(nil, types.A), + )).Description("object to test if super is a superset of"), ), - types.A, + types.Named("result", types.A).Description("`true` if `sub` is a subset of `super`"), ), } -// ObjectUnion creates a new object that is the asymmetric union of two objects var ObjectUnion = &Builtin{ Name: "object.union", + Description: "Creates a new object of the asymmetric union of two objects. " + + "For example: `object.union({\"a\": 1, \"b\": 2, \"c\": {\"d\": 3}}, {\"a\": 7, \"c\": {\"d\": 4, \"e\": 5}})` will result in `{\"a\": 7, \"b\": 2, \"c\": {\"d\": 4, \"e\": 5}}`.", Decl: types.NewFunction( types.Args( - types.NewObject( + types.Named("a", types.NewObject( nil, types.NewDynamicProperty(types.A, types.A), - ), - types.NewObject( + )), + types.Named("b", types.NewObject( nil, types.NewDynamicProperty(types.A, types.A), - ), + )), + ), + types.Named("output", types.A).Description("a new object which is the result of an asymmetric recursive union of two objects where conflicts are resolved by choosing the key from the right-hand object `b`"), + ), // TODO(sr): types.A? ^^^^^^^ (also below) +} + +var ObjectUnionN = &Builtin{ + Name: "object.union_n", + Description: "Creates a new object that is the asymmetric union of all objects merged from left to right. " + + "For example: `object.union_n([{\"a\": 1}, {\"b\": 2}, {\"a\": 3}])` will result in `{\"b\": 2, \"a\": 3}`.", + Decl: types.NewFunction( + types.Args( + types.Named("objects", types.NewArray( + nil, + types.NewObject(nil, types.NewDynamicProperty(types.A, types.A)), + )), ), - types.A, + types.Named("output", types.A).Description("asymmetric recursive union of all objects in `objects`, merged from left to right, where conflicts are resolved by choosing the key from the right-hand object"), ), } -// ObjectRemove Removes specified keys from an object var ObjectRemove = &Builtin{ - Name: "object.remove", + Name: "object.remove", + Description: "Removes specified keys from an object.", Decl: types.NewFunction( types.Args( - types.NewObject( + types.Named("object", types.NewObject( nil, types.NewDynamicProperty(types.A, types.A), - ), - types.NewAny( + )).Description("object to remove keys from"), + types.Named("keys", types.NewAny( types.NewArray(nil, types.A), types.NewSet(types.A), types.NewObject(nil, types.NewDynamicProperty(types.A, types.A)), - ), + )).Description("keys to remove from x"), ), - types.A, + types.Named("output", types.A).Description("result of removing the specified `keys` from `object`"), ), } -// ObjectFilter filters the object by keeping only specified keys var ObjectFilter = &Builtin{ Name: "object.filter", + Description: "Filters the object by keeping only specified keys. " + + "For example: `object.filter({\"a\": {\"b\": \"x\", \"c\": \"y\"}, \"d\": \"z\"}, [\"a\"])` will result in `{\"a\": {\"b\": \"x\", \"c\": \"y\"}}`).", Decl: types.NewFunction( types.Args( - types.NewObject( + types.Named("object", types.NewObject( nil, types.NewDynamicProperty(types.A, types.A), - ), - types.NewAny( + )).Description("object to filter keys"), + types.Named("keys", types.NewAny( types.NewArray(nil, types.A), types.NewSet(types.A), types.NewObject(nil, types.NewDynamicProperty(types.A, types.A)), - ), + )), + ), + types.Named("filtered", types.A).Description("remaining data from `object` with only keys specified in `keys`"), + ), +} + +var ObjectGet = &Builtin{ + Name: "object.get", + Description: "Returns value of an object's key if present, otherwise a default. " + + "If the supplied `key` is an `array`, then `object.get` will search through a nested object or array using each key in turn. " + + "For example: `object.get({\"a\": [{ \"b\": true }]}, [\"a\", 0, \"b\"], false)` results in `true`.", + Decl: types.NewFunction( + types.Args( + types.Named("object", types.NewObject(nil, types.NewDynamicProperty(types.A, types.A))).Description("object to get `key` from"), + types.Named("key", types.A).Description("key to lookup in `object`"), + types.Named("default", types.A).Description("default to use when lookup fails"), + ), + types.Named("value", types.A).Description("`object[key]` if present, otherwise `default`"), + ), +} + +/* + * Encoding + */ +var encoding = category("encoding") + +var JSONMarshal = &Builtin{ + Name: "json.marshal", + Description: "Serializes the input term to JSON.", + Decl: types.NewFunction( + types.Args( + types.Named("x", types.A).Description("the term to serialize"), + ), + types.Named("y", types.S).Description("the JSON string representation of `x`"), + ), + Categories: encoding, +} + +var JSONUnmarshal = &Builtin{ + Name: "json.unmarshal", + Description: "Deserializes the input string.", + Decl: types.NewFunction( + types.Args( + types.Named("x", types.S).Description("a JSON string"), + ), + types.Named("y", types.A).Description("the term deseralized from `x`"), + ), + Categories: encoding, +} + +var JSONIsValid = &Builtin{ + Name: "json.is_valid", + Description: "Verifies the input string is a valid JSON document.", + Decl: types.NewFunction( + types.Args( + types.Named("x", types.S).Description("a JSON string"), ), - types.A, + types.Named("result", types.B).Description("`true` if `x` is valid JSON, `false` otherwise"), ), + Categories: encoding, } -// Base64Encode serializes the input string into base64 encoding. var Base64Encode = &Builtin{ - Name: "base64.encode", + Name: "base64.encode", + Description: "Serializes the input string into base64 encoding.", Decl: types.NewFunction( - types.Args(types.S), - types.S, + types.Args( + types.Named("x", types.S), + ), + types.Named("y", types.S).Description("base64 serialization of `x`"), ), + Categories: encoding, } -// Base64Decode deserializes the base64 encoded input string. var Base64Decode = &Builtin{ - Name: "base64.decode", + Name: "base64.decode", + Description: "Deserializes the base64 encoded input string.", Decl: types.NewFunction( - types.Args(types.S), - types.S, + types.Args( + types.Named("x", types.S), + ), + types.Named("y", types.S).Description("base64 deserialization of `x`"), ), + Categories: encoding, } -// Base64IsValid verifies the input string is base64 encoded. var Base64IsValid = &Builtin{ - Name: "base64.is_valid", + Name: "base64.is_valid", + Description: "Verifies the input string is base64 encoded.", Decl: types.NewFunction( - types.Args(types.S), - types.B, + types.Args( + types.Named("x", types.S), + ), + types.Named("result", types.B).Description("`true` if `x` is valid base64 encoded value, `false` otherwise"), ), + Categories: encoding, } -// Base64UrlEncode serializes the input string into base64url encoding. var Base64UrlEncode = &Builtin{ - Name: "base64url.encode", + Name: "base64url.encode", + Description: "Serializes the input string into base64url encoding.", Decl: types.NewFunction( - types.Args(types.S), - types.S, + types.Args( + types.Named("x", types.S), + ), + types.Named("y", types.S).Description("base64url serialization of `x`"), ), + Categories: encoding, } -// Base64UrlEncodeNoPad serializes the input string into base64url encoding without padding. var Base64UrlEncodeNoPad = &Builtin{ - Name: "base64url.encode_no_pad", + Name: "base64url.encode_no_pad", + Description: "Serializes the input string into base64url encoding without padding.", Decl: types.NewFunction( - types.Args(types.S), - types.S, + types.Args( + types.Named("x", types.S), + ), + types.Named("y", types.S).Description("base64url serialization of `x`"), ), + Categories: encoding, } -// Base64UrlDecode deserializes the base64url encoded input string. var Base64UrlDecode = &Builtin{ - Name: "base64url.decode", + Name: "base64url.decode", + Description: "Deserializes the base64url encoded input string.", Decl: types.NewFunction( - types.Args(types.S), - types.S, + types.Args( + types.Named("x", types.S), + ), + types.Named("y", types.S).Description("base64url deserialization of `x`"), ), + Categories: encoding, } -// URLQueryDecode decodes a URL encoded input string. var URLQueryDecode = &Builtin{ - Name: "urlquery.decode", + Name: "urlquery.decode", + Description: "Decodes a URL-encoded input string.", Decl: types.NewFunction( - types.Args(types.S), - types.S, + types.Args( + types.Named("x", types.S), + ), + types.Named("y", types.S).Description("URL-encoding deserialization of `x`"), ), + Categories: encoding, } -// URLQueryEncode encodes the input string into a URL encoded string. var URLQueryEncode = &Builtin{ - Name: "urlquery.encode", + Name: "urlquery.encode", + Description: "Encodes the input string into a URL-encoded string.", Decl: types.NewFunction( - types.Args(types.S), - types.S, + types.Args( + types.Named("x", types.S), + ), + types.Named("y", types.S).Description("URL-encoding serialization of `x`"), ), + Categories: encoding, } -// URLQueryEncodeObject encodes the given JSON into a URL encoded query string. var URLQueryEncodeObject = &Builtin{ - Name: "urlquery.encode_object", + Name: "urlquery.encode_object", + Description: "Encodes the given object into a URL encoded query string.", Decl: types.NewFunction( types.Args( - types.NewObject( + types.Named("object", types.NewObject( nil, types.NewDynamicProperty( types.S, types.NewAny( types.S, types.NewArray(nil, types.S), - types.NewSet(types.S))))), - types.S, + types.NewSet(types.S)))))), + types.Named("y", types.S).Description("the URL-encoded serialization of `object`"), ), + Categories: encoding, } -// URLQueryDecodeObject decodes the given URL query string into an object. var URLQueryDecodeObject = &Builtin{ - Name: "urlquery.decode_object", + Name: "urlquery.decode_object", + Description: "Decodes the given URL query string into an object.", Decl: types.NewFunction( - types.Args(types.S), - types.NewObject(nil, types.NewDynamicProperty( + types.Args( + types.Named("x", types.S).Description("the query string"), + ), + types.Named("object", types.NewObject(nil, types.NewDynamicProperty( types.S, - types.NewArray(nil, types.S))), + types.NewArray(nil, types.S)))).Description("the resulting object"), ), + Categories: encoding, } -// YAMLMarshal serializes the input term. var YAMLMarshal = &Builtin{ - Name: "yaml.marshal", + Name: "yaml.marshal", + Description: "Serializes the input term to YAML.", Decl: types.NewFunction( - types.Args(types.A), - types.S, + types.Args( + types.Named("x", types.A).Description("the term to serialize"), + ), + types.Named("y", types.S).Description("the YAML string representation of `x`"), ), + Categories: encoding, } -// YAMLUnmarshal deserializes the input string. var YAMLUnmarshal = &Builtin{ - Name: "yaml.unmarshal", + Name: "yaml.unmarshal", + Description: "Deserializes the input string.", Decl: types.NewFunction( - types.Args(types.S), - types.A, + types.Args( + types.Named("x", types.S).Description("a YAML string"), + ), + types.Named("y", types.A).Description("the term deseralized from `x`"), ), + Categories: encoding, } // YAMLIsValid verifies the input string is a valid YAML document. var YAMLIsValid = &Builtin{ - Name: "yaml.is_valid", + Name: "yaml.is_valid", + Description: "Verifies the input string is a valid YAML document.", Decl: types.NewFunction( - types.Args(types.S), - types.B, + types.Args( + types.Named("x", types.S).Description("a YAML string"), + ), + types.Named("result", types.B).Description("`true` if `x` is valid YAML, `false` otherwise"), ), + Categories: encoding, } -// HexEncode serializes the input string into hex encoding. var HexEncode = &Builtin{ - Name: "hex.encode", + Name: "hex.encode", + Description: "Serializes the input string using hex-encoding.", Decl: types.NewFunction( - types.Args(types.S), - types.S, + types.Args( + types.Named("x", types.S), + ), + types.Named("y", types.S).Description("serialization of `x` using hex-encoding"), ), + Categories: encoding, } -// HexDecode deserializes the hex encoded input string. var HexDecode = &Builtin{ - Name: "hex.decode", + Name: "hex.decode", + Description: "Deserializes the hex-encoded input string.", Decl: types.NewFunction( - types.Args(types.S), - types.S, + types.Args( + types.Named("x", types.S).Description("a hex-encoded string"), + ), + types.Named("y", types.S).Description("deseralized from `x`"), ), + Categories: encoding, } /** * Tokens */ +var tokensCat = category("tokens") -// JWTDecode decodes a JSON Web Token and outputs it as an Object. var JWTDecode = &Builtin{ - Name: "io.jwt.decode", + Name: "io.jwt.decode", + Description: "Decodes a JSON Web Token and outputs it as an object.", Decl: types.NewFunction( - types.Args(types.S), - types.NewArray([]types.Type{ + types.Args( + types.Named("jwt", types.S).Description("JWT token to decode"), + ), + types.Named("output", types.NewArray([]types.Type{ types.NewObject(nil, types.NewDynamicProperty(types.A, types.A)), types.NewObject(nil, types.NewDynamicProperty(types.A, types.A)), types.S, - }, nil), + }, nil)).Description("`[header, payload, sig]`, where `header` and `payload` are objects; `sig` is the hexadecimal representation of the signature on the token."), ), + Categories: tokensCat, } -// JWTVerifyRS256 verifies if a RS256 JWT signature is valid or not. var JWTVerifyRS256 = &Builtin{ - Name: "io.jwt.verify_rs256", + Name: "io.jwt.verify_rs256", + Description: "Verifies if a RS256 JWT signature is valid.", Decl: types.NewFunction( types.Args( - types.S, - types.S, + types.Named("jwt", types.S).Description("JWT token whose signature is to be verified"), + types.Named("certificate", types.S).Description("PEM encoded certificate, PEM encoded public key, or the JWK key (set) used to verify the signature"), ), - types.B, + types.Named("result", types.B).Description("`true` if the signature is valid, `false` otherwise"), ), + Categories: tokensCat, } -// JWTVerifyRS384 verifies if a RS384 JWT signature is valid or not. var JWTVerifyRS384 = &Builtin{ - Name: "io.jwt.verify_rs384", + Name: "io.jwt.verify_rs384", + Description: "Verifies if a RS384 JWT signature is valid.", Decl: types.NewFunction( types.Args( - types.S, - types.S, + types.Named("jwt", types.S).Description("JWT token whose signature is to be verified"), + types.Named("certificate", types.S).Description("PEM encoded certificate, PEM encoded public key, or the JWK key (set) used to verify the signature"), ), - types.B, + types.Named("result", types.B).Description("`true` if the signature is valid, `false` otherwise"), ), + Categories: tokensCat, } -// JWTVerifyRS512 verifies if a RS512 JWT signature is valid or not. var JWTVerifyRS512 = &Builtin{ - Name: "io.jwt.verify_rs512", + Name: "io.jwt.verify_rs512", + Description: "Verifies if a RS512 JWT signature is valid.", Decl: types.NewFunction( types.Args( - types.S, - types.S, + types.Named("jwt", types.S).Description("JWT token whose signature is to be verified"), + types.Named("certificate", types.S).Description("PEM encoded certificate, PEM encoded public key, or the JWK key (set) used to verify the signature"), ), - types.B, + types.Named("result", types.B).Description("`true` if the signature is valid, `false` otherwise"), ), + Categories: tokensCat, } -// JWTVerifyPS256 verifies if a PS256 JWT signature is valid or not. var JWTVerifyPS256 = &Builtin{ - Name: "io.jwt.verify_ps256", + Name: "io.jwt.verify_ps256", + Description: "Verifies if a PS256 JWT signature is valid.", Decl: types.NewFunction( types.Args( - types.S, - types.S, + types.Named("jwt", types.S).Description("JWT token whose signature is to be verified"), + types.Named("certificate", types.S).Description("PEM encoded certificate, PEM encoded public key, or the JWK key (set) used to verify the signature"), ), - types.B, + types.Named("result", types.B).Description("`true` if the signature is valid, `false` otherwise"), ), + Categories: tokensCat, } -// JWTVerifyPS384 verifies if a PS384 JWT signature is valid or not. var JWTVerifyPS384 = &Builtin{ - Name: "io.jwt.verify_ps384", + Name: "io.jwt.verify_ps384", + Description: "Verifies if a PS384 JWT signature is valid.", Decl: types.NewFunction( types.Args( - types.S, - types.S, + types.Named("jwt", types.S).Description("JWT token whose signature is to be verified"), + types.Named("certificate", types.S).Description("PEM encoded certificate, PEM encoded public key, or the JWK key (set) used to verify the signature"), ), - types.B, + types.Named("result", types.B).Description("`true` if the signature is valid, `false` otherwise"), ), + Categories: tokensCat, } -// JWTVerifyPS512 verifies if a PS512 JWT signature is valid or not. var JWTVerifyPS512 = &Builtin{ - Name: "io.jwt.verify_ps512", + Name: "io.jwt.verify_ps512", + Description: "Verifies if a PS512 JWT signature is valid.", Decl: types.NewFunction( types.Args( - types.S, - types.S, + types.Named("jwt", types.S).Description("JWT token whose signature is to be verified"), + types.Named("certificate", types.S).Description("PEM encoded certificate, PEM encoded public key, or the JWK key (set) used to verify the signature"), ), - types.B, + types.Named("result", types.B).Description("`true` if the signature is valid, `false` otherwise"), ), + Categories: tokensCat, } -// JWTVerifyES256 verifies if a ES256 JWT signature is valid or not. var JWTVerifyES256 = &Builtin{ - Name: "io.jwt.verify_es256", + Name: "io.jwt.verify_es256", + Description: "Verifies if a ES256 JWT signature is valid.", Decl: types.NewFunction( types.Args( - types.S, - types.S, + types.Named("jwt", types.S).Description("JWT token whose signature is to be verified"), + types.Named("certificate", types.S).Description("PEM encoded certificate, PEM encoded public key, or the JWK key (set) used to verify the signature"), ), - types.B, + types.Named("result", types.B).Description("`true` if the signature is valid, `false` otherwise"), ), + Categories: tokensCat, } -// JWTVerifyES384 verifies if a ES384 JWT signature is valid or not. var JWTVerifyES384 = &Builtin{ - Name: "io.jwt.verify_es384", + Name: "io.jwt.verify_es384", + Description: "Verifies if a ES384 JWT signature is valid.", Decl: types.NewFunction( types.Args( - types.S, - types.S, + types.Named("jwt", types.S).Description("JWT token whose signature is to be verified"), + types.Named("certificate", types.S).Description("PEM encoded certificate, PEM encoded public key, or the JWK key (set) used to verify the signature"), ), - types.B, + types.Named("result", types.B).Description("`true` if the signature is valid, `false` otherwise"), ), + Categories: tokensCat, } -// JWTVerifyES512 verifies if a ES512 JWT signature is valid or not. var JWTVerifyES512 = &Builtin{ - Name: "io.jwt.verify_es512", + Name: "io.jwt.verify_es512", + Description: "Verifies if a ES512 JWT signature is valid.", Decl: types.NewFunction( types.Args( - types.S, - types.S, + types.Named("jwt", types.S).Description("JWT token whose signature is to be verified"), + types.Named("certificate", types.S).Description("PEM encoded certificate, PEM encoded public key, or the JWK key (set) used to verify the signature"), ), - types.B, + types.Named("result", types.B).Description("`true` if the signature is valid, `false` otherwise"), ), + Categories: tokensCat, } -// JWTVerifyHS256 verifies if a HS256 (secret) JWT signature is valid or not. var JWTVerifyHS256 = &Builtin{ - Name: "io.jwt.verify_hs256", + Name: "io.jwt.verify_hs256", + Description: "Verifies if a HS256 (secret) JWT signature is valid.", Decl: types.NewFunction( types.Args( - types.S, - types.S, + types.Named("jwt", types.S).Description("JWT token whose signature is to be verified"), + types.Named("secret", types.S).Description("plain text secret used to verify the signature"), ), - types.B, + types.Named("result", types.B).Description("`true` if the signature is valid, `false` otherwise"), ), + Categories: tokensCat, } -// JWTVerifyHS384 verifies if a HS384 (secret) JWT signature is valid or not. var JWTVerifyHS384 = &Builtin{ - Name: "io.jwt.verify_hs384", + Name: "io.jwt.verify_hs384", + Description: "Verifies if a HS384 (secret) JWT signature is valid.", Decl: types.NewFunction( types.Args( - types.S, - types.S, + types.Named("jwt", types.S).Description("JWT token whose signature is to be verified"), + types.Named("secret", types.S).Description("plain text secret used to verify the signature"), ), - types.B, + types.Named("result", types.B).Description("`true` if the signature is valid, `false` otherwise"), ), + Categories: tokensCat, } -// JWTVerifyHS512 verifies if a HS512 (secret) JWT signature is valid or not. var JWTVerifyHS512 = &Builtin{ - Name: "io.jwt.verify_hs512", + Name: "io.jwt.verify_hs512", + Description: "Verifies if a HS512 (secret) JWT signature is valid.", Decl: types.NewFunction( types.Args( - types.S, - types.S, + types.Named("jwt", types.S).Description("JWT token whose signature is to be verified"), + types.Named("secret", types.S).Description("plain text secret used to verify the signature"), ), - types.B, + types.Named("result", types.B).Description("`true` if the signature is valid, `false` otherwise"), ), + Categories: tokensCat, } -// JWTDecodeVerify verifies a JWT signature under parameterized constraints and decodes the claims if it is valid. +// Marked non-deterministic because it relies on time internally. var JWTDecodeVerify = &Builtin{ Name: "io.jwt.decode_verify", + Description: `Verifies a JWT signature under parameterized constraints and decodes the claims if it is valid. +Supports the following algorithms: HS256, HS384, HS512, RS256, RS384, RS512, ES256, ES384, ES512, PS256, PS384 and PS512.`, Decl: types.NewFunction( types.Args( - types.S, - types.NewObject(nil, types.NewDynamicProperty(types.S, types.A)), + types.Named("jwt", types.S).Description("JWT token whose signature is to be verified and whose claims are to be checked"), + types.Named("constraints", types.NewObject(nil, types.NewDynamicProperty(types.S, types.A))).Description("claim verification constraints"), ), - types.NewArray([]types.Type{ + types.Named("output", types.NewArray([]types.Type{ types.B, types.NewObject(nil, types.NewDynamicProperty(types.A, types.A)), types.NewObject(nil, types.NewDynamicProperty(types.A, types.A)), - }, nil), + }, nil)).Description("`[valid, header, payload]`: if the input token is verified and meets the requirements of `constraints` then `valid` is `true`; `header` and `payload` are objects containing the JOSE header and the JWT claim set; otherwise, `valid` is `false`, `header` and `payload` are `{}`"), ), + Categories: tokensCat, + Nondeterministic: true, } -// JWTEncodeSignRaw encodes and optionally sign a JSON Web Token. -// Inputs are protected headers, payload, secret +var tokenSign = category("tokensign") + +// Marked non-deterministic because it relies on RNG internally. var JWTEncodeSignRaw = &Builtin{ - Name: "io.jwt.encode_sign_raw", + Name: "io.jwt.encode_sign_raw", + Description: "Encodes and optionally signs a JSON Web Token.", Decl: types.NewFunction( types.Args( - types.S, - types.S, - types.S, + types.Named("headers", types.S).Description("JWS Protected Header"), + types.Named("payload", types.S).Description("JWS Payload"), + types.Named("key", types.S).Description("JSON Web Key (RFC7517)"), ), - types.S, + types.Named("output", types.S).Description("signed JWT"), ), + Categories: tokenSign, + Nondeterministic: true, } -// JWTEncodeSign encodes and optionally sign a JSON Web Token. -// Inputs are protected headers, payload, secret +// Marked non-deterministic because it relies on RNG internally. var JWTEncodeSign = &Builtin{ - Name: "io.jwt.encode_sign", + Name: "io.jwt.encode_sign", + Description: "Encodes and optionally signs a JSON Web Token. Inputs are taken as objects, not encoded strings (see `io.jwt.encode_sign_raw`).", Decl: types.NewFunction( types.Args( - types.NewObject(nil, types.NewDynamicProperty(types.S, types.A)), - types.NewObject(nil, types.NewDynamicProperty(types.S, types.A)), - types.NewObject(nil, types.NewDynamicProperty(types.S, types.A)), + types.Named("headers", types.NewObject(nil, types.NewDynamicProperty(types.S, types.A))).Description("JWS Protected Header"), + types.Named("payload", types.NewObject(nil, types.NewDynamicProperty(types.S, types.A))).Description("JWS Payload"), + types.Named("key", types.NewObject(nil, types.NewDynamicProperty(types.S, types.A))).Description("JSON Web Key (RFC7517)"), ), - types.S, + types.Named("output", types.S).Description("signed JWT"), ), + Categories: tokenSign, + Nondeterministic: true, } /** * Time */ -// NowNanos returns the current time since epoch in nanoseconds. +// Marked non-deterministic because it relies on time directly. var NowNanos = &Builtin{ - Name: "time.now_ns", + Name: "time.now_ns", + Description: "Returns the current time since epoch in nanoseconds.", Decl: types.NewFunction( nil, - types.N, + types.Named("now", types.N).Description("nanoseconds since epoch"), ), + Nondeterministic: true, } -// ParseNanos returns the time in nanoseconds parsed from the string in the given format. var ParseNanos = &Builtin{ - Name: "time.parse_ns", + Name: "time.parse_ns", + Description: "Returns the time in nanoseconds parsed from the string in the given format. `undefined` if the result would be outside the valid time range that can fit within an `int64`.", Decl: types.NewFunction( types.Args( - types.S, - types.S, + types.Named("layout", types.S).Description("format used for parsing, see the [Go `time` package documentation](https://golang.org/pkg/time/#Parse) for more details"), + types.Named("value", types.S).Description("input to parse according to `layout`"), ), - types.N, + types.Named("ns", types.N).Description("`value` in nanoseconds since epoch"), ), } -// ParseRFC3339Nanos returns the time in nanoseconds parsed from the string in RFC3339 format. var ParseRFC3339Nanos = &Builtin{ - Name: "time.parse_rfc3339_ns", + Name: "time.parse_rfc3339_ns", + Description: "Returns the time in nanoseconds parsed from the string in RFC3339 format. `undefined` if the result would be outside the valid time range that can fit within an `int64`.", Decl: types.NewFunction( - types.Args(types.S), - types.N, + types.Args( + types.Named("value", types.S), + ), + types.Named("ns", types.N).Description("`value` in nanoseconds since epoch"), ), } -// ParseDurationNanos returns the duration in nanoseconds represented by a duration string. -// Duration string is similar to the Go time.ParseDuration string var ParseDurationNanos = &Builtin{ - Name: "time.parse_duration_ns", + Name: "time.parse_duration_ns", + Description: "Returns the duration in nanoseconds represented by a string.", Decl: types.NewFunction( - types.Args(types.S), - types.N, + types.Args( + types.Named("duration", types.S).Description("a duration like \"3m\"; seethe [Go `time` package documentation](https://golang.org/pkg/time/#ParseDuration) for more details"), + ), + types.Named("ns", types.N).Description("the `duration` in nanoseconds"), ), } -// Date returns the [year, month, day] for the nanoseconds since epoch. var Date = &Builtin{ - Name: "time.date", + Name: "time.date", + Description: "Returns the `[year, month, day]` for the nanoseconds since epoch.", Decl: types.NewFunction( types.Args( - types.NewAny( + types.Named("x", types.NewAny( types.N, types.NewArray([]types.Type{types.N, types.S}, nil), - ), + )).Description("a number representing the nanoseconds since the epoch (UTC); or a two-element array of the nanoseconds, and a timezone string"), ), - types.NewArray([]types.Type{types.N, types.N, types.N}, nil), + types.Named("date", types.NewArray([]types.Type{types.N, types.N, types.N}, nil)).Description("an array of `year`, `month` (1-12), and `day` (1-31)"), ), } -// Clock returns the [hour, minute, second] of the day for the nanoseconds since epoch. var Clock = &Builtin{ - Name: "time.clock", + Name: "time.clock", + Description: "Returns the `[hour, minute, second]` of the day for the nanoseconds since epoch.", Decl: types.NewFunction( types.Args( - types.NewAny( + types.Named("x", types.NewAny( types.N, types.NewArray([]types.Type{types.N, types.S}, nil), - ), + )).Description("a number representing the nanoseconds since the epoch (UTC); or a two-element array of the nanoseconds, and a timezone string"), ), - types.NewArray([]types.Type{types.N, types.N, types.N}, nil), + types.Named("output", types.NewArray([]types.Type{types.N, types.N, types.N}, nil)). + Description("the `hour`, `minute` (0-59), and `second` (0-59) representing the time of day for the nanoseconds since epoch in the supplied timezone (or UTC)"), ), } -// Weekday returns the day of the week (Monday, Tuesday, ...) for the nanoseconds since epoch. var Weekday = &Builtin{ - Name: "time.weekday", + Name: "time.weekday", + Description: "Returns the day of the week (Monday, Tuesday, ...) for the nanoseconds since epoch.", Decl: types.NewFunction( types.Args( - types.NewAny( + types.Named("x", types.NewAny( types.N, types.NewArray([]types.Type{types.N, types.S}, nil), - ), + )).Description("a number representing the nanoseconds since the epoch (UTC); or a two-element array of the nanoseconds, and a timezone string"), ), - types.S, + types.Named("day", types.S).Description("the weekday represented by `ns` nanoseconds since the epoch in the supplied timezone (or UTC)"), ), } -// AddDate returns the nanoseconds since epoch after adding years, months and days to nanoseconds. var AddDate = &Builtin{ - Name: "time.add_date", + Name: "time.add_date", + Description: "Returns the nanoseconds since epoch after adding years, months and days to nanoseconds. `undefined` if the result would be outside the valid time range that can fit within an `int64`.", Decl: types.NewFunction( types.Args( - types.N, - types.N, - types.N, - types.N, + types.Named("ns", types.N).Description("nanoseconds since the epoch"), + types.Named("years", types.N), + types.Named("months", types.N), + types.Named("days", types.N), ), - types.N, + types.Named("output", types.N).Description("nanoseconds since the epoch representing the input time, with years, months and days added"), ), } -// Diff returns the difference [years, months, days, hours, minutes, seconds] between two unix timestamps in nanoseconds var Diff = &Builtin{ - Name: "time.diff", + Name: "time.diff", + Description: "Returns the difference between two unix timestamps in nanoseconds (with optional timezone strings).", Decl: types.NewFunction( types.Args( - types.NewAny( + types.Named("ns1", types.NewAny( types.N, types.NewArray([]types.Type{types.N, types.S}, nil), - ), - types.NewAny( + )), + types.Named("ns2", types.NewAny( types.N, types.NewArray([]types.Type{types.N, types.S}, nil), - ), + )), ), - types.NewArray([]types.Type{types.N, types.N, types.N, types.N, types.N, types.N}, nil), + types.Named("output", types.NewArray([]types.Type{types.N, types.N, types.N, types.N, types.N, types.N}, nil)).Description("difference between `ns1` and `ns2` (in their supplied timezones, if supplied, or UTC) as array of numbers: `[years, months, days, hours, minutes, seconds]`"), ), } @@ -1826,161 +2203,169 @@ var Diff = &Builtin{ * Crypto. */ -// CryptoX509ParseCertificates returns one or more certificates from the given -// base64 encoded string containing DER encoded certificates that have been -// concatenated. var CryptoX509ParseCertificates = &Builtin{ - Name: "crypto.x509.parse_certificates", + Name: "crypto.x509.parse_certificates", + Description: "Returns one or more certificates from the given base64 encoded string containing DER encoded certificates that have been concatenated.", Decl: types.NewFunction( - types.Args(types.S), - types.NewArray(nil, types.NewObject(nil, types.NewDynamicProperty(types.S, types.A))), + types.Args( + types.Named("certs", types.S).Description("base64 encoded DER or PEM data containing one or more certificates or a PEM string of one or more certificates"), + ), + types.Named("output", types.NewArray(nil, types.NewObject(nil, types.NewDynamicProperty(types.S, types.A)))).Description("parsed X.509 certificates represented as objects"), ), } -// CryptoX509ParseAndVerifyCertificates returns one or more certificates from the given -// string containing PEM or base64 encoded DER certificates after verifying the supplied -// certificates form a complete certificate chain back to a trusted root. -// -// The first certificate is treated as the root and the last is treated as the leaf, -// with all others being treated as intermediates var CryptoX509ParseAndVerifyCertificates = &Builtin{ Name: "crypto.x509.parse_and_verify_certificates", + Description: `Returns one or more certificates from the given string containing PEM +or base64 encoded DER certificates after verifying the supplied certificates form a complete +certificate chain back to a trusted root. + +The first certificate is treated as the root and the last is treated as the leaf, +with all others being treated as intermediates.`, Decl: types.NewFunction( - types.Args(types.S), - types.NewArray([]types.Type{ + types.Args( + types.Named("certs", types.S).Description("base64 encoded DER or PEM data containing two or more certificates where the first is a root CA, the last is a leaf certificate, and all others are intermediate CAs"), + ), + types.Named("output", types.NewArray([]types.Type{ types.B, types.NewArray(nil, types.NewObject(nil, types.NewDynamicProperty(types.S, types.A))), - }, nil), + }, nil)).Description("array of `[valid, certs]`: if the input certificate chain could be verified then `valid` is `true` and `certs` is an array of X.509 certificates represented as objects; if the input certificate chain could not be verified then `valid` is `false` and `certs` is `[]`"), ), } -// CryptoX509ParseCertificateRequest returns a PKCS #10 certificate signing -// request from the given PEM-encoded PKCS#10 certificate signing request. var CryptoX509ParseCertificateRequest = &Builtin{ - Name: "crypto.x509.parse_certificate_request", + Name: "crypto.x509.parse_certificate_request", + Description: "Returns a PKCS #10 certificate signing request from the given PEM-encoded PKCS#10 certificate signing request.", Decl: types.NewFunction( - types.Args(types.S), - types.NewObject(nil, types.NewDynamicProperty(types.S, types.A)), + types.Args( + types.Named("csr", types.S).Description("base64 string containing either a PEM encoded or DER CSR or a string containing a PEM CSR"), + ), + types.Named("output", types.NewObject(nil, types.NewDynamicProperty(types.S, types.A))).Description("X.509 CSR represented as an object"), ), } -// CryptoX509ParseRSAPrivateKey returns a JWK for signing a JWT from the given -// PEM-encoded RSA private key. var CryptoX509ParseRSAPrivateKey = &Builtin{ - Name: "crypto.x509.parse_rsa_private_key", + Name: "crypto.x509.parse_rsa_private_key", + Description: "Returns a JWK for signing a JWT from the given PEM-encoded RSA private key.", Decl: types.NewFunction( - types.Args(types.S), - types.NewObject(nil, types.NewDynamicProperty(types.S, types.A)), + types.Args( + types.Named("pem", types.S).Description("base64 string containing a PEM encoded RSA private key"), + ), + types.Named("output", types.NewObject(nil, types.NewDynamicProperty(types.S, types.A))).Description("JWK as an object"), ), } -// CryptoMd5 returns a string representing the input string hashed with the md5 function var CryptoMd5 = &Builtin{ - Name: "crypto.md5", + Name: "crypto.md5", + Description: "Returns a string representing the input string hashed with the MD5 function", Decl: types.NewFunction( - types.Args(types.S), - types.S, + types.Args( + types.Named("x", types.S), + ), + types.Named("y", types.S).Description("MD5-hash of `x`"), ), } -// CryptoSha1 returns a string representing the input string hashed with the sha1 function var CryptoSha1 = &Builtin{ - Name: "crypto.sha1", + Name: "crypto.sha1", + Description: "Returns a string representing the input string hashed with the SHA1 function", Decl: types.NewFunction( - types.Args(types.S), - types.S, + types.Args( + types.Named("x", types.S), + ), + types.Named("y", types.S).Description("SHA1-hash of `x`"), ), } -// CryptoSha256 returns a string representing the input string hashed with the sha256 function var CryptoSha256 = &Builtin{ - Name: "crypto.sha256", + Name: "crypto.sha256", + Description: "Returns a string representing the input string hashed with the SHA256 function", Decl: types.NewFunction( - types.Args(types.S), - types.S, + types.Args( + types.Named("x", types.S), + ), + types.Named("y", types.S).Description("SHA256-hash of `x`"), ), } -// CryptoHmacMd5 returns a string representing the MD-5 HMAC of the input message using the input key -// Inputs are message, key var CryptoHmacMd5 = &Builtin{ - Name: "crypto.hmac.md5", + Name: "crypto.hmac.md5", + Description: "Returns a string representing the MD5 HMAC of the input message using the input key.", Decl: types.NewFunction( types.Args( - types.S, - types.S, + types.Named("x", types.S).Description("input string"), + types.Named("key", types.S).Description("key to use"), ), - types.S, + types.Named("y", types.S).Description("MD5-HMAC of `x`"), ), } -// CryptoHmacSha1 returns a string representing the SHA-1 HMAC of the input message using the input key -// Inputs are message, key var CryptoHmacSha1 = &Builtin{ - Name: "crypto.hmac.sha1", + Name: "crypto.hmac.sha1", + Description: "Returns a string representing the SHA1 HMAC of the input message using the input key.", Decl: types.NewFunction( types.Args( - types.S, - types.S, + types.Named("x", types.S).Description("input string"), + types.Named("key", types.S).Description("key to use"), ), - types.S, + types.Named("y", types.S).Description("SHA1-HMAC of `x`"), ), } -// CryptoHmacSha256 returns a string representing the SHA-256 HMAC of the input message using the input key -// Inputs are message, key var CryptoHmacSha256 = &Builtin{ - Name: "crypto.hmac.sha256", + Name: "crypto.hmac.sha256", + Description: "Returns a string representing the SHA256 HMAC of the input message using the input key.", Decl: types.NewFunction( types.Args( - types.S, - types.S, + types.Named("x", types.S).Description("input string"), + types.Named("key", types.S).Description("key to use"), ), - types.S, + types.Named("y", types.S).Description("SHA256-HMAC of `x`"), ), } -// CryptoHmacSha512 returns a string representing the SHA-512 HMAC of the input message using the input key -// Inputs are message, key var CryptoHmacSha512 = &Builtin{ - Name: "crypto.hmac.sha512", + Name: "crypto.hmac.sha512", + Description: "Returns a string representing the SHA512 HMAC of the input message using the input key.", Decl: types.NewFunction( types.Args( - types.S, - types.S, + types.Named("x", types.S).Description("input string"), + types.Named("key", types.S).Description("key to use"), ), - types.S, + types.Named("y", types.S).Description("SHA512-HMAC of `x`"), ), } /** * Graphs. */ +var graphs = category("graph") -// WalkBuiltin generates [path, value] tuples for all nested documents -// (recursively). var WalkBuiltin = &Builtin{ - Name: "walk", - Relation: true, + Name: "walk", + Relation: true, + Description: "Generates `[path, value]` tuples for all nested documents of `x` (recursively). Queries can use `walk` to traverse documents nested under `x`.", Decl: types.NewFunction( - types.Args(types.A), - types.NewArray( + types.Args( + types.Named("x", types.A), + ), + types.Named("output", types.NewArray( []types.Type{ types.NewArray(nil, types.A), types.A, }, nil, - ), + )).Description("pairs of `path` and `value`: `path` is an array representing the pointer to `value` in `x`"), ), + Categories: graphs, } -// ReachableBuiltin computes the set of reachable nodes in the graph from a set -// of starting nodes. var ReachableBuiltin = &Builtin{ - Name: "graph.reachable", + Name: "graph.reachable", + Description: "Computes the set of reachable nodes in the graph from a set of starting nodes.", Decl: types.NewFunction( types.Args( - types.NewObject( + types.Named("graph", types.NewObject( nil, types.NewDynamicProperty( types.A, @@ -1988,109 +2373,120 @@ var ReachableBuiltin = &Builtin{ types.NewSet(types.A), types.NewArray(nil, types.A)), )), - types.NewAny(types.NewSet(types.A), types.NewArray(nil, types.A)), + ).Description("object containing a set or array of neighboring vertices"), + types.Named("initial", types.NewAny(types.NewSet(types.A), types.NewArray(nil, types.A))).Description("set or array of root vertices"), ), - types.NewSet(types.A), + types.Named("output", types.NewSet(types.A)).Description("set of vertices reachable from the `initial` vertices in the directed `graph`"), ), } -/** - * Sorting - */ - -// Sort returns a sorted array. -var Sort = &Builtin{ - Name: "sort", +var ReachablePathsBuiltin = &Builtin{ + Name: "graph.reachable_paths", + Description: "Computes the set of reachable paths in the graph from a set of starting nodes.", Decl: types.NewFunction( types.Args( - types.NewAny( - types.NewArray(nil, types.A), - types.NewSet(types.A), - ), + types.Named("graph", types.NewObject( + nil, + types.NewDynamicProperty( + types.A, + types.NewAny( + types.NewSet(types.A), + types.NewArray(nil, types.A)), + )), + ).Description("object containing a set or array of root vertices"), + types.Named("initial", types.NewAny(types.NewSet(types.A), types.NewArray(nil, types.A))).Description("initial paths"), // TODO(sr): copied. is that correct? ), - types.NewArray(nil, types.A), + types.Named("output", types.NewSet(types.NewArray(nil, types.A))).Description("paths reachable from the `initial` vertices in the directed `graph`"), ), } /** * Type */ +var typesCat = category("types") -// IsNumber returns true if the input value is a number var IsNumber = &Builtin{ - Name: "is_number", + Name: "is_number", + Description: "Returns `true` if the input value is a number.", Decl: types.NewFunction( types.Args( - types.A, + types.Named("x", types.A), ), - types.B, + types.Named("result", types.B).Description("`true` if `x` is a number, `false` otherwise."), ), + Categories: typesCat, } -// IsString returns true if the input value is a string. var IsString = &Builtin{ - Name: "is_string", + Name: "is_string", + Description: "Returns `true` if the input value is a string.", Decl: types.NewFunction( types.Args( - types.A, + types.Named("x", types.A), ), - types.B, + types.Named("result", types.B).Description("`true` if `x` is a string, `false` otherwise."), ), + Categories: typesCat, } -// IsBoolean returns true if the input value is a boolean. var IsBoolean = &Builtin{ - Name: "is_boolean", + Name: "is_boolean", + Description: "Returns `true` if the input value is a boolean.", Decl: types.NewFunction( types.Args( - types.A, + types.Named("x", types.A), ), - types.B, + types.Named("result", types.B).Description("`true` if `x` is an boolean, `false` otherwise."), ), + Categories: typesCat, } -// IsArray returns true if the input value is an array. var IsArray = &Builtin{ - Name: "is_array", + Name: "is_array", + Description: "Returns `true` if the input value is an array.", Decl: types.NewFunction( types.Args( - types.A, + types.Named("x", types.A), ), - types.B, + types.Named("result", types.B).Description("`true` if `x` is an array, `false` otherwise."), ), + Categories: typesCat, } -// IsSet returns true if the input value is a set. var IsSet = &Builtin{ - Name: "is_set", + Name: "is_set", + Description: "Returns `true` if the input value is a set.", Decl: types.NewFunction( types.Args( - types.A, + types.Named("x", types.A), ), - types.B, + types.Named("result", types.B).Description("`true` if `x` is a set, `false` otherwise."), ), + Categories: typesCat, } -// IsObject returns true if the input value is an object. var IsObject = &Builtin{ - Name: "is_object", + Name: "is_object", + Description: "Returns true if the input value is an object", Decl: types.NewFunction( types.Args( - types.A, + types.Named("x", types.A), ), - types.B, + types.Named("result", types.B).Description("`true` if `x` is an object, `false` otherwise."), ), + Categories: typesCat, } -// IsNull returns true if the input value is null. var IsNull = &Builtin{ - Name: "is_null", + Name: "is_null", + Description: "Returns `true` if the input value is null.", Decl: types.NewFunction( types.Args( - types.A, + types.Named("x", types.A), ), - types.B, + types.Named("result", types.B).Description("`true` if `x` is null, `false` otherwise."), ), + Categories: typesCat, } /** @@ -2099,194 +2495,282 @@ var IsNull = &Builtin{ // TypeNameBuiltin returns the type of the input. var TypeNameBuiltin = &Builtin{ - Name: "type_name", + Name: "type_name", + Description: "Returns the type of its input value.", Decl: types.NewFunction( types.Args( - types.NewAny( - types.A, - ), + types.Named("x", types.A), ), - types.S, + types.Named("type", types.S).Description(`one of "null", "boolean", "number", "string", "array", "object", "set"`), ), + Categories: typesCat, } /** * HTTP Request */ -// HTTPSend returns a HTTP response to the given HTTP request. +// Marked non-deterministic because HTTP request results can be non-deterministic. var HTTPSend = &Builtin{ - Name: "http.send", + Name: "http.send", + Description: "Returns a HTTP response to the given HTTP request.", Decl: types.NewFunction( types.Args( - types.NewObject(nil, types.NewDynamicProperty(types.S, types.A)), + types.Named("request", types.NewObject(nil, types.NewDynamicProperty(types.S, types.A))), ), - types.NewObject(nil, types.NewDynamicProperty(types.A, types.A)), + types.Named("response", types.NewObject(nil, types.NewDynamicProperty(types.A, types.A))), ), + Nondeterministic: true, } /** - * Rego + * GraphQL */ -// RegoParseModule parses the input Rego file and returns a JSON representation -// of the AST. -var RegoParseModule = &Builtin{ - Name: "rego.parse_module", +// GraphQLParse returns a pair of AST objects from parsing/validation. +var GraphQLParse = &Builtin{ + Name: "graphql.parse", + Description: "Returns AST objects for a given GraphQL query and schema after validating the query against the schema. Returns undefined if errors were encountered during parsing or validation. The query and/or schema can be either GraphQL strings or AST objects from the other GraphQL builtin functions.", Decl: types.NewFunction( types.Args( - types.S, - types.S, + types.Named("query", types.NewAny(types.S, types.NewObject(nil, types.NewDynamicProperty(types.A, types.A)))), + types.Named("schema", types.NewAny(types.S, types.NewObject(nil, types.NewDynamicProperty(types.A, types.A)))), ), - types.NewObject(nil, types.NewDynamicProperty(types.S, types.A)), // TODO(tsandall): import AST schema + types.Named("output", types.NewArray([]types.Type{ + types.NewObject(nil, types.NewDynamicProperty(types.A, types.A)), + types.NewObject(nil, types.NewDynamicProperty(types.A, types.A)), + }, nil)).Description("`output` is of the form `[query_ast, schema_ast]`. If the GraphQL query is valid given the provided schema, then `query_ast` and `schema_ast` are objects describing the ASTs for the query and schema."), ), } -/** - * OPA - */ +// GraphQLParseAndVerify returns a boolean and a pair of AST object from parsing/validation. +var GraphQLParseAndVerify = &Builtin{ + Name: "graphql.parse_and_verify", + Description: "Returns a boolean indicating success or failure alongside the parsed ASTs for a given GraphQL query and schema after validating the query against the schema. The query and/or schema can be either GraphQL strings or AST objects from the other GraphQL builtin functions.", + Decl: types.NewFunction( + types.Args( + types.Named("query", types.NewAny(types.S, types.NewObject(nil, types.NewDynamicProperty(types.A, types.A)))), + types.Named("schema", types.NewAny(types.S, types.NewObject(nil, types.NewDynamicProperty(types.A, types.A)))), + ), + types.Named("output", types.NewArray([]types.Type{ + types.B, + types.NewObject(nil, types.NewDynamicProperty(types.A, types.A)), + types.NewObject(nil, types.NewDynamicProperty(types.A, types.A)), + }, nil)).Description(" `output` is of the form `[valid, query_ast, schema_ast]`. If the query is valid given the provided schema, then `valid` is `true`, and `query_ast` and `schema_ast` are objects describing the ASTs for the GraphQL query and schema. Otherwise, `valid` is `false` and `query_ast` and `schema_ast` are `{}`."), + ), +} -// OPARuntime returns an object containing OPA runtime information such as the -// configuration that OPA was booted with. -var OPARuntime = &Builtin{ - Name: "opa.runtime", +// GraphQLParseQuery parses the input GraphQL query and returns a JSON +// representation of its AST. +var GraphQLParseQuery = &Builtin{ + Name: "graphql.parse_query", + Description: "Returns an AST object for a GraphQL query.", Decl: types.NewFunction( - nil, - types.NewObject(nil, types.NewDynamicProperty(types.S, types.A)), + types.Args( + types.Named("query", types.S), + ), + types.Named("output", types.NewObject(nil, types.NewDynamicProperty(types.A, types.A))).Description("AST object for the GraphQL query."), ), } -/** - * Trace - */ +// GraphQLParseSchema parses the input GraphQL schema and returns a JSON +// representation of its AST. +var GraphQLParseSchema = &Builtin{ + Name: "graphql.parse_schema", + Description: "Returns an AST object for a GraphQL schema.", + Decl: types.NewFunction( + types.Args( + types.Named("schema", types.S), + ), + types.Named("output", types.NewObject(nil, types.NewDynamicProperty(types.A, types.A))).Description("AST object for the GraphQL schema."), + ), +} -// Trace prints a note that is included in the query explanation. -var Trace = &Builtin{ - Name: "trace", +// GraphQLIsValid returns true if a GraphQL query is valid with a given +// schema, and returns false for all other inputs. +var GraphQLIsValid = &Builtin{ + Name: "graphql.is_valid", + Description: "Checks that a GraphQL query is valid against a given schema. The query and/or schema can be either GraphQL strings or AST objects from the other GraphQL builtin functions.", Decl: types.NewFunction( types.Args( - types.S, + types.Named("query", types.NewAny(types.S, types.NewObject(nil, types.NewDynamicProperty(types.A, types.A)))), + types.Named("schema", types.NewAny(types.S, types.NewObject(nil, types.NewDynamicProperty(types.A, types.A)))), ), - types.B, + types.Named("output", types.B).Description("`true` if the query is valid under the given schema. `false` otherwise."), ), } /** - * Set + * Rego */ -// Intersection returns the intersection of the given input sets -var Intersection = &Builtin{ - Name: "intersection", +var RegoParseModule = &Builtin{ + Name: "rego.parse_module", + Description: "Parses the input Rego string and returns an object representation of the AST.", Decl: types.NewFunction( types.Args( - types.NewSet(types.NewSet(types.A)), + types.Named("filename", types.S).Description("file name to attach to AST nodes' locations"), + types.Named("rego", types.S).Description("Rego module"), ), - types.NewSet(types.A), + types.Named("output", types.NewObject(nil, types.NewDynamicProperty(types.S, types.A))), // TODO(tsandall): import AST schema ), } -// Union returns the union of the given input sets -var Union = &Builtin{ - Name: "union", +var RegoMetadataChain = &Builtin{ + Name: "rego.metadata.chain", + Description: `Returns the chain of metadata for the active rule. +Ordered starting at the active rule, going outward to the most distant node in its package ancestry. +A chain entry is a JSON document with two members: "path", an array representing the path of the node; and "annotations", a JSON document containing the annotations declared for the node. +The first entry in the chain always points to the active rule, even if it has no declared annotations (in which case the "annotations" member is not present).`, + Decl: types.NewFunction( + types.Args(), + types.Named("chain", types.NewArray(nil, types.A)).Description("each array entry represents a node in the path ancestry (chain) of the active rule that also has declared annotations"), + ), +} + +// RegoMetadataRule returns the metadata for the active rule +var RegoMetadataRule = &Builtin{ + Name: "rego.metadata.rule", + Description: "Returns annotations declared for the active rule and using the _rule_ scope.", + Decl: types.NewFunction( + types.Args(), + types.Named("output", types.A).Description("\"rule\" scope annotations for this rule; empty object if no annotations exist"), + ), +} + +/** + * OPA + */ + +// Marked non-deterministic because of unpredictable config/environment-dependent results. +var OPARuntime = &Builtin{ + Name: "opa.runtime", + Description: "Returns an object that describes the runtime environment where OPA is deployed.", + Decl: types.NewFunction( + nil, + types.Named("output", types.NewObject(nil, types.NewDynamicProperty(types.S, types.A))). + Description("includes a `config` key if OPA was started with a configuration file; an `env` key containing the environment variables that the OPA process was started with; includes `version` and `commit` keys containing the version and build commit of OPA."), + ), + Nondeterministic: true, +} + +/** + * Trace + */ +var tracing = category("tracing") + +var Trace = &Builtin{ + Name: "trace", + Description: "Emits `note` as a `Note` event in the query explanation. Query explanations show the exact expressions evaluated by OPA during policy execution. For example, `trace(\"Hello There!\")` includes `Note \"Hello There!\"` in the query explanation. To include variables in the message, use `sprintf`. For example, `person := \"Bob\"; trace(sprintf(\"Hello There! %v\", [person]))` will emit `Note \"Hello There! Bob\"` inside of the explanation.", Decl: types.NewFunction( types.Args( - types.NewSet(types.NewSet(types.A)), + types.Named("note", types.S).Description("the note to include"), ), - types.NewSet(types.A), + types.Named("result", types.B).Description("always `true`"), ), + Categories: tracing, } /** * Glob */ -// GlobMatch - not to be confused with regex.globs_match - parses and matches strings against the glob notation. var GlobMatch = &Builtin{ - Name: "glob.match", + Name: "glob.match", + Description: "Parses and matches strings against the glob notation. Not to be confused with `regex.globs_match`.", Decl: types.NewFunction( types.Args( - types.S, - types.NewArray(nil, types.S), - types.S, + types.Named("pattern", types.S), + types.Named("delimiters", types.NewAny( + types.NewArray(nil, types.S), + types.NewNull(), + )).Description("glob pattern delimiters, e.g. `[\".\", \":\"]`, defaults to `[\".\"]` if unset. If `delimiters` is `null`, glob match without delimiter."), + types.Named("match", types.S), ), - types.B, + types.Named("result", types.B).Description("true if `match` can be found in `pattern` which is separated by `delimiters`"), ), } -// GlobQuoteMeta returns a string which represents a version of the pattern where all asterisks have been escaped. var GlobQuoteMeta = &Builtin{ - Name: "glob.quote_meta", + Name: "glob.quote_meta", + Description: "Returns a string which represents a version of the pattern where all asterisks have been escaped.", Decl: types.NewFunction( types.Args( - types.S, + types.Named("pattern", types.S), ), - types.S, + types.Named("output", types.S).Description("the escaped string of `pattern`"), ), + // TODO(sr): example for this was: Calling ``glob.quote_meta("*.github.com", output)`` returns ``\\*.github.com`` as ``output``. } /** * Networking */ -// NetCIDRIntersects checks if a cidr intersects with another cidr and returns true or false var NetCIDRIntersects = &Builtin{ - Name: "net.cidr_intersects", + Name: "net.cidr_intersects", + Description: "Checks if a CIDR intersects with another CIDR (e.g. `192.168.0.0/16` overlaps with `192.168.1.0/24`). Supports both IPv4 and IPv6 notations.", Decl: types.NewFunction( types.Args( - types.S, - types.S, + types.Named("cidr1", types.S), + types.Named("cidr2", types.S), ), - types.B, + types.Named("result", types.B), ), } -// NetCIDRExpand returns a set of hosts inside the specified cidr. var NetCIDRExpand = &Builtin{ - Name: "net.cidr_expand", + Name: "net.cidr_expand", + Description: "Expands CIDR to set of hosts (e.g., `net.cidr_expand(\"192.168.0.0/30\")` generates 4 hosts: `{\"192.168.0.0\", \"192.168.0.1\", \"192.168.0.2\", \"192.168.0.3\"}`).", Decl: types.NewFunction( types.Args( - types.S, + types.Named("cidr", types.S), ), - types.NewSet(types.S), + types.Named("hosts", types.NewSet(types.S)).Description("set of IP addresses the CIDR `cidr` expands to"), ), } -// NetCIDRContains checks if a cidr or ip is contained within another cidr and returns true or false var NetCIDRContains = &Builtin{ - Name: "net.cidr_contains", + Name: "net.cidr_contains", + Description: "Checks if a CIDR or IP is contained within another CIDR. `output` is `true` if `cidr_or_ip` (e.g. `127.0.0.64/26` or `127.0.0.1`) is contained within `cidr` (e.g. `127.0.0.1/24`) and `false` otherwise. Supports both IPv4 and IPv6 notations.", Decl: types.NewFunction( types.Args( - types.S, - types.S, + types.Named("cidr", types.S), + types.Named("cidr_or_ip", types.S), ), - types.B, + types.Named("result", types.B), ), } -// NetCIDRContainsMatches checks if collections of cidrs or ips are contained within another collection of cidrs and returns matches. var NetCIDRContainsMatches = &Builtin{ Name: "net.cidr_contains_matches", + Description: "Checks if collections of cidrs or ips are contained within another collection of cidrs and returns matches. " + + "This function is similar to `net.cidr_contains` except it allows callers to pass collections of CIDRs or IPs as arguments and returns the matches (as opposed to a boolean result indicating a match between two CIDRs/IPs).", Decl: types.NewFunction( - types.Args(netCidrContainsMatchesOperandType, netCidrContainsMatchesOperandType), - types.NewSet(types.NewArray([]types.Type{types.A, types.A}, nil)), + types.Args( + types.Named("cidrs", netCidrContainsMatchesOperandType), + types.Named("cidrs_or_ips", netCidrContainsMatchesOperandType), + ), + types.Named("output", types.NewSet(types.NewArray([]types.Type{types.A, types.A}, nil))).Description("tuples identifying matches where `cidrs_or_ips` are contained within `cidrs`"), ), } -// NetCIDRMerge merges IP addresses and subnets into the smallest possible list of CIDRs. var NetCIDRMerge = &Builtin{ Name: "net.cidr_merge", + Description: "Merges IP addresses and subnets into the smallest possible list of CIDRs (e.g., `net.cidr_merge([\"192.0.128.0/24\", \"192.0.129.0/24\"])` generates `{\"192.0.128.0/23\"}`." + + `This function merges adjacent subnets where possible, those contained within others and also removes any duplicates. +Supports both IPv4 and IPv6 notations. IPv6 inputs need a prefix length (e.g. "/128").`, Decl: types.NewFunction( - types.Args(netCidrMergeOperandType), - types.NewSet(types.S), + types.Args( + types.Named("addrs", types.NewAny( + types.NewArray(nil, types.NewAny(types.S)), + types.NewSet(types.S), + )).Description("CIDRs or IP addresses"), + ), + types.Named("output", types.NewSet(types.S)).Description("smallest possible set of CIDRs obtained after merging the provided list of IP addresses and subnets in `addrs`"), ), } -var netCidrMergeOperandType = types.NewAny( - types.NewArray(nil, types.NewAny(types.S)), - types.NewSet(types.S), -) - var netCidrContainsMatchesOperandType = types.NewAny( types.S, types.NewArray(nil, types.NewAny( @@ -2306,44 +2790,43 @@ var netCidrContainsMatchesOperandType = types.NewAny( )), ) -// NetLookupIPAddr returns the set of IP addresses (as strings, both v4 and v6) -// that the passed-in name (string) resolves to using the standard name resolution -// mechanisms available. +// Marked non-deterministic because DNS resolution results can be non-deterministic. var NetLookupIPAddr = &Builtin{ - Name: "net.lookup_ip_addr", + Name: "net.lookup_ip_addr", + Description: "Returns the set of IP addresses (both v4 and v6) that the passed-in `name` resolves to using the standard name resolution mechanisms available.", Decl: types.NewFunction( - types.Args(types.S), - types.NewSet(types.S), + types.Args( + types.Named("name", types.S).Description("domain name to resolve"), + ), + types.Named("addrs", types.NewSet(types.S)).Description("IP addresses (v4 and v6) that `name` resolves to"), ), + Nondeterministic: true, } /** * Semantic Versions */ -// SemVerIsValid validiates a the term is a valid SemVer as a string, returns -// false for all other input var SemVerIsValid = &Builtin{ - Name: "semver.is_valid", + Name: "semver.is_valid", + Description: "Validates that the input is a valid SemVer string.", Decl: types.NewFunction( types.Args( - types.A, + types.Named("vsn", types.A), ), - types.B, + types.Named("result", types.B).Description("`true` if `vsn` is a valid SemVer; `false` otherwise"), ), } -// SemVerCompare compares valid SemVer formatted version strings. Given two -// version strings, if A < B returns -1, if A > B returns 1. If A == B, returns -// 0 var SemVerCompare = &Builtin{ - Name: "semver.compare", + Name: "semver.compare", + Description: "Compares valid SemVer formatted version strings.", Decl: types.NewFunction( types.Args( - types.S, - types.S, + types.Named("a", types.S), + types.Named("b", types.S), ), - types.N, + types.Named("result", types.N).Description("`-1` if `a < b`; `1` if `a > b`; `0` if `a == b`"), ), } @@ -2382,6 +2865,7 @@ var SetDiff = &Builtin{ ), types.NewSet(types.A), ), + deprecated: true, } // NetCIDROverlap has been replaced by the `net.cidr_contains` built-in. @@ -2394,6 +2878,7 @@ var NetCIDROverlap = &Builtin{ ), types.B, ), + deprecated: true, } // CastArray checks the underlying type of the input. If it is array or set, an array @@ -2404,6 +2889,7 @@ var CastArray = &Builtin{ types.Args(types.A), types.NewArray(nil, types.A), ), + deprecated: true, } // CastSet checks the underlying type of the input. @@ -2416,6 +2902,7 @@ var CastSet = &Builtin{ types.Args(types.A), types.NewSet(types.A), ), + deprecated: true, } // CastString returns input if it is a string; if not returns error. @@ -2426,6 +2913,7 @@ var CastString = &Builtin{ types.Args(types.A), types.S, ), + deprecated: true, } // CastBoolean returns input if it is a boolean; if not returns error. @@ -2435,6 +2923,7 @@ var CastBoolean = &Builtin{ types.Args(types.A), types.B, ), + deprecated: true, } // CastNull returns null if input is null; if not returns error. @@ -2444,6 +2933,7 @@ var CastNull = &Builtin{ types.Args(types.A), types.NewNull(), ), + deprecated: true, } // CastObject returns the given object if it is null; throws an error otherwise @@ -2453,6 +2943,7 @@ var CastObject = &Builtin{ types.Args(types.A), types.NewObject(nil, types.NewDynamicProperty(types.A, types.A)), ), + deprecated: true, } // RegexMatchDeprecated declares `re_match` which has been deprecated. Use `regex.match` instead. @@ -2465,15 +2956,72 @@ var RegexMatchDeprecated = &Builtin{ ), types.B, ), + deprecated: true, +} + +// All takes a list and returns true if all of the items +// are true. A collection of length 0 returns true. +var All = &Builtin{ + Name: "all", + Decl: types.NewFunction( + types.Args( + types.NewAny( + types.NewSet(types.A), + types.NewArray(nil, types.A), + ), + ), + types.B, + ), + deprecated: true, +} + +// Any takes a collection and returns true if any of the items +// is true. A collection of length 0 returns false. +var Any = &Builtin{ + Name: "any", + Decl: types.NewFunction( + types.Args( + types.NewAny( + types.NewSet(types.A), + types.NewArray(nil, types.A), + ), + ), + types.B, + ), + deprecated: true, } // Builtin represents a built-in function supported by OPA. Every built-in // function is uniquely identified by a name. type Builtin struct { - Name string `json:"name"` // Unique name of built-in function, e.g., (arg1,arg2,...,argN) - Decl *types.Function `json:"decl"` // Built-in function type declaration. - Infix string `json:"infix,omitempty"` // Unique name of infix operator. Default should be unset. - Relation bool `json:"relation,omitempty"` // Indicates if the built-in acts as a relation. + Name string `json:"name"` // Unique name of built-in function, e.g., (arg1,arg2,...,argN) + Description string `json:"description,omitempty"` // Description of what the built-in function does. + + // Categories of the built-in function. Omitted for namespaced + // built-ins, i.e. "array.concat" is taken to be of the "array" category. + // "minus" for example, is part of two categories: numbers and sets. (NOTE(sr): aspirational) + Categories []string `json:"categories,omitempty"` + + Decl *types.Function `json:"decl"` // Built-in function type declaration. + Infix string `json:"infix,omitempty"` // Unique name of infix operator. Default should be unset. + Relation bool `json:"relation,omitempty"` // Indicates if the built-in acts as a relation. + deprecated bool // Indicates if the built-in has been deprecated. + Nondeterministic bool `json:"nondeterministic,omitempty"` // Indicates if the built-in returns non-deterministic results. +} + +// category is a helper for specifying a Builtin's Categories +func category(cs ...string) []string { + return cs +} + +// IsDeprecated returns true if the Builtin function is deprecated and will be removed in a future release. +func (b *Builtin) IsDeprecated() bool { + return b.deprecated +} + +// IsDeterministic returns true if the Builtin function returns non-deterministic results. +func (b *Builtin) IsNondeterministic() bool { + return b.Nondeterministic } // Expr creates a new expression for the built-in with the given operands. diff --git a/vendor/github.com/open-policy-agent/opa/ast/capabilities.go b/vendor/github.com/open-policy-agent/opa/ast/capabilities.go index a6b162ed..4869767a 100644 --- a/vendor/github.com/open-policy-agent/opa/ast/capabilities.go +++ b/vendor/github.com/open-policy-agent/opa/ast/capabilities.go @@ -5,14 +5,19 @@ package ast import ( + "bytes" + "fmt" "io" + "os" "sort" + "strings" + caps "github.com/open-policy-agent/opa/capabilities" "github.com/open-policy-agent/opa/internal/wasm/sdk/opa/capabilities" "github.com/open-policy-agent/opa/util" ) -// Capabilities defines a structure containing data that describes the capablilities +// Capabilities defines a structure containing data that describes the capabilities // or features supported by a particular version of OPA. type Capabilities struct { Builtins []*Builtin `json:"builtins"` @@ -44,7 +49,8 @@ func CapabilitiesForThisVersion() *Capabilities { f.WasmABIVersions = append(f.WasmABIVersions, WasmABIVersion{Version: vers[0], Minor: vers[1]}) } - f.Builtins = append(f.Builtins, Builtins...) + f.Builtins = make([]*Builtin, len(Builtins)) + copy(f.Builtins, Builtins) sort.Slice(f.Builtins, func(i, j int) bool { return f.Builtins[i].Name < f.Builtins[j].Name }) @@ -63,3 +69,48 @@ func LoadCapabilitiesJSON(r io.Reader) (*Capabilities, error) { var c Capabilities return &c, d.Decode(&c) } + +// LoadCapabilitiesVersion loads a JSON serialized capabilities structure from the specific version. +func LoadCapabilitiesVersion(version string) (*Capabilities, error) { + cvs, err := LoadCapabilitiesVersions() + if err != nil { + return nil, err + } + + for _, cv := range cvs { + if cv == version { + cont, err := caps.FS.ReadFile(cv + ".json") + if err != nil { + return nil, err + } + + return LoadCapabilitiesJSON(bytes.NewReader(cont)) + } + + } + return nil, fmt.Errorf("no capabilities version found %v", version) +} + +// LoadCapabilitiesFile loads a JSON serialized capabilities structure from a file. +func LoadCapabilitiesFile(file string) (*Capabilities, error) { + fd, err := os.Open(file) + if err != nil { + return nil, err + } + defer fd.Close() + return LoadCapabilitiesJSON(fd) +} + +// LoadCapabilitiesVersions loads all capabilities versions +func LoadCapabilitiesVersions() ([]string, error) { + ents, err := caps.FS.ReadDir(".") + if err != nil { + return nil, err + } + + capabilitiesVersions := make([]string, 0, len(ents)) + for _, ent := range ents { + capabilitiesVersions = append(capabilitiesVersions, strings.Replace(ent.Name(), ".json", "", 1)) + } + return capabilitiesVersions, nil +} diff --git a/vendor/github.com/open-policy-agent/opa/ast/check.go b/vendor/github.com/open-policy-agent/opa/ast/check.go index 1e0971bc..fd35d017 100644 --- a/vendor/github.com/open-policy-agent/opa/ast/check.go +++ b/vendor/github.com/open-policy-agent/opa/ast/check.go @@ -136,16 +136,8 @@ func (tc *typeChecker) CheckBody(env *TypeEnv, body Body) (*TypeEnv, Errors) { // CheckTypes runs type checking on the rules returns a TypeEnv if no errors // are found. The resulting TypeEnv wraps the provided one. The resulting // TypeEnv will be able to resolve types of refs that refer to rules. -func (tc *typeChecker) CheckTypes(env *TypeEnv, sorted []util.T) (*TypeEnv, Errors) { +func (tc *typeChecker) CheckTypes(env *TypeEnv, sorted []util.T, as *AnnotationSet) (*TypeEnv, Errors) { env = tc.newEnv(env) - var as *annotationSet - if tc.ss != nil { - var errs Errors - as, errs = buildAnnotationSet(sorted) - if len(errs) > 0 { - return env, errs - } - } for _, s := range sorted { tc.checkRule(env, as, s.(*Rule)) } @@ -181,7 +173,7 @@ func (tc *typeChecker) checkClosures(env *TypeEnv, expr *Expr) Errors { return result } -func (tc *typeChecker) checkRule(env *TypeEnv, as *annotationSet, rule *Rule) { +func (tc *typeChecker) checkRule(env *TypeEnv, as *AnnotationSet, rule *Rule) { env = env.wrap() @@ -275,6 +267,9 @@ func (tc *typeChecker) checkRule(env *TypeEnv, as *annotationSet, rule *Rule) { } func (tc *typeChecker) checkExpr(env *TypeEnv, expr *Expr) *Error { + if err := tc.checkExprWith(env, expr, 0); err != nil { + return err + } if !expr.IsCall() { return nil } @@ -319,17 +314,19 @@ func (tc *typeChecker) checkExprBuiltin(env *TypeEnv, expr *Expr) *Error { } fargs := ftpe.FuncArgs() + namedFargs := ftpe.NamedFuncArgs() if ftpe.Result() != nil { fargs.Args = append(fargs.Args, ftpe.Result()) + namedFargs.Args = append(namedFargs.Args, ftpe.NamedResult()) } if len(args) > len(fargs.Args) && fargs.Variadic == nil { - return newArgError(expr.Location, name, "too many arguments", pre, fargs) + return newArgError(expr.Location, name, "too many arguments", pre, namedFargs) } if len(args) < len(ftpe.FuncArgs().Args) { - return newArgError(expr.Location, name, "too few arguments", pre, fargs) + return newArgError(expr.Location, name, "too few arguments", pre, namedFargs) } for i := range args { @@ -338,7 +335,7 @@ func (tc *typeChecker) checkExprBuiltin(env *TypeEnv, expr *Expr) *Error { for i := range args { post[i] = env.Get(args[i]) } - return newArgError(expr.Location, name, "invalid argument(s)", post, fargs) + return newArgError(expr.Location, name, "invalid argument(s)", post, namedFargs) } } @@ -373,6 +370,27 @@ func (tc *typeChecker) checkExprEq(env *TypeEnv, expr *Expr) *Error { return nil } +func (tc *typeChecker) checkExprWith(env *TypeEnv, expr *Expr, i int) *Error { + if i == len(expr.With) { + return nil + } + + target, value := expr.With[i].Target, expr.With[i].Value + targetType, valueType := env.Get(target), env.Get(value) + + if t, ok := targetType.(*types.Function); ok { // built-in function replacement + switch v := valueType.(type) { + case *types.Function: // ...by function + if !unifies(targetType, valueType) { + return newArgError(expr.With[i].Loc(), target.Value.(Ref), "arity mismatch", v.Args(), t.NamedFuncArgs()) + } + default: // ... by value, nothing to check + } + } + + return tc.checkExprWith(env, expr, i+1) +} + func unify2(env *TypeEnv, a *Term, typeA types.Type, b *Term, typeB types.Type) bool { nilA := types.Nil(typeA) @@ -618,11 +636,11 @@ func (rc *refChecker) Visit(x interface{}) bool { } func (rc *refChecker) checkApply(curr *TypeEnv, ref Ref) *Error { - if tpe := curr.Get(ref); tpe != nil { - if _, ok := tpe.(*types.Function); ok { - return newRefErrUnsupported(ref[0].Location, rc.varRewriter(ref), len(ref)-1, tpe) - } + switch tpe := curr.Get(ref).(type) { + case *types.Function: // NOTE(sr): We don't support first-class functions, except for `with`. + return newRefErrUnsupported(ref[0].Location, rc.varRewriter(ref), len(ref)-1, tpe) } + return nil } @@ -805,7 +823,17 @@ func unifies(a, b types.Type) bool { } return unifies(types.Values(a), types.Values(b)) case *types.Function: - // TODO(tsandall): revisit once functions become first-class values. + // NOTE(sr): variadic functions can only be internal ones, and we've forbidden + // their replacement via `with`; so we disregard variadic here + if types.Arity(a) == types.Arity(b) { + b := b.(*types.Function) + for i := range a.FuncArgs().Args { + if !unifies(a.FuncArgs().Arg(i), b.FuncArgs().Arg(i)) { + return false + } + } + return true + } return false default: panic("unreachable") @@ -1170,7 +1198,7 @@ func getObjectType(ref Ref, o types.Type, rule *Rule, d *types.DynamicProperty) return getObjectTypeRec(keys, o, d), nil } -func getRuleAnnotation(as *annotationSet, rule *Rule) (result []*SchemaAnnotation) { +func getRuleAnnotation(as *AnnotationSet, rule *Rule) (result []*SchemaAnnotation) { for _, x := range as.GetSubpackagesScope(rule.Module.Package.Path) { result = append(result, x.Schemas...) @@ -1215,158 +1243,3 @@ func processAnnotation(ss *SchemaSet, annot *SchemaAnnotation, rule *Rule, allow func errAnnotationRedeclared(a *Annotations, other *Location) *Error { return NewError(TypeErr, a.Location, "%v annotation redeclared: %v", a.Scope, other) } - -type annotationSet struct { - byRule map[*Rule][]*Annotations - byPackage map[*Package]*Annotations - byPath *annotationTreeNode -} - -func buildAnnotationSet(rules []util.T) (*annotationSet, Errors) { - as := newAnnotationSet() - processed := map[*Module]struct{}{} - var errs Errors - for _, x := range rules { - module := x.(*Rule).Module - if _, ok := processed[module]; ok { - continue - } - processed[module] = struct{}{} - for _, a := range module.Annotations { - if err := as.Add(a); err != nil { - errs = append(errs, err) - } - } - } - if len(errs) > 0 { - return nil, errs - } - return as, nil -} - -func newAnnotationSet() *annotationSet { - return &annotationSet{ - byRule: map[*Rule][]*Annotations{}, - byPackage: map[*Package]*Annotations{}, - byPath: newAnnotationTree(), - } -} - -func (as *annotationSet) Add(a *Annotations) *Error { - switch a.Scope { - case annotationScopeRule: - rule := a.node.(*Rule) - as.byRule[rule] = append(as.byRule[rule], a) - case annotationScopePackage: - pkg := a.node.(*Package) - if exist, ok := as.byPackage[pkg]; ok { - return errAnnotationRedeclared(a, exist.Location) - } - as.byPackage[pkg] = a - case annotationScopeDocument: - rule := a.node.(*Rule) - path := rule.Path() - x := as.byPath.Get(path) - if x != nil { - return errAnnotationRedeclared(a, x.Value.Location) - } - as.byPath.Insert(path, a) - case annotationScopeSubpackages: - pkg := a.node.(*Package) - x := as.byPath.Get(pkg.Path) - if x != nil { - return errAnnotationRedeclared(a, x.Value.Location) - } - as.byPath.Insert(pkg.Path, a) - } - return nil -} - -func (as *annotationSet) GetRuleScope(r *Rule) []*Annotations { - if as == nil { - return nil - } - return as.byRule[r] -} - -func (as *annotationSet) GetSubpackagesScope(path Ref) []*Annotations { - if as == nil { - return nil - } - return as.byPath.Ancestors(path) -} - -func (as *annotationSet) GetDocumentScope(path Ref) *Annotations { - if as == nil { - return nil - } - if node := as.byPath.Get(path); node != nil { - return node.Value - } - return nil -} - -func (as *annotationSet) GetPackageScope(pkg *Package) *Annotations { - if as == nil { - return nil - } - return as.byPackage[pkg] -} - -type annotationTreeNode struct { - Value *Annotations - Children map[Value]*annotationTreeNode // we assume key elements are hashable (vars and strings only!) -} - -func newAnnotationTree() *annotationTreeNode { - return &annotationTreeNode{ - Value: nil, - Children: map[Value]*annotationTreeNode{}, - } -} - -func (t *annotationTreeNode) Insert(path Ref, value *Annotations) { - node := t - for _, k := range path { - child, ok := node.Children[k.Value] - if !ok { - child = newAnnotationTree() - node.Children[k.Value] = child - } - node = child - } - node.Value = value -} - -func (t *annotationTreeNode) Get(path Ref) *annotationTreeNode { - node := t - for _, k := range path { - if node == nil { - return nil - } - child, ok := node.Children[k.Value] - if !ok { - return nil - } - node = child - } - return node -} - -func (t *annotationTreeNode) Ancestors(path Ref) (result []*Annotations) { - node := t - for _, k := range path { - if node == nil { - return result - } - child, ok := node.Children[k.Value] - if !ok { - return result - } - if child.Value != nil { - result = append(result, child.Value) - } - node = child - } - return result -} diff --git a/vendor/github.com/open-policy-agent/opa/ast/compare.go b/vendor/github.com/open-policy-agent/opa/ast/compare.go index 685082da..ca9fab7a 100644 --- a/vendor/github.com/open-policy-agent/opa/ast/compare.go +++ b/vendor/github.com/open-policy-agent/opa/ast/compare.go @@ -201,6 +201,9 @@ func Compare(a, b interface{}) int { case *SomeDecl: b := b.(*SomeDecl) return a.Compare(b) + case *Every: + b := b.(*Every) + return a.Compare(b) case *With: b := b.(*With) return a.Compare(b) @@ -272,6 +275,8 @@ func sortOrder(x interface{}) int { return 100 case *SomeDecl: return 101 + case *Every: + return 102 case *With: return 110 case *Head: diff --git a/vendor/github.com/open-policy-agent/opa/ast/compile.go b/vendor/github.com/open-policy-agent/opa/ast/compile.go index e0c27e4b..a650033b 100644 --- a/vendor/github.com/open-policy-agent/opa/ast/compile.go +++ b/vendor/github.com/open-policy-agent/opa/ast/compile.go @@ -11,6 +11,7 @@ import ( "strconv" "strings" + "github.com/open-policy-agent/opa/ast/location" "github.com/open-policy-agent/opa/internal/debug" "github.com/open-policy-agent/opa/internal/gojsonschema" "github.com/open-policy-agent/opa/metrics" @@ -103,12 +104,17 @@ type Compiler struct { builtins map[string]*Builtin // universe of built-in functions customBuiltins map[string]*Builtin // user-supplied custom built-in functions (deprecated: use capabilities) unsafeBuiltinsMap map[string]struct{} // user-supplied set of unsafe built-ins functions to block (deprecated: use capabilities) + deprecatedBuiltinsMap map[string]struct{} // set of deprecated, but not removed, built-in functions enablePrintStatements bool // indicates if print statements should be elided (default) comprehensionIndices map[*Term]*ComprehensionIndex // comprehension key index initialized bool // indicates if init() has been called debug debug.Debug // emits debug information produced during compilation schemaSet *SchemaSet // user-supplied schemas for input and data documents inputType types.Type // global input type retrieved from schema set + annotationSet *AnnotationSet // hierarchical set of annotations + strict bool // enforce strict compilation checks + keepModules bool // whether to keep the unprocessed, parse modules (below) + parsedModules map[string]*Module // parsed, but otherwise unprocessed modules, kept track of when keepModules is true } // CompilerStage defines the interface for stages in the compiler. @@ -239,11 +245,12 @@ func NewCompiler() *Compiler { }, func(x util.T) int { return x.(Ref).Hash() }), - maxErrs: CompileErrorLimitDefault, - after: map[string][]CompilerStageDefinition{}, - unsafeBuiltinsMap: map[string]struct{}{}, - comprehensionIndices: map[*Term]*ComprehensionIndex{}, - debug: debug.Discard(), + maxErrs: CompileErrorLimitDefault, + after: map[string][]CompilerStageDefinition{}, + unsafeBuiltinsMap: map[string]struct{}{}, + deprecatedBuiltinsMap: map[string]struct{}{}, + comprehensionIndices: map[*Term]*ComprehensionIndex{}, + debug: debug.Discard(), } c.ModuleTree = NewModuleTree(nil) @@ -258,6 +265,9 @@ func NewCompiler() *Compiler { // load additional modules. If any stages run before resolution, they // need to be re-run after resolution. {"ResolveRefs", "compile_stage_resolve_refs", c.resolveAllRefs}, + {"CheckKeywordOverrides", "compile_stage_check_keyword_overrides", c.checkKeywordOverrides}, + {"CheckDuplicateImports", "compile_stage_check_duplicate_imports", c.checkDuplicateImports}, + {"RemoveImports", "compile_stage_remove_imports", c.removeImports}, {"SetModuleTree", "compile_stage_set_module_tree", c.setModuleTree}, {"SetRuleTree", "compile_stage_set_rule_tree", c.setRuleTree}, // The local variable generator must be initialized after references are @@ -268,6 +278,9 @@ func NewCompiler() *Compiler { {"CheckVoidCalls", "compile_stage_check_void_calls", c.checkVoidCalls}, {"RewritePrintCalls", "compile_stage_rewrite_print_calls", c.rewritePrintCalls}, {"RewriteExprTerms", "compile_stage_rewrite_expr_terms", c.rewriteExprTerms}, + {"ParseMetadataBlocks", "compile_stage_parse_metadata_blocks", c.parseMetadataBlocks}, + {"SetAnnotationSet", "compile_stage_set_annotationset", c.setAnnotationSet}, + {"RewriteRegoMetadataCalls", "compile_stage_rewrite_rego_metadata_calls", c.rewriteRegoMetadataCalls}, {"SetGraph", "compile_stage_set_graph", c.setGraph}, {"RewriteComprehensionTerms", "compile_stage_rewrite_comprehension_terms", c.rewriteComprehensionTerms}, {"RewriteRefsInHead", "compile_stage_rewrite_refs_in_head", c.rewriteRefsInHead}, @@ -279,8 +292,9 @@ func NewCompiler() *Compiler { {"RewriteEquals", "compile_stage_rewrite_equals", c.rewriteEquals}, {"RewriteDynamicTerms", "compile_stage_rewrite_dynamic_terms", c.rewriteDynamicTerms}, {"CheckRecursion", "compile_stage_check_recursion", c.checkRecursion}, - {"CheckTypes", "compile_stage_check_types", c.checkTypes}, + {"CheckTypes", "compile_stage_check_types", c.checkTypes}, // must be run after CheckRecursion {"CheckUnsafeBuiltins", "compile_state_check_unsafe_builtins", c.checkUnsafeBuiltins}, + {"CheckDeprecatedBuiltins", "compile_state_check_deprecated_builtins", c.checkDeprecatedBuiltins}, {"BuildRuleIndices", "compile_stage_rebuild_indices", c.buildRuleIndices}, {"BuildComprehensionIndices", "compile_stage_rebuild_comprehension_indices", c.buildComprehensionIndices}, } @@ -367,6 +381,28 @@ func (c *Compiler) WithUnsafeBuiltins(unsafeBuiltins map[string]struct{}) *Compi return c } +// WithStrict enables strict mode in the compiler. +func (c *Compiler) WithStrict(strict bool) *Compiler { + c.strict = strict + return c +} + +// WithKeepModules enables retaining unprocessed modules in the compiler. +// Note that the modules aren't copied on the way in or out -- so when +// accessing them via ParsedModules(), mutations will occur in the module +// map that was passed into Compile().` +func (c *Compiler) WithKeepModules(y bool) *Compiler { + c.keepModules = y + return c +} + +// ParsedModules returns the parsed, unprocessed modules from the compiler. +// It is `nil` if keeping modules wasn't enabled via `WithKeepModules(true)`. +// The map includes all modules loaded via the ModuleLoader, if one was used. +func (c *Compiler) ParsedModules() map[string]*Module { + return c.parsedModules +} + // QueryCompiler returns a new QueryCompiler object. func (c *Compiler) QueryCompiler() QueryCompiler { c.init() @@ -382,10 +418,20 @@ func (c *Compiler) Compile(modules map[string]*Module) { c.init() c.Modules = make(map[string]*Module, len(modules)) + c.sorted = make([]string, 0, len(modules)) + + if c.keepModules { + c.parsedModules = make(map[string]*Module, len(modules)) + } else { + c.parsedModules = nil + } for k, v := range modules { c.Modules[k] = v.Copy() c.sorted = append(c.sorted, k) + if c.parsedModules != nil { + c.parsedModules[k] = v + } } sort.Strings(c.sorted) @@ -429,16 +475,16 @@ func (c *Compiler) GetArity(ref Ref) int { // // E.g., given the following module: // -// package a.b.c +// package a.b.c // -// p[k] = v { ... } # rule1 -// p[k1] = v1 { ... } # rule2 +// p[k] = v { ... } # rule1 +// p[k1] = v1 { ... } # rule2 // // The following calls yield the rules on the right. // -// GetRulesExact("data.a.b.c.p") => [rule1, rule2] -// GetRulesExact("data.a.b.c.p.x") => nil -// GetRulesExact("data.a.b.c") => nil +// GetRulesExact("data.a.b.c.p") => [rule1, rule2] +// GetRulesExact("data.a.b.c.p.x") => nil +// GetRulesExact("data.a.b.c") => nil func (c *Compiler) GetRulesExact(ref Ref) (rules []*Rule) { node := c.RuleTree @@ -456,16 +502,16 @@ func (c *Compiler) GetRulesExact(ref Ref) (rules []*Rule) { // // E.g., given the following module: // -// package a.b.c +// package a.b.c // -// p[k] = v { ... } # rule1 -// p[k1] = v1 { ... } # rule2 +// p[k] = v { ... } # rule1 +// p[k1] = v1 { ... } # rule2 // // The following calls yield the rules on the right. // -// GetRulesForVirtualDocument("data.a.b.c.p") => [rule1, rule2] -// GetRulesForVirtualDocument("data.a.b.c.p.x") => [rule1, rule2] -// GetRulesForVirtualDocument("data.a.b.c") => nil +// GetRulesForVirtualDocument("data.a.b.c.p") => [rule1, rule2] +// GetRulesForVirtualDocument("data.a.b.c.p.x") => [rule1, rule2] +// GetRulesForVirtualDocument("data.a.b.c") => nil func (c *Compiler) GetRulesForVirtualDocument(ref Ref) (rules []*Rule) { node := c.RuleTree @@ -486,17 +532,17 @@ func (c *Compiler) GetRulesForVirtualDocument(ref Ref) (rules []*Rule) { // // E.g., given the following module: // -// package a.b.c +// package a.b.c // -// p[x] = y { ... } # rule1 -// p[k] = v { ... } # rule2 -// q { ... } # rule3 +// p[x] = y { ... } # rule1 +// p[k] = v { ... } # rule2 +// q { ... } # rule3 // // The following calls yield the rules on the right. // -// GetRulesWithPrefix("data.a.b.c.p") => [rule1, rule2] -// GetRulesWithPrefix("data.a.b.c.p.a") => nil -// GetRulesWithPrefix("data.a.b.c") => [rule1, rule2, rule3] +// GetRulesWithPrefix("data.a.b.c.p") => [rule1, rule2] +// GetRulesWithPrefix("data.a.b.c.p.a") => nil +// GetRulesWithPrefix("data.a.b.c") => [rule1, rule2, rule3] func (c *Compiler) GetRulesWithPrefix(ref Ref) (rules []*Rule) { node := c.RuleTree @@ -535,18 +581,18 @@ func extractRules(s []util.T) (rules []*Rule) { // // E.g., given the following module: // -// package a.b.c +// package a.b.c // -// p[x] = y { q[x] = y; ... } # rule1 -// q[x] = y { ... } # rule2 +// p[x] = y { q[x] = y; ... } # rule1 +// q[x] = y { ... } # rule2 // // The following calls yield the rules on the right. // -// GetRules("data.a.b.c.p") => [rule1] -// GetRules("data.a.b.c.p.x") => [rule1] -// GetRules("data.a.b.c.q") => [rule2] -// GetRules("data.a.b.c") => [rule1, rule2] -// GetRules("data.a.b.d") => nil +// GetRules("data.a.b.c.p") => [rule1] +// GetRules("data.a.b.c.p.x") => [rule1] +// GetRules("data.a.b.c.q") => [rule2] +// GetRules("data.a.b.c") => [rule1, rule2] +// GetRules("data.a.b.d") => nil func (c *Compiler) GetRules(ref Ref) (rules []*Rule) { set := map[*Rule]struct{}{} @@ -581,34 +627,34 @@ func (c *Compiler) GetRulesDynamic(ref Ref) []*Rule { // // E.g., given the following modules: // -// package a.b.c +// package a.b.c // -// r1 = 1 # rule1 +// r1 = 1 # rule1 // // and: // -// package a.d.c +// package a.d.c // -// r2 = 2 # rule2 +// r2 = 2 # rule2 // // The following calls yield the rules on the right. // -// GetRulesDynamicWithOpts("data.a[x].c[y]", opts) => [rule1, rule2] -// GetRulesDynamicWithOpts("data.a[x].c.r2", opts) => [rule2] -// GetRulesDynamicWithOpts("data.a.b[x][y]", opts) => [rule1] +// GetRulesDynamicWithOpts("data.a[x].c[y]", opts) => [rule1, rule2] +// GetRulesDynamicWithOpts("data.a[x].c.r2", opts) => [rule2] +// GetRulesDynamicWithOpts("data.a.b[x][y]", opts) => [rule1] // // Using the RulesOptions parameter, the inclusion of hidden modules can be // controlled: // // With // -// package system.main +// package system.main // -// r3 = 3 # rule3 +// r3 = 3 # rule3 // // We'd get this result: // -// GetRulesDynamicWithOpts("data[x]", RulesOptions{IncludeHiddenModules: true}) => [rule1, rule2, rule3] +// GetRulesDynamicWithOpts("data[x]", RulesOptions{IncludeHiddenModules: true}) => [rule1, rule2, rule3] // // Without the options, it would be excluded. func (c *Compiler) GetRulesDynamicWithOpts(ref Ref, opts RulesOptions) []*Rule { @@ -617,7 +663,8 @@ func (c *Compiler) GetRulesDynamicWithOpts(ref Ref, opts RulesOptions) []*Rule { set := map[*Rule]struct{}{} var walk func(node *TreeNode, i int) walk = func(node *TreeNode, i int) { - if i >= len(ref) { + switch { + case i >= len(ref): // We've reached the end of the reference and want to collect everything // under this "prefix". node.DepthFirst(func(descendant *TreeNode) bool { @@ -627,7 +674,8 @@ func (c *Compiler) GetRulesDynamicWithOpts(ref Ref, opts RulesOptions) []*Rule { } return descendant.Hide }) - } else if i == 0 || IsConstant(ref[i].Value) { + + case i == 0 || IsConstant(ref[i].Value): // The head of the ref is always grounded. In case another part of the // ref is also grounded, we can lookup the exact child. If it's not found // we can immediately return... @@ -641,7 +689,8 @@ func (c *Compiler) GetRulesDynamicWithOpts(ref Ref, opts RulesOptions) []*Rule { // Otherwise, we continue using the child node. walk(child, i+1) } - } else { + + default: // This part of the ref is a dynamic term. We can't know what it refers // to and will just need to try all of the children. for _, child := range node.Children { @@ -789,15 +838,11 @@ func (c *Compiler) checkRuleConflicts() { kinds := map[DocKind]struct{}{} defaultRules := 0 arities := map[int]struct{}{} - declared := false for _, rule := range node.Values { r := rule.(*Rule) kinds[r.Head.DocKind()] = struct{}{} arities[len(r.Head.Args)] = struct{}{} - if r.Head.Assign { - declared = true - } if r.Default { defaultRules++ } @@ -805,9 +850,7 @@ func (c *Compiler) checkRuleConflicts() { name := Var(node.Key.(String)) - if declared && len(node.Values) > 1 { - c.err(NewError(TypeErr, node.Values[0].(*Rule).Loc(), "rule named %v redeclared at %v", name, node.Values[1].(*Rule).Loc())) - } else if len(kinds) > 1 || len(arities) > 1 { + if len(kinds) > 1 || len(arities) > 1 { c.err(NewError(TypeErr, node.Values[0].(*Rule).Loc(), "conflicting rules named %v found", name)) } else if defaultRules > 1 { c.err(NewError(TypeErr, node.Values[0].(*Rule).Loc(), "multiple default rules named %s found", name)) @@ -886,7 +929,7 @@ func arityMismatchError(env *TypeEnv, f Ref, expr *Expr, exp, act int) *Error { for i, op := range expr.Operands() { have[i] = env.Get(op) } - return newArgError(expr.Loc(), f, "arity mismatch", have, want.FuncArgs()) + return newArgError(expr.Loc(), f, "arity mismatch", have, want.NamedFuncArgs()) } if act != 1 { return NewError(TypeErr, expr.Loc(), "function %v has arity %d, got %d arguments", f, exp, act) @@ -1010,7 +1053,25 @@ func mergeSchemas(schemas ...*gojsonschema.SubSchema) (*gojsonschema.SubSchema, return result, nil } -func parseSchema(schema interface{}) (types.Type, error) { +type schemaParser struct { + definitionCache map[string]*cachedDef +} + +type cachedDef struct { + properties []*types.StaticProperty +} + +func newSchemaParser() *schemaParser { + return &schemaParser{ + definitionCache: map[string]*cachedDef{}, + } +} + +func (parser *schemaParser) parseSchema(schema interface{}) (types.Type, error) { + return parser.parseSchemaWithPropertyKey(schema, "") +} + +func (parser *schemaParser) parseSchemaWithPropertyKey(schema interface{}, propertyKey string) (types.Type, error) { subSchema, ok := schema.(*gojsonschema.SubSchema) if !ok { return nil, fmt.Errorf("unexpected schema type %v", subSchema) @@ -1018,7 +1079,10 @@ func parseSchema(schema interface{}) (types.Type, error) { // Handle referenced schemas, returns directly when a $ref is found if subSchema.RefSchema != nil { - return parseSchema(subSchema.RefSchema) + if existing, ok := parser.definitionCache[subSchema.Ref.String()]; ok { + return types.NewObject(existing.properties, nil), nil + } + return parser.parseSchemaWithPropertyKey(subSchema.RefSchema, subSchema.Ref.String()) } // Handle anyOf @@ -1030,7 +1094,7 @@ func parseSchema(schema interface{}) (types.Type, error) { copySchema := *subSchema copySchemaRef := ©Schema copySchemaRef.AnyOf = nil - coreType, err := parseSchema(copySchemaRef) + coreType, err := parser.parseSchema(copySchemaRef) if err != nil { return nil, fmt.Errorf("unexpected schema type %v: %w", subSchema, err) } @@ -1045,7 +1109,7 @@ func parseSchema(schema interface{}) (types.Type, error) { // Iterate through every property of AnyOf and add it to orType for _, pSchema := range subSchema.AnyOf { - newtype, err := parseSchema(pSchema) + newtype, err := parser.parseSchema(pSchema) if err != nil { return nil, fmt.Errorf("unexpected schema type %v: %w", pSchema, err) } @@ -1068,12 +1132,12 @@ func parseSchema(schema interface{}) (types.Type, error) { if err != nil { return nil, err } - return parseSchema(objectOrArrayResult) + return parser.parseSchema(objectOrArrayResult) } else if subSchema.Types.String() != allOfResult.Types.String() { return nil, fmt.Errorf("unable to merge these schemas") } } - return parseSchema(allOfResult) + return parser.parseSchema(allOfResult) } if subSchema.Types.IsTyped() { @@ -1088,15 +1152,28 @@ func parseSchema(schema interface{}) (types.Type, error) { } else if subSchema.Types.Contains("object") { if len(subSchema.PropertiesChildren) > 0 { - staticProps := make([]*types.StaticProperty, 0, len(subSchema.PropertiesChildren)) + def := &cachedDef{ + properties: make([]*types.StaticProperty, 0, len(subSchema.PropertiesChildren)), + } + for _, pSchema := range subSchema.PropertiesChildren { + def.properties = append(def.properties, types.NewStaticProperty(pSchema.Property, nil)) + } + if propertyKey != "" { + parser.definitionCache[propertyKey] = def + } for _, pSchema := range subSchema.PropertiesChildren { - newtype, err := parseSchema(pSchema) + newtype, err := parser.parseSchema(pSchema) if err != nil { return nil, fmt.Errorf("unexpected schema type %v: %w", pSchema, err) } - staticProps = append(staticProps, types.NewStaticProperty(pSchema.Property, newtype)) + for i, prop := range def.properties { + if prop.Key == pSchema.Property { + def.properties[i].Value = newtype + break + } + } } - return types.NewObject(staticProps, nil), nil + return types.NewObject(def.properties, nil), nil } return types.NewObject(nil, types.NewDynamicProperty(types.A, types.A)), nil @@ -1104,7 +1181,7 @@ func parseSchema(schema interface{}) (types.Type, error) { if len(subSchema.ItemsChildren) > 0 { if subSchema.ItemsChildrenIsSingleSchema { iSchema := subSchema.ItemsChildren[0] - newtype, err := parseSchema(iSchema) + newtype, err := parser.parseSchema(iSchema) if err != nil { return nil, fmt.Errorf("unexpected schema type %v", iSchema) } @@ -1113,7 +1190,7 @@ func parseSchema(schema interface{}) (types.Type, error) { newTypes := make([]types.Type, 0, len(subSchema.ItemsChildren)) for i := 0; i != len(subSchema.ItemsChildren); i++ { iSchema := subSchema.ItemsChildren[i] - newtype, err := parseSchema(iSchema) + newtype, err := parser.parseSchema(iSchema) if err != nil { return nil, fmt.Errorf("unexpected schema type %v", iSchema) } @@ -1128,17 +1205,31 @@ func parseSchema(schema interface{}) (types.Type, error) { // Assume types if not specified in schema if len(subSchema.PropertiesChildren) > 0 { if err := subSchema.Types.Add("object"); err == nil { - return parseSchema(subSchema) + return parser.parseSchema(subSchema) } } else if len(subSchema.ItemsChildren) > 0 { if err := subSchema.Types.Add("array"); err == nil { - return parseSchema(subSchema) + return parser.parseSchema(subSchema) } } return types.A, nil } +func (c *Compiler) setAnnotationSet() { + // Sorting modules by name for stable error reporting + sorted := make([]*Module, 0, len(c.Modules)) + for _, mName := range c.sorted { + sorted = append(sorted, c.Modules[mName]) + } + + as, errs := BuildAnnotationSet(sorted) + for _, err := range errs { + c.err(err) + } + c.annotationSet = as +} + // checkTypes runs the type checker on all rules. The type checker builds a // TypeEnv that is stored on the compiler. func (c *Compiler) checkTypes() { @@ -1148,7 +1239,7 @@ func (c *Compiler) checkTypes() { WithSchemaSet(c.schemaSet). WithInputType(c.inputType). WithVarRewriter(rewriteVarsInRef(c.RewrittenVars)) - env, errs := checker.CheckTypes(c.TypeEnv, sorted) + env, errs := checker.CheckTypes(c.TypeEnv, sorted, c.annotationSet) for _, err := range errs { c.err(err) } @@ -1164,6 +1255,15 @@ func (c *Compiler) checkUnsafeBuiltins() { } } +func (c *Compiler) checkDeprecatedBuiltins() { + for _, name := range c.sorted { + errs := checkDeprecatedBuiltins(c.deprecatedBuiltinsMap, c.Modules[name], c.strict) + for _, err := range errs { + c.err(err) + } + } +} + func (c *Compiler) runStage(metricName string, f func()) { if c.metrics != nil { c.metrics.Timer(metricName).Start() @@ -1193,10 +1293,10 @@ func (c *Compiler) compile() { if c.Failed() { return } - for _, s := range c.after[s.name] { - err := c.runStageAfter(s.MetricName, s.Stage) - if err != nil { + for _, a := range c.after[s.name] { + if err := c.runStageAfter(a.MetricName, a.Stage); err != nil { c.err(err) + return } } } @@ -1216,6 +1316,9 @@ func (c *Compiler) init() { for _, bi := range c.capabilities.Builtins { c.builtins[bi.Name] = bi + if c.strict && bi.IsDeprecated() { + c.deprecatedBuiltinsMap[bi.Name] = struct{}{} + } } for name, bi := range c.customBuiltins { @@ -1253,9 +1356,7 @@ func (c *Compiler) err(err *Error) { func (c *Compiler) getExports() *util.HashMap { rules := util.NewHashMap(func(a, b util.T) bool { - r1 := a.(Ref) - r2 := a.(Ref) - return r1.Equal(r2) + return a.(Ref).Equal(b.(Ref)) }, func(v util.T) int { return v.(Ref).Hash() }) @@ -1277,6 +1378,69 @@ func (c *Compiler) getExports() *util.HashMap { return rules } +func (c *Compiler) GetAnnotationSet() *AnnotationSet { + return c.annotationSet +} + +func (c *Compiler) checkDuplicateImports() { + if !c.strict { + return + } + + for _, name := range c.sorted { + mod := c.Modules[name] + processedImports := map[Var]*Import{} + + for _, imp := range mod.Imports { + name := imp.Name() + + if processed, conflict := processedImports[name]; conflict { + c.err(NewError(CompileErr, imp.Location, "import must not shadow %v", processed)) + } else { + processedImports[name] = imp + } + } + } +} + +func (c *Compiler) checkKeywordOverrides() { + for _, name := range c.sorted { + mod := c.Modules[name] + errs := checkKeywordOverrides(mod, c.strict) + for _, err := range errs { + c.err(err) + } + } +} + +func checkKeywordOverrides(node interface{}, strict bool) Errors { + if !strict { + return nil + } + + errors := Errors{} + + WalkRules(node, func(rule *Rule) bool { + name := rule.Head.Name.String() + if RootDocumentRefs.Contains(RefTerm(VarTerm(name))) { + errors = append(errors, NewError(CompileErr, rule.Location, "rules must not shadow %v (use a different rule name)", name)) + } + return true + }) + + WalkExprs(node, func(expr *Expr) bool { + if expr.IsAssignment() { + name := expr.Operand(0).String() + if RootDocumentRefs.Contains(RefTerm(VarTerm(name))) { + errors = append(errors, NewError(CompileErr, expr.Location, "variables must not shadow %v (use a different variable name)", name)) + } + } + return false + }) + + return errors +} + // resolveAllRefs resolves references in expressions to their fully qualified values. // // For instance, given the following module: @@ -1308,8 +1472,20 @@ func (c *Compiler) resolveAllRefs() { return false }) - // Once imports have been resolved, they are no longer needed. - mod.Imports = nil + if c.strict { // check for unused imports + for _, imp := range mod.Imports { + path := imp.Path.Value.(Ref) + if FutureRootDocument.Equal(path[0]) { + continue // ignore future imports + } + + for v, u := range globals { + if v.Equal(imp.Name()) && !u.used { + c.err(NewError(CompileErr, imp.Location, "%s unused", imp.String())) + } + } + } + } } if c.moduleLoader != nil { @@ -1327,6 +1503,9 @@ func (c *Compiler) resolveAllRefs() { for id, module := range parsed { c.Modules[id] = module.Copy() c.sorted = append(c.sorted, id) + if c.parsedModules != nil { + c.parsedModules[id] = module + } } sort.Strings(c.sorted) @@ -1334,6 +1513,12 @@ func (c *Compiler) resolveAllRefs() { } } +func (c *Compiler) removeImports() { + for name := range c.Modules { + c.Modules[name].Imports = nil + } +} + func (c *Compiler) initLocalVarGen() { c.localvargen = newLocalVarGeneratorForModuleSet(c.sorted, c.Modules) } @@ -1378,12 +1563,14 @@ func (c *Compiler) rewritePrintCalls() { WalkRules(mod, func(r *Rule) bool { safe := r.Head.Args.Vars() safe.Update(ReservedVars) - WalkBodies(r, func(b Body) bool { + vis := func(b Body) bool { for _, err := range rewritePrintCalls(c.localvargen, c.GetArity, safe, b) { c.err(err) } return false - }) + } + WalkBodies(r.Head, vis) + WalkBodies(r.Body, vis) return false }) } @@ -1412,11 +1599,11 @@ func checkVoidCalls(env *TypeEnv, x interface{}) Errors { // // For example, given the following print statement: // -// print("the value of x is:", input.x) +// print("the value of x is:", input.x) // // The expression would be rewritten to: // -// print({__local0__ | __local0__ = "the value of x is:"}, {__local1__ | __local1__ = input.x}) +// print({__local0__ | __local0__ = "the value of x is:"}, {__local1__ | __local1__ = input.x}) func rewritePrintCalls(gen *localVarGenerator, getArity func(Ref) int, globals VarSet, body Body) Errors { var errs Errors @@ -1424,7 +1611,7 @@ func rewritePrintCalls(gen *localVarGenerator, getArity func(Ref) int, globals V // Visit comprehension bodies recursively to ensure print statements inside // those bodies only close over variables that are safe. for i := range body { - if ContainsComprehensions(body[i]) { + if ContainsClosures(body[i]) { safe := outputVarsForBody(body[:i], getArity, globals) safe.Update(globals) WalkClosures(body[i], func(x interface{}) bool { @@ -1435,6 +1622,9 @@ func rewritePrintCalls(gen *localVarGenerator, getArity func(Ref) int, globals V errs = rewritePrintCalls(gen, getArity, safe, x.Body) case *ObjectComprehension: errs = rewritePrintCalls(gen, getArity, safe, x.Body) + case *Every: + safe.Update(x.KeyValueVars()) + errs = rewritePrintCalls(gen, getArity, safe, x.Body) } return true }) @@ -1496,6 +1686,8 @@ func erasePrintCalls(node interface{}) { x.Body = erasePrintCallsInBody(x.Body) case *ObjectComprehension: x.Body = erasePrintCallsInBody(x.Body) + case *Every: + x.Body = erasePrintCallsInBody(x.Body) } return false }).Walk(node) @@ -1603,6 +1795,222 @@ func (c *Compiler) rewriteDynamicTerms() { } } +func (c *Compiler) parseMetadataBlocks() { + // Only parse annotations if rego.metadata built-ins are called + regoMetadataCalled := false + for _, name := range c.sorted { + mod := c.Modules[name] + WalkExprs(mod, func(expr *Expr) bool { + if isRegoMetadataChainCall(expr) || isRegoMetadataRuleCall(expr) { + regoMetadataCalled = true + } + return regoMetadataCalled + }) + + if regoMetadataCalled { + break + } + } + + if regoMetadataCalled { + // NOTE: Possible optimization: only parse annotations for modules on the path of rego.metadata-calling module + for _, name := range c.sorted { + mod := c.Modules[name] + + if len(mod.Annotations) == 0 { + var errs Errors + mod.Annotations, errs = parseAnnotations(mod.Comments) + errs = append(errs, attachAnnotationsNodes(mod)...) + for _, err := range errs { + c.err(err) + } + } + } + } +} + +func (c *Compiler) rewriteRegoMetadataCalls() { + eqFactory := newEqualityFactory(c.localvargen) + + _, chainFuncAllowed := c.builtins[RegoMetadataChain.Name] + _, ruleFuncAllowed := c.builtins[RegoMetadataRule.Name] + + for _, name := range c.sorted { + mod := c.Modules[name] + + WalkRules(mod, func(rule *Rule) bool { + var firstChainCall *Expr + var firstRuleCall *Expr + + WalkExprs(rule, func(expr *Expr) bool { + if chainFuncAllowed && firstChainCall == nil && isRegoMetadataChainCall(expr) { + firstChainCall = expr + } else if ruleFuncAllowed && firstRuleCall == nil && isRegoMetadataRuleCall(expr) { + firstRuleCall = expr + } + return firstChainCall != nil && firstRuleCall != nil + }) + + chainCalled := firstChainCall != nil + ruleCalled := firstRuleCall != nil + + if chainCalled || ruleCalled { + body := make(Body, 0, len(rule.Body)+2) + + var metadataChainVar Var + if chainCalled { + // Create and inject metadata chain for rule + + chain, err := createMetadataChain(c.annotationSet.Chain(rule)) + if err != nil { + c.err(err) + return false + } + + chain.Location = firstChainCall.Location + eq := eqFactory.Generate(chain) + metadataChainVar = eq.Operands()[0].Value.(Var) + body.Append(eq) + } + + var metadataRuleVar Var + if ruleCalled { + // Create and inject metadata for rule + + var metadataRuleTerm *Term + + a := getPrimaryRuleAnnotations(c.annotationSet, rule) + if a != nil { + annotObj, err := a.toObject() + if err != nil { + c.err(err) + return false + } + metadataRuleTerm = NewTerm(*annotObj) + } else { + // If rule has no annotations, assign an empty object + metadataRuleTerm = ObjectTerm() + } + + metadataRuleTerm.Location = firstRuleCall.Location + eq := eqFactory.Generate(metadataRuleTerm) + metadataRuleVar = eq.Operands()[0].Value.(Var) + body.Append(eq) + } + + for _, expr := range rule.Body { + body.Append(expr) + } + rule.Body = body + + vis := func(b Body) bool { + for _, err := range rewriteRegoMetadataCalls(&metadataChainVar, &metadataRuleVar, b, &c.RewrittenVars) { + c.err(err) + } + return false + } + WalkBodies(rule.Head, vis) + WalkBodies(rule.Body, vis) + } + + return false + }) + } +} + +func getPrimaryRuleAnnotations(as *AnnotationSet, rule *Rule) *Annotations { + annots := as.GetRuleScope(rule) + + if len(annots) == 0 { + return nil + } + + // Sort by annotation location; chain must start with annotations declared closest to rule, then going outward + sort.SliceStable(annots, func(i, j int) bool { + return annots[i].Location.Compare(annots[j].Location) > 0 + }) + + return annots[0] +} + +func rewriteRegoMetadataCalls(metadataChainVar *Var, metadataRuleVar *Var, body Body, rewrittenVars *map[Var]Var) Errors { + var errs Errors + + WalkClosures(body, func(x interface{}) bool { + switch x := x.(type) { + case *ArrayComprehension: + errs = rewriteRegoMetadataCalls(metadataChainVar, metadataRuleVar, x.Body, rewrittenVars) + case *SetComprehension: + errs = rewriteRegoMetadataCalls(metadataChainVar, metadataRuleVar, x.Body, rewrittenVars) + case *ObjectComprehension: + errs = rewriteRegoMetadataCalls(metadataChainVar, metadataRuleVar, x.Body, rewrittenVars) + case *Every: + errs = rewriteRegoMetadataCalls(metadataChainVar, metadataRuleVar, x.Body, rewrittenVars) + } + return true + }) + + for i := range body { + expr := body[i] + var metadataVar Var + + if metadataChainVar != nil && isRegoMetadataChainCall(expr) { + metadataVar = *metadataChainVar + } else if metadataRuleVar != nil && isRegoMetadataRuleCall(expr) { + metadataVar = *metadataRuleVar + } else { + continue + } + + // NOTE(johanfylling): An alternative strategy would be to walk the body and replace all operands[0] + // usages with *metadataChainVar + operands := expr.Operands() + var newExpr *Expr + if len(operands) > 0 { // There is an output var to rewrite + rewrittenVar := operands[0] + newExpr = Equality.Expr(rewrittenVar, NewTerm(metadataVar)) + } else { // No output var, just rewrite expr to metadataVar + newExpr = NewExpr(NewTerm(metadataVar)) + } + + newExpr.Generated = true + newExpr.Location = expr.Location + body.Set(newExpr, i) + } + + return errs +} + +func isRegoMetadataChainCall(x *Expr) bool { + return x.IsCall() && x.Operator().Equal(RegoMetadataChain.Ref()) +} + +func isRegoMetadataRuleCall(x *Expr) bool { + return x.IsCall() && x.Operator().Equal(RegoMetadataRule.Ref()) +} + +func createMetadataChain(chain []*AnnotationsRef) (*Term, *Error) { + + metaArray := NewArray() + for _, link := range chain { + p := link.Path.toArray(). + Slice(1, -1) // Dropping leading 'data' element of path + obj := NewObject( + Item(StringTerm("path"), NewTerm(p)), + ) + if link.Annotations != nil { + annotObj, err := link.Annotations.toObject() + if err != nil { + return nil, err + } + obj.Insert(StringTerm("annotations"), NewTerm(*annotObj)) + } + metaArray = metaArray.Append(NewTerm(obj)) + } + + return NewTerm(metaArray), nil +} + func (c *Compiler) rewriteLocalVars() { for _, name := range c.sorted { @@ -1624,6 +2032,7 @@ func (c *Compiler) rewriteLocalVars() { nestedXform := &rewriteNestedHeadVarLocalTransform{ gen: gen, RewrittenVars: c.RewrittenVars, + strict: c.strict, } NewGenericVisitor(nestedXform.Visit).Walk(rule.Head) @@ -1647,7 +2056,7 @@ func (c *Compiler) rewriteLocalVars() { c.rewriteLocalArgVars(gen, stack, rule) - body, declared, errs := rewriteLocalVars(gen, stack, used, rule.Body) + body, declared, errs := rewriteLocalVars(gen, stack, used, rule.Body, c.strict) for _, err := range errs { c.err(err) } @@ -1684,6 +2093,7 @@ type rewriteNestedHeadVarLocalTransform struct { gen *localVarGenerator errs Errors RewrittenVars map[Var]Var + strict bool } func (xform *rewriteNestedHeadVarLocalTransform) Visit(x interface{}) bool { @@ -1713,13 +2123,13 @@ func (xform *rewriteNestedHeadVarLocalTransform) Visit(x interface{}) bool { term.Value = cpy stop = true case *ArrayComprehension: - xform.errs = rewriteDeclaredVarsInArrayComprehension(xform.gen, stack, x, xform.errs) + xform.errs = rewriteDeclaredVarsInArrayComprehension(xform.gen, stack, x, xform.errs, xform.strict) stop = true case *SetComprehension: - xform.errs = rewriteDeclaredVarsInSetComprehension(xform.gen, stack, x, xform.errs) + xform.errs = rewriteDeclaredVarsInSetComprehension(xform.gen, stack, x, xform.errs, xform.strict) stop = true case *ObjectComprehension: - xform.errs = rewriteDeclaredVarsInObjectComprehension(xform.gen, stack, x, xform.errs) + xform.errs = rewriteDeclaredVarsInObjectComprehension(xform.gen, stack, x, xform.errs, xform.strict) stop = true } @@ -1778,7 +2188,9 @@ func (vis *ruleArgLocalRewriter) Visit(x interface{}) Visitor { switch v := t.Value.(type) { case Var: gv, ok := vis.stack.Declared(v) - if !ok { + if ok { + vis.stack.Seen(v) + } else { gv = vis.gen.Generate() vis.stack.Insert(v, gv, argVar) } @@ -1818,7 +2230,7 @@ func (c *Compiler) rewriteWithModifiers() { if !ok { return x, nil } - body, err := rewriteWithModifiersInBody(c, f, body) + body, err := rewriteWithModifiersInBody(c, c.unsafeBuiltinsMap, f, body) if err != nil { c.err(err) } @@ -1926,6 +2338,7 @@ func (qc *queryCompiler) Compile(query Body) (Body, error) { metricName string f func(*QueryContext, Body) (Body, error) }{ + {"CheckKeywordOverrides", "query_compile_stage_check_keyword_overrides", qc.checkKeywordOverrides}, {"ResolveRefs", "query_compile_stage_resolve_refs", qc.resolveRefs}, {"RewriteLocalVars", "query_compile_stage_rewrite_local_vars", qc.rewriteLocalVars}, {"CheckVoidCalls", "query_compile_stage_check_void_calls", qc.checkVoidCalls}, @@ -1938,6 +2351,7 @@ func (qc *queryCompiler) Compile(query Body) (Body, error) { {"RewriteDynamicTerms", "query_compile_stage_rewrite_dynamic_terms", qc.rewriteDynamicTerms}, {"CheckTypes", "query_compile_stage_check_types", qc.checkTypes}, {"CheckUnsafeBuiltins", "query_compile_stage_check_unsafe_builtins", qc.checkUnsafeBuiltins}, + {"CheckDeprecatedBuiltins", "query_compile_stage_check_deprecated_builtins", qc.checkDeprecatedBuiltins}, {"BuildComprehensionIndex", "query_compile_stage_build_comprehension_index", qc.buildComprehensionIndices}, } @@ -1973,9 +2387,16 @@ func (qc *queryCompiler) applyErrorLimit(err error) error { return err } +func (qc *queryCompiler) checkKeywordOverrides(_ *QueryContext, body Body) (Body, error) { + if errs := checkKeywordOverrides(body, qc.compiler.strict); len(errs) > 0 { + return nil, errs + } + return body, nil +} + func (qc *queryCompiler) resolveRefs(qctx *QueryContext, body Body) (Body, error) { - var globals map[Var]Ref + var globals map[Var]*usedRef if qctx != nil { pkg := qctx.Package @@ -2025,7 +2446,7 @@ func (qc *queryCompiler) rewriteExprTerms(_ *QueryContext, body Body) (Body, err func (qc *queryCompiler) rewriteLocalVars(_ *QueryContext, body Body) (Body, error) { gen := newLocalVarGenerator("q", body) stack := newLocalDeclaredVars() - body, _, err := rewriteLocalVars(gen, stack, nil, body) + body, _, err := rewriteLocalVars(gen, stack, nil, body, qc.compiler.strict) if len(err) != 0 { return nil, err } @@ -2088,13 +2509,22 @@ func (qc *queryCompiler) checkTypes(_ *QueryContext, body Body) (Body, error) { } func (qc *queryCompiler) checkUnsafeBuiltins(_ *QueryContext, body Body) (Body, error) { - var unsafe map[string]struct{} + errs := checkUnsafeBuiltins(qc.unsafeBuiltinsMap(), body) + if len(errs) > 0 { + return nil, errs + } + return body, nil +} + +func (qc *queryCompiler) unsafeBuiltinsMap() map[string]struct{} { if qc.unsafeBuiltins != nil { - unsafe = qc.unsafeBuiltins - } else { - unsafe = qc.compiler.unsafeBuiltinsMap + return qc.unsafeBuiltins } - errs := checkUnsafeBuiltins(unsafe, body) + return qc.compiler.unsafeBuiltinsMap +} + +func (qc *queryCompiler) checkDeprecatedBuiltins(_ *QueryContext, body Body) (Body, error) { + errs := checkDeprecatedBuiltins(qc.compiler.deprecatedBuiltinsMap, body, qc.compiler.strict) if len(errs) > 0 { return nil, errs } @@ -2103,7 +2533,7 @@ func (qc *queryCompiler) checkUnsafeBuiltins(_ *QueryContext, body Body) (Body, func (qc *queryCompiler) rewriteWithModifiers(_ *QueryContext, body Body) (Body, error) { f := newEqualityFactory(newLocalVarGenerator("q", body)) - body, err := rewriteWithModifiersInBody(qc.compiler, f, body) + body, err := rewriteWithModifiersInBody(qc.compiler, qc.unsafeBuiltinsMap(), f, body) if err != nil { return nil, Errors{err} } @@ -2368,7 +2798,13 @@ func NewModuleTree(mods map[string]*Module) *ModuleTreeNode { root := &ModuleTreeNode{ Children: map[Value]*ModuleTreeNode{}, } - for _, m := range mods { + names := make([]string, 0, len(mods)) + for name := range mods { + names = append(names, name) + } + sort.Strings(names) + for _, name := range names { + m := mods[name] node := root for i, x := range m.Package.Path { c, ok := node.Children[x.Value] @@ -2484,6 +2920,19 @@ func (n *TreeNode) Child(k Value) *TreeNode { return nil } +// Find dereferences ref along the tree +func (n *TreeNode) Find(ref Ref) *TreeNode { + node := n + for _, r := range ref { + child := node.Child(r.Value) + if child == nil { + return nil + } + node = child + } + return node +} + // DepthFirst performs a depth-first traversal of the rule tree rooted at n. If // f returns true, traversal will not continue to the children of n. func (n *TreeNode) DepthFirst(f func(node *TreeNode) bool) { @@ -2537,7 +2986,7 @@ func NewGraph(modules map[string]*Module, list func(Ref) []*Rule) *Graph { }) } - // Walk over all rules, add them to graph, and build adjencency lists. + // Walk over all rules, add them to graph, and build adjacency lists. for _, module := range modules { WalkRules(module, func(a *Rule) bool { graph.addNode(a) @@ -2757,13 +3206,10 @@ func (vs unsafeVars) Slice() (result []unsafePair) { // contains a mapping of expressions to unsafe variables in those expressions. func reorderBodyForSafety(builtins map[string]*Builtin, arity func(Ref) int, globals VarSet, body Body) (Body, unsafeVars) { - body, unsafe := reorderBodyForClosures(arity, globals, body) - if len(unsafe) != 0 { - return nil, unsafe - } - - reordered := Body{} + bodyVars := body.Vars(SafetyCheckVisitorParams) + reordered := make(Body, 0, len(body)) safe := VarSet{} + unsafe := unsafeVars{} for _, e := range body { for v := range e.Vars(SafetyCheckVisitorParams) { @@ -2783,10 +3229,23 @@ func reorderBodyForSafety(builtins map[string]*Builtin, arity func(Ref) int, glo continue } - safe.Update(outputVarsForExpr(e, arity, safe)) + ovs := outputVarsForExpr(e, arity, safe) + + // check closures: is this expression closing over variables that + // haven't been made safe by what's already included in `reordered`? + vs := unsafeVarsInClosures(e, arity, safe) + cv := vs.Intersect(bodyVars).Diff(globals) + uv := cv.Diff(outputVarsForBody(reordered, arity, safe)) + + if len(uv) > 0 { + if uv.Equal(ovs) { // special case "closure-self" + continue + } + unsafe.Set(e, uv) + } for v := range unsafe[e] { - if safe.Contains(v) { + if ovs.Contains(v) || safe.Contains(v) { delete(unsafe[e], v) } } @@ -2794,10 +3253,11 @@ func reorderBodyForSafety(builtins map[string]*Builtin, arity func(Ref) int, glo if len(unsafe[e]) == 0 { delete(unsafe, e) reordered.Append(e) + safe.Update(ovs) // this expression's outputs are safe } } - if len(reordered) == n { + if len(reordered) == n { // fixed point, could not add any expr of body break } } @@ -2832,7 +3292,8 @@ type bodySafetyTransformer struct { } func (xform *bodySafetyTransformer) Visit(x interface{}) bool { - if term, ok := x.(*Term); ok { + switch term := x.(type) { + case *Term: switch x := term.Value.(type) { case *object: cpy, _ := x.Map(func(k, v *Term) (*Term, *Term, error) { @@ -2862,6 +3323,12 @@ func (xform *bodySafetyTransformer) Visit(x interface{}) bool { xform.reorderSetComprehensionSafety(x) return true } + case *Expr: + if ev, ok := term.Terms.(*Every); ok { + xform.globals.Update(ev.KeyValueVars()) + ev.Body = xform.reorderComprehensionSafety(NewVarSet(), ev.Body) + return true + } } return false } @@ -2897,51 +3364,20 @@ func (xform *bodySafetyTransformer) reorderSetComprehensionSafety(sc *SetCompreh sc.Body = xform.reorderComprehensionSafety(sc.Term.Vars(), sc.Body) } -// reorderBodyForClosures returns a copy of the body ordered such that -// expressions (such as array comprehensions) that close over variables are ordered -// after other expressions that contain the same variable in an output position. -func reorderBodyForClosures(arity func(Ref) int, globals VarSet, body Body) (Body, unsafeVars) { - - reordered := Body{} - unsafe := unsafeVars{} - - for { - n := len(reordered) - - for _, e := range body { - if reordered.Contains(e) { - continue - } - - // Collect vars that are contained in closures within this - // expression. - vs := VarSet{} - WalkClosures(e, func(x interface{}) bool { - vis := &VarVisitor{vars: vs} - vis.Walk(x) - return true - }) - - // Compute vars that are closed over from the body but not yet - // contained in the output position of an expression in the reordered - // body. These vars are considered unsafe. - cv := vs.Intersect(body.Vars(SafetyCheckVisitorParams)).Diff(globals) - uv := cv.Diff(outputVarsForBody(reordered, arity, globals)) - - if len(uv) == 0 { - reordered = append(reordered, e) - delete(unsafe, e) - } else { - unsafe.Set(e, uv) - } - } - - if len(reordered) == n { - break +// unsafeVarsInClosures collects vars that are contained in closures within +// this expression. +func unsafeVarsInClosures(e *Expr, arity func(Ref) int, safe VarSet) VarSet { + vs := VarSet{} + WalkClosures(e, func(x interface{}) bool { + vis := &VarVisitor{vars: vs} + if ev, ok := x.(*Every); ok { + vis.Walk(ev.Body) + return true } - } - - return reordered, unsafe + vis.Walk(x) + return true + }) + return vs } // OutputVarsFromBody returns all variables which are the "output" for @@ -2975,15 +3411,11 @@ func outputVarsForExpr(expr *Expr, arity func(Ref) int, safe VarSet) VarSet { // With modifier inputs must be safe. for _, with := range expr.With { - unsafe := false - WalkVars(with, func(v Var) bool { - if !safe.Contains(v) { - unsafe = true - return true - } - return false - }) - if unsafe { + vis := NewVarVisitor().WithParams(SafetyCheckVisitorParams) + vis.Walk(with) + vars := vis.Vars() + unsafe := vars.Diff(safe) + if len(unsafe) > 0 { return VarSet{} } } @@ -3007,6 +3439,8 @@ func outputVarsForExpr(expr *Expr, arity func(Ref) int, safe VarSet) VarSet { } return outputVarsForExprCall(expr, ar, safe, terms) + case *Every: + return outputVarsForTerms(terms.Domain, safe) default: panic("illegal expression") } @@ -3034,13 +3468,13 @@ func outputVarsForExprCall(expr *Expr, arity int, safe VarSet, terms []*Term) Va return output } - vis := NewVarVisitor().WithParams(VarVisitorParams{ + params := VarVisitorParams{ SkipClosures: true, SkipSets: true, SkipObjectKeys: true, SkipRefHead: true, - }) - + } + vis := NewVarVisitor().WithParams(params) vis.Walk(Args(terms[:numInputTerms])) unsafe := vis.Vars().Diff(output).Diff(safe) @@ -3048,19 +3482,13 @@ func outputVarsForExprCall(expr *Expr, arity int, safe VarSet, terms []*Term) Va return VarSet{} } - vis = NewVarVisitor().WithParams(VarVisitorParams{ - SkipRefHead: true, - SkipSets: true, - SkipObjectKeys: true, - SkipClosures: true, - }) - + vis = NewVarVisitor().WithParams(params) vis.Walk(Args(terms[numInputTerms:])) output.Update(vis.vars) return output } -func outputVarsForTerms(expr *Expr, safe VarSet) VarSet { +func outputVarsForTerms(expr interface{}, safe VarSet) VarSet { output := VarSet{} WalkTerms(expr, func(x *Term) bool { switch r := x.Value.(type) { @@ -3126,31 +3554,24 @@ func (l *localVarGenerator) Generate() Var { } } -func getGlobals(pkg *Package, rules []Var, imports []*Import) map[Var]Ref { +func getGlobals(pkg *Package, rules []Var, imports []*Import) map[Var]*usedRef { - globals := map[Var]Ref{} + globals := map[Var]*usedRef{} // Populate globals with exports within the package. for _, v := range rules { global := append(Ref{}, pkg.Path...) global = append(global, &Term{Value: String(v)}) - globals[v] = global + globals[v] = &usedRef{ref: global} } // Populate globals with imports. - for _, i := range imports { - if len(i.Alias) > 0 { - path := i.Path.Value.(Ref) - globals[i.Alias] = path - } else { - path := i.Path.Value.(Ref) - if len(path) == 1 { - globals[path[0].Value.(Var)] = path - } else { - v := path[len(path)-1].Value.(String) - globals[Var(v)] = path - } + for _, imp := range imports { + path := imp.Path.Value.(Ref) + if FutureRootDocument.Equal(path[0]) { + continue // ignore future imports } + globals[imp.Name()] = &usedRef{ref: path} } return globals @@ -3163,14 +3584,14 @@ func requiresEval(x *Term) bool { return ContainsRefs(x) || ContainsComprehensions(x) } -func resolveRef(globals map[Var]Ref, ignore *declaredVarStack, ref Ref) Ref { +func resolveRef(globals map[Var]*usedRef, ignore *declaredVarStack, ref Ref) Ref { r := Ref{} for i, x := range ref { switch v := x.Value.(type) { case Var: if g, ok := globals[v]; ok && !ignore.Contains(v) { - cpy := g.Copy() + cpy := g.ref.Copy() for i := range cpy { cpy[i].SetLocation(x.Location) } @@ -3179,6 +3600,7 @@ func resolveRef(globals map[Var]Ref, ignore *declaredVarStack, ref Ref) Ref { } else { r = append(r, NewTerm(cpy).SetLocation(x.Location)) } + g.used = true } else { r = append(r, x) } @@ -3192,7 +3614,12 @@ func resolveRef(globals map[Var]Ref, ignore *declaredVarStack, ref Ref) Ref { return r } -func resolveRefsInRule(globals map[Var]Ref, rule *Rule) error { +type usedRef struct { + ref Ref + used bool +} + +func resolveRefsInRule(globals map[Var]*usedRef, rule *Rule) error { ignore := &declaredVarStack{} vars := NewVarSet() @@ -3255,15 +3682,15 @@ func resolveRefsInRule(globals map[Var]Ref, rule *Rule) error { return nil } -func resolveRefsInBody(globals map[Var]Ref, ignore *declaredVarStack, body Body) Body { - r := Body{} +func resolveRefsInBody(globals map[Var]*usedRef, ignore *declaredVarStack, body Body) Body { + r := make([]*Expr, 0, len(body)) for _, expr := range body { r = append(r, resolveRefsInExpr(globals, ignore, expr)) } return r } -func resolveRefsInExpr(globals map[Var]Ref, ignore *declaredVarStack, expr *Expr) *Expr { +func resolveRefsInExpr(globals map[Var]*usedRef, ignore *declaredVarStack, expr *Expr) *Expr { cpy := *expr switch ts := expr.Terms.(type) { case *Term: @@ -3278,6 +3705,20 @@ func resolveRefsInExpr(globals map[Var]Ref, ignore *declaredVarStack, expr *Expr if val, ok := ts.Symbols[0].Value.(Call); ok { cpy.Terms = &SomeDecl{Symbols: []*Term{CallTerm(resolveRefsInTermSlice(globals, ignore, val)...)}} } + case *Every: + locals := NewVarSet() + if ts.Key != nil { + locals.Update(ts.Key.Vars()) + } + locals.Update(ts.Value.Vars()) + ignore.Push(locals) + cpy.Terms = &Every{ + Key: ts.Key.Copy(), // TODO(sr): do more? + Value: ts.Value.Copy(), // TODO(sr): do more? + Domain: resolveRefsInTerm(globals, ignore, ts.Domain), + Body: resolveRefsInBody(globals, ignore, ts.Body), + } + ignore.Pop() } for _, w := range cpy.With { w.Target = resolveRefsInTerm(globals, ignore, w.Target) @@ -3286,14 +3727,15 @@ func resolveRefsInExpr(globals map[Var]Ref, ignore *declaredVarStack, expr *Expr return &cpy } -func resolveRefsInTerm(globals map[Var]Ref, ignore *declaredVarStack, term *Term) *Term { +func resolveRefsInTerm(globals map[Var]*usedRef, ignore *declaredVarStack, term *Term) *Term { switch v := term.Value.(type) { case Var: if g, ok := globals[v]; ok && !ignore.Contains(v) { - cpy := g.Copy() + cpy := g.ref.Copy() for i := range cpy { cpy[i].SetLocation(term.Location) } + g.used = true return NewTerm(cpy).SetLocation(term.Location) } return term @@ -3358,7 +3800,7 @@ func resolveRefsInTerm(globals map[Var]Ref, ignore *declaredVarStack, term *Term } } -func resolveRefsInTermArray(globals map[Var]Ref, ignore *declaredVarStack, terms *Array) []*Term { +func resolveRefsInTermArray(globals map[Var]*usedRef, ignore *declaredVarStack, terms *Array) []*Term { cpy := make([]*Term, terms.Len()) for i := 0; i < terms.Len(); i++ { cpy[i] = resolveRefsInTerm(globals, ignore, terms.Elem(i)) @@ -3366,7 +3808,7 @@ func resolveRefsInTermArray(globals map[Var]Ref, ignore *declaredVarStack, terms return cpy } -func resolveRefsInTermSlice(globals map[Var]Ref, ignore *declaredVarStack, terms []*Term) []*Term { +func resolveRefsInTermSlice(globals map[Var]*usedRef, ignore *declaredVarStack, terms []*Term) []*Term { cpy := make([]*Term, len(terms)) for i := 0; i < len(terms); i++ { cpy[i] = resolveRefsInTerm(globals, ignore, terms[i]) @@ -3525,11 +3967,14 @@ func rewriteEquals(x interface{}) { func rewriteDynamics(f *equalityFactory, body Body) Body { result := make(Body, 0, len(body)) for _, expr := range body { - if expr.IsEquality() { + switch { + case expr.IsEquality(): result = rewriteDynamicsEqExpr(f, expr, result) - } else if expr.IsCall() { + case expr.IsCall(): result = rewriteDynamicsCallExpr(f, expr, result) - } else { + case expr.IsEvery(): + result = rewriteDynamicsEveryExpr(f, expr, result) + default: result = rewriteDynamicsTermExpr(f, expr, result) } } @@ -3559,6 +4004,13 @@ func rewriteDynamicsCallExpr(f *equalityFactory, expr *Expr, result Body) Body { return appendExpr(result, expr) } +func rewriteDynamicsEveryExpr(f *equalityFactory, expr *Expr, result Body) Body { + ev := expr.Terms.(*Every) + result, ev.Domain = rewriteDynamicsOne(expr, f, ev.Domain, result) + ev.Body = rewriteDynamics(f, ev.Body) + return appendExpr(result, expr) +} + func rewriteDynamicsTermExpr(f *equalityFactory, expr *Expr, result Body) Body { term := expr.Terms.(*Term) result, expr.Terms = rewriteDynamicsInTerm(expr, f, term, result) @@ -3705,6 +4157,21 @@ func expandExpr(gen *localVarGenerator, expr *Expr) (result []*Expr) { result = append(result, extras...) } result = append(result, expr) + case *Every: + var extras []*Expr + if _, ok := terms.Domain.Value.(Call); ok { + extras, terms.Domain = expandExprTerm(gen, terms.Domain) + } else { + term := NewTerm(gen.Generate()).SetLocation(terms.Domain.Location) + eq := Equality.Expr(term, terms.Domain).SetLocation(terms.Domain.Location) + eq.Generated = true + eq.With = expr.With + extras = append(extras, eq) + terms.Domain = term + } + terms.Body = rewriteExprTermsInBody(gen, terms.Body) + result = append(result, extras...) + result = append(result, expr) } return } @@ -3822,8 +4289,8 @@ type localDeclaredVars struct { // rewritten contains a mapping of *all* user-defined variables // that have been rewritten whereas vars contains the state - // from the current query (not not any nested queries, and all - // vars seen). + // from the current query (not any nested queries, and all vars + // seen). rewritten map[Var]Var } @@ -3841,6 +4308,7 @@ type declaredVarSet struct { vs map[Var]Var reverse map[Var]Var occurrence map[Var]varOccurrence + count map[Var]int } func newDeclaredVarSet() *declaredVarSet { @@ -3848,6 +4316,7 @@ func newDeclaredVarSet() *declaredVarSet { vs: map[Var]Var{}, reverse: map[Var]Var{}, occurrence: map[Var]varOccurrence{}, + count: map[Var]int{}, } } @@ -3879,6 +4348,8 @@ func (s localDeclaredVars) Insert(x, y Var, occurrence varOccurrence) { elem.reverse[y] = x elem.occurrence[x] = occurrence + elem.count[x] = 1 + // If the variable has been rewritten (where x != y, with y being // the generated value), store it in the map of rewritten vars. // Assume that the generated values are unique for the compilation. @@ -3913,6 +4384,30 @@ func (s localDeclaredVars) GlobalOccurrence(x Var) (varOccurrence, bool) { return newVar, false } +// Seen marks x as seen by incrementing its counter +func (s localDeclaredVars) Seen(x Var) { + for i := len(s.vars) - 1; i >= 0; i-- { + dvs := s.vars[i] + if c, ok := dvs.count[x]; ok { + dvs.count[x] = c + 1 + return + } + } + + s.vars[len(s.vars)-1].count[x] = 1 +} + +// Count returns how many times x has been seen +func (s localDeclaredVars) Count(x Var) int { + for i := len(s.vars) - 1; i >= 0; i-- { + if c, ok := s.vars[i].count[x]; ok { + return c + } + } + + return 0 +} + // rewriteLocalVars rewrites bodies to remove assignment/declaration // expressions. For example: // @@ -3923,24 +4418,27 @@ func (s localDeclaredVars) GlobalOccurrence(x Var) (varOccurrence, bool) { // __local0__ = 1; p[__local0__] // // During rewriting, assignees are validated to prevent use before declaration. -func rewriteLocalVars(g *localVarGenerator, stack *localDeclaredVars, used VarSet, body Body) (Body, map[Var]Var, Errors) { +func rewriteLocalVars(g *localVarGenerator, stack *localDeclaredVars, used VarSet, body Body, strict bool) (Body, map[Var]Var, Errors) { var errs Errors - body, errs = rewriteDeclaredVarsInBody(g, stack, used, body, errs) + body, errs = rewriteDeclaredVarsInBody(g, stack, used, body, errs, strict) return body, stack.Pop().vs, errs } -func rewriteDeclaredVarsInBody(g *localVarGenerator, stack *localDeclaredVars, used VarSet, body Body, errs Errors) (Body, Errors) { +func rewriteDeclaredVarsInBody(g *localVarGenerator, stack *localDeclaredVars, used VarSet, body Body, errs Errors, strict bool) (Body, Errors) { var cpy Body for i := range body { var expr *Expr - if body[i].IsAssignment() { - expr, errs = rewriteDeclaredAssignment(g, stack, body[i], errs) - } else if _, ok := body[i].Terms.(*SomeDecl); ok { - expr, errs = rewriteSomeDeclStatement(g, stack, body[i], errs) - } else { - expr, errs = rewriteDeclaredVarsInExpr(g, stack, body[i], errs) + switch { + case body[i].IsAssignment(): + expr, errs = rewriteDeclaredAssignment(g, stack, body[i], errs, strict) + case body[i].IsSome(): + expr, errs = rewriteSomeDeclStatement(g, stack, body[i], errs, strict) + case body[i].IsEvery(): + expr, errs = rewriteEveryStatement(g, stack, body[i], errs, strict) + default: + expr, errs = rewriteDeclaredVarsInExpr(g, stack, body[i], errs, strict) } if expr != nil { cpy.Append(expr) @@ -3954,9 +4452,45 @@ func rewriteDeclaredVarsInBody(g *localVarGenerator, stack *localDeclaredVars, u cpy.Append(NewExpr(BooleanTerm(true))) } + errs = checkUnusedAssignedVars(body[0].Loc(), stack, used, errs, strict) return cpy, checkUnusedDeclaredVars(body[0].Loc(), stack, used, cpy, errs) } +func checkUnusedAssignedVars(loc *Location, stack *localDeclaredVars, used VarSet, errs Errors, strict bool) Errors { + + if !strict || len(errs) > 0 { + return errs + } + + dvs := stack.Peek() + unused := NewVarSet() + + for v, occ := range dvs.occurrence { + // A var that was assigned in this scope must have been seen (used) more than once (the time of assignment) in + // the same, or nested, scope to be counted as used. + if !v.IsWildcard() && occ == assignedVar && stack.Count(v) <= 1 { + unused.Add(dvs.vs[v]) + } + } + + rewrittenUsed := NewVarSet() + for v := range used { + if gv, ok := stack.Declared(v); ok { + rewrittenUsed.Add(gv) + } else { + rewrittenUsed.Add(v) + } + } + + unused = unused.Diff(rewrittenUsed) + + for _, gv := range unused.Sorted() { + errs = append(errs, NewError(CompileErr, loc, "assigned var %v unused", dvs.reverse[gv])) + } + + return errs +} + func checkUnusedDeclaredVars(loc *Location, stack *localDeclaredVars, used VarSet, cpy Body, errs Errors) Errors { // NOTE(tsandall): Do not generate more errors if there are existing @@ -3987,13 +4521,53 @@ func checkUnusedDeclaredVars(loc *Location, stack *localDeclaredVars, used VarSe unused := declared.Diff(bodyvars).Diff(used) for _, gv := range unused.Sorted() { - errs = append(errs, NewError(CompileErr, loc, "declared var %v unused", dvs.reverse[gv])) + rv := dvs.reverse[gv] + if !rv.IsGenerated() { + errs = append(errs, NewError(CompileErr, loc, "declared var %v unused", rv)) + } } return errs } -func rewriteSomeDeclStatement(g *localVarGenerator, stack *localDeclaredVars, expr *Expr, errs Errors) (*Expr, Errors) { +func rewriteEveryStatement(g *localVarGenerator, stack *localDeclaredVars, expr *Expr, errs Errors, strict bool) (*Expr, Errors) { + e := expr.Copy() + every := e.Terms.(*Every) + + errs = rewriteDeclaredVarsInTermRecursive(g, stack, every.Domain, errs, strict) + + stack.Push() + defer stack.Pop() + + // if the key exists, rewrite + if every.Key != nil { + if v := every.Key.Value.(Var); !v.IsWildcard() { + gv, err := rewriteDeclaredVar(g, stack, v, declaredVar) + if err != nil { + return nil, append(errs, NewError(CompileErr, every.Loc(), err.Error())) + } + every.Key.Value = gv + } + } else { // if the key doesn't exist, add dummy local + every.Key = NewTerm(g.Generate()) + } + + // value is always present + if v := every.Value.Value.(Var); !v.IsWildcard() { + gv, err := rewriteDeclaredVar(g, stack, v, declaredVar) + if err != nil { + return nil, append(errs, NewError(CompileErr, every.Loc(), err.Error())) + } + every.Value.Value = gv + } + + used := NewVarSet() + every.Body, errs = rewriteDeclaredVarsInBody(g, stack, used, every.Body, errs, strict) + + return rewriteDeclaredVarsInExpr(g, stack, e, errs, strict) +} + +func rewriteSomeDeclStatement(g *localVarGenerator, stack *localDeclaredVars, expr *Expr, errs Errors, strict bool) (*Expr, Errors) { e := expr.Copy() decl := e.Terms.(*SomeDecl) for i := range decl.Symbols { @@ -4031,20 +4605,20 @@ func rewriteSomeDeclStatement(g *localVarGenerator, stack *localDeclaredVars, ex return nil, append(errs, NewError(CompileErr, decl.Loc(), err.Error())) } } - return rewriteDeclaredVarsInExpr(g, stack, e, errs) + return rewriteDeclaredVarsInExpr(g, stack, e, errs, strict) } } return nil, errs } -func rewriteDeclaredVarsInExpr(g *localVarGenerator, stack *localDeclaredVars, expr *Expr, errs Errors) (*Expr, Errors) { +func rewriteDeclaredVarsInExpr(g *localVarGenerator, stack *localDeclaredVars, expr *Expr, errs Errors, strict bool) (*Expr, Errors) { vis := NewGenericVisitor(func(x interface{}) bool { var stop bool switch x := x.(type) { case *Term: - stop, errs = rewriteDeclaredVarsInTerm(g, stack, x, errs) + stop, errs = rewriteDeclaredVarsInTerm(g, stack, x, errs, strict) case *With: - _, errs = rewriteDeclaredVarsInTerm(g, stack, x.Value, errs) + errs = rewriteDeclaredVarsInTermRecursive(g, stack, x.Value, errs, strict) stop = true } return stop @@ -4053,7 +4627,7 @@ func rewriteDeclaredVarsInExpr(g *localVarGenerator, stack *localDeclaredVars, e return expr, errs } -func rewriteDeclaredAssignment(g *localVarGenerator, stack *localDeclaredVars, expr *Expr, errs Errors) (*Expr, Errors) { +func rewriteDeclaredAssignment(g *localVarGenerator, stack *localDeclaredVars, expr *Expr, errs Errors, strict bool) (*Expr, Errors) { if expr.Negated { errs = append(errs, NewError(CompileErr, expr.Location, "cannot assign vars inside negated expression")) @@ -4069,10 +4643,10 @@ func rewriteDeclaredAssignment(g *localVarGenerator, stack *localDeclaredVars, e // Rewrite terms on right hand side capture seen vars and recursively // process comprehensions before left hand side is processed. Also // rewrite with modifier. - errs = rewriteDeclaredVarsInTermRecursive(g, stack, expr.Operand(1), errs) + errs = rewriteDeclaredVarsInTermRecursive(g, stack, expr.Operand(1), errs, strict) for _, w := range expr.With { - errs = rewriteDeclaredVarsInTermRecursive(g, stack, w.Value, errs) + errs = rewriteDeclaredVarsInTermRecursive(g, stack, w.Value, errs, strict) } // Rewrite vars on left hand side with unique names. Catch redeclaration @@ -4119,11 +4693,12 @@ func rewriteDeclaredAssignment(g *localVarGenerator, stack *localDeclaredVars, e return expr, errs } -func rewriteDeclaredVarsInTerm(g *localVarGenerator, stack *localDeclaredVars, term *Term, errs Errors) (bool, Errors) { +func rewriteDeclaredVarsInTerm(g *localVarGenerator, stack *localDeclaredVars, term *Term, errs Errors, strict bool) (bool, Errors) { switch v := term.Value.(type) { case Var: if gv, ok := stack.Declared(v); ok { term.Value = gv + stack.Seen(v) } else if stack.Occurrence(v) == newVar { stack.Insert(v, v, seenVar) } @@ -4138,69 +4713,90 @@ func rewriteDeclaredVarsInTerm(g *localVarGenerator, stack *localDeclaredVars, t return true, errs } return false, errs + case Call: + ref := v[0] + WalkVars(ref, func(v Var) bool { + if gv, ok := stack.Declared(v); ok && !gv.Equal(v) { + // We will rewrite the ref of a function call, which is never ok since we don't have first-class functions. + errs = append(errs, NewError(CompileErr, term.Location, "called function %s shadowed", ref)) + return true + } + return false + }) + return false, errs case *object: cpy, _ := v.Map(func(k, v *Term) (*Term, *Term, error) { kcpy := k.Copy() - errs = rewriteDeclaredVarsInTermRecursive(g, stack, kcpy, errs) - errs = rewriteDeclaredVarsInTermRecursive(g, stack, v, errs) + errs = rewriteDeclaredVarsInTermRecursive(g, stack, kcpy, errs, strict) + errs = rewriteDeclaredVarsInTermRecursive(g, stack, v, errs, strict) return kcpy, v, nil }) term.Value = cpy case Set: cpy, _ := v.Map(func(elem *Term) (*Term, error) { elemcpy := elem.Copy() - errs = rewriteDeclaredVarsInTermRecursive(g, stack, elemcpy, errs) + errs = rewriteDeclaredVarsInTermRecursive(g, stack, elemcpy, errs, strict) return elemcpy, nil }) term.Value = cpy case *ArrayComprehension: - errs = rewriteDeclaredVarsInArrayComprehension(g, stack, v, errs) + errs = rewriteDeclaredVarsInArrayComprehension(g, stack, v, errs, strict) case *SetComprehension: - errs = rewriteDeclaredVarsInSetComprehension(g, stack, v, errs) + errs = rewriteDeclaredVarsInSetComprehension(g, stack, v, errs, strict) case *ObjectComprehension: - errs = rewriteDeclaredVarsInObjectComprehension(g, stack, v, errs) + errs = rewriteDeclaredVarsInObjectComprehension(g, stack, v, errs, strict) default: return false, errs } return true, errs } -func rewriteDeclaredVarsInTermRecursive(g *localVarGenerator, stack *localDeclaredVars, term *Term, errs Errors) Errors { +func rewriteDeclaredVarsInTermRecursive(g *localVarGenerator, stack *localDeclaredVars, term *Term, errs Errors, strict bool) Errors { WalkNodes(term, func(n Node) bool { var stop bool switch n := n.(type) { case *With: - _, errs = rewriteDeclaredVarsInTerm(g, stack, n.Value, errs) + errs = rewriteDeclaredVarsInTermRecursive(g, stack, n.Value, errs, strict) stop = true case *Term: - stop, errs = rewriteDeclaredVarsInTerm(g, stack, n, errs) + stop, errs = rewriteDeclaredVarsInTerm(g, stack, n, errs, strict) } return stop }) return errs } -func rewriteDeclaredVarsInArrayComprehension(g *localVarGenerator, stack *localDeclaredVars, v *ArrayComprehension, errs Errors) Errors { +func rewriteDeclaredVarsInArrayComprehension(g *localVarGenerator, stack *localDeclaredVars, v *ArrayComprehension, errs Errors, strict bool) Errors { + used := NewVarSet() + used.Update(v.Term.Vars()) + stack.Push() - v.Body, errs = rewriteDeclaredVarsInBody(g, stack, nil, v.Body, errs) - errs = rewriteDeclaredVarsInTermRecursive(g, stack, v.Term, errs) + v.Body, errs = rewriteDeclaredVarsInBody(g, stack, used, v.Body, errs, strict) + errs = rewriteDeclaredVarsInTermRecursive(g, stack, v.Term, errs, strict) stack.Pop() return errs } -func rewriteDeclaredVarsInSetComprehension(g *localVarGenerator, stack *localDeclaredVars, v *SetComprehension, errs Errors) Errors { +func rewriteDeclaredVarsInSetComprehension(g *localVarGenerator, stack *localDeclaredVars, v *SetComprehension, errs Errors, strict bool) Errors { + used := NewVarSet() + used.Update(v.Term.Vars()) + stack.Push() - v.Body, errs = rewriteDeclaredVarsInBody(g, stack, nil, v.Body, errs) - errs = rewriteDeclaredVarsInTermRecursive(g, stack, v.Term, errs) + v.Body, errs = rewriteDeclaredVarsInBody(g, stack, used, v.Body, errs, strict) + errs = rewriteDeclaredVarsInTermRecursive(g, stack, v.Term, errs, strict) stack.Pop() return errs } -func rewriteDeclaredVarsInObjectComprehension(g *localVarGenerator, stack *localDeclaredVars, v *ObjectComprehension, errs Errors) Errors { +func rewriteDeclaredVarsInObjectComprehension(g *localVarGenerator, stack *localDeclaredVars, v *ObjectComprehension, errs Errors, strict bool) Errors { + used := NewVarSet() + used.Update(v.Key.Vars()) + used.Update(v.Value.Vars()) + stack.Push() - v.Body, errs = rewriteDeclaredVarsInBody(g, stack, nil, v.Body, errs) - errs = rewriteDeclaredVarsInTermRecursive(g, stack, v.Key, errs) - errs = rewriteDeclaredVarsInTermRecursive(g, stack, v.Value, errs) + v.Body, errs = rewriteDeclaredVarsInBody(g, stack, used, v.Body, errs, strict) + errs = rewriteDeclaredVarsInTermRecursive(g, stack, v.Key, errs, strict) + errs = rewriteDeclaredVarsInTermRecursive(g, stack, v.Value, errs, strict) stack.Pop() return errs } @@ -4224,10 +4820,10 @@ func rewriteDeclaredVar(g *localVarGenerator, stack *localDeclaredVars, v Var, o // rewriteWithModifiersInBody will rewrite the body so that with modifiers do // not contain terms that require evaluation as values. If this function // encounters an invalid with modifier target then it will raise an error. -func rewriteWithModifiersInBody(c *Compiler, f *equalityFactory, body Body) (Body, *Error) { +func rewriteWithModifiersInBody(c *Compiler, unsafeBuiltinsMap map[string]struct{}, f *equalityFactory, body Body) (Body, *Error) { var result Body for i := range body { - exprs, err := rewriteWithModifier(c, f, body[i]) + exprs, err := rewriteWithModifier(c, unsafeBuiltinsMap, f, body[i]) if err != nil { return nil, err } @@ -4242,63 +4838,122 @@ func rewriteWithModifiersInBody(c *Compiler, f *equalityFactory, body Body) (Bod return result, nil } -func rewriteWithModifier(c *Compiler, f *equalityFactory, expr *Expr) ([]*Expr, *Error) { +func rewriteWithModifier(c *Compiler, unsafeBuiltinsMap map[string]struct{}, f *equalityFactory, expr *Expr) ([]*Expr, *Error) { var result []*Expr for i := range expr.With { - err := validateTarget(c, expr.With[i].Target) + eval, err := validateWith(c, unsafeBuiltinsMap, expr, i) if err != nil { return nil, err } - if requiresEval(expr.With[i].Value) { + if eval { eq := f.Generate(expr.With[i].Value) result = append(result, eq) expr.With[i].Value = eq.Operand(0) } } - // If any of the with modifiers in this expression were rewritten then result - // will be non-empty. In this case, the expression will have been modified and - // it should also be added to the result. - if len(result) > 0 { - result = append(result, expr) - } - return result, nil + return append(result, expr), nil } -func validateTarget(c *Compiler, term *Term) *Error { - if !isInputRef(term) && !isDataRef(term) { - return NewError(TypeErr, term.Location, "with keyword target must start with %v or %v", InputRootDocument, DefaultRootDocument) +func validateWith(c *Compiler, unsafeBuiltinsMap map[string]struct{}, expr *Expr, i int) (bool, *Error) { + target, value := expr.With[i].Target, expr.With[i].Value + + // Ensure that values that are built-ins are rewritten to Ref (not Var) + if v, ok := value.Value.(Var); ok { + if _, ok := c.builtins[v.String()]; ok { + value.Value = Ref([]*Term{NewTerm(v)}) + } + } + isBuiltinRefOrVar, err := isBuiltinRefOrVar(c.builtins, unsafeBuiltinsMap, target) + if err != nil { + return false, err } - if isDataRef(term) { - ref := term.Value.(Ref) + switch { + case isDataRef(target): + ref := target.Value.(Ref) node := c.RuleTree for i := 0; i < len(ref)-1; i++ { child := node.Child(ref[i].Value) if child == nil { break } else if len(child.Values) > 0 { - return NewError(CompileErr, term.Loc(), "with keyword cannot partially replace virtual document(s)") + return false, NewError(CompileErr, target.Loc(), "with keyword cannot partially replace virtual document(s)") } node = child } if node != nil { + // NOTE(sr): at this point in the compiler stages, we don't have a fully-populated + // TypeEnv yet -- so we have to make do with this check to see if the replacement + // target is a function. It's probably wrong for arity-0 functions, but those are + // and edge case anyways. if child := node.Child(ref[len(ref)-1].Value); child != nil { - for _, value := range child.Values { - if len(value.(*Rule).Head.Args) > 0 { - return NewError(CompileErr, term.Loc(), "with keyword cannot replace functions") + for _, v := range child.Values { + if len(v.(*Rule).Head.Args) > 0 { + if ok, err := validateWithFunctionValue(c.builtins, unsafeBuiltinsMap, c.RuleTree, value); err != nil || ok { + return false, err // may be nil + } } } } } + case isInputRef(target): // ok, valid + case isBuiltinRefOrVar: + + // NOTE(sr): first we ensure that parsed Var builtins (`count`, `concat`, etc) + // are rewritten to their proper Ref convention + if v, ok := target.Value.(Var); ok { + target.Value = Ref([]*Term{NewTerm(v)}) + } + targetRef := target.Value.(Ref) + bi := c.builtins[targetRef.String()] // safe because isBuiltinRefOrVar checked this + if err := validateWithBuiltinTarget(bi, targetRef, target.Loc()); err != nil { + return false, err + } + + if ok, err := validateWithFunctionValue(c.builtins, unsafeBuiltinsMap, c.RuleTree, value); err != nil || ok { + return false, err // may be nil + } + default: + return false, NewError(TypeErr, target.Location, "with keyword target must reference existing %v, %v, or a function", InputRootDocument, DefaultRootDocument) + } + return requiresEval(value), nil +} + +func validateWithBuiltinTarget(bi *Builtin, target Ref, loc *location.Location) *Error { + switch bi.Name { + case Equality.Name, + RegoMetadataChain.Name, + RegoMetadataRule.Name: + return NewError(CompileErr, loc, "with keyword replacing built-in function: replacement of %q invalid", bi.Name) + } + + switch { + case target.HasPrefix(Ref([]*Term{VarTerm("internal")})): + return NewError(CompileErr, loc, "with keyword replacing built-in function: replacement of internal function %q invalid", target) + + case bi.Relation: + return NewError(CompileErr, loc, "with keyword replacing built-in function: target must not be a relation") + + case bi.Decl.Result() == nil: + return NewError(CompileErr, loc, "with keyword replacing built-in function: target must not be a void function") } return nil } +func validateWithFunctionValue(bs map[string]*Builtin, unsafeMap map[string]struct{}, ruleTree *TreeNode, value *Term) (bool, *Error) { + if v, ok := value.Value.(Ref); ok { + if ruleTree.Find(v) != nil { // ref exists in rule tree + return true, nil + } + } + return isBuiltinRefOrVar(bs, unsafeMap, value) +} + func isInputRef(term *Term) bool { if ref, ok := term.Value.(Ref); ok { if ref.HasPrefix(InputRootRef) { @@ -4317,6 +4972,18 @@ func isDataRef(term *Term) bool { return false } +func isBuiltinRefOrVar(bs map[string]*Builtin, unsafeBuiltinsMap map[string]struct{}, term *Term) (bool, *Error) { + switch v := term.Value.(type) { + case Ref, Var: + if _, ok := unsafeBuiltinsMap[v.String()]; ok { + return false, NewError(CompileErr, term.Location, "with keyword replacing built-in function: target must not be unsafe: %q", v) + } + _, ok := bs[v.String()] + return ok, nil + } + return false, nil +} + func isVirtual(node *TreeNode, ref Ref) bool { for i := 0; i < len(ref); i++ { child := node.Child(ref[i].Value) @@ -4396,6 +5063,25 @@ func checkUnsafeBuiltins(unsafeBuiltinsMap map[string]struct{}, node interface{} return errs } +func checkDeprecatedBuiltins(deprecatedBuiltinsMap map[string]struct{}, node interface{}, strict bool) Errors { + // Early out; deprecatedBuiltinsMap is only populated in strict-mode. + if !strict { + return nil + } + + errs := make(Errors, 0) + WalkExprs(node, func(x *Expr) bool { + if x.IsCall() { + operator := x.Operator().String() + if _, ok := deprecatedBuiltinsMap[operator]; ok { + errs = append(errs, NewError(TypeErr, x.Loc(), "deprecated built-in function calls in expression: %v", operator)) + } + } + return false + }) + return errs +} + func rewriteVarsInRef(vars ...map[Var]Var) varRewriter { return func(node Ref) Ref { i, _ := TransformVars(node, func(v Var) (Value, error) { diff --git a/vendor/github.com/open-policy-agent/opa/ast/compilehelper.go b/vendor/github.com/open-policy-agent/opa/ast/compilehelper.go index ca75dfab..dd48884f 100644 --- a/vendor/github.com/open-policy-agent/opa/ast/compilehelper.go +++ b/vendor/github.com/open-policy-agent/opa/ast/compilehelper.go @@ -13,6 +13,7 @@ func CompileModules(modules map[string]string) (*Compiler, error) { // CompileOpts defines a set of options for the compiler. type CompileOpts struct { EnablePrintStatements bool + ParserOptions ParserOptions } // CompileModulesWithOpt takes a set of Rego modules represented as strings and @@ -24,7 +25,7 @@ func CompileModulesWithOpt(modules map[string]string, opts CompileOpts) (*Compil for f, module := range modules { var pm *Module var err error - if pm, err = ParseModule(f, module); err != nil { + if pm, err = ParseModuleWithOpts(f, module, opts.ParserOptions); err != nil { return nil, err } parsed[f] = pm diff --git a/vendor/github.com/open-policy-agent/opa/ast/errors.go b/vendor/github.com/open-policy-agent/opa/ast/errors.go index 11348b3d..066dfcdd 100644 --- a/vendor/github.com/open-policy-agent/opa/ast/errors.go +++ b/vendor/github.com/open-policy-agent/opa/ast/errors.go @@ -121,12 +121,3 @@ func NewError(code string, loc *Location, f string, a ...interface{}) *Error { Message: fmt.Sprintf(f, a...), } } - -var ( - errPartialRuleAssignOperator = fmt.Errorf("partial rules must use = operator (not := operator)") - errFunctionAssignOperator = fmt.Errorf("functions must use = operator (not := operator)") -) - -func errTermAssignOperator(x interface{}) error { - return fmt.Errorf("cannot assign to %v", TypeName(x)) -} diff --git a/vendor/github.com/open-policy-agent/opa/ast/fuzz.go b/vendor/github.com/open-policy-agent/opa/ast/fuzz.go index 6ff7e35a..10651d31 100644 --- a/vendor/github.com/open-policy-agent/opa/ast/fuzz.go +++ b/vendor/github.com/open-policy-agent/opa/ast/fuzz.go @@ -1,3 +1,4 @@ +//go:build gofuzz // +build gofuzz package ast diff --git a/vendor/github.com/open-policy-agent/opa/ast/internal/scanner/scanner.go b/vendor/github.com/open-policy-agent/opa/ast/internal/scanner/scanner.go index 9402749e..7b3f22f0 100644 --- a/vendor/github.com/open-policy-agent/opa/ast/internal/scanner/scanner.go +++ b/vendor/github.com/open-policy-agent/opa/ast/internal/scanner/scanner.go @@ -96,6 +96,11 @@ func (s *Scanner) Keyword(lit string) tokens.Token { // AddKeyword adds a string -> token mapping to this Scanner instance. func (s *Scanner) AddKeyword(kw string, tok tokens.Token) { s.keywords[kw] = tok + + switch tok { + case tokens.Every: // importing 'every' means also importing 'in' + s.keywords["in"] = tokens.In + } } // WithKeywords returns a new copy of the Scanner struct `s`, with the set @@ -112,6 +117,21 @@ func (s *Scanner) WithKeywords(kws map[string]tokens.Token) *Scanner { return &cpy } +// WithoutKeywords returns a new copy of the Scanner struct `s`, with the +// set of known keywords being that of `s` with `kws` removed. +// The previously known keywords are returned for a convenient reset. +func (s *Scanner) WithoutKeywords(kws map[string]tokens.Token) (*Scanner, map[string]tokens.Token) { + cpy := *s + kw := s.keywords + cpy.keywords = make(map[string]tokens.Token, len(s.keywords)-len(kws)) + for kw, tok := range s.keywords { + if _, ok := kws[kw]; !ok { + cpy.AddKeyword(kw, tok) + } + } + return &cpy, kw +} + // Scan will increment the scanners position in the source // code until the next token is found. The token, starting position // of the token, string literal, and any errors encountered are diff --git a/vendor/github.com/open-policy-agent/opa/ast/internal/tokens/tokens.go b/vendor/github.com/open-policy-agent/opa/ast/internal/tokens/tokens.go index ce053c18..29bf971d 100644 --- a/vendor/github.com/open-policy-agent/opa/ast/internal/tokens/tokens.go +++ b/vendor/github.com/open-policy-agent/opa/ast/internal/tokens/tokens.go @@ -65,6 +65,10 @@ const ( Lte Dot Semicolon + + Every + Contains + If ) var strings = [...]string{ @@ -112,6 +116,9 @@ var strings = [...]string{ Lte: "lte", Dot: ".", Semicolon: ";", + Every: "every", + Contains: "contains", + If: "if", } var keywords = map[string]Token{ @@ -136,3 +143,9 @@ func Keywords() map[string]Token { } return cpy } + +// IsKeyword returns if a token is a keyword +func IsKeyword(tok Token) bool { + _, ok := keywords[strings[tok]] + return ok +} diff --git a/vendor/github.com/open-policy-agent/opa/ast/location/location.go b/vendor/github.com/open-policy-agent/opa/ast/location/location.go index 13ae6e35..5bdce013 100644 --- a/vendor/github.com/open-policy-agent/opa/ast/location/location.go +++ b/vendor/github.com/open-policy-agent/opa/ast/location/location.go @@ -39,7 +39,7 @@ func (loc *Location) Errorf(f string, a ...interface{}) error { // Wrapf returns a new error value that wraps an existing error with a message formatted // to include the location info (e.g., line, column, filename, etc.) func (loc *Location) Wrapf(err error, f string, a ...interface{}) error { - return errors.Wrap(err, loc.Format(f, a...)) + return fmt.Errorf(loc.Format(f, a...)+": %w", err) } // Format returns a formatted string prefixed with the location information. diff --git a/vendor/github.com/open-policy-agent/opa/ast/parser.go b/vendor/github.com/open-policy-agent/opa/ast/parser.go index 0990a6a5..fd1b4072 100644 --- a/vendor/github.com/open-policy-agent/opa/ast/parser.go +++ b/vendor/github.com/open-policy-agent/opa/ast/parser.go @@ -10,11 +10,12 @@ import ( "fmt" "io" "math/big" + "net/url" "regexp" + "sort" "strconv" "strings" - "github.com/pkg/errors" "gopkg.in/yaml.v2" "github.com/open-policy-agent/opa/ast/internal/scanner" @@ -94,10 +95,12 @@ func (e *parsedTermCacheItem) String() string { // ParserOptions defines the options for parsing Rego statements. type ParserOptions struct { - Capabilities *Capabilities - ProcessAnnotation bool - AllFutureKeywords bool - FutureKeywords []string + Capabilities *Capabilities + ProcessAnnotation bool + AllFutureKeywords bool + FutureKeywords []string + SkipRules bool + unreleasedKeywords bool // TODO(sr): cleanup } // NewParser creates and initializes a Parser. @@ -133,12 +136,12 @@ func (p *Parser) WithProcessAnnotation(processAnnotation bool) *Parser { // WithFutureKeywords enables "future" keywords, i.e., keywords that can // be imported via // -// import future.keywords.kw -// import future.keywords.other +// import future.keywords.kw +// import future.keywords.other // // but in a more direct way. The equivalent of this import would be // -// WithFutureKeywords("kw", "other") +// WithFutureKeywords("kw", "other") func (p *Parser) WithFutureKeywords(kws ...string) *Parser { p.po.FutureKeywords = kws return p @@ -147,25 +150,31 @@ func (p *Parser) WithFutureKeywords(kws ...string) *Parser { // WithAllFutureKeywords enables all "future" keywords, i.e., the // ParserOption equivalent of // -// import future.keywords +// import future.keywords func (p *Parser) WithAllFutureKeywords(yes bool) *Parser { p.po.AllFutureKeywords = yes return p } +// withUnreleasedKeywords allows using keywords that haven't surfaced +// as future keywords (see above) yet, but have tests that require +// them to be parsed +func (p *Parser) withUnreleasedKeywords(yes bool) *Parser { + p.po.unreleasedKeywords = yes + return p +} + // WithCapabilities sets the capabilities structure on the parser. func (p *Parser) WithCapabilities(c *Capabilities) *Parser { p.po.Capabilities = c return p } -const ( - annotationScopePackage = "package" - annotationScopeImport = "import" - annotationScopeRule = "rule" - annotationScopeDocument = "document" - annotationScopeSubpackages = "subpackages" -) +// WithSkipRules instructs the parser not to attempt to parse Rule statements. +func (p *Parser) WithSkipRules(skip bool) *Parser { + p.po.SkipRules = skip + return p +} func (p *Parser) parsedTermCacheLookup() (*Term, *state) { l := p.s.loc.Offset @@ -206,6 +215,24 @@ func (p *Parser) futureParser() *Parser { return &q } +// presentParser returns a shallow copy of `p` with an empty +// cache, and a scanner that knows none of the future keywords. +// It is used to successfully parse keyword imports, like +// +// import future.keywords.in +// +// even when the parser has already been informed about the +// future keyword "in". This parser won't error out because +// "in" is an identifier. +func (p *Parser) presentParser() (*Parser, map[string]tokens.Token) { + var cpy map[string]tokens.Token + q := *p + q.s = p.save() + q.s.s, cpy = p.s.s.WithoutKeywords(futureKeywords) + q.cache = parsedTermCache{} + return &q, cpy +} + // Parse will read the Rego source and parse statements and // comments as they are found. Any errors encountered while // parsing will be accumulated and returned as a list of Errors. @@ -301,18 +328,21 @@ func (p *Parser) Parse() ([]Statement, []*Comment, Errors) { } p.restore(s) - s = p.save() - if rules := p.parseRules(); rules != nil { - for i := range rules { - stmts = append(stmts, rules[i]) + if !p.po.SkipRules { + s = p.save() + + if rules := p.parseRules(); rules != nil { + for i := range rules { + stmts = append(stmts, rules[i]) + } + continue + } else if len(p.s.errors) > 0 { + break } - continue - } else if len(p.s.errors) > 0 { - break - } - p.restore(s) + p.restore(s) + } if body := p.parseQuery(true, tokens.EOF); body != nil { stmts = append(stmts, body) @@ -331,34 +361,54 @@ func (p *Parser) Parse() ([]Statement, []*Comment, Errors) { func (p *Parser) parseAnnotations(stmts []Statement) []Statement { + annotStmts, errs := parseAnnotations(p.s.comments) + for _, err := range errs { + p.error(err.Location, err.Message) + } + + for _, annotStmt := range annotStmts { + stmts = append(stmts, annotStmt) + } + + return stmts +} + +func parseAnnotations(comments []*Comment) ([]*Annotations, Errors) { + var hint = []byte("METADATA") var curr *metadataParser var blocks []*metadataParser - for i := 0; i < len(p.s.comments); i++ { + for i := 0; i < len(comments); i++ { if curr != nil { - if p.s.comments[i].Location.Row == p.s.comments[i-1].Location.Row+1 && p.s.comments[i].Location.Col == 1 { - curr.Append(p.s.comments[i]) + if comments[i].Location.Row == comments[i-1].Location.Row+1 && comments[i].Location.Col == 1 { + curr.Append(comments[i]) continue } curr = nil } - if bytes.HasPrefix(bytes.TrimSpace(p.s.comments[i].Text), hint) { - curr = newMetadataParser(p.s.comments[i].Location) + if bytes.HasPrefix(bytes.TrimSpace(comments[i].Text), hint) { + curr = newMetadataParser(comments[i].Location) blocks = append(blocks, curr) } } + var stmts []*Annotations + var errs Errors for _, b := range blocks { a, err := b.Parse() if err != nil { - p.error(b.loc, err.Error()) + errs = append(errs, &Error{ + Code: ParseErr, + Message: err.Error(), + Location: b.loc, + }) } else { stmts = append(stmts, a) } } - return stmts + return stmts, errs } func (p *Parser) parsePackage() *Package { @@ -433,8 +483,8 @@ func (p *Parser) parseImport() *Import { p.error(p.s.Loc(), "expected ident") return nil } - - term := p.parseTerm() + q, prev := p.presentParser() + term := q.parseTerm() if term != nil { switch v := term.Value.(type) { case Var: @@ -449,6 +499,9 @@ func (p *Parser) parseImport() *Import { imp.Path = term } } + // keep advanced parser state, reset known keywords + p.s = q.s + p.s.s = q.s.s.WithKeywords(prev) if imp.Path == nil { p.error(p.s.Loc(), "expected path") @@ -500,7 +553,8 @@ func (p *Parser) parseRules() []*Rule { return nil } - if rule.Head = p.parseHead(rule.Default); rule.Head == nil { + usesContains := false + if rule.Head, usesContains = p.parseHead(rule.Default); rule.Head == nil { return nil } @@ -513,23 +567,57 @@ func (p *Parser) parseRules() []*Rule { return []*Rule{&rule} } - if p.s.tok == tokens.LBrace { + hasIf := false + if p.s.tok == tokens.If { + hasIf = true + } + + if hasIf && !usesContains && rule.Head.Key != nil && rule.Head.Value == nil { + p.illegal("invalid for partial set rule %s (use `contains`)", rule.Head.Name) + return nil + } + + switch { + case hasIf: + p.scan() + s := p.save() + if expr := p.parseLiteral(); expr != nil { + // NOTE(sr): set literals are never false or undefined, so parsing this as + // p if { true } + // ^^^^^^^^ set of one element, `true` + // isn't valid. + isSetLiteral := false + if t, ok := expr.Terms.(*Term); ok { + _, isSetLiteral = t.Value.(Set) + } + // expr.Term is []*Term or Every + if !isSetLiteral { + rule.Body.Append(expr) + break + } + } + + // parsing as literal didn't work out, expect '{ BODY }' + p.restore(s) + fallthrough + + case p.s.tok == tokens.LBrace: p.scan() if rule.Body = p.parseBody(tokens.RBrace); rule.Body == nil { return nil } p.scan() - } else { + + case usesContains: + rule.Body = NewBody(NewExpr(BooleanTerm(true).SetLocation(rule.Location)).SetLocation(rule.Location)) + return []*Rule{&rule} + + default: return nil } if p.s.tok == tokens.Else { - if rule.Head.Assign { - p.error(p.s.Loc(), "else keyword cannot be used on rule declared with := operator") - return nil - } - if rule.Head.Key != nil { p.error(p.s.Loc(), "else keyword cannot be used on partial rules") return nil @@ -593,9 +681,9 @@ func (p *Parser) parseElse(head *Head) *Rule { p.scan() switch p.s.tok { - case tokens.LBrace: + case tokens.LBrace, tokens.If: // no value, but a body follows directly rule.Head.Value = BooleanTerm(true) - case tokens.Unify: + case tokens.Assign, tokens.Unify: p.scan() rule.Head.Value = p.parseTermInfixCall() if rule.Head.Value == nil { @@ -607,6 +695,30 @@ func (p *Parser) parseElse(head *Head) *Rule { return nil } + hasIf := p.s.tok == tokens.If + + if hasIf { + p.scan() + s := p.save() + if expr := p.parseLiteral(); expr != nil { + // NOTE(sr): set literals are never false or undefined, so parsing this as + // p if false else if { true } + // ^^^^^^^^ set of one element, `true` + // isn't valid. + isSetLiteral := false + if t, ok := expr.Terms.(*Term); ok { + _, isSetLiteral = t.Value.(Set) + } + // expr.Term is []*Term or Every + if !isSetLiteral { + rule.Body.Append(expr) + setLocRecursive(rule.Body, rule.Location) + return &rule + } + } + p.restore(s) + } + if p.s.tok != tokens.LBrace { rule.Body = NewBody(NewExpr(BooleanTerm(true))) setLocRecursive(rule.Body, rule.Location) @@ -629,7 +741,7 @@ func (p *Parser) parseElse(head *Head) *Rule { return &rule } -func (p *Parser) parseHead(defaultRule bool) *Head { +func (p *Parser) parseHead(defaultRule bool) (*Head, bool) { var head Head head.SetLoc(p.s.Loc()) @@ -651,13 +763,13 @@ func (p *Parser) parseHead(defaultRule bool) *Head { if p.s.tok != tokens.RParen { head.Args = p.parseTermList(tokens.RParen, nil) if head.Args == nil { - return nil + return nil, false } } p.scan() if p.s.tok == tokens.LBrack { - return nil + return nil, false } } @@ -676,30 +788,39 @@ func (p *Parser) parseHead(defaultRule bool) *Head { p.scan() } - if p.s.tok == tokens.Unify { + switch p.s.tok { + case tokens.Contains: + p.scan() + head.Key = p.parseTermInfixCall() + if head.Key == nil { + p.illegal("expected rule key term (e.g., %s contains { ... })", head.Name) + } + + return &head, true + case tokens.Unify: p.scan() head.Value = p.parseTermInfixCall() if head.Value == nil { p.illegal("expected rule value term (e.g., %s[%s] = { ... })", head.Name, head.Key) } - } else if p.s.tok == tokens.Assign { - - if defaultRule { - p.error(p.s.Loc(), "default rules must use = operator (not := operator)") - return nil - } else if head.Key != nil { - p.error(p.s.Loc(), "partial rules must use = operator (not := operator)") - return nil - } else if len(head.Args) > 0 { - p.error(p.s.Loc(), "functions must use = operator (not := operator)") - return nil - } + case tokens.Assign: + s := p.save() p.scan() head.Assign = true head.Value = p.parseTermInfixCall() if head.Value == nil { - p.illegal("expected rule value term (e.g., %s := { ... })", head.Name) + p.restore(s) + switch { + case len(head.Args) > 0: + p.illegal("expected function value term (e.g., %s(...) := { ... })", head.Name) + case head.Key != nil: + p.illegal("expected partial rule value term (e.g., %s[...] := { ... })", head.Name) + case defaultRule: + p.illegal("expected default rule value term (e.g., default %s := )", head.Name) + default: + p.illegal("expected rule value term (e.g., %s := { ... })", head.Name) + } } } @@ -707,7 +828,7 @@ func (p *Parser) parseHead(defaultRule bool) *Head { head.Value = BooleanTerm(true).SetLocation(head.Location) } - return &head + return &head, false } func (p *Parser) parseBody(end tokens.Token) Body { @@ -763,14 +884,26 @@ func (p *Parser) parseLiteral() (expr *Expr) { }() var negated bool + if p.s.tok == tokens.Not { + p.scan() + negated = true + } + switch p.s.tok { case tokens.Some: + if negated { + p.illegal("illegal negation of 'some'") + return nil + } return p.parseSome() - case tokens.Not: - p.scan() - negated = true - fallthrough + case tokens.Every: + if negated { + p.illegal("illegal negation of 'every'") + return nil + } + return p.parseEvery() default: + s := p.save() expr := p.parseExpr() if expr != nil { expr.Negated = negated @@ -779,6 +912,20 @@ func (p *Parser) parseLiteral() (expr *Expr) { return nil } } + // If we find a plain `every` identifier, attempt to parse an every expression, + // add hint if it succeeds. + if term, ok := expr.Terms.(*Term); ok && Var("every").Equal(term.Value) { + var hint bool + t := p.save() + p.restore(s) + if expr := p.futureParser().parseEvery(); expr != nil { + _, hint = expr.Terms.(*Every) + } + p.restore(t) + if hint { + p.hint("`import future.keywords.every` for `every x in xs { ... }` expressions") + } + } return expr } return nil @@ -801,7 +948,8 @@ func (p *Parser) parseWith() []*With { return nil } - if with.Target = p.parseTerm(); with.Target == nil { + with.Target = p.parseTerm() + if with.Target == nil { return nil } @@ -847,14 +995,29 @@ func (p *Parser) parseSome() *Expr { if term := p.parseTermInfixCall(); term != nil { if call, ok := term.Value.(Call); ok { switch call[0].String() { - case Member.Name, MemberWithKey.Name: // OK + case Member.Name: + if len(call) != 3 { + p.illegal("illegal domain") + return nil + } + case MemberWithKey.Name: + if len(call) != 4 { + p.illegal("illegal domain") + return nil + } default: p.illegal("expected `x in xs` or `x, y in xs` expression") return nil } decl.Symbols = []*Term{term} - return NewExpr(decl).SetLocation(decl.Location) + expr := NewExpr(decl).SetLocation(decl.Location) + if p.s.tok == tokens.With { + if expr.With = p.parseWith(); expr.With == nil { + return nil + } + } + return expr } } @@ -898,6 +1061,72 @@ func (p *Parser) parseSome() *Expr { return NewExpr(decl).SetLocation(decl.Location) } +func (p *Parser) parseEvery() *Expr { + qb := &Every{} + qb.SetLoc(p.s.Loc()) + + // TODO(sr): We'd get more accurate error messages if we didn't rely on + // parseTermInfixCall here, but parsed "var [, var] in term" manually. + p.scan() + term := p.parseTermInfixCall() + if term == nil { + return nil + } + call, ok := term.Value.(Call) + if !ok { + p.illegal("expected `x[, y] in xs { ... }` expression") + return nil + } + switch call[0].String() { + case Member.Name: // x in xs + if len(call) != 3 { + p.illegal("illegal domain") + return nil + } + qb.Value = call[1] + qb.Domain = call[2] + case MemberWithKey.Name: // k, v in xs + if len(call) != 4 { + p.illegal("illegal domain") + return nil + } + qb.Key = call[1] + qb.Value = call[2] + qb.Domain = call[3] + if _, ok := qb.Key.Value.(Var); !ok { + p.illegal("expected key to be a variable") + return nil + } + default: + p.illegal("expected `x[, y] in xs { ... }` expression") + return nil + } + if _, ok := qb.Value.Value.(Var); !ok { + p.illegal("expected value to be a variable") + return nil + } + if p.s.tok == tokens.LBrace { // every x in xs { ... } + p.scan() + body := p.parseBody(tokens.RBrace) + if body == nil { + return nil + } + p.scan() + qb.Body = body + expr := NewExpr(qb).SetLocation(qb.Location) + + if p.s.tok == tokens.With { + if expr.With = p.parseWith(); expr.With == nil { + return nil + } + } + return expr + } + + p.illegal("missing body") + return nil +} + func (p *Parser) parseExpr() *Expr { lhs := p.parseTermInfixCall() @@ -961,7 +1190,6 @@ func (p *Parser) parseTermIn(lhs *Term, keyVal bool, offset int) *Term { } } p.restore(s) - return nil } if op := p.parseTermOpName(Member.Ref(), tokens.In); op != nil { if rhs := p.parseTermRelation(nil, p.s.loc.Offset); rhs != nil { @@ -1099,7 +1327,7 @@ func (p *Parser) parseTerm() *Term { term = p.parseNumber() case tokens.String: term = p.parseString() - case tokens.Ident: + case tokens.Ident, tokens.Contains: // NOTE(sr): contains anywhere BUT in rule heads gets no special treatment term = p.parseVar() case tokens.LBrack: term = p.parseArray() @@ -1717,7 +1945,10 @@ func (p *Parser) illegal(note string, a ...interface{}) { } tokType := "token" - if p.s.tok >= tokens.Package && p.s.tok <= tokens.False { + if tokens.IsKeyword(p.s.tok) { + tokType = "keyword" + } + if _, ok := futureKeywords[p.s.tok.String()]; ok { tokType = "keyword" } @@ -1848,9 +2079,17 @@ func (p *Parser) validateDefaultRuleValue(rule *Rule) bool { return valid } +// We explicitly use yaml unmarshalling, to accommodate for the '_' in 'related_resources', +// which isn't handled properly by json for some reason. type rawAnnotation struct { - Scope string `json:"scope"` - Schemas []rawSchemaAnnotation `json:"schemas"` + Scope string `yaml:"scope"` + Title string `yaml:"title"` + Description string `yaml:"description"` + Organizations []string `yaml:"organizations"` + RelatedResources []interface{} `yaml:"related_resources"` + Authors []interface{} `yaml:"authors"` + Schemas []rawSchemaAnnotation `yaml:"schemas"` + Custom map[string]interface{} `yaml:"custom"` } type rawSchemaAnnotation map[string]interface{} @@ -1899,12 +2138,20 @@ func (b *metadataParser) Parse() (*Annotations, error) { var result Annotations result.Scope = raw.Scope + result.Title = raw.Title + result.Description = raw.Description + result.Organizations = raw.Organizations - for _, pair := range raw.Schemas { - var k string - var v interface{} - for k, v = range pair { + for _, v := range raw.RelatedResources { + rr, err := parseRelatedResource(v) + if err != nil { + return nil, fmt.Errorf("invalid related-resource definition %s: %w", v, err) } + result.RelatedResources = append(result.RelatedResources, rr) + } + + for _, pair := range raw.Schemas { + k, v := unwrapPair(pair) var a SchemaAnnotation var err error @@ -1923,7 +2170,7 @@ func (b *metadataParser) Parse() (*Annotations, error) { case map[interface{}]interface{}: w, err := convertYAMLMapKeyTypes(v, nil) if err != nil { - return nil, errors.Wrap(err, "invalid schema definition") + return nil, fmt.Errorf("invalid schema definition: %w", err) } a.Definition = &w default: @@ -1933,10 +2180,33 @@ func (b *metadataParser) Parse() (*Annotations, error) { result.Schemas = append(result.Schemas, &a) } + for _, v := range raw.Authors { + author, err := parseAuthor(v) + if err != nil { + return nil, fmt.Errorf("invalid author definition %s: %w", v, err) + } + result.Authors = append(result.Authors, author) + } + + result.Custom = make(map[string]interface{}) + for k, v := range raw.Custom { + val, err := convertYAMLMapKeyTypes(v, nil) + if err != nil { + return nil, err + } + result.Custom[k] = val + } + result.Location = b.loc return &result, nil } +func unwrapPair(pair map[string]interface{}) (k string, v interface{}) { + for k, v = range pair { + } + return +} + var errInvalidSchemaRef = fmt.Errorf("invalid schema reference") // NOTE(tsandall): 'schema' is not registered as a root because it's not @@ -1961,6 +2231,96 @@ func parseSchemaRef(s string) (Ref, error) { return nil, errInvalidSchemaRef } +func parseRelatedResource(rr interface{}) (*RelatedResourceAnnotation, error) { + rr, err := convertYAMLMapKeyTypes(rr, nil) + if err != nil { + return nil, err + } + + switch rr := rr.(type) { + case string: + if len(rr) > 0 { + u, err := url.Parse(rr) + if err != nil { + return nil, err + } + return &RelatedResourceAnnotation{Ref: *u}, nil + } + return nil, fmt.Errorf("ref URL may not be empty string") + case map[string]interface{}: + description := strings.TrimSpace(getSafeString(rr, "description")) + ref := strings.TrimSpace(getSafeString(rr, "ref")) + if len(ref) > 0 { + u, err := url.Parse(ref) + if err != nil { + return nil, err + } + return &RelatedResourceAnnotation{Description: description, Ref: *u}, nil + } + return nil, fmt.Errorf("'ref' value required in object") + } + + return nil, fmt.Errorf("invalid value type, must be string or map") +} + +func parseAuthor(a interface{}) (*AuthorAnnotation, error) { + a, err := convertYAMLMapKeyTypes(a, nil) + if err != nil { + return nil, err + } + + switch a := a.(type) { + case string: + return parseAuthorString(a) + case map[string]interface{}: + name := strings.TrimSpace(getSafeString(a, "name")) + email := strings.TrimSpace(getSafeString(a, "email")) + if len(name) > 0 || len(email) > 0 { + return &AuthorAnnotation{name, email}, nil + } + return nil, fmt.Errorf("'name' and/or 'email' values required in object") + } + + return nil, fmt.Errorf("invalid value type, must be string or map") +} + +func getSafeString(m map[string]interface{}, k string) string { + if v, found := m[k]; found { + if s, ok := v.(string); ok { + return s + } + } + return "" +} + +const emailPrefix = "<" +const emailSuffix = ">" + +// parseAuthor parses a string into an AuthorAnnotation. If the last word of the input string is enclosed within <>, +// it is extracted as the author's email. The email may not contain whitelines, as it then will be interpreted as +// multiple words. +func parseAuthorString(s string) (*AuthorAnnotation, error) { + parts := strings.Fields(s) + + if len(parts) == 0 { + return nil, fmt.Errorf("author is an empty string") + } + + namePartCount := len(parts) + trailing := parts[namePartCount-1] + var email string + if len(trailing) >= len(emailPrefix)+len(emailSuffix) && strings.HasPrefix(trailing, emailPrefix) && + strings.HasSuffix(trailing, emailSuffix) { + email = trailing[len(emailPrefix):] + email = email[0 : len(email)-len(emailSuffix)] + namePartCount = namePartCount - 1 + } + + name := strings.Join(parts[0:namePartCount], " ") + + return &AuthorAnnotation{Name: name, Email: email}, nil +} + func convertYAMLMapKeyTypes(x interface{}, path []string) (interface{}, error) { var err error switch x := x.(type) { @@ -1993,23 +2353,22 @@ func convertYAMLMapKeyTypes(x interface{}, path []string) (interface{}, error) { // futureKeywords is the source of truth for future keywords that will // eventually become standard keywords inside of Rego. var futureKeywords = map[string]tokens.Token{ - "in": tokens.In, + "in": tokens.In, + "every": tokens.Every, + "contains": tokens.Contains, + "if": tokens.If, } func (p *Parser) futureImport(imp *Import, allowedFutureKeywords map[string]tokens.Token) { path := imp.Path.Value.(Ref) - if len(path) == 1 { - p.errorf(imp.Path.Location, "invalid import, use `import future.keywords` or `import.future.keywords.in`") - return - } - if !path[1].Equal(StringTerm("keywords")) { + if len(path) == 1 || !path[1].Equal(StringTerm("keywords")) { p.errorf(imp.Path.Location, "invalid import, must be `future.keywords`") return } if imp.Alias != "" { - p.errorf(imp.Path.Location, "future keyword imports cannot be aliased") + p.errorf(imp.Path.Location, "`future` imports cannot be aliased") return } @@ -2017,6 +2376,7 @@ func (p *Parser) futureImport(imp *Import, allowedFutureKeywords map[string]toke for k := range allowedFutureKeywords { kwds = append(kwds, k) } + switch len(path) { case 2: // all keywords imported, nothing to do case 3: // one keyword imported @@ -2028,6 +2388,7 @@ func (p *Parser) futureImport(imp *Import, allowedFutureKeywords map[string]toke keyword := string(kw) _, ok = allowedFutureKeywords[keyword] if !ok { + sort.Strings(kwds) // so the error message is stable p.errorf(imp.Path.Location, "unexpected keyword, must be one of %v", kwds) return } diff --git a/vendor/github.com/open-policy-agent/opa/ast/parser_ext.go b/vendor/github.com/open-policy-agent/opa/ast/parser_ext.go index 749dbd62..41eb4443 100644 --- a/vendor/github.com/open-policy-agent/opa/ast/parser_ext.go +++ b/vendor/github.com/open-policy-agent/opa/ast/parser_ext.go @@ -12,17 +12,22 @@ package ast import ( "bytes" + "errors" "fmt" "strings" "unicode" - - "github.com/pkg/errors" ) // MustParseBody returns a parsed body. // If an error occurs during parsing, panic. func MustParseBody(input string) Body { - parsed, err := ParseBody(input) + return MustParseBodyWithOpts(input, ParserOptions{}) +} + +// MustParseBodyWithOpts returns a parsed body. +// If an error occurs during parsing, panic. +func MustParseBodyWithOpts(input string, opts ParserOptions) Body { + parsed, err := ParseBodyWithOpts(input, opts) if err != nil { panic(err) } @@ -52,7 +57,13 @@ func MustParseImports(input string) []*Import { // MustParseModule returns a parsed module. // If an error occurs during parsing, panic. func MustParseModule(input string) *Module { - parsed, err := ParseModule("", input) + return MustParseModuleWithOpts(input, ParserOptions{}) +} + +// MustParseModuleWithOpts returns a parsed module. +// If an error occurs during parsing, panic. +func MustParseModuleWithOpts(input string, opts ParserOptions) *Module { + parsed, err := ParseModuleWithOpts("", input, opts) if err != nil { panic(err) } @@ -161,30 +172,15 @@ func ParseRuleFromExpr(module *Module, expr *Expr) (*Rule, error) { return nil, errors.New("expression cannot be used for rule head") } - if expr.IsAssignment() { - - lhs, rhs := expr.Operand(0), expr.Operand(1) - if lhs == nil || rhs == nil { - return nil, errors.New("assignment requires two operands") - } - - rule, err := ParseCompleteDocRuleFromAssignmentExpr(module, lhs, rhs) - - if err == nil { - rule.Location = expr.Location - rule.Head.Location = expr.Location - return rule, nil - } else if _, ok := lhs.Value.(Call); ok { - return nil, errFunctionAssignOperator - } else if _, ok := lhs.Value.(Ref); ok { - return nil, errPartialRuleAssignOperator - } - - return nil, errTermAssignOperator(lhs.Value) - } - if expr.IsEquality() { return parseCompleteRuleFromEq(module, expr) + } else if expr.IsAssignment() { + rule, err := parseCompleteRuleFromEq(module, expr) + if err != nil { + return nil, err + } + rule.Head.Assign = true + return rule, nil } if _, ok := BuiltinMap[expr.Operator().String()]; ok { @@ -435,10 +431,13 @@ func ParseModuleWithOpts(filename, input string, popts ParserOptions) (*Module, // ParseBody returns exactly one body. // If multiple bodies are parsed, an error is returned. func ParseBody(input string) (Body, error) { - return ParseBodyWithOpts(input, ParserOptions{}) + return ParseBodyWithOpts(input, ParserOptions{SkipRules: true}) } +// ParseBodyWithOpts returns exactly one body. It does _not_ set SkipRules: true on its own, +// but respects whatever ParserOptions it's been given. func ParseBodyWithOpts(input string, popts ParserOptions) (Body, error) { + stmts, _, err := ParseStatementsWithOpts("", input, popts) if err != nil { return nil, err @@ -467,7 +466,7 @@ func ParseBodyWithOpts(input string, popts ParserOptions) (Body, error) { func ParseExpr(input string) (*Expr, error) { body, err := ParseBody(input) if err != nil { - return nil, errors.Wrap(err, "failed to parse expression") + return nil, fmt.Errorf("failed to parse expression: %w", err) } if len(body) != 1 { return nil, fmt.Errorf("expected exactly one expression but got: %v", body) @@ -494,7 +493,7 @@ func ParsePackage(input string) (*Package, error) { func ParseTerm(input string) (*Term, error) { body, err := ParseBody(input) if err != nil { - return nil, errors.Wrap(err, "failed to parse term") + return nil, fmt.Errorf("failed to parse term: %w", err) } if len(body) != 1 { return nil, fmt.Errorf("expected exactly one term but got: %v", body) @@ -510,7 +509,7 @@ func ParseTerm(input string) (*Term, error) { func ParseRef(input string) (Ref, error) { term, err := ParseTerm(input) if err != nil { - return nil, errors.Wrap(err, "failed to parse ref") + return nil, fmt.Errorf("failed to parse ref: %w", err) } ref, ok := term.Value.(Ref) if !ok { @@ -566,7 +565,9 @@ func ParseStatementsWithOpts(filename, input string, popts ParserOptions) ([]Sta WithProcessAnnotation(popts.ProcessAnnotation). WithFutureKeywords(popts.FutureKeywords...). WithAllFutureKeywords(popts.AllFutureKeywords). - WithCapabilities(popts.Capabilities) + WithCapabilities(popts.Capabilities). + WithSkipRules(popts.SkipRules). + withUnreleasedKeywords(popts.unreleasedKeywords) stmts, comments, errs := parser.Parse() @@ -585,14 +586,15 @@ func parseModule(filename string, stmts []Statement, comments []*Comment) (*Modu var errs Errors - _package, ok := stmts[0].(*Package) + pkg, ok := stmts[0].(*Package) if !ok { loc := stmts[0].Loc() errs = append(errs, NewError(ParseErr, loc, "package expected")) } mod := &Module{ - Package: _package, + Package: pkg, + stmts: stmts, } // The comments slice only holds comments that were not their own statements. @@ -632,34 +634,7 @@ func parseModule(filename string, stmts []Statement, comments []*Comment) (*Modu return nil, errs } - // Find first non-annotation statement following each annotation and attach - // the annotation to that statement. - for _, a := range mod.Annotations { - for _, stmt := range stmts { - _, ok := stmt.(*Annotations) - if !ok { - if stmt.Loc().Row > a.Location.Row { - a.node = stmt - break - } - } - } - - if a.Scope == "" { - switch a.node.(type) { - case *Rule: - a.Scope = annotationScopeRule - case *Package: - a.Scope = annotationScopePackage - case *Import: - a.Scope = annotationScopeImport - } - } - - if err := validateAnnotationScopeAttachment(a); err != nil { - errs = append(errs, err) - } - } + errs = append(errs, attachAnnotationsNodes(mod)...) if len(errs) > 0 { return nil, errs @@ -668,24 +643,6 @@ func parseModule(filename string, stmts []Statement, comments []*Comment) (*Modu return mod, nil } -func validateAnnotationScopeAttachment(a *Annotations) *Error { - - switch a.Scope { - case annotationScopeRule, annotationScopeDocument: - if _, ok := a.node.(*Rule); ok { - return nil - } - return newScopeAttachmentErr(a, "rule") - case annotationScopePackage, annotationScopeSubpackages: - if _, ok := a.node.(*Package); ok { - return nil - } - return newScopeAttachmentErr(a, "package") - } - - return NewError(ParseErr, a.Loc(), "invalid annotation scope '%v'", a.Scope) -} - func newScopeAttachmentErr(a *Annotations, want string) *Error { var have string if a.node != nil { diff --git a/vendor/github.com/open-policy-agent/opa/ast/policy.go b/vendor/github.com/open-policy-agent/opa/ast/policy.go index 26b6d78e..04ebe56f 100644 --- a/vendor/github.com/open-policy-agent/opa/ast/policy.go +++ b/vendor/github.com/open-policy-agent/opa/ast/policy.go @@ -144,6 +144,7 @@ type ( Annotations []*Annotations `json:"annotations,omitempty"` Rules []*Rule `json:"rules,omitempty"` Comments []*Comment `json:"comments,omitempty"` + stmts []Statement } // Comment contains the raw text from the comment in the definition. @@ -152,21 +153,6 @@ type ( Location *Location } - // Annotations represents metadata attached to other AST nodes such as rules. - Annotations struct { - Location *Location `json:"-"` - Scope string `json:"scope"` - Schemas []*SchemaAnnotation `json:"schemas,omitempty"` - node Node - } - - // SchemaAnnotation contains a schema declaration for the document identified by the path. - SchemaAnnotation struct { - Path Ref `json:"path"` - Schema Ref `json:"schema,omitempty"` - Definition *interface{} `json:"definition,omitempty"` - } - // Package represents the namespace of the documents produced // by rules inside the module. Package struct { @@ -231,6 +217,14 @@ type ( Symbols []*Term `json:"symbols"` } + Every struct { + Location *Location `json:"-"` + Key *Term `json:"key"` + Value *Term `json:"value"` + Domain *Term `json:"domain"` + Body Body `json:"body"` + } + // With represents a modifier on an expression. With struct { Location *Location `json:"-"` @@ -239,122 +233,6 @@ type ( } ) -func (s *Annotations) String() string { - bs, _ := json.Marshal(s) - return string(bs) -} - -// Loc returns the location of this annotation. -func (s *Annotations) Loc() *Location { - return s.Location -} - -// SetLoc updates the location of this annotation. -func (s *Annotations) SetLoc(l *Location) { - s.Location = l -} - -// Compare returns an integer indicating if s is less than, equal to, or greater -// than other. -func (s *Annotations) Compare(other *Annotations) int { - - if cmp := scopeCompare(s.Scope, other.Scope); cmp != 0 { - return cmp - } - - max := len(s.Schemas) - if len(other.Schemas) < max { - max = len(other.Schemas) - } - - for i := 0; i < max; i++ { - if cmp := s.Schemas[i].Compare(other.Schemas[i]); cmp != 0 { - return cmp - } - } - - if len(s.Schemas) > len(other.Schemas) { - return 1 - } else if len(s.Schemas) < len(other.Schemas) { - return -1 - } - - return 0 -} - -// Copy returns a deep copy of s. -func (s *Annotations) Copy(node Node) *Annotations { - cpy := *s - cpy.Schemas = make([]*SchemaAnnotation, len(s.Schemas)) - for i := range cpy.Schemas { - cpy.Schemas[i] = s.Schemas[i].Copy() - } - cpy.node = node - return &cpy -} - -// Copy returns a deep copy of s. -func (s *SchemaAnnotation) Copy() *SchemaAnnotation { - cpy := *s - return &cpy -} - -// Compare returns an integer indicating if s is less than, equal to, or greater -// than other. -func (s *SchemaAnnotation) Compare(other *SchemaAnnotation) int { - - if cmp := s.Path.Compare(other.Path); cmp != 0 { - return cmp - } - - if cmp := s.Schema.Compare(other.Schema); cmp != 0 { - return cmp - } - - if s.Definition != nil && other.Definition == nil { - return -1 - } else if s.Definition == nil && other.Definition != nil { - return 1 - } else if s.Definition != nil && other.Definition != nil { - return util.Compare(*s.Definition, *other.Definition) - } - - return 0 -} - -func (s *SchemaAnnotation) String() string { - bs, _ := json.Marshal(s) - return string(bs) -} - -func scopeCompare(s1, s2 string) int { - - o1 := scopeOrder(s1) - o2 := scopeOrder(s2) - - if o2 < o1 { - return 1 - } else if o2 > o1 { - return -1 - } - - if s1 < s2 { - return -1 - } else if s2 < s1 { - return 1 - } - - return 0 -} - -func scopeOrder(s string) int { - switch s { - case annotationScopeRule: - return 1 - } - return 0 -} - // Compare returns an integer indicating whether mod is less than, equal to, // or greater than other. func (mod *Module) Compare(other *Module) int { @@ -383,32 +261,22 @@ func (mod *Module) Copy() *Module { cpy := *mod cpy.Rules = make([]*Rule, len(mod.Rules)) - var nodes map[Node]Node - - if len(mod.Annotations) > 0 { - nodes = make(map[Node]Node) - } + nodes := make(map[Node]Node, len(mod.Rules)+len(mod.Imports)+1 /* package */) for i := range mod.Rules { cpy.Rules[i] = mod.Rules[i].Copy() cpy.Rules[i].Module = &cpy - if nodes != nil { - nodes[mod.Rules[i]] = cpy.Rules[i] - } + nodes[mod.Rules[i]] = cpy.Rules[i] } cpy.Imports = make([]*Import, len(mod.Imports)) for i := range mod.Imports { cpy.Imports[i] = mod.Imports[i].Copy() - if nodes != nil { - nodes[mod.Imports[i]] = cpy.Imports[i] - } + nodes[mod.Imports[i]] = cpy.Imports[i] } cpy.Package = mod.Package.Copy() - if nodes != nil { - nodes[mod.Package] = cpy.Package - } + nodes[mod.Package] = cpy.Package cpy.Annotations = make([]*Annotations, len(mod.Annotations)) for i := range mod.Annotations { @@ -420,6 +288,11 @@ func (mod *Module) Copy() *Module { cpy.Comments[i] = mod.Comments[i].Copy() } + cpy.stmts = make([]Statement, len(mod.stmts)) + for i := range mod.stmts { + cpy.stmts[i] = nodes[mod.stmts[i]] + } + return &cpy } @@ -914,7 +787,7 @@ func (a Args) Copy() Args { } func (a Args) String() string { - var buf []string + buf := make([]string, 0, len(a)) for _, t := range a { buf = append(buf, t.String()) } @@ -1058,7 +931,7 @@ func (body Body) SetLoc(loc *Location) { } func (body Body) String() string { - var buf []string + buf := make([]string, 0, len(body)) for _, v := range body { buf = append(buf, v.String()) } @@ -1075,6 +948,11 @@ func (body Body) Vars(params VarVisitorParams) VarSet { // NewExpr returns a new Expr object. func NewExpr(terms interface{}) *Expr { + switch terms.(type) { + case *SomeDecl, *Every, *Term, []*Term: // ok + default: + panic("unreachable") + } return &Expr{ Negated: false, Terms: terms, @@ -1153,6 +1031,10 @@ func (expr *Expr) Compare(other *Expr) int { if cmp := Compare(t, other.Terms.(*SomeDecl)); cmp != 0 { return cmp } + case *Every: + if cmp := Compare(t, other.Terms.(*Every)); cmp != 0 { + return cmp + } } return withSliceCompare(expr.With, other.With) @@ -1166,14 +1048,28 @@ func (expr *Expr) sortOrder() int { return 1 case []*Term: return 2 + case *Every: + return 3 } return -1 } +// CopyWithoutTerms returns a deep copy of expr without its Terms +func (expr *Expr) CopyWithoutTerms() *Expr { + cpy := *expr + + cpy.With = make([]*With, len(expr.With)) + for i := range expr.With { + cpy.With[i] = expr.With[i].Copy() + } + + return &cpy +} + // Copy returns a deep copy of expr. func (expr *Expr) Copy() *Expr { - cpy := *expr + cpy := expr.CopyWithoutTerms() switch ts := expr.Terms.(type) { case *SomeDecl: @@ -1186,14 +1082,11 @@ func (expr *Expr) Copy() *Expr { cpy.Terms = cpyTs case *Term: cpy.Terms = ts.Copy() + case *Every: + cpy.Terms = ts.Copy() } - cpy.With = make([]*With, len(expr.With)) - for i := range expr.With { - cpy.With[i] = expr.With[i].Copy() - } - - return &cpy + return cpy } // Hash returns the hash code of the Expr. @@ -1248,6 +1141,18 @@ func (expr *Expr) IsCall() bool { return ok } +// IsEvery returns true if this expression is an 'every' expression. +func (expr *Expr) IsEvery() bool { + _, ok := expr.Terms.(*Every) + return ok +} + +// IsSome returns true if this expression is a 'some' expression. +func (expr *Expr) IsSome() bool { + _, ok := expr.Terms.(*SomeDecl) + return ok +} + // Operator returns the name of the function or built-in this expression refers // to. If this expression is not a function call, returns nil. func (expr *Expr) Operator() Ref { @@ -1333,7 +1238,7 @@ func (expr *Expr) SetLoc(loc *Location) { } func (expr *Expr) String() string { - var buf []string + buf := make([]string, 0, 2+len(expr.With)) if expr.Negated { buf = append(buf, "not") } @@ -1344,9 +1249,7 @@ func (expr *Expr) String() string { } else { buf = append(buf, Call(t).String()) } - case *Term: - buf = append(buf, t.String()) - case *SomeDecl: + case fmt.Stringer: buf = append(buf, t.String()) } @@ -1422,6 +1325,62 @@ func (d *SomeDecl) Hash() int { return termSliceHash(d.Symbols) } +func (q *Every) String() string { + if q.Key != nil { + return fmt.Sprintf("every %s, %s in %s { %s }", + q.Key, + q.Value, + q.Domain, + q.Body) + } + return fmt.Sprintf("every %s in %s { %s }", + q.Value, + q.Domain, + q.Body) +} + +func (q *Every) Loc() *Location { + return q.Location +} + +func (q *Every) SetLoc(l *Location) { + q.Location = l +} + +// Copy returns a deep copy of d. +func (q *Every) Copy() *Every { + cpy := *q + cpy.Key = q.Key.Copy() + cpy.Value = q.Value.Copy() + cpy.Domain = q.Domain.Copy() + cpy.Body = q.Body.Copy() + return &cpy +} + +func (q *Every) Compare(other *Every) int { + for _, terms := range [][2]*Term{ + {q.Key, other.Key}, + {q.Value, other.Value}, + {q.Domain, other.Domain}, + } { + if d := Compare(terms[0], terms[1]); d != 0 { + return d + } + } + return q.Body.Compare(other.Body) +} + +// KeyValueVars returns the key and val arguments of an `every` +// expression, if they are non-nil and not wildcards. +func (q *Every) KeyValueVars() VarSet { + vis := &VarVisitor{vars: VarSet{}} + if q.Key != nil { + vis.Walk(q.Key) + } + vis.Walk(q.Value) + return vis.vars +} + func (w *With) String() string { return "with " + w.Target.String() + " as " + w.Value.String() } @@ -1503,6 +1462,8 @@ func Copy(x interface{}) interface{} { return x.Copy() case *SomeDecl: return x.Copy() + case *Every: + return x.Copy() case *Term: return x.Copy() case *ArrayComprehension: diff --git a/vendor/github.com/open-policy-agent/opa/ast/schema.go b/vendor/github.com/open-policy-agent/opa/ast/schema.go index 76bd4756..8c96ac62 100644 --- a/vendor/github.com/open-policy-agent/opa/ast/schema.go +++ b/vendor/github.com/open-policy-agent/opa/ast/schema.go @@ -54,7 +54,7 @@ func loadSchema(raw interface{}, allowNet []string) (types.Type, error) { return nil, err } - tpe, err := parseSchema(jsonSchema.RootSchema) + tpe, err := newSchemaParser().parseSchema(jsonSchema.RootSchema) if err != nil { return nil, fmt.Errorf("type checking: %w", err) } diff --git a/vendor/github.com/open-policy-agent/opa/ast/term.go b/vendor/github.com/open-policy-agent/opa/ast/term.go index ce254685..73131b51 100644 --- a/vendor/github.com/open-policy-agent/opa/ast/term.go +++ b/vendor/github.com/open-policy-agent/opa/ast/term.go @@ -17,6 +17,7 @@ import ( "sort" "strconv" "strings" + "sync" "github.com/OneOfOne/xxhash" "github.com/pkg/errors" @@ -493,6 +494,20 @@ func ContainsComprehensions(v interface{}) bool { return found } +// ContainsClosures returns true if the Value v contains closures. +func ContainsClosures(v interface{}) bool { + found := false + WalkClosures(v, func(x interface{}) bool { + switch x.(type) { + case *ArrayComprehension, *ObjectComprehension, *SetComprehension, *Every: + found = true + return found + } + return found + }) + return found +} + // IsScalar returns true if the AST value is a scalar. func IsScalar(v Value) bool { switch v.(type) { @@ -878,19 +893,16 @@ func (ref Ref) Append(term *Term) Ref { // existing elements are shifted to the right. If pos > len(ref)+1 this // function panics. func (ref Ref) Insert(x *Term, pos int) Ref { - if pos == len(ref) { + switch { + case pos == len(ref): return ref.Append(x) - } else if pos > len(ref)+1 { + case pos > len(ref)+1: panic("illegal index") } cpy := make(Ref, len(ref)+1) - for i := 0; i < pos; i++ { - cpy[i] = ref[i] - } + copy(cpy, ref[:pos]) cpy[pos] = x - for i := pos; i < len(ref); i++ { - cpy[i+1] = ref[i] - } + copy(cpy[pos+1:], ref[pos:]) return cpy } @@ -904,9 +916,8 @@ func (ref Ref) Extend(other Ref) Ref { head.Value = String(head.Value.(Var)) offset := len(ref) dst[offset] = head - for i := range other[1:] { - dst[offset+i+1] = other[i+1] - } + + copy(dst[offset+1:], other[1:]) return dst } @@ -917,10 +928,7 @@ func (ref Ref) Concat(terms []*Term) Ref { } cpy := make(Ref, len(ref)+len(terms)) copy(cpy, ref) - - for i := range terms { - cpy[len(ref)+i] = terms[i] - } + copy(cpy[len(ref):], terms) return cpy } @@ -1064,25 +1072,43 @@ func (ref Ref) String() string { } // OutputVars returns a VarSet containing variables that would be bound by evaluating -// this expression in isolation. +// this expression in isolation. func (ref Ref) OutputVars() VarSet { vis := NewVarVisitor().WithParams(VarVisitorParams{SkipRefHead: true}) vis.Walk(ref) return vis.Vars() } +func (ref Ref) toArray() *Array { + a := NewArray() + for _, term := range ref { + if _, ok := term.Value.(String); ok { + a = a.Append(term) + } else { + a = a.Append(StringTerm(term.Value.String())) + } + } + return a +} + // QueryIterator defines the interface for querying AST documents with references. type QueryIterator func(map[Var]Value, Value) error // ArrayTerm creates a new Term with an Array value. func ArrayTerm(a ...*Term) *Term { - return &Term{Value: &Array{elems: a, hash: 0, ground: termSliceIsGround(a)}} + return NewTerm(NewArray(a...)) } // NewArray creates an Array with the terms provided. The array will // use the provided term slice. func NewArray(a ...*Term) *Array { - return &Array{elems: a, hash: 0, ground: termSliceIsGround(a)} + hs := make([]int, len(a)) + for i, e := range a { + hs[i] = e.Value.Hash() + } + arr := &Array{elems: a, hashs: hs, ground: termSliceIsGround(a)} + arr.rehash() + return arr } // Array represents an array as defined by the language. Arrays are similar to the @@ -1090,14 +1116,18 @@ func NewArray(a ...*Term) *Array { // and References. type Array struct { elems []*Term + hashs []int // element hashes hash int ground bool } // Copy returns a deep copy of arr. func (arr *Array) Copy() *Array { + cpy := make([]int, len(arr.elems)) + copy(cpy, arr.hashs) return &Array{ elems: termSliceCopy(arr.elems), + hashs: cpy, hash: arr.hash, ground: arr.IsGround()} } @@ -1159,16 +1189,12 @@ func (arr *Array) Sorted() *Array { } sort.Sort(termSlice(cpy)) a := NewArray(cpy...) - a.hash = arr.hash + a.hashs = arr.hashs return a } // Hash returns the hash code for the Value. func (arr *Array) Hash() int { - if arr.hash == 0 { - arr.hash = termSliceHash(arr.elems) - } - return arr.hash } @@ -1208,11 +1234,19 @@ func (arr *Array) Elem(i int) *Term { return arr.elems[i] } +// rehash updates the cached hash of arr. +func (arr *Array) rehash() { + arr.hash = 0 + for _, h := range arr.hashs { + arr.hash += h + } +} + // set sets the element i of arr. func (arr *Array) set(i int, v *Term) { arr.ground = arr.ground && v.IsGround() arr.elems[i] = v - arr.hash = 0 + arr.hashs[i] = v.Value.Hash() } // Slice returns a slice of arr starting from i index to j. -1 @@ -1221,15 +1255,21 @@ func (arr *Array) set(i int, v *Term) { // the other. func (arr *Array) Slice(i, j int) *Array { var elems []*Term + var hashs []int if j == -1 { elems = arr.elems[i:] + hashs = arr.hashs[i:] } else { elems = arr.elems[i:j] + hashs = arr.hashs[i:j] } // If arr is ground, the slice is, too. // If it's not, the slice could still be. gr := arr.ground || termSliceIsGround(elems) - return &Array{elems: elems, ground: gr} + + s := &Array{elems: elems, hashs: hashs, ground: gr} + s.rehash() + return s } // Iter calls f on each element in arr. If f returns an error, @@ -1266,7 +1306,8 @@ func (arr *Array) Foreach(f func(*Term)) { func (arr *Array) Append(v *Term) *Array { cpy := *arr cpy.elems = append(arr.elems, v) - cpy.hash = 0 + cpy.hashs = append(arr.hashs, v.Value.Hash()) + cpy.hash = arr.hash + v.Value.Hash() cpy.ground = arr.ground && v.IsGround() return &cpy } @@ -1305,10 +1346,11 @@ func newset(n int) *set { keys = make([]*Term, 0, n) } return &set{ - elems: make(map[int]*Term, n), - keys: keys, - hash: 0, - ground: true, + elems: make(map[int]*Term, n), + keys: keys, + hash: 0, + ground: true, + sortGuard: new(sync.Once), } } @@ -1321,10 +1363,11 @@ func SetTerm(t ...*Term) *Term { } type set struct { - elems map[int]*Term - keys []*Term - hash int - ground bool + elems map[int]*Term + keys []*Term + hash int + ground bool + sortGuard *sync.Once // Prevents race condition around sorting. } // Copy returns a deep copy of s. @@ -1345,11 +1388,6 @@ func (s *set) IsGround() bool { // Hash returns a hash code for s. func (s *set) Hash() int { - if s.hash == 0 { - s.Foreach(func(x *Term) { - s.hash += x.Hash() - }) - } return s.hash } @@ -1359,7 +1397,7 @@ func (s *set) String() string { } var b strings.Builder b.WriteRune('{') - for i := range s.keys { + for i := range s.sortedKeys() { if i > 0 { b.WriteString(", ") } @@ -1369,6 +1407,13 @@ func (s *set) String() string { return b.String() } +func (s *set) sortedKeys() []*Term { + s.sortGuard.Do(func() { + sort.Sort(termSlice(s.keys)) + }) + return s.keys +} + // Compare compares s to other, return <0, 0, or >0 if it is less than, equal to, // or greater than other. func (s *set) Compare(other Value) int { @@ -1380,7 +1425,7 @@ func (s *set) Compare(other Value) int { return 1 } t := other.(*set) - return termSliceCompare(s.keys, t.keys) + return termSliceCompare(s.sortedKeys(), t.sortedKeys()) } // Find returns the set or dereferences the element itself. @@ -1446,7 +1491,7 @@ func (s *set) Add(t *Term) { // Iter calls f on each element in s. If f returns an error, iteration stops // and the return value is the error. func (s *set) Iter(f func(*Term) error) error { - for i := range s.keys { + for i := range s.sortedKeys() { if err := f(s.keys[i]); err != nil { return err } @@ -1522,26 +1567,26 @@ func (s *set) MarshalJSON() ([]byte, error) { if s.keys == nil { return []byte(`[]`), nil } - return json.Marshal(s.keys) + return json.Marshal(s.sortedKeys()) } // Sorted returns an Array that contains the sorted elements of s. func (s *set) Sorted() *Array { cpy := make([]*Term, len(s.keys)) - for i := range s.keys { - cpy[i] = s.keys[i] - } - sort.Sort(termSlice(cpy)) + copy(cpy, s.sortedKeys()) return NewArray(cpy...) } // Slice returns a slice of terms contained in the set. func (s *set) Slice() []*Term { - return s.keys + return s.sortedKeys() } +// NOTE(philipc): We assume a many-readers, single-writer model here. +// This method should NOT be used concurrently, or else we risk data races. func (s *set) insert(x *Term) { hash := x.Hash() + insertHash := hash // This `equal` utility is duplicated and manually inlined a number of // time in this file. Inlining it avoids heap allocations, so it makes // a big performance difference: some operations like lookup become twice @@ -1619,27 +1664,23 @@ func (s *set) insert(x *Term) { equal = func(y Value) bool { return Compare(x, y) == 0 } } - for curr, ok := s.elems[hash]; ok; { + for curr, ok := s.elems[insertHash]; ok; { if equal(curr.Value) { return } - hash++ - curr, ok = s.elems[hash] + insertHash++ + curr, ok = s.elems[insertHash] } - s.elems[hash] = x - i := sort.Search(len(s.keys), func(i int) bool { return Compare(x, s.keys[i]) < 0 }) - if i < len(s.keys) { - // insert at position `i`: - s.keys = append(s.keys, nil) // add some space - copy(s.keys[i+1:], s.keys[i:]) // move things over - s.keys[i] = x // drop it in position - } else { - s.keys = append(s.keys, x) - } + s.elems[insertHash] = x + // O(1) insertion, but we'll have to re-sort the keys later. + s.keys = append(s.keys, x) + // Reset the sync.Once instance. + // See https://github.com/golang/go/issues/25955 for why we do it this way. + s.sortGuard = new(sync.Once) - s.hash = 0 + s.hash += hash s.ground = s.ground && x.IsGround() } @@ -1751,7 +1792,7 @@ type Object interface { MergeWith(other Object, conflictResolver func(v1, v2 *Term) (*Term, bool)) (Object, bool) Filter(filter Object) (Object, error) Keys() []*Term - Elem(i int) (*Term, *Term) + KeysIterator() ObjectKeysIterator get(k *Term) *objectElem // To prevent external implementations } @@ -1774,7 +1815,8 @@ type object struct { keys objectElemSlice ground int // number of key and value grounds. Counting is // required to support insert's key-value replace. - hash int + hash int + sortGuard *sync.Once // Prevents race condition around sorting. } func newobject(n int) *object { @@ -1783,10 +1825,11 @@ func newobject(n int) *object { keys = make(objectElemSlice, 0, n) } return &object{ - elems: make(map[int]*objectElem, n), - keys: keys, - ground: 0, - hash: 0, + elems: make(map[int]*objectElem, n), + keys: keys, + ground: 0, + hash: 0, + sortGuard: new(sync.Once), } } @@ -1808,6 +1851,13 @@ func Item(key, value *Term) [2]*Term { return [2]*Term{key, value} } +func (obj *object) sortedKeys() objectElemSlice { + obj.sortGuard.Do(func() { + sort.Sort(obj.keys) + }) + return obj.keys +} + // Compare compares obj to other, return <0, 0, or >0 if it is less than, equal to, // or greater than other. func (obj *object) Compare(other Value) int { @@ -1820,29 +1870,32 @@ func (obj *object) Compare(other Value) int { } a := obj b := other.(*object) - minLen := len(a.keys) - if len(b.keys) < len(a.keys) { - minLen = len(b.keys) + // Ensure that keys are in canonical sorted order before use! + akeys := a.sortedKeys() + bkeys := b.sortedKeys() + minLen := len(akeys) + if len(b.keys) < len(akeys) { + minLen = len(bkeys) } for i := 0; i < minLen; i++ { - keysCmp := Compare(a.keys[i].key, b.keys[i].key) + keysCmp := Compare(akeys[i].key, bkeys[i].key) if keysCmp < 0 { return -1 } if keysCmp > 0 { return 1 } - valA := a.keys[i].value - valB := b.keys[i].value + valA := akeys[i].value + valB := bkeys[i].value valCmp := Compare(valA, valB) if valCmp != 0 { return valCmp } } - if len(a.keys) < len(b.keys) { + if len(akeys) < len(bkeys) { return -1 } - if len(b.keys) < len(a.keys) { + if len(bkeys) < len(akeys) { return 1 } return 0 @@ -1874,14 +1927,6 @@ func (obj *object) Get(k *Term) *Term { // Hash returns the hash code for the Value. func (obj *object) Hash() int { - if obj.hash == 0 { - for h, curr := range obj.elems { - for ; curr != nil; curr = curr.next { - obj.hash += h - obj.hash += curr.value.Hash() - } - } - } return obj.hash } @@ -1926,7 +1971,7 @@ func (obj *object) Intersect(other Object) [][3]*Term { // Iter calls the function f for each key-value pair in the object. If f // returns an error, iteration stops and the error is returned. func (obj *object) Iter(f func(*Term, *Term) error) error { - for _, node := range obj.keys { + for _, node := range obj.sortedKeys() { if err := f(node.key, node.value); err != nil { return err } @@ -1978,21 +2023,22 @@ func (obj *object) Map(f func(*Term, *Term) (*Term, *Term, error)) (Object, erro func (obj *object) Keys() []*Term { keys := make([]*Term, len(obj.keys)) - for i, elem := range obj.keys { + for i, elem := range obj.sortedKeys() { keys[i] = elem.key } return keys } -func (obj *object) Elem(i int) (*Term, *Term) { - return obj.keys[i].key, obj.keys[i].value +// Returns an iterator over the obj's keys. +func (obj *object) KeysIterator() ObjectKeysIterator { + return newobjectKeysIterator(obj) } // MarshalJSON returns JSON encoded bytes representing obj. func (obj *object) MarshalJSON() ([]byte, error) { sl := make([][2]*Term, obj.Len()) - for i, node := range obj.keys { + for i, node := range obj.sortedKeys() { sl[i] = Item(node.key, node.value) } return json.Marshal(sl) @@ -2072,7 +2118,7 @@ func (obj object) String() string { var b strings.Builder b.WriteRune('{') - for i, elem := range obj.keys { + for i, elem := range obj.sortedKeys() { if i > 0 { b.WriteString(", ") } @@ -2172,6 +2218,8 @@ func (obj *object) get(k *Term) *objectElem { return nil } +// NOTE(philipc): We assume a many-readers, single-writer model here. +// This method should NOT be used concurrently, or else we risk data races. func (obj *object) insert(k, v *Term) { hash := k.Hash() head := obj.elems[hash] @@ -2266,7 +2314,6 @@ func (obj *object) insert(k, v *Term) { } curr.value = v - obj.hash = 0 return } } @@ -2276,16 +2323,12 @@ func (obj *object) insert(k, v *Term) { next: head, } obj.elems[hash] = elem - i := sort.Search(len(obj.keys), func(i int) bool { return Compare(elem.key, obj.keys[i].key) < 0 }) - if i < len(obj.keys) { - // insert at position `i`: - obj.keys = append(obj.keys, nil) // add some space - copy(obj.keys[i+1:], obj.keys[i:]) // move things over - obj.keys[i] = elem // drop it in position - } else { - obj.keys = append(obj.keys, elem) - } - obj.hash = 0 + // O(1) insertion, but we'll have to re-sort the keys later. + obj.keys = append(obj.keys, elem) + // Reset the sync.Once instance. + // See https://github.com/golang/go/issues/25955 for why we do it this way. + obj.sortGuard = new(sync.Once) + obj.hash += hash + v.Hash() if k.IsGround() { obj.ground++ @@ -2360,6 +2403,36 @@ func filterObject(o Value, filter Value) (Value, error) { } } +// NOTE(philipc): The only way to get an ObjectKeyIterator should be +// from an Object. This ensures that the iterator can have implementation- +// specific details internally, with no contracts except to the very +// limited interface. +type ObjectKeysIterator interface { + Next() (*Term, bool) +} + +type objectKeysIterator struct { + obj *object + numKeys int + index int +} + +func newobjectKeysIterator(o *object) ObjectKeysIterator { + return &objectKeysIterator{ + obj: o, + numKeys: o.Len(), + index: 0, + } +} + +func (oki *objectKeysIterator) Next() (*Term, bool) { + if oki.index == oki.numKeys || oki.numKeys == 0 { + return nil, false + } + oki.index++ + return oki.obj.sortedKeys()[oki.index-1].key, true +} + // ArrayComprehension represents an array comprehension as defined in the language. type ArrayComprehension struct { Term *Term `json:"term"` diff --git a/vendor/github.com/open-policy-agent/opa/ast/transform.go b/vendor/github.com/open-policy-agent/opa/ast/transform.go index c7fa4c8f..c25f7bc3 100644 --- a/vendor/github.com/open-policy-agent/opa/ast/transform.go +++ b/vendor/github.com/open-policy-agent/opa/ast/transform.go @@ -172,6 +172,26 @@ func Transform(t Transformer, x interface{}) (interface{}, error) { if y.Terms, err = transformTerm(t, ts); err != nil { return nil, err } + case *Every: + if ts.Key != nil { + ts.Key, err = transformTerm(t, ts.Key) + if err != nil { + return nil, err + } + } + ts.Value, err = transformTerm(t, ts.Value) + if err != nil { + return nil, err + } + ts.Domain, err = transformTerm(t, ts.Domain) + if err != nil { + return nil, err + } + ts.Body, err = transformBody(t, ts.Body) + if err != nil { + return nil, err + } + y.Terms = ts } for i, w := range y.With { w, err := Transform(t, w) diff --git a/vendor/github.com/open-policy-agent/opa/ast/varset.go b/vendor/github.com/open-policy-agent/opa/ast/varset.go index 16dc3f58..14f53149 100644 --- a/vendor/github.com/open-policy-agent/opa/ast/varset.go +++ b/vendor/github.com/open-policy-agent/opa/ast/varset.go @@ -91,7 +91,7 @@ func (s VarSet) Update(vs VarSet) { } func (s VarSet) String() string { - tmp := []string{} + tmp := make([]string, 0, len(s)) for v := range s { tmp = append(tmp, string(v)) } diff --git a/vendor/github.com/open-policy-agent/opa/ast/visit.go b/vendor/github.com/open-policy-agent/opa/ast/visit.go index 139c4de3..02c393cd 100644 --- a/vendor/github.com/open-policy-agent/opa/ast/visit.go +++ b/vendor/github.com/open-policy-agent/opa/ast/visit.go @@ -88,14 +88,12 @@ func walk(v Visitor, x interface{}) { } case *Expr: switch ts := x.Terms.(type) { - case *SomeDecl: + case *Term, *SomeDecl, *Every: Walk(w, ts) case []*Term: for _, t := range ts { Walk(w, t) } - case *Term: - Walk(w, ts) } for i := range x.With { Walk(w, x.With[i]) @@ -136,6 +134,13 @@ func walk(v Visitor, x interface{}) { for _, t := range x { Walk(w, t) } + case *Every: + if x.Key != nil { + Walk(w, x.Key) + } + Walk(w, x.Value) + Walk(w, x.Domain) + Walk(w, x.Body) } } @@ -155,8 +160,8 @@ func WalkVars(x interface{}, f func(Var) bool) { // returns true, AST nodes under the last node will not be visited. func WalkClosures(x interface{}, f func(interface{}) bool) { vis := &GenericVisitor{func(x interface{}) bool { - switch x.(type) { - case *ArrayComprehension, *ObjectComprehension, *SetComprehension: + switch x := x.(type) { + case *ArrayComprehension, *ObjectComprehension, *SetComprehension, *Every: return f(x) } return false @@ -319,14 +324,12 @@ func (vis *GenericVisitor) Walk(x interface{}) { } case *Expr: switch ts := x.Terms.(type) { - case *SomeDecl: + case *Term, *SomeDecl, *Every: vis.Walk(ts) case []*Term: for _, t := range ts { vis.Walk(t) } - case *Term: - vis.Walk(ts) } for i := range x.With { vis.Walk(x.With[i]) @@ -367,6 +370,13 @@ func (vis *GenericVisitor) Walk(x interface{}) { for _, t := range x { vis.Walk(t) } + case *Every: + if x.Key != nil { + vis.Walk(x.Key) + } + vis.Walk(x.Value) + vis.Walk(x.Domain) + vis.Walk(x.Body) } } @@ -440,14 +450,12 @@ func (vis *BeforeAfterVisitor) Walk(x interface{}) { } case *Expr: switch ts := x.Terms.(type) { - case *SomeDecl: + case *Term, *SomeDecl, *Every: vis.Walk(ts) case []*Term: for _, t := range ts { vis.Walk(t) } - case *Term: - vis.Walk(ts) } for i := range x.With { vis.Walk(x.With[i]) @@ -488,6 +496,13 @@ func (vis *BeforeAfterVisitor) Walk(x interface{}) { for _, t := range x { vis.Walk(t) } + case *Every: + if x.Key != nil { + vis.Walk(x.Key) + } + vis.Walk(x.Value) + vis.Walk(x.Domain) + vis.Walk(x.Body) } } @@ -527,6 +542,8 @@ func (vis *VarVisitor) Vars() VarSet { return vis.vars } +// visit determines if the VarVisitor will recurse into x: if it returns `true`, +// the visitor will _skip_ that branch of the AST func (vis *VarVisitor) visit(v interface{}) bool { if vis.params.SkipObjectKeys { if o, ok := v.(Object); ok { @@ -545,9 +562,15 @@ func (vis *VarVisitor) visit(v interface{}) bool { } } if vis.params.SkipClosures { - switch v.(type) { + switch v := v.(type) { case *ArrayComprehension, *ObjectComprehension, *SetComprehension: return true + case *Expr: + if ev, ok := v.Terms.(*Every); ok { + vis.Walk(ev.Domain) + // We're _not_ walking ev.Body -- that's the closure here + return true + } } } if vis.params.SkipWithTarget { @@ -585,6 +608,20 @@ func (vis *VarVisitor) visit(v interface{}) bool { vis.Walk(v[i]) } return true + case *With: + if ref, ok := v.Target.Value.(Ref); ok { + for _, t := range ref[1:] { + vis.Walk(t) + } + } + if ref, ok := v.Value.Value.(Ref); ok { + for _, t := range ref[1:] { + vis.Walk(t) + } + } else { + vis.Walk(v.Value) + } + return true } } if v, ok := v.(Var); ok { @@ -643,14 +680,12 @@ func (vis *VarVisitor) Walk(x interface{}) { } case *Expr: switch ts := x.Terms.(type) { - case *SomeDecl: + case *Term, *SomeDecl, *Every: vis.Walk(ts) case []*Term: for _, t := range ts { vis.Walk(t) } - case *Term: - vis.Walk(ts) } for i := range x.With { vis.Walk(x.With[i]) @@ -691,5 +726,12 @@ func (vis *VarVisitor) Walk(x interface{}) { for _, t := range x { vis.Walk(t) } + case *Every: + if x.Key != nil { + vis.Walk(x.Key) + } + vis.Walk(x.Value) + vis.Walk(x.Domain) + vis.Walk(x.Body) } } diff --git a/vendor/github.com/open-policy-agent/opa/bundle/bundle.go b/vendor/github.com/open-policy-agent/opa/bundle/bundle.go index 13ec7fe5..1f3791ef 100644 --- a/vendor/github.com/open-policy-agent/opa/bundle/bundle.go +++ b/vendor/github.com/open-policy-agent/opa/bundle/bundle.go @@ -11,6 +11,7 @@ import ( "compress/gzip" "encoding/hex" "encoding/json" + "errors" "fmt" "io" "net/url" @@ -18,8 +19,6 @@ import ( "reflect" "strings" - "github.com/pkg/errors" - "github.com/open-policy-agent/opa/ast" "github.com/open-policy-agent/opa/format" "github.com/open-policy-agent/opa/internal/file/archive" @@ -32,12 +31,17 @@ import ( const ( RegoExt = ".rego" WasmFile = "policy.wasm" + PlanFile = "plan.json" ManifestExt = ".manifest" SignaturesFile = "signatures.json" + patchFile = "patch.json" dataFile = "data.json" yamlDataFile = "data.yaml" + ymlDataFile = "data.yml" defaultHashingAlg = "SHA-256" DefaultSizeLimitBytes = (1024 * 1024 * 1024) // limit bundle reads to 1GB to protect against gzip bombs + DeltaBundleType = "delta" + SnapshotBundleType = "snapshot" ) // Bundle represents a loaded bundle. The bundle can contain data and policies. @@ -48,6 +52,32 @@ type Bundle struct { Modules []ModuleFile Wasm []byte // Deprecated. Use WasmModules instead WasmModules []WasmModuleFile + PlanModules []PlanModuleFile + Patch Patch + Etag string + Raw []Raw + + lazyLoadingMode bool + sizeLimitBytes int64 +} + +// Raw contains raw bytes representing the bundle's content +type Raw struct { + Path string + Value []byte +} + +// Patch contains an array of objects wherein each object represents the patch operation to be +// applied to the bundle data. +type Patch struct { + Data []PatchOperation `json:"data,omitempty"` +} + +// PatchOperation models a single patch operation against a document. +type PatchOperation struct { + Op string `json:"op"` + Path string `json:"path"` + Value interface{} `json:"value"` } // SignaturesConfig represents an array of JWTs that encapsulate the signatures for the bundle. @@ -129,21 +159,11 @@ func (m Manifest) Equal(other Manifest) bool { return false } - if len(m.WasmResolvers) != len(other.WasmResolvers) { - return false - } - - for i := 0; i < len(m.WasmResolvers); i++ { - if m.WasmResolvers[i] != other.WasmResolvers[i] { - return false - } - } - if !reflect.DeepEqual(m.Metadata, other.Metadata) { return false } - return m.rootSet().Equal(other.rootSet()) + return m.equalWasmResolversAndRoots(other) } // Copy returns a deep copy of the manifest. @@ -171,7 +191,7 @@ func (m Manifest) Copy() Manifest { func (m Manifest) String() string { m.Init() - return fmt.Sprintf("", m.Revision, *m.Roots, m.WasmResolvers) + return fmt.Sprintf("", m.Revision, *m.Roots, m.WasmResolvers, m.Metadata) } func (m Manifest) rootSet() stringSet { @@ -184,6 +204,20 @@ func (m Manifest) rootSet() stringSet { return stringSet(rs) } +func (m Manifest) equalWasmResolversAndRoots(other Manifest) bool { + if len(m.WasmResolvers) != len(other.WasmResolvers) { + return false + } + + for i := 0; i < len(m.WasmResolvers); i++ { + if m.WasmResolvers[i] != other.WasmResolvers[i] { + return false + } + } + + return m.rootSet().Equal(other.rootSet()) +} + type stringSet map[string]struct{} func (ss stringSet) Equal(other stringSet) bool { @@ -222,12 +256,7 @@ func (m *Manifest) validateAndInjectDefaults(b Bundle) error { for _, module := range b.Modules { found := false if path, err := module.Parsed.Package.Path.Ptr(); err == nil { - for i := range roots { - if strings.HasPrefix(path, roots[i]) { - found = true - break - } - } + found = RootPathsContain(roots, path) } if !found { return fmt.Errorf("manifest roots %v do not permit '%v' in module '%v'", roots, module.Parsed.Package, module.Path) @@ -248,15 +277,7 @@ func (m *Manifest) validateAndInjectDefaults(b Bundle) error { } // Ensure wasm module entrypoint in within bundle roots - found := false - for i := range roots { - if strings.HasPrefix(wmConfig.Entrypoint, roots[i]) { - found = true - break - } - } - - if !found { + if !RootPathsContain(roots, wmConfig.Entrypoint) { return fmt.Errorf("manifest roots %v do not permit '%v' entrypoint for wasm module '%v'", roots, wmConfig.Entrypoint, wmConfig.Module) } @@ -268,17 +289,28 @@ func (m *Manifest) validateAndInjectDefaults(b Bundle) error { wasmModuleToEps[wmConfig.Module] = wmConfig.Entrypoint } + // Validate data patches in bundle. + for _, patch := range b.Patch.Data { + path := strings.Trim(patch.Path, "/") + if !RootPathsContain(roots, path) { + return fmt.Errorf("manifest roots %v do not permit data patch at path '%s'", roots, path) + } + } + + if b.lazyLoadingMode { + return nil + } + // Validate data in bundle. return dfs(b.Data, "", func(path string, node interface{}) (bool, error) { path = strings.Trim(path, "/") - for i := range roots { - if strings.HasPrefix(path, roots[i]) { - return true, nil - } + if RootPathsContain(roots, path) { + return true, nil } + if _, ok := node.(map[string]interface{}); ok { for i := range roots { - if strings.HasPrefix(roots[i], path) { + if RootPathsContain(strings.Split(path, "/"), roots[i]) { return false, nil } } @@ -303,6 +335,17 @@ type WasmModuleFile struct { Raw []byte } +// PlanModuleFile represents a single plan module contained in a bundle. +// +// NOTE(tsandall): currently the plans are just opaque binary blobs. In the +// future we could inject the entrypoints so that the plans could be executed +// inside of OPA proper like we do for Wasm modules. +type PlanModuleFile struct { + URL string + Path string + Raw []byte +} + // Reader contains the reader to load the bundle from. type Reader struct { loader DirectoryLoader @@ -314,6 +357,9 @@ type Reader struct { processAnnotations bool files map[string]FileInfo // files in the bundle signature payload sizeLimitBytes int64 + etag string + lazyLoadingMode bool + name string } // NewReader is deprecated. Use NewCustomReader instead. @@ -378,38 +424,60 @@ func (r *Reader) WithSizeLimitBytes(n int64) *Reader { return r } +// WithBundleEtag sets the given etag value on the bundle +func (r *Reader) WithBundleEtag(etag string) *Reader { + r.etag = etag + return r +} + +// WithBundleName specifies the bundle name +func (r *Reader) WithBundleName(name string) *Reader { + r.name = name + return r +} + +// WithLazyLoadingMode sets the bundle loading mode. If true, +// bundles will be read in lazy mode. In this mode, data files in the bundle will not be +// deserialized and the check to validate that the bundle data does not contain paths +// outside the bundle's roots will not be performed while reading the bundle. +func (r *Reader) WithLazyLoadingMode(yes bool) *Reader { + r.lazyLoadingMode = yes + return r +} + // Read returns a new Bundle loaded from the reader. func (r *Reader) Read() (Bundle, error) { var bundle Bundle var descriptors []*Descriptor var err error + var raw []Raw - bundle.Data = map[string]interface{}{} - - bundle.Signatures, descriptors, err = listSignaturesAndDescriptors(r.loader, r.skipVerify, r.sizeLimitBytes) + bundle.Signatures, bundle.Patch, descriptors, err = preProcessBundle(r.loader, r.skipVerify, r.sizeLimitBytes) if err != nil { return bundle, err } - err = r.checkSignaturesAndDescriptors(bundle.Signatures) - if err != nil { - return bundle, err + bundle.lazyLoadingMode = r.lazyLoadingMode + bundle.sizeLimitBytes = r.sizeLimitBytes + + if bundle.Type() == SnapshotBundleType { + err = r.checkSignaturesAndDescriptors(bundle.Signatures) + if err != nil { + return bundle, err + } + + bundle.Data = map[string]interface{}{} } for _, f := range descriptors { - var buf bytes.Buffer - n, err := f.Read(&buf, r.sizeLimitBytes) - f.Close() // always close, even on error - - if err != nil && err != io.EOF { + buf, err := readFile(f, r.sizeLimitBytes) + if err != nil { return bundle, err - } else if err == nil && n >= r.sizeLimitBytes { - return bundle, fmt.Errorf("bundle file exceeded max size (%v bytes)", r.sizeLimitBytes-1) } // verify the file content - if !bundle.Signatures.isEmpty() { + if bundle.Type() == SnapshotBundleType && !bundle.Signatures.isEmpty() { path := f.Path() if r.baseDir != "" { path = f.URL() @@ -431,6 +499,17 @@ func (r *Reader) Read() (Bundle, error) { if strings.HasSuffix(path, RegoExt) { fullPath := r.fullPath(path) + bs := buf.Bytes() + + if r.lazyLoadingMode { + p := fullPath + if r.name != "" { + p = modulePathWithPrefix(r.name, fullPath) + } + + raw = append(raw, Raw{Path: p, Value: bs}) + } + r.metrics.Timer(metrics.RegoModuleParse).Start() module, err := ast.ParseModuleWithOpts(fullPath, buf.String(), ast.ParserOptions{ProcessAnnotation: r.processAnnotations}) r.metrics.Timer(metrics.RegoModuleParse).Stop() @@ -441,18 +520,28 @@ func (r *Reader) Read() (Bundle, error) { mf := ModuleFile{ URL: f.URL(), Path: fullPath, - Raw: buf.Bytes(), + Raw: bs, Parsed: module, } bundle.Modules = append(bundle.Modules, mf) - } else if filepath.Base(path) == WasmFile { bundle.WasmModules = append(bundle.WasmModules, WasmModuleFile{ URL: f.URL(), Path: r.fullPath(path), Raw: buf.Bytes(), }) + } else if filepath.Base(path) == PlanFile { + bundle.PlanModules = append(bundle.PlanModules, PlanModuleFile{ + URL: f.URL(), + Path: r.fullPath(path), + Raw: buf.Bytes(), + }) } else if filepath.Base(path) == dataFile { + if r.lazyLoadingMode { + raw = append(raw, Raw{Path: path, Value: buf.Bytes()}) + continue + } + var value interface{} r.metrics.Timer(metrics.RegoDataParse).Start() @@ -460,14 +549,18 @@ func (r *Reader) Read() (Bundle, error) { r.metrics.Timer(metrics.RegoDataParse).Stop() if err != nil { - return bundle, errors.Wrapf(err, "bundle load failed on %v", r.fullPath(path)) + return bundle, fmt.Errorf("bundle load failed on %v: %w", r.fullPath(path), err) } if err := insertValue(&bundle, path, value); err != nil { return bundle, err } - } else if filepath.Base(path) == yamlDataFile { + } else if filepath.Base(path) == yamlDataFile || filepath.Base(path) == ymlDataFile { + if r.lazyLoadingMode { + raw = append(raw, Raw{Path: path, Value: buf.Bytes()}) + continue + } var value interface{} @@ -476,7 +569,7 @@ func (r *Reader) Read() (Bundle, error) { r.metrics.Timer(metrics.RegoDataParse).Stop() if err != nil { - return bundle, errors.Wrapf(err, "bundle load failed on %v", r.fullPath(path)) + return bundle, fmt.Errorf("bundle load failed on %v: %w", r.fullPath(path), err) } if err := insertValue(&bundle, path, value); err != nil { @@ -485,13 +578,27 @@ func (r *Reader) Read() (Bundle, error) { } else if strings.HasSuffix(path, ManifestExt) { if err := util.NewJSONDecoder(&buf).Decode(&bundle.Manifest); err != nil { - return bundle, errors.Wrap(err, "bundle load failed on manifest decode") + return bundle, fmt.Errorf("bundle load failed on manifest decode: %w", err) } } } + if bundle.Type() == DeltaBundleType { + if len(bundle.Data) != 0 { + return bundle, fmt.Errorf("delta bundle expected to contain only patch file but data files found") + } + + if len(bundle.Modules) != 0 { + return bundle, fmt.Errorf("delta bundle expected to contain only patch file but policy files found") + } + + if len(bundle.WasmModules) != 0 { + return bundle, fmt.Errorf("delta bundle expected to contain only patch file but wasm files found") + } + } + // check if the bundle signatures specify any files that weren't found in the bundle - if len(r.files) != 0 { + if bundle.Type() == SnapshotBundleType && len(r.files) != 0 { extra := []string{} for k := range r.files { extra = append(extra, k) @@ -524,21 +631,24 @@ func (r *Reader) Read() (Bundle, error) { b, err := json.Marshal(&bundle.Manifest) if err != nil { - return bundle, errors.Wrap(err, "bundle load failed on manifest marshal") + return bundle, fmt.Errorf("bundle load failed on manifest marshal: %w", err) } err = util.UnmarshalJSON(b, &metadata) if err != nil { - return bundle, errors.Wrap(err, "bundle load failed on manifest unmarshal") + return bundle, fmt.Errorf("bundle load failed on manifest unmarshal: %w", err) } // For backwards compatibility always write to the old unnamed manifest path // This will *not* be correct if >1 bundle is in use... if err := bundle.insertData(legacyManifestStoragePath, metadata); err != nil { - return bundle, errors.Wrapf(err, "bundle load failed on %v", legacyRevisionStoragePath) + return bundle, fmt.Errorf("bundle load failed on %v: %w", legacyRevisionStoragePath, err) } } + bundle.Etag = r.etag + bundle.Raw = raw + return bundle, nil } @@ -632,36 +742,48 @@ func (w *Writer) Write(bundle Bundle) error { gw := gzip.NewWriter(w.w) tw := tar.NewWriter(gw) - var buf bytes.Buffer + bundleType := bundle.Type() - if err := json.NewEncoder(&buf).Encode(bundle.Data); err != nil { - return err - } + if bundleType == SnapshotBundleType { + var buf bytes.Buffer - if err := archive.WriteFile(tw, "data.json", buf.Bytes()); err != nil { - return err - } + if err := json.NewEncoder(&buf).Encode(bundle.Data); err != nil { + return err + } - for _, module := range bundle.Modules { - path := module.URL - if w.usePath { - path = module.Path + if err := archive.WriteFile(tw, "data.json", buf.Bytes()); err != nil { + return err + } + + for _, module := range bundle.Modules { + path := module.URL + if w.usePath { + path = module.Path + } + + if err := archive.WriteFile(tw, path, module.Raw); err != nil { + return err + } } - if err := archive.WriteFile(tw, path, module.Raw); err != nil { + if err := w.writeWasm(tw, bundle); err != nil { return err } - } - if err := w.writeWasm(tw, bundle); err != nil { - return err - } + if err := writeSignatures(tw, bundle); err != nil { + return err + } - if err := writeManifest(tw, bundle); err != nil { - return err + if err := w.writePlan(tw, bundle); err != nil { + return err + } + } else if bundleType == DeltaBundleType { + if err := writePatch(tw, bundle); err != nil { + return err + } } - if err := writeSignatures(tw, bundle); err != nil { + if err := writeManifest(tw, bundle); err != nil { return err } @@ -695,6 +817,22 @@ func (w *Writer) writeWasm(tw *tar.Writer, bundle Bundle) error { return nil } +func (w *Writer) writePlan(tw *tar.Writer, bundle Bundle) error { + for _, wm := range bundle.PlanModules { + path := wm.URL + if w.usePath { + path = wm.Path + } + + err := archive.WriteFile(tw, path, wm.Raw) + if err != nil { + return err + } + } + + return nil +} + func writeManifest(tw *tar.Writer, bundle Bundle) error { if bundle.Manifest.Equal(Manifest{}) { @@ -710,6 +848,17 @@ func writeManifest(tw *tar.Writer, bundle Bundle) error { return archive.WriteFile(tw, ManifestExt, buf.Bytes()) } +func writePatch(tw *tar.Writer, bundle Bundle) error { + + var buf bytes.Buffer + + if err := json.NewEncoder(&buf).Encode(bundle.Patch); err != nil { + return err + } + + return archive.WriteFile(tw, patchFile, buf.Bytes()) +} + func writeSignatures(tw *tar.Writer, bundle Bundle) error { if bundle.Signatures.isEmpty() { @@ -750,11 +899,37 @@ func hashBundleFiles(hash SignatureHasher, b *Bundle) ([]FileInfo, error) { files = append(files, NewFile(strings.TrimPrefix(wasmModule.Path, "/"), hex.EncodeToString(bs), defaultHashingAlg)) } - bs, err = hash.HashFile(b.Manifest) - if err != nil { - return files, err + for _, planmodule := range b.PlanModules { + bs, err := hash.HashFile(planmodule.Raw) + if err != nil { + return files, err + } + files = append(files, NewFile(strings.TrimPrefix(planmodule.Path, "/"), hex.EncodeToString(bs), defaultHashingAlg)) + } + + // If the manifest is essentially empty, don't add it to the signatures since it + // won't be written to the bundle. Otherwise: + // parse the manifest into a JSON structure; + // then recursively order the fields of all objects alphabetically and then apply + // the hash function to result to compute the hash. + if !b.Manifest.Equal(Manifest{}) { + mbs, err := json.Marshal(b.Manifest) + if err != nil { + return files, err + } + + var result map[string]interface{} + if err := util.Unmarshal(mbs, &result); err != nil { + return files, err + } + + bs, err = hash.HashFile(result) + if err != nil { + return files, err + } + + files = append(files, NewFile(strings.TrimPrefix(ManifestExt, "/"), hex.EncodeToString(bs), defaultHashingAlg)) } - files = append(files, NewFile(strings.TrimPrefix(ManifestExt, "/"), hex.EncodeToString(bs), defaultHashingAlg)) return files, err } @@ -958,6 +1133,14 @@ func (b *Bundle) readData(key []string) *interface{} { return &child } +// Type returns the type of the bundle. +func (b *Bundle) Type() string { + if len(b.Patch.Data) != 0 { + return DeltaBundleType + } + return SnapshotBundleType +} + func mktree(path []string, value interface{}) (map[string]interface{}, error) { if len(path) == 0 { // For 0 length path the value is the full tree. @@ -1017,6 +1200,7 @@ func Merge(bundles []*Bundle) (*Bundle, error) { result.Manifest.WasmResolvers = append(result.Manifest.WasmResolvers, b.Manifest.WasmResolvers...) result.WasmModules = append(result.WasmModules, b.WasmModules...) + result.PlanModules = append(result.PlanModules, b.PlanModules...) } @@ -1072,7 +1256,13 @@ func rootContains(root []string, other []string) bool { } func insertValue(b *Bundle, path string, value interface{}) error { + if err := b.insertData(getNormalizedPath(path), value); err != nil { + return fmt.Errorf("bundle load failed on %v: %w", path, err) + } + return nil +} +func getNormalizedPath(path string) []string { // Remove leading / and . characters from the directory path. If the bundle // was written with OPA then the paths will contain a leading slash. On the // other hand, if the path is empty, filepath.Dir will return '.'. @@ -1083,10 +1273,7 @@ func insertValue(b *Bundle, path string, value interface{}) error { if dirpath != "" { key = strings.Split(dirpath, "/") } - if err := b.insertData(key, value); err != nil { - return errors.Wrapf(err, "bundle load failed on %v", path) - } - return nil + return key } func dfs(value interface{}, path string, fn func(string, interface{}) (bool, error)) error { @@ -1127,9 +1314,10 @@ func IsStructuredDoc(name string) bool { filepath.Base(name) == SignaturesFile || filepath.Base(name) == ManifestExt } -func listSignaturesAndDescriptors(loader DirectoryLoader, skipVerify bool, sizeLimitBytes int64) (SignaturesConfig, []*Descriptor, error) { +func preProcessBundle(loader DirectoryLoader, skipVerify bool, sizeLimitBytes int64) (SignaturesConfig, Patch, []*Descriptor, error) { descriptors := []*Descriptor{} var signatures SignaturesConfig + var patch Patch for { f, err := loader.NextFile() @@ -1138,26 +1326,54 @@ func listSignaturesAndDescriptors(loader DirectoryLoader, skipVerify bool, sizeL } if err != nil { - return signatures, nil, errors.Wrap(err, "bundle read failed") + return signatures, patch, nil, fmt.Errorf("bundle read failed: %w", err) } // check for the signatures file if !skipVerify && strings.HasSuffix(f.Path(), SignaturesFile) { - var buf bytes.Buffer - n, err := f.Read(&buf, sizeLimitBytes) - f.Close() // always close, even on error - if err != nil && err != io.EOF { - return signatures, nil, err - } else if err == nil && n >= sizeLimitBytes { - return signatures, nil, fmt.Errorf("bundle signatures file exceeded max size (%v bytes)", sizeLimitBytes-1) + buf, err := readFile(f, sizeLimitBytes) + if err != nil { + return signatures, patch, nil, err } if err := util.NewJSONDecoder(&buf).Decode(&signatures); err != nil { - return signatures, nil, errors.Wrap(err, "bundle load failed on signatures decode") + return signatures, patch, nil, fmt.Errorf("bundle load failed on signatures decode: %w", err) } } else if !strings.HasSuffix(f.Path(), SignaturesFile) { descriptors = append(descriptors, f) + + if filepath.Base(f.Path()) == patchFile { + + var b bytes.Buffer + tee := io.TeeReader(f.reader, &b) + f.reader = tee + + buf, err := readFile(f, sizeLimitBytes) + if err != nil { + return signatures, patch, nil, err + } + + if err := util.NewJSONDecoder(&buf).Decode(&patch); err != nil { + return signatures, patch, nil, fmt.Errorf("bundle load failed on patch decode: %w", err) + } + + f.reader = &b + } } } - return signatures, descriptors, nil + return signatures, patch, descriptors, nil +} + +func readFile(f *Descriptor, sizeLimitBytes int64) (bytes.Buffer, error) { + var buf bytes.Buffer + n, err := f.Read(&buf, sizeLimitBytes) + f.Close() // always close, even on error + + if err != nil && err != io.EOF { + return buf, err + } else if err == nil && n >= sizeLimitBytes { + return buf, fmt.Errorf("bundle file '%v' exceeded max size (%v bytes)", strings.TrimPrefix(f.Path(), "/"), sizeLimitBytes-1) + } + + return buf, nil } diff --git a/vendor/github.com/open-policy-agent/opa/bundle/file.go b/vendor/github.com/open-policy-agent/opa/bundle/file.go index 040c0af3..5793a48f 100644 --- a/vendor/github.com/open-policy-agent/opa/bundle/file.go +++ b/vendor/github.com/open-policy-agent/opa/bundle/file.go @@ -4,14 +4,18 @@ import ( "archive/tar" "bytes" "compress/gzip" + "fmt" "io" "os" "path" "path/filepath" + "sort" "strings" "sync" - "github.com/pkg/errors" + "github.com/open-policy-agent/opa/loader/filter" + + "github.com/open-policy-agent/opa/storage" ) // Descriptor contains information about a file and @@ -43,7 +47,7 @@ func (f *lazyFile) Read(b []byte) (int, error) { if f.file == nil { if f.file, err = os.Open(f.path); err != nil { - return 0, errors.Wrapf(err, "failed to open file %s", f.path) + return 0, fmt.Errorf("failed to open file %s: %w", f.path, err) } } @@ -110,12 +114,14 @@ type DirectoryLoader interface { // NextFile must return io.EOF if there is no next value. The returned // descriptor should *always* be closed when no longer needed. NextFile() (*Descriptor, error) + WithFilter(filter filter.LoaderFilter) DirectoryLoader } type dirLoader struct { - root string - files []string - idx int + root string + files []string + idx int + filter filter.LoaderFilter } // NewDirectoryLoader returns a basic DirectoryLoader implementation @@ -141,6 +147,12 @@ func NewDirectoryLoader(root string) DirectoryLoader { return &d } +// WithFilter specifies the filter object to use to filter files while loading bundles +func (d *dirLoader) WithFilter(filter filter.LoaderFilter) DirectoryLoader { + d.filter = filter + return d +} + // NextFile iterates to the next file in the directory tree // and returns a file Descriptor for the file. func (d *dirLoader) NextFile() (*Descriptor, error) { @@ -149,12 +161,19 @@ func (d *dirLoader) NextFile() (*Descriptor, error) { d.files = []string{} err := filepath.Walk(d.root, func(path string, info os.FileInfo, err error) error { if info != nil && info.Mode().IsRegular() { + if d.filter != nil && d.filter(filepath.ToSlash(path), info, getdepth(path, false)) { + return nil + } d.files = append(d.files, filepath.ToSlash(path)) + } else if info != nil && info.Mode().IsDir() { + if d.filter != nil && d.filter(filepath.ToSlash(path), info, getdepth(path, true)) { + return filepath.SkipDir + } } return nil }) if err != nil { - return nil, errors.Wrap(err, "failed to list files") + return nil, fmt.Errorf("failed to list files: %w", err) } } @@ -188,11 +207,15 @@ type tarballLoader struct { tr *tar.Reader files []file idx int + filter filter.LoaderFilter + skipDir map[string]struct{} } type file struct { name string reader io.Reader + path storage.Path + raw []byte } // NewTarballLoader is deprecated. Use NewTarballLoaderWithBaseURL instead. @@ -214,13 +237,19 @@ func NewTarballLoaderWithBaseURL(r io.Reader, baseURL string) DirectoryLoader { return &l } +// WithFilter specifies the filter object to use to filter files while loading bundles +func (t *tarballLoader) WithFilter(filter filter.LoaderFilter) DirectoryLoader { + t.filter = filter + return t +} + // NextFile iterates to the next file in the directory tree // and returns a file Descriptor for the file. func (t *tarballLoader) NextFile() (*Descriptor, error) { if t.tr == nil { gr, err := gzip.NewReader(t.r) if err != nil { - return nil, errors.Wrap(err, "archive read failed") + return nil, fmt.Errorf("archive read failed: %w", err) } t.tr = tar.NewReader(gr) @@ -229,6 +258,10 @@ func (t *tarballLoader) NextFile() (*Descriptor, error) { if t.files == nil { t.files = []file{} + if t.skipDir == nil { + t.skipDir = map[string]struct{}{} + } + for { header, err := t.tr.Next() if err == io.EOF { @@ -241,16 +274,48 @@ func (t *tarballLoader) NextFile() (*Descriptor, error) { // Keep iterating on the archive until we find a normal file if header.Typeflag == tar.TypeReg { + + if t.filter != nil { + + if t.filter(filepath.ToSlash(header.Name), header.FileInfo(), getdepth(header.Name, false)) { + continue + } + + basePath := strings.Trim(filepath.Dir(filepath.ToSlash(header.Name)), "/") + + // check if the directory is to be skipped + if _, ok := t.skipDir[basePath]; ok { + continue + } + + match := false + for p := range t.skipDir { + if strings.HasPrefix(basePath, p) { + match = true + break + } + } + + if match { + continue + } + } + f := file{name: header.Name} var buf bytes.Buffer if _, err := io.Copy(&buf, t.tr); err != nil { - return nil, errors.Wrapf(err, "failed to copy file %s", header.Name) + return nil, fmt.Errorf("failed to copy file %s: %w", header.Name, err) } f.reader = &buf t.files = append(t.files, f) + } else if header.Typeflag == tar.TypeDir { + cleanedPath := filepath.ToSlash(header.Name) + if t.filter != nil && t.filter(cleanedPath, header.FileInfo(), getdepth(header.Name, true)) { + t.skipDir[strings.Trim(cleanedPath, "/")] = struct{}{} + } } } } @@ -266,3 +331,83 @@ func (t *tarballLoader) NextFile() (*Descriptor, error) { return newDescriptor(path.Join(t.baseURL, f.name), f.name, f.reader), nil } + +// Next implements the storage.Iterator interface. +// It iterates to the next policy or data file in the directory tree +// and returns a storage.Update for the file. +func (it *iterator) Next() (*storage.Update, error) { + + if it.files == nil { + it.files = []file{} + + for _, item := range it.raw { + f := file{name: item.Path} + + fpath := strings.TrimLeft(filepath.ToSlash(filepath.Dir(f.name)), "/.") + if strings.HasSuffix(f.name, RegoExt) { + fpath = strings.Trim(f.name, "/") + } + + p, ok := storage.ParsePathEscaped("/" + fpath) + if !ok { + return nil, fmt.Errorf("storage path invalid: %v", f.name) + } + f.path = p + + f.raw = item.Value + + it.files = append(it.files, f) + } + + sortFilePathAscend(it.files) + } + + // If done reading files then just return io.EOF + // errors for each NextFile() call + if it.idx >= len(it.files) { + return nil, io.EOF + } + + f := it.files[it.idx] + it.idx++ + + isPolicy := false + if strings.HasSuffix(f.name, RegoExt) { + isPolicy = true + } + + return &storage.Update{ + Path: f.path, + Value: f.raw, + IsPolicy: isPolicy, + }, nil +} + +type iterator struct { + raw []Raw + files []file + idx int +} + +func NewIterator(raw []Raw) storage.Iterator { + it := iterator{ + raw: raw, + } + return &it +} + +func sortFilePathAscend(files []file) { + sort.Slice(files, func(i, j int) bool { + return len(files[i].path) < len(files[j].path) + }) +} + +func getdepth(path string, isDir bool) int { + if isDir { + cleanedPath := strings.Trim(filepath.ToSlash(path), "/") + return len(strings.Split(cleanedPath, "/")) + } + + basePath := strings.Trim(filepath.Dir(filepath.ToSlash(path)), "/") + return len(strings.Split(basePath, "/")) +} diff --git a/vendor/github.com/open-policy-agent/opa/bundle/filefs.go b/vendor/github.com/open-policy-agent/opa/bundle/filefs.go index f587456e..5f331739 100644 --- a/vendor/github.com/open-policy-agent/opa/bundle/filefs.go +++ b/vendor/github.com/open-policy-agent/opa/bundle/filefs.go @@ -7,7 +7,10 @@ import ( "fmt" "io" "io/fs" + "path/filepath" "sync" + + "github.com/open-policy-agent/opa/loader/filter" ) const ( @@ -19,6 +22,7 @@ type dirLoaderFS struct { filesystem fs.FS files []string idx int + filter filter.LoaderFilter } // NewFSLoader returns a basic DirectoryLoader implementation @@ -28,11 +32,6 @@ func NewFSLoader(filesystem fs.FS) (DirectoryLoader, error) { filesystem: filesystem, } - err := fs.WalkDir(d.filesystem, defaultFSLoaderRoot, d.walkDir) - if err != nil { - return nil, fmt.Errorf("failed to list files: %w", err) - } - return &d, nil } @@ -41,19 +40,46 @@ func (d *dirLoaderFS) walkDir(path string, dirEntry fs.DirEntry, err error) erro return err } - if dirEntry != nil && dirEntry.Type().IsRegular() { - d.files = append(d.files, path) - } + if dirEntry != nil { + info, err := dirEntry.Info() + if err != nil { + return err + } + + if dirEntry.Type().IsRegular() { + if d.filter != nil && d.filter(filepath.ToSlash(path), info, getdepth(path, false)) { + return nil + } + d.files = append(d.files, path) + } else if dirEntry.Type().IsDir() { + if d.filter != nil && d.filter(filepath.ToSlash(path), info, getdepth(path, true)) { + return fs.SkipDir + } + } + } return nil } +// WithFilter specifies the filter object to use to filter files while loading bundles +func (d *dirLoaderFS) WithFilter(filter filter.LoaderFilter) DirectoryLoader { + d.filter = filter + return d +} + // NextFile iterates to the next file in the directory tree // and returns a file Descriptor for the file. func (d *dirLoaderFS) NextFile() (*Descriptor, error) { d.Lock() defer d.Unlock() + if d.files == nil { + err := fs.WalkDir(d.filesystem, defaultFSLoaderRoot, d.walkDir) + if err != nil { + return nil, fmt.Errorf("failed to list files: %w", err) + } + } + // If done reading files then just return io.EOF // errors for each NextFile() call if d.idx >= len(d.files) { diff --git a/vendor/github.com/open-policy-agent/opa/bundle/store.go b/vendor/github.com/open-policy-agent/opa/bundle/store.go index 0dc77964..ab8af205 100644 --- a/vendor/github.com/open-policy-agent/opa/bundle/store.go +++ b/vendor/github.com/open-policy-agent/opa/bundle/store.go @@ -9,8 +9,11 @@ import ( "encoding/base64" "encoding/json" "fmt" + "path/filepath" + "strings" "github.com/open-policy-agent/opa/ast" + "github.com/open-policy-agent/opa/internal/json/patch" "github.com/open-policy-agent/opa/metrics" "github.com/open-policy-agent/opa/storage" "github.com/open-policy-agent/opa/util" @@ -26,6 +29,11 @@ func ManifestStoragePath(name string) storage.Path { return append(BundlesBasePath, name, "manifest") } +// EtagStoragePath is the storage path used for the given named bundle etag. +func EtagStoragePath(name string) storage.Path { + return append(BundlesBasePath, name, "etag") +} + func namedBundlePath(name string) storage.Path { return append(BundlesBasePath, name) } @@ -77,6 +85,11 @@ func WriteManifestToStore(ctx context.Context, store storage.Store, txn storage. return write(ctx, store, txn, ManifestStoragePath(name), manifest) } +// WriteEtagToStore will write the bundle etag into the storage. This function is called when the bundle is activated. +func WriteEtagToStore(ctx context.Context, store storage.Store, txn storage.Transaction, name, etag string) error { + return write(ctx, store, txn, EtagStoragePath(name), etag) +} + func write(ctx context.Context, store storage.Store, txn storage.Transaction, path storage.Path, value interface{}) error { if err := util.RoundTrip(&value); err != nil { return err @@ -99,10 +112,22 @@ func write(ctx context.Context, store storage.Store, txn storage.Transaction, pa func EraseManifestFromStore(ctx context.Context, store storage.Store, txn storage.Transaction, name string) error { path := namedBundlePath(name) err := store.Write(ctx, txn, storage.RemoveOp, path, nil) - if err != nil && !storage.IsNotFound(err) { - return err + return suppressNotFound(err) +} + +// eraseBundleEtagFromStore will remove the bundle etag from storage. This function is called +// when the bundle is deactivated. +func eraseBundleEtagFromStore(ctx context.Context, store storage.Store, txn storage.Transaction, name string) error { + path := EtagStoragePath(name) + err := store.Write(ctx, txn, storage.RemoveOp, path, nil) + return suppressNotFound(err) +} + +func suppressNotFound(err error) error { + if err == nil || storage.IsNotFound(err) { + return nil } - return nil + return err } func writeWasmModulesToStore(ctx context.Context, store storage.Store, txn storage.Transaction, name string, b *Bundle) error { @@ -121,10 +146,7 @@ func eraseWasmModulesFromStore(ctx context.Context, store storage.Store, txn sto path := wasmModulePath(name) err := store.Write(ctx, txn, storage.RemoveOp, path, nil) - if err != nil && !storage.IsNotFound(err) { - return err - } - return nil + return suppressNotFound(err) } // ReadWasmMetadataFromStore will read Wasm module resolver metadata from the store. @@ -235,11 +257,7 @@ func ReadBundleMetadataFromStore(ctx context.Context, store storage.Store, txn s func readMetadataFromStore(ctx context.Context, store storage.Store, txn storage.Transaction, path storage.Path) (map[string]interface{}, error) { value, err := store.Read(ctx, txn, path) if err != nil { - if storageErr, ok := err.(*storage.Error); ok && storageErr.Code == storage.NotFoundErr { - return nil, nil - } - - return nil, err + return nil, suppressNotFound(err) } data, ok := value.(map[string]interface{}) @@ -250,6 +268,27 @@ func readMetadataFromStore(ctx context.Context, store storage.Store, txn storage return data, nil } +// ReadBundleEtagFromStore returns the etag for the specified bundle. +// If the bundle is not activated, this function will return +// storage NotFound error. +func ReadBundleEtagFromStore(ctx context.Context, store storage.Store, txn storage.Transaction, name string) (string, error) { + return readEtagFromStore(ctx, store, txn, EtagStoragePath(name)) +} + +func readEtagFromStore(ctx context.Context, store storage.Store, txn storage.Transaction, path storage.Path) (string, error) { + value, err := store.Read(ctx, txn, path) + if err != nil { + return "", err + } + + str, ok := value.(string) + if !ok { + return "", fmt.Errorf("corrupt bundle etag") + } + + return str, nil +} + // ActivateOpts defines options for the Activate API call. type ActivateOpts struct { Ctx context.Context @@ -283,13 +322,13 @@ type DeactivateOpts struct { func Deactivate(opts *DeactivateOpts) error { erase := map[string]struct{}{} for name := range opts.BundleNames { - if roots, err := ReadBundleRootsFromStore(opts.Ctx, opts.Store, opts.Txn, name); err == nil { - for _, root := range roots { - erase[root] = struct{}{} - } - } else if !storage.IsNotFound(err) { + roots, err := ReadBundleRootsFromStore(opts.Ctx, opts.Store, opts.Txn, name) + if suppressNotFound(err) != nil { return err } + for _, root := range roots { + erase[root] = struct{}{} + } } _, err := eraseBundles(opts.Ctx, opts.Store, opts.Txn, opts.BundleNames, erase) return err @@ -300,21 +339,28 @@ func activateBundles(opts *ActivateOpts) error { // Build collections of bundle names, modules, and roots to erase erase := map[string]struct{}{} names := map[string]struct{}{} + deltaBundles := map[string]*Bundle{} + snapshotBundles := map[string]*Bundle{} for name, b := range opts.Bundles { - names[name] = struct{}{} + if b.Type() == DeltaBundleType { + deltaBundles[name] = b + } else { + snapshotBundles[name] = b + names[name] = struct{}{} - if roots, err := ReadBundleRootsFromStore(opts.Ctx, opts.Store, opts.Txn, name); err == nil { + roots, err := ReadBundleRootsFromStore(opts.Ctx, opts.Store, opts.Txn, name) + if suppressNotFound(err) != nil { + return err + } for _, root := range roots { erase[root] = struct{}{} } - } else if !storage.IsNotFound(err) { - return err - } - // Erase data at new roots to prepare for writing the new data - for _, root := range *b.Manifest.Roots { - erase[root] = struct{}{} + // Erase data at new roots to prepare for writing the new data + for _, root := range *b.Manifest.Roots { + erase[root] = struct{}{} + } } } @@ -325,23 +371,77 @@ func activateBundles(opts *ActivateOpts) error { return err } + if len(deltaBundles) != 0 { + err := activateDeltaBundles(opts, deltaBundles) + if err != nil { + return err + } + } + // Erase data and policies at new + old roots, and remove the old - // manifests before activating a new bundles. + // manifests before activating a new snapshot bundle. remaining, err := eraseBundles(opts.Ctx, opts.Store, opts.Txn, names, erase) if err != nil { return err } - for _, b := range opts.Bundles { - // Write data from each new bundle into the store. Only write under the - // roots contained in their manifest. This should be done *before* the - // policies so that path conflict checks can occur. - if err := writeData(opts.Ctx, opts.Store, opts.Txn, *b.Manifest.Roots, b.Data); err != nil { - return err + // Validate data in bundle does not contain paths outside the bundle's roots. + for _, b := range snapshotBundles { + + if b.lazyLoadingMode { + + for _, item := range b.Raw { + path := filepath.ToSlash(item.Path) + + if filepath.Base(path) == dataFile || filepath.Base(path) == yamlDataFile { + var val map[string]json.RawMessage + err = util.Unmarshal(item.Value, &val) + if err == nil { + err = doDFS(val, filepath.Dir(strings.Trim(path, "/")), *b.Manifest.Roots) + if err != nil { + return err + } + } else { + // Build an object for the value + p := getNormalizedPath(path) + + if len(p) == 0 { + return fmt.Errorf("root value must be object") + } + + // verify valid YAML or JSON value + var x interface{} + err := util.Unmarshal(item.Value, &x) + if err != nil { + return err + } + + value := item.Value + dir := map[string]json.RawMessage{} + for i := len(p) - 1; i > 0; i-- { + dir[p[i]] = value + + bs, err := json.Marshal(dir) + if err != nil { + return err + } + + value = bs + dir = map[string]json.RawMessage{} + } + dir[p[0]] = value + + err = doDFS(dir, filepath.Dir(strings.Trim(path, "/")), *b.Manifest.Roots) + if err != nil { + return err + } + } + } + } } } - // Write and compile the modules all at once to avoid having to re-do work. + // Compile the modules all at once to avoid having to re-do work. remainingAndExtra := make(map[string]*ast.Module) for name, mod := range remaining { remainingAndExtra[name] = mod @@ -350,21 +450,26 @@ func activateBundles(opts *ActivateOpts) error { remainingAndExtra[name] = mod } - err = writeModules(opts.Ctx, opts.Store, opts.Txn, opts.Compiler, opts.Metrics, opts.Bundles, remainingAndExtra, opts.legacy) + err = compileModules(opts.Compiler, opts.Metrics, snapshotBundles, remainingAndExtra, opts.legacy) if err != nil { return err } - for name, b := range opts.Bundles { - // Always write manifests to the named location. If the plugin is in the older style config - // then also write to the old legacy unnamed location. - if err := WriteManifestToStore(opts.Ctx, opts.Store, opts.Txn, name, b.Manifest); err != nil { + if err := writeDataAndModules(opts.Ctx, opts.Store, opts.Txn, opts.TxnCtx, snapshotBundles, opts.legacy); err != nil { + return err + } + + if err := ast.CheckPathConflicts(opts.Compiler, storage.NonEmpty(opts.Ctx, opts.Store, opts.Txn)); len(err) > 0 { + return err + } + + for name, b := range snapshotBundles { + if err := writeManifestToStore(opts, name, b.Manifest); err != nil { return err } - if opts.legacy { - if err := LegacyWriteManifestToStore(opts.Ctx, opts.Store, opts.Txn, b.Manifest); err != nil { - return err - } + + if err := writeEtagToStore(opts, name, b.Etag); err != nil { + return err } if err := writeWasmModulesToStore(opts.Ctx, opts.Store, opts.Txn, name, b); err != nil { @@ -375,6 +480,107 @@ func activateBundles(opts *ActivateOpts) error { return nil } +func doDFS(obj map[string]json.RawMessage, path string, roots []string) error { + if len(roots) == 1 && roots[0] == "" { + return nil + } + + for key := range obj { + + newPath := filepath.Join(strings.Trim(path, "/"), key) + + // Note: filepath.Join can return paths with '\' separators, always use + // filepath.ToSlash to keep them normalized. + newPath = strings.TrimLeft(filepath.ToSlash(newPath), "/.") + + contains := false + prefix := false + if RootPathsContain(roots, newPath) { + contains = true + } else { + for i := range roots { + if strings.HasPrefix(strings.Trim(roots[i], "/"), newPath) { + prefix = true + break + } + } + } + + if !contains && !prefix { + return fmt.Errorf("manifest roots %v do not permit data at path '/%s' (hint: check bundle directory structure)", roots, newPath) + } + + if contains { + continue + } + + var next map[string]json.RawMessage + err := util.Unmarshal(obj[key], &next) + if err != nil { + return fmt.Errorf("manifest roots %v do not permit data at path '/%s' (hint: check bundle directory structure)", roots, newPath) + } + + if err := doDFS(next, newPath, roots); err != nil { + return err + } + } + return nil +} + +func activateDeltaBundles(opts *ActivateOpts, bundles map[string]*Bundle) error { + + // Check that the manifest roots and wasm resolvers in the delta bundle + // match with those currently in the store + for name, b := range bundles { + value, err := opts.Store.Read(opts.Ctx, opts.Txn, ManifestStoragePath(name)) + if err != nil { + if storage.IsNotFound(err) { + continue + } + return err + } + + bs, err := json.Marshal(value) + if err != nil { + return fmt.Errorf("corrupt manifest data: %w", err) + } + + var manifest Manifest + + err = util.UnmarshalJSON(bs, &manifest) + if err != nil { + return fmt.Errorf("corrupt manifest data: %w", err) + } + + if !b.Manifest.equalWasmResolversAndRoots(manifest) { + return fmt.Errorf("delta bundle '%s' has wasm resolvers or manifest roots that are different from those in the store", name) + } + } + + for _, b := range bundles { + err := applyPatches(opts.Ctx, opts.Store, opts.Txn, b.Patch.Data) + if err != nil { + return err + } + } + + if err := ast.CheckPathConflicts(opts.Compiler, storage.NonEmpty(opts.Ctx, opts.Store, opts.Txn)); len(err) > 0 { + return err + } + + for name, b := range bundles { + if err := writeManifestToStore(opts, name, b.Manifest); err != nil { + return err + } + + if err := writeEtagToStore(opts, name, b.Etag); err != nil { + return err + } + } + + return nil +} + // erase bundles by name and roots. This will clear all policies and data at its roots and remove its // manifest from storage. func eraseBundles(ctx context.Context, store storage.Store, txn storage.Transaction, names map[string]struct{}, roots map[string]struct{}) (map[string]*ast.Module, error) { @@ -389,15 +595,19 @@ func eraseBundles(ctx context.Context, store storage.Store, txn storage.Transact } for name := range names { - if err := EraseManifestFromStore(ctx, store, txn, name); err != nil && !storage.IsNotFound(err) { + if err := EraseManifestFromStore(ctx, store, txn, name); suppressNotFound(err) != nil { + return nil, err + } + + if err := LegacyEraseManifestFromStore(ctx, store, txn); suppressNotFound(err) != nil { return nil, err } - if err := LegacyEraseManifestFromStore(ctx, store, txn); err != nil && !storage.IsNotFound(err) { + if err := eraseBundleEtagFromStore(ctx, store, txn, name); suppressNotFound(err) != nil { return nil, err } - if err := eraseWasmModulesFromStore(ctx, store, txn, name); err != nil && !storage.IsNotFound(err) { + if err := eraseWasmModulesFromStore(ctx, store, txn, name); suppressNotFound(err) != nil { return nil, err } } @@ -411,11 +621,10 @@ func eraseData(ctx context.Context, store storage.Store, txn storage.Transaction if !ok { return fmt.Errorf("manifest root path invalid: %v", root) } + if len(path) > 0 { - if err := store.Write(ctx, txn, storage.RemoveOp, path, nil); err != nil { - if !storage.IsNotFound(err) { - return err - } + if err := store.Write(ctx, txn, storage.RemoveOp, path, nil); suppressNotFound(err) != nil { + return err } } } @@ -462,6 +671,70 @@ func erasePolicies(ctx context.Context, store storage.Store, txn storage.Transac return remaining, nil } +func writeManifestToStore(opts *ActivateOpts, name string, manifest Manifest) error { + // Always write manifests to the named location. If the plugin is in the older style config + // then also write to the old legacy unnamed location. + if err := WriteManifestToStore(opts.Ctx, opts.Store, opts.Txn, name, manifest); err != nil { + return err + } + + if opts.legacy { + if err := LegacyWriteManifestToStore(opts.Ctx, opts.Store, opts.Txn, manifest); err != nil { + return err + } + } + + return nil +} + +func writeEtagToStore(opts *ActivateOpts, name, etag string) error { + if err := WriteEtagToStore(opts.Ctx, opts.Store, opts.Txn, name, etag); err != nil { + return err + } + + return nil +} + +func writeDataAndModules(ctx context.Context, store storage.Store, txn storage.Transaction, txnCtx *storage.Context, bundles map[string]*Bundle, legacy bool) error { + params := storage.WriteParams + params.Context = txnCtx + + for name, b := range bundles { + if len(b.Raw) == 0 { + // Write data from each new bundle into the store. Only write under the + // roots contained in their manifest. + if err := writeData(ctx, store, txn, *b.Manifest.Roots, b.Data); err != nil { + return err + } + + for _, mf := range b.Modules { + var path string + + // For backwards compatibility, in legacy mode, upsert policies to + // the unprefixed path. + if legacy { + path = mf.Path + } else { + path = modulePathWithPrefix(name, mf.Path) + } + + if err := store.UpsertPolicy(ctx, txn, path, mf.Raw); err != nil { + return err + } + } + } else { + params.BasePaths = *b.Manifest.Roots + + err := store.Truncate(ctx, txn, params, NewIterator(b.Raw)) + if err != nil { + return fmt.Errorf("store truncate failed for bundle '%s': %v", name, err) + } + } + } + + return nil +} + func writeData(ctx context.Context, store storage.Store, txn storage.Transaction, roots []string, data map[string]interface{}) error { for _, root := range roots { path, ok := storage.ParsePathEscaped("/" + root) @@ -482,6 +755,43 @@ func writeData(ctx context.Context, store storage.Store, txn storage.Transaction return nil } +func compileModules(compiler *ast.Compiler, m metrics.Metrics, bundles map[string]*Bundle, extraModules map[string]*ast.Module, legacy bool) error { + + m.Timer(metrics.RegoModuleCompile).Start() + defer m.Timer(metrics.RegoModuleCompile).Stop() + + modules := map[string]*ast.Module{} + + // preserve any modules already on the compiler + for name, module := range compiler.Modules { + modules[name] = module + } + + // preserve any modules passed in from the store + for name, module := range extraModules { + modules[name] = module + } + + // include all the new bundle modules + for bundleName, b := range bundles { + if legacy { + for _, mf := range b.Modules { + modules[mf.Path] = mf.Parsed + } + } else { + for name, module := range b.ParsedModules(bundleName) { + modules[name] = module + } + } + } + + if compiler.Compile(modules); compiler.Failed() { + return compiler.Errors + } + + return nil +} + func writeModules(ctx context.Context, store storage.Store, txn storage.Transaction, compiler *ast.Compiler, m metrics.Metrics, bundles map[string]*Bundle, extraModules map[string]*ast.Module, legacy bool) error { m.Timer(metrics.RegoModuleCompile).Start() @@ -557,7 +867,7 @@ func lookup(path storage.Path, data map[string]interface{}) (interface{}, bool) func hasRootsOverlap(ctx context.Context, store storage.Store, txn storage.Transaction, bundles map[string]*Bundle) error { collisions := map[string][]string{} allBundles, err := ReadBundleNamesFromStore(ctx, store, txn) - if err != nil && !storage.IsNotFound(err) { + if suppressNotFound(err) != nil { return err } @@ -566,7 +876,7 @@ func hasRootsOverlap(ctx context.Context, store storage.Store, txn storage.Trans // Build a map of roots for existing bundles already in the system for _, name := range allBundles { roots, err := ReadBundleRootsFromStore(ctx, store, txn, name) - if err != nil && !storage.IsNotFound(err) { + if suppressNotFound(err) != nil { return err } allRoots[name] = roots @@ -607,6 +917,47 @@ func hasRootsOverlap(ctx context.Context, store storage.Store, txn storage.Trans return nil } +func applyPatches(ctx context.Context, store storage.Store, txn storage.Transaction, patches []PatchOperation) error { + for _, pat := range patches { + + // construct patch path + path, ok := patch.ParsePatchPathEscaped("/" + strings.Trim(pat.Path, "/")) + if !ok { + return fmt.Errorf("error parsing patch path") + } + + var op storage.PatchOp + switch pat.Op { + case "upsert": + op = storage.AddOp + + _, err := store.Read(ctx, txn, path[:len(path)-1]) + if err != nil { + if !storage.IsNotFound(err) { + return err + } + + if err := storage.MakeDir(ctx, store, txn, path[:len(path)-1]); err != nil { + return err + } + } + case "remove": + op = storage.RemoveOp + case "replace": + op = storage.ReplaceOp + default: + return fmt.Errorf("bad patch operation: %v", pat.Op) + } + + // apply the patch + if err := store.Write(ctx, txn, op, path, pat.Value); err != nil { + return err + } + } + + return nil +} + // Helpers for the older single (unnamed) bundle style manifest storage. // LegacyManifestStoragePath is the older unnamed bundle path for manifests to be stored. diff --git a/vendor/github.com/open-policy-agent/opa/bundle/verify.go b/vendor/github.com/open-policy-agent/opa/bundle/verify.go index f76baf2c..e85be835 100644 --- a/vendor/github.com/open-policy-agent/opa/bundle/verify.go +++ b/vendor/github.com/open-policy-agent/opa/bundle/verify.go @@ -16,8 +16,6 @@ import ( "github.com/open-policy-agent/opa/internal/jwx/jws" "github.com/open-policy-agent/opa/internal/jwx/jws/verify" "github.com/open-policy-agent/opa/util" - - "github.com/pkg/errors" ) const defaultVerifierID = "_default" @@ -92,12 +90,12 @@ func verifyJWTSignature(token string, bvc *VerificationConfig) (*DecodedSignatur var decodedHeader []byte if decodedHeader, err = base64.RawURLEncoding.DecodeString(parts[0]); err != nil { - return nil, errors.Wrap(err, "failed to base64 decode JWT headers") + return nil, fmt.Errorf("failed to base64 decode JWT headers: %w", err) } var hdr jws.StandardHeaders if err := json.Unmarshal(decodedHeader, &hdr); err != nil { - return nil, errors.Wrap(err, "failed to parse JWT headers") + return nil, fmt.Errorf("failed to parse JWT headers: %w", err) } payload, err := base64.RawURLEncoding.DecodeString(parts[1]) diff --git a/vendor/github.com/open-policy-agent/opa/capabilities/capabilities.go b/vendor/github.com/open-policy-agent/opa/capabilities/capabilities.go new file mode 100644 index 00000000..1266df1d --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/capabilities/capabilities.go @@ -0,0 +1,18 @@ +// Copyright 2021 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +//go:build go1.16 +// +build go1.16 + +package capabilities + +import ( + "embed" +) + +// FS contains the embedded capabilities/ directory of the built version, +// which has all the capabilities of previous versions: +// "v0.18.0.json" contains the capabilities JSON of version v0.18.0, etc +//go:embed *.json +var FS embed.FS diff --git a/vendor/github.com/open-policy-agent/opa/capabilities/v0.17.0.json b/vendor/github.com/open-policy-agent/opa/capabilities/v0.17.0.json new file mode 100644 index 00000000..b536350d --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/capabilities/v0.17.0.json @@ -0,0 +1,2392 @@ +{ + "builtins": [ + { + "name": "abs", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "all", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "and", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "\u0026" + }, + { + "name": "any", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "array.concat", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "array.slice", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "assign", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": ":=" + }, + { + "name": "base64.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "cast_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "cast_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "cast_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "null" + }, + "type": "function" + } + }, + { + "name": "cast_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "cast_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "cast_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "concat", + "decl": { + "args": [ + { + "type": "string" + }, + { + "of": [ + { + "of": { + "type": "string" + }, + "type": "set" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "count", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "crypto.md5", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha1", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha256", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "div", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "/" + }, + { + "name": "endswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "eq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "=" + }, + { + "name": "equal", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "==" + }, + { + "name": "format_int", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "glob.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "glob.quote_meta", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "gt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e" + }, + { + "name": "gte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e=" + }, + { + "name": "http.send", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "indexof", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "intersection", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode_verify", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign_raw", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_number", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "json.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "lower", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "lt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c" + }, + { + "name": "lte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c=" + }, + { + "name": "max", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "min", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "minus", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + "type": "function" + }, + "infix": "-" + }, + { + "name": "mul", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "*" + }, + { + "name": "neq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "!=" + }, + { + "name": "net.cidr_contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_expand", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_intersects", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_overlap", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "object.get", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "opa.runtime", + "decl": { + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "or", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "|" + }, + { + "name": "plus", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "+" + }, + { + "name": "product", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "re_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.find_all_string_submatch_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.find_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.globs_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.template_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "rego.parse_module", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "rem", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "%" + }, + { + "name": "replace", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "round", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "set_diff", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "sort", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "sprintf", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "startswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "strings.replace_n", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "string" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "substring", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "sum", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.clock", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.date", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.now_ns", + "decl": { + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_duration_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_ns", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_rfc3339_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.weekday", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "to_number", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "type": "string" + }, + { + "type": "boolean" + }, + { + "type": "null" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "trace", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "trim", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_left", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_prefix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_right", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_space", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_suffix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "type_name", + "decl": { + "args": [ + { + "of": [ + { + "type": "any" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "union", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "units.parse_bytes", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "upper", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode_object", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "walk", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "static": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "function" + }, + "relation": true + }, + { + "name": "yaml.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "yaml.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + } + ] +} diff --git a/vendor/github.com/open-policy-agent/opa/capabilities/v0.17.1.json b/vendor/github.com/open-policy-agent/opa/capabilities/v0.17.1.json new file mode 100644 index 00000000..b536350d --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/capabilities/v0.17.1.json @@ -0,0 +1,2392 @@ +{ + "builtins": [ + { + "name": "abs", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "all", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "and", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "\u0026" + }, + { + "name": "any", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "array.concat", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "array.slice", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "assign", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": ":=" + }, + { + "name": "base64.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "cast_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "cast_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "cast_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "null" + }, + "type": "function" + } + }, + { + "name": "cast_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "cast_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "cast_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "concat", + "decl": { + "args": [ + { + "type": "string" + }, + { + "of": [ + { + "of": { + "type": "string" + }, + "type": "set" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "count", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "crypto.md5", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha1", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha256", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "div", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "/" + }, + { + "name": "endswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "eq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "=" + }, + { + "name": "equal", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "==" + }, + { + "name": "format_int", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "glob.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "glob.quote_meta", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "gt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e" + }, + { + "name": "gte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e=" + }, + { + "name": "http.send", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "indexof", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "intersection", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode_verify", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign_raw", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_number", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "json.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "lower", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "lt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c" + }, + { + "name": "lte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c=" + }, + { + "name": "max", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "min", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "minus", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + "type": "function" + }, + "infix": "-" + }, + { + "name": "mul", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "*" + }, + { + "name": "neq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "!=" + }, + { + "name": "net.cidr_contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_expand", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_intersects", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_overlap", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "object.get", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "opa.runtime", + "decl": { + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "or", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "|" + }, + { + "name": "plus", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "+" + }, + { + "name": "product", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "re_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.find_all_string_submatch_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.find_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.globs_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.template_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "rego.parse_module", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "rem", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "%" + }, + { + "name": "replace", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "round", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "set_diff", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "sort", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "sprintf", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "startswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "strings.replace_n", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "string" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "substring", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "sum", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.clock", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.date", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.now_ns", + "decl": { + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_duration_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_ns", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_rfc3339_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.weekday", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "to_number", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "type": "string" + }, + { + "type": "boolean" + }, + { + "type": "null" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "trace", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "trim", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_left", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_prefix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_right", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_space", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_suffix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "type_name", + "decl": { + "args": [ + { + "of": [ + { + "type": "any" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "union", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "units.parse_bytes", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "upper", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode_object", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "walk", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "static": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "function" + }, + "relation": true + }, + { + "name": "yaml.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "yaml.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + } + ] +} diff --git a/vendor/github.com/open-policy-agent/opa/capabilities/v0.17.2.json b/vendor/github.com/open-policy-agent/opa/capabilities/v0.17.2.json new file mode 100644 index 00000000..1fbd4dbc --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/capabilities/v0.17.2.json @@ -0,0 +1,2525 @@ +{ + "builtins": [ + { + "name": "abs", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "all", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "and", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "\u0026" + }, + { + "name": "any", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "array.concat", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "array.slice", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "assign", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": ":=" + }, + { + "name": "base64.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "cast_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "cast_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "cast_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "null" + }, + "type": "function" + } + }, + { + "name": "cast_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "cast_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "cast_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "concat", + "decl": { + "args": [ + { + "type": "string" + }, + { + "of": [ + { + "of": { + "type": "string" + }, + "type": "set" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "count", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "crypto.md5", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha1", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha256", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "div", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "/" + }, + { + "name": "endswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "eq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "=" + }, + { + "name": "equal", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "==" + }, + { + "name": "format_int", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "glob.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "glob.quote_meta", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "gt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e" + }, + { + "name": "gte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e=" + }, + { + "name": "http.send", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "indexof", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "intersection", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode_verify", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign_raw", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_number", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "json.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "lower", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "lt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c" + }, + { + "name": "lte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c=" + }, + { + "name": "max", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "min", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "minus", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + "type": "function" + }, + "infix": "-" + }, + { + "name": "mul", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "*" + }, + { + "name": "neq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "!=" + }, + { + "name": "net.cidr_contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_expand", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_intersects", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_overlap", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "object.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.get", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.union", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "opa.runtime", + "decl": { + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "or", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "|" + }, + { + "name": "plus", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "+" + }, + { + "name": "product", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "re_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.find_all_string_submatch_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.find_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.globs_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.template_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "rego.parse_module", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "rem", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "%" + }, + { + "name": "replace", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "round", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "set_diff", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "sort", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "sprintf", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "startswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "strings.replace_n", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "string" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "substring", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "sum", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.clock", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.date", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.now_ns", + "decl": { + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_duration_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_ns", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_rfc3339_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.weekday", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "to_number", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "type": "string" + }, + { + "type": "boolean" + }, + { + "type": "null" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "trace", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "trim", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_left", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_prefix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_right", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_space", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_suffix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "type_name", + "decl": { + "args": [ + { + "of": [ + { + "type": "any" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "union", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "units.parse_bytes", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "upper", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode_object", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "walk", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "static": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "function" + }, + "relation": true + }, + { + "name": "yaml.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "yaml.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + } + ] +} diff --git a/vendor/github.com/open-policy-agent/opa/capabilities/v0.17.3.json b/vendor/github.com/open-policy-agent/opa/capabilities/v0.17.3.json new file mode 100644 index 00000000..1fbd4dbc --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/capabilities/v0.17.3.json @@ -0,0 +1,2525 @@ +{ + "builtins": [ + { + "name": "abs", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "all", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "and", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "\u0026" + }, + { + "name": "any", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "array.concat", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "array.slice", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "assign", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": ":=" + }, + { + "name": "base64.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "cast_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "cast_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "cast_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "null" + }, + "type": "function" + } + }, + { + "name": "cast_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "cast_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "cast_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "concat", + "decl": { + "args": [ + { + "type": "string" + }, + { + "of": [ + { + "of": { + "type": "string" + }, + "type": "set" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "count", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "crypto.md5", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha1", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha256", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "div", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "/" + }, + { + "name": "endswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "eq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "=" + }, + { + "name": "equal", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "==" + }, + { + "name": "format_int", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "glob.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "glob.quote_meta", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "gt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e" + }, + { + "name": "gte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e=" + }, + { + "name": "http.send", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "indexof", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "intersection", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode_verify", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign_raw", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_number", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "json.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "lower", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "lt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c" + }, + { + "name": "lte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c=" + }, + { + "name": "max", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "min", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "minus", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + "type": "function" + }, + "infix": "-" + }, + { + "name": "mul", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "*" + }, + { + "name": "neq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "!=" + }, + { + "name": "net.cidr_contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_expand", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_intersects", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_overlap", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "object.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.get", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.union", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "opa.runtime", + "decl": { + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "or", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "|" + }, + { + "name": "plus", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "+" + }, + { + "name": "product", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "re_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.find_all_string_submatch_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.find_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.globs_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.template_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "rego.parse_module", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "rem", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "%" + }, + { + "name": "replace", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "round", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "set_diff", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "sort", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "sprintf", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "startswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "strings.replace_n", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "string" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "substring", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "sum", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.clock", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.date", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.now_ns", + "decl": { + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_duration_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_ns", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_rfc3339_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.weekday", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "to_number", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "type": "string" + }, + { + "type": "boolean" + }, + { + "type": "null" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "trace", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "trim", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_left", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_prefix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_right", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_space", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_suffix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "type_name", + "decl": { + "args": [ + { + "of": [ + { + "type": "any" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "union", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "units.parse_bytes", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "upper", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode_object", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "walk", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "static": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "function" + }, + "relation": true + }, + { + "name": "yaml.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "yaml.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + } + ] +} diff --git a/vendor/github.com/open-policy-agent/opa/capabilities/v0.18.0.json b/vendor/github.com/open-policy-agent/opa/capabilities/v0.18.0.json new file mode 100644 index 00000000..8b8174d1 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/capabilities/v0.18.0.json @@ -0,0 +1,2685 @@ +{ + "builtins": [ + { + "name": "abs", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "all", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "and", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "\u0026" + }, + { + "name": "any", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "array.concat", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "array.slice", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "assign", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": ":=" + }, + { + "name": "base64.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "bits.and", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.lsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.negate", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.or", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.rsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.xor", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "cast_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "cast_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "cast_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "null" + }, + "type": "function" + } + }, + { + "name": "cast_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "cast_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "cast_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "concat", + "decl": { + "args": [ + { + "type": "string" + }, + { + "of": [ + { + "of": { + "type": "string" + }, + "type": "set" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "count", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "crypto.md5", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha1", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha256", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "div", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "/" + }, + { + "name": "endswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "eq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "=" + }, + { + "name": "equal", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "==" + }, + { + "name": "format_int", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "glob.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "glob.quote_meta", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "gt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e" + }, + { + "name": "gte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e=" + }, + { + "name": "http.send", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "indexof", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "intersection", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode_verify", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign_raw", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_number", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "json.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "lower", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "lt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c" + }, + { + "name": "lte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c=" + }, + { + "name": "max", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "min", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "minus", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + "type": "function" + }, + "infix": "-" + }, + { + "name": "mul", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "*" + }, + { + "name": "neq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "!=" + }, + { + "name": "net.cidr_contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_expand", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_intersects", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_overlap", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "object.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.get", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.union", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "opa.runtime", + "decl": { + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "or", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "|" + }, + { + "name": "plus", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "+" + }, + { + "name": "product", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "re_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.find_all_string_submatch_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.find_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.globs_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.template_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "rego.parse_module", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "rem", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "%" + }, + { + "name": "replace", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "round", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "set_diff", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "sort", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "sprintf", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "startswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "strings.replace_n", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "string" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "substring", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "sum", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.clock", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.date", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.now_ns", + "decl": { + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_duration_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_ns", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_rfc3339_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.weekday", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "to_number", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "type": "string" + }, + { + "type": "boolean" + }, + { + "type": "null" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "trace", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "trim", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_left", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_prefix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_right", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_space", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_suffix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "type_name", + "decl": { + "args": [ + { + "of": [ + { + "type": "any" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "union", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "units.parse_bytes", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "upper", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode_object", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "walk", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "static": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "function" + }, + "relation": true + }, + { + "name": "yaml.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "yaml.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + } + ] +} diff --git a/vendor/github.com/open-policy-agent/opa/capabilities/v0.19.0-rc1.json b/vendor/github.com/open-policy-agent/opa/capabilities/v0.19.0-rc1.json new file mode 100644 index 00000000..d660b620 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/capabilities/v0.19.0-rc1.json @@ -0,0 +1,2835 @@ +{ + "builtins": [ + { + "name": "abs", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "all", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "and", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "\u0026" + }, + { + "name": "any", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "array.concat", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "array.slice", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "assign", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": ":=" + }, + { + "name": "base64.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "bits.and", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.lsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.negate", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.or", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.rsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.xor", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "cast_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "cast_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "cast_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "null" + }, + "type": "function" + } + }, + { + "name": "cast_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "cast_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "cast_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "concat", + "decl": { + "args": [ + { + "type": "string" + }, + { + "of": [ + { + "of": { + "type": "string" + }, + "type": "set" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "count", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "crypto.md5", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha1", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha256", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "div", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "/" + }, + { + "name": "endswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "eq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "=" + }, + { + "name": "equal", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "==" + }, + { + "name": "format_int", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "glob.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "glob.quote_meta", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "gt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e" + }, + { + "name": "gte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e=" + }, + { + "name": "http.send", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "indexof", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "intersection", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode_verify", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign_raw", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_number", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "json.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "lower", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "lt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c" + }, + { + "name": "lte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c=" + }, + { + "name": "max", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "min", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "minus", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + "type": "function" + }, + "infix": "-" + }, + { + "name": "mul", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "*" + }, + { + "name": "neq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "!=" + }, + { + "name": "net.cidr_contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_contains_matches", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "static": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_expand", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_intersects", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_overlap", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "object.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.get", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.union", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "opa.runtime", + "decl": { + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "or", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "|" + }, + { + "name": "plus", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "+" + }, + { + "name": "product", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "re_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.find_all_string_submatch_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.find_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.globs_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.template_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "rego.parse_module", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "rem", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "%" + }, + { + "name": "replace", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "round", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "set_diff", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "sort", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "sprintf", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "startswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "strings.replace_n", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "string" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "substring", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "sum", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.clock", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.date", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.now_ns", + "decl": { + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_duration_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_ns", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_rfc3339_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.weekday", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "to_number", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "type": "string" + }, + { + "type": "boolean" + }, + { + "type": "null" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "trace", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "trim", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_left", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_prefix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_right", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_space", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_suffix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "type_name", + "decl": { + "args": [ + { + "of": [ + { + "type": "any" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "union", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "units.parse_bytes", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "upper", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode_object", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "walk", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "static": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "function" + }, + "relation": true + }, + { + "name": "yaml.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "yaml.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + } + ] +} diff --git a/vendor/github.com/open-policy-agent/opa/capabilities/v0.19.0.json b/vendor/github.com/open-policy-agent/opa/capabilities/v0.19.0.json new file mode 100644 index 00000000..fa9ae4e4 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/capabilities/v0.19.0.json @@ -0,0 +1,2858 @@ +{ + "builtins": [ + { + "name": "abs", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "all", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "and", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "\u0026" + }, + { + "name": "any", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "array.concat", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "array.slice", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "assign", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": ":=" + }, + { + "name": "base64.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "bits.and", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.lsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.negate", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.or", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.rsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.xor", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "cast_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "cast_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "cast_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "null" + }, + "type": "function" + } + }, + { + "name": "cast_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "cast_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "cast_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "concat", + "decl": { + "args": [ + { + "type": "string" + }, + { + "of": [ + { + "of": { + "type": "string" + }, + "type": "set" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "count", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "crypto.md5", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha1", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha256", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "div", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "/" + }, + { + "name": "endswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "eq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "=" + }, + { + "name": "equal", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "==" + }, + { + "name": "format_int", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "glob.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "glob.quote_meta", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "gt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e" + }, + { + "name": "gte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e=" + }, + { + "name": "http.send", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "indexof", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "intersection", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode_verify", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign_raw", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_number", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "json.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "lower", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "lt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c" + }, + { + "name": "lte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c=" + }, + { + "name": "max", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "min", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "minus", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + "type": "function" + }, + "infix": "-" + }, + { + "name": "mul", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "*" + }, + { + "name": "neq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "!=" + }, + { + "name": "net.cidr_contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_contains_matches", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "static": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_expand", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_intersects", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_overlap", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "object.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.get", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.union", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "opa.runtime", + "decl": { + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "or", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "|" + }, + { + "name": "plus", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "+" + }, + { + "name": "product", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "re_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.find_all_string_submatch_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.find_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.globs_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.template_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "rego.parse_module", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "rem", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "%" + }, + { + "name": "replace", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "round", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "set_diff", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "sort", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "sprintf", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "startswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "strings.replace_n", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "string" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "substring", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "sum", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.add_date", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.clock", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.date", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.now_ns", + "decl": { + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_duration_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_ns", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_rfc3339_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.weekday", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "to_number", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "type": "string" + }, + { + "type": "boolean" + }, + { + "type": "null" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "trace", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "trim", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_left", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_prefix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_right", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_space", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_suffix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "type_name", + "decl": { + "args": [ + { + "of": [ + { + "type": "any" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "union", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "units.parse_bytes", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "upper", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode_object", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "walk", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "static": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "function" + }, + "relation": true + }, + { + "name": "yaml.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "yaml.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + } + ] +} diff --git a/vendor/github.com/open-policy-agent/opa/capabilities/v0.19.1.json b/vendor/github.com/open-policy-agent/opa/capabilities/v0.19.1.json new file mode 100644 index 00000000..fa9ae4e4 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/capabilities/v0.19.1.json @@ -0,0 +1,2858 @@ +{ + "builtins": [ + { + "name": "abs", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "all", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "and", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "\u0026" + }, + { + "name": "any", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "array.concat", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "array.slice", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "assign", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": ":=" + }, + { + "name": "base64.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "bits.and", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.lsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.negate", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.or", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.rsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.xor", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "cast_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "cast_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "cast_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "null" + }, + "type": "function" + } + }, + { + "name": "cast_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "cast_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "cast_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "concat", + "decl": { + "args": [ + { + "type": "string" + }, + { + "of": [ + { + "of": { + "type": "string" + }, + "type": "set" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "count", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "crypto.md5", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha1", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha256", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "div", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "/" + }, + { + "name": "endswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "eq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "=" + }, + { + "name": "equal", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "==" + }, + { + "name": "format_int", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "glob.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "glob.quote_meta", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "gt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e" + }, + { + "name": "gte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e=" + }, + { + "name": "http.send", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "indexof", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "intersection", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode_verify", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign_raw", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_number", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "json.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "lower", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "lt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c" + }, + { + "name": "lte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c=" + }, + { + "name": "max", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "min", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "minus", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + "type": "function" + }, + "infix": "-" + }, + { + "name": "mul", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "*" + }, + { + "name": "neq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "!=" + }, + { + "name": "net.cidr_contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_contains_matches", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "static": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_expand", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_intersects", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_overlap", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "object.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.get", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.union", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "opa.runtime", + "decl": { + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "or", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "|" + }, + { + "name": "plus", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "+" + }, + { + "name": "product", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "re_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.find_all_string_submatch_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.find_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.globs_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.template_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "rego.parse_module", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "rem", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "%" + }, + { + "name": "replace", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "round", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "set_diff", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "sort", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "sprintf", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "startswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "strings.replace_n", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "string" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "substring", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "sum", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.add_date", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.clock", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.date", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.now_ns", + "decl": { + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_duration_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_ns", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_rfc3339_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.weekday", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "to_number", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "type": "string" + }, + { + "type": "boolean" + }, + { + "type": "null" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "trace", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "trim", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_left", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_prefix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_right", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_space", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_suffix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "type_name", + "decl": { + "args": [ + { + "of": [ + { + "type": "any" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "union", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "units.parse_bytes", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "upper", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode_object", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "walk", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "static": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "function" + }, + "relation": true + }, + { + "name": "yaml.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "yaml.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + } + ] +} diff --git a/vendor/github.com/open-policy-agent/opa/capabilities/v0.19.2.json b/vendor/github.com/open-policy-agent/opa/capabilities/v0.19.2.json new file mode 100644 index 00000000..fa9ae4e4 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/capabilities/v0.19.2.json @@ -0,0 +1,2858 @@ +{ + "builtins": [ + { + "name": "abs", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "all", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "and", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "\u0026" + }, + { + "name": "any", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "array.concat", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "array.slice", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "assign", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": ":=" + }, + { + "name": "base64.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "bits.and", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.lsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.negate", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.or", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.rsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.xor", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "cast_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "cast_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "cast_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "null" + }, + "type": "function" + } + }, + { + "name": "cast_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "cast_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "cast_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "concat", + "decl": { + "args": [ + { + "type": "string" + }, + { + "of": [ + { + "of": { + "type": "string" + }, + "type": "set" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "count", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "crypto.md5", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha1", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha256", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "div", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "/" + }, + { + "name": "endswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "eq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "=" + }, + { + "name": "equal", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "==" + }, + { + "name": "format_int", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "glob.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "glob.quote_meta", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "gt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e" + }, + { + "name": "gte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e=" + }, + { + "name": "http.send", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "indexof", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "intersection", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode_verify", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign_raw", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_number", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "json.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "lower", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "lt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c" + }, + { + "name": "lte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c=" + }, + { + "name": "max", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "min", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "minus", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + "type": "function" + }, + "infix": "-" + }, + { + "name": "mul", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "*" + }, + { + "name": "neq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "!=" + }, + { + "name": "net.cidr_contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_contains_matches", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "static": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_expand", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_intersects", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_overlap", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "object.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.get", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.union", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "opa.runtime", + "decl": { + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "or", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "|" + }, + { + "name": "plus", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "+" + }, + { + "name": "product", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "re_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.find_all_string_submatch_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.find_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.globs_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.template_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "rego.parse_module", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "rem", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "%" + }, + { + "name": "replace", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "round", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "set_diff", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "sort", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "sprintf", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "startswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "strings.replace_n", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "string" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "substring", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "sum", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.add_date", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.clock", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.date", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.now_ns", + "decl": { + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_duration_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_ns", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_rfc3339_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.weekday", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "to_number", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "type": "string" + }, + { + "type": "boolean" + }, + { + "type": "null" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "trace", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "trim", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_left", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_prefix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_right", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_space", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_suffix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "type_name", + "decl": { + "args": [ + { + "of": [ + { + "type": "any" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "union", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "units.parse_bytes", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "upper", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode_object", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "walk", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "static": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "function" + }, + "relation": true + }, + { + "name": "yaml.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "yaml.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + } + ] +} diff --git a/vendor/github.com/open-policy-agent/opa/capabilities/v0.20.0.json b/vendor/github.com/open-policy-agent/opa/capabilities/v0.20.0.json new file mode 100644 index 00000000..f2d29068 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/capabilities/v0.20.0.json @@ -0,0 +1,3064 @@ +{ + "builtins": [ + { + "name": "abs", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "all", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "and", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "\u0026" + }, + { + "name": "any", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "array.concat", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "array.slice", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "assign", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": ":=" + }, + { + "name": "base64.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "bits.and", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.lsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.negate", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.or", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.rsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.xor", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "cast_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "cast_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "cast_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "null" + }, + "type": "function" + } + }, + { + "name": "cast_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "cast_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "cast_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "concat", + "decl": { + "args": [ + { + "type": "string" + }, + { + "of": [ + { + "of": { + "type": "string" + }, + "type": "set" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "count", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "crypto.md5", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha1", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha256", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "div", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "/" + }, + { + "name": "endswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "eq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "=" + }, + { + "name": "equal", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "==" + }, + { + "name": "format_int", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "glob.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "glob.quote_meta", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "graph.reachable", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "gt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e" + }, + { + "name": "gte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e=" + }, + { + "name": "http.send", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "indexof", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "intersection", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode_verify", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign_raw", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_number", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "json.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "lower", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "lt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c" + }, + { + "name": "lte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c=" + }, + { + "name": "max", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "min", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "minus", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + "type": "function" + }, + "infix": "-" + }, + { + "name": "mul", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "*" + }, + { + "name": "neq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "!=" + }, + { + "name": "net.cidr_contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_contains_matches", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "static": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_expand", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_intersects", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_overlap", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "object.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.get", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.union", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "opa.runtime", + "decl": { + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "or", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "|" + }, + { + "name": "plus", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "+" + }, + { + "name": "product", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "re_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.find_all_string_submatch_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.find_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.globs_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.template_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "rego.parse_module", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "rem", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "%" + }, + { + "name": "replace", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "round", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "set_diff", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "sort", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "sprintf", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "startswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "strings.replace_n", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "string" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "substring", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "sum", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.add_date", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.clock", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.date", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.now_ns", + "decl": { + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_duration_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_ns", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_rfc3339_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.weekday", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "to_number", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "type": "string" + }, + { + "type": "boolean" + }, + { + "type": "null" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "trace", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "trim", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_left", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_prefix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_right", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_space", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_suffix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "type_name", + "decl": { + "args": [ + { + "of": [ + { + "type": "any" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "union", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "units.parse_bytes", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "upper", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode_object", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "uuid.rfc4122", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "walk", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "static": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "function" + }, + "relation": true + }, + { + "name": "yaml.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "yaml.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + } + ] +} diff --git a/vendor/github.com/open-policy-agent/opa/capabilities/v0.20.1.json b/vendor/github.com/open-policy-agent/opa/capabilities/v0.20.1.json new file mode 100644 index 00000000..f2d29068 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/capabilities/v0.20.1.json @@ -0,0 +1,3064 @@ +{ + "builtins": [ + { + "name": "abs", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "all", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "and", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "\u0026" + }, + { + "name": "any", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "array.concat", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "array.slice", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "assign", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": ":=" + }, + { + "name": "base64.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "bits.and", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.lsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.negate", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.or", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.rsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.xor", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "cast_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "cast_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "cast_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "null" + }, + "type": "function" + } + }, + { + "name": "cast_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "cast_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "cast_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "concat", + "decl": { + "args": [ + { + "type": "string" + }, + { + "of": [ + { + "of": { + "type": "string" + }, + "type": "set" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "count", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "crypto.md5", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha1", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha256", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "div", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "/" + }, + { + "name": "endswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "eq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "=" + }, + { + "name": "equal", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "==" + }, + { + "name": "format_int", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "glob.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "glob.quote_meta", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "graph.reachable", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "gt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e" + }, + { + "name": "gte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e=" + }, + { + "name": "http.send", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "indexof", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "intersection", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode_verify", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign_raw", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_number", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "json.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "lower", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "lt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c" + }, + { + "name": "lte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c=" + }, + { + "name": "max", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "min", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "minus", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + "type": "function" + }, + "infix": "-" + }, + { + "name": "mul", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "*" + }, + { + "name": "neq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "!=" + }, + { + "name": "net.cidr_contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_contains_matches", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "static": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_expand", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_intersects", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_overlap", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "object.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.get", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.union", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "opa.runtime", + "decl": { + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "or", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "|" + }, + { + "name": "plus", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "+" + }, + { + "name": "product", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "re_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.find_all_string_submatch_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.find_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.globs_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.template_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "rego.parse_module", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "rem", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "%" + }, + { + "name": "replace", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "round", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "set_diff", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "sort", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "sprintf", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "startswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "strings.replace_n", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "string" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "substring", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "sum", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.add_date", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.clock", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.date", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.now_ns", + "decl": { + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_duration_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_ns", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_rfc3339_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.weekday", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "to_number", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "type": "string" + }, + { + "type": "boolean" + }, + { + "type": "null" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "trace", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "trim", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_left", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_prefix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_right", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_space", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_suffix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "type_name", + "decl": { + "args": [ + { + "of": [ + { + "type": "any" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "union", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "units.parse_bytes", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "upper", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode_object", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "uuid.rfc4122", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "walk", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "static": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "function" + }, + "relation": true + }, + { + "name": "yaml.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "yaml.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + } + ] +} diff --git a/vendor/github.com/open-policy-agent/opa/capabilities/v0.20.2.json b/vendor/github.com/open-policy-agent/opa/capabilities/v0.20.2.json new file mode 100644 index 00000000..f2d29068 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/capabilities/v0.20.2.json @@ -0,0 +1,3064 @@ +{ + "builtins": [ + { + "name": "abs", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "all", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "and", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "\u0026" + }, + { + "name": "any", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "array.concat", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "array.slice", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "assign", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": ":=" + }, + { + "name": "base64.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "bits.and", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.lsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.negate", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.or", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.rsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.xor", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "cast_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "cast_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "cast_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "null" + }, + "type": "function" + } + }, + { + "name": "cast_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "cast_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "cast_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "concat", + "decl": { + "args": [ + { + "type": "string" + }, + { + "of": [ + { + "of": { + "type": "string" + }, + "type": "set" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "count", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "crypto.md5", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha1", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha256", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "div", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "/" + }, + { + "name": "endswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "eq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "=" + }, + { + "name": "equal", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "==" + }, + { + "name": "format_int", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "glob.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "glob.quote_meta", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "graph.reachable", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "gt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e" + }, + { + "name": "gte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e=" + }, + { + "name": "http.send", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "indexof", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "intersection", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode_verify", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign_raw", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_number", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "json.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "lower", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "lt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c" + }, + { + "name": "lte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c=" + }, + { + "name": "max", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "min", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "minus", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + "type": "function" + }, + "infix": "-" + }, + { + "name": "mul", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "*" + }, + { + "name": "neq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "!=" + }, + { + "name": "net.cidr_contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_contains_matches", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "static": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_expand", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_intersects", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_overlap", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "object.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.get", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.union", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "opa.runtime", + "decl": { + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "or", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "|" + }, + { + "name": "plus", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "+" + }, + { + "name": "product", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "re_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.find_all_string_submatch_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.find_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.globs_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.template_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "rego.parse_module", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "rem", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "%" + }, + { + "name": "replace", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "round", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "set_diff", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "sort", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "sprintf", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "startswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "strings.replace_n", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "string" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "substring", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "sum", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.add_date", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.clock", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.date", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.now_ns", + "decl": { + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_duration_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_ns", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_rfc3339_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.weekday", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "to_number", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "type": "string" + }, + { + "type": "boolean" + }, + { + "type": "null" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "trace", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "trim", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_left", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_prefix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_right", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_space", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_suffix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "type_name", + "decl": { + "args": [ + { + "of": [ + { + "type": "any" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "union", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "units.parse_bytes", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "upper", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode_object", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "uuid.rfc4122", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "walk", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "static": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "function" + }, + "relation": true + }, + { + "name": "yaml.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "yaml.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + } + ] +} diff --git a/vendor/github.com/open-policy-agent/opa/capabilities/v0.20.3.json b/vendor/github.com/open-policy-agent/opa/capabilities/v0.20.3.json new file mode 100644 index 00000000..f2d29068 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/capabilities/v0.20.3.json @@ -0,0 +1,3064 @@ +{ + "builtins": [ + { + "name": "abs", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "all", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "and", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "\u0026" + }, + { + "name": "any", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "array.concat", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "array.slice", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "assign", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": ":=" + }, + { + "name": "base64.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "bits.and", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.lsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.negate", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.or", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.rsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.xor", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "cast_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "cast_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "cast_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "null" + }, + "type": "function" + } + }, + { + "name": "cast_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "cast_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "cast_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "concat", + "decl": { + "args": [ + { + "type": "string" + }, + { + "of": [ + { + "of": { + "type": "string" + }, + "type": "set" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "count", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "crypto.md5", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha1", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha256", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "div", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "/" + }, + { + "name": "endswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "eq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "=" + }, + { + "name": "equal", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "==" + }, + { + "name": "format_int", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "glob.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "glob.quote_meta", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "graph.reachable", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "gt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e" + }, + { + "name": "gte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e=" + }, + { + "name": "http.send", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "indexof", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "intersection", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode_verify", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign_raw", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_number", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "json.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "lower", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "lt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c" + }, + { + "name": "lte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c=" + }, + { + "name": "max", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "min", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "minus", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + "type": "function" + }, + "infix": "-" + }, + { + "name": "mul", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "*" + }, + { + "name": "neq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "!=" + }, + { + "name": "net.cidr_contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_contains_matches", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "static": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_expand", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_intersects", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_overlap", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "object.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.get", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.union", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "opa.runtime", + "decl": { + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "or", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "|" + }, + { + "name": "plus", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "+" + }, + { + "name": "product", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "re_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.find_all_string_submatch_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.find_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.globs_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.template_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "rego.parse_module", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "rem", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "%" + }, + { + "name": "replace", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "round", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "set_diff", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "sort", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "sprintf", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "startswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "strings.replace_n", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "string" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "substring", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "sum", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.add_date", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.clock", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.date", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.now_ns", + "decl": { + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_duration_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_ns", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_rfc3339_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.weekday", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "to_number", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "type": "string" + }, + { + "type": "boolean" + }, + { + "type": "null" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "trace", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "trim", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_left", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_prefix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_right", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_space", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_suffix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "type_name", + "decl": { + "args": [ + { + "of": [ + { + "type": "any" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "union", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "units.parse_bytes", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "upper", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode_object", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "uuid.rfc4122", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "walk", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "static": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "function" + }, + "relation": true + }, + { + "name": "yaml.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "yaml.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + } + ] +} diff --git a/vendor/github.com/open-policy-agent/opa/capabilities/v0.20.4.json b/vendor/github.com/open-policy-agent/opa/capabilities/v0.20.4.json new file mode 100644 index 00000000..f2d29068 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/capabilities/v0.20.4.json @@ -0,0 +1,3064 @@ +{ + "builtins": [ + { + "name": "abs", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "all", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "and", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "\u0026" + }, + { + "name": "any", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "array.concat", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "array.slice", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "assign", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": ":=" + }, + { + "name": "base64.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "bits.and", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.lsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.negate", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.or", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.rsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.xor", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "cast_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "cast_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "cast_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "null" + }, + "type": "function" + } + }, + { + "name": "cast_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "cast_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "cast_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "concat", + "decl": { + "args": [ + { + "type": "string" + }, + { + "of": [ + { + "of": { + "type": "string" + }, + "type": "set" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "count", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "crypto.md5", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha1", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha256", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "div", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "/" + }, + { + "name": "endswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "eq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "=" + }, + { + "name": "equal", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "==" + }, + { + "name": "format_int", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "glob.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "glob.quote_meta", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "graph.reachable", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "gt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e" + }, + { + "name": "gte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e=" + }, + { + "name": "http.send", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "indexof", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "intersection", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode_verify", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign_raw", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_number", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "json.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "lower", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "lt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c" + }, + { + "name": "lte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c=" + }, + { + "name": "max", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "min", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "minus", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + "type": "function" + }, + "infix": "-" + }, + { + "name": "mul", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "*" + }, + { + "name": "neq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "!=" + }, + { + "name": "net.cidr_contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_contains_matches", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "static": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_expand", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_intersects", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_overlap", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "object.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.get", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.union", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "opa.runtime", + "decl": { + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "or", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "|" + }, + { + "name": "plus", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "+" + }, + { + "name": "product", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "re_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.find_all_string_submatch_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.find_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.globs_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.template_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "rego.parse_module", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "rem", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "%" + }, + { + "name": "replace", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "round", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "set_diff", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "sort", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "sprintf", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "startswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "strings.replace_n", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "string" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "substring", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "sum", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.add_date", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.clock", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.date", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.now_ns", + "decl": { + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_duration_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_ns", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_rfc3339_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.weekday", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "to_number", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "type": "string" + }, + { + "type": "boolean" + }, + { + "type": "null" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "trace", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "trim", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_left", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_prefix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_right", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_space", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_suffix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "type_name", + "decl": { + "args": [ + { + "of": [ + { + "type": "any" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "union", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "units.parse_bytes", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "upper", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode_object", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "uuid.rfc4122", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "walk", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "static": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "function" + }, + "relation": true + }, + { + "name": "yaml.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "yaml.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + } + ] +} diff --git a/vendor/github.com/open-policy-agent/opa/capabilities/v0.20.5.json b/vendor/github.com/open-policy-agent/opa/capabilities/v0.20.5.json new file mode 100644 index 00000000..f2d29068 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/capabilities/v0.20.5.json @@ -0,0 +1,3064 @@ +{ + "builtins": [ + { + "name": "abs", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "all", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "and", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "\u0026" + }, + { + "name": "any", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "array.concat", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "array.slice", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "assign", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": ":=" + }, + { + "name": "base64.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "bits.and", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.lsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.negate", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.or", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.rsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.xor", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "cast_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "cast_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "cast_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "null" + }, + "type": "function" + } + }, + { + "name": "cast_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "cast_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "cast_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "concat", + "decl": { + "args": [ + { + "type": "string" + }, + { + "of": [ + { + "of": { + "type": "string" + }, + "type": "set" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "count", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "crypto.md5", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha1", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha256", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "div", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "/" + }, + { + "name": "endswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "eq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "=" + }, + { + "name": "equal", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "==" + }, + { + "name": "format_int", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "glob.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "glob.quote_meta", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "graph.reachable", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "gt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e" + }, + { + "name": "gte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e=" + }, + { + "name": "http.send", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "indexof", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "intersection", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode_verify", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign_raw", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_number", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "json.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "lower", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "lt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c" + }, + { + "name": "lte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c=" + }, + { + "name": "max", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "min", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "minus", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + "type": "function" + }, + "infix": "-" + }, + { + "name": "mul", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "*" + }, + { + "name": "neq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "!=" + }, + { + "name": "net.cidr_contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_contains_matches", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "static": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_expand", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_intersects", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_overlap", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "object.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.get", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.union", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "opa.runtime", + "decl": { + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "or", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "|" + }, + { + "name": "plus", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "+" + }, + { + "name": "product", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "re_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.find_all_string_submatch_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.find_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.globs_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.template_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "rego.parse_module", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "rem", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "%" + }, + { + "name": "replace", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "round", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "set_diff", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "sort", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "sprintf", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "startswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "strings.replace_n", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "string" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "substring", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "sum", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.add_date", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.clock", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.date", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.now_ns", + "decl": { + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_duration_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_ns", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_rfc3339_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.weekday", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "to_number", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "type": "string" + }, + { + "type": "boolean" + }, + { + "type": "null" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "trace", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "trim", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_left", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_prefix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_right", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_space", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_suffix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "type_name", + "decl": { + "args": [ + { + "of": [ + { + "type": "any" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "union", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "units.parse_bytes", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "upper", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode_object", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "uuid.rfc4122", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "walk", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "static": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "function" + }, + "relation": true + }, + { + "name": "yaml.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "yaml.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + } + ] +} diff --git a/vendor/github.com/open-policy-agent/opa/capabilities/v0.21.0.json b/vendor/github.com/open-policy-agent/opa/capabilities/v0.21.0.json new file mode 100644 index 00000000..18d1af3d --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/capabilities/v0.21.0.json @@ -0,0 +1,3086 @@ +{ + "builtins": [ + { + "name": "abs", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "all", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "and", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "\u0026" + }, + { + "name": "any", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "array.concat", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "array.slice", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "assign", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": ":=" + }, + { + "name": "base64.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "bits.and", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.lsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.negate", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.or", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.rsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.xor", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "cast_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "cast_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "cast_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "null" + }, + "type": "function" + } + }, + { + "name": "cast_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "cast_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "cast_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "concat", + "decl": { + "args": [ + { + "type": "string" + }, + { + "of": [ + { + "of": { + "type": "string" + }, + "type": "set" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "count", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "crypto.md5", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha1", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha256", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificate_request", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "div", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "/" + }, + { + "name": "endswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "eq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "=" + }, + { + "name": "equal", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "==" + }, + { + "name": "format_int", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "glob.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "glob.quote_meta", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "graph.reachable", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "gt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e" + }, + { + "name": "gte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e=" + }, + { + "name": "http.send", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "indexof", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "intersection", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode_verify", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign_raw", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_number", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "json.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "lower", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "lt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c" + }, + { + "name": "lte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c=" + }, + { + "name": "max", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "min", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "minus", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + "type": "function" + }, + "infix": "-" + }, + { + "name": "mul", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "*" + }, + { + "name": "neq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "!=" + }, + { + "name": "net.cidr_contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_contains_matches", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "static": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_expand", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_intersects", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_overlap", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "object.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.get", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.union", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "opa.runtime", + "decl": { + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "or", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "|" + }, + { + "name": "plus", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "+" + }, + { + "name": "product", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "re_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.find_all_string_submatch_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.find_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.globs_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.template_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "rego.parse_module", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "rem", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "%" + }, + { + "name": "replace", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "round", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "set_diff", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "sort", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "sprintf", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "startswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "strings.replace_n", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "string" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "substring", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "sum", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.add_date", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.clock", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.date", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.now_ns", + "decl": { + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_duration_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_ns", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_rfc3339_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.weekday", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "to_number", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "type": "string" + }, + { + "type": "boolean" + }, + { + "type": "null" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "trace", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "trim", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_left", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_prefix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_right", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_space", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_suffix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "type_name", + "decl": { + "args": [ + { + "of": [ + { + "type": "any" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "union", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "units.parse_bytes", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "upper", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode_object", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "uuid.rfc4122", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "walk", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "static": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "function" + }, + "relation": true + }, + { + "name": "yaml.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "yaml.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + } + ] +} diff --git a/vendor/github.com/open-policy-agent/opa/capabilities/v0.21.1.json b/vendor/github.com/open-policy-agent/opa/capabilities/v0.21.1.json new file mode 100644 index 00000000..18d1af3d --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/capabilities/v0.21.1.json @@ -0,0 +1,3086 @@ +{ + "builtins": [ + { + "name": "abs", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "all", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "and", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "\u0026" + }, + { + "name": "any", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "array.concat", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "array.slice", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "assign", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": ":=" + }, + { + "name": "base64.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "bits.and", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.lsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.negate", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.or", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.rsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.xor", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "cast_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "cast_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "cast_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "null" + }, + "type": "function" + } + }, + { + "name": "cast_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "cast_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "cast_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "concat", + "decl": { + "args": [ + { + "type": "string" + }, + { + "of": [ + { + "of": { + "type": "string" + }, + "type": "set" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "count", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "crypto.md5", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha1", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha256", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificate_request", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "div", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "/" + }, + { + "name": "endswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "eq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "=" + }, + { + "name": "equal", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "==" + }, + { + "name": "format_int", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "glob.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "glob.quote_meta", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "graph.reachable", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "gt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e" + }, + { + "name": "gte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e=" + }, + { + "name": "http.send", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "indexof", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "intersection", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode_verify", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign_raw", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_number", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "json.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "lower", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "lt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c" + }, + { + "name": "lte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c=" + }, + { + "name": "max", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "min", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "minus", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + "type": "function" + }, + "infix": "-" + }, + { + "name": "mul", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "*" + }, + { + "name": "neq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "!=" + }, + { + "name": "net.cidr_contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_contains_matches", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "static": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_expand", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_intersects", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_overlap", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "object.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.get", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.union", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "opa.runtime", + "decl": { + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "or", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "|" + }, + { + "name": "plus", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "+" + }, + { + "name": "product", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "re_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.find_all_string_submatch_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.find_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.globs_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.template_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "rego.parse_module", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "rem", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "%" + }, + { + "name": "replace", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "round", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "set_diff", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "sort", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "sprintf", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "startswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "strings.replace_n", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "string" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "substring", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "sum", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.add_date", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.clock", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.date", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.now_ns", + "decl": { + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_duration_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_ns", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_rfc3339_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.weekday", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "to_number", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "type": "string" + }, + { + "type": "boolean" + }, + { + "type": "null" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "trace", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "trim", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_left", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_prefix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_right", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_space", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_suffix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "type_name", + "decl": { + "args": [ + { + "of": [ + { + "type": "any" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "union", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "units.parse_bytes", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "upper", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode_object", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "uuid.rfc4122", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "walk", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "static": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "function" + }, + "relation": true + }, + { + "name": "yaml.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "yaml.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + } + ] +} diff --git a/vendor/github.com/open-policy-agent/opa/capabilities/v0.22.0.json b/vendor/github.com/open-policy-agent/opa/capabilities/v0.22.0.json new file mode 100644 index 00000000..c3f06714 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/capabilities/v0.22.0.json @@ -0,0 +1,3137 @@ +{ + "builtins": [ + { + "name": "abs", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "all", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "and", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "\u0026" + }, + { + "name": "any", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "array.concat", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "array.slice", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "assign", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": ":=" + }, + { + "name": "base64.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "bits.and", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.lsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.negate", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.or", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.rsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.xor", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "cast_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "cast_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "cast_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "null" + }, + "type": "function" + } + }, + { + "name": "cast_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "cast_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "cast_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "concat", + "decl": { + "args": [ + { + "type": "string" + }, + { + "of": [ + { + "of": { + "type": "string" + }, + "type": "set" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "count", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "crypto.md5", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha1", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha256", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificate_request", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "div", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "/" + }, + { + "name": "endswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "eq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "=" + }, + { + "name": "equal", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "==" + }, + { + "name": "format_int", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "glob.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "glob.quote_meta", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "graph.reachable", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "gt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e" + }, + { + "name": "gte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e=" + }, + { + "name": "http.send", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "indexof", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "intersection", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode_verify", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign_raw", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_number", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "json.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "lower", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "lt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c" + }, + { + "name": "lte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c=" + }, + { + "name": "max", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "min", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "minus", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + "type": "function" + }, + "infix": "-" + }, + { + "name": "mul", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "*" + }, + { + "name": "neq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "!=" + }, + { + "name": "net.cidr_contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_contains_matches", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "static": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_expand", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_intersects", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_overlap", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "numbers.range", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "object.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.get", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.union", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "opa.runtime", + "decl": { + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "or", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "|" + }, + { + "name": "plus", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "+" + }, + { + "name": "product", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "re_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.find_all_string_submatch_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.find_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.globs_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.template_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "rego.parse_module", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "rem", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "%" + }, + { + "name": "replace", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "round", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.compare", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.is_valid", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "set_diff", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "sort", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "sprintf", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "startswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "strings.replace_n", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "string" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "substring", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "sum", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.add_date", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.clock", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.date", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.now_ns", + "decl": { + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_duration_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_ns", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_rfc3339_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.weekday", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "to_number", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "type": "string" + }, + { + "type": "boolean" + }, + { + "type": "null" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "trace", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "trim", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_left", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_prefix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_right", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_space", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_suffix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "type_name", + "decl": { + "args": [ + { + "of": [ + { + "type": "any" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "union", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "units.parse_bytes", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "upper", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode_object", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "uuid.rfc4122", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "walk", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "static": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "function" + }, + "relation": true + }, + { + "name": "yaml.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "yaml.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + } + ] +} diff --git a/vendor/github.com/open-policy-agent/opa/capabilities/v0.23.0.json b/vendor/github.com/open-policy-agent/opa/capabilities/v0.23.0.json new file mode 100644 index 00000000..8eeceda5 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/capabilities/v0.23.0.json @@ -0,0 +1,3168 @@ +{ + "builtins": [ + { + "name": "abs", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "all", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "and", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "\u0026" + }, + { + "name": "any", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "array.concat", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "array.slice", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "assign", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": ":=" + }, + { + "name": "base64.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "bits.and", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.lsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.negate", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.or", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.rsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.xor", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "cast_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "cast_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "cast_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "null" + }, + "type": "function" + } + }, + { + "name": "cast_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "cast_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "cast_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "concat", + "decl": { + "args": [ + { + "type": "string" + }, + { + "of": [ + { + "of": { + "type": "string" + }, + "type": "set" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "count", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "crypto.md5", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha1", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha256", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificate_request", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "div", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "/" + }, + { + "name": "endswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "eq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "=" + }, + { + "name": "equal", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "==" + }, + { + "name": "format_int", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "glob.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "glob.quote_meta", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "graph.reachable", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "gt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e" + }, + { + "name": "gte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e=" + }, + { + "name": "http.send", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "indexof", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "intersection", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode_verify", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign_raw", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_number", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "json.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "lower", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "lt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c" + }, + { + "name": "lte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c=" + }, + { + "name": "max", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "min", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "minus", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + "type": "function" + }, + "infix": "-" + }, + { + "name": "mul", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "*" + }, + { + "name": "neq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "!=" + }, + { + "name": "net.cidr_contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_contains_matches", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "static": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_expand", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_intersects", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_overlap", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "numbers.range", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "object.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.get", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.union", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "opa.runtime", + "decl": { + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "or", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "|" + }, + { + "name": "plus", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "+" + }, + { + "name": "product", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "re_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.find_all_string_submatch_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.find_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.globs_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.template_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "rego.parse_module", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "rem", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "%" + }, + { + "name": "replace", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "round", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.compare", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.is_valid", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "set_diff", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "sort", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "sprintf", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "startswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "strings.replace_n", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "string" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "substring", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "sum", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.add_date", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.clock", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.date", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.now_ns", + "decl": { + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_duration_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_ns", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_rfc3339_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.weekday", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "to_number", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "type": "string" + }, + { + "type": "boolean" + }, + { + "type": "null" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "trace", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "trim", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_left", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_prefix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_right", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_space", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_suffix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "type_name", + "decl": { + "args": [ + { + "of": [ + { + "type": "any" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "union", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "units.parse_bytes", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "upper", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode_object", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "uuid.rfc4122", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "walk", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "static": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "function" + }, + "relation": true + }, + { + "name": "yaml.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "yaml.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + } + ] +} diff --git a/vendor/github.com/open-policy-agent/opa/capabilities/v0.23.1.json b/vendor/github.com/open-policy-agent/opa/capabilities/v0.23.1.json new file mode 100644 index 00000000..8eeceda5 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/capabilities/v0.23.1.json @@ -0,0 +1,3168 @@ +{ + "builtins": [ + { + "name": "abs", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "all", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "and", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "\u0026" + }, + { + "name": "any", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "array.concat", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "array.slice", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "assign", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": ":=" + }, + { + "name": "base64.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "bits.and", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.lsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.negate", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.or", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.rsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.xor", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "cast_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "cast_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "cast_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "null" + }, + "type": "function" + } + }, + { + "name": "cast_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "cast_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "cast_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "concat", + "decl": { + "args": [ + { + "type": "string" + }, + { + "of": [ + { + "of": { + "type": "string" + }, + "type": "set" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "count", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "crypto.md5", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha1", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha256", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificate_request", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "div", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "/" + }, + { + "name": "endswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "eq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "=" + }, + { + "name": "equal", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "==" + }, + { + "name": "format_int", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "glob.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "glob.quote_meta", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "graph.reachable", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "gt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e" + }, + { + "name": "gte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e=" + }, + { + "name": "http.send", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "indexof", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "intersection", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode_verify", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign_raw", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_number", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "json.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "lower", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "lt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c" + }, + { + "name": "lte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c=" + }, + { + "name": "max", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "min", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "minus", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + "type": "function" + }, + "infix": "-" + }, + { + "name": "mul", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "*" + }, + { + "name": "neq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "!=" + }, + { + "name": "net.cidr_contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_contains_matches", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "static": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_expand", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_intersects", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_overlap", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "numbers.range", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "object.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.get", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.union", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "opa.runtime", + "decl": { + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "or", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "|" + }, + { + "name": "plus", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "+" + }, + { + "name": "product", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "re_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.find_all_string_submatch_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.find_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.globs_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.template_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "rego.parse_module", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "rem", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "%" + }, + { + "name": "replace", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "round", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.compare", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.is_valid", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "set_diff", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "sort", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "sprintf", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "startswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "strings.replace_n", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "string" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "substring", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "sum", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.add_date", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.clock", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.date", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.now_ns", + "decl": { + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_duration_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_ns", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_rfc3339_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.weekday", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "to_number", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "type": "string" + }, + { + "type": "boolean" + }, + { + "type": "null" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "trace", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "trim", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_left", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_prefix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_right", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_space", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_suffix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "type_name", + "decl": { + "args": [ + { + "of": [ + { + "type": "any" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "union", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "units.parse_bytes", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "upper", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode_object", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "uuid.rfc4122", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "walk", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "static": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "function" + }, + "relation": true + }, + { + "name": "yaml.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "yaml.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + } + ] +} diff --git a/vendor/github.com/open-policy-agent/opa/capabilities/v0.23.2.json b/vendor/github.com/open-policy-agent/opa/capabilities/v0.23.2.json new file mode 100644 index 00000000..8eeceda5 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/capabilities/v0.23.2.json @@ -0,0 +1,3168 @@ +{ + "builtins": [ + { + "name": "abs", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "all", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "and", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "\u0026" + }, + { + "name": "any", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "array.concat", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "array.slice", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "assign", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": ":=" + }, + { + "name": "base64.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "bits.and", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.lsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.negate", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.or", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.rsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.xor", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "cast_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "cast_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "cast_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "null" + }, + "type": "function" + } + }, + { + "name": "cast_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "cast_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "cast_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "concat", + "decl": { + "args": [ + { + "type": "string" + }, + { + "of": [ + { + "of": { + "type": "string" + }, + "type": "set" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "count", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "crypto.md5", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha1", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha256", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificate_request", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "div", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "/" + }, + { + "name": "endswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "eq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "=" + }, + { + "name": "equal", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "==" + }, + { + "name": "format_int", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "glob.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "glob.quote_meta", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "graph.reachable", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "gt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e" + }, + { + "name": "gte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e=" + }, + { + "name": "http.send", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "indexof", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "intersection", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode_verify", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign_raw", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_number", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "json.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "lower", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "lt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c" + }, + { + "name": "lte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c=" + }, + { + "name": "max", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "min", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "minus", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + "type": "function" + }, + "infix": "-" + }, + { + "name": "mul", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "*" + }, + { + "name": "neq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "!=" + }, + { + "name": "net.cidr_contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_contains_matches", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "static": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_expand", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_intersects", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_overlap", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "numbers.range", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "object.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.get", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.union", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "opa.runtime", + "decl": { + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "or", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "|" + }, + { + "name": "plus", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "+" + }, + { + "name": "product", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "re_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.find_all_string_submatch_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.find_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.globs_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.template_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "rego.parse_module", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "rem", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "%" + }, + { + "name": "replace", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "round", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.compare", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.is_valid", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "set_diff", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "sort", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "sprintf", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "startswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "strings.replace_n", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "string" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "substring", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "sum", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.add_date", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.clock", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.date", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.now_ns", + "decl": { + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_duration_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_ns", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_rfc3339_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.weekday", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "to_number", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "type": "string" + }, + { + "type": "boolean" + }, + { + "type": "null" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "trace", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "trim", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_left", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_prefix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_right", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_space", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_suffix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "type_name", + "decl": { + "args": [ + { + "of": [ + { + "type": "any" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "union", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "units.parse_bytes", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "upper", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode_object", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "uuid.rfc4122", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "walk", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "static": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "function" + }, + "relation": true + }, + { + "name": "yaml.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "yaml.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + } + ] +} diff --git a/vendor/github.com/open-policy-agent/opa/capabilities/v0.24.0.json b/vendor/github.com/open-policy-agent/opa/capabilities/v0.24.0.json new file mode 100644 index 00000000..ccca1210 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/capabilities/v0.24.0.json @@ -0,0 +1,3243 @@ +{ + "builtins": [ + { + "name": "abs", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "all", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "and", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "\u0026" + }, + { + "name": "any", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "array.concat", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "array.slice", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "assign", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": ":=" + }, + { + "name": "base64.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "base64url.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "bits.and", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.lsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.negate", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.or", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.rsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.xor", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "cast_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "cast_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "cast_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "null" + }, + "type": "function" + } + }, + { + "name": "cast_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "cast_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "cast_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "concat", + "decl": { + "args": [ + { + "type": "string" + }, + { + "of": [ + { + "of": { + "type": "string" + }, + "type": "set" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "count", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "crypto.md5", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha1", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha256", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificate_request", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "div", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "/" + }, + { + "name": "endswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "eq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "=" + }, + { + "name": "equal", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "==" + }, + { + "name": "format_int", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "glob.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "glob.quote_meta", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "graph.reachable", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "gt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e" + }, + { + "name": "gte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e=" + }, + { + "name": "http.send", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "indexof", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "intersection", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode_verify", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign_raw", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_number", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "json.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "lower", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "lt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c" + }, + { + "name": "lte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c=" + }, + { + "name": "max", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "min", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "minus", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + "type": "function" + }, + "infix": "-" + }, + { + "name": "mul", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "*" + }, + { + "name": "neq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "!=" + }, + { + "name": "net.cidr_contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_contains_matches", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "static": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_expand", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_intersects", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_merge", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_overlap", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "numbers.range", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "object.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.get", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.union", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "opa.runtime", + "decl": { + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "or", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "|" + }, + { + "name": "plus", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "+" + }, + { + "name": "product", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "re_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.find_all_string_submatch_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.find_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.globs_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.template_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "rego.parse_module", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "rem", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "%" + }, + { + "name": "replace", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "round", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.compare", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.is_valid", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "set_diff", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "sort", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "sprintf", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "startswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "strings.replace_n", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "string" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "substring", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "sum", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.add_date", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.clock", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.date", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.now_ns", + "decl": { + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_duration_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_ns", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_rfc3339_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.weekday", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "to_number", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "type": "string" + }, + { + "type": "boolean" + }, + { + "type": "null" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "trace", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "trim", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_left", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_prefix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_right", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_space", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_suffix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "type_name", + "decl": { + "args": [ + { + "of": [ + { + "type": "any" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "union", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "units.parse_bytes", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "upper", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode_object", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "dynamic": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode_object", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "uuid.rfc4122", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "walk", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "static": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "function" + }, + "relation": true + }, + { + "name": "yaml.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "yaml.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + } + ] +} diff --git a/vendor/github.com/open-policy-agent/opa/capabilities/v0.25.0-rc1.json b/vendor/github.com/open-policy-agent/opa/capabilities/v0.25.0-rc1.json new file mode 100644 index 00000000..9dc7c080 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/capabilities/v0.25.0-rc1.json @@ -0,0 +1,3271 @@ +{ + "builtins": [ + { + "name": "abs", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "all", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "and", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "\u0026" + }, + { + "name": "any", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "array.concat", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "array.slice", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "assign", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": ":=" + }, + { + "name": "base64.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "base64url.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "bits.and", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.lsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.negate", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.or", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.rsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.xor", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "cast_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "cast_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "cast_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "null" + }, + "type": "function" + } + }, + { + "name": "cast_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "cast_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "cast_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "concat", + "decl": { + "args": [ + { + "type": "string" + }, + { + "of": [ + { + "of": { + "type": "string" + }, + "type": "set" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "count", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "crypto.md5", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha1", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha256", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificate_request", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "div", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "/" + }, + { + "name": "endswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "eq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "=" + }, + { + "name": "equal", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "==" + }, + { + "name": "format_int", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "glob.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "glob.quote_meta", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "graph.reachable", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "gt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e" + }, + { + "name": "gte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e=" + }, + { + "name": "http.send", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "indexof", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "intersection", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode_verify", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign_raw", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_number", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "json.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "lower", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "lt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c" + }, + { + "name": "lte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c=" + }, + { + "name": "max", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "min", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "minus", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + "type": "function" + }, + "infix": "-" + }, + { + "name": "mul", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "*" + }, + { + "name": "neq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "!=" + }, + { + "name": "net.cidr_contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_contains_matches", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "static": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_expand", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_intersects", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_merge", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_overlap", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "numbers.range", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "object.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.get", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.union", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "opa.runtime", + "decl": { + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "or", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "|" + }, + { + "name": "plus", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "+" + }, + { + "name": "product", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "re_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.find_all_string_submatch_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.find_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.globs_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.template_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "rego.parse_module", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "rem", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "%" + }, + { + "name": "replace", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "round", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.compare", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.is_valid", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "set_diff", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "sort", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "sprintf", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "startswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "strings.replace_n", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "string" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "substring", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "sum", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.add_date", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.clock", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.date", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.now_ns", + "decl": { + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_duration_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_ns", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_rfc3339_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.weekday", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "to_number", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "type": "string" + }, + { + "type": "boolean" + }, + { + "type": "null" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "trace", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "trim", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_left", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_prefix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_right", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_space", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_suffix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "type_name", + "decl": { + "args": [ + { + "of": [ + { + "type": "any" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "union", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "units.parse_bytes", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "upper", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode_object", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "dynamic": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode_object", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "uuid.rfc4122", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "walk", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "static": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "function" + }, + "relation": true + }, + { + "name": "yaml.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "yaml.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "yaml.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + } + ] +} diff --git a/vendor/github.com/open-policy-agent/opa/capabilities/v0.25.0-rc2.json b/vendor/github.com/open-policy-agent/opa/capabilities/v0.25.0-rc2.json new file mode 100644 index 00000000..b5570417 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/capabilities/v0.25.0-rc2.json @@ -0,0 +1,3313 @@ +{ + "builtins": [ + { + "name": "abs", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "all", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "and", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "\u0026" + }, + { + "name": "any", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "array.concat", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "array.slice", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "assign", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": ":=" + }, + { + "name": "base64.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "base64url.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode_no_pad", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "bits.and", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.lsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.negate", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.or", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.rsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.xor", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "cast_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "cast_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "cast_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "null" + }, + "type": "function" + } + }, + { + "name": "cast_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "cast_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "cast_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "concat", + "decl": { + "args": [ + { + "type": "string" + }, + { + "of": [ + { + "of": { + "type": "string" + }, + "type": "set" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "count", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "crypto.md5", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha1", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha256", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificate_request", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "div", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "/" + }, + { + "name": "endswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "eq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "=" + }, + { + "name": "equal", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "==" + }, + { + "name": "format_int", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "glob.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "glob.quote_meta", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "graph.reachable", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "gt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e" + }, + { + "name": "gte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e=" + }, + { + "name": "hex.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "hex.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "http.send", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "indexof", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "intersection", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode_verify", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign_raw", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_number", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "json.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "lower", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "lt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c" + }, + { + "name": "lte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c=" + }, + { + "name": "max", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "min", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "minus", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + "type": "function" + }, + "infix": "-" + }, + { + "name": "mul", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "*" + }, + { + "name": "neq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "!=" + }, + { + "name": "net.cidr_contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_contains_matches", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "static": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_expand", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_intersects", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_merge", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_overlap", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "numbers.range", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "object.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.get", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.union", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "opa.runtime", + "decl": { + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "or", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "|" + }, + { + "name": "plus", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "+" + }, + { + "name": "product", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "re_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.find_all_string_submatch_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.find_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.globs_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.template_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "rego.parse_module", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "rem", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "%" + }, + { + "name": "replace", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "round", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.compare", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.is_valid", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "set_diff", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "sort", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "sprintf", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "startswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "strings.replace_n", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "string" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "substring", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "sum", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.add_date", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.clock", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.date", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.now_ns", + "decl": { + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_duration_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_ns", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_rfc3339_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.weekday", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "to_number", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "type": "string" + }, + { + "type": "boolean" + }, + { + "type": "null" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "trace", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "trim", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_left", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_prefix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_right", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_space", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_suffix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "type_name", + "decl": { + "args": [ + { + "of": [ + { + "type": "any" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "union", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "units.parse_bytes", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "upper", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode_object", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "dynamic": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode_object", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "uuid.rfc4122", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "walk", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "static": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "function" + }, + "relation": true + }, + { + "name": "yaml.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "yaml.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "yaml.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + } + ] +} diff --git a/vendor/github.com/open-policy-agent/opa/capabilities/v0.25.0-rc3.json b/vendor/github.com/open-policy-agent/opa/capabilities/v0.25.0-rc3.json new file mode 100644 index 00000000..b5570417 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/capabilities/v0.25.0-rc3.json @@ -0,0 +1,3313 @@ +{ + "builtins": [ + { + "name": "abs", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "all", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "and", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "\u0026" + }, + { + "name": "any", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "array.concat", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "array.slice", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "assign", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": ":=" + }, + { + "name": "base64.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "base64url.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode_no_pad", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "bits.and", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.lsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.negate", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.or", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.rsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.xor", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "cast_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "cast_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "cast_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "null" + }, + "type": "function" + } + }, + { + "name": "cast_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "cast_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "cast_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "concat", + "decl": { + "args": [ + { + "type": "string" + }, + { + "of": [ + { + "of": { + "type": "string" + }, + "type": "set" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "count", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "crypto.md5", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha1", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha256", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificate_request", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "div", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "/" + }, + { + "name": "endswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "eq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "=" + }, + { + "name": "equal", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "==" + }, + { + "name": "format_int", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "glob.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "glob.quote_meta", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "graph.reachable", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "gt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e" + }, + { + "name": "gte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e=" + }, + { + "name": "hex.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "hex.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "http.send", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "indexof", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "intersection", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode_verify", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign_raw", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_number", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "json.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "lower", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "lt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c" + }, + { + "name": "lte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c=" + }, + { + "name": "max", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "min", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "minus", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + "type": "function" + }, + "infix": "-" + }, + { + "name": "mul", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "*" + }, + { + "name": "neq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "!=" + }, + { + "name": "net.cidr_contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_contains_matches", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "static": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_expand", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_intersects", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_merge", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_overlap", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "numbers.range", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "object.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.get", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.union", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "opa.runtime", + "decl": { + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "or", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "|" + }, + { + "name": "plus", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "+" + }, + { + "name": "product", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "re_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.find_all_string_submatch_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.find_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.globs_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.template_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "rego.parse_module", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "rem", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "%" + }, + { + "name": "replace", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "round", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.compare", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.is_valid", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "set_diff", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "sort", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "sprintf", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "startswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "strings.replace_n", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "string" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "substring", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "sum", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.add_date", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.clock", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.date", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.now_ns", + "decl": { + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_duration_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_ns", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_rfc3339_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.weekday", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "to_number", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "type": "string" + }, + { + "type": "boolean" + }, + { + "type": "null" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "trace", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "trim", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_left", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_prefix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_right", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_space", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_suffix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "type_name", + "decl": { + "args": [ + { + "of": [ + { + "type": "any" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "union", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "units.parse_bytes", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "upper", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode_object", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "dynamic": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode_object", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "uuid.rfc4122", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "walk", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "static": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "function" + }, + "relation": true + }, + { + "name": "yaml.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "yaml.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "yaml.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + } + ] +} diff --git a/vendor/github.com/open-policy-agent/opa/capabilities/v0.25.0-rc4.json b/vendor/github.com/open-policy-agent/opa/capabilities/v0.25.0-rc4.json new file mode 100644 index 00000000..b5570417 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/capabilities/v0.25.0-rc4.json @@ -0,0 +1,3313 @@ +{ + "builtins": [ + { + "name": "abs", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "all", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "and", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "\u0026" + }, + { + "name": "any", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "array.concat", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "array.slice", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "assign", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": ":=" + }, + { + "name": "base64.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "base64url.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode_no_pad", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "bits.and", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.lsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.negate", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.or", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.rsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.xor", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "cast_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "cast_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "cast_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "null" + }, + "type": "function" + } + }, + { + "name": "cast_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "cast_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "cast_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "concat", + "decl": { + "args": [ + { + "type": "string" + }, + { + "of": [ + { + "of": { + "type": "string" + }, + "type": "set" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "count", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "crypto.md5", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha1", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha256", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificate_request", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "div", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "/" + }, + { + "name": "endswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "eq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "=" + }, + { + "name": "equal", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "==" + }, + { + "name": "format_int", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "glob.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "glob.quote_meta", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "graph.reachable", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "gt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e" + }, + { + "name": "gte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e=" + }, + { + "name": "hex.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "hex.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "http.send", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "indexof", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "intersection", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode_verify", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign_raw", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_number", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "json.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "lower", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "lt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c" + }, + { + "name": "lte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c=" + }, + { + "name": "max", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "min", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "minus", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + "type": "function" + }, + "infix": "-" + }, + { + "name": "mul", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "*" + }, + { + "name": "neq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "!=" + }, + { + "name": "net.cidr_contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_contains_matches", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "static": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_expand", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_intersects", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_merge", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_overlap", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "numbers.range", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "object.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.get", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.union", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "opa.runtime", + "decl": { + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "or", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "|" + }, + { + "name": "plus", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "+" + }, + { + "name": "product", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "re_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.find_all_string_submatch_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.find_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.globs_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.template_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "rego.parse_module", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "rem", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "%" + }, + { + "name": "replace", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "round", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.compare", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.is_valid", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "set_diff", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "sort", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "sprintf", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "startswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "strings.replace_n", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "string" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "substring", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "sum", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.add_date", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.clock", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.date", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.now_ns", + "decl": { + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_duration_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_ns", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_rfc3339_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.weekday", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "to_number", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "type": "string" + }, + { + "type": "boolean" + }, + { + "type": "null" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "trace", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "trim", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_left", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_prefix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_right", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_space", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_suffix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "type_name", + "decl": { + "args": [ + { + "of": [ + { + "type": "any" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "union", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "units.parse_bytes", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "upper", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode_object", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "dynamic": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode_object", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "uuid.rfc4122", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "walk", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "static": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "function" + }, + "relation": true + }, + { + "name": "yaml.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "yaml.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "yaml.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + } + ] +} diff --git a/vendor/github.com/open-policy-agent/opa/capabilities/v0.25.0.json b/vendor/github.com/open-policy-agent/opa/capabilities/v0.25.0.json new file mode 100644 index 00000000..650eed4b --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/capabilities/v0.25.0.json @@ -0,0 +1,3355 @@ +{ + "builtins": [ + { + "name": "abs", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "all", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "and", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "\u0026" + }, + { + "name": "any", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "array.concat", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "array.slice", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "assign", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": ":=" + }, + { + "name": "base64.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "base64url.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode_no_pad", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "bits.and", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.lsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.negate", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.or", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.rsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.xor", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "cast_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "cast_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "cast_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "null" + }, + "type": "function" + } + }, + { + "name": "cast_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "cast_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "cast_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "concat", + "decl": { + "args": [ + { + "type": "string" + }, + { + "of": [ + { + "of": { + "type": "string" + }, + "type": "set" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "count", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "crypto.md5", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha1", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha256", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificate_request", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "div", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "/" + }, + { + "name": "endswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "eq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "=" + }, + { + "name": "equal", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "==" + }, + { + "name": "format_int", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "glob.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "glob.quote_meta", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "graph.reachable", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "gt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e" + }, + { + "name": "gte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e=" + }, + { + "name": "hex.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "hex.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "http.send", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "indexof", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "intersection", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode_verify", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign_raw", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_number", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "json.patch", + "decl": { + "args": [ + { + "type": "any" + }, + { + "dynamic": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "static": [ + { + "key": "op", + "value": { + "type": "string" + } + }, + { + "key": "path", + "value": { + "type": "any" + } + } + ], + "type": "object" + }, + "type": "array" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "lower", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "lt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c" + }, + { + "name": "lte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c=" + }, + { + "name": "max", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "min", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "minus", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + "type": "function" + }, + "infix": "-" + }, + { + "name": "mul", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "*" + }, + { + "name": "neq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "!=" + }, + { + "name": "net.cidr_contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_contains_matches", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "static": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_expand", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_intersects", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_merge", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_overlap", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "numbers.range", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "object.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.get", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.union", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "opa.runtime", + "decl": { + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "or", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "|" + }, + { + "name": "plus", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "+" + }, + { + "name": "product", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "re_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.find_all_string_submatch_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.find_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.globs_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.template_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "rego.parse_module", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "rem", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "%" + }, + { + "name": "replace", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "round", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.compare", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.is_valid", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "set_diff", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "sort", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "sprintf", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "startswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "strings.replace_n", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "string" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "substring", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "sum", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.add_date", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.clock", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.date", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.now_ns", + "decl": { + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_duration_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_ns", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_rfc3339_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.weekday", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "to_number", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "type": "string" + }, + { + "type": "boolean" + }, + { + "type": "null" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "trace", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "trim", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_left", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_prefix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_right", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_space", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_suffix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "type_name", + "decl": { + "args": [ + { + "of": [ + { + "type": "any" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "union", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "units.parse_bytes", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "upper", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode_object", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "dynamic": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode_object", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "uuid.rfc4122", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "walk", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "static": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "function" + }, + "relation": true + }, + { + "name": "yaml.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "yaml.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "yaml.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + } + ] +} diff --git a/vendor/github.com/open-policy-agent/opa/capabilities/v0.25.1.json b/vendor/github.com/open-policy-agent/opa/capabilities/v0.25.1.json new file mode 100644 index 00000000..650eed4b --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/capabilities/v0.25.1.json @@ -0,0 +1,3355 @@ +{ + "builtins": [ + { + "name": "abs", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "all", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "and", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "\u0026" + }, + { + "name": "any", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "array.concat", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "array.slice", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "assign", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": ":=" + }, + { + "name": "base64.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "base64url.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode_no_pad", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "bits.and", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.lsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.negate", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.or", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.rsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.xor", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "cast_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "cast_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "cast_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "null" + }, + "type": "function" + } + }, + { + "name": "cast_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "cast_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "cast_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "concat", + "decl": { + "args": [ + { + "type": "string" + }, + { + "of": [ + { + "of": { + "type": "string" + }, + "type": "set" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "count", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "crypto.md5", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha1", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha256", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificate_request", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "div", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "/" + }, + { + "name": "endswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "eq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "=" + }, + { + "name": "equal", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "==" + }, + { + "name": "format_int", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "glob.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "glob.quote_meta", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "graph.reachable", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "gt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e" + }, + { + "name": "gte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e=" + }, + { + "name": "hex.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "hex.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "http.send", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "indexof", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "intersection", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode_verify", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign_raw", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_number", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "json.patch", + "decl": { + "args": [ + { + "type": "any" + }, + { + "dynamic": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "static": [ + { + "key": "op", + "value": { + "type": "string" + } + }, + { + "key": "path", + "value": { + "type": "any" + } + } + ], + "type": "object" + }, + "type": "array" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "lower", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "lt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c" + }, + { + "name": "lte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c=" + }, + { + "name": "max", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "min", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "minus", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + "type": "function" + }, + "infix": "-" + }, + { + "name": "mul", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "*" + }, + { + "name": "neq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "!=" + }, + { + "name": "net.cidr_contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_contains_matches", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "static": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_expand", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_intersects", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_merge", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_overlap", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "numbers.range", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "object.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.get", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.union", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "opa.runtime", + "decl": { + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "or", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "|" + }, + { + "name": "plus", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "+" + }, + { + "name": "product", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "re_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.find_all_string_submatch_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.find_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.globs_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.template_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "rego.parse_module", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "rem", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "%" + }, + { + "name": "replace", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "round", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.compare", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.is_valid", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "set_diff", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "sort", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "sprintf", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "startswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "strings.replace_n", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "string" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "substring", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "sum", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.add_date", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.clock", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.date", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.now_ns", + "decl": { + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_duration_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_ns", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_rfc3339_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.weekday", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "to_number", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "type": "string" + }, + { + "type": "boolean" + }, + { + "type": "null" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "trace", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "trim", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_left", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_prefix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_right", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_space", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_suffix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "type_name", + "decl": { + "args": [ + { + "of": [ + { + "type": "any" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "union", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "units.parse_bytes", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "upper", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode_object", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "dynamic": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode_object", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "uuid.rfc4122", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "walk", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "static": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "function" + }, + "relation": true + }, + { + "name": "yaml.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "yaml.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "yaml.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + } + ] +} diff --git a/vendor/github.com/open-policy-agent/opa/capabilities/v0.25.2.json b/vendor/github.com/open-policy-agent/opa/capabilities/v0.25.2.json new file mode 100644 index 00000000..650eed4b --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/capabilities/v0.25.2.json @@ -0,0 +1,3355 @@ +{ + "builtins": [ + { + "name": "abs", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "all", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "and", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "\u0026" + }, + { + "name": "any", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "array.concat", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "array.slice", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "assign", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": ":=" + }, + { + "name": "base64.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "base64url.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode_no_pad", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "bits.and", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.lsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.negate", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.or", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.rsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.xor", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "cast_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "cast_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "cast_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "null" + }, + "type": "function" + } + }, + { + "name": "cast_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "cast_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "cast_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "concat", + "decl": { + "args": [ + { + "type": "string" + }, + { + "of": [ + { + "of": { + "type": "string" + }, + "type": "set" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "count", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "crypto.md5", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha1", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha256", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificate_request", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "div", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "/" + }, + { + "name": "endswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "eq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "=" + }, + { + "name": "equal", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "==" + }, + { + "name": "format_int", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "glob.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "glob.quote_meta", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "graph.reachable", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "gt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e" + }, + { + "name": "gte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e=" + }, + { + "name": "hex.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "hex.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "http.send", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "indexof", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "intersection", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode_verify", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign_raw", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_number", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "json.patch", + "decl": { + "args": [ + { + "type": "any" + }, + { + "dynamic": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "static": [ + { + "key": "op", + "value": { + "type": "string" + } + }, + { + "key": "path", + "value": { + "type": "any" + } + } + ], + "type": "object" + }, + "type": "array" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "lower", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "lt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c" + }, + { + "name": "lte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c=" + }, + { + "name": "max", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "min", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "minus", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + "type": "function" + }, + "infix": "-" + }, + { + "name": "mul", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "*" + }, + { + "name": "neq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "!=" + }, + { + "name": "net.cidr_contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_contains_matches", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "static": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_expand", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_intersects", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_merge", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_overlap", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "numbers.range", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "object.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.get", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.union", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "opa.runtime", + "decl": { + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "or", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "|" + }, + { + "name": "plus", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "+" + }, + { + "name": "product", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "re_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.find_all_string_submatch_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.find_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.globs_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.template_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "rego.parse_module", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "rem", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "%" + }, + { + "name": "replace", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "round", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.compare", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.is_valid", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "set_diff", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "sort", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "sprintf", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "startswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "strings.replace_n", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "string" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "substring", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "sum", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.add_date", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.clock", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.date", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.now_ns", + "decl": { + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_duration_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_ns", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_rfc3339_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.weekday", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "to_number", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "type": "string" + }, + { + "type": "boolean" + }, + { + "type": "null" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "trace", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "trim", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_left", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_prefix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_right", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_space", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_suffix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "type_name", + "decl": { + "args": [ + { + "of": [ + { + "type": "any" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "union", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "units.parse_bytes", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "upper", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode_object", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "dynamic": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode_object", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "uuid.rfc4122", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "walk", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "static": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "function" + }, + "relation": true + }, + { + "name": "yaml.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "yaml.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "yaml.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + } + ] +} diff --git a/vendor/github.com/open-policy-agent/opa/capabilities/v0.26.0.json b/vendor/github.com/open-policy-agent/opa/capabilities/v0.26.0.json new file mode 100644 index 00000000..6910546b --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/capabilities/v0.26.0.json @@ -0,0 +1,3383 @@ +{ + "builtins": [ + { + "name": "abs", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "all", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "and", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "\u0026" + }, + { + "name": "any", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "array.concat", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "array.slice", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "assign", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": ":=" + }, + { + "name": "base64.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "base64url.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode_no_pad", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "bits.and", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.lsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.negate", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.or", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.rsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.xor", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "cast_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "cast_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "cast_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "null" + }, + "type": "function" + } + }, + { + "name": "cast_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "cast_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "cast_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "ceil", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "concat", + "decl": { + "args": [ + { + "type": "string" + }, + { + "of": [ + { + "of": { + "type": "string" + }, + "type": "set" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "count", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "crypto.md5", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha1", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha256", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificate_request", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "div", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "/" + }, + { + "name": "endswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "eq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "=" + }, + { + "name": "equal", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "==" + }, + { + "name": "floor", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "format_int", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "glob.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "glob.quote_meta", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "graph.reachable", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "gt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e" + }, + { + "name": "gte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e=" + }, + { + "name": "hex.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "hex.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "http.send", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "indexof", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "intersection", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode_verify", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign_raw", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_number", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "json.patch", + "decl": { + "args": [ + { + "type": "any" + }, + { + "dynamic": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "static": [ + { + "key": "op", + "value": { + "type": "string" + } + }, + { + "key": "path", + "value": { + "type": "any" + } + } + ], + "type": "object" + }, + "type": "array" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "lower", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "lt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c" + }, + { + "name": "lte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c=" + }, + { + "name": "max", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "min", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "minus", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + "type": "function" + }, + "infix": "-" + }, + { + "name": "mul", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "*" + }, + { + "name": "neq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "!=" + }, + { + "name": "net.cidr_contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_contains_matches", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "static": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_expand", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_intersects", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_merge", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_overlap", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "numbers.range", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "object.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.get", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.union", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "opa.runtime", + "decl": { + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "or", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "|" + }, + { + "name": "plus", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "+" + }, + { + "name": "product", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "re_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.find_all_string_submatch_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.find_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.globs_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.template_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "rego.parse_module", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "rem", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "%" + }, + { + "name": "replace", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "round", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.compare", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.is_valid", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "set_diff", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "sort", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "sprintf", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "startswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "strings.replace_n", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "string" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "substring", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "sum", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.add_date", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.clock", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.date", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.now_ns", + "decl": { + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_duration_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_ns", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_rfc3339_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.weekday", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "to_number", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "type": "string" + }, + { + "type": "boolean" + }, + { + "type": "null" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "trace", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "trim", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_left", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_prefix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_right", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_space", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_suffix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "type_name", + "decl": { + "args": [ + { + "of": [ + { + "type": "any" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "union", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "units.parse_bytes", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "upper", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode_object", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "dynamic": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode_object", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "uuid.rfc4122", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "walk", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "static": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "function" + }, + "relation": true + }, + { + "name": "yaml.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "yaml.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "yaml.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + } + ] +} diff --git a/vendor/github.com/open-policy-agent/opa/capabilities/v0.27.0.json b/vendor/github.com/open-policy-agent/opa/capabilities/v0.27.0.json new file mode 100644 index 00000000..b9046a6e --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/capabilities/v0.27.0.json @@ -0,0 +1,3389 @@ +{ + "builtins": [ + { + "name": "abs", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "all", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "and", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "\u0026" + }, + { + "name": "any", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "array.concat", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "array.slice", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "assign", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": ":=" + }, + { + "name": "base64.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "base64url.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode_no_pad", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "bits.and", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.lsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.negate", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.or", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.rsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.xor", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "cast_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "cast_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "cast_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "null" + }, + "type": "function" + } + }, + { + "name": "cast_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "cast_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "cast_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "ceil", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "concat", + "decl": { + "args": [ + { + "type": "string" + }, + { + "of": [ + { + "of": { + "type": "string" + }, + "type": "set" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "count", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "crypto.md5", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha1", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha256", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificate_request", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "div", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "/" + }, + { + "name": "endswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "eq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "=" + }, + { + "name": "equal", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "==" + }, + { + "name": "floor", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "format_int", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "glob.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "glob.quote_meta", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "graph.reachable", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "gt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e" + }, + { + "name": "gte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e=" + }, + { + "name": "hex.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "hex.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "http.send", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "indexof", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "intersection", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode_verify", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign_raw", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_number", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "json.patch", + "decl": { + "args": [ + { + "type": "any" + }, + { + "dynamic": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "static": [ + { + "key": "op", + "value": { + "type": "string" + } + }, + { + "key": "path", + "value": { + "type": "any" + } + } + ], + "type": "object" + }, + "type": "array" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "lower", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "lt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c" + }, + { + "name": "lte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c=" + }, + { + "name": "max", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "min", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "minus", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + "type": "function" + }, + "infix": "-" + }, + { + "name": "mul", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "*" + }, + { + "name": "neq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "!=" + }, + { + "name": "net.cidr_contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_contains_matches", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "static": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_expand", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_intersects", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_merge", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_overlap", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "numbers.range", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "object.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.get", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.union", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "opa.runtime", + "decl": { + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "or", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "|" + }, + { + "name": "plus", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "+" + }, + { + "name": "product", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "re_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.find_all_string_submatch_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.find_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.globs_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.template_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "rego.parse_module", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "rem", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "%" + }, + { + "name": "replace", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "round", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.compare", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.is_valid", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "set_diff", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "sort", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "sprintf", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "startswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "strings.replace_n", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "string" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "substring", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "sum", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.add_date", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.clock", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.date", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.now_ns", + "decl": { + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_duration_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_ns", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_rfc3339_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.weekday", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "to_number", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "type": "string" + }, + { + "type": "boolean" + }, + { + "type": "null" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "trace", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "trim", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_left", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_prefix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_right", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_space", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_suffix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "type_name", + "decl": { + "args": [ + { + "of": [ + { + "type": "any" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "union", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "units.parse_bytes", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "upper", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode_object", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "dynamic": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode_object", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "uuid.rfc4122", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "walk", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "static": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "function" + }, + "relation": true + }, + { + "name": "yaml.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "yaml.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "yaml.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + } + ], + "wasm_abi_versions": [ + { + "version": 1, + "minor_version": 1 + } + ] +} diff --git a/vendor/github.com/open-policy-agent/opa/capabilities/v0.27.1.json b/vendor/github.com/open-policy-agent/opa/capabilities/v0.27.1.json new file mode 100644 index 00000000..b9046a6e --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/capabilities/v0.27.1.json @@ -0,0 +1,3389 @@ +{ + "builtins": [ + { + "name": "abs", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "all", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "and", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "\u0026" + }, + { + "name": "any", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "array.concat", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "array.slice", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "assign", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": ":=" + }, + { + "name": "base64.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "base64url.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode_no_pad", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "bits.and", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.lsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.negate", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.or", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.rsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.xor", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "cast_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "cast_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "cast_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "null" + }, + "type": "function" + } + }, + { + "name": "cast_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "cast_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "cast_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "ceil", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "concat", + "decl": { + "args": [ + { + "type": "string" + }, + { + "of": [ + { + "of": { + "type": "string" + }, + "type": "set" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "count", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "crypto.md5", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha1", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha256", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificate_request", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "div", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "/" + }, + { + "name": "endswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "eq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "=" + }, + { + "name": "equal", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "==" + }, + { + "name": "floor", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "format_int", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "glob.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "glob.quote_meta", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "graph.reachable", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "gt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e" + }, + { + "name": "gte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e=" + }, + { + "name": "hex.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "hex.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "http.send", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "indexof", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "intersection", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode_verify", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign_raw", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_number", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "json.patch", + "decl": { + "args": [ + { + "type": "any" + }, + { + "dynamic": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "static": [ + { + "key": "op", + "value": { + "type": "string" + } + }, + { + "key": "path", + "value": { + "type": "any" + } + } + ], + "type": "object" + }, + "type": "array" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "lower", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "lt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c" + }, + { + "name": "lte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c=" + }, + { + "name": "max", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "min", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "minus", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + "type": "function" + }, + "infix": "-" + }, + { + "name": "mul", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "*" + }, + { + "name": "neq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "!=" + }, + { + "name": "net.cidr_contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_contains_matches", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "static": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_expand", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_intersects", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_merge", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_overlap", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "numbers.range", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "object.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.get", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.union", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "opa.runtime", + "decl": { + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "or", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "|" + }, + { + "name": "plus", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "+" + }, + { + "name": "product", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "re_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.find_all_string_submatch_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.find_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.globs_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.template_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "rego.parse_module", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "rem", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "%" + }, + { + "name": "replace", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "round", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.compare", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.is_valid", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "set_diff", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "sort", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "sprintf", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "startswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "strings.replace_n", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "string" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "substring", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "sum", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.add_date", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.clock", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.date", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.now_ns", + "decl": { + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_duration_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_ns", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_rfc3339_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.weekday", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "to_number", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "type": "string" + }, + { + "type": "boolean" + }, + { + "type": "null" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "trace", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "trim", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_left", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_prefix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_right", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_space", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_suffix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "type_name", + "decl": { + "args": [ + { + "of": [ + { + "type": "any" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "union", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "units.parse_bytes", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "upper", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode_object", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "dynamic": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode_object", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "uuid.rfc4122", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "walk", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "static": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "function" + }, + "relation": true + }, + { + "name": "yaml.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "yaml.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "yaml.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + } + ], + "wasm_abi_versions": [ + { + "version": 1, + "minor_version": 1 + } + ] +} diff --git a/vendor/github.com/open-policy-agent/opa/capabilities/v0.28.0.json b/vendor/github.com/open-policy-agent/opa/capabilities/v0.28.0.json new file mode 100644 index 00000000..7ff34daf --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/capabilities/v0.28.0.json @@ -0,0 +1,3458 @@ +{ + "builtins": [ + { + "name": "abs", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "all", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "and", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "\u0026" + }, + { + "name": "any", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "array.concat", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "array.slice", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "assign", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": ":=" + }, + { + "name": "base64.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "base64url.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode_no_pad", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "bits.and", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.lsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.negate", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.or", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.rsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.xor", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "cast_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "cast_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "cast_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "null" + }, + "type": "function" + } + }, + { + "name": "cast_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "cast_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "cast_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "ceil", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "concat", + "decl": { + "args": [ + { + "type": "string" + }, + { + "of": [ + { + "of": { + "type": "string" + }, + "type": "set" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "count", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "crypto.md5", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha1", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha256", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificate_request", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "div", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "/" + }, + { + "name": "endswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "eq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "=" + }, + { + "name": "equal", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "==" + }, + { + "name": "floor", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "format_int", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "glob.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "glob.quote_meta", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "graph.reachable", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "gt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e" + }, + { + "name": "gte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e=" + }, + { + "name": "hex.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "hex.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "http.send", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "indexof", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "intersection", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode_verify", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign_raw", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_number", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "json.patch", + "decl": { + "args": [ + { + "type": "any" + }, + { + "dynamic": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "static": [ + { + "key": "op", + "value": { + "type": "string" + } + }, + { + "key": "path", + "value": { + "type": "any" + } + } + ], + "type": "object" + }, + "type": "array" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "lower", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "lt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c" + }, + { + "name": "lte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c=" + }, + { + "name": "max", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "min", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "minus", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + "type": "function" + }, + "infix": "-" + }, + { + "name": "mul", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "*" + }, + { + "name": "neq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "!=" + }, + { + "name": "net.cidr_contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_contains_matches", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "static": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_expand", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_intersects", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_merge", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_overlap", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "numbers.range", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "object.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.get", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.union", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "opa.runtime", + "decl": { + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "or", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "|" + }, + { + "name": "plus", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "+" + }, + { + "name": "product", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "re_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.find_all_string_submatch_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.find_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.globs_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.template_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "rego.parse_module", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "rem", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "%" + }, + { + "name": "replace", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "round", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.compare", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.is_valid", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "set_diff", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "sort", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "sprintf", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "startswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "strings.replace_n", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "string" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "substring", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "sum", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.add_date", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.clock", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.date", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.diff", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.now_ns", + "decl": { + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_duration_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_ns", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_rfc3339_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.weekday", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "to_number", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "type": "string" + }, + { + "type": "boolean" + }, + { + "type": "null" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "trace", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "trim", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_left", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_prefix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_right", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_space", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_suffix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "type_name", + "decl": { + "args": [ + { + "of": [ + { + "type": "any" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "union", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "units.parse_bytes", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "upper", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode_object", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "dynamic": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode_object", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "uuid.rfc4122", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "walk", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "static": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "function" + }, + "relation": true + }, + { + "name": "yaml.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "yaml.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "yaml.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + } + ], + "wasm_abi_versions": [ + { + "version": 1, + "minor_version": 1 + } + ] +} diff --git a/vendor/github.com/open-policy-agent/opa/capabilities/v0.29.0.json b/vendor/github.com/open-policy-agent/opa/capabilities/v0.29.0.json new file mode 100644 index 00000000..7ff34daf --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/capabilities/v0.29.0.json @@ -0,0 +1,3458 @@ +{ + "builtins": [ + { + "name": "abs", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "all", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "and", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "\u0026" + }, + { + "name": "any", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "array.concat", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "array.slice", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "assign", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": ":=" + }, + { + "name": "base64.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "base64url.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode_no_pad", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "bits.and", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.lsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.negate", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.or", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.rsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.xor", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "cast_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "cast_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "cast_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "null" + }, + "type": "function" + } + }, + { + "name": "cast_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "cast_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "cast_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "ceil", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "concat", + "decl": { + "args": [ + { + "type": "string" + }, + { + "of": [ + { + "of": { + "type": "string" + }, + "type": "set" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "count", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "crypto.md5", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha1", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha256", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificate_request", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "div", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "/" + }, + { + "name": "endswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "eq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "=" + }, + { + "name": "equal", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "==" + }, + { + "name": "floor", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "format_int", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "glob.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "glob.quote_meta", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "graph.reachable", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "gt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e" + }, + { + "name": "gte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e=" + }, + { + "name": "hex.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "hex.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "http.send", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "indexof", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "intersection", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode_verify", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign_raw", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_number", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "json.patch", + "decl": { + "args": [ + { + "type": "any" + }, + { + "dynamic": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "static": [ + { + "key": "op", + "value": { + "type": "string" + } + }, + { + "key": "path", + "value": { + "type": "any" + } + } + ], + "type": "object" + }, + "type": "array" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "lower", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "lt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c" + }, + { + "name": "lte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c=" + }, + { + "name": "max", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "min", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "minus", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + "type": "function" + }, + "infix": "-" + }, + { + "name": "mul", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "*" + }, + { + "name": "neq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "!=" + }, + { + "name": "net.cidr_contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_contains_matches", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "static": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_expand", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_intersects", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_merge", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_overlap", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "numbers.range", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "object.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.get", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.union", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "opa.runtime", + "decl": { + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "or", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "|" + }, + { + "name": "plus", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "+" + }, + { + "name": "product", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "re_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.find_all_string_submatch_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.find_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.globs_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.template_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "rego.parse_module", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "rem", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "%" + }, + { + "name": "replace", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "round", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.compare", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.is_valid", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "set_diff", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "sort", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "sprintf", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "startswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "strings.replace_n", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "string" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "substring", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "sum", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.add_date", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.clock", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.date", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.diff", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.now_ns", + "decl": { + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_duration_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_ns", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_rfc3339_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.weekday", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "to_number", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "type": "string" + }, + { + "type": "boolean" + }, + { + "type": "null" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "trace", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "trim", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_left", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_prefix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_right", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_space", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_suffix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "type_name", + "decl": { + "args": [ + { + "of": [ + { + "type": "any" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "union", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "units.parse_bytes", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "upper", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode_object", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "dynamic": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode_object", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "uuid.rfc4122", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "walk", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "static": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "function" + }, + "relation": true + }, + { + "name": "yaml.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "yaml.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "yaml.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + } + ], + "wasm_abi_versions": [ + { + "version": 1, + "minor_version": 1 + } + ] +} diff --git a/vendor/github.com/open-policy-agent/opa/capabilities/v0.29.1.json b/vendor/github.com/open-policy-agent/opa/capabilities/v0.29.1.json new file mode 100644 index 00000000..7ff34daf --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/capabilities/v0.29.1.json @@ -0,0 +1,3458 @@ +{ + "builtins": [ + { + "name": "abs", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "all", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "and", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "\u0026" + }, + { + "name": "any", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "array.concat", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "array.slice", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "assign", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": ":=" + }, + { + "name": "base64.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "base64url.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode_no_pad", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "bits.and", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.lsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.negate", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.or", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.rsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.xor", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "cast_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "cast_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "cast_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "null" + }, + "type": "function" + } + }, + { + "name": "cast_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "cast_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "cast_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "ceil", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "concat", + "decl": { + "args": [ + { + "type": "string" + }, + { + "of": [ + { + "of": { + "type": "string" + }, + "type": "set" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "count", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "crypto.md5", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha1", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha256", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificate_request", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "div", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "/" + }, + { + "name": "endswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "eq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "=" + }, + { + "name": "equal", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "==" + }, + { + "name": "floor", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "format_int", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "glob.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "glob.quote_meta", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "graph.reachable", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "gt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e" + }, + { + "name": "gte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e=" + }, + { + "name": "hex.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "hex.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "http.send", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "indexof", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "intersection", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode_verify", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign_raw", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_number", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "json.patch", + "decl": { + "args": [ + { + "type": "any" + }, + { + "dynamic": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "static": [ + { + "key": "op", + "value": { + "type": "string" + } + }, + { + "key": "path", + "value": { + "type": "any" + } + } + ], + "type": "object" + }, + "type": "array" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "lower", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "lt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c" + }, + { + "name": "lte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c=" + }, + { + "name": "max", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "min", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "minus", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + "type": "function" + }, + "infix": "-" + }, + { + "name": "mul", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "*" + }, + { + "name": "neq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "!=" + }, + { + "name": "net.cidr_contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_contains_matches", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "static": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_expand", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_intersects", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_merge", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_overlap", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "numbers.range", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "object.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.get", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.union", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "opa.runtime", + "decl": { + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "or", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "|" + }, + { + "name": "plus", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "+" + }, + { + "name": "product", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "re_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.find_all_string_submatch_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.find_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.globs_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.template_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "rego.parse_module", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "rem", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "%" + }, + { + "name": "replace", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "round", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.compare", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.is_valid", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "set_diff", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "sort", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "sprintf", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "startswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "strings.replace_n", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "string" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "substring", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "sum", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.add_date", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.clock", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.date", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.diff", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.now_ns", + "decl": { + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_duration_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_ns", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_rfc3339_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.weekday", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "to_number", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "type": "string" + }, + { + "type": "boolean" + }, + { + "type": "null" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "trace", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "trim", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_left", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_prefix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_right", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_space", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_suffix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "type_name", + "decl": { + "args": [ + { + "of": [ + { + "type": "any" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "union", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "units.parse_bytes", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "upper", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode_object", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "dynamic": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode_object", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "uuid.rfc4122", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "walk", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "static": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "function" + }, + "relation": true + }, + { + "name": "yaml.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "yaml.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "yaml.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + } + ], + "wasm_abi_versions": [ + { + "version": 1, + "minor_version": 1 + } + ] +} diff --git a/vendor/github.com/open-policy-agent/opa/capabilities/v0.29.2.json b/vendor/github.com/open-policy-agent/opa/capabilities/v0.29.2.json new file mode 100644 index 00000000..7ff34daf --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/capabilities/v0.29.2.json @@ -0,0 +1,3458 @@ +{ + "builtins": [ + { + "name": "abs", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "all", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "and", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "\u0026" + }, + { + "name": "any", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "array.concat", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "array.slice", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "assign", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": ":=" + }, + { + "name": "base64.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "base64url.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode_no_pad", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "bits.and", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.lsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.negate", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.or", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.rsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.xor", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "cast_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "cast_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "cast_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "null" + }, + "type": "function" + } + }, + { + "name": "cast_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "cast_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "cast_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "ceil", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "concat", + "decl": { + "args": [ + { + "type": "string" + }, + { + "of": [ + { + "of": { + "type": "string" + }, + "type": "set" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "count", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "crypto.md5", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha1", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha256", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificate_request", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "div", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "/" + }, + { + "name": "endswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "eq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "=" + }, + { + "name": "equal", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "==" + }, + { + "name": "floor", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "format_int", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "glob.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "glob.quote_meta", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "graph.reachable", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "gt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e" + }, + { + "name": "gte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e=" + }, + { + "name": "hex.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "hex.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "http.send", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "indexof", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "intersection", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode_verify", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign_raw", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_number", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "json.patch", + "decl": { + "args": [ + { + "type": "any" + }, + { + "dynamic": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "static": [ + { + "key": "op", + "value": { + "type": "string" + } + }, + { + "key": "path", + "value": { + "type": "any" + } + } + ], + "type": "object" + }, + "type": "array" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "lower", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "lt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c" + }, + { + "name": "lte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c=" + }, + { + "name": "max", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "min", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "minus", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + "type": "function" + }, + "infix": "-" + }, + { + "name": "mul", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "*" + }, + { + "name": "neq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "!=" + }, + { + "name": "net.cidr_contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_contains_matches", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "static": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_expand", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_intersects", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_merge", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_overlap", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "numbers.range", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "object.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.get", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.union", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "opa.runtime", + "decl": { + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "or", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "|" + }, + { + "name": "plus", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "+" + }, + { + "name": "product", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "re_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.find_all_string_submatch_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.find_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.globs_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.template_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "rego.parse_module", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "rem", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "%" + }, + { + "name": "replace", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "round", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.compare", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.is_valid", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "set_diff", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "sort", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "sprintf", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "startswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "strings.replace_n", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "string" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "substring", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "sum", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.add_date", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.clock", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.date", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.diff", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.now_ns", + "decl": { + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_duration_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_ns", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_rfc3339_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.weekday", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "to_number", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "type": "string" + }, + { + "type": "boolean" + }, + { + "type": "null" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "trace", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "trim", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_left", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_prefix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_right", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_space", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_suffix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "type_name", + "decl": { + "args": [ + { + "of": [ + { + "type": "any" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "union", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "units.parse_bytes", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "upper", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode_object", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "dynamic": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode_object", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "uuid.rfc4122", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "walk", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "static": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "function" + }, + "relation": true + }, + { + "name": "yaml.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "yaml.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "yaml.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + } + ], + "wasm_abi_versions": [ + { + "version": 1, + "minor_version": 1 + } + ] +} diff --git a/vendor/github.com/open-policy-agent/opa/capabilities/v0.29.3.json b/vendor/github.com/open-policy-agent/opa/capabilities/v0.29.3.json new file mode 100644 index 00000000..7ff34daf --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/capabilities/v0.29.3.json @@ -0,0 +1,3458 @@ +{ + "builtins": [ + { + "name": "abs", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "all", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "and", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "\u0026" + }, + { + "name": "any", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "array.concat", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "array.slice", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "assign", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": ":=" + }, + { + "name": "base64.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "base64url.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode_no_pad", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "bits.and", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.lsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.negate", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.or", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.rsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.xor", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "cast_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "cast_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "cast_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "null" + }, + "type": "function" + } + }, + { + "name": "cast_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "cast_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "cast_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "ceil", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "concat", + "decl": { + "args": [ + { + "type": "string" + }, + { + "of": [ + { + "of": { + "type": "string" + }, + "type": "set" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "count", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "crypto.md5", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha1", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha256", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificate_request", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "div", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "/" + }, + { + "name": "endswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "eq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "=" + }, + { + "name": "equal", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "==" + }, + { + "name": "floor", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "format_int", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "glob.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "glob.quote_meta", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "graph.reachable", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "gt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e" + }, + { + "name": "gte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e=" + }, + { + "name": "hex.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "hex.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "http.send", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "indexof", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "intersection", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode_verify", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign_raw", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_number", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "json.patch", + "decl": { + "args": [ + { + "type": "any" + }, + { + "dynamic": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "static": [ + { + "key": "op", + "value": { + "type": "string" + } + }, + { + "key": "path", + "value": { + "type": "any" + } + } + ], + "type": "object" + }, + "type": "array" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "lower", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "lt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c" + }, + { + "name": "lte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c=" + }, + { + "name": "max", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "min", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "minus", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + "type": "function" + }, + "infix": "-" + }, + { + "name": "mul", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "*" + }, + { + "name": "neq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "!=" + }, + { + "name": "net.cidr_contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_contains_matches", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "static": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_expand", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_intersects", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_merge", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_overlap", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "numbers.range", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "object.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.get", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.union", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "opa.runtime", + "decl": { + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "or", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "|" + }, + { + "name": "plus", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "+" + }, + { + "name": "product", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "re_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.find_all_string_submatch_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.find_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.globs_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.template_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "rego.parse_module", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "rem", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "%" + }, + { + "name": "replace", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "round", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.compare", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.is_valid", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "set_diff", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "sort", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "sprintf", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "startswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "strings.replace_n", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "string" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "substring", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "sum", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.add_date", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.clock", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.date", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.diff", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.now_ns", + "decl": { + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_duration_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_ns", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_rfc3339_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.weekday", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "to_number", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "type": "string" + }, + { + "type": "boolean" + }, + { + "type": "null" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "trace", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "trim", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_left", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_prefix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_right", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_space", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_suffix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "type_name", + "decl": { + "args": [ + { + "of": [ + { + "type": "any" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "union", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "units.parse_bytes", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "upper", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode_object", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "dynamic": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode_object", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "uuid.rfc4122", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "walk", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "static": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "function" + }, + "relation": true + }, + { + "name": "yaml.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "yaml.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "yaml.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + } + ], + "wasm_abi_versions": [ + { + "version": 1, + "minor_version": 1 + } + ] +} diff --git a/vendor/github.com/open-policy-agent/opa/capabilities/v0.29.4.json b/vendor/github.com/open-policy-agent/opa/capabilities/v0.29.4.json new file mode 100644 index 00000000..7ff34daf --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/capabilities/v0.29.4.json @@ -0,0 +1,3458 @@ +{ + "builtins": [ + { + "name": "abs", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "all", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "and", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "\u0026" + }, + { + "name": "any", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "array.concat", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "array.slice", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "assign", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": ":=" + }, + { + "name": "base64.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "base64url.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode_no_pad", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "bits.and", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.lsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.negate", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.or", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.rsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.xor", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "cast_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "cast_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "cast_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "null" + }, + "type": "function" + } + }, + { + "name": "cast_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "cast_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "cast_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "ceil", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "concat", + "decl": { + "args": [ + { + "type": "string" + }, + { + "of": [ + { + "of": { + "type": "string" + }, + "type": "set" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "count", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "crypto.md5", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha1", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha256", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificate_request", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "div", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "/" + }, + { + "name": "endswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "eq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "=" + }, + { + "name": "equal", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "==" + }, + { + "name": "floor", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "format_int", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "glob.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "glob.quote_meta", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "graph.reachable", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "gt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e" + }, + { + "name": "gte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e=" + }, + { + "name": "hex.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "hex.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "http.send", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "indexof", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "intersection", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode_verify", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign_raw", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_number", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "json.patch", + "decl": { + "args": [ + { + "type": "any" + }, + { + "dynamic": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "static": [ + { + "key": "op", + "value": { + "type": "string" + } + }, + { + "key": "path", + "value": { + "type": "any" + } + } + ], + "type": "object" + }, + "type": "array" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "lower", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "lt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c" + }, + { + "name": "lte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c=" + }, + { + "name": "max", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "min", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "minus", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + "type": "function" + }, + "infix": "-" + }, + { + "name": "mul", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "*" + }, + { + "name": "neq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "!=" + }, + { + "name": "net.cidr_contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_contains_matches", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "static": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_expand", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_intersects", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_merge", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_overlap", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "numbers.range", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "object.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.get", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.union", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "opa.runtime", + "decl": { + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "or", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "|" + }, + { + "name": "plus", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "+" + }, + { + "name": "product", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "re_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.find_all_string_submatch_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.find_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.globs_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.template_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "rego.parse_module", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "rem", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "%" + }, + { + "name": "replace", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "round", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.compare", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.is_valid", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "set_diff", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "sort", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "sprintf", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "startswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "strings.replace_n", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "string" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "substring", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "sum", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.add_date", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.clock", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.date", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.diff", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.now_ns", + "decl": { + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_duration_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_ns", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_rfc3339_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.weekday", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "to_number", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "type": "string" + }, + { + "type": "boolean" + }, + { + "type": "null" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "trace", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "trim", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_left", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_prefix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_right", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_space", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_suffix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "type_name", + "decl": { + "args": [ + { + "of": [ + { + "type": "any" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "union", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "units.parse_bytes", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "upper", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode_object", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "dynamic": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode_object", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "uuid.rfc4122", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "walk", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "static": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "function" + }, + "relation": true + }, + { + "name": "yaml.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "yaml.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "yaml.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + } + ], + "wasm_abi_versions": [ + { + "version": 1, + "minor_version": 1 + } + ] +} diff --git a/vendor/github.com/open-policy-agent/opa/capabilities/v0.30.0.json b/vendor/github.com/open-policy-agent/opa/capabilities/v0.30.0.json new file mode 100644 index 00000000..7ff34daf --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/capabilities/v0.30.0.json @@ -0,0 +1,3458 @@ +{ + "builtins": [ + { + "name": "abs", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "all", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "and", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "\u0026" + }, + { + "name": "any", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "array.concat", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "array.slice", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "assign", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": ":=" + }, + { + "name": "base64.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "base64url.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode_no_pad", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "bits.and", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.lsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.negate", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.or", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.rsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.xor", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "cast_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "cast_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "cast_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "null" + }, + "type": "function" + } + }, + { + "name": "cast_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "cast_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "cast_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "ceil", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "concat", + "decl": { + "args": [ + { + "type": "string" + }, + { + "of": [ + { + "of": { + "type": "string" + }, + "type": "set" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "count", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "crypto.md5", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha1", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha256", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificate_request", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "div", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "/" + }, + { + "name": "endswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "eq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "=" + }, + { + "name": "equal", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "==" + }, + { + "name": "floor", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "format_int", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "glob.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "glob.quote_meta", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "graph.reachable", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "gt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e" + }, + { + "name": "gte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e=" + }, + { + "name": "hex.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "hex.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "http.send", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "indexof", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "intersection", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode_verify", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign_raw", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_number", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "json.patch", + "decl": { + "args": [ + { + "type": "any" + }, + { + "dynamic": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "static": [ + { + "key": "op", + "value": { + "type": "string" + } + }, + { + "key": "path", + "value": { + "type": "any" + } + } + ], + "type": "object" + }, + "type": "array" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "lower", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "lt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c" + }, + { + "name": "lte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c=" + }, + { + "name": "max", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "min", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "minus", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + "type": "function" + }, + "infix": "-" + }, + { + "name": "mul", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "*" + }, + { + "name": "neq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "!=" + }, + { + "name": "net.cidr_contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_contains_matches", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "static": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_expand", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_intersects", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_merge", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_overlap", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "numbers.range", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "object.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.get", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.union", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "opa.runtime", + "decl": { + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "or", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "|" + }, + { + "name": "plus", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "+" + }, + { + "name": "product", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "re_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.find_all_string_submatch_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.find_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.globs_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.template_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "rego.parse_module", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "rem", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "%" + }, + { + "name": "replace", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "round", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.compare", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.is_valid", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "set_diff", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "sort", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "sprintf", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "startswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "strings.replace_n", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "string" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "substring", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "sum", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.add_date", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.clock", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.date", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.diff", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.now_ns", + "decl": { + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_duration_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_ns", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_rfc3339_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.weekday", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "to_number", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "type": "string" + }, + { + "type": "boolean" + }, + { + "type": "null" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "trace", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "trim", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_left", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_prefix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_right", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_space", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_suffix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "type_name", + "decl": { + "args": [ + { + "of": [ + { + "type": "any" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "union", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "units.parse_bytes", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "upper", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode_object", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "dynamic": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode_object", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "uuid.rfc4122", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "walk", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "static": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "function" + }, + "relation": true + }, + { + "name": "yaml.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "yaml.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "yaml.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + } + ], + "wasm_abi_versions": [ + { + "version": 1, + "minor_version": 1 + } + ] +} diff --git a/vendor/github.com/open-policy-agent/opa/capabilities/v0.30.1.json b/vendor/github.com/open-policy-agent/opa/capabilities/v0.30.1.json new file mode 100644 index 00000000..7ff34daf --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/capabilities/v0.30.1.json @@ -0,0 +1,3458 @@ +{ + "builtins": [ + { + "name": "abs", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "all", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "and", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "\u0026" + }, + { + "name": "any", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "array.concat", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "array.slice", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "assign", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": ":=" + }, + { + "name": "base64.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "base64url.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode_no_pad", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "bits.and", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.lsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.negate", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.or", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.rsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.xor", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "cast_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "cast_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "cast_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "null" + }, + "type": "function" + } + }, + { + "name": "cast_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "cast_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "cast_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "ceil", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "concat", + "decl": { + "args": [ + { + "type": "string" + }, + { + "of": [ + { + "of": { + "type": "string" + }, + "type": "set" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "count", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "crypto.md5", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha1", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha256", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificate_request", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "div", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "/" + }, + { + "name": "endswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "eq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "=" + }, + { + "name": "equal", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "==" + }, + { + "name": "floor", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "format_int", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "glob.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "glob.quote_meta", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "graph.reachable", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "gt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e" + }, + { + "name": "gte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e=" + }, + { + "name": "hex.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "hex.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "http.send", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "indexof", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "intersection", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode_verify", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign_raw", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_number", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "json.patch", + "decl": { + "args": [ + { + "type": "any" + }, + { + "dynamic": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "static": [ + { + "key": "op", + "value": { + "type": "string" + } + }, + { + "key": "path", + "value": { + "type": "any" + } + } + ], + "type": "object" + }, + "type": "array" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "lower", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "lt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c" + }, + { + "name": "lte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c=" + }, + { + "name": "max", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "min", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "minus", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + "type": "function" + }, + "infix": "-" + }, + { + "name": "mul", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "*" + }, + { + "name": "neq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "!=" + }, + { + "name": "net.cidr_contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_contains_matches", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "static": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_expand", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_intersects", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_merge", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_overlap", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "numbers.range", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "object.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.get", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.union", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "opa.runtime", + "decl": { + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "or", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "|" + }, + { + "name": "plus", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "+" + }, + { + "name": "product", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "re_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.find_all_string_submatch_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.find_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.globs_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.template_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "rego.parse_module", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "rem", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "%" + }, + { + "name": "replace", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "round", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.compare", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.is_valid", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "set_diff", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "sort", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "sprintf", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "startswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "strings.replace_n", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "string" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "substring", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "sum", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.add_date", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.clock", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.date", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.diff", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.now_ns", + "decl": { + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_duration_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_ns", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_rfc3339_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.weekday", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "to_number", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "type": "string" + }, + { + "type": "boolean" + }, + { + "type": "null" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "trace", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "trim", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_left", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_prefix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_right", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_space", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_suffix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "type_name", + "decl": { + "args": [ + { + "of": [ + { + "type": "any" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "union", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "units.parse_bytes", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "upper", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode_object", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "dynamic": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode_object", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "uuid.rfc4122", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "walk", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "static": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "function" + }, + "relation": true + }, + { + "name": "yaml.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "yaml.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "yaml.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + } + ], + "wasm_abi_versions": [ + { + "version": 1, + "minor_version": 1 + } + ] +} diff --git a/vendor/github.com/open-policy-agent/opa/capabilities/v0.30.2.json b/vendor/github.com/open-policy-agent/opa/capabilities/v0.30.2.json new file mode 100644 index 00000000..7ff34daf --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/capabilities/v0.30.2.json @@ -0,0 +1,3458 @@ +{ + "builtins": [ + { + "name": "abs", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "all", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "and", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "\u0026" + }, + { + "name": "any", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "array.concat", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "array.slice", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "assign", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": ":=" + }, + { + "name": "base64.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "base64url.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode_no_pad", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "bits.and", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.lsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.negate", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.or", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.rsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.xor", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "cast_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "cast_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "cast_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "null" + }, + "type": "function" + } + }, + { + "name": "cast_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "cast_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "cast_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "ceil", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "concat", + "decl": { + "args": [ + { + "type": "string" + }, + { + "of": [ + { + "of": { + "type": "string" + }, + "type": "set" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "count", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "crypto.md5", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha1", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha256", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificate_request", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "div", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "/" + }, + { + "name": "endswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "eq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "=" + }, + { + "name": "equal", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "==" + }, + { + "name": "floor", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "format_int", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "glob.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "glob.quote_meta", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "graph.reachable", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "gt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e" + }, + { + "name": "gte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e=" + }, + { + "name": "hex.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "hex.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "http.send", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "indexof", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "intersection", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode_verify", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign_raw", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_number", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "json.patch", + "decl": { + "args": [ + { + "type": "any" + }, + { + "dynamic": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "static": [ + { + "key": "op", + "value": { + "type": "string" + } + }, + { + "key": "path", + "value": { + "type": "any" + } + } + ], + "type": "object" + }, + "type": "array" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "lower", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "lt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c" + }, + { + "name": "lte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c=" + }, + { + "name": "max", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "min", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "minus", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + "type": "function" + }, + "infix": "-" + }, + { + "name": "mul", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "*" + }, + { + "name": "neq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "!=" + }, + { + "name": "net.cidr_contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_contains_matches", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "static": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_expand", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_intersects", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_merge", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_overlap", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "numbers.range", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "object.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.get", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.union", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "opa.runtime", + "decl": { + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "or", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "|" + }, + { + "name": "plus", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "+" + }, + { + "name": "product", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "re_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.find_all_string_submatch_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.find_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.globs_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.template_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "rego.parse_module", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "rem", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "%" + }, + { + "name": "replace", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "round", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.compare", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.is_valid", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "set_diff", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "sort", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "sprintf", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "startswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "strings.replace_n", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "string" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "substring", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "sum", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.add_date", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.clock", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.date", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.diff", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.now_ns", + "decl": { + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_duration_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_ns", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_rfc3339_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.weekday", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "to_number", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "type": "string" + }, + { + "type": "boolean" + }, + { + "type": "null" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "trace", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "trim", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_left", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_prefix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_right", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_space", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_suffix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "type_name", + "decl": { + "args": [ + { + "of": [ + { + "type": "any" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "union", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "units.parse_bytes", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "upper", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode_object", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "dynamic": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode_object", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "uuid.rfc4122", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "walk", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "static": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "function" + }, + "relation": true + }, + { + "name": "yaml.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "yaml.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "yaml.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + } + ], + "wasm_abi_versions": [ + { + "version": 1, + "minor_version": 1 + } + ] +} diff --git a/vendor/github.com/open-policy-agent/opa/capabilities/v0.31.0.json b/vendor/github.com/open-policy-agent/opa/capabilities/v0.31.0.json new file mode 100644 index 00000000..7aec357e --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/capabilities/v0.31.0.json @@ -0,0 +1,3512 @@ +{ + "builtins": [ + { + "name": "abs", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "all", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "and", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "\u0026" + }, + { + "name": "any", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "array.concat", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "array.slice", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "assign", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": ":=" + }, + { + "name": "base64.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "base64url.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode_no_pad", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "bits.and", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.lsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.negate", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.or", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.rsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.xor", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "cast_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "cast_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "cast_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "null" + }, + "type": "function" + } + }, + { + "name": "cast_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "cast_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "cast_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "ceil", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "concat", + "decl": { + "args": [ + { + "type": "string" + }, + { + "of": [ + { + "of": { + "type": "string" + }, + "type": "set" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "count", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "crypto.md5", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha1", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha256", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_and_verify_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificate_request", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "div", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "/" + }, + { + "name": "endswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "eq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "=" + }, + { + "name": "equal", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "==" + }, + { + "name": "floor", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "format_int", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "glob.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "glob.quote_meta", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "graph.reachable", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "gt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e" + }, + { + "name": "gte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e=" + }, + { + "name": "hex.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "hex.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "http.send", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "indexof", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "intersection", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode_verify", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign_raw", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_number", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "json.patch", + "decl": { + "args": [ + { + "type": "any" + }, + { + "dynamic": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "static": [ + { + "key": "op", + "value": { + "type": "string" + } + }, + { + "key": "path", + "value": { + "type": "any" + } + } + ], + "type": "object" + }, + "type": "array" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "lower", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "lt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c" + }, + { + "name": "lte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c=" + }, + { + "name": "max", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "min", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "minus", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + "type": "function" + }, + "infix": "-" + }, + { + "name": "mul", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "*" + }, + { + "name": "neq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "!=" + }, + { + "name": "net.cidr_contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_contains_matches", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "static": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_expand", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_intersects", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_merge", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_overlap", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "numbers.range", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "object.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.get", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.union", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "opa.runtime", + "decl": { + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "or", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "|" + }, + { + "name": "plus", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "+" + }, + { + "name": "product", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "rand.intn", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "re_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.find_all_string_submatch_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.find_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.globs_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.template_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "rego.parse_module", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "rem", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "%" + }, + { + "name": "replace", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "round", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.compare", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.is_valid", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "set_diff", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "sort", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "sprintf", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "startswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "strings.replace_n", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "string" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "substring", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "sum", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.add_date", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.clock", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.date", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.diff", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.now_ns", + "decl": { + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_duration_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_ns", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_rfc3339_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.weekday", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "to_number", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "type": "string" + }, + { + "type": "boolean" + }, + { + "type": "null" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "trace", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "trim", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_left", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_prefix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_right", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_space", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_suffix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "type_name", + "decl": { + "args": [ + { + "of": [ + { + "type": "any" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "union", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "units.parse_bytes", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "upper", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode_object", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "dynamic": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode_object", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "uuid.rfc4122", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "walk", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "static": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "function" + }, + "relation": true + }, + { + "name": "yaml.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "yaml.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "yaml.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + } + ], + "wasm_abi_versions": [ + { + "version": 1, + "minor_version": 1 + }, + { + "version": 1, + "minor_version": 2 + } + ] +} diff --git a/vendor/github.com/open-policy-agent/opa/capabilities/v0.32.0.json b/vendor/github.com/open-policy-agent/opa/capabilities/v0.32.0.json new file mode 100644 index 00000000..7aec357e --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/capabilities/v0.32.0.json @@ -0,0 +1,3512 @@ +{ + "builtins": [ + { + "name": "abs", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "all", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "and", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "\u0026" + }, + { + "name": "any", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "array.concat", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "array.slice", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "assign", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": ":=" + }, + { + "name": "base64.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "base64url.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode_no_pad", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "bits.and", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.lsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.negate", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.or", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.rsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.xor", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "cast_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "cast_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "cast_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "null" + }, + "type": "function" + } + }, + { + "name": "cast_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "cast_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "cast_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "ceil", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "concat", + "decl": { + "args": [ + { + "type": "string" + }, + { + "of": [ + { + "of": { + "type": "string" + }, + "type": "set" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "count", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "crypto.md5", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha1", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha256", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_and_verify_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificate_request", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "div", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "/" + }, + { + "name": "endswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "eq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "=" + }, + { + "name": "equal", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "==" + }, + { + "name": "floor", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "format_int", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "glob.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "glob.quote_meta", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "graph.reachable", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "gt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e" + }, + { + "name": "gte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e=" + }, + { + "name": "hex.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "hex.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "http.send", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "indexof", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "intersection", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode_verify", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign_raw", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_number", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "json.patch", + "decl": { + "args": [ + { + "type": "any" + }, + { + "dynamic": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "static": [ + { + "key": "op", + "value": { + "type": "string" + } + }, + { + "key": "path", + "value": { + "type": "any" + } + } + ], + "type": "object" + }, + "type": "array" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "lower", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "lt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c" + }, + { + "name": "lte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c=" + }, + { + "name": "max", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "min", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "minus", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + "type": "function" + }, + "infix": "-" + }, + { + "name": "mul", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "*" + }, + { + "name": "neq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "!=" + }, + { + "name": "net.cidr_contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_contains_matches", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "static": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_expand", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_intersects", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_merge", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_overlap", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "numbers.range", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "object.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.get", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.union", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "opa.runtime", + "decl": { + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "or", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "|" + }, + { + "name": "plus", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "+" + }, + { + "name": "product", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "rand.intn", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "re_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.find_all_string_submatch_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.find_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.globs_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.template_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "rego.parse_module", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "rem", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "%" + }, + { + "name": "replace", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "round", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.compare", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.is_valid", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "set_diff", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "sort", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "sprintf", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "startswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "strings.replace_n", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "string" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "substring", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "sum", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.add_date", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.clock", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.date", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.diff", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.now_ns", + "decl": { + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_duration_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_ns", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_rfc3339_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.weekday", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "to_number", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "type": "string" + }, + { + "type": "boolean" + }, + { + "type": "null" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "trace", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "trim", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_left", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_prefix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_right", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_space", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_suffix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "type_name", + "decl": { + "args": [ + { + "of": [ + { + "type": "any" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "union", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "units.parse_bytes", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "upper", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode_object", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "dynamic": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode_object", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "uuid.rfc4122", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "walk", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "static": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "function" + }, + "relation": true + }, + { + "name": "yaml.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "yaml.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "yaml.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + } + ], + "wasm_abi_versions": [ + { + "version": 1, + "minor_version": 1 + }, + { + "version": 1, + "minor_version": 2 + } + ] +} diff --git a/vendor/github.com/open-policy-agent/opa/capabilities/v0.32.1.json b/vendor/github.com/open-policy-agent/opa/capabilities/v0.32.1.json new file mode 100644 index 00000000..7aec357e --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/capabilities/v0.32.1.json @@ -0,0 +1,3512 @@ +{ + "builtins": [ + { + "name": "abs", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "all", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "and", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "\u0026" + }, + { + "name": "any", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "array.concat", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "array.slice", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "assign", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": ":=" + }, + { + "name": "base64.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "base64url.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode_no_pad", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "bits.and", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.lsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.negate", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.or", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.rsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.xor", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "cast_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "cast_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "cast_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "null" + }, + "type": "function" + } + }, + { + "name": "cast_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "cast_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "cast_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "ceil", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "concat", + "decl": { + "args": [ + { + "type": "string" + }, + { + "of": [ + { + "of": { + "type": "string" + }, + "type": "set" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "count", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "crypto.md5", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha1", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha256", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_and_verify_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificate_request", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "div", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "/" + }, + { + "name": "endswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "eq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "=" + }, + { + "name": "equal", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "==" + }, + { + "name": "floor", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "format_int", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "glob.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "glob.quote_meta", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "graph.reachable", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "gt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e" + }, + { + "name": "gte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e=" + }, + { + "name": "hex.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "hex.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "http.send", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "indexof", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "intersection", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode_verify", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign_raw", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_number", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "json.patch", + "decl": { + "args": [ + { + "type": "any" + }, + { + "dynamic": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "static": [ + { + "key": "op", + "value": { + "type": "string" + } + }, + { + "key": "path", + "value": { + "type": "any" + } + } + ], + "type": "object" + }, + "type": "array" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "lower", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "lt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c" + }, + { + "name": "lte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c=" + }, + { + "name": "max", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "min", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "minus", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + "type": "function" + }, + "infix": "-" + }, + { + "name": "mul", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "*" + }, + { + "name": "neq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "!=" + }, + { + "name": "net.cidr_contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_contains_matches", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "static": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_expand", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_intersects", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_merge", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_overlap", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "numbers.range", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "object.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.get", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.union", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "opa.runtime", + "decl": { + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "or", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "|" + }, + { + "name": "plus", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "+" + }, + { + "name": "product", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "rand.intn", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "re_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.find_all_string_submatch_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.find_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.globs_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.template_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "rego.parse_module", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "rem", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "%" + }, + { + "name": "replace", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "round", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.compare", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.is_valid", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "set_diff", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "sort", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "sprintf", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "startswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "strings.replace_n", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "string" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "substring", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "sum", + "decl": { + "args": [ + { + "of": [ + { + "of": { + "type": "number" + }, + "type": "set" + }, + { + "dynamic": { + "type": "number" + }, + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.add_date", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.clock", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.date", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.diff", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.now_ns", + "decl": { + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_duration_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_ns", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_rfc3339_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.weekday", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "to_number", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "type": "string" + }, + { + "type": "boolean" + }, + { + "type": "null" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "trace", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "trim", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_left", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_prefix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_right", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_space", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_suffix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "type_name", + "decl": { + "args": [ + { + "of": [ + { + "type": "any" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "union", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "units.parse_bytes", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "upper", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode_object", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "dynamic": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode_object", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "uuid.rfc4122", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "walk", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "static": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "function" + }, + "relation": true + }, + { + "name": "yaml.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "yaml.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "yaml.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + } + ], + "wasm_abi_versions": [ + { + "version": 1, + "minor_version": 1 + }, + { + "version": 1, + "minor_version": 2 + } + ] +} diff --git a/vendor/github.com/open-policy-agent/opa/capabilities/v0.33.0.json b/vendor/github.com/open-policy-agent/opa/capabilities/v0.33.0.json new file mode 100644 index 00000000..cc391d8a --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/capabilities/v0.33.0.json @@ -0,0 +1,3534 @@ +{ + "builtins": [ + { + "name": "abs", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "all", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "and", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "\u0026" + }, + { + "name": "any", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "array.concat", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "array.slice", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "assign", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": ":=" + }, + { + "name": "base64.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "base64url.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode_no_pad", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "bits.and", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.lsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.negate", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.or", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.rsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.xor", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "cast_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "cast_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "cast_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "null" + }, + "type": "function" + } + }, + { + "name": "cast_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "cast_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "cast_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "ceil", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "concat", + "decl": { + "args": [ + { + "type": "string" + }, + { + "of": [ + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "count", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "crypto.md5", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha1", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha256", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_and_verify_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificate_request", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_rsa_private_key", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "div", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "/" + }, + { + "name": "endswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "eq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "=" + }, + { + "name": "equal", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "==" + }, + { + "name": "floor", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "format_int", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "glob.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "glob.quote_meta", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "graph.reachable", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "gt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e" + }, + { + "name": "gte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e=" + }, + { + "name": "hex.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "hex.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "http.send", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "indexof", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "intersection", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode_verify", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign_raw", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_number", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "json.patch", + "decl": { + "args": [ + { + "type": "any" + }, + { + "dynamic": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "static": [ + { + "key": "op", + "value": { + "type": "string" + } + }, + { + "key": "path", + "value": { + "type": "any" + } + } + ], + "type": "object" + }, + "type": "array" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "lower", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "lt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c" + }, + { + "name": "lte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c=" + }, + { + "name": "max", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "min", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "minus", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + "type": "function" + }, + "infix": "-" + }, + { + "name": "mul", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "*" + }, + { + "name": "neq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "!=" + }, + { + "name": "net.cidr_contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_contains_matches", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "static": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_expand", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_intersects", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_merge", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_overlap", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "numbers.range", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "object.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.get", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.union", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "opa.runtime", + "decl": { + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "or", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "|" + }, + { + "name": "plus", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "+" + }, + { + "name": "product", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + { + "of": { + "type": "number" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "rand.intn", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "re_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.find_all_string_submatch_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.find_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.globs_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.template_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "rego.parse_module", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "rem", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "%" + }, + { + "name": "replace", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "round", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.compare", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.is_valid", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "set_diff", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "sort", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "sprintf", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "startswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "strings.replace_n", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "string" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "substring", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "sum", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + { + "of": { + "type": "number" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.add_date", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.clock", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.date", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.diff", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.now_ns", + "decl": { + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_duration_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_ns", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_rfc3339_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.weekday", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "to_number", + "decl": { + "args": [ + { + "of": [ + { + "type": "null" + }, + { + "type": "boolean" + }, + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "trace", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "trim", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_left", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_prefix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_right", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_space", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_suffix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "type_name", + "decl": { + "args": [ + { + "of": [ + { + "type": "any" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "union", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "units.parse_bytes", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "upper", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode_object", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "dynamic": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode_object", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "uuid.rfc4122", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "walk", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "static": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "function" + }, + "relation": true + }, + { + "name": "yaml.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "yaml.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "yaml.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + } + ], + "wasm_abi_versions": [ + { + "version": 1, + "minor_version": 1 + }, + { + "version": 1, + "minor_version": 2 + } + ] +} diff --git a/vendor/github.com/open-policy-agent/opa/capabilities/v0.33.1.json b/vendor/github.com/open-policy-agent/opa/capabilities/v0.33.1.json new file mode 100644 index 00000000..cc391d8a --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/capabilities/v0.33.1.json @@ -0,0 +1,3534 @@ +{ + "builtins": [ + { + "name": "abs", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "all", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "and", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "\u0026" + }, + { + "name": "any", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "array.concat", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "array.slice", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "assign", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": ":=" + }, + { + "name": "base64.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "base64url.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode_no_pad", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "bits.and", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.lsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.negate", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.or", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.rsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.xor", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "cast_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "cast_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "cast_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "null" + }, + "type": "function" + } + }, + { + "name": "cast_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "cast_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "cast_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "ceil", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "concat", + "decl": { + "args": [ + { + "type": "string" + }, + { + "of": [ + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "count", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "crypto.md5", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha1", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha256", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_and_verify_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificate_request", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_rsa_private_key", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "div", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "/" + }, + { + "name": "endswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "eq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "=" + }, + { + "name": "equal", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "==" + }, + { + "name": "floor", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "format_int", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "glob.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "glob.quote_meta", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "graph.reachable", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "gt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e" + }, + { + "name": "gte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e=" + }, + { + "name": "hex.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "hex.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "http.send", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "indexof", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "intersection", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode_verify", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign_raw", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_number", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "json.patch", + "decl": { + "args": [ + { + "type": "any" + }, + { + "dynamic": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "static": [ + { + "key": "op", + "value": { + "type": "string" + } + }, + { + "key": "path", + "value": { + "type": "any" + } + } + ], + "type": "object" + }, + "type": "array" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "lower", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "lt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c" + }, + { + "name": "lte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c=" + }, + { + "name": "max", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "min", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "minus", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + "type": "function" + }, + "infix": "-" + }, + { + "name": "mul", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "*" + }, + { + "name": "neq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "!=" + }, + { + "name": "net.cidr_contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_contains_matches", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "static": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_expand", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_intersects", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_merge", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_overlap", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "numbers.range", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "object.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.get", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.union", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "opa.runtime", + "decl": { + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "or", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "|" + }, + { + "name": "plus", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "+" + }, + { + "name": "product", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + { + "of": { + "type": "number" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "rand.intn", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "re_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.find_all_string_submatch_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.find_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.globs_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.template_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "rego.parse_module", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "rem", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "%" + }, + { + "name": "replace", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "round", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.compare", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.is_valid", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "set_diff", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "sort", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "sprintf", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "startswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "strings.replace_n", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "string" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "substring", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "sum", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + { + "of": { + "type": "number" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.add_date", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.clock", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.date", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.diff", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.now_ns", + "decl": { + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_duration_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_ns", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_rfc3339_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.weekday", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "to_number", + "decl": { + "args": [ + { + "of": [ + { + "type": "null" + }, + { + "type": "boolean" + }, + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "trace", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "trim", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_left", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_prefix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_right", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_space", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_suffix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "type_name", + "decl": { + "args": [ + { + "of": [ + { + "type": "any" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "union", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "units.parse_bytes", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "upper", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode_object", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "dynamic": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode_object", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "uuid.rfc4122", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "walk", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "static": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "function" + }, + "relation": true + }, + { + "name": "yaml.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "yaml.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "yaml.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + } + ], + "wasm_abi_versions": [ + { + "version": 1, + "minor_version": 1 + }, + { + "version": 1, + "minor_version": 2 + } + ] +} diff --git a/vendor/github.com/open-policy-agent/opa/capabilities/v0.34.0.json b/vendor/github.com/open-policy-agent/opa/capabilities/v0.34.0.json new file mode 100644 index 00000000..91f6512f --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/capabilities/v0.34.0.json @@ -0,0 +1,3602 @@ +{ + "builtins": [ + { + "name": "abs", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "all", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "and", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "\u0026" + }, + { + "name": "any", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "array.concat", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "array.slice", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "assign", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": ":=" + }, + { + "name": "base64.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "base64url.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode_no_pad", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "bits.and", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.lsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.negate", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.or", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.rsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.xor", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "cast_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "cast_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "cast_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "null" + }, + "type": "function" + } + }, + { + "name": "cast_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "cast_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "cast_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "ceil", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "concat", + "decl": { + "args": [ + { + "type": "string" + }, + { + "of": [ + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "count", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "crypto.md5", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha1", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha256", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_and_verify_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificate_request", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_rsa_private_key", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "div", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "/" + }, + { + "name": "endswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "eq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "=" + }, + { + "name": "equal", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "==" + }, + { + "name": "floor", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "format_int", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "glob.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "glob.quote_meta", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "graph.reachable", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "gt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e" + }, + { + "name": "gte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e=" + }, + { + "name": "hex.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "hex.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "http.send", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "indexof", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "internal.member_2", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "in" + }, + { + "name": "internal.member_3", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "in" + }, + { + "name": "internal.print", + "decl": { + "args": [ + { + "dynamic": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "array" + } + ], + "type": "function" + } + }, + { + "name": "intersection", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode_verify", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign_raw", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_number", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "json.patch", + "decl": { + "args": [ + { + "type": "any" + }, + { + "dynamic": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "static": [ + { + "key": "op", + "value": { + "type": "string" + } + }, + { + "key": "path", + "value": { + "type": "any" + } + } + ], + "type": "object" + }, + "type": "array" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "lower", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "lt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c" + }, + { + "name": "lte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c=" + }, + { + "name": "max", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "min", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "minus", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + "type": "function" + }, + "infix": "-" + }, + { + "name": "mul", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "*" + }, + { + "name": "neq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "!=" + }, + { + "name": "net.cidr_contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_contains_matches", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "static": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_expand", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_intersects", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_merge", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_overlap", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "numbers.range", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "object.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.get", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.union", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "opa.runtime", + "decl": { + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "or", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "|" + }, + { + "name": "plus", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "+" + }, + { + "name": "print", + "decl": { + "type": "function", + "variadic": { + "type": "any" + } + } + }, + { + "name": "product", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + { + "of": { + "type": "number" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "rand.intn", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "re_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.find_all_string_submatch_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.find_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.globs_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.template_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "rego.parse_module", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "rem", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "%" + }, + { + "name": "replace", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "round", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.compare", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.is_valid", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "set_diff", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "sort", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "sprintf", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "startswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "strings.replace_n", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "string" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "substring", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "sum", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + { + "of": { + "type": "number" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.add_date", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.clock", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.date", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.diff", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.now_ns", + "decl": { + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_duration_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_ns", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_rfc3339_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.weekday", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "to_number", + "decl": { + "args": [ + { + "of": [ + { + "type": "null" + }, + { + "type": "boolean" + }, + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "trace", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "trim", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_left", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_prefix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_right", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_space", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_suffix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "type_name", + "decl": { + "args": [ + { + "of": [ + { + "type": "any" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "union", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "units.parse_bytes", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "upper", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode_object", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "dynamic": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode_object", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "uuid.rfc4122", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "walk", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "static": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "function" + }, + "relation": true + }, + { + "name": "yaml.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "yaml.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "yaml.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + } + ], + "future_keywords": [ + "in" + ], + "wasm_abi_versions": [ + { + "version": 1, + "minor_version": 1 + }, + { + "version": 1, + "minor_version": 2 + } + ] +} diff --git a/vendor/github.com/open-policy-agent/opa/capabilities/v0.34.1.json b/vendor/github.com/open-policy-agent/opa/capabilities/v0.34.1.json new file mode 100644 index 00000000..91f6512f --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/capabilities/v0.34.1.json @@ -0,0 +1,3602 @@ +{ + "builtins": [ + { + "name": "abs", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "all", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "and", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "\u0026" + }, + { + "name": "any", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "array.concat", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "array.slice", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "assign", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": ":=" + }, + { + "name": "base64.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "base64url.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode_no_pad", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "bits.and", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.lsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.negate", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.or", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.rsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.xor", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "cast_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "cast_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "cast_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "null" + }, + "type": "function" + } + }, + { + "name": "cast_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "cast_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "cast_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "ceil", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "concat", + "decl": { + "args": [ + { + "type": "string" + }, + { + "of": [ + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "count", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "crypto.md5", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha1", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha256", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_and_verify_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificate_request", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_rsa_private_key", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "div", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "/" + }, + { + "name": "endswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "eq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "=" + }, + { + "name": "equal", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "==" + }, + { + "name": "floor", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "format_int", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "glob.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "glob.quote_meta", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "graph.reachable", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "gt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e" + }, + { + "name": "gte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e=" + }, + { + "name": "hex.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "hex.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "http.send", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "indexof", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "internal.member_2", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "in" + }, + { + "name": "internal.member_3", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "in" + }, + { + "name": "internal.print", + "decl": { + "args": [ + { + "dynamic": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "array" + } + ], + "type": "function" + } + }, + { + "name": "intersection", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode_verify", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign_raw", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_number", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "json.patch", + "decl": { + "args": [ + { + "type": "any" + }, + { + "dynamic": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "static": [ + { + "key": "op", + "value": { + "type": "string" + } + }, + { + "key": "path", + "value": { + "type": "any" + } + } + ], + "type": "object" + }, + "type": "array" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "lower", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "lt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c" + }, + { + "name": "lte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c=" + }, + { + "name": "max", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "min", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "minus", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + "type": "function" + }, + "infix": "-" + }, + { + "name": "mul", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "*" + }, + { + "name": "neq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "!=" + }, + { + "name": "net.cidr_contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_contains_matches", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "static": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_expand", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_intersects", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_merge", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_overlap", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "numbers.range", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "object.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.get", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.union", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "opa.runtime", + "decl": { + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "or", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "|" + }, + { + "name": "plus", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "+" + }, + { + "name": "print", + "decl": { + "type": "function", + "variadic": { + "type": "any" + } + } + }, + { + "name": "product", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + { + "of": { + "type": "number" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "rand.intn", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "re_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.find_all_string_submatch_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.find_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.globs_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.template_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "rego.parse_module", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "rem", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "%" + }, + { + "name": "replace", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "round", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.compare", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.is_valid", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "set_diff", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "sort", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "sprintf", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "startswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "strings.replace_n", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "string" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "substring", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "sum", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + { + "of": { + "type": "number" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.add_date", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.clock", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.date", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.diff", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.now_ns", + "decl": { + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_duration_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_ns", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_rfc3339_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.weekday", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "to_number", + "decl": { + "args": [ + { + "of": [ + { + "type": "null" + }, + { + "type": "boolean" + }, + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "trace", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "trim", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_left", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_prefix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_right", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_space", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_suffix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "type_name", + "decl": { + "args": [ + { + "of": [ + { + "type": "any" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "union", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "units.parse_bytes", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "upper", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode_object", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "dynamic": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode_object", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "uuid.rfc4122", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "walk", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "static": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "function" + }, + "relation": true + }, + { + "name": "yaml.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "yaml.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "yaml.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + } + ], + "future_keywords": [ + "in" + ], + "wasm_abi_versions": [ + { + "version": 1, + "minor_version": 1 + }, + { + "version": 1, + "minor_version": 2 + } + ] +} diff --git a/vendor/github.com/open-policy-agent/opa/capabilities/v0.34.2.json b/vendor/github.com/open-policy-agent/opa/capabilities/v0.34.2.json new file mode 100644 index 00000000..91f6512f --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/capabilities/v0.34.2.json @@ -0,0 +1,3602 @@ +{ + "builtins": [ + { + "name": "abs", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "all", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "and", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "\u0026" + }, + { + "name": "any", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "array.concat", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "array.slice", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "assign", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": ":=" + }, + { + "name": "base64.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "base64url.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode_no_pad", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "bits.and", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.lsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.negate", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.or", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.rsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.xor", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "cast_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "cast_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "cast_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "null" + }, + "type": "function" + } + }, + { + "name": "cast_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "cast_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "cast_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "ceil", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "concat", + "decl": { + "args": [ + { + "type": "string" + }, + { + "of": [ + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "count", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "crypto.md5", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha1", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha256", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_and_verify_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificate_request", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_rsa_private_key", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "div", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "/" + }, + { + "name": "endswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "eq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "=" + }, + { + "name": "equal", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "==" + }, + { + "name": "floor", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "format_int", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "glob.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "glob.quote_meta", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "graph.reachable", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "gt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e" + }, + { + "name": "gte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e=" + }, + { + "name": "hex.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "hex.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "http.send", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "indexof", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "internal.member_2", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "in" + }, + { + "name": "internal.member_3", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "in" + }, + { + "name": "internal.print", + "decl": { + "args": [ + { + "dynamic": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "array" + } + ], + "type": "function" + } + }, + { + "name": "intersection", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode_verify", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign_raw", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_number", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "json.patch", + "decl": { + "args": [ + { + "type": "any" + }, + { + "dynamic": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "static": [ + { + "key": "op", + "value": { + "type": "string" + } + }, + { + "key": "path", + "value": { + "type": "any" + } + } + ], + "type": "object" + }, + "type": "array" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "lower", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "lt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c" + }, + { + "name": "lte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c=" + }, + { + "name": "max", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "min", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "minus", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + "type": "function" + }, + "infix": "-" + }, + { + "name": "mul", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "*" + }, + { + "name": "neq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "!=" + }, + { + "name": "net.cidr_contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_contains_matches", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "static": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_expand", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_intersects", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_merge", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_overlap", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "numbers.range", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "object.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.get", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.union", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "opa.runtime", + "decl": { + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "or", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "|" + }, + { + "name": "plus", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "+" + }, + { + "name": "print", + "decl": { + "type": "function", + "variadic": { + "type": "any" + } + } + }, + { + "name": "product", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + { + "of": { + "type": "number" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "rand.intn", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "re_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.find_all_string_submatch_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.find_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.globs_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.template_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "rego.parse_module", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "rem", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "%" + }, + { + "name": "replace", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "round", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.compare", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.is_valid", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "set_diff", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "sort", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "sprintf", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "startswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "strings.replace_n", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "string" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "substring", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "sum", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + { + "of": { + "type": "number" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.add_date", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.clock", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.date", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.diff", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.now_ns", + "decl": { + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_duration_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_ns", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_rfc3339_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.weekday", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "to_number", + "decl": { + "args": [ + { + "of": [ + { + "type": "null" + }, + { + "type": "boolean" + }, + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "trace", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "trim", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_left", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_prefix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_right", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_space", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_suffix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "type_name", + "decl": { + "args": [ + { + "of": [ + { + "type": "any" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "union", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "units.parse_bytes", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "upper", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode_object", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "dynamic": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode_object", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "uuid.rfc4122", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "walk", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "static": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "function" + }, + "relation": true + }, + { + "name": "yaml.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "yaml.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "yaml.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + } + ], + "future_keywords": [ + "in" + ], + "wasm_abi_versions": [ + { + "version": 1, + "minor_version": 1 + }, + { + "version": 1, + "minor_version": 2 + } + ] +} diff --git a/vendor/github.com/open-policy-agent/opa/capabilities/v0.35.0.json b/vendor/github.com/open-policy-agent/opa/capabilities/v0.35.0.json new file mode 100644 index 00000000..612db5ad --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/capabilities/v0.35.0.json @@ -0,0 +1,3619 @@ +{ + "builtins": [ + { + "name": "abs", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "all", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "and", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "\u0026" + }, + { + "name": "any", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "array.concat", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "array.slice", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "assign", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": ":=" + }, + { + "name": "base64.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "base64url.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode_no_pad", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "bits.and", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.lsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.negate", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.or", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.rsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.xor", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "cast_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "cast_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "cast_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "null" + }, + "type": "function" + } + }, + { + "name": "cast_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "cast_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "cast_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "ceil", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "concat", + "decl": { + "args": [ + { + "type": "string" + }, + { + "of": [ + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "count", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "crypto.md5", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha1", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha256", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_and_verify_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificate_request", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_rsa_private_key", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "div", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "/" + }, + { + "name": "endswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "eq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "=" + }, + { + "name": "equal", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "==" + }, + { + "name": "floor", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "format_int", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "glob.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "glob.quote_meta", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "graph.reachable", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "gt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e" + }, + { + "name": "gte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e=" + }, + { + "name": "hex.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "hex.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "http.send", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "indexof", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "internal.member_2", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "in" + }, + { + "name": "internal.member_3", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "in" + }, + { + "name": "internal.print", + "decl": { + "args": [ + { + "dynamic": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "array" + } + ], + "type": "function" + } + }, + { + "name": "intersection", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode_verify", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign_raw", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_number", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "json.patch", + "decl": { + "args": [ + { + "type": "any" + }, + { + "dynamic": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "static": [ + { + "key": "op", + "value": { + "type": "string" + } + }, + { + "key": "path", + "value": { + "type": "any" + } + } + ], + "type": "object" + }, + "type": "array" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "lower", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "lt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c" + }, + { + "name": "lte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c=" + }, + { + "name": "max", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "min", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "minus", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + "type": "function" + }, + "infix": "-" + }, + { + "name": "mul", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "*" + }, + { + "name": "neq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "!=" + }, + { + "name": "net.cidr_contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_contains_matches", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "static": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_expand", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_intersects", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_merge", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_overlap", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.lookup_ip_addr", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "numbers.range", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "object.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.get", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.union", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "opa.runtime", + "decl": { + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "or", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "|" + }, + { + "name": "plus", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "+" + }, + { + "name": "print", + "decl": { + "type": "function", + "variadic": { + "type": "any" + } + } + }, + { + "name": "product", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + { + "of": { + "type": "number" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "rand.intn", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "re_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.find_all_string_submatch_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.find_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.globs_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.template_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "rego.parse_module", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "rem", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "%" + }, + { + "name": "replace", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "round", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.compare", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.is_valid", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "set_diff", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "sort", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "sprintf", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "startswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "strings.replace_n", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "string" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "substring", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "sum", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + { + "of": { + "type": "number" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.add_date", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.clock", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.date", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.diff", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.now_ns", + "decl": { + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_duration_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_ns", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_rfc3339_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.weekday", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "to_number", + "decl": { + "args": [ + { + "of": [ + { + "type": "null" + }, + { + "type": "boolean" + }, + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "trace", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "trim", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_left", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_prefix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_right", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_space", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_suffix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "type_name", + "decl": { + "args": [ + { + "of": [ + { + "type": "any" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "union", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "units.parse_bytes", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "upper", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode_object", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "dynamic": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode_object", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "uuid.rfc4122", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "walk", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "static": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "function" + }, + "relation": true + }, + { + "name": "yaml.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "yaml.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "yaml.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + } + ], + "future_keywords": [ + "in" + ], + "wasm_abi_versions": [ + { + "version": 1, + "minor_version": 1 + }, + { + "version": 1, + "minor_version": 2 + } + ] +} diff --git a/vendor/github.com/open-policy-agent/opa/capabilities/v0.36.0.json b/vendor/github.com/open-policy-agent/opa/capabilities/v0.36.0.json new file mode 100644 index 00000000..bedf0716 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/capabilities/v0.36.0.json @@ -0,0 +1,3721 @@ +{ + "builtins": [ + { + "name": "abs", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "all", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "and", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "\u0026" + }, + { + "name": "any", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "array.concat", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "array.reverse", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "array.slice", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "assign", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": ":=" + }, + { + "name": "base64.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "base64url.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode_no_pad", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "bits.and", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.lsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.negate", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.or", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.rsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.xor", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "cast_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "cast_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "cast_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "null" + }, + "type": "function" + } + }, + { + "name": "cast_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "cast_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "cast_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "ceil", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "concat", + "decl": { + "args": [ + { + "type": "string" + }, + { + "of": [ + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "count", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "crypto.hmac.md5", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.hmac.sha1", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.hmac.sha256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.hmac.sha512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.md5", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha1", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha256", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_and_verify_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificate_request", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_rsa_private_key", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "div", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "/" + }, + { + "name": "endswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "eq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "=" + }, + { + "name": "equal", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "==" + }, + { + "name": "floor", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "format_int", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "glob.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "glob.quote_meta", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "graph.reachable", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "gt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e" + }, + { + "name": "gte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e=" + }, + { + "name": "hex.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "hex.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "http.send", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "indexof", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "internal.member_2", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "in" + }, + { + "name": "internal.member_3", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "in" + }, + { + "name": "internal.print", + "decl": { + "args": [ + { + "dynamic": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "array" + } + ], + "type": "function" + } + }, + { + "name": "intersection", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode_verify", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign_raw", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_number", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "json.patch", + "decl": { + "args": [ + { + "type": "any" + }, + { + "dynamic": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "static": [ + { + "key": "op", + "value": { + "type": "string" + } + }, + { + "key": "path", + "value": { + "type": "any" + } + } + ], + "type": "object" + }, + "type": "array" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "lower", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "lt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c" + }, + { + "name": "lte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c=" + }, + { + "name": "max", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "min", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "minus", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + "type": "function" + }, + "infix": "-" + }, + { + "name": "mul", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "*" + }, + { + "name": "neq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "!=" + }, + { + "name": "net.cidr_contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_contains_matches", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "static": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_expand", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_intersects", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_merge", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_overlap", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.lookup_ip_addr", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "numbers.range", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "object.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.get", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.union", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "opa.runtime", + "decl": { + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "or", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "|" + }, + { + "name": "plus", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "+" + }, + { + "name": "print", + "decl": { + "type": "function", + "variadic": { + "type": "any" + } + } + }, + { + "name": "product", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + { + "of": { + "type": "number" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "rand.intn", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "re_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.find_all_string_submatch_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.find_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.globs_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.template_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "rego.parse_module", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "rem", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "%" + }, + { + "name": "replace", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "round", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.compare", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.is_valid", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "set_diff", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "sort", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "sprintf", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "startswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "strings.replace_n", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "string" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "strings.reverse", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "substring", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "sum", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + { + "of": { + "type": "number" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.add_date", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.clock", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.date", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.diff", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.now_ns", + "decl": { + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_duration_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_ns", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_rfc3339_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.weekday", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "to_number", + "decl": { + "args": [ + { + "of": [ + { + "type": "null" + }, + { + "type": "boolean" + }, + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "trace", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "trim", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_left", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_prefix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_right", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_space", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_suffix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "type_name", + "decl": { + "args": [ + { + "of": [ + { + "type": "any" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "union", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "units.parse_bytes", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "upper", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode_object", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "dynamic": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode_object", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "uuid.rfc4122", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "walk", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "static": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "function" + }, + "relation": true + }, + { + "name": "yaml.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "yaml.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "yaml.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + } + ], + "future_keywords": [ + "in" + ], + "wasm_abi_versions": [ + { + "version": 1, + "minor_version": 1 + }, + { + "version": 1, + "minor_version": 2 + } + ] +} diff --git a/vendor/github.com/open-policy-agent/opa/capabilities/v0.36.1.json b/vendor/github.com/open-policy-agent/opa/capabilities/v0.36.1.json new file mode 100644 index 00000000..bedf0716 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/capabilities/v0.36.1.json @@ -0,0 +1,3721 @@ +{ + "builtins": [ + { + "name": "abs", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "all", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "and", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "\u0026" + }, + { + "name": "any", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "array.concat", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "array.reverse", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "array.slice", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "assign", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": ":=" + }, + { + "name": "base64.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "base64url.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode_no_pad", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "bits.and", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.lsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.negate", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.or", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.rsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.xor", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "cast_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "cast_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "cast_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "null" + }, + "type": "function" + } + }, + { + "name": "cast_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "cast_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "cast_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "ceil", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "concat", + "decl": { + "args": [ + { + "type": "string" + }, + { + "of": [ + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "count", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "crypto.hmac.md5", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.hmac.sha1", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.hmac.sha256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.hmac.sha512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.md5", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha1", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha256", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_and_verify_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificate_request", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_rsa_private_key", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "div", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "/" + }, + { + "name": "endswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "eq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "=" + }, + { + "name": "equal", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "==" + }, + { + "name": "floor", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "format_int", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "glob.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "glob.quote_meta", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "graph.reachable", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "gt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e" + }, + { + "name": "gte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e=" + }, + { + "name": "hex.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "hex.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "http.send", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "indexof", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "internal.member_2", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "in" + }, + { + "name": "internal.member_3", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "in" + }, + { + "name": "internal.print", + "decl": { + "args": [ + { + "dynamic": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "array" + } + ], + "type": "function" + } + }, + { + "name": "intersection", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode_verify", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign_raw", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_number", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "json.patch", + "decl": { + "args": [ + { + "type": "any" + }, + { + "dynamic": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "static": [ + { + "key": "op", + "value": { + "type": "string" + } + }, + { + "key": "path", + "value": { + "type": "any" + } + } + ], + "type": "object" + }, + "type": "array" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "lower", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "lt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c" + }, + { + "name": "lte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c=" + }, + { + "name": "max", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "min", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "minus", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + "type": "function" + }, + "infix": "-" + }, + { + "name": "mul", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "*" + }, + { + "name": "neq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "!=" + }, + { + "name": "net.cidr_contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_contains_matches", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "static": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_expand", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_intersects", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_merge", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_overlap", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.lookup_ip_addr", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "numbers.range", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "object.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.get", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.union", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "opa.runtime", + "decl": { + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "or", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "|" + }, + { + "name": "plus", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "+" + }, + { + "name": "print", + "decl": { + "type": "function", + "variadic": { + "type": "any" + } + } + }, + { + "name": "product", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + { + "of": { + "type": "number" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "rand.intn", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "re_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.find_all_string_submatch_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.find_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.globs_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.template_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "rego.parse_module", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "rem", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "%" + }, + { + "name": "replace", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "round", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.compare", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.is_valid", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "set_diff", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "sort", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "sprintf", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "startswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "strings.replace_n", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "string" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "strings.reverse", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "substring", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "sum", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + { + "of": { + "type": "number" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.add_date", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.clock", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.date", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.diff", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.now_ns", + "decl": { + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_duration_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_ns", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_rfc3339_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.weekday", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "to_number", + "decl": { + "args": [ + { + "of": [ + { + "type": "null" + }, + { + "type": "boolean" + }, + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "trace", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "trim", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_left", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_prefix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_right", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_space", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_suffix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "type_name", + "decl": { + "args": [ + { + "of": [ + { + "type": "any" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "union", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "units.parse_bytes", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "upper", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode_object", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "dynamic": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode_object", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "uuid.rfc4122", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "walk", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "static": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "function" + }, + "relation": true + }, + { + "name": "yaml.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "yaml.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "yaml.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + } + ], + "future_keywords": [ + "in" + ], + "wasm_abi_versions": [ + { + "version": 1, + "minor_version": 1 + }, + { + "version": 1, + "minor_version": 2 + } + ] +} diff --git a/vendor/github.com/open-policy-agent/opa/capabilities/v0.37.0.json b/vendor/github.com/open-policy-agent/opa/capabilities/v0.37.0.json new file mode 100644 index 00000000..e8736d8d --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/capabilities/v0.37.0.json @@ -0,0 +1,3825 @@ +{ + "builtins": [ + { + "name": "abs", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "all", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "and", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "\u0026" + }, + { + "name": "any", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "array.concat", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "array.reverse", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "array.slice", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "assign", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": ":=" + }, + { + "name": "base64.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "base64url.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode_no_pad", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "bits.and", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.lsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.negate", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.or", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.rsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.xor", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "cast_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "cast_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "cast_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "null" + }, + "type": "function" + } + }, + { + "name": "cast_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "cast_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "cast_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "ceil", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "concat", + "decl": { + "args": [ + { + "type": "string" + }, + { + "of": [ + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "count", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "crypto.hmac.md5", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.hmac.sha1", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.hmac.sha256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.hmac.sha512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.md5", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha1", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha256", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_and_verify_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificate_request", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_rsa_private_key", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "div", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "/" + }, + { + "name": "endswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "eq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "=" + }, + { + "name": "equal", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "==" + }, + { + "name": "floor", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "format_int", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "glob.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "glob.quote_meta", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "graph.reachable", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "graph.reachable_paths", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "gt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e" + }, + { + "name": "gte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e=" + }, + { + "name": "hex.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "hex.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "http.send", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "indexof", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "indexof_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "internal.member_2", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "in" + }, + { + "name": "internal.member_3", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "in" + }, + { + "name": "internal.print", + "decl": { + "args": [ + { + "dynamic": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "array" + } + ], + "type": "function" + } + }, + { + "name": "intersection", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode_verify", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign_raw", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_number", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "json.patch", + "decl": { + "args": [ + { + "type": "any" + }, + { + "dynamic": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "static": [ + { + "key": "op", + "value": { + "type": "string" + } + }, + { + "key": "path", + "value": { + "type": "any" + } + } + ], + "type": "object" + }, + "type": "array" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "lower", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "lt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c" + }, + { + "name": "lte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c=" + }, + { + "name": "max", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "min", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "minus", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + "type": "function" + }, + "infix": "-" + }, + { + "name": "mul", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "*" + }, + { + "name": "neq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "!=" + }, + { + "name": "net.cidr_contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_contains_matches", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "static": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_expand", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_intersects", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_merge", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_overlap", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.lookup_ip_addr", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "numbers.range", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "object.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.get", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.union", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.union_n", + "decl": { + "args": [ + { + "dynamic": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "opa.runtime", + "decl": { + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "or", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "|" + }, + { + "name": "plus", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "+" + }, + { + "name": "print", + "decl": { + "type": "function", + "variadic": { + "type": "any" + } + } + }, + { + "name": "product", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + { + "of": { + "type": "number" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "rand.intn", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "re_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.find_all_string_submatch_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.find_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.globs_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.template_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "rego.parse_module", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "rem", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "%" + }, + { + "name": "replace", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "round", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.compare", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.is_valid", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "set_diff", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "sort", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "sprintf", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "startswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "strings.replace_n", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "string" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "strings.reverse", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "substring", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "sum", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + { + "of": { + "type": "number" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.add_date", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.clock", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.date", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.diff", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.now_ns", + "decl": { + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_duration_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_ns", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_rfc3339_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.weekday", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "to_number", + "decl": { + "args": [ + { + "of": [ + { + "type": "null" + }, + { + "type": "boolean" + }, + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "trace", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "trim", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_left", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_prefix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_right", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_space", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_suffix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "type_name", + "decl": { + "args": [ + { + "of": [ + { + "type": "any" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "union", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "units.parse_bytes", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "upper", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode_object", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "dynamic": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode_object", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "uuid.rfc4122", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "walk", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "static": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "function" + }, + "relation": true + }, + { + "name": "yaml.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "yaml.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "yaml.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + } + ], + "future_keywords": [ + "in" + ], + "wasm_abi_versions": [ + { + "version": 1, + "minor_version": 1 + }, + { + "version": 1, + "minor_version": 2 + } + ] +} diff --git a/vendor/github.com/open-policy-agent/opa/capabilities/v0.37.1.json b/vendor/github.com/open-policy-agent/opa/capabilities/v0.37.1.json new file mode 100644 index 00000000..e8736d8d --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/capabilities/v0.37.1.json @@ -0,0 +1,3825 @@ +{ + "builtins": [ + { + "name": "abs", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "all", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "and", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "\u0026" + }, + { + "name": "any", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "array.concat", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "array.reverse", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "array.slice", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "assign", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": ":=" + }, + { + "name": "base64.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "base64url.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode_no_pad", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "bits.and", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.lsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.negate", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.or", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.rsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.xor", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "cast_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "cast_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "cast_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "null" + }, + "type": "function" + } + }, + { + "name": "cast_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "cast_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "cast_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "ceil", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "concat", + "decl": { + "args": [ + { + "type": "string" + }, + { + "of": [ + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "count", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "crypto.hmac.md5", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.hmac.sha1", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.hmac.sha256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.hmac.sha512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.md5", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha1", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha256", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_and_verify_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificate_request", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_rsa_private_key", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "div", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "/" + }, + { + "name": "endswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "eq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "=" + }, + { + "name": "equal", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "==" + }, + { + "name": "floor", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "format_int", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "glob.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "glob.quote_meta", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "graph.reachable", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "graph.reachable_paths", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "gt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e" + }, + { + "name": "gte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e=" + }, + { + "name": "hex.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "hex.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "http.send", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "indexof", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "indexof_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "internal.member_2", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "in" + }, + { + "name": "internal.member_3", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "in" + }, + { + "name": "internal.print", + "decl": { + "args": [ + { + "dynamic": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "array" + } + ], + "type": "function" + } + }, + { + "name": "intersection", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode_verify", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign_raw", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_number", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "json.patch", + "decl": { + "args": [ + { + "type": "any" + }, + { + "dynamic": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "static": [ + { + "key": "op", + "value": { + "type": "string" + } + }, + { + "key": "path", + "value": { + "type": "any" + } + } + ], + "type": "object" + }, + "type": "array" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "lower", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "lt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c" + }, + { + "name": "lte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c=" + }, + { + "name": "max", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "min", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "minus", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + "type": "function" + }, + "infix": "-" + }, + { + "name": "mul", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "*" + }, + { + "name": "neq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "!=" + }, + { + "name": "net.cidr_contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_contains_matches", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "static": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_expand", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_intersects", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_merge", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_overlap", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.lookup_ip_addr", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "numbers.range", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "object.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.get", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.union", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.union_n", + "decl": { + "args": [ + { + "dynamic": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "opa.runtime", + "decl": { + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "or", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "|" + }, + { + "name": "plus", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "+" + }, + { + "name": "print", + "decl": { + "type": "function", + "variadic": { + "type": "any" + } + } + }, + { + "name": "product", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + { + "of": { + "type": "number" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "rand.intn", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "re_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.find_all_string_submatch_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.find_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.globs_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.template_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "rego.parse_module", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "rem", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "%" + }, + { + "name": "replace", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "round", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.compare", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.is_valid", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "set_diff", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "sort", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "sprintf", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "startswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "strings.replace_n", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "string" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "strings.reverse", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "substring", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "sum", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + { + "of": { + "type": "number" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.add_date", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.clock", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.date", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.diff", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.now_ns", + "decl": { + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_duration_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_ns", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_rfc3339_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.weekday", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "to_number", + "decl": { + "args": [ + { + "of": [ + { + "type": "null" + }, + { + "type": "boolean" + }, + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "trace", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "trim", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_left", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_prefix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_right", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_space", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_suffix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "type_name", + "decl": { + "args": [ + { + "of": [ + { + "type": "any" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "union", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "units.parse_bytes", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "upper", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode_object", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "dynamic": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode_object", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "uuid.rfc4122", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "walk", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "static": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "function" + }, + "relation": true + }, + { + "name": "yaml.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "yaml.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "yaml.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + } + ], + "future_keywords": [ + "in" + ], + "wasm_abi_versions": [ + { + "version": 1, + "minor_version": 1 + }, + { + "version": 1, + "minor_version": 2 + } + ] +} diff --git a/vendor/github.com/open-policy-agent/opa/capabilities/v0.37.2.json b/vendor/github.com/open-policy-agent/opa/capabilities/v0.37.2.json new file mode 100644 index 00000000..e8736d8d --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/capabilities/v0.37.2.json @@ -0,0 +1,3825 @@ +{ + "builtins": [ + { + "name": "abs", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "all", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "and", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "\u0026" + }, + { + "name": "any", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "array.concat", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "array.reverse", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "array.slice", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "assign", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": ":=" + }, + { + "name": "base64.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "base64url.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode_no_pad", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "bits.and", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.lsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.negate", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.or", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.rsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.xor", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "cast_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "cast_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "cast_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "null" + }, + "type": "function" + } + }, + { + "name": "cast_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "cast_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "cast_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "ceil", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "concat", + "decl": { + "args": [ + { + "type": "string" + }, + { + "of": [ + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "count", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "crypto.hmac.md5", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.hmac.sha1", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.hmac.sha256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.hmac.sha512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.md5", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha1", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha256", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_and_verify_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificate_request", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_rsa_private_key", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "div", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "/" + }, + { + "name": "endswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "eq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "=" + }, + { + "name": "equal", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "==" + }, + { + "name": "floor", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "format_int", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "glob.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "glob.quote_meta", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "graph.reachable", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "graph.reachable_paths", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "gt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e" + }, + { + "name": "gte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e=" + }, + { + "name": "hex.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "hex.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "http.send", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "indexof", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "indexof_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "internal.member_2", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "in" + }, + { + "name": "internal.member_3", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "in" + }, + { + "name": "internal.print", + "decl": { + "args": [ + { + "dynamic": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "array" + } + ], + "type": "function" + } + }, + { + "name": "intersection", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode_verify", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign_raw", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_number", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "json.patch", + "decl": { + "args": [ + { + "type": "any" + }, + { + "dynamic": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "static": [ + { + "key": "op", + "value": { + "type": "string" + } + }, + { + "key": "path", + "value": { + "type": "any" + } + } + ], + "type": "object" + }, + "type": "array" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "lower", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "lt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c" + }, + { + "name": "lte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c=" + }, + { + "name": "max", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "min", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "minus", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + "type": "function" + }, + "infix": "-" + }, + { + "name": "mul", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "*" + }, + { + "name": "neq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "!=" + }, + { + "name": "net.cidr_contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_contains_matches", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "static": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_expand", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_intersects", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_merge", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_overlap", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.lookup_ip_addr", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "numbers.range", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "object.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.get", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.union", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.union_n", + "decl": { + "args": [ + { + "dynamic": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "opa.runtime", + "decl": { + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "or", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "|" + }, + { + "name": "plus", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "+" + }, + { + "name": "print", + "decl": { + "type": "function", + "variadic": { + "type": "any" + } + } + }, + { + "name": "product", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + { + "of": { + "type": "number" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "rand.intn", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "re_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.find_all_string_submatch_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.find_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.globs_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.template_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "rego.parse_module", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "rem", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "%" + }, + { + "name": "replace", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "round", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.compare", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.is_valid", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "set_diff", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "sort", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "sprintf", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "startswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "strings.replace_n", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "string" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "strings.reverse", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "substring", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "sum", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + { + "of": { + "type": "number" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.add_date", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.clock", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.date", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.diff", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.now_ns", + "decl": { + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_duration_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_ns", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_rfc3339_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.weekday", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "to_number", + "decl": { + "args": [ + { + "of": [ + { + "type": "null" + }, + { + "type": "boolean" + }, + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "trace", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "trim", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_left", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_prefix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_right", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_space", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_suffix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "type_name", + "decl": { + "args": [ + { + "of": [ + { + "type": "any" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "union", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "units.parse_bytes", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "upper", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode_object", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "dynamic": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode_object", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "uuid.rfc4122", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "walk", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "static": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "function" + }, + "relation": true + }, + { + "name": "yaml.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "yaml.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "yaml.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + } + ], + "future_keywords": [ + "in" + ], + "wasm_abi_versions": [ + { + "version": 1, + "minor_version": 1 + }, + { + "version": 1, + "minor_version": 2 + } + ] +} diff --git a/vendor/github.com/open-policy-agent/opa/capabilities/v0.38.0.json b/vendor/github.com/open-policy-agent/opa/capabilities/v0.38.0.json new file mode 100644 index 00000000..d5edb2da --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/capabilities/v0.38.0.json @@ -0,0 +1,3826 @@ +{ + "builtins": [ + { + "name": "abs", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "all", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "and", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "\u0026" + }, + { + "name": "any", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "array.concat", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "array.reverse", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "array.slice", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "assign", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": ":=" + }, + { + "name": "base64.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "base64url.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode_no_pad", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "bits.and", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.lsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.negate", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.or", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.rsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.xor", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "cast_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "cast_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "cast_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "null" + }, + "type": "function" + } + }, + { + "name": "cast_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "cast_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "cast_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "ceil", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "concat", + "decl": { + "args": [ + { + "type": "string" + }, + { + "of": [ + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "count", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "crypto.hmac.md5", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.hmac.sha1", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.hmac.sha256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.hmac.sha512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.md5", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha1", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha256", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_and_verify_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificate_request", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_rsa_private_key", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "div", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "/" + }, + { + "name": "endswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "eq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "=" + }, + { + "name": "equal", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "==" + }, + { + "name": "floor", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "format_int", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "glob.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "glob.quote_meta", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "graph.reachable", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "graph.reachable_paths", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "gt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e" + }, + { + "name": "gte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e=" + }, + { + "name": "hex.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "hex.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "http.send", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "indexof", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "indexof_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "internal.member_2", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "in" + }, + { + "name": "internal.member_3", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "in" + }, + { + "name": "internal.print", + "decl": { + "args": [ + { + "dynamic": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "array" + } + ], + "type": "function" + } + }, + { + "name": "intersection", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode_verify", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign_raw", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_number", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "json.patch", + "decl": { + "args": [ + { + "type": "any" + }, + { + "dynamic": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "static": [ + { + "key": "op", + "value": { + "type": "string" + } + }, + { + "key": "path", + "value": { + "type": "any" + } + } + ], + "type": "object" + }, + "type": "array" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "lower", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "lt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c" + }, + { + "name": "lte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c=" + }, + { + "name": "max", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "min", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "minus", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + "type": "function" + }, + "infix": "-" + }, + { + "name": "mul", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "*" + }, + { + "name": "neq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "!=" + }, + { + "name": "net.cidr_contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_contains_matches", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "static": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_expand", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_intersects", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_merge", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_overlap", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.lookup_ip_addr", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "numbers.range", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "object.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.get", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.union", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.union_n", + "decl": { + "args": [ + { + "dynamic": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "opa.runtime", + "decl": { + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "or", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "|" + }, + { + "name": "plus", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "+" + }, + { + "name": "print", + "decl": { + "type": "function", + "variadic": { + "type": "any" + } + } + }, + { + "name": "product", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + { + "of": { + "type": "number" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "rand.intn", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "re_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.find_all_string_submatch_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.find_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.globs_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.template_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "rego.parse_module", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "rem", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "%" + }, + { + "name": "replace", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "round", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.compare", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.is_valid", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "set_diff", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "sort", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "sprintf", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "startswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "strings.replace_n", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "string" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "strings.reverse", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "substring", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "sum", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + { + "of": { + "type": "number" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.add_date", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.clock", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.date", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.diff", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.now_ns", + "decl": { + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_duration_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_ns", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_rfc3339_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.weekday", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "to_number", + "decl": { + "args": [ + { + "of": [ + { + "type": "null" + }, + { + "type": "boolean" + }, + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "trace", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "trim", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_left", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_prefix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_right", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_space", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_suffix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "type_name", + "decl": { + "args": [ + { + "of": [ + { + "type": "any" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "union", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "units.parse_bytes", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "upper", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode_object", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "dynamic": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode_object", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "uuid.rfc4122", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "walk", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "static": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "function" + }, + "relation": true + }, + { + "name": "yaml.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "yaml.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "yaml.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + } + ], + "future_keywords": [ + "every", + "in" + ], + "wasm_abi_versions": [ + { + "version": 1, + "minor_version": 1 + }, + { + "version": 1, + "minor_version": 2 + } + ] +} diff --git a/vendor/github.com/open-policy-agent/opa/capabilities/v0.38.1.json b/vendor/github.com/open-policy-agent/opa/capabilities/v0.38.1.json new file mode 100644 index 00000000..d5edb2da --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/capabilities/v0.38.1.json @@ -0,0 +1,3826 @@ +{ + "builtins": [ + { + "name": "abs", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "all", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "and", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "\u0026" + }, + { + "name": "any", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "array.concat", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "array.reverse", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "array.slice", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "assign", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": ":=" + }, + { + "name": "base64.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "base64url.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode_no_pad", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "bits.and", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.lsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.negate", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.or", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.rsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.xor", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "cast_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "cast_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "cast_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "null" + }, + "type": "function" + } + }, + { + "name": "cast_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "cast_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "cast_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "ceil", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "concat", + "decl": { + "args": [ + { + "type": "string" + }, + { + "of": [ + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "count", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "crypto.hmac.md5", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.hmac.sha1", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.hmac.sha256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.hmac.sha512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.md5", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha1", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha256", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_and_verify_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificate_request", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_rsa_private_key", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "div", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "/" + }, + { + "name": "endswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "eq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "=" + }, + { + "name": "equal", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "==" + }, + { + "name": "floor", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "format_int", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "glob.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "glob.quote_meta", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "graph.reachable", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "graph.reachable_paths", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "gt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e" + }, + { + "name": "gte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e=" + }, + { + "name": "hex.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "hex.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "http.send", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "indexof", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "indexof_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "internal.member_2", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "in" + }, + { + "name": "internal.member_3", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "in" + }, + { + "name": "internal.print", + "decl": { + "args": [ + { + "dynamic": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "array" + } + ], + "type": "function" + } + }, + { + "name": "intersection", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode_verify", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign_raw", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_number", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "json.patch", + "decl": { + "args": [ + { + "type": "any" + }, + { + "dynamic": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "static": [ + { + "key": "op", + "value": { + "type": "string" + } + }, + { + "key": "path", + "value": { + "type": "any" + } + } + ], + "type": "object" + }, + "type": "array" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "lower", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "lt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c" + }, + { + "name": "lte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c=" + }, + { + "name": "max", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "min", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "minus", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + "type": "function" + }, + "infix": "-" + }, + { + "name": "mul", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "*" + }, + { + "name": "neq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "!=" + }, + { + "name": "net.cidr_contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_contains_matches", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "static": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_expand", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_intersects", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_merge", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_overlap", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.lookup_ip_addr", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "numbers.range", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "object.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.get", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.union", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.union_n", + "decl": { + "args": [ + { + "dynamic": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "opa.runtime", + "decl": { + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "or", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "|" + }, + { + "name": "plus", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "+" + }, + { + "name": "print", + "decl": { + "type": "function", + "variadic": { + "type": "any" + } + } + }, + { + "name": "product", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + { + "of": { + "type": "number" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "rand.intn", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "re_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.find_all_string_submatch_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.find_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.globs_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.template_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "rego.parse_module", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "rem", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "%" + }, + { + "name": "replace", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "round", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.compare", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.is_valid", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "set_diff", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "sort", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "sprintf", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "startswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "strings.replace_n", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "string" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "strings.reverse", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "substring", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "sum", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + { + "of": { + "type": "number" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.add_date", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.clock", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.date", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.diff", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.now_ns", + "decl": { + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_duration_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_ns", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_rfc3339_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.weekday", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "to_number", + "decl": { + "args": [ + { + "of": [ + { + "type": "null" + }, + { + "type": "boolean" + }, + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "trace", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "trim", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_left", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_prefix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_right", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_space", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_suffix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "type_name", + "decl": { + "args": [ + { + "of": [ + { + "type": "any" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "union", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "units.parse_bytes", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "upper", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode_object", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "dynamic": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode_object", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "uuid.rfc4122", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "walk", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "static": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "function" + }, + "relation": true + }, + { + "name": "yaml.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "yaml.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "yaml.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + } + ], + "future_keywords": [ + "every", + "in" + ], + "wasm_abi_versions": [ + { + "version": 1, + "minor_version": 1 + }, + { + "version": 1, + "minor_version": 2 + } + ] +} diff --git a/vendor/github.com/open-policy-agent/opa/capabilities/v0.39.0.json b/vendor/github.com/open-policy-agent/opa/capabilities/v0.39.0.json new file mode 100644 index 00000000..d5edb2da --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/capabilities/v0.39.0.json @@ -0,0 +1,3826 @@ +{ + "builtins": [ + { + "name": "abs", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "all", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "and", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "\u0026" + }, + { + "name": "any", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "array.concat", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "array.reverse", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "array.slice", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "assign", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": ":=" + }, + { + "name": "base64.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "base64url.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode_no_pad", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "bits.and", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.lsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.negate", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.or", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.rsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.xor", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "cast_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "cast_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "cast_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "null" + }, + "type": "function" + } + }, + { + "name": "cast_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "cast_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "cast_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "ceil", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "concat", + "decl": { + "args": [ + { + "type": "string" + }, + { + "of": [ + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "count", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "crypto.hmac.md5", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.hmac.sha1", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.hmac.sha256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.hmac.sha512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.md5", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha1", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha256", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_and_verify_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificate_request", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_rsa_private_key", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "div", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "/" + }, + { + "name": "endswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "eq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "=" + }, + { + "name": "equal", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "==" + }, + { + "name": "floor", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "format_int", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "glob.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "glob.quote_meta", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "graph.reachable", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "graph.reachable_paths", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "gt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e" + }, + { + "name": "gte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e=" + }, + { + "name": "hex.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "hex.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "http.send", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "indexof", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "indexof_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "internal.member_2", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "in" + }, + { + "name": "internal.member_3", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "in" + }, + { + "name": "internal.print", + "decl": { + "args": [ + { + "dynamic": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "array" + } + ], + "type": "function" + } + }, + { + "name": "intersection", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode_verify", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign_raw", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_number", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "json.patch", + "decl": { + "args": [ + { + "type": "any" + }, + { + "dynamic": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "static": [ + { + "key": "op", + "value": { + "type": "string" + } + }, + { + "key": "path", + "value": { + "type": "any" + } + } + ], + "type": "object" + }, + "type": "array" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "lower", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "lt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c" + }, + { + "name": "lte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c=" + }, + { + "name": "max", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "min", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "minus", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + "type": "function" + }, + "infix": "-" + }, + { + "name": "mul", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "*" + }, + { + "name": "neq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "!=" + }, + { + "name": "net.cidr_contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_contains_matches", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "static": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_expand", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_intersects", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_merge", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_overlap", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.lookup_ip_addr", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "numbers.range", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "object.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.get", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.union", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.union_n", + "decl": { + "args": [ + { + "dynamic": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "opa.runtime", + "decl": { + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "or", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "|" + }, + { + "name": "plus", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "+" + }, + { + "name": "print", + "decl": { + "type": "function", + "variadic": { + "type": "any" + } + } + }, + { + "name": "product", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + { + "of": { + "type": "number" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "rand.intn", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "re_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.find_all_string_submatch_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.find_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.globs_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.template_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "rego.parse_module", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "rem", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "%" + }, + { + "name": "replace", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "round", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.compare", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.is_valid", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "set_diff", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "sort", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "sprintf", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "startswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "strings.replace_n", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "string" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "strings.reverse", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "substring", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "sum", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + { + "of": { + "type": "number" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.add_date", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.clock", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.date", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.diff", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.now_ns", + "decl": { + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_duration_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_ns", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_rfc3339_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.weekday", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "to_number", + "decl": { + "args": [ + { + "of": [ + { + "type": "null" + }, + { + "type": "boolean" + }, + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "trace", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "trim", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_left", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_prefix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_right", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_space", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_suffix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "type_name", + "decl": { + "args": [ + { + "of": [ + { + "type": "any" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "union", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "units.parse_bytes", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "upper", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode_object", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "dynamic": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode_object", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "uuid.rfc4122", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "walk", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "static": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "function" + }, + "relation": true + }, + { + "name": "yaml.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "yaml.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "yaml.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + } + ], + "future_keywords": [ + "every", + "in" + ], + "wasm_abi_versions": [ + { + "version": 1, + "minor_version": 1 + }, + { + "version": 1, + "minor_version": 2 + } + ] +} diff --git a/vendor/github.com/open-policy-agent/opa/capabilities/v0.40.0.json b/vendor/github.com/open-policy-agent/opa/capabilities/v0.40.0.json new file mode 100644 index 00000000..2f9e6e64 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/capabilities/v0.40.0.json @@ -0,0 +1,3847 @@ +{ + "builtins": [ + { + "name": "abs", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "all", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "and", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "\u0026" + }, + { + "name": "any", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "array.concat", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "array.reverse", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "array.slice", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "assign", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": ":=" + }, + { + "name": "base64.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "base64url.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode_no_pad", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "bits.and", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.lsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.negate", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.or", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.rsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.xor", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "cast_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "cast_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "cast_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "null" + }, + "type": "function" + } + }, + { + "name": "cast_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "cast_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "cast_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "ceil", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "concat", + "decl": { + "args": [ + { + "type": "string" + }, + { + "of": [ + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "count", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "crypto.hmac.md5", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.hmac.sha1", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.hmac.sha256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.hmac.sha512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.md5", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha1", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha256", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_and_verify_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificate_request", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_rsa_private_key", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "div", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "/" + }, + { + "name": "endswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "eq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "=" + }, + { + "name": "equal", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "==" + }, + { + "name": "floor", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "format_int", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "glob.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "glob.quote_meta", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "graph.reachable", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "graph.reachable_paths", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "gt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e" + }, + { + "name": "gte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e=" + }, + { + "name": "hex.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "hex.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "http.send", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "indexof", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "indexof_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "internal.member_2", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "in" + }, + { + "name": "internal.member_3", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "in" + }, + { + "name": "internal.print", + "decl": { + "args": [ + { + "dynamic": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "array" + } + ], + "type": "function" + } + }, + { + "name": "intersection", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode_verify", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign_raw", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_number", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "json.patch", + "decl": { + "args": [ + { + "type": "any" + }, + { + "dynamic": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "static": [ + { + "key": "op", + "value": { + "type": "string" + } + }, + { + "key": "path", + "value": { + "type": "any" + } + } + ], + "type": "object" + }, + "type": "array" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "lower", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "lt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c" + }, + { + "name": "lte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c=" + }, + { + "name": "max", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "min", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "minus", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + "type": "function" + }, + "infix": "-" + }, + { + "name": "mul", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "*" + }, + { + "name": "neq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "!=" + }, + { + "name": "net.cidr_contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_contains_matches", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "static": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_expand", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_intersects", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_merge", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_overlap", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.lookup_ip_addr", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "numbers.range", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "object.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.get", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.union", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.union_n", + "decl": { + "args": [ + { + "dynamic": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "opa.runtime", + "decl": { + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "or", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "|" + }, + { + "name": "plus", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "+" + }, + { + "name": "print", + "decl": { + "type": "function", + "variadic": { + "type": "any" + } + } + }, + { + "name": "product", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + { + "of": { + "type": "number" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "rand.intn", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "re_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.find_all_string_submatch_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.find_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.globs_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.template_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "rego.metadata.chain", + "decl": { + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "rego.metadata.rule", + "decl": { + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "rego.parse_module", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "rem", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "%" + }, + { + "name": "replace", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "round", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.compare", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.is_valid", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "set_diff", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "sort", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "sprintf", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "startswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "strings.replace_n", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "string" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "strings.reverse", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "substring", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "sum", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + { + "of": { + "type": "number" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.add_date", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.clock", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.date", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.diff", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.now_ns", + "decl": { + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_duration_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_ns", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_rfc3339_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.weekday", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "to_number", + "decl": { + "args": [ + { + "of": [ + { + "type": "null" + }, + { + "type": "boolean" + }, + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "trace", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "trim", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_left", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_prefix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_right", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_space", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_suffix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "type_name", + "decl": { + "args": [ + { + "of": [ + { + "type": "any" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "union", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "units.parse_bytes", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "upper", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode_object", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "dynamic": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode_object", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "uuid.rfc4122", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "walk", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "static": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "function" + }, + "relation": true + }, + { + "name": "yaml.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "yaml.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "yaml.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + } + ], + "future_keywords": [ + "every", + "in" + ], + "wasm_abi_versions": [ + { + "version": 1, + "minor_version": 1 + }, + { + "version": 1, + "minor_version": 2 + } + ] +} diff --git a/vendor/github.com/open-policy-agent/opa/capabilities/v0.41.0.json b/vendor/github.com/open-policy-agent/opa/capabilities/v0.41.0.json new file mode 100644 index 00000000..cf56e6ca --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/capabilities/v0.41.0.json @@ -0,0 +1,4007 @@ +{ + "builtins": [ + { + "name": "abs", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "all", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "and", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "\u0026" + }, + { + "name": "any", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "array.concat", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "array.reverse", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "array.slice", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "assign", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": ":=" + }, + { + "name": "base64.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "base64url.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode_no_pad", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "bits.and", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.lsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.negate", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.or", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.rsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.xor", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "cast_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "cast_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "cast_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "null" + }, + "type": "function" + } + }, + { + "name": "cast_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "cast_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "cast_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "ceil", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "concat", + "decl": { + "args": [ + { + "type": "string" + }, + { + "of": [ + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "count", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "crypto.hmac.md5", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.hmac.sha1", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.hmac.sha256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.hmac.sha512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.md5", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha1", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha256", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_and_verify_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificate_request", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_rsa_private_key", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "div", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "/" + }, + { + "name": "endswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "eq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "=" + }, + { + "name": "equal", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "==" + }, + { + "name": "floor", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "format_int", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "glob.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "glob.quote_meta", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "graph.reachable", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "graph.reachable_paths", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "graphql.is_valid", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "graphql.parse", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "static": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "graphql.parse_and_verify", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "graphql.parse_query", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "graphql.parse_schema", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "gt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e" + }, + { + "name": "gte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e=" + }, + { + "name": "hex.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "hex.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "http.send", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "indexof", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "indexof_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "internal.member_2", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "in" + }, + { + "name": "internal.member_3", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "in" + }, + { + "name": "internal.print", + "decl": { + "args": [ + { + "dynamic": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "array" + } + ], + "type": "function" + } + }, + { + "name": "intersection", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode_verify", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign_raw", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_number", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "json.patch", + "decl": { + "args": [ + { + "type": "any" + }, + { + "dynamic": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "static": [ + { + "key": "op", + "value": { + "type": "string" + } + }, + { + "key": "path", + "value": { + "type": "any" + } + } + ], + "type": "object" + }, + "type": "array" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "lower", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "lt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c" + }, + { + "name": "lte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c=" + }, + { + "name": "max", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "min", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "minus", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + "type": "function" + }, + "infix": "-" + }, + { + "name": "mul", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "*" + }, + { + "name": "neq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "!=" + }, + { + "name": "net.cidr_contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_contains_matches", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "static": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_expand", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_intersects", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_merge", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_overlap", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.lookup_ip_addr", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "numbers.range", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "object.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.get", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.union", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.union_n", + "decl": { + "args": [ + { + "dynamic": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "opa.runtime", + "decl": { + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "or", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "|" + }, + { + "name": "plus", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "+" + }, + { + "name": "print", + "decl": { + "type": "function", + "variadic": { + "type": "any" + } + } + }, + { + "name": "product", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + { + "of": { + "type": "number" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "rand.intn", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "re_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.find_all_string_submatch_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.find_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.globs_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.template_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "rego.metadata.chain", + "decl": { + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "rego.metadata.rule", + "decl": { + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "rego.parse_module", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "rem", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "%" + }, + { + "name": "replace", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "round", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.compare", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.is_valid", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "set_diff", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "sort", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "sprintf", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "startswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "strings.replace_n", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "string" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "strings.reverse", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "substring", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "sum", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + { + "of": { + "type": "number" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.add_date", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.clock", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.date", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.diff", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.now_ns", + "decl": { + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_duration_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_ns", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_rfc3339_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.weekday", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "to_number", + "decl": { + "args": [ + { + "of": [ + { + "type": "null" + }, + { + "type": "boolean" + }, + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "trace", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "trim", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_left", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_prefix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_right", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_space", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_suffix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "type_name", + "decl": { + "args": [ + { + "of": [ + { + "type": "any" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "union", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "units.parse", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "units.parse_bytes", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "upper", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode_object", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "dynamic": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode_object", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "uuid.rfc4122", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "walk", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "static": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "function" + }, + "relation": true + }, + { + "name": "yaml.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "yaml.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "yaml.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + } + ], + "future_keywords": [ + "every", + "in" + ], + "wasm_abi_versions": [ + { + "version": 1, + "minor_version": 1 + }, + { + "version": 1, + "minor_version": 2 + } + ] +} diff --git a/vendor/github.com/open-policy-agent/opa/capabilities/v0.42.0.json b/vendor/github.com/open-policy-agent/opa/capabilities/v0.42.0.json new file mode 100644 index 00000000..2a57613f --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/capabilities/v0.42.0.json @@ -0,0 +1,4076 @@ +{ + "builtins": [ + { + "name": "abs", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "all", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "and", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "\u0026" + }, + { + "name": "any", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "array.concat", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "array.reverse", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "array.slice", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "assign", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": ":=" + }, + { + "name": "base64.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "base64url.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode_no_pad", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "bits.and", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.lsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.negate", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.or", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.rsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.xor", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "cast_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "cast_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "cast_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "null" + }, + "type": "function" + } + }, + { + "name": "cast_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "cast_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "cast_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "ceil", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "concat", + "decl": { + "args": [ + { + "type": "string" + }, + { + "of": [ + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "count", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "crypto.hmac.md5", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.hmac.sha1", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.hmac.sha256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.hmac.sha512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.md5", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha1", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha256", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_and_verify_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificate_request", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_rsa_private_key", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "div", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "/" + }, + { + "name": "endswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "eq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "=" + }, + { + "name": "equal", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "==" + }, + { + "name": "floor", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "format_int", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "glob.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "glob.quote_meta", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "graph.reachable", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "graph.reachable_paths", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "graphql.is_valid", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "graphql.parse", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "static": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "graphql.parse_and_verify", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "graphql.parse_query", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "graphql.parse_schema", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "gt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e" + }, + { + "name": "gte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e=" + }, + { + "name": "hex.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "hex.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "http.send", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "indexof", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "indexof_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "internal.member_2", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "in" + }, + { + "name": "internal.member_3", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "in" + }, + { + "name": "internal.print", + "decl": { + "args": [ + { + "dynamic": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "array" + } + ], + "type": "function" + } + }, + { + "name": "intersection", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode_verify", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign_raw", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_number", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "json.patch", + "decl": { + "args": [ + { + "type": "any" + }, + { + "dynamic": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "static": [ + { + "key": "op", + "value": { + "type": "string" + } + }, + { + "key": "path", + "value": { + "type": "any" + } + } + ], + "type": "object" + }, + "type": "array" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "lower", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "lt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c" + }, + { + "name": "lte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c=" + }, + { + "name": "max", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "min", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "minus", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + "type": "function" + }, + "infix": "-" + }, + { + "name": "mul", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "*" + }, + { + "name": "neq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "!=" + }, + { + "name": "net.cidr_contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_contains_matches", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "static": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_expand", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_intersects", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_merge", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_overlap", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.lookup_ip_addr", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "numbers.range", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "object.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.get", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.subset", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.union", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.union_n", + "decl": { + "args": [ + { + "dynamic": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "opa.runtime", + "decl": { + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "or", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "|" + }, + { + "name": "plus", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "+" + }, + { + "name": "print", + "decl": { + "type": "function", + "variadic": { + "type": "any" + } + } + }, + { + "name": "product", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + { + "of": { + "type": "number" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "rand.intn", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "re_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.find_all_string_submatch_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.find_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.globs_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.template_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "rego.metadata.chain", + "decl": { + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "rego.metadata.rule", + "decl": { + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "rego.parse_module", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "rem", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "%" + }, + { + "name": "replace", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "round", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.compare", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.is_valid", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "set_diff", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "sort", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "sprintf", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "startswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "strings.replace_n", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "string" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "strings.reverse", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "substring", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "sum", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + { + "of": { + "type": "number" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.add_date", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.clock", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.date", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.diff", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.now_ns", + "decl": { + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_duration_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_ns", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_rfc3339_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.weekday", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "to_number", + "decl": { + "args": [ + { + "of": [ + { + "type": "null" + }, + { + "type": "boolean" + }, + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "trace", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "trim", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_left", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_prefix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_right", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_space", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_suffix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "type_name", + "decl": { + "args": [ + { + "of": [ + { + "type": "any" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "union", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "units.parse", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "units.parse_bytes", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "upper", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode_object", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "dynamic": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode_object", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "uuid.rfc4122", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "walk", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "static": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "function" + }, + "relation": true + }, + { + "name": "yaml.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "yaml.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "yaml.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + } + ], + "future_keywords": [ + "contains", + "every", + "if", + "in" + ], + "wasm_abi_versions": [ + { + "version": 1, + "minor_version": 1 + }, + { + "version": 1, + "minor_version": 2 + } + ] +} diff --git a/vendor/github.com/open-policy-agent/opa/capabilities/v0.42.1.json b/vendor/github.com/open-policy-agent/opa/capabilities/v0.42.1.json new file mode 100644 index 00000000..2a57613f --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/capabilities/v0.42.1.json @@ -0,0 +1,4076 @@ +{ + "builtins": [ + { + "name": "abs", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "all", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "and", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "\u0026" + }, + { + "name": "any", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "array.concat", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "array.reverse", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "array.slice", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "assign", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": ":=" + }, + { + "name": "base64.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "base64url.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode_no_pad", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "bits.and", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.lsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.negate", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.or", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.rsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.xor", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "cast_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "cast_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "cast_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "null" + }, + "type": "function" + } + }, + { + "name": "cast_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "cast_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "cast_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "ceil", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "concat", + "decl": { + "args": [ + { + "type": "string" + }, + { + "of": [ + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "count", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "crypto.hmac.md5", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.hmac.sha1", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.hmac.sha256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.hmac.sha512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.md5", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha1", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha256", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_and_verify_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificate_request", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_rsa_private_key", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "div", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "/" + }, + { + "name": "endswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "eq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "=" + }, + { + "name": "equal", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "==" + }, + { + "name": "floor", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "format_int", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "glob.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "glob.quote_meta", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "graph.reachable", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "graph.reachable_paths", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "graphql.is_valid", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "graphql.parse", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "static": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "graphql.parse_and_verify", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "graphql.parse_query", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "graphql.parse_schema", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "gt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e" + }, + { + "name": "gte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e=" + }, + { + "name": "hex.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "hex.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "http.send", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "indexof", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "indexof_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "internal.member_2", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "in" + }, + { + "name": "internal.member_3", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "in" + }, + { + "name": "internal.print", + "decl": { + "args": [ + { + "dynamic": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "array" + } + ], + "type": "function" + } + }, + { + "name": "intersection", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode_verify", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign_raw", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_number", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "json.patch", + "decl": { + "args": [ + { + "type": "any" + }, + { + "dynamic": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "static": [ + { + "key": "op", + "value": { + "type": "string" + } + }, + { + "key": "path", + "value": { + "type": "any" + } + } + ], + "type": "object" + }, + "type": "array" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "lower", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "lt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c" + }, + { + "name": "lte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c=" + }, + { + "name": "max", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "min", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "minus", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + "type": "function" + }, + "infix": "-" + }, + { + "name": "mul", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "*" + }, + { + "name": "neq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "!=" + }, + { + "name": "net.cidr_contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_contains_matches", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "static": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_expand", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_intersects", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_merge", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_overlap", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.lookup_ip_addr", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "numbers.range", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "object.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.get", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.subset", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.union", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.union_n", + "decl": { + "args": [ + { + "dynamic": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "opa.runtime", + "decl": { + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "or", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "|" + }, + { + "name": "plus", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "+" + }, + { + "name": "print", + "decl": { + "type": "function", + "variadic": { + "type": "any" + } + } + }, + { + "name": "product", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + { + "of": { + "type": "number" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "rand.intn", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "re_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.find_all_string_submatch_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.find_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.globs_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.template_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "rego.metadata.chain", + "decl": { + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "rego.metadata.rule", + "decl": { + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "rego.parse_module", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "rem", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "%" + }, + { + "name": "replace", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "round", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.compare", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.is_valid", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "set_diff", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "sort", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "sprintf", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "startswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "strings.replace_n", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "string" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "strings.reverse", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "substring", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "sum", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + { + "of": { + "type": "number" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.add_date", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.clock", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.date", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.diff", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.now_ns", + "decl": { + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_duration_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_ns", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_rfc3339_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.weekday", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "to_number", + "decl": { + "args": [ + { + "of": [ + { + "type": "null" + }, + { + "type": "boolean" + }, + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "trace", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "trim", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_left", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_prefix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_right", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_space", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_suffix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "type_name", + "decl": { + "args": [ + { + "of": [ + { + "type": "any" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "union", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "units.parse", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "units.parse_bytes", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "upper", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode_object", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "dynamic": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode_object", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "uuid.rfc4122", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "walk", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "static": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "function" + }, + "relation": true + }, + { + "name": "yaml.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "yaml.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "yaml.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + } + ], + "future_keywords": [ + "contains", + "every", + "if", + "in" + ], + "wasm_abi_versions": [ + { + "version": 1, + "minor_version": 1 + }, + { + "version": 1, + "minor_version": 2 + } + ] +} diff --git a/vendor/github.com/open-policy-agent/opa/capabilities/v0.42.2.json b/vendor/github.com/open-policy-agent/opa/capabilities/v0.42.2.json new file mode 100644 index 00000000..2a57613f --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/capabilities/v0.42.2.json @@ -0,0 +1,4076 @@ +{ + "builtins": [ + { + "name": "abs", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "all", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "and", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "\u0026" + }, + { + "name": "any", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "array.concat", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "array.reverse", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "array.slice", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "assign", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": ":=" + }, + { + "name": "base64.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "base64url.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode_no_pad", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "bits.and", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.lsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.negate", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.or", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.rsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.xor", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "cast_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "cast_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "cast_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "null" + }, + "type": "function" + } + }, + { + "name": "cast_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "cast_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "cast_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "ceil", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "concat", + "decl": { + "args": [ + { + "type": "string" + }, + { + "of": [ + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "count", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "crypto.hmac.md5", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.hmac.sha1", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.hmac.sha256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.hmac.sha512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.md5", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha1", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha256", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_and_verify_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificate_request", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_rsa_private_key", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "div", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "/" + }, + { + "name": "endswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "eq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "=" + }, + { + "name": "equal", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "==" + }, + { + "name": "floor", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "format_int", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "glob.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "glob.quote_meta", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "graph.reachable", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "graph.reachable_paths", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "graphql.is_valid", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "graphql.parse", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "static": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "graphql.parse_and_verify", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "graphql.parse_query", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "graphql.parse_schema", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "gt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e" + }, + { + "name": "gte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e=" + }, + { + "name": "hex.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "hex.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "http.send", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "indexof", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "indexof_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "internal.member_2", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "in" + }, + { + "name": "internal.member_3", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "in" + }, + { + "name": "internal.print", + "decl": { + "args": [ + { + "dynamic": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "array" + } + ], + "type": "function" + } + }, + { + "name": "intersection", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode_verify", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign_raw", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_number", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "json.patch", + "decl": { + "args": [ + { + "type": "any" + }, + { + "dynamic": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "static": [ + { + "key": "op", + "value": { + "type": "string" + } + }, + { + "key": "path", + "value": { + "type": "any" + } + } + ], + "type": "object" + }, + "type": "array" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "lower", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "lt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c" + }, + { + "name": "lte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c=" + }, + { + "name": "max", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "min", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "minus", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + "type": "function" + }, + "infix": "-" + }, + { + "name": "mul", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "*" + }, + { + "name": "neq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "!=" + }, + { + "name": "net.cidr_contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_contains_matches", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "static": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_expand", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_intersects", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_merge", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_overlap", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.lookup_ip_addr", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "numbers.range", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "object.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.get", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.subset", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.union", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.union_n", + "decl": { + "args": [ + { + "dynamic": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "opa.runtime", + "decl": { + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "or", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "|" + }, + { + "name": "plus", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "+" + }, + { + "name": "print", + "decl": { + "type": "function", + "variadic": { + "type": "any" + } + } + }, + { + "name": "product", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + { + "of": { + "type": "number" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "rand.intn", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "re_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.find_all_string_submatch_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.find_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.globs_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.template_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "rego.metadata.chain", + "decl": { + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "rego.metadata.rule", + "decl": { + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "rego.parse_module", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "rem", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "%" + }, + { + "name": "replace", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "round", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.compare", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.is_valid", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "set_diff", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "sort", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "sprintf", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "startswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "strings.replace_n", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "string" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "strings.reverse", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "substring", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "sum", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + { + "of": { + "type": "number" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.add_date", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.clock", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.date", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.diff", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.now_ns", + "decl": { + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_duration_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_ns", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_rfc3339_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.weekday", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "to_number", + "decl": { + "args": [ + { + "of": [ + { + "type": "null" + }, + { + "type": "boolean" + }, + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "trace", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "trim", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_left", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_prefix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_right", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_space", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_suffix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "type_name", + "decl": { + "args": [ + { + "of": [ + { + "type": "any" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "union", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "units.parse", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "units.parse_bytes", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "upper", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode_object", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "dynamic": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode_object", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "uuid.rfc4122", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "walk", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "static": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "function" + }, + "relation": true + }, + { + "name": "yaml.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "yaml.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "yaml.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + } + ], + "future_keywords": [ + "contains", + "every", + "if", + "in" + ], + "wasm_abi_versions": [ + { + "version": 1, + "minor_version": 1 + }, + { + "version": 1, + "minor_version": 2 + } + ] +} diff --git a/vendor/github.com/open-policy-agent/opa/capabilities/v0.43.0.json b/vendor/github.com/open-policy-agent/opa/capabilities/v0.43.0.json new file mode 100644 index 00000000..f61c6f85 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/capabilities/v0.43.0.json @@ -0,0 +1,4079 @@ +{ + "builtins": [ + { + "name": "abs", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "all", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "and", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "\u0026" + }, + { + "name": "any", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "array.concat", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "array.reverse", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "array.slice", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "assign", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": ":=" + }, + { + "name": "base64.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "base64url.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode_no_pad", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "bits.and", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.lsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.negate", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.or", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.rsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.xor", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "cast_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "cast_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "cast_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "null" + }, + "type": "function" + } + }, + { + "name": "cast_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "cast_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "cast_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "ceil", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "concat", + "decl": { + "args": [ + { + "type": "string" + }, + { + "of": [ + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "count", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "crypto.hmac.md5", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.hmac.sha1", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.hmac.sha256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.hmac.sha512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.md5", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha1", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha256", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_and_verify_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificate_request", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_rsa_private_key", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "div", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "/" + }, + { + "name": "endswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "eq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "=" + }, + { + "name": "equal", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "==" + }, + { + "name": "floor", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "format_int", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "glob.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "of": [ + { + "type": "null" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + } + ], + "type": "any" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "glob.quote_meta", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "graph.reachable", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "graph.reachable_paths", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "graphql.is_valid", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "graphql.parse", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "static": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "graphql.parse_and_verify", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "graphql.parse_query", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "graphql.parse_schema", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "gt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e" + }, + { + "name": "gte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e=" + }, + { + "name": "hex.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "hex.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "http.send", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "indexof", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "indexof_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "internal.member_2", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "in" + }, + { + "name": "internal.member_3", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "in" + }, + { + "name": "internal.print", + "decl": { + "args": [ + { + "dynamic": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "array" + } + ], + "type": "function" + } + }, + { + "name": "intersection", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode_verify", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign_raw", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_number", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "json.patch", + "decl": { + "args": [ + { + "type": "any" + }, + { + "dynamic": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "static": [ + { + "key": "op", + "value": { + "type": "string" + } + }, + { + "key": "path", + "value": { + "type": "any" + } + } + ], + "type": "object" + }, + "type": "array" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "lower", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "lt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c" + }, + { + "name": "lte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c=" + }, + { + "name": "max", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "min", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "minus", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + "type": "function" + }, + "infix": "-" + }, + { + "name": "mul", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "*" + }, + { + "name": "neq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "!=" + }, + { + "name": "net.cidr_contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_contains_matches", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "static": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_expand", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_intersects", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_merge", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_overlap", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.lookup_ip_addr", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "numbers.range", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "object.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.get", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.subset", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.union", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.union_n", + "decl": { + "args": [ + { + "dynamic": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "opa.runtime", + "decl": { + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "or", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "|" + }, + { + "name": "plus", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "+" + }, + { + "name": "print", + "decl": { + "type": "function", + "variadic": { + "type": "any" + } + } + }, + { + "name": "product", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + { + "of": { + "type": "number" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "rand.intn", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "re_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.find_all_string_submatch_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.find_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.globs_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.template_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "rego.metadata.chain", + "decl": { + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "rego.metadata.rule", + "decl": { + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "rego.parse_module", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "rem", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "%" + }, + { + "name": "replace", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "round", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.compare", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.is_valid", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "set_diff", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "sort", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "sprintf", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "startswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "strings.replace_n", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "string" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "strings.reverse", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "substring", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "sum", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + { + "of": { + "type": "number" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.add_date", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.clock", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.date", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.diff", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.now_ns", + "decl": { + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_duration_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_ns", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_rfc3339_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.weekday", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "to_number", + "decl": { + "args": [ + { + "of": [ + { + "type": "null" + }, + { + "type": "boolean" + }, + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "trace", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "trim", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_left", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_prefix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_right", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_space", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_suffix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "type_name", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "union", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "units.parse", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "units.parse_bytes", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "upper", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode_object", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "dynamic": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode_object", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "uuid.rfc4122", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "walk", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "static": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "function" + }, + "relation": true + }, + { + "name": "yaml.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "yaml.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "yaml.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + } + ], + "future_keywords": [ + "contains", + "every", + "if", + "in" + ], + "wasm_abi_versions": [ + { + "version": 1, + "minor_version": 1 + }, + { + "version": 1, + "minor_version": 2 + } + ] +} diff --git a/vendor/github.com/open-policy-agent/opa/capabilities/v0.43.1.json b/vendor/github.com/open-policy-agent/opa/capabilities/v0.43.1.json new file mode 100644 index 00000000..f61c6f85 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/capabilities/v0.43.1.json @@ -0,0 +1,4079 @@ +{ + "builtins": [ + { + "name": "abs", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "all", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "and", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "\u0026" + }, + { + "name": "any", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "array.concat", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "array.reverse", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "array.slice", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "assign", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": ":=" + }, + { + "name": "base64.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "base64url.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode_no_pad", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "bits.and", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.lsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.negate", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.or", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.rsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.xor", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "cast_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "cast_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "cast_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "null" + }, + "type": "function" + } + }, + { + "name": "cast_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "cast_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "cast_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "ceil", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "concat", + "decl": { + "args": [ + { + "type": "string" + }, + { + "of": [ + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "count", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "crypto.hmac.md5", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.hmac.sha1", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.hmac.sha256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.hmac.sha512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.md5", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha1", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha256", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_and_verify_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificate_request", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_rsa_private_key", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "div", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "/" + }, + { + "name": "endswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "eq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "=" + }, + { + "name": "equal", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "==" + }, + { + "name": "floor", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "format_int", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "glob.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "of": [ + { + "type": "null" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + } + ], + "type": "any" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "glob.quote_meta", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "graph.reachable", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "graph.reachable_paths", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "graphql.is_valid", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "graphql.parse", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "static": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "graphql.parse_and_verify", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "graphql.parse_query", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "graphql.parse_schema", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "gt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e" + }, + { + "name": "gte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e=" + }, + { + "name": "hex.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "hex.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "http.send", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "indexof", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "indexof_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "internal.member_2", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "in" + }, + { + "name": "internal.member_3", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "in" + }, + { + "name": "internal.print", + "decl": { + "args": [ + { + "dynamic": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "array" + } + ], + "type": "function" + } + }, + { + "name": "intersection", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode_verify", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.encode_sign_raw", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_number", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "json.patch", + "decl": { + "args": [ + { + "type": "any" + }, + { + "dynamic": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "static": [ + { + "key": "op", + "value": { + "type": "string" + } + }, + { + "key": "path", + "value": { + "type": "any" + } + } + ], + "type": "object" + }, + "type": "array" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "lower", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "lt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c" + }, + { + "name": "lte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c=" + }, + { + "name": "max", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "min", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "minus", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + "type": "function" + }, + "infix": "-" + }, + { + "name": "mul", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "*" + }, + { + "name": "neq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "!=" + }, + { + "name": "net.cidr_contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_contains_matches", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "static": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_expand", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_intersects", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_merge", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_overlap", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.lookup_ip_addr", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "numbers.range", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "object.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.get", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.subset", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.union", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.union_n", + "decl": { + "args": [ + { + "dynamic": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "opa.runtime", + "decl": { + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "or", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "|" + }, + { + "name": "plus", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "+" + }, + { + "name": "print", + "decl": { + "type": "function", + "variadic": { + "type": "any" + } + } + }, + { + "name": "product", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + { + "of": { + "type": "number" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "rand.intn", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "re_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.find_all_string_submatch_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.find_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.globs_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.template_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "rego.metadata.chain", + "decl": { + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "rego.metadata.rule", + "decl": { + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "rego.parse_module", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "rem", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "%" + }, + { + "name": "replace", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "round", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.compare", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.is_valid", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "set_diff", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "sort", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "sprintf", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "startswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "strings.replace_n", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "string" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "strings.reverse", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "substring", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "sum", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + { + "of": { + "type": "number" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.add_date", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.clock", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.date", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.diff", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.now_ns", + "decl": { + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_duration_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_ns", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_rfc3339_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.weekday", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "to_number", + "decl": { + "args": [ + { + "of": [ + { + "type": "null" + }, + { + "type": "boolean" + }, + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "trace", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "trim", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_left", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_prefix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_right", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_space", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_suffix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "type_name", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "union", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "units.parse", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "units.parse_bytes", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "upper", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode_object", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "dynamic": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode_object", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "uuid.rfc4122", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "walk", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "static": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "function" + }, + "relation": true + }, + { + "name": "yaml.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "yaml.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "yaml.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + } + ], + "future_keywords": [ + "contains", + "every", + "if", + "in" + ], + "wasm_abi_versions": [ + { + "version": 1, + "minor_version": 1 + }, + { + "version": 1, + "minor_version": 2 + } + ] +} diff --git a/vendor/github.com/open-policy-agent/opa/capabilities/v0.44.0.json b/vendor/github.com/open-policy-agent/opa/capabilities/v0.44.0.json new file mode 100644 index 00000000..a44b9511 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/capabilities/v0.44.0.json @@ -0,0 +1,4190 @@ +{ + "builtins": [ + { + "name": "abs", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "all", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "and", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "\u0026" + }, + { + "name": "any", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "array.concat", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "array.reverse", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "array.slice", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "assign", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": ":=" + }, + { + "name": "base64.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "base64url.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode_no_pad", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "bits.and", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.lsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.negate", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.or", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.rsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.xor", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "cast_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "cast_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "cast_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "null" + }, + "type": "function" + } + }, + { + "name": "cast_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "cast_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "cast_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "ceil", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "concat", + "decl": { + "args": [ + { + "type": "string" + }, + { + "of": [ + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "count", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "crypto.hmac.md5", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.hmac.sha1", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.hmac.sha256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.hmac.sha512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.md5", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha1", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha256", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_and_verify_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificate_request", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_rsa_private_key", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "div", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "/" + }, + { + "name": "endswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "eq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "=" + }, + { + "name": "equal", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "==" + }, + { + "name": "floor", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "format_int", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "glob.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "of": [ + { + "type": "null" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + } + ], + "type": "any" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "glob.quote_meta", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "graph.reachable", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "graph.reachable_paths", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "graphql.is_valid", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "graphql.parse", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "static": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "graphql.parse_and_verify", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "graphql.parse_query", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "graphql.parse_schema", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "gt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e" + }, + { + "name": "gte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e=" + }, + { + "name": "hex.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "hex.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "http.send", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + }, + "nondeterministic": true + }, + { + "name": "indexof", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "indexof_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "internal.member_2", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "in" + }, + { + "name": "internal.member_3", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "in" + }, + { + "name": "internal.print", + "decl": { + "args": [ + { + "dynamic": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "array" + } + ], + "type": "function" + } + }, + { + "name": "intersection", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode_verify", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "array" + }, + "type": "function" + }, + "nondeterministic": true + }, + { + "name": "io.jwt.encode_sign", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + }, + "nondeterministic": true + }, + { + "name": "io.jwt.encode_sign_raw", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + }, + "nondeterministic": true + }, + { + "name": "io.jwt.verify_es256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_number", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "json.patch", + "decl": { + "args": [ + { + "type": "any" + }, + { + "dynamic": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "static": [ + { + "key": "op", + "value": { + "type": "string" + } + }, + { + "key": "path", + "value": { + "type": "any" + } + } + ], + "type": "object" + }, + "type": "array" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "lower", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "lt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c" + }, + { + "name": "lte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c=" + }, + { + "name": "max", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "min", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "minus", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + "type": "function" + }, + "infix": "-" + }, + { + "name": "mul", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "*" + }, + { + "name": "neq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "!=" + }, + { + "name": "net.cidr_contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_contains_matches", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "static": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_expand", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_intersects", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_merge", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_overlap", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.lookup_ip_addr", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + }, + "nondeterministic": true + }, + { + "name": "numbers.range", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "object.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.get", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.subset", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.union", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.union_n", + "decl": { + "args": [ + { + "dynamic": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "opa.runtime", + "decl": { + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + }, + "nondeterministic": true + }, + { + "name": "or", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "|" + }, + { + "name": "plus", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "+" + }, + { + "name": "print", + "decl": { + "type": "function", + "variadic": { + "type": "any" + } + } + }, + { + "name": "product", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + { + "of": { + "type": "number" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "rand.intn", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "nondeterministic": true + }, + { + "name": "re_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.find_all_string_submatch_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.find_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.globs_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.template_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "rego.metadata.chain", + "decl": { + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "rego.metadata.rule", + "decl": { + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "rego.parse_module", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "rem", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "%" + }, + { + "name": "replace", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "round", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.compare", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.is_valid", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "set_diff", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "sort", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "sprintf", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "startswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "strings.any_prefix_match", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "strings.any_suffix_match", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "strings.replace_n", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "string" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "strings.reverse", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "substring", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "sum", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + { + "of": { + "type": "number" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.add_date", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.clock", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.date", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.diff", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.now_ns", + "decl": { + "result": { + "type": "number" + }, + "type": "function" + }, + "nondeterministic": true + }, + { + "name": "time.parse_duration_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_ns", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_rfc3339_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.weekday", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "to_number", + "decl": { + "args": [ + { + "of": [ + { + "type": "null" + }, + { + "type": "boolean" + }, + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "trace", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "trim", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_left", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_prefix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_right", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_space", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_suffix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "type_name", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "union", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "units.parse", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "units.parse_bytes", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "upper", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode_object", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "dynamic": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode_object", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "uuid.rfc4122", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + }, + "nondeterministic": true + }, + { + "name": "walk", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "static": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "function" + }, + "relation": true + }, + { + "name": "yaml.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "yaml.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "yaml.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + } + ], + "future_keywords": [ + "contains", + "every", + "if", + "in" + ], + "wasm_abi_versions": [ + { + "version": 1, + "minor_version": 1 + }, + { + "version": 1, + "minor_version": 2 + } + ] +} diff --git a/vendor/github.com/open-policy-agent/opa/capabilities/v0.45.0.json b/vendor/github.com/open-policy-agent/opa/capabilities/v0.45.0.json new file mode 100644 index 00000000..1d28f21b --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/capabilities/v0.45.0.json @@ -0,0 +1,4306 @@ +{ + "builtins": [ + { + "name": "abs", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "all", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "and", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "\u0026" + }, + { + "name": "any", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "array.concat", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "array.reverse", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "array.slice", + "decl": { + "args": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "assign", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": ":=" + }, + { + "name": "base64.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "base64url.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "base64url.encode_no_pad", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "bits.and", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.lsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.negate", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.or", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.rsh", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "bits.xor", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "cast_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "cast_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "cast_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "null" + }, + "type": "function" + } + }, + { + "name": "cast_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "cast_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "cast_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "ceil", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "concat", + "decl": { + "args": [ + { + "type": "string" + }, + { + "of": [ + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "count", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "crypto.hmac.md5", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.hmac.sha1", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.hmac.sha256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.hmac.sha512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.md5", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha1", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.sha256", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_and_verify_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificate_request", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_certificates", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "crypto.x509.parse_rsa_private_key", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "div", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "/" + }, + { + "name": "endswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "eq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "=" + }, + { + "name": "equal", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "==" + }, + { + "name": "floor", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "format_int", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "glob.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "of": [ + { + "type": "null" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + } + ], + "type": "any" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "glob.quote_meta", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "graph.reachable", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "graph.reachable_paths", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "graphql.is_valid", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "graphql.parse", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "graphql.parse_and_verify", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "graphql.parse_query", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "graphql.parse_schema", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "gt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e" + }, + { + "name": "gte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003e=" + }, + { + "name": "hex.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "hex.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "http.send", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + }, + "nondeterministic": true + }, + { + "name": "indexof", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "indexof_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "internal.member_2", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "in" + }, + { + "name": "internal.member_3", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "in" + }, + { + "name": "internal.print", + "decl": { + "args": [ + { + "dynamic": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "array" + } + ], + "type": "function" + } + }, + { + "name": "intersection", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "static": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "io.jwt.decode_verify", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "static": [ + { + "type": "boolean" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "type": "array" + }, + "type": "function" + }, + "nondeterministic": true + }, + { + "name": "io.jwt.encode_sign", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + }, + "nondeterministic": true + }, + { + "name": "io.jwt.encode_sign_raw", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + }, + "nondeterministic": true + }, + { + "name": "io.jwt.verify_es256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_es512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_hs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_ps512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs256", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs384", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "io.jwt.verify_rs512", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_array", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_boolean", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_null", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_number", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_object", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_set", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "is_string", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "json.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "json.patch", + "decl": { + "args": [ + { + "type": "any" + }, + { + "dynamic": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "static": [ + { + "key": "op", + "value": { + "type": "string" + } + }, + { + "key": "path", + "value": { + "type": "any" + } + } + ], + "type": "object" + }, + "type": "array" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "json.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "lower", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "lt", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c" + }, + { + "name": "lte", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "\u003c=" + }, + { + "name": "max", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "min", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "minus", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": [ + { + "type": "number" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + "type": "function" + }, + "infix": "-" + }, + { + "name": "mul", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "*" + }, + { + "name": "neq", + "decl": { + "args": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + }, + "infix": "!=" + }, + { + "name": "net.cidr_contains", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_contains_matches", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "static": [ + { + "type": "any" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_expand", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_intersects", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.cidr_merge", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "of": [ + { + "type": "string" + } + ], + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "net.cidr_overlap", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "net.lookup_ip_addr", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "of": { + "type": "string" + }, + "type": "set" + }, + "type": "function" + }, + "nondeterministic": true + }, + { + "name": "numbers.range", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "object.filter", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.get", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "type": "any" + }, + { + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.remove", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.subset", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.union", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "object.union_n", + "decl": { + "args": [ + { + "dynamic": { + "dynamic": { + "key": { + "type": "any" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "array" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "opa.runtime", + "decl": { + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + }, + "nondeterministic": true + }, + { + "name": "or", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + }, + "infix": "|" + }, + { + "name": "plus", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "+" + }, + { + "name": "print", + "decl": { + "type": "function", + "variadic": { + "type": "any" + } + } + }, + { + "name": "product", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + { + "of": { + "type": "number" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "rand.intn", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "nondeterministic": true + }, + { + "name": "re_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.find_all_string_submatch_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.find_n", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "number" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.globs_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "regex.replace", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "regex.split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "regex.template_match", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "rego.metadata.chain", + "decl": { + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "rego.metadata.rule", + "decl": { + "result": { + "type": "any" + }, + "type": "function" + } + }, + { + "name": "rego.parse_module", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "any" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "rem", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + }, + "infix": "%" + }, + { + "name": "replace", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "round", + "decl": { + "args": [ + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.compare", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "semver.is_valid", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "set_diff", + "decl": { + "args": [ + { + "of": { + "type": "any" + }, + "type": "set" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "sort", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "of": { + "type": "any" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "split", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + "type": "function" + } + }, + { + "name": "sprintf", + "decl": { + "args": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "any" + }, + "type": "array" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "startswith", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "strings.any_prefix_match", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "strings.any_suffix_match", + "decl": { + "args": [ + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "strings.replace_n", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "type": "string" + } + }, + "type": "object" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "strings.reverse", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "substring", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "sum", + "decl": { + "args": [ + { + "of": [ + { + "dynamic": { + "type": "number" + }, + "type": "array" + }, + { + "of": { + "type": "number" + }, + "type": "set" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.add_date", + "decl": { + "args": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.clock", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.date", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.diff", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + }, + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "static": [ + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + }, + { + "type": "number" + } + ], + "type": "array" + }, + "type": "function" + } + }, + { + "name": "time.now_ns", + "decl": { + "result": { + "type": "number" + }, + "type": "function" + }, + "nondeterministic": true + }, + { + "name": "time.parse_duration_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_ns", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.parse_rfc3339_ns", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "time.weekday", + "decl": { + "args": [ + { + "of": [ + { + "type": "number" + }, + { + "static": [ + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "array" + } + ], + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "to_number", + "decl": { + "args": [ + { + "of": [ + { + "type": "null" + }, + { + "type": "boolean" + }, + { + "type": "number" + }, + { + "type": "string" + } + ], + "type": "any" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "trace", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "trim", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_left", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_prefix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_right", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_space", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "trim_suffix", + "decl": { + "args": [ + { + "type": "string" + }, + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "type_name", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "union", + "decl": { + "args": [ + { + "of": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "set" + } + ], + "result": { + "of": { + "type": "any" + }, + "type": "set" + }, + "type": "function" + } + }, + { + "name": "units.parse", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "units.parse_bytes", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "number" + }, + "type": "function" + } + }, + { + "name": "upper", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.decode_object", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "dynamic": { + "type": "string" + }, + "type": "array" + } + }, + "type": "object" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "urlquery.encode_object", + "decl": { + "args": [ + { + "dynamic": { + "key": { + "type": "string" + }, + "value": { + "of": [ + { + "type": "string" + }, + { + "dynamic": { + "type": "string" + }, + "type": "array" + }, + { + "of": { + "type": "string" + }, + "type": "set" + } + ], + "type": "any" + } + }, + "type": "object" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "uuid.rfc4122", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "string" + }, + "type": "function" + }, + "nondeterministic": true + }, + { + "name": "walk", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "static": [ + { + "dynamic": { + "type": "any" + }, + "type": "array" + }, + { + "type": "any" + } + ], + "type": "array" + }, + "type": "function" + }, + "relation": true + }, + { + "name": "yaml.is_valid", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "boolean" + }, + "type": "function" + } + }, + { + "name": "yaml.marshal", + "decl": { + "args": [ + { + "type": "any" + } + ], + "result": { + "type": "string" + }, + "type": "function" + } + }, + { + "name": "yaml.unmarshal", + "decl": { + "args": [ + { + "type": "string" + } + ], + "result": { + "type": "any" + }, + "type": "function" + } + } + ], + "future_keywords": [ + "contains", + "every", + "if", + "in" + ], + "wasm_abi_versions": [ + { + "version": 1, + "minor_version": 1 + }, + { + "version": 1, + "minor_version": 2 + } + ] +} diff --git a/vendor/github.com/open-policy-agent/opa/format/format.go b/vendor/github.com/open-policy-agent/opa/format/format.go index dbe165d5..f549f4ec 100644 --- a/vendor/github.com/open-policy-agent/opa/format/format.go +++ b/vendor/github.com/open-policy-agent/opa/format/format.go @@ -12,8 +12,23 @@ import ( "sort" "github.com/open-policy-agent/opa/ast" + "github.com/open-policy-agent/opa/internal/future" ) +// Opts lets you control the code formatting via `AstWithOpts()`. +type Opts struct { + // IgnoreLocations instructs the formatter not to use the AST nodes' locations + // into account when laying out the code: notably, when the input is the result + // of partial evaluation, arguments maybe have been shuffled around, but still + // carry along their original source locations. + IgnoreLocations bool +} + +// defaultLocationFile is the file name used in `Ast()` for terms +// without a location, as could happen when pretty-printing the +// results of partial eval. +const defaultLocationFile = "__format_default__" + // Source formats a Rego source file. The bytes provided must describe a complete // Rego module. If they don't, Source will return an error resulting from the attempt // to parse the bytes. @@ -43,7 +58,10 @@ func MustAst(x interface{}) []byte { // element, Ast returns nil and an error. If AST nodes are missing locations // an arbitrary location will be used. func Ast(x interface{}) ([]byte, error) { + return AstWithOpts(x, Opts{}) +} +func AstWithOpts(x interface{}, opts Opts) ([]byte, error) { // The node has to be deep copied because it may be mutated below. Alternatively, // we could avoid the copy by checking if mutation will occur first. For now, // since format is not latency sensitive, just deep copy in all cases. @@ -55,7 +73,18 @@ func Ast(x interface{}) ([]byte, error) { // or internal.member_3, it will sugarize them into usage of the `in` // operator. It has to ensure that the proper future keyword import is // present. - extraFutureKeywordImports := map[string]bool{} + extraFutureKeywordImports := map[string]struct{}{} + + // When the future keyword "contains" is imported, all the pretty-printed + // modules will use that format for partial sets. + // NOTE(sr): For ref-head rules, this will be the default behaviour, since + // we need "contains" to disambiguate complete rules from partial sets. + useContainsKW := false + + // Same logic applies as for "contains": if `future.keywords.if` (or all + // future keywords) is imported, we'll render rules that can use `if` with + // `if`. + useIf := false // Preprocess the AST. Set any required defaults and calculate // values required for printing the formatted output. @@ -69,13 +98,26 @@ func Ast(x interface{}) ([]byte, error) { unmangleWildcardVar(wildcards, n) case *ast.Expr: - if n.IsCall() && - ast.Member.Ref().Equal(n.Operator()) || - ast.MemberWithKey.Ref().Equal(n.Operator()) { - extraFutureKeywordImports["in"] = true + switch { + case n.IsCall() && ast.Member.Ref().Equal(n.Operator()) || ast.MemberWithKey.Ref().Equal(n.Operator()): + extraFutureKeywordImports["in"] = struct{}{} + case n.IsEvery(): + extraFutureKeywordImports["every"] = struct{}{} + } + + case *ast.Import: + switch { + case future.IsAllFutureKeywords(n): + useContainsKW = true + useIf = true + case future.IsFutureKeyword(n, "contains"): + useContainsKW = true + case future.IsFutureKeyword(n, "if"): + useIf = true } } - if x.Loc() == nil { + + if opts.IgnoreLocations || x.Loc() == nil { x.SetLoc(defaultLocation(x)) } return false @@ -90,21 +132,26 @@ func Ast(x interface{}) ([]byte, error) { for kw := range extraFutureKeywordImports { x.Imports = ensureFutureKeywordImport(x.Imports, kw) } - w.writeModule(x) + w.writeModule(x, useContainsKW, useIf) case *ast.Package: w.writePackage(x, nil) case *ast.Import: w.writeImports([]*ast.Import{x}, nil) case *ast.Rule: - w.writeRule(x, false, nil) + w.writeRule(x, false /* isElse */, useContainsKW, useIf, nil) case *ast.Head: - w.writeHead(x, false, false, nil) + w.writeHead(x, + false, // isDefault + false, // isExpandedConst + useContainsKW, + useIf, + nil) case ast.Body: w.writeBody(x, nil) case *ast.Expr: w.writeExpr(x, nil) case *ast.With: - w.writeWith(x, nil) + w.writeWith(x, nil, false) case *ast.Term: w.writeTerm(x, nil) case ast.Value: @@ -154,7 +201,7 @@ func squashTrailingNewlines(bs []byte) []byte { } func defaultLocation(x ast.Node) *ast.Location { - return ast.NewLocation([]byte(x.String()), "", 1, 1) + return ast.NewLocation([]byte(x.String()), defaultLocationFile, 1, 1) } type writer struct { @@ -167,7 +214,7 @@ type writer struct { delay bool } -func (w *writer) writeModule(module *ast.Module) { +func (w *writer) writeModule(module *ast.Module, useContainsKW, useIf bool) { var pkg *ast.Package var others []interface{} var comments []*ast.Comment @@ -206,7 +253,7 @@ func (w *writer) writeModule(module *ast.Module) { imports, others = gatherImports(others) comments = w.writeImports(imports, comments) rules, others = gatherRules(others) - comments = w.writeRules(rules, comments) + comments = w.writeRules(rules, useContainsKW, useIf, comments) } for i, c := range comments { @@ -236,16 +283,16 @@ func (w *writer) writeComments(comments []*ast.Comment) { } } -func (w *writer) writeRules(rules []*ast.Rule, comments []*ast.Comment) []*ast.Comment { +func (w *writer) writeRules(rules []*ast.Rule, useContainsKW, useIf bool, comments []*ast.Comment) []*ast.Comment { for _, rule := range rules { comments = w.insertComments(comments, rule.Location) - comments = w.writeRule(rule, false, comments) + comments = w.writeRule(rule, false, useContainsKW, useIf, comments) w.blankLine() } return comments } -func (w *writer) writeRule(rule *ast.Rule, isElse bool, comments []*ast.Comment) []*ast.Comment { +func (w *writer) writeRule(rule *ast.Rule, isElse, useContainsKW, useIf bool, comments []*ast.Comment) []*ast.Comment { if rule == nil { return comments } @@ -264,13 +311,30 @@ func (w *writer) writeRule(rule *ast.Rule, isElse bool, comments []*ast.Comment) // pretend that the rule has no body in this case. isExpandedConst := rule.Body.Equal(ast.NewBody(ast.NewExpr(ast.BooleanTerm(true)))) && rule.Else == nil - comments = w.writeHead(rule.Head, rule.Default, isExpandedConst, comments) + comments = w.writeHead(rule.Head, rule.Default, isExpandedConst, useContainsKW, useIf, comments) + + // this excludes partial sets UNLESS `contains` is used + partialSetException := useContainsKW || rule.Head.Value != nil if (len(rule.Body) == 0 || isExpandedConst) && !isElse { w.endLine() return comments } + if useIf && partialSetException { + w.write(" if") + if len(rule.Body) == 1 { + if rule.Body[0].Location.Row == rule.Head.Location.Row { + w.write(" ") + comments = w.writeExpr(rule.Body[0], comments) + w.endLine() + if rule.Else != nil { + comments = w.writeElse(rule, useContainsKW, useIf, comments) + } + return comments + } + } + } w.write(" {") w.endLine() w.up() @@ -281,8 +345,10 @@ func (w *writer) writeRule(rule *ast.Rule, isElse bool, comments []*ast.Comment) if len(rule.Head.Args) > 0 { closeLoc = closingLoc('(', ')', '{', '}', rule.Location) - } else { + } else if rule.Head.Key != nil { closeLoc = closingLoc('[', ']', '{', '}', rule.Location) + } else { + closeLoc = closingLoc(0, 0, '{', '}', rule.Location) } comments = w.insertComments(comments, closeLoc) @@ -291,12 +357,12 @@ func (w *writer) writeRule(rule *ast.Rule, isElse bool, comments []*ast.Comment) w.startLine() w.write("}") if rule.Else != nil { - comments = w.writeElse(rule, comments) + comments = w.writeElse(rule, useContainsKW, useIf, comments) } return comments } -func (w *writer) writeElse(rule *ast.Rule, comments []*ast.Comment) []*ast.Comment { +func (w *writer) writeElse(rule *ast.Rule, useContainsKW, useIf bool, comments []*ast.Comment) []*ast.Comment { // If there was nothing else on the line before the "else" starts // then preserve this style of else block, otherwise it will be // started as an "inline" else eg: @@ -357,10 +423,10 @@ func (w *writer) writeElse(rule *ast.Rule, comments []*ast.Comment) []*ast.Comme rule.Else.Head.Value.Location = rule.Else.Head.Location } - return w.writeRule(rule.Else, true, comments) + return w.writeRule(rule.Else, true, useContainsKW, useIf, comments) } -func (w *writer) writeHead(head *ast.Head, isDefault bool, isExpandedConst bool, comments []*ast.Comment) []*ast.Comment { +func (w *writer) writeHead(head *ast.Head, isDefault, isExpandedConst, useContainsKW, useIf bool, comments []*ast.Comment) []*ast.Comment { w.write(head.Name.String()) if len(head.Args) > 0 { w.write("(") @@ -372,9 +438,14 @@ func (w *writer) writeHead(head *ast.Head, isDefault bool, isExpandedConst bool, w.write(")") } if head.Key != nil { - w.write("[") - comments = w.writeTerm(head.Key, comments) - w.write("]") + if useContainsKW && head.Value == nil { + w.write(" contains ") + comments = w.writeTerm(head.Key, comments) + } else { // no `if` for p[x] notation + w.write("[") + comments = w.writeTerm(head.Key, comments) + w.write("]") + } } if head.Value != nil && (head.Key != nil || ast.Compare(head.Value, ast.BooleanTerm(true)) != 0 || isExpandedConst || isDefault) { if head.Assign { @@ -400,10 +471,19 @@ func (w *writer) insertComments(comments []*ast.Comment, loc *ast.Location) []*a func (w *writer) writeBody(body ast.Body, comments []*ast.Comment) []*ast.Comment { comments = w.insertComments(comments, body.Loc()) - offset := 0 for i, expr := range body { - if i > 0 && expr.Location.Row-body[i-1].Location.Row-offset > 1 { - w.blankLine() + // Insert a blank line in before the expression if it was not right + // after the previous expression. + if i > 0 { + lastRow := body[i-1].Location.Row + for _, c := range body[i-1].Location.Text { + if c == '\n' { + lastRow++ + } + } + if expr.Location.Row > lastRow+1 { + w.blankLine() + } } w.startLine() @@ -426,6 +506,8 @@ func (w *writer) writeExpr(expr *ast.Expr, comments []*ast.Comment) []*ast.Comme switch t := expr.Terms.(type) { case *ast.SomeDecl: comments = w.writeSomeDecl(t, comments) + case *ast.Every: + comments = w.writeEvery(t, comments) case []*ast.Term: comments = w.writeFunctionCall(expr, comments) case *ast.Term: @@ -434,7 +516,9 @@ func (w *writer) writeExpr(expr *ast.Expr, comments []*ast.Comment) []*ast.Comme var indented bool for i, with := range expr.With { - if i > 0 && with.Location.Row-expr.With[i-1].Location.Row > 0 { + if i == 0 || with.Location.Row == expr.With[i-1].Location.Row { // we're on the same line + comments = w.writeWith(with, comments, false) + } else { // we're on a new line if !indented { indented = true @@ -443,8 +527,8 @@ func (w *writer) writeExpr(expr *ast.Expr, comments []*ast.Comment) []*ast.Comme } w.endLine() w.startLine() + comments = w.writeWith(with, comments, true) } - comments = w.writeWith(with, comments) } return comments @@ -481,6 +565,27 @@ func (w *writer) writeSomeDecl(decl *ast.SomeDecl, comments []*ast.Comment) []*a return comments } +func (w *writer) writeEvery(every *ast.Every, comments []*ast.Comment) []*ast.Comment { + comments = w.insertComments(comments, every.Location) + w.write("every ") + if every.Key != nil { + comments = w.writeTerm(every.Key, comments) + w.write(", ") + } + comments = w.writeTerm(every.Value, comments) + w.write(" in ") + comments = w.writeTerm(every.Domain, comments) + w.write(" {") + comments = w.writeComprehensionBody('{', '}', every.Body, every.Loc(), every.Loc(), comments) + + if len(every.Body) == 1 && + every.Body[0].Location.Row == every.Location.Row { + w.write(" ") + } + w.write("}") + return comments +} + func (w *writer) writeFunctionCall(expr *ast.Expr, comments []*ast.Comment) []*ast.Comment { terms := expr.Terms.([]*ast.Term) @@ -527,9 +632,12 @@ func (w *writer) writeFunctionCallPlain(terms []*ast.Term, comments []*ast.Comme return w.writeIterable(args, loc, closingLoc(0, 0, '(', ')', loc), comments, w.listWriter()) } -func (w *writer) writeWith(with *ast.With, comments []*ast.Comment) []*ast.Comment { +func (w *writer) writeWith(with *ast.With, comments []*ast.Comment, indented bool) []*ast.Comment { comments = w.insertComments(comments, with.Location) - w.write(" with ") + if !indented { + w.write(" ") + } + w.write("with ") comments = w.writeTerm(with.Target, comments) w.write(" as ") return w.writeTerm(with.Value, comments) @@ -755,7 +863,7 @@ func (w *writer) writeComprehension(open, close byte, term *ast.Term, body ast.B } func (w *writer) writeComprehensionBody(open, close byte, body ast.Body, term, compr *ast.Location, comments []*ast.Comment) []*ast.Comment { - var exprs []interface{} + exprs := make([]interface{}, 0, len(body)) for _, expr := range body { exprs = append(exprs, expr) } @@ -872,24 +980,40 @@ func (w *writer) listWriter() entryWriter { // location: anything on the same line will be put into a slice. func groupIterable(elements []interface{}, last *ast.Location) [][]interface{} { // Generated vars occur in the AST when we're rendering the result of - // partial evaluation in a bundle build with optimization. For those vars, - // there is no location, and the grouping based on source location will - // yield a bad result. So if there's a generated variable among elements, - // we'll render the elements all in one line. - vis := ast.NewVarVisitor() + // partial evaluation in a bundle build with optimization. + // Those variables, and wildcard variables have the "default location", + // set in `Ast()`). That is no proper file location, and the grouping + // based on source location will yield a bad result. + // Another case is generated variables: they do have proper file locations, + // but their row/col information may no longer match their AST location. + // So, for generated variables, we also don't trust the location, but + // keep them ungrouped. + def := false // default location found? for _, elem := range elements { - vis.Walk(elem) - } - for v := range vis.Vars() { - if v.IsGenerated() { + ast.WalkTerms(elem, func(t *ast.Term) bool { + if t.Location.File == defaultLocationFile { + def = true + return true + } + return false + }) + ast.WalkVars(elem, func(v ast.Var) bool { + if v.IsGenerated() { + def = true + return true + } + return false + }) + if def { // return as-is return [][]interface{}{elements} } } sort.Slice(elements, func(i, j int) bool { return locLess(elements[i], elements[j]) }) + var lines [][]interface{} - var cur []interface{} + cur := make([]interface{}, 0, len(elements)) for i, t := range elements { elem := t loc := getLoc(elem) @@ -1190,12 +1314,16 @@ func (w *writer) down() { } func ensureFutureKeywordImport(imps []*ast.Import, kw string) []*ast.Import { - allKeywords := ast.MustParseTerm("future.keywords") - kwPath := ast.MustParseTerm("future.keywords." + kw) for _, imp := range imps { - if allKeywords.Equal(imp.Path) || imp.Path.Equal(kwPath) { + if future.IsAllFutureKeywords(imp) || + future.IsFutureKeyword(imp, kw) || + (future.IsFutureKeyword(imp, "every") && kw == "in") { // "every" implies "in", so we don't need to add both return imps } } - return append(imps, &ast.Import{Path: kwPath}) + imp := &ast.Import{ + Path: ast.MustParseTerm("future.keywords." + kw), + } + imp.Location = defaultLocation(imp) + return append(imps, imp) } diff --git a/vendor/github.com/open-policy-agent/opa/internal/compiler/wasm/opa/callgraph.csv b/vendor/github.com/open-policy-agent/opa/internal/compiler/wasm/opa/callgraph.csv index 2af55db7..43be6c5b 100644 --- a/vendor/github.com/open-policy-agent/opa/internal/compiler/wasm/opa/callgraph.csv +++ b/vendor/github.com/open-policy-agent/opa/internal/compiler/wasm/opa/callgraph.csv @@ -227,9 +227,9 @@ opa_to_number,opa_number_int opa_to_number,opa_atof64 opa_to_number,opa_number_ref opa_base64_is_valid,opa_value_type +opa_base64_is_valid,opa_boolean opa_base64_is_valid,base64_gen_decode opa_base64_is_valid,free -opa_base64_is_valid,opa_boolean base64_gen_decode,memset base64_gen_decode,malloc base64_gen_decode,free @@ -251,15 +251,19 @@ opa_json_unmarshal,opa_json_parse opa_json_marshal,opa_json_dump opa_json_marshal,strlen opa_json_marshal,opa_string_allocated +opa_json_is_valid,opa_value_type +opa_json_is_valid,opa_json_parse +opa_json_is_valid,opa_free +opa_json_is_valid,opa_boolean opa_runtime_error,opa_itoa opa_runtime_error,opa_strlen opa_runtime_error,opa_malloc opa_runtime_error,snprintf_ opa_runtime_error,opa_abort builtin_graph_reachable,opa_value_type -builtin_graph_reachable,opa_set builtin_graph_reachable,opa_array builtin_graph_reachable,opa_array_append +builtin_graph_reachable,opa_set builtin_graph_reachable,opa_value_get builtin_graph_reachable,opa_set_get builtin_graph_reachable,opa_set_add @@ -519,7 +523,7 @@ builtin_object_filter,opa_value_get builtin_object_filter,opa_object_get builtin_object_filter,opa_object_insert builtin_object_get,opa_value_type -builtin_object_get,opa_object_get +builtin_object_get,opa_value_get builtin_object_remove,opa_value_type builtin_object_remove,opa_set builtin_object_remove,opa_value_iter @@ -566,6 +570,19 @@ opa_sets_union,opa_value_type opa_sets_union,opa_set opa_sets_union,opa_set_add opa_sets_union,opa_value_free +opa_strings_any_prefix_match,opa_value_type +opa_strings_any_prefix_match,opa_value_iter +opa_strings_any_prefix_match,opa_value_get +opa_strings_any_prefix_match,opa_strings_any_prefix_match +opa_strings_any_prefix_match,opa_value_free +opa_strings_any_prefix_match,opa_strncmp +opa_strings_any_prefix_match,opa_boolean +opa_strings_any_suffix_match,opa_value_type +opa_strings_any_suffix_match,opa_value_iter +opa_strings_any_suffix_match,opa_value_get +opa_strings_any_suffix_match,opa_strings_any_suffix_match +opa_strings_any_suffix_match,opa_value_free +opa_strings_any_suffix_match,opa_boolean opa_strings_concat,opa_value_type opa_strings_concat,opa_malloc opa_strings_concat,memcpy @@ -937,24 +954,23 @@ opa_glob_match,opa_value_get opa_glob_match,operator\20new\28unsigned\20long\29 opa_glob_match,memcpy opa_glob_match,operator\20delete\28void*\29 +opa_glob_match,void\20std::__1::vector\2c\20std::__1::allocator\20>\2c\20std::__1::allocator\2c\20std::__1::allocator\20>\20>\20>::__push_back_slow_path\2c\20std::__1::allocator\20>\20>\28std::__1::basic_string\2c\20std::__1::allocator\20>&&\29 opa_glob_match,opa_builtin_cache_get opa_glob_match,opa_builtin_cache_set opa_glob_match,std::__1::basic_string\2c\20std::__1::allocator\20>::basic_string\28std::__1::basic_string\2c\20std::__1::allocator\20>\20const&\29 opa_glob_match,std::__1::__hash_iterator\2c\20std::__1::allocator\20>\20>\2c\20void*>*>\20std::__1::__hash_table\2c\20std::__1::allocator\20>\20>\2c\20std::__1::__unordered_map_hasher\2c\20std::__1::allocator\20>\20>\2c\20std::__1::hash\2c\20std::__1::equal_to\2c\20true>\2c\20std::__1::__unordered_map_equal\2c\20std::__1::allocator\20>\20>\2c\20std::__1::equal_to\2c\20std::__1::hash\2c\20true>\2c\20std::__1::allocator\2c\20std::__1::allocator\20>\20>\20>\20>::find\28cache_key\20const&\29 opa_glob_match,std::__1::basic_string\2c\20std::__1::allocator\20>::operator=\28std::__1::basic_string\2c\20std::__1::allocator\20>\20const&\29 opa_glob_match,glob_translate\28char\20const*\2c\20unsigned\20long\2c\20std::__1::vector\2c\20std::__1::allocator\20>\2c\20std::__1::allocator\2c\20std::__1::allocator\20>\20>\20>\20const&\2c\20std::__1::basic_string\2c\20std::__1::allocator\20>*\29 -opa_glob_match,cache\28\29 opa_glob_match,std::__1::pair\2c\20std::__1::allocator\20>\20>::pair\2c\20std::__1::allocator\20>&\2c\20false>\28cache_key&\2c\20std::__1::basic_string\2c\20std::__1::allocator\20>&\29 opa_glob_match,std::__1::pair\2c\20std::__1::allocator\20>\20>\2c\20void*>*>\2c\20bool>\20std::__1::__hash_table\2c\20std::__1::allocator\20>\20>\2c\20std::__1::__unordered_map_hasher\2c\20std::__1::allocator\20>\20>\2c\20std::__1::hash\2c\20std::__1::equal_to\2c\20true>\2c\20std::__1::__unordered_map_equal\2c\20std::__1::allocator\20>\20>\2c\20std::__1::equal_to\2c\20std::__1::hash\2c\20true>\2c\20std::__1::allocator\2c\20std::__1::allocator\20>\20>\20>\20>::__emplace_unique_key_args\2c\20std::__1::allocator\20>\20>\20>\28cache_key\20const&\2c\20std::__1::pair\2c\20std::__1::allocator\20>\20>&&\29 -opa_glob_match,std::__1::pair\2c\20std::__1::allocator\20>\20>::~pair\28\29 opa_glob_match,opa_string opa_glob_match,opa_regex_match opa_glob_match,abort +void\20std::__1::vector\2c\20std::__1::allocator\20>\2c\20std::__1::allocator\2c\20std::__1::allocator\20>\20>\20>::__push_back_slow_path\2c\20std::__1::allocator\20>\20>\28std::__1::basic_string\2c\20std::__1::allocator\20>&&\29,operator\20new\28unsigned\20long\29 +void\20std::__1::vector\2c\20std::__1::allocator\20>\2c\20std::__1::allocator\2c\20std::__1::allocator\20>\20>\20>::__push_back_slow_path\2c\20std::__1::allocator\20>\20>\28std::__1::basic_string\2c\20std::__1::allocator\20>&&\29,operator\20delete\28void*\29 +void\20std::__1::vector\2c\20std::__1::allocator\20>\2c\20std::__1::allocator\2c\20std::__1::allocator\20>\20>\20>::__push_back_slow_path\2c\20std::__1::allocator\20>\20>\28std::__1::basic_string\2c\20std::__1::allocator\20>&&\29,abort std::__1::__hash_iterator\2c\20std::__1::allocator\20>\20>\2c\20void*>*>\20std::__1::__hash_table\2c\20std::__1::allocator\20>\20>\2c\20std::__1::__unordered_map_hasher\2c\20std::__1::allocator\20>\20>\2c\20std::__1::hash\2c\20std::__1::equal_to\2c\20true>\2c\20std::__1::__unordered_map_equal\2c\20std::__1::allocator\20>\20>\2c\20std::__1::equal_to\2c\20std::__1::hash\2c\20true>\2c\20std::__1::allocator\2c\20std::__1::allocator\20>\20>\20>\20>::find\28cache_key\20const&\29,std::__1::__unordered_map_hasher\2c\20std::__1::allocator\20>\20>\2c\20std::__1::hash\2c\20std::__1::equal_to\2c\20true>::operator\28\29\28cache_key\20const&\29\20const std::__1::__hash_iterator\2c\20std::__1::allocator\20>\20>\2c\20void*>*>\20std::__1::__hash_table\2c\20std::__1::allocator\20>\20>\2c\20std::__1::__unordered_map_hasher\2c\20std::__1::allocator\20>\20>\2c\20std::__1::hash\2c\20std::__1::equal_to\2c\20true>\2c\20std::__1::__unordered_map_equal\2c\20std::__1::allocator\20>\20>\2c\20std::__1::equal_to\2c\20std::__1::hash\2c\20true>\2c\20std::__1::allocator\2c\20std::__1::allocator\20>\20>\20>\20>::find\28cache_key\20const&\29,std::__1::equal_to::operator\28\29\28cache_key\20const&\2c\20cache_key\20const&\29\20const -cache\28\29,opa_builtin_cache_get -cache\28\29,operator\20new\28unsigned\20long\29 -cache\28\29,opa_builtin_cache_set std::__1::pair\2c\20std::__1::allocator\20>\20>::pair\2c\20std::__1::allocator\20>&\2c\20false>\28cache_key&\2c\20std::__1::basic_string\2c\20std::__1::allocator\20>&\29,std::__1::basic_string\2c\20std::__1::allocator\20>::basic_string\28std::__1::basic_string\2c\20std::__1::allocator\20>\20const&\29 std::__1::pair\2c\20std::__1::allocator\20>\20>::pair\2c\20std::__1::allocator\20>&\2c\20false>\28cache_key&\2c\20std::__1::basic_string\2c\20std::__1::allocator\20>&\29,operator\20new\28unsigned\20long\29 std::__1::pair\2c\20std::__1::allocator\20>\20>::pair\2c\20std::__1::allocator\20>&\2c\20false>\28cache_key&\2c\20std::__1::basic_string\2c\20std::__1::allocator\20>&\29,abort @@ -963,7 +979,6 @@ std::__1::pair\2c\20std::__1::allocator\20>\20>\2c\20void*>*>\2c\20bool>\20std::__1::__hash_table\2c\20std::__1::allocator\20>\20>\2c\20std::__1::__unordered_map_hasher\2c\20std::__1::allocator\20>\20>\2c\20std::__1::hash\2c\20std::__1::equal_to\2c\20true>\2c\20std::__1::__unordered_map_equal\2c\20std::__1::allocator\20>\20>\2c\20std::__1::equal_to\2c\20std::__1::hash\2c\20true>\2c\20std::__1::allocator\2c\20std::__1::allocator\20>\20>\20>\20>::__emplace_unique_key_args\2c\20std::__1::allocator\20>\20>\20>\28cache_key\20const&\2c\20std::__1::pair\2c\20std::__1::allocator\20>\20>&&\29,operator\20new\28unsigned\20long\29 std::__1::pair\2c\20std::__1::allocator\20>\20>\2c\20void*>*>\2c\20bool>\20std::__1::__hash_table\2c\20std::__1::allocator\20>\20>\2c\20std::__1::__unordered_map_hasher\2c\20std::__1::allocator\20>\20>\2c\20std::__1::hash\2c\20std::__1::equal_to\2c\20true>\2c\20std::__1::__unordered_map_equal\2c\20std::__1::allocator\20>\20>\2c\20std::__1::equal_to\2c\20std::__1::hash\2c\20true>\2c\20std::__1::allocator\2c\20std::__1::allocator\20>\20>\20>\20>::__emplace_unique_key_args\2c\20std::__1::allocator\20>\20>\20>\28cache_key\20const&\2c\20std::__1::pair\2c\20std::__1::allocator\20>\20>&&\29,std::__1::__next_prime\28unsigned\20long\29 std::__1::pair\2c\20std::__1::allocator\20>\20>\2c\20void*>*>\2c\20bool>\20std::__1::__hash_table\2c\20std::__1::allocator\20>\20>\2c\20std::__1::__unordered_map_hasher\2c\20std::__1::allocator\20>\20>\2c\20std::__1::hash\2c\20std::__1::equal_to\2c\20true>\2c\20std::__1::__unordered_map_equal\2c\20std::__1::allocator\20>\20>\2c\20std::__1::equal_to\2c\20std::__1::hash\2c\20true>\2c\20std::__1::allocator\2c\20std::__1::allocator\20>\20>\20>\20>::__emplace_unique_key_args\2c\20std::__1::allocator\20>\20>\20>\28cache_key\20const&\2c\20std::__1::pair\2c\20std::__1::allocator\20>\20>&&\29,std::__1::__hash_table\2c\20std::__1::allocator\20>\20>\2c\20std::__1::__unordered_map_hasher\2c\20std::__1::allocator\20>\20>\2c\20std::__1::hash\2c\20std::__1::equal_to\2c\20true>\2c\20std::__1::__unordered_map_equal\2c\20std::__1::allocator\20>\20>\2c\20std::__1::equal_to\2c\20std::__1::hash\2c\20true>\2c\20std::__1::allocator\2c\20std::__1::allocator\20>\20>\20>\20>::__rehash\28unsigned\20long\29 -std::__1::pair\2c\20std::__1::allocator\20>\20>::~pair\28\29,operator\20delete\28void*\29 std::__1::equal_to::operator\28\29\28cache_key\20const&\2c\20cache_key\20const&\29\20const,memcmp std::__1::__hash_table\2c\20std::__1::allocator\20>\20>\2c\20std::__1::__unordered_map_hasher\2c\20std::__1::allocator\20>\20>\2c\20std::__1::hash\2c\20std::__1::equal_to\2c\20true>\2c\20std::__1::__unordered_map_equal\2c\20std::__1::allocator\20>\20>\2c\20std::__1::equal_to\2c\20std::__1::hash\2c\20true>\2c\20std::__1::allocator\2c\20std::__1::allocator\20>\20>\20>\20>::__rehash\28unsigned\20long\29,operator\20new\28unsigned\20long\29 std::__1::__hash_table\2c\20std::__1::allocator\20>\20>\2c\20std::__1::__unordered_map_hasher\2c\20std::__1::allocator\20>\20>\2c\20std::__1::hash\2c\20std::__1::equal_to\2c\20true>\2c\20std::__1::__unordered_map_equal\2c\20std::__1::allocator\20>\20>\2c\20std::__1::equal_to\2c\20std::__1::hash\2c\20true>\2c\20std::__1::allocator\2c\20std::__1::allocator\20>\20>\20>\20>::__rehash\28unsigned\20long\29,operator\20delete\28void*\29 diff --git a/vendor/github.com/open-policy-agent/opa/internal/compiler/wasm/opa/opa.go b/vendor/github.com/open-policy-agent/opa/internal/compiler/wasm/opa/opa.go index 8c1c9d90..10b33840 100644 --- a/vendor/github.com/open-policy-agent/opa/internal/compiler/wasm/opa/opa.go +++ b/vendor/github.com/open-policy-agent/opa/internal/compiler/wasm/opa/opa.go @@ -1,51 +1,27 @@ -// Copyright 2018 The OPA Authors. All rights reserved. +// Copyright 2021 The OPA Authors. All rights reserved. // Use of this source code is governed by an Apache2 // license that can be found in the LICENSE file. -// THIS FILE IS GENERATED. DO NOT EDIT. - // Package opa contains bytecode for the OPA-WASM library. package opa import ( - "bytes" - "compress/gzip" - "io/ioutil" - "sync" + _ "embed" ) -var ( - bytesOnce sync.Once - bs []byte - callGraphCSVOnce sync.Once - callGraphCSV []byte -) +//go:embed opa.wasm +var wasmBase []byte + +//go:embed callgraph.csv +var callGraphCSV []byte // Bytes returns the OPA-WASM bytecode. -func Bytes() ([]byte, error) { - var err error - bytesOnce.Do(func() { - gr, err := gzip.NewReader(bytes.NewBuffer(gzipped)) - if err != nil { - return - } - bs, err = ioutil.ReadAll(gr) - }) - return bs, err +func Bytes() []byte { + return wasmBase } // CallGraphCSV returns a CSV representation of the // OPA-WASM bytecode's call graph: 'caller,callee' -func CallGraphCSV() ([]byte, error) { - var err error - callGraphCSVOnce.Do(func() { - cg, err := gzip.NewReader(bytes.NewBuffer(gzippedCallGraphCSV)) - if err != nil { - return - } - callGraphCSV, err = ioutil.ReadAll(cg) - }) - return callGraphCSV, err +func CallGraphCSV() []byte { + return callGraphCSV } -var gzipped = []byte("\x1F\x8B\x08\x00\x00\x00\x00\x00\x00\xFF\xEC\xBD\x0B\xB4\x65\x57\x55\x20\xBA\xBE\x7B\xEF\x73\xF6\xD9\xF7\x9E\xAA\x54\xA5\x8A\xDC\x20\x6B\x6F\x79\xA3\x2B\x9A\x52\xB4\xB5\xE2\x0B\xF0\xA8\x75\x47\xDF\xAA\xDC\x14\x50\xA8\xA0\xAF\x19\xFD\x1E\x89\x1A\xBB\xDD\x37\x40\x55\xE5\xA6\x8C\xC3\xA2\xCE\x2D\x2A\x90\xF0\x33\xD8\x2A\x1F\x45\x09\x10\x0D\xD8\x44\xA0\x45\x7E\xCA\x4B\x44\x14\x68\x51\x71\x88\x43\x10\xD4\xA0\x68\xD3\x08\x18\x21\xF0\x90\x5F\x7A\xCC\xCF\x5A\x7B\xED\x73\xCE\xBD\x55\x37\x84\x8F\x48\x29\xB9\x67\xAF\xBD\xF6\xFA\xCC\x39\xD7\xFC\xAC\x35\xD7\x9C\xE2\xEA\xEB\x9E\x2C\x85\x10\xF2\x3D\x72\xE9\x2A\x3D\x99\x4C\xE4\xE4\x2A\x15\xFE\x2B\xAE\x32\x93\x09\xFC\x91\xF8\x3F\x39\xB9\xCA\x4E\xA8\x40\xE3\x7F\xF3\xC9\x84\x9F\xB1\x9E\x9C\x5C\x95\x51\x41\xAC\x29\x27\x57\x09\x39\x89\x15\xE1\x51\x5C\xA5\x26\xA7\xE5\xE4\x2A\x7D\x1A\x9F\xE5\x69\xEC\xEB\xB4\xB8\x4A\x9E\x92\xA7\xAE\x2A\x27\xDD\x3F\x39\xB9\xAA\x80\xBF\xA7\x92\xDF\xFD\x7E\x70\x20\xD0\x00\xF4\x0F\x7F\x07\xC9\xC7\xEA\x58\xA6\xAF\x79\xCA\xC9\xD1\x53\x8F\x5D\xFD\xA4\x1F\xB9\xFE\x27\xAE\x5D\xFF\x89\xA7\x3C\x4C\xC8\xE9\xA2\xEF\x12\x62\xBA\xE8\xBB\xC5\x60\xBA\xE8\xDF\x8B\x72\xBA\xE8\x7B\xC4\x10\x8A\x06\x50\x74\xF5\x8F\x3C\xF5\xC4\xBA\x30\xFA\x2D\xF6\x4D\x36\xFC\x93\x02\xFF\x8B\xFF\x84\x95\x12\x7F\x0A\x41\x05\x72\xA4\x2A\xAB\x94\x5D\xB0\x76\x80\xFF\x6F\xAD\x96\xD6\x5A\x21\xB1\xC2\xA2\x94\x42\x96\xF0\x60\xAD\x19\x19\x6B\xA4\x55\x0B\x0B\x0B\xCA\x2E\x8C\x46\x26\xB4\x4F\x6D\x5B\x49\xBD\x84\x86\xE1\x19\x7A\x1E\x73\x57\x02\xEA\xD1\x4F\x2C\x0F\xFF\x84\x1C\xD0\x08\x79\x88\xE1\x5B\x23\x73\x65\xE5\x0E\x69\x61\x78\x36\x57\x23\x2B\x77\x5A\x39\xB2\xA3\x91\xDD\xA9\x84\x94\xCA\xE6\x2A\x13\x56\x19\xA3\xAC\xCC\x95\x12\x72\x24\xB4\x95\x42\x59\x69\x8D\xD4\x42\x59\x6B\x2F\x18\x0C\xB4\xDE\xB5\x7B\xF7\x85\x62\x20\x17\x60\xF4\x40\x61\x42\x08\x5B\xE6\x42\x66\xB9\xCE\x06\xC3\x6C\x30\x30\x66\x20\x74\x0E\xE5\x5A\x0E\x06\x52\x0E\xC4\x48\x74\x23\x54\x46\x18\xF8\xA7\x4C\x6E\xAD\xCD\x73\x9D\xE7\xB9\xD4\x5A\xE4\x52\x0C\xB4\x90\x52\x0C\x32\x95\xED\xC9\x32\xAD\x75\x96\x65\x7B\xB2\x3D\x99\x2E\xCB\x32\xCB\xB0\x44\x6B\x59\x16\xA5\xDC\x33\x14\x02\x3A\x50\x03\x6B\x8D\x35\x0B\x52\xE6\x42\x48\x25\x84\x2C\xF3\xA1\xC8\x73\x51\x55\x30\x63\xAD\xB5\xCA\xF2\x5C\x9B\x52\xEE\x11\x5A\x68\xAD\x4D\xAE\x33\x5D\xEC\x95\x32\x53\x72\x71\xD1\x9A\xAC\xC8\x95\x51\x4A\xA9\x07\xD9\xC1\x40\x6A\xA1\xF7\x08\x25\xA7\xFF\x5D\x24\x61\xB8\x17\x2A\x25\x95\xB5\x85\x19\x56\xA2\x1A\x28\x01\x5F\x55\x76\xA0\x88\x06\x06\x83\xA1\x28\x07\x80\x3C\x65\xAD\x12\x39\x8E\x15\x5A\x2C\x01\x02\x42\x1A\x69\x06\x43\xEC\x29\xCF\xF2\x5C\x29\x83\x08\x81\xFF\x37\x7B\x64\x5E\xEC\xD1\xB9\x28\x73\x91\x1B\x61\x8D\x55\x6A\x58\x0D\xCC\xC8\x8C\x84\x55\x16\x00\x05\xE8\x91\xA5\x10\x62\x20\xA4\x92\x52\x29\xAB\x07\xC2\x0A\x25\x95\xB1\x66\x20\x80\xCC\xA4\x18\x4A\x25\xC5\x60\x68\x07\x66\x60\xAC\x19\x4A\xEC\x0C\x3E\x34\x56\x1E\x93\x57\x5E\x69\xB5\x14\x2A\x5B\x54\x13\xE9\x37\x36\x4C\x39\x11\xFE\xEE\xDB\xF2\x32\x7F\x8F\xDA\x99\x3D\xF9\x9A\x27\x3F\xF5\xC4\x4F\x29\x31\x06\x92\xBF\xE6\xE4\xD5\xD7\x3E\xE9\x47\xD7\x6F\x78\xD2\x53\xAE\xF9\x49\x71\xE9\x10\x8A\x9E\x7C\xF5\xB5\xD7\x3E\xF5\x47\xC5\x7F\xBC\xB0\xF7\xFE\xBA\x6B\xD6\x9F\xF4\x13\x4F\x39\x76\xFD\xBA\xD8\x5F\x84\x17\xE2\x3B\xB0\x8D\xFF\x72\xCD\xD5\xC7\x9E\x74\x6C\xFD\x04\xD4\x11\xFF\xF7\x22\x14\x9D\xBC\xFA\xDA\xEB\xAF\x79\xD2\xB1\xAB\x4F\x5C\x77\x8D\xB8\x62\xA1\x2B\xF9\xB1\xEB\x9F\x7C\x4C\x3C\xA1\x82\x82\xF6\xBA\xA7\x3E\x85\x9E\x1F\xBF\x7B\xA6\xA7\x1F\xBB\x7A\xFD\x6A\xF1\xB0\xA5\x99\xF2\x6B\x9E\xB2\x7E\xE2\xA7\x8E\x3D\xF5\x27\x9E\xB2\x2E\xBE\x6B\x4F\xEF\xED\x7F\xBE\x66\xFD\x49\x27\xAE\xB9\xEE\xFA\x6B\xD7\xC5\x77\x2F\xC4\xF6\x69\x04\x87\x71\xC4\x3F\x7E\xE2\x9A\x6B\xC4\x13\xCB\x27\xD1\x78\x7F\xE4\xEA\xEB\xAE\xD1\xB2\x3F\xFE\xFF\x7C\xCD\xBA\xF8\xE1\x9D\xDD\x68\xAF\xFE\xB1\x1F\x7B\xD2\xB1\xAB\xD7\xFF\x8B\xF8\x5B\xB9\xBB\x2B\x3D\x71\xCD\x93\x9F\x7A\xF2\x1A\x7A\xF1\x77\x72\xF0\x0B\x52\x0A\x2F\xCB\xD5\x37\xCA\xC7\x3C\xEE\x07\x3E\x25\xEF\x95\xCF\x55\xCF\x53\x2F\xD0\xBF\xA5\xDE\xA8\x5E\xA8\xDF\xA0\xDE\xA4\x5E\xA8\x7E\x51\xBD\x58\xBD\x48\xFD\x96\x7E\xA3\xFE\x23\xFD\x27\xFA\x0F\xF5\xBB\xF5\x1B\xF4\xFB\xF5\x5F\xEA\x0F\xE8\x0F\xEA\x5B\xF5\x19\xF3\x74\x73\xD6\x7C\x42\xDF\xA3\x3F\xA9\xEF\xD5\x9F\xD1\x9F\xD5\x9F\xD3\x5F\xD0\xAF\x32\xBF\x67\x6E\x33\xBF\x6A\xDE\x6E\x7E\xCD\xFC\xBE\xF9\x03\xF3\x0E\xF3\x37\xE6\xA3\xE6\xE3\xE6\x5E\xF3\x19\xF3\x05\xF3\x12\xFB\xCB\xF6\x57\xEC\xCB\xEC\x6F\xD8\xD7\xD8\x5F\xB5\xAF\xB0\xB7\xD9\xD7\xDA\x5F\xB7\xFF\xCD\xBE\xD2\xBE\xCA\xFE\x77\xFB\x9B\xF6\xF5\x76\xF8\xC5\x0F\x5D\xF0\x26\xFB\x0E\xA9\xD4\x44\x9E\xFE\xD6\x8D\x8D\x8D\x0D\xE1\xC7\x6B\x8D\x7C\x28\xFD\x14\xB5\x52\x07\xBB\xFF\x73\x62\xFC\x5B\x67\xA0\xFC\x54\xBB\x60\x84\x54\xDA\x96\x50\xB8\x4F\x99\x4A\x2C\x8B\x5A\x8F\x4C\x09\x9F\xC0\x4F\x7D\xD0\xE9\x65\x79\xAA\xD6\x4E\xFA\x51\x0B\x75\x0A\xA7\xDA\xF1\x6F\x3F\x63\x63\x43\x38\xD5\x36\x0A\x3F\x5B\xAD\xC4\xC8\x94\xA5\x13\xDF\xAB\x8A\xF1\x1F\x42\xD3\xB5\x1A\x69\x7C\x1E\xC5\x67\x35\xF5\x2C\x4B\xA7\xC3\x43\xE9\xA4\x1F\xB7\x34\x5A\xA7\xCA\x37\x6B\x69\x27\x5B\x4C\x23\x4E\xE0\xA7\xDB\x05\x0D\x6B\xA5\x1C\xFF\xEE\xD3\xA1\x21\x18\xA6\x38\xA0\x0A\xA7\xBD\x18\xFF\x3D\xB6\x26\x7D\xD1\x8E\xFF\x0E\xDE\xAA\x83\x4E\xEE\x53\xC5\x4A\x25\xFC\xBB\xCE\xDE\xB8\x21\xC6\x2F\x80\x52\x9E\x3A\x16\x8B\xDA\x78\x53\x2B\x7D\x90\xC1\xE1\x54\xBB\x4F\x85\xBE\xF4\xA1\x4A\x38\x3D\xFE\xE3\xA7\xF3\x50\x46\xA6\x74\x3A\xAD\xF6\x3F\xA1\xBF\xF1\x27\x36\x68\x20\xCA\x17\x6D\xAD\x9C\xF1\xB2\x6D\x0C\xF6\xB0\x5A\x89\x12\xE6\xFC\x91\x8D\x00\x80\xFE\xA8\x47\xBD\x51\x8F\x7A\xA3\x1E\xCD\x8E\x9A\xC7\xDD\x58\x1E\x79\x37\x6A\xE3\xD5\x3A\x8C\xA8\x51\x2B\x95\xD0\x07\x9D\x4A\x26\x71\xB8\x32\x4E\x53\xC9\xD4\x78\xF7\x29\xD3\xA8\x4A\x94\xD8\x6A\x6D\xCB\x30\x74\x3B\x3B\xEE\x14\x0C\x5F\x01\xD4\xC9\xAF\x11\xEA\xEE\xF9\xB2\x50\x27\xBF\xEA\xA8\xBB\xE7\x81\x45\xDD\xBF\x48\x69\x26\x5E\xD4\x72\x33\x4C\x31\xB0\x25\x7C\xEB\x4D\xAD\xA1\xAE\xA6\xBA\x72\xA5\x12\x4E\xD2\x1C\x34\x82\xF4\x8D\xF8\xED\xE4\xCA\x4A\x96\x49\x71\x0D\x23\x00\xF0\x02\x94\x25\xF3\x0F\x00\xEF\x48\x95\x58\x0F\xA0\x45\xC0\x31\x04\x1C\x15\x7A\x61\x7C\x05\x00\x69\x04\x50\xDA\xB9\x9E\xEE\x56\x87\x0E\x01\x3E\x3A\xC2\xC7\x94\xA1\x6F\x43\xF0\x91\x5F\x91\xA9\xCB\x2B\xBE\x26\x53\xA7\x6E\xCF\x7F\xEA\x2F\xEB\x4F\x7D\x6A\xF2\xA0\x37\xD1\xE0\xC6\x1F\x44\x56\x2D\x69\x84\x8A\x60\xA1\xBB\xD1\xE9\x30\x3A\x43\x94\x2B\x9D\x81\x41\xBD\x17\xBE\x82\x9F\xA6\x31\x71\x18\x0A\x20\x21\xDB\x46\x3B\x15\xE7\x3F\xFE\x53\x6A\x1F\x88\xF2\x0C\x92\xF7\x87\xA0\x00\x86\xF8\x6C\x35\x35\xC4\xE9\x41\x1A\x61\x4A\x1E\x57\x25\xBD\xAC\xF5\x48\x05\xDE\x50\x09\x7E\xC6\x51\xD6\x06\x06\xED\x4D\x8D\x88\xD4\x07\x9D\x71\x32\xE1\x13\xEA\x70\x45\x02\xAD\xE1\xF2\xFD\x42\xAE\x54\x0A\x19\x51\x2D\xA7\xD0\x56\x62\xC3\xB2\xDC\x64\x01\x4B\x04\x83\x47\x16\xD1\xEB\x81\x9E\xB1\x65\xFC\x69\x1A\x99\x00\x06\x1A\x0D\xCB\x57\x85\xE5\xFB\xAE\x08\x9A\xF2\xC3\x52\xEA\xED\xF1\xD7\x1E\x93\xD4\x9B\x33\x49\x75\x98\xC8\x8F\x4B\x71\x80\xC2\x4B\xEE\x1C\x79\x26\x33\x46\xC6\x5D\x64\x8C\x5E\xC4\x4A\x92\xBB\x1B\x41\x6F\x49\xA9\x93\xFE\xE9\xEA\xE1\x72\xC1\x01\x7F\x5C\x68\xC7\x6F\xC6\x2E\xC5\xE1\x50\x65\x8A\x15\x3D\x42\xCA\x09\xCD\x09\x29\x49\x8D\xDF\x06\xD5\x90\xEA\xA0\x05\xA7\xC6\xBF\x0D\x05\x44\xF2\x08\x24\x15\x1B\x2B\xBF\x8F\x3E\xD6\xEA\xA0\x53\x11\x18\x5E\x1E\xC1\x95\xA2\x9C\x48\xBF\x5D\xA9\x75\x04\x70\x79\x93\x94\x6A\x16\xB6\x0C\x55\x64\xB9\x8D\x40\x28\x8A\x4E\x00\xA8\x20\x00\xA0\xE1\x44\x00\x7C\x1A\xDE\x3A\x41\xCC\x36\x11\x04\x7F\xFC\xEC\x4E\x10\x38\xC5\x8C\x59\xF4\xA6\xFF\xC0\x0E\xE4\xA6\xB3\xF3\x07\xF2\xFC\xE7\x9C\xC7\x40\x7E\x7A\x1E\xB1\xCD\x19\x47\x2A\x11\x7D\x32\x16\xDD\x1F\xCB\x73\xD2\xB1\xE0\x48\x50\x2E\xB3\x74\x9A\xA2\x81\x07\xB8\xEF\x67\x6F\xA7\xEF\xB7\xF5\x11\xA0\x02\x40\xA8\xE3\x5A\x38\xC9\xBF\xA2\xC0\x40\x3E\x8C\x0C\x7F\xFC\x0F\xD4\x29\xFD\xED\x23\x09\x58\x7F\x18\x9C\x0B\x0A\x83\x82\xE1\x7D\xF6\xE9\xBD\xE1\xF1\x5F\x05\xB2\x7D\xE4\xEF\x7C\xD2\xF1\x06\x3E\xE3\x8E\xFC\x6B\x52\x22\xD2\x1D\xEE\x54\x8A\xBB\x77\xF6\x39\xC5\xF4\x1C\x74\x9C\x83\xC1\x19\x80\x50\x71\xC0\xBB\x79\xA8\x22\x0C\x55\x38\xED\x4C\x3A\xD4\x7F\xC1\xA1\xB1\x1E\xE1\xCC\xEC\x50\x35\x0D\x15\x5A\xF4\xB7\xA4\x43\x15\x3C\x54\x4D\x8A\x08\x01\xCA\xF0\x5F\x98\xF5\x59\xE4\x06\xBA\x37\x13\xFD\x35\xC0\xC6\xD9\xB3\xDB\xC0\xC6\xC7\x9E\x75\x1E\xD8\xF8\x93\xAD\xB1\xB1\xBD\x39\xE8\x6E\x0E\x28\xD0\xC2\x3C\x0C\xCE\xE3\xC3\xDD\x3C\x3E\x37\x9F\xAA\xF6\xA9\x51\x23\xBC\x39\x5E\xC1\x9C\xC4\xCC\x84\xFE\x29\xC5\x99\x09\x38\x9B\x42\xCA\x9D\xF7\x6F\x42\xA8\xE3\x8C\xDF\x0F\x6D\xE3\x33\xFD\xBC\x3F\xD3\x4C\xD0\xF5\x8C\xF9\xE8\xA2\x35\xBE\xD9\x04\x7E\x0B\x26\x40\x42\x26\x0A\x4D\x7B\x18\xC7\x94\x3E\x80\xC5\x0B\x92\xAE\x65\x05\x48\xCD\xC8\x53\x03\x7A\x3A\x4B\xFF\x36\xD1\x79\x40\x58\x9A\x19\x61\x19\x0D\x19\x51\x8B\xF8\xB1\xDC\xE4\x63\x01\x1F\xD3\x20\xE8\x63\x55\xBE\x5E\x29\x35\x51\x89\x91\xAF\xA3\x0A\x60\x36\x9B\x89\x86\x07\x95\x3C\x40\x65\x50\x1B\xC1\xC0\x7A\x0B\xC9\x50\xA7\x9C\x0E\xBF\x89\x21\x5C\xA2\x45\x63\x97\x27\x4F\xC0\x4D\x01\xEB\xF4\xB2\xB8\x4C\xC3\xEA\x71\x16\x0D\xFD\x26\xFB\x61\xE0\xA0\x2E\xBB\x4C\x0B\x97\xD5\xB6\x0C\x5F\x15\x4D\x16\xBE\xCA\xE8\xAB\x02\xBE\xCA\x9C\xA5\x0F\xEC\x65\xBA\x70\xB6\xCE\xA0\xA1\xEC\x69\xB7\x33\x5C\x0D\x76\x18\xBE\x77\xCF\x5C\x76\x37\x35\xF8\x41\x76\xBB\xD7\xEB\xDE\x9C\x00\x55\xD8\x19\x82\xB3\x4C\x40\xC5\xCA\x19\x7E\x6D\x97\xE5\xA9\xC6\x3E\x81\xD4\xA6\x0E\xD9\xA6\xFC\x51\xC2\xB5\x9C\x86\x10\xA0\x65\x4A\xB3\x05\x0D\x1A\xE8\x4A\xAF\xFB\x53\x2D\xEE\x80\xF4\x94\x7B\xEA\x54\xFB\x1B\x48\x91\x9F\xB4\x68\x79\x81\x6A\xF6\x59\x20\xA8\xCD\x16\xC0\xF6\xC9\x7F\x11\xCA\xDE\x1D\x64\x3B\xFF\xD2\xB1\x0C\x90\xAD\x1C\x20\x53\x1D\x44\xBE\xED\xC4\xF8\x63\xA8\x37\xFF\x13\xB7\x93\x3C\x8D\xFF\x19\xFF\xFB\x71\xFC\xEF\xA7\xF1\xBF\xB8\x28\xA0\x0F\x27\xC7\x9F\x9A\x2E\x71\x66\xA9\xDF\x98\x13\x50\xF4\xC9\xF9\xAD\x94\x1F\xFB\x3A\x9E\xF9\xA7\xB6\x9C\xF9\x3F\xF7\x4A\x68\x9A\x73\x00\xC0\x33\xA7\xD9\x5E\x4F\xFA\xA5\x4C\x35\x91\xFE\xEC\x58\x4B\x79\x77\x98\x6C\x18\xDF\x47\x43\x65\xD0\xD1\xFE\xB1\x6B\x1E\x85\x92\xEB\x8D\xB2\x4E\xC4\x25\x2A\xFE\x5F\xCF\x30\xBE\x77\x06\xAE\xF7\x9E\x27\x5C\xEF\xDD\x84\xA2\xFE\x42\x4A\x39\x2B\x5D\x52\xED\x2F\x00\x93\x39\x29\x4F\x90\x0C\x90\x1E\xDF\x93\x68\xB6\x74\x6C\xEE\x12\x5D\x2C\x4F\x7E\x38\xEA\x39\x92\xA4\x8A\xC1\x41\x2A\x60\x07\x41\xB0\x98\x76\xFC\xA9\x29\x2D\x5E\xED\x53\x68\x9B\x7F\x31\xD9\xCE\x71\x92\x51\xF9\x91\x14\x95\x61\x93\x65\x5A\x0F\x78\x81\x9A\xA7\xCB\xB0\xB9\xB2\x29\x39\x75\x33\x3A\x54\xC5\xF1\xE0\x5C\xD5\xC1\xF9\x13\x7C\x42\xD5\x1B\x86\x66\xCB\x2C\xD2\x24\x3E\xA4\x98\x20\x70\x6A\x92\x6A\x60\x2F\x6B\x82\xC6\x87\xE0\x83\x8B\x9C\xE6\x49\xCE\xA0\xAA\x6B\xBA\xDF\x82\xDC\xAC\x85\xF9\xD2\xF8\x85\xAA\x53\x27\xDE\x23\x66\x01\x43\xD3\x37\xA9\x54\x33\x24\xC8\x48\xB6\x86\x2D\x92\x7D\xED\xF8\x15\x1B\x1D\xD8\xE0\x0D\x0A\x57\x35\xBE\x6D\x83\xE0\x93\x96\x71\x55\x06\xA4\xDA\x2F\xF6\x35\x12\xFE\x08\x22\x7C\x89\xBB\x07\x06\xA0\x01\x2D\xFB\x1D\x2D\xAA\xC5\x3B\x48\x2C\xE8\xFD\x42\xF8\xC9\x75\x4E\xEC\x17\xA2\x31\xC7\xC1\x34\xF5\xC7\xDA\xFD\x02\xA4\x03\xFD\x35\xC7\x0F\xE3\x7E\x04\x09\xF2\x5A\xA0\x32\x80\x7B\x86\x93\xB6\x91\x2B\x95\x1A\xA1\xF5\x5C\x1B\x50\xB3\xDE\xD5\xE9\xBE\xEF\x11\x1D\x60\xDE\x62\xB4\x99\xC8\xD3\x29\xD1\xE8\x64\x35\x48\xDE\x42\x44\x4B\x59\x38\xD3\x36\x16\x86\xF5\x9D\x87\x2A\x09\xAB\x0D\x37\x12\x0E\xC5\x8E\x32\xA8\x12\x60\x41\x5B\x38\x99\xB3\x5E\xB6\xCE\xC0\x4C\x64\x4B\x2A\xC1\xCF\xD0\x26\x1A\xB4\x81\xF2\x38\x5F\x16\x3F\x58\x49\x97\x23\x68\x9A\xE2\x8E\x65\xFD\xCC\x27\x54\xC0\xFF\x68\x7B\x25\x03\x00\x15\x5E\x1E\xAF\x33\x1C\x53\x01\xD3\x1B\xD0\xB8\x80\x0E\x0A\xFF\x25\x79\xBC\xB6\xB4\x51\xE2\x14\x0C\x52\xFA\x1D\xAD\x9F\xF8\x8D\xD3\x2E\xBF\xFD\x7A\x97\x2F\xE7\x4F\x68\xC4\xD2\xE5\x02\xA4\xF9\xCE\xF0\x62\xF9\x86\x53\xCB\xC2\x89\xA5\x26\x56\x91\x58\x85\x5F\xC8\xA5\x3A\x77\xD6\x19\xAF\xDA\x06\x48\xA1\x04\x2A\xCC\x56\x50\x93\x31\xED\x4C\xFB\xF0\x69\x09\x83\x75\xC5\x4A\x25\x5D\xE1\xF5\x71\x60\x1B\xC8\xC6\x07\x5E\xAF\x56\xC2\x15\xFE\x0B\xF2\x78\x5D\xD0\xAE\x1B\x68\x65\x6D\x83\x00\xB4\xCE\x22\x36\xFD\x0E\xC0\xEA\xF1\xCB\xF1\xB7\x4A\xCA\x77\x26\xE5\x3A\x29\xBF\x20\x29\x37\x49\xF9\xAE\x58\x5E\x38\x09\x6F\x24\x8E\x1F\xB8\x38\x60\x4D\x21\xA5\x01\xAC\x8C\x3F\xD6\x36\xD2\x49\xFC\x8C\xBF\xC1\x39\xC0\xB8\x90\xAC\x26\x2D\xB0\xC0\x91\x22\xF4\xF6\xB4\x9E\xAC\xFC\xB9\xA1\x2C\x03\xDD\xB8\x1E\xDD\x00\xE5\xA0\xA2\x13\x76\xAF\x88\xB5\x3A\x03\x23\xF3\x3F\xD0\x2E\x54\x78\x08\xCA\xFF\x24\x2E\x80\xC9\x01\xB1\x03\xD8\x22\x8C\x01\x15\x42\x2F\x60\x0A\xBA\xCE\xBC\xA8\x73\x2F\xEA\x82\x88\x10\x68\x64\xB0\x52\xA1\x60\xF2\xAF\x7C\xCD\x8D\x1B\xC2\x0D\xFC\xDE\x75\xBF\xF7\x7A\x5F\x8E\x5F\x02\xCC\xA0\x19\x32\xD5\xBA\xDC\x8B\xC3\xCE\xFA\xFB\xE4\xF1\x66\xB0\x72\xBC\x1A\xB8\x21\x7D\xB1\xE6\x06\x7E\x78\x6D\xDB\xC0\xAB\x23\xD5\x00\x94\xC7\xCB\x05\x2C\xC9\x1C\x37\xA5\xEA\x9C\x48\x4E\x5F\x59\x0D\xE0\x09\xA8\xAF\xAD\x0B\xDE\x8D\x83\x4A\x85\x1B\xF8\xEF\x38\x5C\x15\xAE\xF0\xE6\x50\x55\xE0\x80\x33\x18\xB0\x74\x19\xD4\xC5\x31\x97\x61\x5D\xD2\x82\xAC\x78\x35\xC2\x37\x57\x54\x16\x04\xE0\x3E\x25\x0E\x08\x5C\xC2\x23\x5B\x3A\xB5\x3C\xB9\x0C\x61\x30\x86\x89\x2B\xFF\xA0\x16\x0A\x04\x2B\xB9\xAC\x21\x63\x0B\x08\x84\x7C\xA5\x32\x4E\xD4\x40\x97\xB9\xBF\x1C\xD5\xCD\xFD\x42\xC2\x2F\x0B\xD0\x3F\x54\x59\xEA\xBF\xA4\x5D\xD9\x1A\x6C\x6D\x98\x06\x22\xB1\xAE\x5C\x89\x83\x23\x58\xD3\x66\x30\xE0\x18\x80\xEC\xEF\x02\x20\x91\xE5\xAE\xF8\xC9\xE5\x04\xE4\x66\xE8\x77\x30\x98\x61\x0D\xDE\x4D\x35\xE9\xAF\x4B\xDE\x01\xDB\x73\x12\x40\x98\xB9\x81\x33\x6B\x2E\xF3\x66\xFD\x44\x9D\xC1\xD0\x80\xB6\x6C\x3D\x60\x78\xE2\xF8\x01\x45\xB9\xBF\x1C\xE4\x4F\x4E\x90\x05\x09\xD9\x08\x57\x1D\xA9\x0A\xA2\x85\x8B\x5D\xE9\x8A\x43\x55\xE6\xB4\xBF\xB8\xAD\x07\x4C\x15\x60\xBA\x0D\x91\x32\x4A\x00\x8A\xEC\x53\x86\x9C\xA2\x0C\x5C\xC0\x06\xA8\x82\xBA\x94\x2B\xC7\xAB\xA1\xCB\x02\x55\x48\xA4\x8A\x1C\xA9\x62\xE8\x06\x2E\x27\xAA\x30\x2B\x1D\xA7\x1E\xC2\x94\x86\xC8\x75\x87\x30\x93\x21\xB0\x20\xB3\x52\x0D\x9C\x84\x71\x03\x89\x99\x43\x15\x8E\xCE\x0D\x88\x22\x06\x50\x6F\x40\x8C\x18\x80\xDE\x94\xAE\xA0\x15\x39\xF4\xFA\xD1\x55\x46\x94\x70\xF1\x01\x21\x46\xBA\x64\xB9\x00\x58\x56\xF5\x00\xF0\xEB\xEC\x4A\x25\x47\x45\x89\x58\x5F\xA9\x72\xA7\x80\x3B\x48\x80\x4C\xEE\x94\xCB\x7C\xB1\xEE\x32\xBF\xF1\x25\x7D\xDC\x17\x27\x4F\x3C\x5C\xD0\xC7\x96\x3F\x96\x88\xDA\x0C\x65\x28\x80\x0F\xA9\x04\x06\xC7\xB4\x39\xA8\xAD\x33\xB5\x74\x79\x25\x91\xC5\x99\x8A\xCA\x14\xAA\x1A\x34\x0E\xE8\x0F\x31\x61\xE7\xF6\x07\xF5\xE1\x53\x8B\x7A\x85\xAB\x0E\x55\x9A\xA4\x92\xB3\x6B\xCD\x10\xF8\x00\xC2\xEA\x38\xAC\x60\x67\xFD\xE4\x3A\x2F\x9C\x58\xC3\xA1\xF8\xD3\x08\x19\x37\xF4\xA7\x8F\xAF\xE1\xAA\x11\x20\xFA\x9C\x00\x1E\xBE\xD8\xBA\x01\xFC\x42\x2E\x99\xEF\x17\xC8\x0F\x72\x62\x0B\xD2\x2F\xB4\xC8\xB3\x88\xFD\xE1\x9F\x0C\xDB\x6B\x40\xBA\x96\x5E\x3A\xB1\x06\x00\x2F\x60\x95\x54\x40\x80\xD6\x0D\x9D\x59\x6B\x1B\xC1\x5F\x09\xFC\x8A\x4E\xA3\x0D\x8F\x7B\x64\xCA\x28\xD7\x0E\x87\x05\x0B\x18\x1E\xA1\x05\x08\xAB\x71\xE4\x8B\x16\xE5\xD6\x65\xC8\x2C\x0D\x30\x45\xD7\x99\x82\x1F\x52\xD2\x3C\xC0\x9A\xC6\x3C\x9D\x42\x74\x2A\x85\x49\x54\x0A\x83\x84\x2A\xE7\xA9\x14\xCE\xD4\x16\xD7\x49\x4F\xB1\x90\xF3\x14\x0B\xC9\x8A\x85\x9C\x52\x2C\x2C\xF0\x31\xBB\x52\xE9\x11\xED\x89\x73\xEB\xD4\xCF\x8C\xCE\xB2\xAD\xA6\x0D\x34\x6D\x3A\x9D\x05\x98\xC7\x66\x3A\xCB\x05\xB4\x9B\x47\x3B\xEC\xBC\xB9\xBE\x3B\x29\xEB\xF6\xEA\xFB\xA5\x57\x72\xE9\xDE\xB4\x74\x72\x9D\x7F\xC8\xC9\x79\xD5\xE7\x97\xCA\x2B\xB8\xF4\x21\x60\xA6\x8D\xC7\xEF\x27\x8D\x7A\x59\x5C\xA6\x80\xA2\x8A\x96\x7F\x95\x03\xFC\xEA\x80\x12\xE5\x0B\xFB\xC6\x46\x1E\xC4\x24\xED\x86\xBD\x3F\xD5\xF5\xED\xF8\x7D\x34\x47\x67\xC6\xEF\x64\x4B\x2C\x87\x56\x46\x40\xF7\x07\x54\x01\x4A\x12\x58\x13\xB9\x13\x07\x94\x70\xF9\xF8\x35\x1B\xA8\xFD\xE6\xFB\x54\x51\x13\x4D\x64\xA4\xDD\xFF\x79\x6A\x2E\xBC\x37\xD8\x7C\x79\x6A\x2C\xEC\x14\x71\x54\xD0\xDC\xC8\x77\xA3\x36\xF1\xD7\xA8\xCC\x69\x1F\xA3\x7C\x8C\xF0\x13\x2F\xC6\xF8\xC9\x45\xF0\xD3\x8B\xF1\x99\xE4\xC1\x8B\xF1\xD3\x7B\x8F\x5E\x8C\xCF\x4E\x15\x78\x31\xBE\x11\x8B\xCA\x17\x6D\x7A\x08\x31\x75\xD2\x34\x41\x2F\x12\xA3\x74\xB9\x2C\xD8\xA1\x43\xA0\xF7\xC7\x77\x09\x19\x9F\x55\xE9\xBB\xBD\x33\xDE\xC4\x01\x55\xF3\x16\xB6\x5A\x62\xF1\xF8\x7F\x84\x13\xA1\xC8\xE0\xA6\x4F\x25\x9E\x32\x6F\x60\x53\x3B\x79\x86\x96\x5B\xDA\xDB\xA8\xF5\x1B\xAF\x05\x5C\xBE\x21\x35\xB2\x5E\xC4\x5B\xD3\x9B\x1F\x46\xDD\x68\x65\x11\xFA\x7B\xBE\x5A\x6B\x4C\x60\x25\x7E\xDC\xFA\x0D\xE9\x37\xD4\xF8\x56\x34\x71\xBC\xAE\x2D\x2A\x6D\xE3\xD6\x69\x67\xDB\x26\xF3\x4F\x83\x35\xD5\x3A\xEB\x9F\xD6\x12\x1F\x1C\xB7\x2E\xF3\xA7\x43\xE9\xE9\xB4\x74\x12\x4A\x27\x49\x29\x15\xC1\xB3\x45\x15\xD2\xFF\x1E\x71\x4F\x50\x0A\x90\xAB\x3E\x26\x9E\x02\x03\x03\xCF\x83\x6A\x47\x1C\x0B\xF8\xB5\xA5\xE3\xB7\xD3\xA4\xD9\x62\x89\x3E\xE8\x32\x6A\x5F\xC0\x30\x81\x29\xA0\xDE\xB7\x21\x0F\xB7\x3C\x7C\x2F\xDB\xB4\x14\xD4\x66\x67\x51\xE9\x0D\xAA\x36\x68\x62\x69\x33\xFD\xEA\x34\xC2\x0C\xF5\x3C\x97\xAD\x79\x7D\xBC\xD1\x2E\x6B\xBD\x3A\xE9\xF5\xB5\xE3\x17\x06\x15\x04\x44\xD3\x50\x81\x40\x01\xA9\x4B\x5A\x61\x01\x65\x9E\x4E\x73\xAC\x3E\xE8\x1F\x09\x3A\x8F\xB3\x4E\x1E\x45\xE9\x85\x3D\xA1\xBC\x54\x07\x03\xE8\x40\x53\xC8\x5A\xD4\xB9\xFC\x86\x3C\x84\x2A\xC4\xA8\x75\xBA\x45\x05\xC1\x15\x2E\xF3\x8F\x3C\x44\x9A\x95\x6C\x1B\x4D\xDC\x7D\x08\xE2\xA9\xF2\xE6\x24\xFC\x1D\x79\xB5\x7E\x02\xEA\x62\xE9\x82\x57\x27\xF9\x2D\x94\xCA\x50\x9A\xAD\xC3\xDF\xC5\x13\x97\x0B\xE5\x8A\x4A\x82\x7A\xD0\xD6\x43\x57\x00\x0B\x2C\xD1\x90\x02\xF8\x00\x77\x2F\x79\x91\x80\x2D\xB4\xA0\xA4\x12\xA5\x1B\x10\xAD\x21\x4A\x34\xA8\x16\x8A\x95\x13\x52\x53\x4A\xA7\xDC\xD0\x0D\xD6\x80\x6D\x0C\xC8\x89\xE3\xF9\x2A\x50\xA1\x2D\x9F\xFC\xC0\x10\x7D\x3C\x32\x73\x74\x08\x36\xFE\xCB\xB9\x34\xFF\xE5\xF6\xF6\xE6\x6D\xF5\x76\x93\xA6\x03\x7E\x83\xB4\xFB\xA5\xFB\xEE\xBB\x4F\xE3\x69\xAD\x57\xEB\x5E\x3F\xC5\x5B\x30\x9D\x24\x58\x68\x96\x29\x27\x0B\xF2\x55\x7A\xFD\x58\x20\x43\xD2\x01\x32\x32\x9A\x40\x3D\x01\x5A\xB0\x5E\x9D\x6C\xA3\x06\xA2\x61\x51\x0A\xAF\x98\x4C\x1E\x75\xBC\x7B\x25\x71\xBD\x7A\xB3\x8E\x32\x8F\x3E\xDD\xBB\xEE\x2F\x3E\x79\xA2\x57\x4D\x51\x35\xB5\x4E\xAA\xF6\xB7\x4C\xBD\x36\x64\x91\x69\x32\x28\x9E\x06\x0A\x92\xBA\x92\x17\x0B\xAE\xD0\x79\x03\x4B\xD6\x2B\xBE\x87\x51\xF8\x87\x71\xAB\xD2\x3F\x92\x04\x84\x89\xE3\xDB\x2F\x64\x23\xA6\x07\x27\x69\x66\xEB\xFE\x11\x58\x00\x0C\xD3\xF8\x47\x5E\x2E\xB4\x33\x4E\x00\xAD\xD2\xE0\x4A\xD6\xC3\x40\x2F\x61\x3B\xD9\x65\x40\x6F\x08\xB9\xD2\x99\x2F\x03\xED\xEF\xF9\xAA\x12\xD9\x7B\xEE\x07\x91\x5D\x9A\x6E\xDB\x26\x4A\x5E\x27\x79\xDE\xD1\xED\xB6\x3E\x94\x36\xC8\xDE\xDB\xF5\x80\x1B\x9B\x62\xFC\x32\xA0\x3F\xEA\xA1\x7C\x4B\xA2\x37\xDC\x23\x3A\xC1\x20\x5F\xED\x8C\xBF\x4B\xB4\x7E\x38\x7E\xDE\x19\x94\xFE\xEA\xD5\x24\x2D\x62\x81\x18\xDF\xCC\xE7\x3C\x77\x89\x96\x7E\x13\xE7\x0D\xBF\x35\xFF\x00\xE2\x97\xA4\xC4\xD4\x80\x2F\xCD\x07\x60\xC0\x25\x0C\x7E\x01\x4A\x87\xA1\x86\x40\xF1\x50\x4E\xFA\x5B\x9E\x71\x23\x1E\xAC\x3E\x9B\x76\xDE\x14\x69\x2F\xCE\xF8\x7B\x82\x92\x56\xDE\x64\x65\x3E\x49\x40\x91\x1D\xAA\xC4\xF8\xAF\xCF\xE0\x7E\xED\xF8\x03\xF1\xA4\x4B\xC6\x85\x36\xE3\x11\x25\xA3\x47\x94\x3E\xE7\x39\x16\x1F\x82\xC9\xC4\xE1\x49\x26\x0E\x70\xE4\x46\x24\xE7\xB9\x11\xA9\xF9\x6E\x44\x92\x1D\xDD\xB8\x61\x72\x74\xA3\xF1\x23\x86\x55\x3C\x64\xCB\xC8\xDB\x05\xB7\x6E\x5D\xE6\x35\xB4\x6E\x9A\x9C\x9C\x40\xD0\x95\x8A\x39\xB5\x9E\x99\xA1\xA6\x19\x76\xBB\x6C\x96\x26\x8A\x7B\xAE\x3A\x4E\xB5\x29\xC8\x9D\x05\x57\x54\x41\x43\x2D\xC3\xCC\x49\x26\x60\x53\x30\x73\x8D\xBA\xB3\x8D\x00\xD3\x9B\x00\x6C\xEA\x33\x00\x18\xFC\x6E\x0A\xDE\xD7\xE4\xF1\xE0\x58\xB0\x0D\xDB\x87\x9B\x8A\x0E\x58\xDD\xD0\x3A\x40\x96\x09\x24\xA1\xD9\xBA\x08\xC2\xCB\x85\xEE\xB6\x6C\x7D\x33\xAC\xCC\x69\x0B\x30\xED\xF2\xF1\x9F\x50\xB7\x19\xBC\xC8\x10\x3B\xC1\xFF\xEC\x0E\x23\xB3\x89\xC3\x65\x38\x6E\xE4\x01\x55\x10\xB9\x01\x9B\x54\x7E\x3F\x2E\x50\xDC\x91\x80\x77\x63\x72\xA7\x82\xBA\x62\x0D\xD7\xEC\xD1\x8A\x76\xC7\x6A\xC5\x5B\xB8\xA0\x0A\xF8\x87\x4D\x7D\x46\x3B\xD4\x8A\xB6\x1A\xC6\xCF\x26\x88\xD0\x92\xC6\x5E\x7B\x0D\xA2\xF3\xDA\xA5\x42\x50\xBD\x1A\x4F\x76\xC7\xB5\x74\xC1\xF9\x28\x34\xDA\x1B\xC7\x2A\xF1\x79\xC5\x87\x85\xA2\x31\x54\xDE\x00\x65\x82\x89\xCF\x07\x13\x34\xAD\xEF\x98\x1A\x1F\x61\x6B\xAD\x51\xCE\x6E\xDA\x3F\x4C\x20\x38\x1A\x42\xF3\xF4\xC2\xD4\x16\x7D\xDB\x66\x86\xD5\x60\x73\x71\x08\x28\x84\x94\x8B\xA3\x81\xF5\x1B\xC0\xE6\x94\x77\x27\x10\x6E\x7F\x2F\x0E\xC5\xD1\x86\x36\x55\x00\xBB\x82\x11\x66\x30\x42\x19\x4C\x5B\xE1\x9F\x00\x8B\x45\x0A\xAE\xAE\xA8\x7A\x52\x55\xE1\xB9\x2F\x4D\x4A\x75\x93\x92\x34\x29\x05\x84\x8C\xE7\x2E\xA1\x27\x3A\x48\x4F\x81\x0A\x6F\xC1\xF4\xB1\xE8\xA9\x55\xBE\x4D\x4B\x1B\xD9\x38\x20\x0F\x87\xD1\x10\x40\x43\x33\x05\x7D\x14\x5A\x6B\x34\x23\x03\xB0\x4B\xDF\xC6\x2D\x4F\x45\xF2\xF8\x83\x38\x73\xEB\x9B\x43\x15\x50\xBB\x3B\x0A\x8C\x0D\x59\x3E\xB6\x69\x09\x08\xD2\x59\xA7\xD7\x9C\x39\x5A\x91\xCF\x0C\x6E\xB5\xFF\xBF\xED\xC2\xE3\x45\x96\xFC\xEB\x3D\x9C\xEB\x9F\x88\xFF\x15\x49\x89\xC8\x84\xCC\xA8\x77\x86\x69\xD8\x34\x48\x8A\x9C\x71\x6A\xCD\xE9\xD6\x9B\x2B\x2A\x43\xD0\x7D\x2E\x5A\xDC\x15\x9E\x92\x8F\x2F\x15\x72\xBA\x40\x4D\x17\xE8\xA4\x00\x8B\xC0\xC8\xC0\xC6\xC9\x5D\x12\xE1\x57\x6B\xEC\x5D\x96\x29\xA6\xBC\x74\xD2\x59\xFF\x25\x71\x64\x09\x44\xA6\x42\xC0\xA0\x5B\x24\xC3\x8D\x7C\x87\xA0\x3E\xD4\xCE\x7D\xE6\xE4\x12\x09\xD7\xB3\x16\xD4\x3D\x99\xE0\x70\x16\x4D\xFA\xA0\xDA\xE4\xFF\x70\xA6\x8D\x04\xB8\xFF\x27\x53\x86\x7F\x83\x32\xFD\x37\x2C\xCF\xF1\x2F\x2F\x8B\xF0\x53\xF1\x5F\x01\xFF\xD1\x92\x1E\x6C\x99\xD1\x2E\x47\x0A\xE4\x91\x7F\xDF\xCF\xDE\x48\x47\x83\xE3\x67\x21\x13\x19\x75\x50\x83\x49\xAA\xC5\x99\x6F\x4A\xFF\xB9\x17\xF4\xBF\x29\xFB\xDF\xE8\xA9\x6F\xEC\x15\xD5\xD0\xBF\xEA\x85\xF4\x8D\xE5\x6F\x86\xE1\x1B\x8B\xDF\x18\xFE\x46\xEC\x53\xBB\x56\xAA\x41\xDA\xDF\xC0\x7F\xE4\x37\xFA\xFD\x0D\x02\x73\x55\xD4\x23\x41\xD0\x8C\x9F\x43\x88\x17\x78\x6E\xC4\x15\x10\x57\xF0\x5A\xD2\x6B\x5A\x80\x71\x99\x5D\x72\x18\x5B\x0B\x38\x5D\xC0\x43\xE4\xDF\x8D\xC7\xBE\xA1\xBC\xE8\x3D\x0D\x7A\x4F\xC3\xDE\x53\xD9\x7B\x1A\xF5\x9E\xAA\x45\x66\xA8\x81\x61\x3B\x09\xF2\x00\xB7\x4B\xEE\xA2\x3E\xF1\xED\x73\x3A\x76\x4E\xDB\x56\x04\xA9\x3E\x43\xE9\x68\x2B\xC8\x9D\x3B\x73\x99\x6D\x72\xA6\x26\x7C\x46\x0E\x33\xA4\x76\xE1\x59\x78\xF4\x0A\xA1\x83\x2C\xD4\xA1\xFD\xA4\xA5\x83\x23\xE5\x4F\x1D\x8F\x7B\xA3\xC6\x89\xB6\xC9\x81\x0D\xB7\x4D\x11\xF7\x44\x65\x8B\x67\x14\xF1\x59\xC1\xB3\xEA\x9E\x35\x3C\xEB\xF0\x9C\x39\x81\x9B\xEB\xB4\x09\x6D\xD9\xC3\xBC\x45\x32\x11\x6D\x4D\x38\x93\x5C\x19\xF7\xEB\xE3\x76\x1D\x6D\x01\xE2\x77\xC6\x29\xD6\x85\xC3\x59\x31\x6B\xFE\xBC\x33\x1F\xF4\x04\xB2\xDB\x3F\x18\x0E\x3D\x01\x07\xE3\x3B\x50\x3D\x2A\xFC\x04\x39\x2B\x35\x0D\xB3\xA8\x73\xE4\x03\xDE\xB6\x08\x83\x62\xFC\x6A\x46\x0E\x4C\x89\x5E\xE6\x60\x6C\xF0\xD9\x4A\x23\xC8\x8B\xD9\x29\x97\xAF\x05\x78\x12\x34\xA5\x33\xDD\x28\x78\x1C\x78\x78\xD9\x14\xC4\x8E\xE3\x82\x2F\xBC\xA3\xE3\xB9\x06\xD0\xEF\x5F\xD7\xF9\xDB\xD1\xE9\x22\x9D\x20\x08\x74\x7C\xCF\x5D\x41\x5B\xC7\xE9\x88\x51\xCA\xE1\xBC\x9C\xF6\xA3\x76\xFC\x9A\x64\x72\xFE\x35\xAF\x48\xDD\xF7\xF6\x01\xD7\xC2\x4F\x5D\xE1\xF2\xF1\x6B\x51\x09\xC7\x79\xCD\xE1\x44\xA0\xE7\x5C\x8A\x03\x46\x21\x90\x27\xFF\x7A\x0F\xE7\xFA\x47\x95\x65\x9E\xE7\x8A\x4B\x74\x9E\xE7\x26\xB7\x59\x1E\x66\x34\x82\x5F\x1E\x7F\x65\xF0\x6B\x04\xBF\x2C\xFC\x1A\xD2\x1E\xB9\xCB\x7D\x75\x39\x1E\x50\xB8\xDC\x0F\xE0\x57\x37\xEB\x1E\x36\xFD\x1F\xDF\x9C\xBA\x64\xFA\xAC\xAD\x2D\x81\x39\xE2\xD2\xD6\xC1\x91\xC2\xD9\xF8\xB5\x9D\xFB\xF5\x08\x60\x55\x38\x3B\xFE\x0D\x24\xB4\xA2\x9C\x06\x9C\x2A\xD3\x4F\x88\x78\x54\x87\x1B\x5C\xF1\x57\x90\x66\xDC\xAD\xB8\x12\x68\x03\x08\x38\x77\x66\x2D\xD0\x70\xEF\x70\x53\x94\xCF\x07\xC5\x71\xFA\x6A\x5B\x3C\xDB\x3C\xDD\x2E\x54\x42\x2A\x6D\x4C\x56\xD8\xA2\x28\xF2\xA2\x1C\xFF\x39\x99\x37\xC1\xA7\x7D\x31\x3A\xAE\x93\x22\x38\x6A\xD8\x05\x4C\xAF\x35\x26\x98\x5C\xB8\xC7\x84\x6B\xDF\xA4\x6B\x5F\xD3\x71\x39\x2D\x7F\x93\x2E\x7F\x85\x47\x23\x74\x42\x92\x27\x2B\x14\xE7\x1B\x9F\x55\x8B\x0B\x26\x3E\xEB\x16\x79\xC0\xE6\xCB\x5F\xD3\xF2\x57\xDB\x58\xFE\xCA\x99\xF1\x5F\x9C\x89\x1E\x38\x30\xB3\x1C\xE7\x09\x4B\xF1\xF7\x36\x18\x16\xB5\x0C\xE6\x9E\x0E\xDB\xBA\x6F\x43\xAB\x37\xF7\xD7\xB7\x0B\xCA\x0A\xDC\x3D\xC4\xF3\x7B\xE0\xAE\x96\xBC\x25\x9C\x70\xF9\xF8\xED\x58\x4F\xB2\x2A\xAD\x9C\x24\x83\x00\x44\xD3\x88\x34\xCE\xD0\x96\xF4\x83\xC3\x95\x18\xFF\xD5\x99\xC0\xB6\x85\x93\xFD\xCF\x03\x22\x93\xEE\x7F\xB2\x5D\xB0\x42\x6B\x21\x35\xF9\xA4\xE5\xE3\xDF\x4F\x3C\x8C\xFE\x80\x7E\xB3\xCD\xA7\x4A\xA7\xCA\x0D\xB0\xC7\xC3\xED\xAA\x5D\x2B\xC1\xA0\x05\xC3\x93\x0C\x10\xA7\xFC\xE0\x10\x2A\x9C\xDC\x4B\x18\x03\x89\xC1\x58\xAF\x37\xF2\x11\xEA\xC4\x30\x01\xBA\xD3\x38\xFD\x2D\x1F\x42\xA1\x76\xFA\xBB\x52\x2A\x1E\xC1\x54\x35\x60\xBB\x34\x7F\x18\x8E\x53\xE3\x77\xF4\x3B\x82\xD9\x1A\x25\x59\x95\x9E\x3B\xBC\x50\xB3\x3A\x3C\x67\x0A\x68\x9D\xCC\x6D\x79\x9B\x53\xF8\x7F\xBA\x3D\x0D\xD7\xB9\xF5\x2A\x27\x0E\xA8\xBD\x80\x63\xBE\x62\x00\x4A\x36\x3A\x4E\x5D\xEC\xD4\xB2\xB8\x4C\x8F\xD1\x09\x89\x3C\x91\x92\xB6\xF1\xF8\xC9\x75\x8B\x76\x5B\xAD\xCB\xED\xB6\xFE\xD7\x16\x6D\x49\xBC\x0E\x47\x76\x44\xAD\x83\x39\x09\x74\x66\x51\x96\x6B\x6F\x83\x47\x6E\x23\x1F\x53\xD1\x72\x96\xEB\x8D\xE5\xB3\xA2\xAC\x12\x7E\xB2\x58\x76\xFB\xEA\xFA\x38\xDD\x2C\x40\xC6\xC2\x32\xDF\xC3\x8B\x53\xC7\xD7\xEA\xDC\x4B\xBC\x49\x94\x39\xD5\x36\xF0\x1A\x3B\x76\xAA\xA5\xFD\x7C\x5C\xA1\xA1\x28\x65\x07\xA1\x56\x9F\x29\x84\xD2\x4E\x33\xA0\x63\x76\xA8\x78\x18\xE5\xA8\x9F\xB4\x78\x7F\x8A\x68\x02\x7A\x9D\x6A\x1D\x5D\x09\x40\xD0\x4D\x5A\xBC\xF9\x05\xF8\xB6\x07\x68\x8F\x0B\xCD\x49\x97\x1D\x50\x60\x2D\x65\x7C\x92\xB9\x86\x9A\x12\x5E\xEF\xF9\x07\x3A\x77\xDD\xA7\x8C\xFF\x10\xEE\x3F\x9A\x5A\xF9\x4F\x89\x5A\x7A\x5D\x6B\xFF\x31\x51\x67\x5E\xA1\x30\xF4\x86\x41\x69\xBE\x09\xCA\xF9\xA0\xBC\x17\x5A\xF0\xBA\x56\xFE\xD3\x00\x3F\x55\x6B\xFF\x49\x80\x9F\x44\x99\x07\xB5\xF3\xD6\x65\x01\xDC\x4E\xB7\x4E\xC6\x07\xD5\xFA\xBF\x27\x4C\x04\xB7\xF2\x03\xCA\x78\x51\xFE\x8E\xE9\x4E\xA2\x92\xA3\x6B\xF4\x00\x0C\x5B\x8B\xA4\xE4\xD1\xEE\x21\xA1\xC6\xD0\x25\x56\xB1\xD6\x36\x96\x50\x95\x11\xAA\x2C\xA0\x2A\x67\x54\x15\x80\x2A\xDC\x4D\x08\x47\x94\x19\x60\x6B\xE0\x05\xA8\x07\x78\x3C\x15\xB0\x95\x21\xB6\x60\x2A\x56\x1F\x04\xE1\xDF\x36\x26\x62\xCB\x26\xD8\x32\xA1\x28\xDD\x69\x0F\xB5\x64\x6F\x63\x3D\x94\xAA\x4E\x06\xD2\x81\x57\xC0\x16\x08\xB6\xDA\x96\x6E\x40\xD8\x82\x5E\xA7\x5A\x07\x6D\xAC\xB6\x6E\xC0\xDE\x67\x88\xAD\x3C\x62\xCB\x3A\xE1\x0A\xC2\x16\xEF\x24\xDB\xB5\xB6\x31\x80\xAD\x70\xFB\x45\xE3\x84\x69\x33\x10\xA7\x4B\xDB\x05\x78\x80\xA6\x41\xC6\x17\x7C\x78\x66\xA0\x73\x85\xF3\x0E\xAE\x0A\xD8\x60\x4B\x9E\x74\x71\x66\xA1\x54\xB5\x38\xC9\xA9\x52\xDD\xE2\xA9\x44\x52\x5A\x1B\x3C\x65\x33\xE1\x94\x2D\x4C\x36\xF6\xD7\x9F\x29\x7F\xD2\x9B\x30\x50\x12\x12\x0B\xC0\x2A\x3D\x95\xB7\xE5\xEF\x19\xDC\xF2\xED\x94\xA4\xFD\x42\xE2\xE9\x94\x90\xB4\x85\x74\x89\x2E\xC6\xFF\x83\x45\x2C\x6E\xB0\xF2\x79\xAD\x21\x3A\x42\xF7\x9B\x11\x6D\x03\xB6\x41\x57\xB2\x44\x4A\x12\x48\x29\x63\x52\xCA\xE3\xAA\x47\xD5\xC5\x02\x58\x0B\xDE\xB3\xB1\x09\x1D\x59\xA4\x23\xD4\x2B\xF4\x41\x97\x3B\x3C\x9A\x0B\x74\x24\x13\x3A\xD2\xA1\x28\x3C\x47\x3A\x92\x09\x1D\xE9\x48\x47\x32\xA1\x23\xCB\x1E\x76\x81\x8E\x60\xCA\xB5\x44\xB7\x13\xC1\xBD\x4E\xB5\xCE\x0A\x15\x7A\x31\x16\x0C\xD6\x2C\xD2\x91\x24\xAA\x02\x3A\xCA\xF9\x44\x62\xAD\x6D\x34\xAE\x7A\xB3\x52\x29\xD2\x13\x0B\xD4\x1B\x3B\x1A\xC2\xBD\x1B\xD4\x13\x73\x76\x2F\xD4\xD0\x31\xFA\x18\xEA\x94\x2E\x70\x42\xE1\xF8\x35\x2D\x55\x2D\x4E\x70\xAA\x54\x43\x69\x9F\x86\x34\xCC\x29\x38\x15\xE2\x56\xEE\x73\x9E\x19\xF5\x6F\xC0\x4B\x98\x7A\x1C\xC1\x94\x22\x49\x8D\xF4\xA6\xAF\x23\x0B\x7A\xE3\x45\x72\xB8\x99\x63\x76\x42\x29\xDD\xD5\xF9\x40\x21\x66\x86\x42\x6A\x36\x51\x3A\x22\x31\xEC\x84\x33\x4B\x24\x85\x97\xE8\xBF\x92\x3B\xDD\x82\x25\xC2\x38\xD3\x09\x91\xD8\x50\x94\x2C\x92\x58\x4B\x76\xA5\xAA\x2B\x8D\xE0\x2C\x58\x92\x05\x22\x01\x32\x04\x75\xC8\x04\x22\xD1\xED\x54\xEB\x3D\xAF\x98\x19\x22\xD1\x33\x44\xA2\x23\x91\x68\xDF\x74\xAC\x1D\x8F\x2F\x0E\x28\xDA\x4E\x94\xEC\x5D\xCE\x87\x16\xC1\x7F\x99\x0E\xB6\x07\x78\xC8\x3D\x24\x53\x78\xE8\x9B\x43\xF8\x87\x0D\x65\xB7\x35\xD8\x87\x53\x60\xCF\xBF\xA1\xC0\x3E\x3C\x3F\xB0\xBB\x41\xD4\x6D\xA0\x0D\x32\x70\xB7\x86\x5B\x39\x05\xB7\xEC\x1B\x0A\x6E\xE5\x79\x92\xEB\x07\xC5\x7C\x7A\x8D\x82\x64\x08\xC6\xCB\x45\x46\x59\xAB\xED\x9C\x7F\x52\x94\x48\xA9\x87\x2B\x73\x6E\x90\x4F\x93\xEA\xF0\x1B\x0A\xE4\xF7\x93\x54\xEF\x07\xDC\x06\xFF\x16\xE1\xE6\x3F\x21\xFA\x80\xD3\xDB\x07\x5C\xF1\x6F\x12\x70\x9F\x9C\x02\x9C\xDA\x3E\xE0\xFE\x4D\x0A\x15\x32\xF3\x36\x93\x2A\xFA\xBC\x00\xF7\x8D\x25\x55\xCE\x17\x70\x9F\xEE\x00\xF7\xB0\xCB\x85\x8C\xBF\x54\x90\x34\x9A\xDD\x59\x06\xD0\x84\xF2\x55\xEB\xB5\xBF\x73\x03\xCF\xB6\xD8\xB1\x65\x5A\xE7\x54\x3D\x70\x4F\x01\x7B\xA5\x32\xFF\x06\xC1\xEC\xD4\x7E\x51\x75\xFB\x0A\xFB\xC5\x42\x84\xB5\xC2\x5E\x85\x83\x9A\xC1\x31\x43\x76\xAE\x0C\xE7\x58\xFD\xD3\x9A\xBC\xFE\x86\x22\xE2\xFB\xA7\xC9\x93\xDD\x8D\x4E\xCB\x93\x99\x8B\x9F\xBB\x44\x2F\x72\xCD\x1F\xA3\x17\xD6\x7F\x35\xD2\x4E\xFC\xE9\xD9\x33\x8C\xE8\x6F\x35\x69\x17\x72\x21\xB5\x32\x99\xCD\x43\x94\xBE\xE0\xB0\x92\x6C\xBE\xA9\x64\xF3\x8D\xCF\x2E\x1A\x1B\xCD\x70\xF2\xF6\x0B\x9B\x6F\x32\x3D\xBB\xF4\x74\x7A\x49\xDB\x39\x60\x97\x5A\x30\x49\xD5\x5C\x33\x5C\x4D\x19\xCA\x6A\xAE\x19\xAE\xE6\x9A\xE1\xD9\x26\x66\x38\x6F\xBE\xD9\x4D\xCD\xF0\xDE\xE6\x9B\xE9\x99\xE1\x96\xF0\x62\x3B\x33\x1C\x1D\xE3\x9D\x4C\xF5\x80\x4F\x8B\x48\xF1\x1F\x43\x5F\xC8\xF8\x53\x07\xCC\xB1\x6D\xCB\x07\x0B\xEF\x4A\x0E\x19\xDE\x9D\xFC\xFE\xC3\xEE\xB7\xFF\x80\xF0\x7F\x25\x3C\x3A\xD3\x8F\xFF\x24\x9E\x45\x53\xDC\xA3\xA4\xC2\xD9\xB4\x02\x36\xF2\xA7\x49\x23\x9F\x17\xFE\x8B\xC2\xDF\xD8\xD5\xA1\x53\x8C\xCF\xE4\x32\x9B\x04\x4C\xDB\x6E\x11\x66\x84\xE9\x9C\x30\x9D\x01\xA6\x0B\xC6\xF4\x20\x62\x1A\x9D\xB2\x73\xC0\x74\xC6\x1B\x77\x79\x82\xE9\x1C\x31\x3D\xE4\x8D\xBB\x81\xB3\x6D\x93\xCF\xDD\xB8\xCB\x5D\x7F\x6B\x2D\x9F\xBB\x71\x97\xCF\xDD\xB8\x1B\x6E\xB2\x71\x97\x11\xA6\x07\x9B\x6E\xDC\x65\x80\xE9\x8C\x31\x5D\xF4\x36\xEE\x06\x84\xE9\x81\x8B\x1B\x77\x16\x31\x6D\x9D\x9A\xB3\x02\xF1\x38\x13\xE3\x36\xD8\x24\x70\x14\xF3\xA7\x6F\x42\xF6\xFC\x20\xEB\x2F\x9D\x86\x6C\xE9\x14\x7E\x22\x9D\x72\x66\x07\xF2\x35\x94\xAB\x82\xDC\xDD\xCF\x07\xAC\xEA\x1B\x17\xAC\xEA\x3C\x09\x56\xCF\x21\xD8\xBC\x74\x79\x39\x42\x0F\x36\x16\x0C\xB7\x68\x69\x27\x28\x06\x7E\x2F\x84\x64\xE9\xB3\x7E\x1B\x22\xD1\x35\x9A\xE0\xAB\xE6\xB3\xFE\x6E\x63\x5B\x31\xEB\xD7\x09\x7C\xF5\x2C\xEB\x7F\xE0\x77\x60\x37\x63\xFD\xEA\x5C\xAC\x3F\x89\xAE\x72\xFE\xAC\xFF\x53\x1D\xEB\xFF\xFB\x8E\xF5\xDF\xDB\xB1\xFE\x7D\xC8\xF9\xE1\xD7\x25\x97\x0B\x13\x50\x61\x53\x19\xC0\xAC\x39\x61\xDF\xE5\x19\x23\xF3\xCD\xAE\x52\xB3\x1B\x5B\x12\x63\xC7\x90\x9F\x52\x40\x67\x63\x56\xF0\x52\x88\xD3\x7E\x6F\xEB\xD1\x77\x5B\xFB\x71\x8B\xD7\x85\xF9\xF2\x30\x22\xA6\x68\x9D\x1A\xFF\x59\xC4\xE0\xD3\xE8\x42\xAB\x70\x76\xFC\x7E\x64\x63\x19\x35\x57\x1B\x97\x8D\x7F\x1F\x0F\xB4\xED\xF8\x2F\xE9\x2E\x52\x25\xE3\x12\x34\x33\x4B\xD0\x26\x4B\xB0\x7F\xDE\x63\xA6\xB7\x60\x33\xBE\x86\x31\xAB\xB8\x0D\xBC\x44\xA7\xE5\x02\x83\x0C\x44\x2A\x31\xB3\x8A\x9B\x99\x55\xDC\xCC\x5C\xC5\xCD\x24\x54\x32\xA0\x3B\xF2\x26\x50\x09\xE8\x63\xB5\x89\xAB\x10\x7A\x9D\x6A\x9D\xEF\xC0\xF7\x56\x61\x77\xDE\x63\x66\xCE\x7B\x4C\x38\xEF\x71\xC6\x5F\x3E\xBD\x0A\x67\x95\xB3\x99\x8B\xA3\xBC\x26\xFD\x46\x10\x0E\xA8\x56\x09\xBF\x51\xF4\xB4\x45\x45\x9D\xAA\xA8\x2D\x72\xEB\xD4\x6A\xB7\x92\x55\x87\xA6\xE4\x04\x75\x4A\x85\xEB\xDC\xCF\xCF\x75\x78\x9A\xF1\xE1\xA9\xFD\xCA\x1C\x9E\x66\x5B\x1F\x9E\xDA\xF3\x3B\x3C\x35\xBD\xC3\xD3\xFE\x22\x4E\x0F\x4F\xC5\x2C\x7A\xF6\x29\x76\xD3\x11\x44\xF2\x5E\x94\x8F\xEE\x39\x13\xC8\x70\x18\x1A\xD7\x98\x8C\x6B\x4C\xD2\x05\x7D\x09\x2B\x4C\xD0\x0A\xC3\x4B\x3A\x89\xEF\xC0\x63\x36\x69\x6C\xDC\x2E\xC3\xCF\x71\xAF\x19\x79\x40\xED\xDD\xB2\xB5\x9B\x35\x60\x46\x1C\x50\xB7\xBE\xE2\x26\x18\xEB\x41\x8C\x51\xE3\x3F\x72\xEB\x4D\x1B\xE2\x80\xFA\xEC\xAD\x58\xE8\xEF\xA5\xC7\x7B\xE8\x11\x1A\xFF\x08\xBF\x79\x27\xBD\x79\x1F\x3F\xBE\x97\x1E\xDF\xD3\x55\x7C\x27\xBF\x79\x25\xBD\x79\x13\x3F\xBE\x9E\x1E\x5F\xDB\x55\x7C\x25\xBF\xB9\x99\xDE\xBC\x88\x1F\x7F\x9E\x1E\x9F\xDF\x55\xBC\x99\xDF\x7C\xE4\xA5\x34\xC6\x97\xF2\x18\xE9\xF1\x9E\x97\x76\x63\xA4\x9F\x4E\xF8\xF1\xFA\x01\x75\x1B\x4D\x10\x20\xFE\x85\x5B\xE3\xCF\x7B\xBB\x9F\x1F\xEC\x7E\xBE\xB7\xFB\xF9\x3B\xDD\xCF\xD7\x77\x3F\x5F\xD2\xFD\xFC\xF9\xEE\xE7\x17\x5E\xDA\xB5\xDB\x0D\xE3\x9D\xAF\x88\x3F\xEF\xEA\x7E\xBE\xA9\xFB\xF9\x5A\xF8\x59\x96\xC2\x8B\x7D\x84\x86\xF2\x5E\xF5\x8D\x87\x84\x48\x62\x5F\xB7\x18\x78\x1E\x3B\xEA\x51\xD8\x07\xB1\x4F\x7D\x18\x9A\x70\xE2\x28\x5D\x85\x87\x92\x1B\x6F\x8D\x25\x8A\x4A\x6E\xEB\x4A\x74\x2D\xFD\xDD\xF0\x08\x2A\xCC\x3E\xF5\x76\x7A\xB3\x4A\xE1\x2B\xEC\xBA\xBF\x1B\x9A\x43\x66\xA4\xFC\xAE\x96\x3C\x7E\x12\xEF\x7E\xB2\x7D\xC7\x14\x94\xD7\x69\x0C\x8C\x43\xFC\x57\xEE\x53\xC8\xBA\xF1\x94\x0D\x6F\x6D\x63\xE8\x50\x33\x32\xEC\xD9\x68\xA6\x3F\x34\x78\x8D\xA8\xE5\xAB\x31\x12\xDE\x64\x7C\xA9\xC2\x09\x30\x9B\x77\xE1\x55\x45\x6C\x97\x62\x13\x36\xDA\x65\x4E\xAC\xF9\xF5\x16\x39\x12\x74\x9C\x39\x4D\xFE\x4D\xE3\x96\xFC\x47\x24\x5D\xE7\xB2\xB8\x2D\x41\xC3\x30\x2D\x45\xDC\x8D\xC7\xF0\x19\xCC\x98\x1B\x60\x65\x93\x42\x13\xF6\xEB\x2B\xF4\x11\xD4\x7C\xF2\x4B\x91\xC1\x69\x2A\xA5\xEF\x16\x42\x23\x1D\xB9\xF9\xE3\x13\x2A\x8F\x00\x72\x78\x58\xE5\x2D\x9E\xF1\x49\xF4\xA0\x3E\x28\xC8\xEB\xF3\x1D\xBF\xD4\x79\x7D\x52\x43\x58\x1B\x26\x01\x33\xC3\x87\x92\xC1\xC7\xF3\x19\xC9\x92\x3C\xC4\xC2\x8B\x02\xDD\x07\xCA\xCF\x1B\x39\x98\x47\x0E\x7E\xBD\x6D\x14\xBA\x6A\xCF\x10\x86\x9E\x21\x0C\xDD\x27\x0C\xD3\x11\x86\x9E\x21\x0C\x14\xF8\xE3\x16\x2F\x4C\xEF\x6A\xEB\x0C\xA3\x52\xD4\x39\x5E\xB5\x6A\x0A\xBA\x0D\xE2\x64\x3D\x40\x04\x34\xD2\x29\xBC\x54\xE4\x0B\xF8\x42\xBA\xEC\x70\xB8\xDA\x13\xC2\xA0\xE0\xAD\x65\xC0\x8B\xA4\xDD\xDE\x01\x3C\x09\xD0\x6F\x47\xAD\xC3\x28\xCA\x03\x3C\x5B\xF7\x23\xC4\xB9\xF1\x7B\x99\x84\x8A\x43\x95\x8A\x38\x40\x8B\x46\x12\x0D\x0D\xF0\x53\x7B\xB8\xD2\xCE\x70\x48\x68\x14\x30\xB8\x0F\x05\x0A\x98\x1B\x00\x05\x81\x0D\x81\xDF\x3A\x89\xF1\x49\xD0\x67\x76\x80\x9A\x83\x13\xB8\x9D\x46\xCE\xDA\xA7\xDA\x46\xC2\x64\x90\x08\x07\x6C\x99\xF8\x1B\x5A\xB2\x4A\x24\x4A\x69\xDA\x5B\xD5\x38\xE7\xB9\x23\x95\x5B\x8F\x54\x6D\x6F\xA4\x18\xE2\x1B\x06\x66\x31\xE4\xC2\xEC\xA8\x6C\x1C\x55\x6F\x14\x62\xEB\x51\xC8\x6D\x8D\xA2\x2C\xCB\xEF\x96\x72\xE2\xF8\x3E\xA5\x23\xA2\xC3\x68\xFB\x68\xD4\xCA\xD5\xA5\xF1\xAF\x70\x50\x38\xD6\x32\x9C\x2C\xFF\x1D\x6A\x6C\xBE\x00\x65\xEB\x85\x1B\x3D\xE7\x67\xB5\xEE\x91\xC5\x41\x1B\xE5\x25\xE7\x51\x8F\xE2\x4D\xEC\x97\x6A\xE2\x83\x12\x89\x4C\x95\x5D\x43\x81\xC2\x61\xED\xBC\x07\x07\xCD\x71\x1E\xBE\x0F\x6A\x8B\x7D\x54\x58\x8B\xF8\x21\xAD\x32\xFE\x10\xF5\xC0\xF0\x21\xBE\x31\xE5\x1E\x41\x2B\x15\x0B\xF7\xA9\xD0\x6E\xF9\xAD\xC2\x89\x57\x87\xC0\x0C\x49\x05\xD2\x52\xD1\x7B\xB4\x7C\x62\x77\x35\xB7\xA7\x0D\x81\xC6\xF3\xEA\xF1\xFB\xD8\xCD\x57\xD0\x02\x8E\xDF\x63\x44\x87\x37\x9D\x89\xBA\x31\xB0\x1F\x8A\xDE\x90\xA8\x46\x6F\xE8\xC7\x95\x10\xC1\x92\xF2\x62\xBF\x78\x2F\xB4\x54\x09\xFF\x3E\xF8\x3B\x7E\x0D\x5D\x97\x07\x00\x7D\x18\x85\xCA\x17\xB0\xF8\xB5\x5C\x6C\x0E\xA8\x9F\xBF\x2D\xCA\xA5\x9B\xE9\x27\x45\xF5\x03\x7A\xB8\xF5\x36\x62\x2B\x18\x32\x56\x7A\x49\x5F\x23\x63\x8F\xE9\x19\x38\xEA\xBE\xFF\xD4\xD9\x1E\x7B\x93\x97\xD3\x48\x00\x77\x61\x0B\xB7\x2C\x78\x58\xF0\x03\x9B\x2A\x77\x86\x18\x21\x21\xD0\x5E\x59\x7E\x46\x29\xDD\xA5\x60\xB9\x4B\x6C\x76\xAD\x99\x83\xB0\x72\xB0\xEA\x47\x74\x76\x64\xF0\xC9\x92\x42\x77\x91\xBD\x61\x98\xB5\x76\x1A\xCD\x6E\xBA\x02\x8F\x16\x83\xD3\x44\xAF\x8D\x68\x7D\xD8\xA1\xE5\x69\x4A\xFF\x88\x76\xFC\xE6\x2E\xE6\xF7\x23\x66\x82\x8F\x73\xE8\x8B\x24\x82\x38\xF4\x7F\x89\x2E\x1A\x83\x1A\x6F\x71\x6A\xF9\xBE\xFB\xEE\xBB\x6F\xF1\x87\xE8\xF2\xE8\xED\x34\x7B\x6A\xF8\xEF\x82\x2F\xBF\x74\xE6\x32\x4D\x0E\x60\xE3\xD6\x3B\xFF\xCF\xBF\x84\x41\x9D\xE8\x80\xC6\x3B\x98\xE5\x47\x93\xB5\xC0\xD8\x51\x54\xFD\x7D\x53\x23\x2D\x93\xA1\xBE\x23\x5D\x41\x0A\x6F\x24\xFA\xDF\x7A\x66\x1A\x75\xD1\xDF\x25\xBA\xCB\xDD\xCF\xDB\x22\x62\x7A\x8C\x94\x1E\x22\x83\x73\x44\xF0\x7F\x88\xB3\xE6\x68\xE0\x21\x80\xB0\x8A\xAB\x83\xAE\x2B\x88\xF1\x5B\x02\x22\xBA\x48\xD0\xE4\x61\xC9\xD1\x4A\xA6\x23\x94\x3C\xED\x01\x18\xCD\x66\xA3\x40\xBE\xB5\x65\xEF\xB7\xEA\x2E\xC2\xD1\xDC\xEE\xB7\x08\x99\x1E\xC9\x67\xD4\x8E\x6F\x3E\xDB\x0B\x19\x1F\x42\x7E\x53\x68\x47\xAE\xF3\xD1\x18\x2E\x34\x04\x96\x7F\xC7\x73\x7B\x44\x16\x42\x67\x9A\x7E\x0B\x7C\xE5\x1A\xC3\x51\x72\x34\x2D\x9D\xC4\xE6\x6E\xC0\x10\x4E\x86\xF2\xE9\xA9\xE8\xF5\xBD\x4E\x42\x4A\x13\x8C\xF5\x24\x81\xDF\x8E\x9F\x7B\x16\xDD\x5F\xBD\x5C\xA7\xE5\x42\xDA\x47\x81\x3E\x74\x61\xE8\xCF\x3B\x8B\x00\x38\xC7\xD0\x3B\x92\xC5\x05\xE7\x8C\x1F\xA6\x03\xFB\x99\xB3\x5B\x0C\x0C\x3E\x09\x01\x66\x7A\xB1\x08\x5E\xA9\xCE\x81\x21\x8E\x7C\x13\x30\x93\x06\x94\x05\x42\x4D\x42\x63\x37\x53\x48\xDB\x12\x52\x22\x42\x4A\x95\x18\xFF\x79\xC8\x90\x12\xB3\x90\x12\x7E\xE8\x74\x0A\xA9\x4D\x91\xAC\x7B\x90\xE2\x68\x58\x18\x95\xEC\xCB\x85\x94\x28\xAF\x3F\xC7\x4A\x52\xF3\x33\x20\x6C\x05\x91\x0F\xCD\x81\xC8\x14\x7E\x4E\xDF\xCF\x5E\x41\x59\xB9\x2D\x4A\x2A\xEA\xFE\xB3\x53\xDD\xDF\xF2\xA2\x73\x76\xBF\x31\x37\xE0\x7D\x88\x50\x16\x42\xFD\xCE\x8D\x62\xCF\x3D\xAB\xB4\x67\xC5\x3D\xFF\xE5\x2F\xCD\xF6\x1C\xA2\xA5\x4F\x85\x63\x7D\x20\x01\xF0\x2F\x53\x00\xF8\xFC\x0B\xBF\x5A\x00\x38\x7B\xB6\x0F\x80\xDB\x7E\xF6\xBC\x01\xB0\x69\xCE\x81\x8F\x86\xED\x1E\xFE\x35\x13\xAF\x7F\x5A\x27\xF0\x58\x1B\x23\x7B\x46\xEE\x77\xCE\x91\xFF\xD3\xD3\xE7\x44\xB4\x4F\x66\xF2\xBC\x5F\xEC\xAD\xBF\x7F\xDC\x98\x97\x73\xE0\xCE\x64\xCF\x7A\xCE\x1C\x74\x9C\x83\x09\x21\x30\x30\x4A\x0D\xCD\x41\x27\xFC\x9A\x6F\x28\xC3\x1C\x4C\x27\x03\xC2\xD8\x03\x13\x41\xD3\xD3\x74\x2C\x05\xB4\x75\x27\x56\x97\x1A\xDB\x63\x2E\x78\x93\xC7\xFA\x21\x72\x88\xC0\x5C\x2E\x4A\xE6\x96\x32\x86\xB0\x37\xAD\xC0\xDE\xCA\x9C\xF5\x39\x45\x6D\xCD\xC3\x16\x75\xBE\x8A\x37\x15\x30\x04\xBC\xF5\x37\x90\xA3\x79\x0E\xA6\xC2\xF2\x99\x8D\x1B\x37\x9E\xBF\x71\x97\xB8\x4C\x02\x70\x6F\x68\xA7\x0A\xC6\xE8\xE4\xED\x72\xB0\x02\xF3\x18\x7F\x55\xB8\xCC\xE5\x5E\xAE\xB7\x78\x22\x22\xBD\x7C\xB8\xA4\x2D\x51\xBC\x62\x47\xD1\x56\xCB\xC0\x7A\x25\x8E\xCB\x59\x2F\xA2\x50\x50\x29\xAB\x9B\x37\x23\x8A\x89\xFC\xA2\x1E\xC3\x34\x7D\xE4\xDF\x3B\x85\xEC\x17\xBF\xB0\x97\x8F\xA1\x47\x13\x71\x24\x9D\x18\x48\x08\x28\x24\xE9\x10\x33\x04\xF4\xDA\x39\x11\xAC\xF1\x50\xA6\x23\x20\xF9\xB5\x5E\x04\xFF\x7C\x8E\x45\xF0\x47\x37\x9D\xC7\x22\xF8\x1A\xCF\xE1\xDE\x73\xCC\xE1\xBD\xCF\x3C\x8F\x39\x6C\x99\x0F\xA8\x17\x07\xFD\xDC\xD2\xF0\x33\x53\x04\x11\xB8\xF2\x73\x5F\xD0\xCB\x07\x14\x07\xD2\x93\xC6\x2F\xD6\xF3\x92\x7E\x24\x11\x01\xD3\xC4\x3C\xE8\x6B\x42\xE1\x0D\x39\x92\xB9\x0C\x51\xEB\x39\x0A\x6C\x8C\x50\x4F\x21\xC3\x75\xD9\xE5\x9C\x91\x04\xBE\x4E\xBD\x0C\x00\x7B\x79\xB2\x96\xF8\xE6\xAA\xE6\x50\x70\xF2\x0A\x0E\x7D\x23\xC6\xFF\x8B\x8E\xDD\x2A\xE1\x3F\xD7\xD3\x54\x9C\xE1\xA0\x35\x1C\xF6\xBC\x39\xCF\x9E\x9C\xF1\x93\x2B\x69\x1F\xE3\xFC\x7B\xF8\xF8\x36\x7B\x80\xF1\x97\x31\x56\x7F\xD9\xA9\x81\x9A\xE2\x6D\x4E\x05\xF1\xEF\x0B\xAC\x8F\x4B\x59\x4C\xC2\x36\x42\x2F\x47\x89\x0A\x69\xC8\x38\xD3\x0F\xC7\x2F\xE5\x48\x48\x76\x9F\x2A\xBC\x3C\xC4\xE1\x02\x4D\x9D\x3B\x19\x22\x42\x3B\xE3\x2C\x5D\x54\xA3\x28\x47\x1C\x32\x09\x23\x95\x15\xE1\x32\x2B\x6D\x18\x38\x8D\x11\xD8\x92\x30\x4A\x93\x76\x21\x33\x94\xCC\xB8\x84\xB7\x78\x7D\xBD\xAD\xF3\xE0\x6A\x86\xFD\x4E\xDA\x46\xAF\x72\xAC\x02\x4B\xE1\x9F\x68\x58\x74\x57\x9B\x37\x23\xA6\x13\xAF\x95\x4E\x96\x2F\x50\x52\x4F\x18\xFF\x32\xBD\x05\x9E\x26\x37\x45\x2B\xDA\x89\x5E\xA0\x2A\xD5\x25\xF4\x50\x49\xDC\xA5\xB9\xA9\xCD\x42\x66\x3E\x4A\x80\xE0\x9F\xF7\x6B\x80\x2E\x3C\x8A\x1D\xBF\x04\xFB\x48\x4B\x5E\x84\xF0\x98\x49\xB5\xD6\xEB\xCC\xDF\x72\x5B\xFC\xA0\xF7\xF1\x2F\x60\xC9\xAF\x27\x6F\x37\x36\xA6\xDE\x6E\x35\xD0\x12\xC3\x08\x97\x4E\x95\x3F\x25\x4D\x04\x0B\xEE\x47\x89\xCE\x05\x46\xE8\xB9\xD9\xF3\x94\xC0\x60\x9B\x9A\x03\xBC\x10\x9A\x6B\x72\x9D\x1E\x7F\x36\x92\x39\x5D\x23\x8F\xD4\xDD\x25\x65\x63\x8C\xFC\x9A\x56\x18\xA8\x1E\xC3\x23\xBB\xE4\x4A\xF7\x7C\xAF\xB6\x2C\x0B\xC4\x41\x86\x75\xE7\xEE\x46\x89\xB5\x95\x29\xA1\x11\xA6\x68\x95\xCC\x44\xAF\x54\x9C\x77\xD0\x69\x1A\x2C\x98\x32\x7C\x74\xFB\xB9\x70\xDE\x8E\xC1\x1E\x9C\x09\x24\x04\x75\xDF\x16\xE8\x77\x64\x62\xB0\x30\x76\xA2\x30\x2B\x95\x8A\x21\xAF\x44\x2F\xE4\x15\x9F\x70\x62\x46\x45\x83\x8B\x02\xA0\x30\xA7\x37\x13\xA2\x5C\xCD\x26\x5D\xE4\x80\x58\x86\x43\x80\x85\x1B\xF7\x81\x34\x97\x69\xE9\xA5\xD9\x14\x5B\x74\x88\x70\x19\xED\x01\x2C\x0F\xE9\xEF\xF8\xE7\xF0\xBF\x73\x3B\xE7\xC8\x5E\x9C\x62\x32\x5B\x96\xA7\x9A\x0C\x13\xF5\x3C\x3E\x46\xF4\x42\x0A\xF9\x13\xAD\x2C\xE3\x89\x57\xD9\x1B\xCF\xCC\xE4\x99\xC8\x0E\xC7\xB8\x68\x29\x6A\x30\xB9\xB7\x90\x3A\xA2\x85\x23\x4C\x60\x54\x01\x90\x24\xAB\x8D\x5E\x6A\x4C\x44\x95\x85\xB1\x71\x4A\x19\xBD\xC4\x13\xD4\xCE\xD2\x0C\x28\x1A\xB5\x8D\x38\xE4\x5F\xE3\xCF\x6F\xC4\x48\xE0\xCA\x59\x97\x05\x1C\x9A\xE8\xCF\x65\xF1\x08\x3A\xDE\xFE\xD7\x21\xA1\xAA\x01\x9D\x15\x29\x2F\xEB\x60\x99\x05\x4C\x5A\x3E\xAB\x9E\x87\xC9\xCF\x77\xC0\xD4\x1D\x26\x2D\x72\xC8\x14\x93\xCC\xBD\xD0\x4B\x16\xBA\xEA\x44\x90\xA3\x74\x93\x50\xB8\x2C\xEA\xDC\x9B\x1E\x35\x31\x3E\xF3\x2D\xF0\x39\x35\x04\xC6\xA7\xA5\x83\x88\x1C\xF0\x99\x07\x7C\x96\x3C\x4B\x18\x58\xF9\x66\xCE\xE0\x99\xEE\x2F\x66\x87\xFB\xE8\x3C\x94\xE6\xB5\x4A\x1F\xF2\xC3\x95\xEC\x16\x99\x4C\x17\x59\xC7\x2E\xD2\xE0\x7F\x52\x00\xBB\x20\x76\xDF\xAD\xBB\xDA\x50\x54\x07\xC3\xDB\xBF\x36\xC4\x97\xB4\x18\x58\x91\xA6\x22\xD3\x35\x88\xC9\xAC\xBE\xA7\x4B\xD5\x98\x8E\x1C\xB8\x11\x6E\x46\x53\x63\x82\x13\xA3\x1A\x8A\x50\xF0\xB9\x39\x89\xBB\xB6\x39\xDF\x8E\x6C\xBA\x19\x9B\xAD\x67\x6C\x38\x8F\x65\xA0\x98\x9A\x52\xA4\x70\x60\x09\x28\xE6\x66\xD0\x3E\x98\xE5\x5A\x31\x2E\x1F\xE0\x36\x04\xE5\xEB\x00\x26\x03\xC0\x64\x02\x30\xD1\x6F\xD5\xA9\xF2\xDF\xC7\xC4\x98\x7D\x80\xA5\xEB\x16\x21\xB7\xD1\x85\xC5\x78\xBE\x02\x99\xD0\x2D\x58\x31\x85\x64\xE4\x8D\x6F\x22\x2C\x22\x8E\xE8\xB7\xC5\xC0\x7E\x75\xE7\x35\x8F\xD1\xE6\x60\xD9\xA4\x40\xB6\x41\xEE\xDB\xD9\x52\x5C\xB0\x1B\x4C\x1E\x29\x3B\xEE\x73\xE3\x92\xF7\x7D\xA7\xF2\x21\xCB\x5E\x3A\xD5\x40\x8F\xF0\x29\x48\xED\xDF\x3E\x13\xB3\x96\xE1\xB1\x69\x02\x35\x0C\x94\xA8\xD3\x28\x91\xFD\x74\xC8\xAA\xFC\xA1\xF9\x50\x4C\xF0\x3E\x4B\x33\x4E\x8E\xFF\x65\x23\x62\x0A\x5E\xFE\xFF\x1B\x91\x0B\x13\xA0\x1F\xC8\x66\x3F\xDF\x35\xFB\xF3\x73\xD6\x77\x7E\xB8\xDF\x90\xE8\x8B\x33\xBD\x79\x62\x69\xDB\xC5\x87\xB4\xFD\xF8\x90\xF8\x38\x9F\xF1\xE9\x08\x43\x1D\x60\xF8\xF6\xF3\x19\x15\x0A\x06\xDA\x3B\x1D\x35\x1A\x90\xB5\xBA\x34\xFE\x9B\xC0\x6C\x38\x5E\x2B\x56\x22\xC1\x11\x88\x20\x08\x8C\x2E\xC7\xB0\xE9\x05\x07\x15\xDD\x24\x44\x9C\x04\xAD\x20\x91\xCC\x02\xFD\xAC\x44\x45\xF1\x2E\x67\xE3\x85\x3A\x55\xFE\xA9\x8C\xE1\x04\xA7\xE7\x41\x67\x43\x5D\xBE\xE3\x68\xD2\x6C\x92\xF3\xD8\xA1\x93\x2A\xED\x94\x74\x59\x75\xF3\xC3\x15\x99\x27\x36\xE0\x88\xDE\x3E\x93\xD6\x18\xA5\xD3\xC3\x92\x1B\x19\x2C\x96\x5D\xFB\x54\x6D\x9D\x5A\xA9\x4C\x39\x93\x3D\xD9\x59\xCA\x25\x1D\x33\x28\x83\xFE\x15\x84\x61\xF9\xBE\xED\x90\x4B\x10\x9C\xE7\x24\x98\x1E\x7D\x9C\x93\x3C\x70\x35\x8B\xCD\x57\xF3\xD6\x6D\xE2\xB2\x15\xB1\x4D\x11\x90\xF5\x45\x95\x78\x0E\xF4\xE6\x36\x33\x1D\xD2\x3B\xB6\x9C\x11\xC5\x73\xEC\x63\x8A\x74\x83\x2C\xFA\x71\xE6\x87\x2B\xE3\xB2\xE4\x81\x3B\xCA\x39\xC4\x6C\xC3\xBA\xFE\x00\x7B\xC1\x10\xB3\x83\x29\xBD\x31\x27\x7E\x45\x73\xEC\x34\x43\xC5\xA1\x52\x07\x30\xC7\x01\x87\x4A\x55\x07\x5D\x36\xD3\x6A\xB6\x8D\x56\xB3\x79\xAD\x3A\xC5\x14\x95\xAF\x54\xDA\xE5\xB5\x9A\x41\x1F\x14\x92\xEF\xC8\x14\x1A\x17\xE3\xC7\x5E\x94\xFF\x8E\x73\x28\x52\xF4\xA9\x16\xD5\x05\x20\xC1\x9A\x52\x66\x85\xDB\x36\xE5\xA3\x31\x06\x92\x53\x91\x63\xEF\x17\x18\xF4\x08\xD3\x64\x1D\xED\x3C\x93\x9D\x79\x4C\x25\x3C\xF4\xD1\x0F\x61\xD5\xB9\xFD\x7A\x51\x2E\x80\xE1\xF5\xB8\x16\xC3\x93\x0E\x57\xCB\x87\xC5\xF8\x4A\xFE\x64\xDB\x48\x7F\xE1\x11\xF4\x15\x51\x5E\x3A\xB9\xEE\x9F\xB5\xB1\xA1\x8E\xA3\xE7\xAD\x1F\x1C\x22\xE6\xF9\x6D\xBD\xEF\x9D\xF0\x77\x4E\xE8\x21\x5B\x3D\xE1\x84\x7F\x69\xF7\x54\xFE\xAD\x54\xFA\xB4\xC1\x58\xB4\xCB\x8F\xBA\xA9\xD1\xA7\x9C\x7E\x46\xAD\x9D\x7A\x35\x88\x45\x34\xE5\xD0\x7B\x63\xE3\x65\xE8\x66\x02\xE3\x9F\x80\x4E\x7F\xFA\x69\xB7\x4F\xB9\x3C\xDA\x5A\x3B\xBB\x2C\x9E\x50\xE1\xFE\xA5\xC0\x64\x98\x4E\xF9\xFD\x5D\x15\xCA\xC5\x45\x7E\x8E\x68\x22\x66\xD1\x32\x56\x75\xE6\x72\x02\x27\x3A\xE6\xA8\x43\x21\x22\x00\x56\xC3\x98\x62\x75\xE1\x72\xCA\x13\x75\xBA\x6D\x06\xD1\x81\xB6\xA0\x0E\x9E\x06\x6D\x91\x7F\x66\x50\x57\xF3\x60\x36\x3F\x5B\x2A\x79\x1A\xD1\x23\xBD\x7C\x2C\xE1\x62\x99\x15\xB0\xFD\x42\xF8\xFD\x87\x1A\xE3\xE0\x05\xE6\x82\xAA\x51\xB1\x59\xAB\xC1\x40\xD1\x1D\x2E\x6D\x00\xE8\x00\x10\x78\x7A\x31\xA6\x51\xD2\xCB\xC3\xD3\xCE\xDE\xB1\x7C\x9F\x3C\x7B\x6A\xF9\x71\xA7\x62\xA6\x33\xA2\xC2\x65\xE1\xF4\xD3\x00\x6C\x4B\x98\x25\xA9\xFC\x98\xB1\x72\xA2\x4E\xC1\xFF\x74\x7F\x44\xFF\x41\xE0\xBF\x7B\xEE\x0C\x3F\x1E\x15\x47\xA7\x97\x6A\xC3\xA5\x82\x6F\x7A\x50\xFA\x80\xC7\x56\x22\x29\x27\xAD\x06\xE3\x1A\x3E\xAE\x6D\x32\x1A\xED\x91\x4A\x3A\xCB\x95\x1E\x7A\xF0\xE5\x2E\x7B\xE3\xAD\xE8\xE2\x43\x9C\x94\xBC\x8E\x9D\x7D\x39\x45\x31\xA4\x5F\xD0\x32\x45\x58\xE0\xE6\x9A\x0C\xA3\xF7\x76\x5D\x91\xE7\x36\x85\xB7\x86\x31\xFC\xE2\x8B\xE1\xDF\x9B\x1F\x55\xDB\xA4\xCE\xA6\xC3\x29\xC0\xBE\x79\xE3\xCB\x6F\xAD\x8B\x64\x64\xAF\x98\x1E\x55\xEE\x8C\x2B\x5E\x7E\x2B\x65\xBD\x8A\x0F\x34\x36\x19\x5A\xAE\xB3\xD2\x9F\xAE\x07\x2E\xEB\xA2\xF9\x52\x26\x97\x61\x30\xE4\xA8\x49\x0C\x27\x81\x81\x7B\xC9\xF2\xD7\x5E\xB5\x75\xE6\x27\x94\x8B\x80\x9E\x80\x3E\xE9\x88\x36\xC3\x59\xC1\x23\xCE\x81\x62\x51\xE8\x48\x00\x30\x87\xD2\x0F\xAF\x75\xBA\xF5\x8F\x6B\x6B\xD0\x8F\x89\xE4\x82\xD3\x4F\xE6\xE4\xE1\x8A\xD2\xDA\xAB\x83\xAE\xA4\x50\x7A\xA5\xCF\x8F\x33\x17\x2F\x71\x13\xFF\x28\xA7\xE9\x97\xA5\x87\xB7\x37\xB0\x3F\x33\x66\xAC\xF1\x1B\x6F\xFB\x8B\xEF\xBC\x16\x4C\x5A\x5F\x04\x47\xEF\x10\xB0\xCD\x0F\xAF\xED\x13\x19\x00\x65\x78\xED\x1B\x5F\x5E\xE7\x18\x72\xF2\xFF\xC4\x74\x5C\x83\xD2\x0D\xCA\xDF\xB7\x32\x9F\x16\x8B\xE6\x70\x12\x65\xB2\xA7\xF2\x1B\xC1\xCE\x77\x8D\x46\x18\xAA\x11\x07\x98\x96\x68\xE4\x75\xA9\xFF\x95\x37\xB5\x81\x47\x0E\x0D\xD4\x45\x40\x6F\xB2\xAE\x17\x13\xE2\x7A\x67\x64\xDC\x63\xAA\x1E\x19\x7C\x05\x9B\x0C\xAD\xC9\x2C\x98\x39\xE8\xD2\x7E\x2D\x56\x2B\x9D\xE6\x48\x84\x39\x77\x8E\xF7\x65\xB0\x1B\xD4\x0A\xF0\xA8\x15\x94\x5F\x72\x5C\xC2\x2B\x9C\x85\x57\x18\x96\x10\xD5\x77\x3F\xC2\x08\x3B\x14\x57\x8B\x7C\xE8\x39\xDC\x3D\x7E\x71\x51\x88\x5D\x65\xDA\xA6\x68\x69\xF8\x96\x74\x9A\x7E\xD3\x17\x91\x0A\xC6\xD1\xC9\xD1\x26\xE5\x6D\xB9\x2E\x3B\x71\x4E\xDE\xDE\x86\xBD\x51\x72\x67\x28\x76\xE3\x22\x05\x3D\x97\x14\x63\x90\x3B\x99\x09\x58\x6E\x50\xBE\x05\xA0\x1D\xE2\x2D\x27\x4D\x9B\x67\xB5\x9A\x8E\xFE\xAE\xE3\xB6\x8A\x5E\xED\x8C\x88\x11\xE9\x62\x2C\xF0\x01\x8C\x8A\x5C\x43\x19\x8C\x59\xB0\x35\x30\xD2\xCA\x04\xF1\x56\x74\xA3\x2A\x66\x74\x04\x0A\x73\x26\xAE\xC0\x93\x21\x3D\x07\x7E\xBC\x5F\x83\x9B\x03\x00\x60\xF4\x22\xD4\xE8\x2C\x90\xD4\xE2\x40\xEF\xB5\x9E\xD6\x73\x0C\xA6\x4E\x6C\x9B\xC2\x71\x68\xDA\x8C\x04\x60\x00\x62\x16\x80\x48\xF2\xEC\x79\x73\x8C\xDF\xD9\x5C\x55\x93\x18\xBA\x89\xDC\xE3\xF4\x15\x61\x27\x98\xC6\xAF\x5A\x76\x67\xA0\x18\xC1\xA0\xE6\xAA\xB4\xFA\xDA\x15\x14\xA5\x15\xC1\x5E\x89\x12\x1A\x44\x9F\xFF\x2E\x17\x13\x0D\xE7\x57\xE7\x28\x9B\x73\x53\x67\x75\x97\x53\xC8\xF9\x96\xFC\x4C\x55\xBA\xEE\x15\xC7\x34\x2F\x00\x4E\x96\xB5\x40\xBE\x5B\x80\xE9\xAA\x38\xD3\xE6\x21\x8C\x01\x6F\x30\x0A\x62\xD4\x18\x0E\xE3\x7D\x21\xD2\x4A\x28\xFD\x73\x3A\xCC\x7F\x54\x4A\x75\x6E\x59\x6E\xB3\xA4\x5F\x7A\x3A\x69\x36\xC7\x9B\x57\x7E\x6F\x2F\xBF\xEA\xDE\xE5\xD3\xA7\x96\x1F\xF5\xAC\xC6\x2C\xE7\x3F\x54\x09\xFF\x62\xE9\xCC\xED\x8D\x04\x8E\x78\xD2\x93\x7F\x68\x2F\x93\xAB\xF4\x6A\xBD\x96\xFD\xA3\xA2\x5D\x4E\x3B\x11\x53\xCD\xEE\x6A\xC7\xCF\xEA\xCE\x0C\x77\x01\x69\x3E\xE7\x97\x7A\xCE\x4E\x3F\xFF\xBA\x1B\x37\x30\x02\xA8\xA6\xEA\xE4\x2E\x94\x7E\xF0\xC9\x5F\xEC\xFB\x53\x88\x5A\xFA\xDD\x4C\xF5\x9A\x75\xA6\xC9\x95\x78\xC3\x84\x35\x11\xE1\xC4\x1A\xDD\x45\x92\xAD\xDF\x8D\xF7\x5F\xE8\x62\x3D\x9E\xEE\x3A\x71\x40\x8D\x9D\xA6\x17\x7E\xDC\xF2\xBB\x12\x54\x1F\x76\x3F\x22\xBA\xC4\x63\x1B\xD7\x9D\x44\xDC\xA4\x70\x23\x78\xF3\x24\xB7\x9B\x90\x49\x77\x38\x81\x24\x62\xAE\x08\xB9\x21\xD8\xD7\x8D\x69\xD6\x04\x9A\x55\x74\x2B\x2D\x54\x5F\xBB\x02\x73\x52\x52\x1E\x34\x20\x81\xE5\x09\x3B\x50\x31\xAB\xE6\x4C\xBF\xA3\x65\xE2\x02\xDD\x8E\x02\x26\x6F\x0B\xC7\x22\x21\x3C\x30\xF9\x3D\xFF\xEE\x8B\x13\xA0\x66\xCB\xF2\x54\x8D\xD9\x0B\x46\x4E\x62\x90\x57\x3A\xB0\xC9\x42\x3F\x53\x3E\x04\xF7\xB1\xBD\xB2\xE5\xAC\xD3\x3B\x63\x5D\x8C\x9D\x10\x7F\xD6\x76\x91\xB5\x32\x8A\xDB\x4C\x73\xE1\x53\x6E\x2F\xEB\x01\x09\x69\xA8\x31\x5C\x73\x05\xF0\xA9\x02\x7A\x9C\x06\xDB\x30\x2E\x75\xF8\x88\x41\xD7\xC4\x2F\xAF\xA8\xCB\x00\xBC\x32\xEE\x17\x0F\x9C\xAE\x33\xE0\xE9\x61\x68\x28\x95\x31\xB4\x8C\xE1\x84\x1A\xAE\x58\x6B\x86\x2E\xA7\x1B\xF7\x6D\x53\x82\xEA\x68\x5D\x39\xFE\x00\x0D\xDF\x95\xB8\xFD\xED\x72\x66\x9C\x45\xEB\x86\x81\x25\x22\xC3\xCD\xDA\x9A\x32\x3E\x60\x32\x28\x4D\xBA\x97\x85\x51\xE3\xD1\xBA\x4A\x58\xA8\x62\x46\x9B\x97\x7C\xA3\x8B\xE4\x26\x35\xEF\xF9\x02\x5A\xDE\x11\x25\x9E\x89\x99\x49\x12\x59\x76\xCE\x86\x99\xE1\x24\xFF\x00\x75\x76\x0B\x8E\xC9\xFC\xA0\xE8\x57\x9E\xCE\x45\x91\x0B\x9F\xD3\x60\x66\xE5\x44\xE2\x7E\x19\x39\x2C\xEB\x54\x15\xB0\xE9\x03\xFA\x60\xD3\x01\x0F\x48\xD4\x70\xAD\xAF\x26\xB1\x8A\x5A\x4D\x30\xD2\x6C\xAD\x66\xB6\x61\x7A\x06\xDA\xDF\xCA\x79\x5E\x65\xF3\xB8\xB3\x48\x27\x8C\x32\xD0\xC4\x6B\x69\xB8\x8D\x82\x01\x1C\xF0\xF4\x8A\xFD\xCF\x58\x6E\x38\x4A\xC0\xB3\xE9\x12\xE1\x15\xC9\x3B\x3E\xA6\x5D\x6B\xA3\xC0\x31\x09\xDE\x47\x4E\xB5\x21\x3D\x05\x8B\x3D\xED\x82\xC4\xA3\x36\x36\xC9\xE2\xF3\x11\xDD\x0B\xDB\x7E\x9E\xAC\x25\xD9\xA5\x27\x66\xC1\xF9\x53\xE2\xDA\x42\xE1\xCB\x61\xD1\xAD\x8E\x77\x3A\x0B\x20\xDC\x94\xEB\xA0\xF2\x65\xD7\x9A\x84\x5C\xB0\x9A\x8D\xF3\x83\xB6\xE2\x44\xA2\xF0\xE6\x53\x33\xEA\x1C\x96\x8B\x0D\xB1\xD2\x25\x58\x6B\x48\x2B\xA6\x4C\x06\x23\x5D\x0E\x5F\xAF\x3D\x86\x75\x02\xBB\x36\xDB\x23\xDF\x2D\xE6\x7E\xF3\x78\xD7\x38\xF4\x1E\x4B\xD2\x31\xE0\x29\x5F\x02\x83\x9C\xEC\xDA\x04\xC9\xB9\x3B\x07\x1F\x04\x06\x38\x35\x9A\x1C\xD7\xEB\x88\x87\x82\xCC\x61\x14\xC7\x41\x8F\x3D\x40\x20\x0F\xCD\x5B\x1E\x74\xA0\x81\x1E\xF7\xFC\x4F\xE8\xF7\x7E\x4E\xCD\x62\x4A\xD1\x59\x0D\x19\x9D\x7A\xFA\x0D\x69\x80\x89\x42\xF0\x1C\x9D\x81\x42\x20\xF9\x7F\x33\x17\x7B\xC3\x8A\x98\xD7\xB1\x4E\xB9\x36\x6B\x09\xB8\x71\xDE\xD3\x12\xF0\x86\x6F\x7C\xD4\x97\xE8\xBD\x8D\x5D\x16\x3F\x48\x61\x3A\x2E\xD1\xE3\x65\xF1\x03\x95\xF0\xB7\xBF\x26\x9E\xF2\xD2\x96\x39\x71\x59\xD0\x38\x31\xDF\x23\xEE\x80\x1A\x8E\xC8\x0A\x3F\x97\x45\x9D\xBB\x0C\x2C\x7B\x0E\x9A\x69\xBF\x1F\x4D\x50\xC1\x3E\x97\x49\x80\xFB\x39\x98\x83\x61\x16\x4B\x6C\x7A\xE5\x3C\xAA\x1F\xA4\xF1\x8D\x9B\x41\x48\xF3\x6D\x9F\xE6\x06\xDF\x5F\xE9\x12\x4F\x9D\xEA\x9C\xA2\xE2\x1B\x8E\x03\x83\xCA\x9B\x2C\x43\xCE\x30\x27\x67\x57\x83\xEC\x56\x83\x88\xAB\x41\x44\x45\x76\xFA\x62\xED\xCA\xEC\xBE\xF9\xAC\x98\x84\x47\x3C\x74\x6F\xE0\xD7\x2F\x07\x5E\x4C\xCC\x0F\x50\xFA\x46\x25\xED\xF6\x39\x43\xCC\x9D\xCE\x73\xD3\x47\xC9\x31\x05\x95\xF4\xA9\xEC\x54\x05\xF0\xEA\x2D\x97\x86\xE2\x6D\xC0\x51\x63\xD9\xD6\x32\x47\x42\x7A\xD8\x60\x3B\x80\x89\x40\x64\xC9\xBA\x44\x6D\x5D\x86\x52\x13\x13\xC8\x91\xD8\x08\x52\x90\x92\x66\x61\x0E\x1F\x89\x4C\xD2\x1E\xC6\x1F\x7C\x1A\xE6\x31\x6D\x10\x54\x4C\xC1\x0F\x3D\x60\x8D\x20\xBE\xF8\xFA\x73\xAA\xAF\x25\x0B\xED\x8F\xD4\xFD\x61\xA9\x51\xA1\x4F\xDC\x6E\xF6\xE1\xED\x1B\x33\x7E\xDD\x99\x24\x61\x40\x8F\x00\xE9\x02\x1C\x7B\x63\xA4\x70\xCD\xCE\x13\xAE\x59\x60\x7C\x33\x70\xB5\xAD\xCB\x18\xAE\x39\xC1\x35\x73\x39\xFA\xAA\x64\x6B\x01\xAE\x19\x66\x09\x4C\xE0\x9A\x05\xB8\x66\x78\xBD\xCF\x8B\x2B\xC9\x1B\x65\x8A\xA0\x03\x35\xA7\x42\xCA\x6C\x02\x4E\x74\xD6\x7A\x80\x18\x17\xEB\x26\xC0\xB3\xF8\x8E\x74\x43\x7B\xAE\xE9\xE0\x54\x1B\xEE\x81\x38\x39\x65\x2A\x9E\x7B\x20\xDD\x6D\xF4\xA9\x61\x28\xA7\xD7\xDA\xE9\xC1\x50\xAD\x35\x32\x06\xFB\x20\x52\x71\xC1\xAB\xB8\xE0\x55\x3A\x9C\xDB\x15\x29\x63\xD3\x09\xF8\x7C\x74\xE0\xE1\x15\xC8\xF6\x7B\x81\x3A\x88\xA4\x58\x2E\x74\x5A\x8F\xC9\x36\x14\x59\x31\x74\x54\xF9\xB2\x33\x21\xF8\x0E\x66\xFE\x6A\x94\xFF\xC9\xD6\x5B\xCA\xC9\xE2\x0E\x57\x78\x46\x4B\xA9\xB1\x28\x1A\x14\x39\xA4\x69\x27\x1F\x1D\x3B\xE1\xC6\x31\x00\xD5\xFD\x69\x1F\x9D\x98\x9C\xBC\x12\x2D\x7A\x3A\x24\xD6\x38\x63\xE0\x8C\xD3\x50\x92\x6D\x04\x94\x8C\x80\x92\x11\x50\x98\x1E\xE9\xB9\xAA\xE7\x64\x17\x43\x4D\x80\x36\x06\xA2\xF5\xB1\x95\x64\xF9\x40\x47\xEE\xFD\xAC\x26\x72\xDE\xAA\x1B\xFF\x77\x56\x30\x99\x93\xD7\x92\x4C\x24\x6C\x54\x35\xD2\x99\x47\x23\xDB\xC3\x69\xF3\xF2\x55\xF3\x96\x21\x06\x36\x90\xE8\x64\xC2\xFD\xCD\x5B\xE5\x96\xFA\xE3\xEC\x6F\xA6\x0F\x06\x60\x4F\xAA\x53\x14\x18\x06\x5D\xAE\x1A\x93\xA6\xF9\x30\xE5\xDB\x75\xCF\x89\x79\xAB\x2C\x91\xF3\x28\x48\x42\x3D\x64\xEF\x8A\x11\x5C\x63\x62\xFF\xA3\x55\xDC\x2F\xB0\x88\xF0\x12\x03\xBF\x3C\xBA\xE2\x93\xB5\x44\x35\x0E\x5A\x62\xF4\xA6\x03\x1B\x3C\xEA\xB5\xB4\x21\xEE\x80\xE5\xDA\x78\xB8\x70\xD1\xEA\x52\xD8\xEC\xEF\x67\x61\xC3\xBC\xC0\xF3\xD4\x5C\x32\xEB\x6C\x72\xC3\x0B\x6F\xC4\x64\x9E\x13\x73\x30\x8F\x0B\x37\x32\x08\xF6\x7A\x2E\x8A\xF4\xF8\x37\xB7\x78\xCB\x81\x1C\xF2\xC7\x22\x8D\xE7\x14\x14\x24\x73\x99\xCB\xAF\x58\x6A\x32\xE8\x9D\x4C\x37\x8C\x1E\x07\x50\xA3\x54\x2E\xB4\x05\x87\xAA\xBB\xE5\x38\xF2\xC4\x2D\x4A\x37\x0F\x89\xD3\x6A\xFB\xBF\x0E\x2C\x5E\x45\x58\x7C\xE9\xD7\x07\x16\x5F\xFF\x75\x87\xC5\x87\xCE\xA6\x50\x45\xFD\x37\x24\x82\xA7\xD4\xA9\xF3\xF3\xAC\x9E\xB3\x92\x3A\x9F\x4A\xF6\x7C\x2A\xE5\xE7\x53\x29\x3B\x9F\x4A\x72\xA6\xD2\xCB\x64\xAC\x35\xE5\xDB\x28\x7A\x11\xFB\x94\x36\x36\xCB\x4B\xCA\x18\xE7\xCD\xF8\xDD\xBC\x0F\xFE\xAE\x5B\xE0\x39\x8F\xCF\xAF\x7A\x0E\x3C\x67\xF1\xF9\xCF\x7F\xA1\xFF\xFC\xB3\x1B\xF0\x6C\xE3\xF3\xCB\x6E\xEC\xBF\x7F\x2E\x3E\x6B\x7A\xA6\x21\x52\x50\x9A\x63\xC7\xFD\xC6\x6B\xF5\xA1\xF2\x16\x2D\xF5\xC4\x4F\xD8\xD9\x29\x6B\x9D\xBA\x92\xB4\xE9\xB6\x91\x31\xED\x17\xCA\x48\x78\xFA\x74\x48\x02\xC6\x79\x92\x95\x7F\x5C\xDB\x08\x3E\x3D\xE5\x38\x53\x61\x79\xD8\x23\x28\x06\x5F\x35\x69\x6B\xBE\x0F\x72\x67\xF7\x06\xD6\xE5\x6A\x5B\xD3\x9D\x4E\x8A\x50\x4F\x8D\x99\xF9\x8D\x65\x24\x53\x93\x16\x14\xB5\x60\xA8\xE9\x57\x61\x34\x20\x6C\xCC\xB4\xE1\x10\xB2\x51\x69\x63\xB6\xDF\x98\xED\x37\x66\xA1\x31\x94\xCE\x16\x1B\xA3\xEC\x58\xDE\x76\x8D\xC9\x6D\x36\x26\xBB\xC6\xF0\xC8\x19\xDE\x9A\x75\x87\x13\x6C\xE1\x97\x8A\xBF\xB0\x69\x3A\xCE\x29\xBF\x17\xD0\x33\x5C\xF7\x1B\x1B\x1B\xD7\xB4\x4E\xFA\x8D\x77\x8A\xF6\x84\xFF\xE2\x7D\xC0\x13\x02\xD6\x96\xF0\xB9\xC3\xE2\x52\xF9\xB6\x8C\xD0\x88\x97\xF0\x40\x07\xA0\x33\x49\xC9\x29\xD3\x32\xBF\x21\x8F\x53\xF4\xA0\x03\x4A\xB8\x0C\x55\x96\xCC\xDF\x2D\x8F\xFB\xBB\x64\x40\x70\x23\x49\x79\xC8\xFC\x97\xB0\xFC\x50\x25\xA1\x45\x6C\x86\x72\x2A\x4E\xAE\xC4\xD0\x07\x30\x93\x3B\xE5\x11\xD4\x33\x30\x9C\x51\xE6\xB3\x75\x7F\xD7\xE2\x71\x27\xFD\xA3\x8E\x9F\xE0\xC6\xEF\x91\xC7\xFD\xDD\xDC\xB8\x8A\x8D\x73\x18\x2F\xBC\x8E\x43\xBF\x68\xF7\x2D\xE3\xBA\xA4\x47\x09\x7F\x2B\x26\xF4\x16\xD8\x0F\x06\xED\xC1\x24\x6E\xA8\x7A\xD1\x00\xEE\x92\xAB\x74\x09\x23\xF3\x0F\x21\xD0\x97\x44\x70\x49\x4D\x15\x87\x7A\xAE\x26\x3E\xCE\x9D\xCF\x7C\xFC\xD2\xAD\x3F\x66\x60\x7D\x42\x1E\x46\x82\x8C\xB5\x54\xD2\xB7\x4A\x3F\x57\x09\xFC\x50\x3D\xD4\x14\x0A\xE4\x51\xC7\x7D\xB6\xEE\x32\x3F\x5A\xF7\x1B\x77\xEB\xE3\x27\x02\x28\x5D\xE6\x3F\x2B\x8F\xFB\x7B\x18\x90\x9A\x01\x29\x22\x20\x15\x06\xCC\xEA\x83\x54\x30\x48\xE9\x2B\x24\xBD\xC6\xFA\xE7\x4B\x26\xD5\xCD\xA6\x44\x2F\xE6\x00\x21\x01\xD4\x22\xC1\x5A\x1D\xC1\x86\x93\x9A\x62\x6B\x58\x9F\xA3\x6D\x97\xF9\x7B\x25\x1D\xD9\xC4\x1A\x92\xDB\xBC\x05\x49\x4D\xA6\x2F\x52\x1A\x54\xE9\x8B\x0E\xEA\x28\xE2\x0D\x86\x90\x02\xE0\x8E\x00\xB8\x3B\x61\x59\xDD\x23\x18\xBA\x3E\x5B\x3F\x81\x90\x3F\x41\xD9\xCD\xDF\x2A\x29\x44\xC3\x7D\xE2\x08\x45\x6C\xB9\x5C\xA0\xFF\x09\x15\x2E\x72\x21\x7C\xB8\x21\x31\xEB\x3F\x3C\x65\x27\xFD\x5D\xF0\x24\xBC\x0A\x35\x39\x3B\x7D\x52\x15\x9D\xD9\xFC\xE8\xA4\xBF\x1B\xAB\x86\x0F\x63\x4B\x5E\x53\x0E\xB6\x58\x82\x59\x8C\xFD\xCE\x93\xFE\x9E\xB9\x1F\xC4\xF6\xBA\x16\x4C\xF9\xB3\x32\xE1\xE4\x44\x23\x14\x81\x8C\x0E\x0C\x35\xAC\x61\x04\x3C\x59\xCB\xFE\x74\xF0\x49\x00\x7B\x84\xB8\xE8\xA9\x36\xEC\x58\xB3\x37\xC2\x5D\x12\x3A\x38\xC4\x75\xD8\x9C\xA9\x15\x59\x48\x12\x74\x87\x2B\x41\xD3\x0A\x47\x2A\x18\x26\x43\x39\x79\xC5\x12\xF1\xB2\x5F\x07\x53\x25\xC8\x3F\x7F\x1F\x18\x41\x94\x83\xD3\xFF\x24\x30\xE7\x3D\x47\x2A\xE9\xA5\x53\xEB\xFE\xA5\x1B\x1B\xE6\x38\x05\xC1\xA3\xCC\xA5\x7E\xE3\x6E\x71\x25\x67\x12\x46\x90\x82\x6D\xB5\xB1\xFF\x48\x25\xFC\xDD\xA7\x6B\xED\x55\x18\xF2\xC6\x5D\x62\xB5\x52\x69\xAD\xE7\xDC\x85\x61\x81\x36\x0E\xD6\xDA\xC7\x99\xBD\x72\xA6\xDA\xAF\x53\xB5\xF7\xDD\x39\xA1\xAB\x18\x54\xEF\x35\x33\xF5\xEE\xA6\x7A\x7F\x8A\xF5\x6C\xEC\xF6\x6E\x71\xB8\x52\x7E\xE3\x56\x28\xCD\x6A\xF4\x60\x90\xFE\xFD\xC2\xA9\x93\x5E\x1E\xAF\x28\xC1\x9F\x53\x3E\xBB\xD6\xBF\x1D\x8F\xE6\xBE\x53\x8A\xA7\xAE\xF0\x7D\x27\xFF\x0C\x04\x29\x30\xB9\x70\x8C\x2D\xCB\x17\x4B\x99\x4F\xFC\x5B\x15\x59\xBB\xE8\xCE\xA3\xD6\xBC\x7A\xB2\x53\x6D\xA3\xF9\x98\x79\xD7\xB5\x8D\xF5\xF7\x60\x7B\x14\x5B\x48\x1C\x69\x72\x74\x2D\x34\xCE\xFA\x7B\xC3\x0B\xC7\x61\x7A\xE0\x03\xFF\x85\x58\x5D\xF9\x8D\x8D\xB7\xF3\x49\xB5\x70\xD9\x9A\x3F\x7D\xBC\xF5\xB2\xC5\xDC\x7C\x02\xFE\x18\xCA\x33\x4B\xD1\x1B\xF3\xA5\x46\xA1\xD1\x2A\xCA\x17\x7C\x85\x86\xF6\xD9\x39\x43\xE3\x81\xB9\xEC\x3C\xC6\x75\x09\xE8\x5A\xA6\xBB\x99\x07\x92\x4E\xF8\x1B\xDA\x05\xA5\x84\x2C\x3D\xDE\x2C\x20\x9D\xE7\xD3\x46\x99\x89\x3A\x3D\x5F\x31\x03\x42\x3F\xDD\x2E\x14\xA0\x93\x09\x69\x14\x25\xE0\x42\xA7\xCE\xEC\x1C\x4E\x9D\x9A\xDC\x12\xC6\x6F\xA0\xED\xFB\x96\x1C\x22\xF0\xA9\x25\x47\xCF\x22\xF5\xCC\xEC\x79\x65\x8E\xF2\xB2\xEB\xC8\x6E\xB3\x23\x6A\xDC\xCC\x6F\xFC\x68\x95\x8D\xF0\xA6\xD8\x7E\x21\xBD\xBC\x8E\xD2\xF0\x9E\x49\xC2\x10\x1B\x4C\xAB\xF8\xFB\xBF\xFD\xC9\x9B\x6F\x58\x0C\x77\x17\x9C\xC4\xCB\xB3\x21\x78\x20\xFA\xDC\x60\x8D\x9A\x56\xA5\x93\xE8\x77\xA3\x43\x21\x1E\xFE\xFC\xDC\xD9\x8D\xE2\x5A\x0C\x98\x74\x5D\xF8\x4D\x22\xA8\x7B\x54\xFD\x47\xDD\x7F\x34\xFD\x47\xDB\x7F\xCC\xFA\x8F\x39\x3E\xA2\x1B\x22\xFA\x99\x68\x5F\x60\xEC\xC0\x70\x0F\xAB\xA0\x34\x87\xEA\x8E\xE5\xC9\x29\x4C\xC5\x99\xDC\x9C\x09\xFB\xE4\x4E\xDD\xF1\xC4\x4A\x24\xB7\x68\x90\xF7\xA7\x70\xB5\x2E\xFB\xFE\x4A\x85\xD3\x5C\xBE\x2F\x43\x20\x1E\x85\x5B\xC0\xD3\x93\x67\xD7\x48\x50\x96\xE2\x25\x60\x27\xCB\xFB\x52\xF7\x07\xF4\x14\x03\x53\x95\xD2\xF3\x4F\x6A\x85\x4D\xC1\xF0\x28\x5D\x9E\x4A\x3C\xE8\x3A\x22\x26\x0D\xD7\xA2\x76\xAB\x62\xEE\xFB\x5E\x0D\x4D\x35\x34\x58\x69\x18\xF4\x51\x22\x8B\xE6\x33\xAB\xE9\x4B\x56\x06\x4D\x90\x60\x83\x10\x95\x60\xB2\xAF\x35\x0E\x42\xFD\x3B\x67\xBA\x80\xD4\x6F\x4D\x7E\xFF\x7F\xC9\xEF\x3B\x93\xDF\x77\xB1\xAD\x71\x5F\x77\x45\x34\x5C\x26\x7A\x0C\xC5\x53\x7A\xC3\x99\xC4\xFD\xD6\x90\x2F\x49\x71\x6C\xC6\x5B\x9B\xA2\xBC\xD2\x7D\x23\xBA\x23\x46\x58\x15\xEC\xEC\xF9\x40\xB4\x66\x62\x6B\xAB\x62\x3A\xA8\x0E\x26\x3A\x93\x18\xE4\x06\x5D\x10\xB9\x7F\x8C\x09\x44\x8D\xFD\x0C\x4F\xF5\x25\xCF\x48\x52\x66\x95\x9F\x50\x9A\x4E\x8B\xB6\x0A\x49\x90\x72\x9B\x9F\xC6\xB4\xF8\x4A\x53\x96\x7D\x30\x92\xBB\xFB\x89\x5D\xDA\x35\x85\x41\x60\x8A\x96\x16\x60\xDA\xA5\xC0\xA4\x2F\xE4\x7D\xC2\x7B\xC2\x23\xCA\x7E\x89\x03\xAC\x34\x5D\x03\x94\x5C\xE5\x12\x2D\x1A\xB3\x2C\x7E\x10\xA3\x72\x89\xEF\x55\xC5\x7F\xAC\xD8\x9D\xC7\xDC\xCE\x17\x2A\x31\xA7\x47\x84\x2D\x87\xDC\xB4\x33\x90\x95\xC1\x71\x20\x42\x36\xBA\xC0\x0B\x0E\x38\x46\x47\x0F\x78\x17\xE6\x5C\xAD\x89\xCD\x5A\x43\x24\x85\x55\x3D\xBB\x85\xFE\x46\xA5\xD4\x44\xCD\xBA\xEB\x4C\xB3\xF5\x80\x55\x8D\x92\x04\xB1\xBA\x97\x0F\xD1\x3A\xA4\xA2\xF3\x0E\x81\x4D\x95\x5B\x61\x41\x4C\x61\x61\xA4\x4B\x86\x7E\x6C\x69\x1C\x5B\xC2\x87\x1A\x3D\xA3\x34\xBA\x04\x35\x06\x46\x20\x1A\x4B\x58\xB0\x4F\x40\x88\xA7\xED\x95\xC1\x2B\x08\x2F\x9D\xD3\x2F\xB9\xC9\x8D\x67\x9D\xDC\x12\xFE\xB3\x67\x6D\x1D\xB5\x22\x71\xBD\x79\x0C\xB9\x5E\x77\xE7\x08\x32\x46\x9A\x35\xF1\xFE\xC5\xB3\xCE\xF0\xB5\x4D\x62\x51\x14\x51\x90\xA2\xA8\xAD\x92\xDD\x71\x84\x16\xF7\x07\x64\x66\xF8\x80\x54\xE3\x8D\xC1\xE0\x3B\x17\x7C\x20\xB0\xC1\xC6\x2C\xD1\x9D\x0D\x7B\x07\xA5\xB7\xF4\xA6\xB6\xCB\xB2\x2E\x9C\xAE\x07\x98\x0B\x01\xB8\x96\x53\xAB\x4B\x8B\x25\xB2\xE9\x21\x39\x97\xBA\x62\x79\x72\xAA\x19\xB9\x41\xC7\xAF\xD1\xF1\xB0\x46\x57\xBE\x91\x93\xDF\xAB\x8A\x27\xF2\x01\x08\xBF\xC1\x64\x50\xA5\x1B\x12\x2D\x35\xC3\x4A\xB8\xC2\x65\x8F\xAF\x73\x97\xB9\xE2\xFB\xD1\xF6\x41\xC7\xC4\x02\x58\x7B\x41\x3C\x1D\x86\x50\xBA\x61\x37\x0A\x97\x7B\x79\x7C\xA9\x7C\xD7\xC8\xEC\x9C\x28\x9C\x9C\x13\xE3\xB7\x07\x96\x83\x8E\xA0\x6F\xA7\x9D\x37\x14\xEE\xE4\xE7\x6F\x01\xA5\xAB\x4B\x4D\x96\x1C\x9A\xE3\xD5\x68\xAF\xD7\x9B\xA2\x45\x61\x43\xEE\x91\xA0\x1F\xD0\xF0\x06\x40\x25\xB1\xB0\x19\xF2\x6A\x19\xC0\xF4\xBD\xA8\x47\x81\xBF\x0D\xFA\xEB\x66\xC0\xBC\xC4\x0D\x29\xFC\xF8\x3E\x85\xBE\x4E\x18\xE7\x6F\x50\xE1\xB5\xF0\x11\x7A\xA3\xF2\xCB\xA6\x72\xC3\x43\xA8\xD2\x56\x24\x6D\x16\xBC\xA9\x17\x99\xCC\x2B\x90\x25\xE3\x54\x96\x2C\xC2\x02\x19\xD7\x8B\x41\xDA\x0C\x67\x6A\x2C\x50\x8D\x85\xD2\x2D\xB8\xC5\xC4\x43\x36\xAE\x9D\x45\x14\x30\x65\x46\x12\xA6\x74\x95\x1B\x92\x4C\xA9\x77\x8C\xB2\xD2\x55\xE8\x6E\xC4\xFF\x31\xCD\x22\x14\x98\x66\xC1\x2D\xBA\x85\x40\x7D\x55\x46\x85\x15\x57\x59\x85\x02\xB7\x78\xA4\xDE\x31\xB2\x74\xD8\x07\xDF\x37\x3B\xB1\xAD\xE6\x02\xB7\xD3\x5D\xB0\xDA\xEC\x58\x6A\x16\x81\xCE\x16\xEF\xA8\x77\x79\x59\x8F\x96\x65\xBD\x1B\xA6\xEA\x2E\xA8\x2F\x84\x21\x5F\xE0\x76\x1E\xAD\xEC\x28\x23\x2A\x5B\xF0\xA2\x1E\xAB\x83\x6E\x37\x50\xD9\x1E\x77\x21\x52\x19\x74\xEA\x16\x91\x96\xC6\x40\x4B\x7B\xDC\x90\xA9\x6C\xD8\xBD\x59\x80\x37\x63\xB7\xC0\x68\x5C\xA8\x84\xDB\xED\x76\x3D\xBE\x1E\xB9\x5D\x6E\x37\x50\xD9\x22\x50\xD9\xA2\xDB\x0D\x54\xB6\x1B\x87\x88\x03\x28\xDD\x82\x9F\xB8\x1D\x30\x8E\xD5\x25\x37\x02\x2A\xAB\x77\x8C\x0C\xC1\xE7\x4E\x86\x8F\xA6\xC7\xBB\xF8\x51\xF5\xA4\xEA\x08\x80\x09\xD2\x7A\x08\xD2\xBA\xDE\x51\xBA\x1D\x2B\x95\x2A\x67\x70\x0F\x7F\xDC\x60\x05\x4C\xE0\x01\x70\xE3\x51\x99\x50\x5F\x47\x68\xC1\xA9\x75\x80\xAB\xF6\x7C\x09\x0C\x98\x7C\x8F\xB4\xC8\x48\xFB\xDA\x13\x57\xF1\xE5\x12\x57\xB1\x05\x71\x15\x7D\xE2\xDA\xF1\xD5\x26\xAE\x1D\xE7\x45\x5C\x05\x13\xD7\x0E\x24\xAE\xA2\x4F\x5C\x45\x9F\xB8\x8A\xAD\x89\xAB\xC0\x3C\xA0\x32\x12\x97\xA4\x68\xA2\x1D\x51\x95\xA5\x1B\xB9\x32\xE1\x65\x39\x58\x1F\x39\x07\x44\x45\xEF\x40\xE3\x2C\xA8\xBC\x13\x0F\xE6\x9B\x3D\xB2\x44\x07\xA5\x08\x7B\xBA\x4E\xAE\xD8\x61\xBD\x8A\x61\xBB\xB9\x10\x3D\xD4\x7D\x51\x57\x5E\xD6\xC3\xC0\x4C\xAB\x36\x56\x73\x15\x80\xA1\x72\x43\xE8\x73\x88\x9F\xB0\x8B\x10\x0C\xAD\x2A\x5D\x45\xF5\x4A\xA7\x62\xBB\x3A\xE9\x58\xCF\xEB\x58\xCF\xE9\x58\x6F\xDD\xB1\x8E\x1D\xEB\xE9\x8E\x35\x7F\x30\x28\x6F\xDF\x95\x95\x89\x33\x91\xC3\x70\x8D\xCA\x8F\x39\x50\xA7\x76\x8A\xE2\x24\x59\x3C\x30\x5A\x16\x97\x71\xBE\x4F\x45\xEE\x27\x74\x22\xAD\xBC\x5C\xC7\x70\x2D\xDE\xAC\xF3\xB1\x95\xE9\x26\x82\x37\x0D\x42\xF2\x61\xD5\x06\xAF\xF8\x4B\x94\xB8\x4C\xC5\x98\x10\xC1\x1D\x3E\x8C\x39\xC0\x84\x5C\xFA\xE2\xB8\x5D\x48\x7A\x9C\xDE\x6C\x9C\x77\x5F\x3E\xE7\xBB\x66\x40\xB7\x05\x8F\xA5\x19\xE0\x19\x32\x3A\x70\xF3\xA8\xFD\xD0\xA9\xA5\x46\xD1\xE0\x75\x32\x78\xFE\x84\xEE\xB2\x85\xEB\x12\xDB\x98\x44\x33\xD8\x74\x1A\xF8\xBE\x1E\xC4\xD9\x38\x8D\x77\xDE\x0E\xA8\xC2\xEF\x0D\xD1\xF9\xDD\xE0\x8E\xCB\x74\xE1\x94\x3F\xAB\x1E\x2E\xF9\x96\x9C\x71\x03\x50\x88\x1B\xEB\x8A\x03\xA8\xB2\xE2\xF6\x76\x0E\xAF\xF2\xD4\xD2\xE6\x7B\x53\x2A\x19\x00\x93\x0E\x91\xCD\x90\x6E\x42\x83\x5E\xD1\x36\x19\x28\x01\x39\x2A\x01\x50\xA2\xD7\x1B\x85\x8C\x78\x40\x8B\x87\xEE\xBC\xD0\x17\x83\x8A\x07\x62\x9D\x6A\x31\x6A\x33\x7A\xAF\x52\x22\xE7\x21\x82\x46\x39\xDC\xD4\xA4\x2C\x24\x4E\xF9\x63\x14\x87\xFA\x06\xBE\x02\x81\x77\xC1\x29\x0E\x0D\x83\xC5\x59\x98\x92\x53\x2E\x6B\x2C\x6B\xFB\x14\xDD\xD7\x86\xC1\xB5\x54\x6F\x70\x08\x8C\xE7\x95\x4A\x7A\x83\x5A\x5A\x30\x49\x15\x70\xE9\x62\xC6\x24\x2D\x3A\x93\x74\x30\x53\x23\xA7\x1A\x39\x98\xA4\xF9\x2A\x3A\x15\x17\xE8\x53\x66\x56\xA7\x99\x3A\x1B\xA4\x83\xC8\xD2\x95\x1B\xA4\x2C\x1D\x1D\x7D\x07\xFC\x1F\x43\x1A\x97\x69\x72\x6C\x38\x68\xAB\x45\x65\xA9\x58\x71\xA5\xD5\x2A\x07\x48\x25\x4C\x1D\xF9\x18\xB5\xD6\x8C\x5C\xE9\x46\xC8\xD4\xF1\x36\x88\xB9\x03\xD7\xFB\xC2\xB2\xAC\x17\x61\xDE\x6E\x54\x8F\x61\xFC\x23\x57\x02\x53\xCF\x89\xA9\xE7\x14\xD1\xD6\x2D\x02\x53\xDF\xE1\xC6\xC8\xD4\x55\xBC\xCF\x43\x67\x59\x3B\xDC\x80\x99\xFA\xA0\x7B\x43\x99\xB9\x5C\xCE\xCC\x32\x07\x36\xEE\xAA\xC7\xD7\x0B\x20\x77\xBE\x1F\x53\xD1\xE1\x7E\xC6\x22\x30\xF5\x45\xBA\xFB\x39\x46\xA6\x9E\x23\x53\x1F\xB9\x72\x75\xC9\x2D\x74\x4C\x1D\x20\x94\x30\x75\x78\xDC\x84\xA9\xC3\x5B\x60\xEA\x83\xC8\xD4\x83\x0F\x13\xD3\x3B\x11\xC7\x00\x28\x62\x86\xA7\xC9\x3E\xA3\xB2\xC4\xA8\x2C\x31\x2A\x3B\xCD\xA8\xD8\x35\x7B\x0E\xA3\xC2\x68\x6C\x3D\x46\x35\x40\xDF\x44\xD5\x52\x54\x9F\xE9\x35\xCE\xD7\x53\x6D\xC2\xA8\xEC\x2C\xA3\xB2\x2E\x67\x46\x25\x13\x46\x25\xBA\xCB\x3A\x22\x30\x2A\x4E\x68\x5D\xE0\xE5\x5D\x1E\x4B\x83\xB1\x27\x98\x51\xD9\x2D\x18\x55\x4E\xD7\xB3\x03\xA3\x32\x1D\xA3\x3A\xEF\x49\x34\x66\xD3\x69\xE0\x7B\xF6\x81\xE7\x0C\x40\x66\x86\x51\x99\x1E\xA3\xB2\x4C\xFE\xC8\xA8\x06\x2E\x3B\x80\x2A\x1C\x32\xAA\x02\x5E\x15\xF1\xEE\x90\xA2\x9B\x97\x8D\xE8\x18\x15\x0E\x47\x05\x56\xB5\xA0\x0F\xBA\x05\x54\x04\x73\xD0\x1F\xDA\x46\x74\xF6\x0A\x94\x44\x56\x65\x52\x56\x45\x5F\x18\x8E\xF9\xD0\x0C\x22\xAB\x1A\x90\x86\x59\xE7\x50\x05\x59\x15\xA6\xC6\x1B\x00\xAB\x1A\x10\xAB\xCA\x03\xAB\xCA\x51\x82\x44\x56\x65\x79\xF1\x32\xAB\x12\xCD\xA0\xC7\xAA\x06\x61\x70\xCC\xAA\x4C\xC2\xAA\x0A\x60\x57\x29\xAB\xCA\x66\x18\x51\x86\x4B\x90\x98\xCD\x4C\x8D\x82\x6A\x14\xA5\xCB\x5D\x41\xAC\x0A\x88\xA4\xC0\x5B\xB1\x53\xAC\x2A\x9F\x61\x55\x86\x59\x55\x16\x59\x55\xBC\x18\xD7\xE4\xC4\x93\x0A\x6C\x38\xB0\xAA\xAC\x63\x55\x54\x89\x58\x55\x7E\xA4\xCE\x02\xAB\x32\xB3\xAC\x2A\x5B\x6A\x28\x15\x16\xB1\x2A\xC9\xAC\x2A\x77\x23\xBA\x97\xD9\x67\x55\x05\xEB\x9F\xCC\xAA\x86\x1D\xAB\xCA\x3B\xFD\x73\x87\x33\xCC\xAA\x4C\xF7\xA6\x20\xFD\xB3\x60\x56\x55\x04\x56\x25\x03\xAB\xE2\x28\x5D\x29\xAB\x1A\x22\xAB\x2A\xFC\xC4\x65\xCC\xAA\x24\xB2\xAA\x8C\x58\x95\x61\x56\x95\x11\xAB\x32\xCC\xAA\xB2\xCD\x58\x95\x41\x56\x95\x81\x90\x65\x56\xC5\x14\x4F\xC4\x81\x14\xB1\x00\x64\xBD\xC0\xD4\xBC\x9A\x44\xE0\x47\xBD\x20\x8F\xFC\x88\x98\x93\xCB\xBD\x38\xB2\x58\x3A\x75\x47\xBD\xC3\x9B\x7A\x00\xC0\x43\x38\x29\xDE\x15\x46\x38\x55\x2E\x47\x38\xA1\xDA\x31\xE0\xAD\x61\x30\x63\xDC\x8E\x27\x32\xAD\x73\xB1\x42\x77\x18\x06\x91\xA9\x14\x2D\x62\xE5\x16\xB9\xFB\x3B\x9E\x58\x49\x37\x00\x40\x0D\xDC\xA2\x53\x77\x34\x3B\x9E\xD8\x71\x77\x90\x38\x65\x89\x07\xEB\x39\xAE\x4B\xF8\x03\xA3\x33\xE5\x2D\x4A\xCB\x89\x3C\x95\xDE\x8D\x92\xD3\x21\xF4\x7A\x7B\x95\xE2\x12\x5D\xBC\x99\x2F\x9F\x4E\x6D\x55\xFA\xA2\x1D\xDF\x42\x5B\x6A\xC2\xBF\x38\x0D\xDE\x27\xBF\x5D\x17\xE4\x8A\xF4\xD6\x6E\xF3\xA9\xBB\x8C\x3C\xE5\xD4\xF3\xD6\x46\xDF\xCE\x1E\x1E\x2F\x7D\xCB\xEB\xF6\x85\xCD\x78\xE1\x8B\x93\x50\xD8\x3D\x8F\xA7\x9E\xF7\x9E\x0C\xBF\xF5\xB2\xBB\xF9\xF6\xF4\xA5\x5E\xDE\x37\x5D\xF0\xB0\xE9\x82\xEF\xBB\xF9\xF6\xEB\xCA\xCF\x18\xDC\x58\x9D\x13\x53\xF0\x5C\x1B\xAB\x21\xCC\x18\x4A\x1D\x72\xC2\xC2\x9D\xC4\xD1\x57\x77\xCF\x75\x82\xBB\x7D\xCB\xF2\x54\xB3\xC9\xC6\xAB\xE0\x64\xCA\xC9\x78\x1B\x1B\x46\x4C\x87\x2C\x78\x9C\x86\x7E\x03\xE4\x98\x09\xAC\xD0\xAE\x54\x06\x0F\x23\xE6\xEF\xB1\x36\x39\x88\x3E\x54\x17\xF1\xBA\x69\x46\xFB\x5E\x59\x6F\x47\xBC\xE4\x77\xB1\x5B\xB2\xE0\xE4\x63\x78\x8C\x35\xC5\xFE\x91\x41\x8A\x66\x78\xEE\x8E\x6E\x22\xF6\x50\xA5\xB1\xFB\x8C\x66\x20\x67\x67\xA0\xCE\x3D\x03\xBD\xAD\x19\x98\xED\xCC\x40\x9E\xC7\x0C\x54\x32\x83\xA9\x0D\xE6\x47\x92\x67\xB4\x0C\x64\x75\x2A\xB8\x12\xA1\x5B\x79\xE7\xC6\xA0\xD6\xFD\xEB\xFF\x8E\xCF\x30\x11\x99\x74\xD8\xF8\x7F\xE4\xD5\x44\x9F\x1E\x4E\xE4\x69\x3A\x16\x20\x8B\x4F\x46\x45\x4A\x39\x49\x4A\x06\x2A\x52\xAA\x53\xA4\x24\x47\xBA\xC4\xFB\x95\x92\x15\x29\xD9\x29\x52\xBA\xB3\x99\xC3\xDD\x0F\x74\x54\x94\x2D\x71\x77\x19\x74\x10\x22\xE1\x70\x1B\x4C\x25\x8A\x14\x8A\x1E\x4A\x18\x16\x4D\x64\xCC\x2A\x13\x2D\x3E\xC9\x97\x9F\xE6\xC4\xBA\xE3\x64\x6B\x08\xB3\x2E\xC6\x09\x79\xDE\xE3\xE6\x30\x8D\xDA\x0F\x9D\x5C\x6A\x24\x0D\x5E\xAF\xF7\x2E\x99\xAA\xA0\x48\x15\x7C\xEB\x72\x3B\x93\x68\x36\x9F\x46\x8C\x98\xC2\xB3\x71\x0A\xEF\x4D\xA7\x8A\x94\x74\xC5\x1D\x94\x9F\x8B\x14\x29\x45\xBB\xDE\x05\x2A\x52\x06\xF5\x2F\x67\x28\xFF\x08\x86\x59\xC9\xA2\xC5\x27\xE3\x5D\x72\x99\x0C\x20\x5A\x7C\xA0\xEB\x0C\xEA\x21\x06\x9A\x40\x4D\xA5\x04\xCD\xC9\xD0\xB6\x2F\x8A\xAD\x46\xA2\xE2\x32\x4A\xD5\x28\xFA\x62\xC4\x92\xB9\x29\x9C\x6C\x69\x7D\x14\x68\x36\x80\x1D\x32\x40\x63\x52\x92\xC5\x07\xB6\x64\x3D\x74\x92\x2C\x3E\x09\x6A\x94\x9C\xB6\xF8\x08\x2C\x68\x64\xB6\x4E\xBA\xB2\x19\x26\x6A\x94\x24\x73\xB4\x0C\x6A\x94\x70\x23\x8C\x0F\x41\x6A\x94\x9C\x3E\x84\x04\x7B\xAE\xD1\x9B\xDA\x7C\x23\xA8\x93\x43\x9D\x2C\xD6\x41\xF7\x86\xBC\xC6\xF0\x18\x72\x75\xA6\xD9\xAC\xDF\x58\xDE\x35\xA6\xFB\x4D\x00\xC4\xBD\xAC\x2B\xC5\x8E\xFD\x53\x8A\x18\xE8\x1A\xED\xC2\x60\x60\x51\x0D\x93\x36\x2F\x9D\x70\x23\x56\xC4\xAA\x91\xE5\x13\x9E\x11\xFF\x87\xB3\x62\xE2\xE5\x72\x67\xA2\x22\x56\x55\x7C\xA7\x47\x52\x25\xB4\x19\x25\xD8\x8C\xD5\x88\x06\x86\x2D\x90\x1B\x6C\xD1\xE4\xA0\xA2\xAF\x36\xD5\x12\x93\xDD\x1D\xF5\x02\x68\x8A\xAC\x88\x49\x97\xD3\xBA\xCD\x9D\x3D\x5A\x99\xA0\x88\x19\x4F\x5B\x23\xA8\x60\x8C\x9D\xEE\xCE\x32\x64\xAA\x6E\x8D\x58\x11\x1B\x75\x6F\x70\x3D\x16\xCE\x44\x2D\x03\x14\xB1\x85\xC7\xD7\x99\x5B\x20\x45\x8C\x57\x07\x6B\x15\x74\x07\x14\x15\x31\xE3\x27\xAE\x82\x71\xAC\x2E\xB9\x0C\x15\xB1\x6A\xA4\x09\x42\x77\x32\x84\xE8\xBE\x48\xCA\x92\x0A\x62\x49\x05\xB1\xA4\x62\x9A\x25\xE1\xC5\x98\x62\x1E\x4B\x42\x55\x79\x86\x25\x65\xB0\x9A\x8B\x2D\x56\x73\x91\xB0\xA4\x62\x96\x25\x15\x68\xD6\x4C\xB3\xA4\x1D\x1D\x4B\xDA\x31\x6F\x13\xCA\xF2\x58\x60\x51\x47\x96\x54\x6C\xC1\x92\x32\xA2\xA6\xC0\x92\x74\x9F\x25\x9D\xD7\x24\x1A\xBD\xE9\x34\xF0\x3D\xC8\x16\x9E\x0D\x26\x1F\x9F\x66\x49\xBA\xC7\x92\x0A\x4E\x00\xC6\x2C\xC9\x26\x2C\x69\x6A\x13\x4A\x96\x6E\x07\x8C\x62\x47\xC7\x92\x8A\x84\x25\xED\xD0\x07\xDD\x0E\xF6\x78\xD8\x11\x2D\x3B\xDA\x84\x82\x92\xC8\x92\x74\xCA\x92\xE8\x0B\x5D\xF1\x40\x4C\x64\x49\x86\x37\xA1\x32\xA8\x42\xA1\x86\x28\x23\xD4\x84\x78\xD1\xB1\xB6\xCE\x02\x4B\xCA\x58\x26\x33\x4B\x22\xB0\x90\xB9\xEA\xA4\x13\x8D\xE9\xB1\x24\xD3\xB3\xEC\x42\x02\xB6\x6E\x13\x2A\xDB\xD4\xEB\x21\x23\xAF\x87\x2C\xF2\x90\x99\x1A\xB4\x23\x51\xE7\xA5\xCB\xC2\x26\x14\x66\x10\x76\xD9\xAC\x65\x97\x4D\x59\x76\xD2\x69\x66\x28\x16\x2C\x3B\xC9\x01\x3B\x28\x6A\x47\xB8\x17\x9F\x63\xC3\x81\xA1\xD8\xCA\xF2\x39\x29\x57\x22\x86\x92\x1D\xA9\x6D\xB0\xEC\x34\x9D\x2C\xC8\xE4\x64\xC1\xD2\xA9\xA1\xCB\xEE\xA8\xC7\xCB\xA2\x5E\xF4\xB2\xDE\xE5\x2E\xA0\xD9\xF1\xA9\x42\x5E\xD2\x1D\x27\x3A\x00\x71\xD9\x1D\xF1\xDC\x73\xF1\xF6\x26\x0F\x27\xF8\x19\xD8\x1F\x18\xFF\x53\xBA\x6C\xA5\x52\x5E\xD6\xBB\xC1\x02\xA3\x52\x30\x49\x16\x6F\xC7\x4D\xAC\x0C\xEF\xB8\x85\xE3\x49\x84\x7B\x4E\xAE\x29\xBB\x5D\xBE\x82\x47\x85\x19\x9E\xFE\xEE\x1E\xE9\xD2\x9B\xFA\x42\x6F\xEA\x3D\x01\x4C\x00\xE4\xBD\x29\x90\xF7\xC0\x40\xF7\xD6\x7B\x62\x0E\xB4\x99\x1A\x17\x52\x8D\x0B\xE9\xD0\xE3\xC2\xA3\xB1\x6D\x77\xA1\xDB\xB3\x5A\xCD\x26\xB7\xDF\x83\xB8\x28\x32\x44\x85\xE8\x5B\x89\x84\x4C\x46\xCD\x6E\xE0\xF5\x19\x5E\x30\xE6\xFF\x98\x66\x0F\x06\xD1\x69\x2E\x84\xAE\x22\x6A\x76\x57\x39\x87\x4A\x6B\x32\x5E\x49\x61\x18\x05\x36\x78\x04\x61\x45\x13\x00\xE4\x3C\x88\x54\x80\x8B\xDC\x83\xDC\x45\xAB\xCD\xEE\xA5\x66\x0F\xE0\x68\xCF\x1D\xF5\x92\x97\xF5\xC5\x00\x90\x65\x59\x2F\xB8\x8B\xEA\x07\xC3\xE4\x2E\x72\x0F\x3A\x5A\x19\x6C\x8E\x19\xFF\x85\x5E\xD4\x7B\xD5\x41\xB7\x00\x8C\xFF\x5B\xDC\x83\x11\x65\x18\xDD\x67\x0F\xB2\xF7\xBD\x00\x8C\x6F\x71\x39\x33\xFE\xBC\x7B\x83\x60\xDA\xEB\x2E\x64\xC6\x7F\x61\x25\xDC\x82\x5B\x7A\x7C\x7D\xB1\x5B\x72\x0B\xC0\xF8\x17\x80\xE3\x2F\xB8\x3D\xC0\x8A\xF6\x50\x48\xB7\x07\x23\xE3\xBF\xD0\x4F\xDC\x6E\x18\xCB\xEA\x92\xBB\x18\x19\x3F\x02\x19\xC0\x75\x27\x83\x4B\xD1\xE3\x5D\xFC\x28\x4B\x40\xA8\x04\x9C\xC9\xB5\x7A\x77\xE9\x76\x57\xAA\x44\x91\xD2\x2C\xBA\xF1\xE3\xEB\x5D\x48\x6F\xE3\x1F\x40\xEE\x88\x5F\xE3\xBE\x12\x79\x97\xED\xF6\x13\x67\xF9\xC0\x69\x17\x76\x67\x31\x25\xA1\xD3\xDC\x9D\x45\x63\xCB\x69\xEE\xCE\xCE\x31\xF8\xC1\xF0\x82\x85\x2B\xD7\x6A\x0B\x3A\x16\x1B\xFC\xCC\x06\x89\x67\x60\xCA\x43\xE6\x75\x51\x6C\x80\xA8\x4C\xE5\x97\x21\xF9\x65\x48\x7E\x99\x69\xF9\xC5\x21\xBB\xE6\xC8\xAF\x9C\xF7\x5B\x3A\xF9\xA5\x71\x5B\x8F\x7D\xF9\x66\x58\x3F\xFB\xD8\x99\x44\x7E\x99\x59\xF9\x65\xE2\xDE\xE4\x68\x46\x7E\x8D\x7A\xF2\x2B\x0D\x3A\x2C\xC2\xDE\x4F\x06\x7F\x58\x7E\x99\x2D\xE4\x57\x1E\xB7\x8B\xA2\x94\xDE\xEE\x24\x38\x48\xC6\xBC\x69\x50\x54\xCC\x2C\xCE\xC6\x19\x3C\xF7\xE8\xCB\xAF\xAC\x27\xBF\x78\xDB\x0B\x0F\x15\x1A\x4D\xD9\x27\x35\xC9\xAF\x2E\xAE\xD1\x68\x13\xF9\x85\xC3\x91\x41\x82\xED\xD1\x07\xDD\x1E\xB4\x78\x72\x64\x08\x0D\x0A\x2D\xCB\x7B\x93\x7B\x3A\x09\x96\xA5\x12\x8C\xBE\xC8\x78\x53\xAB\xD1\x51\x82\x69\xF4\x11\x31\x75\x0E\x55\x50\x82\xE5\x14\xDB\x66\xD2\x62\x88\x30\xDC\x9B\x64\x09\x36\xB5\x37\x69\x98\xA7\xB3\x04\xDB\xD1\xE8\x9E\x04\xD3\x61\x70\x2C\xC1\xB2\x44\x82\xD9\x64\x6F\x52\x4E\xBB\xA7\x12\x55\x8A\x6E\x6F\x32\x9B\xA9\x61\xA9\x86\x45\xBD\x92\x24\x18\xDE\x3E\x39\x9F\xBD\x49\xE9\x32\x66\x93\x22\x4A\xB0\x8C\xFF\x03\x48\xA2\x8B\xE3\xA4\x28\x32\x9B\x14\x9D\x04\xCB\xE2\xDE\xA4\x04\xEE\x28\x82\x04\x43\xB6\x78\x31\x49\xB0\x9D\xEE\x62\xB7\x73\xB5\x11\xDD\xDE\x64\x90\x60\x17\xBA\x9D\x34\xBB\x9D\xEE\xE2\x39\x12\x2C\xEF\x49\xB0\x70\x2A\x55\xE7\x20\xDC\x58\x82\xE5\x24\xC1\xF6\x92\x04\xCB\x3A\x09\x66\x39\x21\x3E\x71\x53\x1B\xF3\x14\x90\x04\xDB\xEB\x2C\x49\xB0\x1C\xD9\xFB\x5E\x92\x60\xBB\xBD\x01\x5E\xD6\xC9\xA7\x07\xA7\x40\xDE\x05\x03\x7D\x70\xBD\x2B\xA0\xC1\xCE\xD4\xD8\x4D\x35\x76\x03\x95\xEE\x72\xBB\x8F\xC6\xB6\xDD\x6E\xB7\x6B\x9E\x04\xDB\xB5\xB5\x04\xCB\x9D\x65\xD4\xEC\x05\x09\x86\xC2\xCB\xF2\x7F\x4C\xB3\x8B\x24\xD4\x6E\xE8\x2A\xA2\x66\x2F\x4A\x30\x12\x5D\xBC\x96\xC2\x30\x70\xE7\xDA\x1E\x41\x58\xC5\xC3\x8A\xE6\x21\x41\x90\x3D\xC4\x3D\x68\xB5\xD9\xBB\xD4\xEC\x02\x1C\xED\x22\x09\x76\x11\x00\x04\x25\xD8\x83\xEA\x0B\x60\x72\x0F\x72\x0F\x21\x09\xB6\x37\x48\xB0\xDD\x5E\xD4\x0F\xEE\x24\xD8\x05\x77\x44\x39\xB5\x0B\xE5\xD4\x83\x49\x82\x59\x96\x60\xB6\x7B\x83\x60\x7A\xB0\xDB\xCD\x12\x6C\x77\x90\x60\x17\x4D\x49\xB0\x5D\xC0\x8C\x70\xB2\x05\x0E\x82\x44\xCA\x5E\x18\xCB\xEA\x92\xBB\x08\x45\x0A\x02\x19\xC0\x75\x27\x83\x4B\xD1\xE3\x5D\xFC\x08\x16\x17\xC8\x10\x8B\x32\x64\x6F\xE9\xF6\xF6\x25\xD8\x85\xA9\x04\xDB\x1B\x25\x58\x8E\x12\x6C\xAF\x9F\x38\x01\x24\xBA\xBA\xE4\x2E\xC4\xEE\x04\x49\xB0\x8C\xBB\x23\x19\xE5\x32\xEE\x4E\x6C\x26\xC1\x32\xEC\x1D\xF3\x41\x86\x08\x01\x84\x28\xE2\x19\x78\x44\xB3\x07\xB8\xDD\x1E\x66\x72\xAB\x41\xCC\x15\xB0\x74\x0C\xB1\x0B\x81\xC6\x32\xF9\x2A\x67\xC8\xF7\x30\xE3\x7B\xD1\xB9\x2A\xA3\xE6\x66\xE2\x33\x03\xB8\xAA\x14\xB3\x6A\x8A\x92\xD9\x36\x92\x9B\x3E\x8A\x57\xBE\xD0\x81\x59\x3A\x8A\x65\x00\x66\x35\x2C\x9E\x23\x95\xC4\x3C\xC6\x30\xFC\x11\x0E\xBF\x2A\x5D\x05\xC3\x97\x18\xDD\xE9\xBE\x17\xF4\x23\xBA\x00\x2F\xA5\x6D\x85\x11\x1E\x38\xD1\xE9\x56\xE2\x78\x51\xFE\xAA\x96\x66\x32\xBD\x12\xC2\x0E\x1A\xBA\x1C\x93\xCF\x71\xC1\x8E\xE9\xEA\x70\x95\xC1\xAF\xF1\x4A\x95\xB1\x4D\x83\xA6\xFB\x7E\x21\x57\x2A\x1B\x4A\x42\xD4\x1B\xB9\x42\xE6\x3A\x6E\xE7\xCA\xCE\x1F\x43\xB0\xD1\xA3\x7C\x51\x4B\x0E\x64\xD8\x19\xD8\x21\x4B\x2B\x89\xBD\x7E\x9E\x90\xE0\xAD\x8A\xE2\xB9\x0B\x38\x31\xEB\x7B\xCF\x97\x1E\xF0\xA2\x81\xEB\x9F\x41\xCA\x10\x88\x89\xE2\x0E\x26\x63\x4A\x42\x30\xCB\xD5\xD4\x45\x5D\xA6\x51\x43\xEE\x4F\x67\xE6\x9C\x9D\x05\x2F\x5C\x59\x76\x1A\x09\x83\xA2\x2C\x6F\x5F\xC8\xF2\x89\x3C\x9D\xF1\xEE\xE4\xDC\x6C\x3D\x0E\x23\x4B\xF0\xC6\x2E\x87\xD5\xA3\xC8\xED\x61\xB7\xD7\x19\xBC\xF7\x56\x1C\x4E\xCB\xE4\x79\x55\xEC\x1C\x5F\x7C\x86\x1E\x2F\xDE\x05\x25\xEE\x80\x32\x78\x3F\xDA\x50\x82\x7E\xB0\x51\x28\x65\x74\xF8\x55\xC4\x5F\xA3\xF8\x6B\x1C\x7F\xED\x8A\xBF\xF6\x3A\xBD\x5C\x5C\x96\x54\xBF\x38\x6C\x25\x64\x89\x7B\xF8\x9C\x28\x93\x78\x77\xCA\xC4\x08\xF2\xB8\xB7\x8C\x8A\x12\x29\x14\x14\x43\x9E\x27\x86\xB6\xF4\xF8\x0F\x42\x3D\xDC\x40\xC8\xCA\x50\xBD\x4B\x4B\xD1\xBF\x04\x92\xC5\xBD\xC4\x2C\x09\x86\xDD\x45\x5B\x0A\xA7\xD7\xF9\xBC\xD3\x6B\x33\xBB\xB2\xF8\xC8\x03\xD4\x33\x3E\xD7\xD0\x9D\x67\x45\xEF\x5C\x83\xBC\x0C\x66\xCF\x35\x32\x3A\xD7\xC8\xE2\xC9\xE5\xBC\x73\x8D\x50\xE5\x12\x2D\x28\x10\x90\x72\x83\xFE\x91\xC6\x20\xFA\x92\x67\xC0\x67\xF9\x4E\x45\xCD\xF1\x43\xB3\xBE\x17\x6B\xB6\x42\x52\xD3\x99\xB8\xA3\xBC\x52\xA9\xB8\xEF\x3A\xB2\xA8\x79\xE1\x56\x82\xC5\xD3\x5E\x4E\x33\xE0\x4D\x3D\xF4\xA6\xEE\x89\xEA\x51\x2A\xAA\x4B\x3C\x0D\xAD\xCB\xDE\x79\x6F\xAF\x06\x9D\x97\xD6\xC3\xD2\x0D\x5D\x39\xC7\xC5\x90\xA3\xCE\x2E\xE4\x25\xED\x2F\xE6\x78\x28\x4F\xF2\xBA\x2A\x46\xC3\x92\x44\x75\x77\xCC\xCB\x07\x94\x43\x57\xBA\x61\x74\x32\x0C\x42\x9A\xAB\x80\x22\x65\x5D\x79\xA4\xCA\x47\x03\x26\x89\x8A\xE4\xF4\x82\xAB\xDC\xC2\x6A\xB3\xB8\xD4\x94\xC0\xDA\xCA\x3B\xEA\xB1\x97\x78\x4E\x59\x2E\xCB\x7A\xE0\x16\xEA\x9D\xD1\x2D\x7A\xA4\x0E\xBA\x01\xC8\xE3\x0B\xDC\xCE\x3B\xA2\xD4\x2D\x51\x36\x8C\x80\x26\x2F\xE8\x9F\xE9\x96\x9D\x5B\xF4\xA8\xE7\x16\x3D\x00\xD9\xB8\xC3\x8D\xDD\x00\xE4\xF1\x00\xE4\xF1\xC0\x95\x80\x67\x9A\x5D\xBD\x33\xBA\x45\x2F\xBA\x05\x57\x05\x9F\xC2\x2A\x03\x6D\xC3\x86\x33\x5D\xD4\xEC\xF0\x11\x05\x64\x65\x80\xFE\x6D\x77\x88\xBB\x08\x84\xB0\xE0\xAA\xA3\x74\xB3\x6F\x4A\x74\x2E\xA2\xF3\x6A\x44\xB9\xA1\xD3\x18\x8E\xC2\x73\x4E\xF2\x91\xF3\xC9\xC7\x84\xB6\xBE\x49\x3E\xFF\x56\xC8\x87\x50\xAE\xC3\x61\x1E\x28\x49\x78\xFB\x17\x59\x26\xAA\x4F\x81\x15\xA3\x56\x54\xA4\xB1\x75\xD9\x69\x75\x5E\x80\xB9\x9B\x94\xD4\xDD\x75\x23\x1D\xA5\x75\x9F\x12\x7B\xE1\x89\x93\xEB\x46\x06\xB3\xD5\x2F\x26\xEC\xBF\xE4\x44\x48\x5E\xB6\xE3\x77\xA2\xC5\x3A\x4A\x1D\xAB\x8C\xE3\xDC\xF7\x16\x05\x9D\x3A\x38\xA7\xC3\x46\xF3\x71\x4F\xF0\x0B\x65\x59\x45\x06\xB0\xB8\x82\xA8\x11\xD4\x87\x26\x48\x15\xCA\x61\x30\x55\x0D\x8F\x3E\x3D\x5E\x7D\x43\xD5\x48\x63\xC8\x12\xEC\xDC\x19\x4C\xBC\x1F\x46\x7A\x40\x8D\xCA\x4F\xE6\xD9\x42\xE2\xBF\x1A\x3D\x43\xDF\xF4\x1F\x7E\x1C\xFF\xFD\xC3\xA3\x5E\xEE\xE4\x9B\x7E\xBC\x12\xD3\x92\x3D\x75\xF7\x0C\x21\xEB\x64\x6D\xD9\x2F\xB4\x91\x07\x28\x63\xF2\x01\x85\x51\x0B\xD0\x13\x8F\x04\x3F\x08\x2D\xD6\x66\x4F\xB7\x5E\x53\x0E\xCF\x53\xC7\x6B\xE3\xC7\xF1\x0E\xAA\x26\x25\xCF\xAF\xB7\x9D\xB2\x00\xCF\x37\x4C\x3D\x9F\x9A\x7A\xE6\x0E\xC6\xB4\xAD\x8D\x61\x52\x14\x06\x7F\xC2\x40\x31\xB6\x74\x99\xD7\xC7\xE9\x56\xFD\x3A\x1E\x5F\x4F\x7D\x68\x48\x93\x0C\x39\x08\xA6\x8E\x28\xF2\x4E\xB7\x98\x11\xE0\xCD\x00\x8F\x28\xD1\x75\x9E\xA9\x0A\x26\x33\x54\xA1\x0F\xDC\x1E\x26\x44\x97\x5D\xA4\x50\xC5\xCE\xF6\x86\x0E\xD2\xD5\x21\xBE\xD8\xCD\x3E\x58\xB2\x1E\x92\x91\x9B\x79\x13\x03\xDE\xCF\xB2\x2D\x4B\x6C\xCB\x06\xC6\xA6\x66\x6A\x64\x54\x23\x43\x37\x31\x97\x1D\x8D\x6D\xBB\x0C\x13\xA2\xCC\xB0\x39\x4B\x46\x6E\x64\x73\xC6\xA9\xB8\x65\x6E\x89\xF0\xBB\x28\xAD\x81\x95\x65\xD0\x76\xB2\x65\x6E\x88\xD5\x40\x95\xA3\x55\x86\x3D\x16\x31\x9E\x1B\x70\x39\xD3\xE7\x72\xE1\x2A\x51\xE0\x72\x76\x59\xD6\x3B\xDD\x02\x59\xB1\xC8\x2B\x42\x23\x64\x47\x31\xDB\xDB\x09\x6C\x6F\x17\x5B\xB1\xA6\xBB\x32\x34\x22\x93\x5E\x31\xDB\x53\xDD\x9B\x8C\xD8\x1E\xEF\x2E\xA1\x22\xB8\x33\xB0\xBD\x9D\xC0\xF6\x76\x02\xDB\xDB\x19\xEE\x14\x99\xCE\x8A\xCD\xA6\xD9\x1E\xED\x84\x02\x78\xEE\x8C\x3B\xA1\xF8\x18\x36\x46\x25\x2A\x6F\x98\x27\x23\xEE\x84\x3E\xB6\x92\x01\xB7\x3D\x5B\x4C\x12\x9B\x8D\xE1\x5F\x4B\x57\xBA\x01\xBA\xD9\xB9\x21\x2D\x28\x1D\x97\xAE\x2B\xD2\xF3\x22\xE6\x75\x89\x85\x10\x4C\x08\xA7\x2F\x51\xE6\x32\xE0\x35\xC1\x4A\xB8\x58\xCA\x89\x57\x71\x97\x4F\x60\xB8\x03\xAF\x30\x16\x41\xF9\xED\xF0\x32\xBA\x27\x7A\x71\xB9\x18\xBB\xE0\xD3\xE0\x38\x89\xEF\x59\x83\x47\xEC\xE5\x83\xD3\xBA\xD2\x89\x74\xB7\x50\x96\xDF\x0A\x6F\x47\xD1\x2D\x82\xBE\x85\x96\x70\x65\x52\x0B\x6F\x55\xD2\x4E\x78\x12\x9D\x0B\x05\x71\x18\xE9\x55\x74\x2F\x08\x57\xE5\xF0\xFB\x82\xFD\x00\x02\x63\x41\xEE\x83\x29\xE0\xC1\x42\xC6\x34\x4B\xE8\x85\x02\x6B\x5D\xF7\x18\x8D\x05\x46\x63\x99\xD1\x98\x70\xF7\x20\x61\x34\x6A\x8A\xD1\xA8\x29\x46\xA3\xE6\x30\x1A\x8B\x57\x9D\x1B\x43\x49\x24\xF0\xA6\x70\xC9\xE9\x94\x03\x9B\x51\x73\xD8\x0C\x05\x8F\x0B\x6C\x26\xE6\xE5\x92\x21\x3D\x0E\x86\x53\xE1\xF4\x70\x21\x73\x96\x0C\x95\xD8\x2C\xFF\xCD\xF9\xE0\xCB\xBF\x09\x3E\x27\x39\x41\xCE\x2C\xD0\xFE\x45\xA6\xB2\x5F\x75\xC2\x78\xEA\xB2\x71\x92\xFC\xB1\xBB\xC2\xAA\xBA\x8C\x3B\xA9\xCC\x7F\x0F\x4A\xDF\x22\x3D\xB0\x60\x81\x67\xE6\xB6\xDF\x28\xEC\xA1\x73\x09\x92\x94\xD4\x67\x46\xD8\xEB\x46\x72\x40\x34\x52\x07\xA6\xAA\x05\x61\xAF\x4B\x82\x0B\xC6\x1E\x44\x8F\x12\x43\x67\x2B\x3D\x61\x7F\xBF\xA4\x7D\xFE\x4D\x69\x3F\x57\xDA\x9B\x6F\x4A\xFB\x6F\x4A\xFB\x2F\x53\xDA\x1B\x96\xF6\x66\x8E\xB4\x37\x0F\x90\xB4\xFF\x12\x3A\x1E\x9F\xD6\x5B\x6D\xDF\x0E\x0A\x3A\x58\x2A\x8A\x9C\x36\x69\x6B\xB1\x99\x82\x40\xF7\x3B\x3B\x5F\x65\x29\x32\xF2\x55\xAE\x55\xA7\x0A\x90\xDA\x20\x9D\x62\x8D\x00\xA5\xBD\x5C\x9C\xAD\x18\x6B\xA8\xAE\x86\x32\xB5\x8A\x7A\x03\xD4\x80\x05\xE5\xBB\x26\x00\x26\x53\x8E\x87\x50\x96\xDE\xE0\x90\x7C\xD5\x0C\x1D\x0F\xC5\xBC\x53\x52\xDE\xA6\xC3\x1C\x85\x26\x82\xB1\xA8\x3B\xCE\x87\x9E\x79\x23\x10\xC3\x07\x30\x0E\x80\xA5\x58\x4D\xC8\x55\x71\x1C\xE3\x3F\x0A\x71\x32\xC6\x7F\xCC\xA1\x23\xFE\x26\xB9\xF1\x1F\x52\xB4\x3E\x5A\xAA\x49\xD7\x28\xC9\xE1\xD0\x1A\x1E\x65\x27\x82\x5B\xCC\xB8\x8E\x62\xD4\x25\x31\xFE\xB3\x33\x94\x95\xFF\x7F\x81\xC0\x4F\x98\x33\x5D\x2B\xF4\x38\x52\xBA\x5D\x18\x36\x5C\xA7\x6E\x10\x5A\xCE\x23\x06\x6B\x82\x78\x94\x0D\x10\x61\xB2\xE7\x6B\x6F\x33\x37\x08\x69\x99\xF6\x6F\x10\x02\x60\x39\x9A\xBB\x76\x2A\xA6\x85\xD6\xF4\x0B\x23\x50\x05\x81\x9C\xC7\x8C\x08\xA3\x69\xDF\x2C\x39\x85\x35\xCB\x67\xDB\x18\xE7\x59\xCE\x1B\x63\xD8\x07\x4F\xB0\x46\xF9\x80\x92\x7B\x4C\x08\x0B\x0B\x78\x2A\xE2\x75\x26\x3A\xA5\xB1\x2E\x8F\x43\xB5\x4E\xB9\x9C\xB5\x2B\x02\x2E\x3C\xBF\x2D\xDC\xE1\x2A\xCB\xF2\xAF\x24\x1E\x79\xB0\x84\x24\x4A\xA1\x10\xC9\x30\x8C\x75\x3F\x74\x7A\xA9\xD1\x74\xBB\x73\xEA\xE6\xA3\x08\x87\xF6\xAA\xCB\xE1\x4E\x37\x1F\x75\x1B\xA2\xD3\xF0\xC4\xF8\x04\xA7\x23\xC7\xEE\xCE\x83\xEB\x42\xA1\xEA\xF4\x3C\x40\x61\x72\x58\x3C\xB4\xA7\x00\xA5\xE9\xA1\xBD\x76\x0A\x0F\xED\x35\x2F\xA9\x2E\xE1\x7B\x23\x1C\x86\xA3\x13\xA8\x82\x97\xBB\x41\x3B\x97\xFC\x95\xF0\x12\xA3\x36\x97\xDF\x32\xA5\xD1\xDF\xD1\x57\xE9\xA7\x8C\x03\xB9\x95\x71\xB0\x03\xBB\x82\xEF\x05\x7D\x5F\xEE\x87\x95\x10\x17\x36\xAD\x68\x41\xE9\x64\x28\x25\x28\x2E\x34\xDA\x2F\xE1\x90\x9C\x9B\x99\x0F\x37\x92\x93\x2F\xD9\x18\xE3\x38\x0B\x0A\xA7\x41\x48\x17\xB4\x7E\x44\xF9\xCB\x18\x3B\xAD\x5B\xD6\xB3\x2B\x90\x2F\x1B\x60\xE0\x35\x72\x22\x11\x9D\x13\x89\xEA\x92\x6F\xC1\x92\x66\x1A\x55\x4E\x30\x8D\x8A\x80\x4A\x81\x81\x44\x12\x1A\x2D\x92\xDC\x54\x86\x33\x19\xC7\xF3\x1E\x9C\x66\xE9\x64\x79\x43\xCA\x1F\x04\x99\x3E\x22\x1E\x9B\xE0\x7C\x25\x69\x2C\x82\x8E\x3D\x44\x3C\x39\x11\xF1\xE4\x44\xC4\x93\x13\x11\x4F\x4E\x44\x3C\x39\x11\xF1\xE4\x24\xFC\xBA\xD8\x89\xE9\x8E\xF3\xAF\x5A\xC7\x1B\x4A\xC9\x53\x66\xE2\xC4\x9B\x6A\xE9\x0B\xCA\xAE\x08\x24\xB8\x5E\x2B\x27\x7A\xAA\xE8\x8F\x56\xA2\x9C\xB6\x6B\xC4\x5C\xBB\x46\xF4\xEC\x1A\xD1\xD9\x35\x4E\xF8\x53\x6D\xAD\x3D\xE5\x93\x52\x84\xB1\xBE\x41\x22\xA6\x1E\xCC\xD4\x73\xD1\x86\x96\xC6\x84\xE0\x53\x2D\xA7\x78\x87\xA6\x85\x7F\x1A\x51\x44\xDA\x94\xF0\x26\xD2\x82\x71\xFA\x28\xA8\x15\x31\x90\x9E\x2A\xC7\xC9\xDA\x90\x97\x0B\x59\xBE\x55\xCA\x2C\x30\x1B\xCA\xE0\x86\x87\x7C\xCA\x8F\x6A\x0D\x96\x19\x9E\x1A\x5A\x85\x37\x84\x71\x31\x2B\x7F\x8A\xE2\xD2\xB0\xC7\xE2\x8E\x33\x00\x11\x76\x94\xD1\xB5\x72\x86\x12\x21\xD2\xED\xC5\x26\x77\xB9\xBF\x01\x3E\xA0\x80\x94\x37\x70\xD4\x64\xCC\x2E\xEE\x8F\x61\x4B\x49\x23\x7C\xCE\x2D\xD2\x2B\xC6\xC8\x41\x99\x5B\x85\xE4\xAC\x81\xC4\xCB\xF7\x6A\x99\x4D\xBC\x26\x73\x11\x57\x12\x9D\x16\xDA\xC3\x68\x09\xE0\x21\xE1\x21\x0E\xDA\x8F\x02\x05\x75\x4E\x75\x45\xC5\x77\x9D\xC9\xE5\x12\x60\x66\xE2\x7E\x2B\xDD\xCD\x58\xB0\x42\x6B\x2D\x74\x19\xB3\xE8\xE1\xC5\x66\x8E\xF6\x2C\x63\x0B\x7C\x93\x93\xA3\xF6\xD6\xD6\xB1\x54\x31\x78\x5F\x2E\x43\x0D\x52\x93\x3A\xFC\x98\xE0\x9E\x52\xD0\x1D\x1D\xBA\xC0\xCB\xC9\xCC\x1B\x43\xD1\x55\xA1\xF7\xEC\x50\x85\x91\x1F\x3B\x47\x34\x5A\x9D\x51\xC6\x36\xE8\xC3\x6B\x09\xE5\x86\x16\x89\x89\x8B\xC4\xC4\x45\x62\xE2\x22\x31\x71\x91\x98\xB8\x48\x4C\x5C\x24\xA6\x5B\x24\x5D\xC2\xEC\xD2\x15\x30\xF3\x82\x33\x5A\x3B\x53\x0B\x97\x77\x99\x16\xD1\xFC\xCB\xC8\x6F\x28\xCE\xAC\xF3\x1B\x45\xC7\x2E\xCE\x5A\x2E\xBD\x02\xE4\xE1\xD4\x0E\x53\x76\x04\xA7\xA8\x13\xBA\xD7\x44\xFE\x61\x9C\x55\x43\x97\xAF\xD2\xA0\x6C\x68\xE2\x80\x11\xAB\x9A\xB0\xAA\xA7\xB0\x6A\xD8\x8E\x21\xAC\x1A\xA2\xAD\x80\x55\x4E\xFE\x3F\x07\xAB\x51\xFE\x4D\xDA\xC6\x86\xFC\x0B\xB1\x05\xD2\x13\x4C\xC0\xAA\x09\xE2\xD4\x00\xD3\xE0\xC0\xDE\xC6\xE9\x38\x6F\x85\x18\xB5\x84\x51\xCB\x18\xA5\xE4\xD0\x74\x71\x97\x8E\x06\x5C\xC6\x5C\xFA\x50\x05\x12\xBA\xEC\x24\xB5\x8A\x8D\x76\x64\x12\x6E\xD1\xA3\x36\x1F\x4E\x0C\x66\xB7\xEF\x65\xD8\x33\xE0\x99\x3A\x1B\xF6\x0C\x44\xB0\xE3\x31\x75\x94\xEA\xBD\xAD\xC2\xAE\x5F\xD8\x03\x28\x48\x46\xA3\x96\x3B\x69\xF9\x2C\x9C\x53\x9D\x50\x56\x41\xFE\x49\x92\x04\x03\x99\xF1\x46\xD0\x66\x17\xE9\xCE\x73\x45\x76\x6B\x32\x2E\x46\x2F\x39\x38\x44\x6F\x41\x1A\x83\x29\x32\x79\x41\xDA\x84\x16\x61\x01\x3E\x16\x0F\x74\x68\x29\xCA\x20\xFD\x92\x50\x87\xA2\xCB\xA3\xDF\x21\x48\x4D\x67\xD1\xA7\x15\xA8\x0F\x57\xF3\x03\xB9\xF1\x95\x08\x15\xE2\x7D\x4D\x9D\x6B\x4B\xBA\x31\x84\x1E\x09\xBD\x28\x5C\xBD\x43\xA0\x6F\x15\xEA\xA0\x17\xFB\xD4\x6D\xB7\xDD\x04\x88\xF0\xB4\x59\xF4\x0E\x78\x7B\x80\x0A\xCB\xF2\x02\x11\x6B\x38\x31\xFE\x30\x2E\x8D\x37\x0C\x7B\x49\x53\x38\x49\x03\x29\x1C\xA4\x06\x20\x47\xE0\xA9\x63\x30\xD0\x67\xBD\x14\x06\xF1\xEB\x67\x31\x8B\xDE\xAC\x21\x46\x55\xD1\x14\x23\x4B\x2C\xB7\x59\x51\xC6\xE0\x63\xB4\xC0\x0E\x55\x39\x90\xED\x09\xD2\xE5\x39\x12\x95\x06\x52\x25\x49\x30\xFE\x9F\x41\x0D\xE7\x40\xF3\x84\x71\x79\xBC\x29\x96\x38\x07\xA5\x97\x27\x5D\xB1\x34\xFE\x55\x1C\x07\xBB\xD5\x73\xD6\x42\xF8\x68\xFC\xCB\x67\x29\xD9\x1D\x20\x3F\x8B\x56\x26\xE6\xBC\xC3\x21\xAC\x79\x75\xFD\x6A\x25\x46\x05\x20\xC8\xBF\xFB\x65\x30\xA9\x5F\x0B\x8D\xC9\xFD\xE2\xFB\x56\x92\xD9\xD2\x0B\xCA\xC2\xB1\xAB\x1D\x7F\x24\x1D\x1D\xCC\x83\x06\x98\xC1\x00\x15\x0F\x30\x83\x01\xAA\x73\x0F\x50\xF8\x3B\x93\xBE\x47\xD9\x36\x46\x03\x16\xE0\x3E\xBE\x46\x5B\xB7\x50\xCD\xC5\x31\x48\xFF\x50\x12\x85\xBD\x61\x38\xE1\x6F\xF9\xB5\xAE\x09\x6C\xE1\x7B\xB8\x85\xFD\xD8\xC2\xA5\x49\x0B\x0F\x9B\xDF\x42\x6F\xBC\xF6\xCB\x00\x4A\x07\x12\x93\x80\x64\x64\x88\x95\xF2\x06\xD9\xB2\xB8\x4C\xB3\xFE\xC2\xB7\xE6\xE5\x49\xBC\x2F\xD4\xE4\x94\x22\xA0\x31\xFE\xD8\xD1\xCA\xF0\xC5\xBA\xB0\xD5\xE4\x4B\xDA\x9F\xDB\xD1\xFA\x63\xC7\x1B\x3B\x7E\xC9\x59\xDE\xF0\x63\xE9\x0F\x8B\xFA\x04\x76\x80\x27\x89\x14\xBC\x26\xF3\x72\x9D\xAC\x50\x79\x02\x58\x2B\x46\x11\x32\xAE\x40\xD3\xC4\xE5\x4B\x2E\xE3\xE4\x6F\x28\x8A\x42\xEA\x04\xFF\x42\x00\xA9\x97\x11\x42\xF7\x0B\x04\x3A\x05\x01\xDE\x83\xA2\x5D\x2A\xE8\x59\xC5\x56\xCC\x12\x27\x2C\x84\x56\x4C\x68\x05\x99\x87\xFF\xC7\xDF\xE8\x53\x4D\xBA\xE2\xD0\xD7\x49\xD3\xBD\xD4\xA9\xD5\x05\xE3\x6C\x6C\x32\x52\x93\x8E\xD4\x9C\x9B\x82\xE3\xA2\x82\x25\x45\x6E\xD1\x94\x61\x11\x54\x3D\xE1\x9F\xBE\x91\x10\x1C\x5F\x4A\xDE\x6A\x99\xE7\xE9\x32\xCF\xB7\xB7\xCC\x59\x9B\xEB\x8D\x88\x77\x44\xFC\x3F\x24\xE0\xA1\xF0\xC6\x2F\x48\x96\x42\x99\xB2\xB8\x8F\xFF\x46\x6F\xD1\x47\x46\xBB\x58\x8E\x7F\x01\xF0\x2F\xCA\xFB\x72\x3C\xAD\x40\x36\xE9\x31\x83\xD7\x14\xAB\x0C\x0A\x86\x3C\x09\x00\x06\xA0\xAE\x70\xF8\x7A\xD0\x7B\x56\x70\x03\x83\xD3\x26\x1B\xBA\xFE\x46\xE9\x3B\x50\xFF\x01\x49\x84\x5F\x69\x2E\xC5\x64\xDD\x4E\xF8\x0F\x8A\xF1\xED\x67\x71\xFB\x84\xC0\xC0\x97\xBC\xE9\x16\x1E\x7E\x15\x3E\xB8\x74\xBB\x1F\x5C\xBE\xDD\x0F\x9A\xED\x7E\xF0\xC8\xED\x7E\xF0\x88\xED\x7E\xF0\x7F\x6D\xF7\x83\x0F\x88\xED\x7E\xF1\x57\xDB\xFE\xE2\xAF\xB7\xFD\xC5\x77\x6E\xF7\x83\x0F\x6E\xBB\x8B\xCF\x6F\xFB\x8B\x2F\x6E\xFB\x8B\x2F\x6C\xFB\x8B\x6F\xDB\xEE\x07\xDF\xBE\xDD\x0F\x1E\xD5\xFF\x00\xD8\x6C\x46\x97\xE4\x93\x16\x1A\x5A\xB1\x2B\x75\x96\xE4\x06\x07\x8B\x11\xF9\x21\x99\x89\xBF\x9D\x75\x99\x80\x1E\xB6\xD6\x58\xD6\x52\x1F\x46\x82\x06\xCC\x58\xA7\xC6\x1F\x25\x5B\x85\xCC\xAA\x4B\x1D\xE6\x57\x82\xA7\x4B\xDB\xF1\x27\xCF\xC4\x70\x19\xCE\xEE\x53\x0F\x75\x16\xA4\xB7\x42\xE1\x80\x2E\x48\x02\xAB\x7B\xE1\x27\x94\x5D\xD1\x8B\xF1\x7F\x3B\x4B\xA1\x67\x4B\x27\xC7\xFF\x78\x86\xB2\x9E\x13\x03\x44\xF6\x43\x1F\x14\xC1\x34\x16\xCE\x5E\xA2\xDD\x65\x4A\x50\x7E\x8F\xBD\x2C\x50\x2D\x08\xD4\xB1\x8A\x61\x48\x48\x13\x3F\x8C\xFD\x8D\x5B\xFF\xBA\x97\x75\xDA\x1D\x7D\x38\x6E\xFD\xCD\x89\xCA\x17\xCE\x85\x74\x3C\xF6\x01\x49\x8A\x17\xF6\x0A\x0A\x36\x44\xE7\x24\xAE\x00\xCB\xBE\xE0\x88\xB3\x41\x88\x0D\x01\xB2\xE5\x92\x17\x78\xC6\xB5\x0F\xAF\x8B\xCB\x93\xAE\x5C\x72\xB6\xCB\x9B\x87\x01\xD6\xED\x3E\xB6\x17\xA8\x21\xAA\x1C\xDA\x38\x09\x9F\x1D\x5F\x3A\x54\xC9\x32\x80\x1C\xEC\x98\xE5\xD7\x6D\x6C\x6C\xBC\xD6\xA0\xEA\x4A\xBB\xA6\x1A\x64\xF4\xB7\xC2\x9F\x4B\xC4\x27\x6E\xBE\x71\x43\x5C\x26\x04\x6B\x17\x97\x88\xCF\x74\x05\x63\x2C\xF8\x52\x57\xB0\x17\x0B\x9E\xF9\xAC\x58\xF0\x90\xD6\x8B\x7D\xE2\x79\x50\x70\x40\xA0\x92\x62\x9D\x62\xA5\x07\x80\x44\xEE\x72\x39\xF9\xEA\x01\x0A\x9B\xE1\x12\x47\x5C\xC2\xF1\x26\x32\xCC\x76\x32\xCC\x26\x32\xAC\xF0\x23\xB0\x8D\xD9\x65\x3D\x01\xF2\x9A\x1F\x3D\x99\x53\x04\xC3\x40\x53\xFD\x0B\x46\x61\xF7\x21\x6D\x8D\x5B\x12\xA7\x7C\xD0\xC6\x1D\x98\xA4\x03\x03\x14\x81\x8A\xBD\x21\x72\x09\xB4\x42\x34\x78\x69\xD8\xA0\xBD\x07\x0D\x9A\x20\xDD\x67\x49\x4D\x3C\x5C\x62\x10\x31\xBB\x5F\x8C\xBB\x79\xEC\x0D\x5F\x00\x25\x77\xC5\xFB\x62\xB1\x7F\x18\x8B\xD1\x72\x95\x2D\x8C\x11\x68\x39\xB8\xF5\xC6\x7B\x71\x74\xD8\xCC\xF9\x44\x0F\xA8\x87\x3A\xE1\x77\xB5\xC1\x16\x79\x10\xFC\xC2\x5D\xB3\x7D\xA1\xAC\x7C\x85\x94\x76\x42\x46\x95\xDF\x45\x69\x8B\xC1\x82\x1A\xE3\xE9\x29\xE7\xB1\xE3\x5D\x6B\xDA\x8E\x51\x74\x25\x21\xEE\xDC\xEC\x17\x26\x8C\x95\xF4\xA5\x38\xDC\x30\xDD\x7D\x6A\x4C\xF1\x85\x77\x91\xFB\xBA\x49\xB6\x8D\x28\xC1\xC0\xF5\x31\xD8\x02\xE7\xB1\x38\xA0\x76\x39\x15\x15\xFC\xF2\xE3\x7A\x9E\x45\xAB\x93\xB3\x2B\xDC\x90\x27\xE0\xBF\x8C\x32\xA2\x03\xD5\x10\x2C\x8E\x1D\xC5\x30\xCD\xE6\x44\x17\xF6\x80\x94\x5B\x01\xC4\x90\x28\xB7\x19\x9A\x8D\x8E\xF7\xC5\x46\x2D\xDD\x7D\x56\xAC\xE9\x1A\xCE\x86\x8E\x2A\xAE\x21\xEF\x84\xCC\x11\x3B\xC9\x30\x8B\x7B\x50\x6D\x33\xDC\xFA\xE3\x8C\x7E\xCC\x4C\xF0\x7C\x4F\x90\x2D\x12\x00\x26\xFB\x00\x13\xCE\x24\xBB\xC7\x48\x62\xF4\x31\x87\xB2\x1E\x37\xB8\xC5\xB0\xEB\x10\x5A\xC8\x48\xD9\x3A\x69\x1D\x21\x3B\x7E\x79\x58\x23\xF8\x41\x6F\x1F\xA2\xEB\x58\xF7\x3B\xD6\x29\xA6\x24\x9F\x36\xEC\x02\x16\xDC\x29\xA4\x22\xEE\x3D\x4A\x87\xB9\x69\x24\x71\x14\xBA\x5B\x67\xFA\x3B\x81\xF0\x2D\xF5\x9F\xE8\x8F\x2E\xF3\xA7\xDA\x03\x6A\x57\x6A\x78\x77\xFA\xA0\x13\xE3\x8F\xA3\xCB\xBD\x28\xCB\xDF\x79\xB0\x2C\x02\xBA\xEF\x12\xBD\xD8\x48\x60\xFA\x3B\x32\xE5\xC4\xE5\xC2\x39\x3E\xD4\x10\x64\xC6\x35\xAA\x6D\xC8\xD3\xC1\x5F\x4A\xC1\x15\xE0\xF5\x43\xF1\x28\x36\xBA\x5E\x10\x14\x0B\x22\x70\x8C\x49\xAE\x5B\x34\xA3\x55\x3C\x7D\xE2\x62\xF4\x1C\xA1\xDC\x6C\x40\x40\xE2\x8A\x46\x2D\xD5\xC6\xA3\x45\x20\x9C\x5A\x22\xAA\x16\xDC\x2A\x7A\x19\x08\x67\x69\xD5\x5D\xDA\x3A\xCE\x49\xB3\xAF\xA5\x98\x6D\x98\x0B\x98\xC6\x37\xD7\xEA\x47\xA3\xEB\x2F\x05\xE6\x9C\xF4\x3F\xD4\x2E\x5C\x68\x73\x99\xCF\xFC\x33\x19\xDD\xB5\x78\xC6\xA4\x5D\xD0\x22\x57\x9A\x3D\x4F\x2E\xC6\xB3\x9F\x8B\xFD\x98\xE5\xA7\xF1\x0B\x44\xC6\xC7\x8E\x56\x83\xE0\x1C\x40\x64\xAF\x98\xEC\xB3\xC8\xF7\x91\x9B\x29\xF2\x14\xC9\x88\xD8\x9D\x81\x05\xA0\x99\xFA\xD0\x2B\x87\xFC\x3B\xD8\xBC\x03\x84\xAB\x15\x72\xBF\xB5\x21\x8F\x23\x1A\xF8\x9C\x3E\x2E\xE1\x28\xB0\x68\xF6\x92\x1B\xCD\x51\x8C\x1D\x63\x18\x1E\xDE\x00\x41\x8C\x06\x25\xB9\xC3\x8C\x1B\xBB\xD6\x64\x5E\x5D\xDF\xE0\x6E\x83\xF2\x80\x76\xC3\xE3\xA7\x0C\xE9\x6B\xA0\x5A\x5C\xDF\x14\x60\x52\xAE\x2E\xF9\xFB\xEE\xBB\xEF\x3E\xED\x34\x50\x29\xFC\x94\xAB\x4B\xEC\x47\x43\xE9\x37\x43\x03\x43\xBC\x7A\xC3\xD3\x45\xB7\x18\xF4\xE2\x68\x72\x1A\x89\xA2\x9B\x39\x35\x58\x4B\x40\xC9\x18\x25\x00\x37\x1E\x95\xB3\x89\xC9\x8A\x27\x6D\x7B\x49\x32\xEE\x22\x36\x35\x76\x76\xA5\x2A\x02\xAF\x1B\x15\x25\xE3\xE2\x8A\xCA\x26\xA8\x28\x22\x2A\x8A\xAF\x73\x54\x14\xE7\x40\x45\xF1\xE5\xA2\x62\xF0\x95\x45\x45\x1E\x51\x91\x77\xA8\x30\x09\x2A\x16\x23\x2A\xF2\xAF\x0D\x2A\xB6\x81\x0C\xF9\xAF\x1C\x19\x22\x20\x23\xF2\xA8\x09\xF0\xA8\x51\x06\xA0\x03\xCB\x51\x27\x88\x19\x44\xC4\x64\x5F\xDF\x6B\xC4\x89\xF1\x27\x50\x52\x65\xE7\x40\x4F\xF6\xE5\xA2\x27\xFF\xCA\xA1\x87\xAF\xC9\x24\x08\xE2\x49\xD9\x32\xC1\x49\x16\x71\x62\xBF\xBE\x71\x32\xB2\xE7\xC0\x85\xFD\x72\x71\x91\x7D\x65\x97\x8A\x89\x7C\x2B\x86\xCA\x9A\xD1\x74\x00\x2E\xE8\xF1\x80\x3E\x7D\x64\x7E\x92\x2A\x63\x52\x55\x26\xE7\x13\x81\xBE\x2A\x63\xA2\x2A\x03\x78\x11\x57\x34\x3A\x51\x65\xF4\x12\x5D\xBF\xEA\x54\x19\x8C\xB4\x95\xCF\x51\x65\x32\x82\x7B\xE6\x74\x4B\x28\xFE\xB6\xC3\x95\x4C\x69\x06\x6F\x97\x64\x4E\x01\xF0\xBF\x49\x38\x5F\x75\xC2\x49\xC8\xE6\xEF\x5F\x88\x36\xE0\x88\xF6\x6A\x25\x13\x11\xA8\x9F\x09\xBA\x4C\x5C\xE2\xE6\xEB\x1C\x53\xE6\x1C\x98\x32\x5F\x2E\xA6\xEC\x57\x16\x53\x3A\x62\x4A\x97\xCE\xD0\x2E\x94\x77\x2D\xEE\x9C\xEC\x53\x77\xFF\x1D\x22\x0B\x4C\x39\x71\x89\x7E\x1F\x3C\x5D\xA6\xF7\xD1\xD3\x7B\xE8\xC9\x39\x89\xBB\x2A\xFB\xD4\xCD\x1F\x8E\x95\x71\xDF\x45\x6F\x7C\x18\x6B\xF0\xD7\x9F\xA5\xFA\x05\x3D\xDD\x43\x4F\xD3\xE6\xD3\x7C\xEA\x70\x6A\x8D\x9D\xC9\x24\x8E\x8E\xC4\xB6\xB8\x72\x69\xFC\x4F\x7C\x21\x3A\xD2\x8E\xF2\xE2\xE1\xD2\x90\xD7\x4D\x8A\x47\xD3\xC7\xA3\x09\xD6\x8F\xE9\xB4\x1A\x33\x07\x8F\x26\xA0\x81\xEF\x7B\x4E\xE1\xD1\x6C\x85\x47\xD3\xC3\xA3\x4A\xF1\x88\xD1\x6C\x02\x1E\x99\x78\x09\x8F\x6A\x06\x8F\x66\x0A\x8F\xAA\x87\x47\x33\xAB\xD5\x48\x7F\x97\x98\xB1\x64\x6F\xBA\x40\x8E\xE6\xA7\x38\x0F\xFB\x28\x21\x44\xCA\x66\xC6\xEC\x3E\xB5\xAF\xB1\x6D\x93\xB1\x77\xD5\x43\xEB\x1C\xFE\x5C\x8A\x91\x34\x3B\x33\x96\x6E\xCA\x36\x36\xF2\xFE\x8C\x78\xBF\x8D\x31\x37\xB9\xD8\x59\x8A\x12\xCC\x79\xFA\x9D\x05\xF6\x6F\x97\xEA\x82\xD8\x7F\xE1\xEC\x12\x05\x65\x65\xF6\x9F\x31\xFB\x2F\x0E\xA8\x4B\xC9\xA0\xDD\x17\x85\x01\x40\x88\x46\x86\x81\xE8\x70\xEC\x77\x3D\x9D\x29\x89\xE3\xDB\x6A\xDA\x65\xC0\xF0\x8D\x7F\x25\x28\x01\xE6\x7F\x7D\x16\xD7\x49\x98\x4F\xE1\x87\x74\x0D\xF2\x18\x23\x1D\x9D\x0E\xCA\xD5\xB0\x93\x7A\xEC\x78\xA3\x22\xF3\x29\x70\xC3\x0D\x63\x9F\x03\x83\x41\xE6\x53\xA4\xCC\xA7\x20\xE6\x63\xFB\xCC\xC7\x32\xF3\x01\xBB\xA0\x63\x3E\xB6\x63\x3E\xFB\xD4\x2E\xF2\xB5\xDC\x4B\x51\x34\x30\x21\x04\xEE\xFD\x76\x6C\xC7\xF2\xEE\xCB\x5A\xA3\x80\x5C\xF1\xCC\xD4\xF6\xC9\xD5\xBA\xDC\xE9\x35\xA0\xE4\xEB\x41\xC0\x3A\x1B\xC9\x35\x4F\xC9\x95\x02\x56\xA3\x67\x78\xDE\x91\x6B\xDE\x91\xAB\x2D\xD1\x87\x02\xC8\x35\xA3\x91\x58\xA2\xDE\xBA\xA0\x23\x3A\x8C\xC6\xCA\xBE\x4E\xE8\x6F\xDB\x91\x6B\x41\xE4\x9A\x13\xB9\x5A\x0C\x98\x80\xC7\xA2\x91\xED\xD0\xB7\x80\x9C\x04\x09\xA3\xAF\x2B\x24\xE8\x79\x48\xF0\xB2\x56\x5E\x52\x50\x67\x46\x46\xBE\x25\x32\xB4\xCB\x99\x77\x0C\xDC\x20\x41\xC6\x1C\xDE\x61\x37\xE7\x1D\xF3\x90\xA1\x53\x64\xE8\x14\x19\xF9\x3C\x64\xE8\x0E\x19\x61\x12\x18\xF1\xC8\xE5\x84\x14\x9C\x95\xA2\x2D\x1D\x4C\xDC\x5D\xF8\xFA\x70\x0F\x3D\xF9\xBF\x0A\xF4\x98\xC0\xDF\xEE\x07\x8A\xFA\xEB\xE5\x6B\x8C\xA2\x38\x91\x3E\x9A\x42\xF8\x6E\x8E\x4F\xD0\xA0\xBF\x14\xB3\x57\x4A\x0C\x48\xEC\x95\xB2\xCD\x99\x3A\x73\x85\xDF\x1F\x38\x73\x3D\x40\x8D\x19\x3D\xD5\x0B\x50\x9B\x23\x7A\xB5\x2F\xCF\x85\xDE\x61\x50\x48\xF0\xD4\x85\xD0\xAB\xDD\x90\xD1\xAB\x01\xBD\x8A\xD1\xAB\xF1\xD6\x53\x44\x2F\xDE\x7C\x52\x8C\x5E\xD5\x43\xAF\xEA\xD0\x3B\x85\x60\x45\x08\xD6\x7D\x26\x28\x53\xA4\x0E\x01\xA9\x65\x0F\xA9\x8A\x91\xAA\x1C\x22\x5D\x5E\xDF\x8C\xDC\x28\x41\xAA\x4A\x91\xAA\x3A\xA4\x46\xE5\x0D\xAB\xF4\x90\x5A\xD2\x85\x29\x1A\x89\xC5\xD7\x40\xAC\x25\xDE\x6C\x53\x07\x01\x02\x01\xA9\xC3\x04\xA9\x9A\x90\xAA\x12\x26\x48\x31\xCB\x58\x66\x47\xD0\x5B\x5F\xE1\x4A\x49\x41\x5F\x10\xE8\x0B\x06\x7D\x1E\x57\x96\x05\xD0\x47\x76\x4C\xA0\xB7\xE9\xCA\xB2\xB4\xB2\x8A\xFE\xCA\x2A\x78\x65\x0D\x5C\xD6\xBA\x22\x59\x5C\x45\x7F\x71\x15\xC9\xE2\x2A\x08\xC5\x02\x17\x6B\xE0\x7D\xBC\xB0\x8A\x39\x0B\xAB\xE8\x2F\xAC\x62\x8A\xF7\x15\x5B\x2D\xAC\x62\xF3\x85\x55\x60\x34\x5B\x5E\x58\xCC\x53\x68\x61\xD9\x99\x85\x55\x4C\x2D\x2C\xDB\x5B\x58\x05\x2C\x2C\x3A\x3E\x50\xBD\x45\x15\xF4\x85\x73\xE8\xA4\xB8\xE2\xF6\xB1\x5A\xEA\x9F\x0B\x9A\x2E\x29\xA3\xC0\x3B\x59\x89\xEF\x8E\x11\xBA\x94\x06\xE5\x5F\x2C\xF4\x14\x31\x75\xDE\xA7\x08\xBA\xC5\x08\x45\xF1\x14\xC1\x92\x16\x96\xF5\xD4\xAF\x9C\x19\x41\x54\xBF\xF2\xB6\x0E\x59\xA4\x6D\xA2\x7E\xE5\xC8\x8C\xA3\xFA\x95\xBB\x1C\xD4\xAF\x7C\x09\xF4\x34\xF2\x50\xCA\x97\xC8\x85\x94\xD5\x2F\x68\x07\x8F\x4B\xA2\xF5\x6D\xA3\xF5\xCD\xCB\x52\xBB\x9C\x86\xC8\x61\xE0\xE8\x30\xC9\xB4\x98\x53\x32\x0F\xA9\x35\x31\x3B\x60\x56\x57\x51\xDD\xCC\x59\xF6\x87\x94\x19\xC8\x6C\x3E\x88\x1A\xDA\x80\x7C\x8A\x45\x93\x63\x84\xAC\x3C\x30\xBB\x6A\x0D\x2F\xED\x50\xBA\x4D\x20\xCF\x92\xC2\xA7\x97\x0E\x57\x0F\x85\x02\x01\x74\x95\x6E\x04\x54\xA7\xAF\x6F\x6C\x47\x95\x0A\xCC\xDF\xC2\x0D\xE1\x5D\x09\x14\x0B\xAC\x3E\x50\xA4\x74\xA5\xD7\x44\x91\x62\x75\xA9\xF7\x85\xD7\xEB\xFF\x9B\xBD\x77\x81\xB6\x2C\xAB\x0A\x43\xF7\xFA\xEC\xEF\xDA\xFB\x9C\x7D\x3F\xD5\x14\x7D\xDA\xD7\x73\xEF\x81\xE3\x5D\x42\x97\xAF\xC6\x78\x52\x4D\xDA\x4F\xD7\x2A\xBB\xEA\xD6\xED\x02\xE9\x97\x11\xC7\x4B\xC6\x4B\xC6\x23\x0F\x93\xA7\xFB\x16\x4D\x9D\x5B\x65\xD3\x86\xA2\xEE\xED\xA2\x90\x4E\x14\xC5\x80\x1F\x90\x3C\xAB\x0B\xA4\x10\xBD\x09\x6A\x7C\xE8\x43\xA4\xA2\x4F\x41\xFC\x80\x22\x7E\x22\x1A\xE4\x35\x8A\x62\x0C\x6A\x34\x28\x86\x7A\x63\xCE\xB9\xD6\xFE\x9C\x73\x6E\xDD\xEA\xA6\x9A\x88\xA3\x6B\x40\xDF\xB3\xF7\x5E\xFF\x35\xE7\x5C\x73\xCE\x35\x3F\xB5\xF1\x27\x20\xDB\x9C\xD7\x9C\x37\x07\x42\xEA\x98\xB9\x4C\x86\xBD\x1C\x54\x0B\x7B\x09\x98\xA6\xCA\xD8\x43\x88\x1C\xFD\x70\x65\xF2\xF6\x7E\xB8\xCA\x8D\x15\x15\x1B\x4C\x57\x05\xAD\x01\x85\xCF\x0C\x4F\x14\x82\xE3\x2E\xC5\x84\x76\x55\x02\x31\x67\x59\x8C\xDB\xC8\x58\xB3\x20\xAA\x7A\x20\xAA\x41\x21\x7C\xCE\x03\xF4\x1C\x5B\xDF\x82\x16\x01\x14\xB3\xF5\xC3\xDB\xA9\x64\x96\xAD\x4F\x16\xC3\x55\x42\x44\xD1\x9F\x3B\x28\x23\x21\x5C\x25\x2D\x5C\xE5\x90\x0C\xE1\x2A\x71\x70\x15\x32\x5B\xAF\x98\xAD\x67\x28\x23\x92\xC4\x10\xC5\x7E\x09\x4F\x62\xF3\x43\xDC\x7C\x35\xD8\x7C\xE9\x37\x9F\xC4\x3C\x2C\xD7\x6D\x7E\xB4\x70\xF3\x25\x6D\x7E\x6B\x03\xC2\xF1\xB1\x13\xEE\x36\xF1\x5B\xEF\x6C\xFD\x23\x3C\x62\xBA\xAD\x5F\xBC\xE7\x11\xFB\x82\xE6\x60\x4E\x38\xBB\x66\x1A\xAC\x61\xFA\x67\x03\xEF\x84\x46\xD3\xC0\x19\xC4\x56\x20\xA2\x9C\x9A\xD4\xDA\xAA\x29\xBB\xDE\x72\xF0\xEA\xC2\x22\x81\x76\xDA\x35\x22\x9D\xB9\x5D\x6D\xAA\x18\xB4\x3D\x3F\xA5\xFD\xA1\x0F\x08\x51\xCE\x6C\x23\xA6\xFF\xB0\xB3\x45\xEB\x5B\x91\x34\x4D\x43\x10\x66\x01\x2B\x2B\x22\xA7\xCE\x7C\xDF\x23\x73\xCE\x86\xE6\xC4\xF0\xBB\xA0\xE7\x90\x70\x25\xCA\xCA\xD3\x33\x91\x4F\xFC\x0C\xC0\xB0\xAB\xB5\x98\x56\x2E\x53\x67\xEB\xEC\xA7\x58\x3B\x73\x61\x5A\x71\xD8\xB5\x9C\x88\x44\xEB\x5E\xE7\x86\xA9\xD9\x3F\xD9\xAD\x28\x44\x60\xC8\x36\xBD\xA9\xB1\x88\x9F\x91\xCD\x87\xA5\x5C\x19\xF2\x1D\x29\x69\x31\xF0\x98\x68\x6A\xB5\xCE\x1E\x79\x14\xF1\x1A\x3B\xC9\xDD\x66\xC6\x6E\x61\xB4\xDF\xB9\xF6\x44\x8E\xAD\x6C\x4F\x64\xB9\xF8\x44\x56\x9E\x43\xA0\x01\xF1\x89\xCC\x86\xD0\x47\xC8\xD6\x31\x67\x62\xC9\xF2\x95\xA6\xF4\x7E\xEE\x44\x0E\xD9\x70\xB0\x4B\x49\x81\x7C\x63\xEF\x4C\x0E\xBB\x33\xB9\xA7\xCB\x48\x86\xBA\x8C\x84\x63\xC9\xB6\x67\x33\x07\xCB\xF7\xBA\x0C\xA6\x70\x83\x33\x59\xB5\x67\xB2\xD7\x65\x14\x50\x3C\x91\x33\x59\xCF\x9D\xC9\x9A\xCE\x64\xCD\x23\xF1\x67\x72\x0C\xDA\x9F\xC9\xC6\xD3\xC5\x10\x4C\xEF\x4C\x8E\xE7\xCE\xE4\x19\x5D\x46\xE4\xBD\xD1\x86\xF8\xB3\xE8\xAE\xFE\xB7\xD9\x66\xC4\xE7\x20\xAA\x5D\x62\x70\x79\xC2\xD9\x09\x2C\x32\x15\x81\xB0\x6F\x15\xE3\x9C\x17\x81\x93\x13\xEC\x63\x14\x12\x1C\x0A\xEE\x72\x86\x0B\x01\x1B\x5B\xBB\xC1\x39\x52\xDB\x7E\x5B\x9B\xFB\xB6\xDA\x7E\xBB\x63\xE6\x9B\x73\xC0\x70\x76\x4E\xB9\x17\x92\x0D\x04\xE6\x35\x72\x38\x3D\x49\x86\x22\x6B\x92\x0E\x4A\xE9\x2C\x6C\x9C\x5A\xD2\xF9\x04\x6B\x08\x38\xE7\xA9\x6B\x78\xB3\xD6\x08\x0C\xE1\x40\x41\x19\x38\x05\xA5\x46\x38\x88\x20\xEA\xE9\x26\xC3\x16\x08\xA2\x4E\x31\x19\xB5\x9C\x1D\x31\x62\x7D\xC5\x24\x85\x33\xAC\x5D\x1C\x29\xC9\x7C\x5A\x15\x11\x17\xCA\x2E\x2B\x5E\x31\xA9\x40\xF7\x80\x20\x22\x8F\x3C\x44\x11\x6F\x38\x84\x07\x29\x9B\xCD\x78\x95\x56\x3B\x1D\xBF\xDB\xEF\x95\x42\x6F\x77\x29\xD8\x9D\x87\x83\xFD\x72\xAF\x42\x66\xD7\x4A\x6D\x7D\x98\x8B\x7B\x82\xE7\xE1\x9F\x2F\x13\x77\xF1\x1F\x20\x25\xF1\x31\x67\x36\xB5\xDA\x90\xF5\x93\x33\x10\x6B\x0D\xD4\x4B\xFB\xAA\x1D\x46\x6C\xC5\xBF\x68\x4F\x38\x46\x95\x42\xC2\xE2\x2E\x05\xD5\x51\x16\x83\x04\x84\x4B\x17\xC9\x41\x0D\xD4\x9A\x5C\x05\x75\x28\x28\x71\xB9\x1F\x42\x44\x9F\x4E\x68\xC9\xE6\x2C\xFF\x0A\xC1\x39\x4B\xC3\xE3\x2E\x27\x90\x66\x98\xBC\x1D\xB4\x07\x4B\x88\xCA\xC7\x5A\xBB\x72\xB6\xCD\x12\x34\xD6\x16\x7A\x98\x21\x50\x9D\xE5\xD6\xE0\xAB\x71\xD6\x5D\x4E\x7D\x8D\x90\xED\xD3\xB2\x9B\x57\xC5\x62\x41\x82\x2B\x18\xE8\x51\xC9\xEA\xF1\x8F\xD8\x07\xD9\xEA\xAD\x4A\x2C\xB6\x14\xE1\x38\x0A\xC1\xA8\x0C\x84\xCC\x94\x0E\xB3\x24\xCA\xB2\x2C\x8B\xD3\xCC\x38\x53\xB0\xBC\xFD\xC5\xD6\x44\xC2\x59\x06\xE5\xA9\xCF\xB5\x3F\xBF\x83\x1C\x2B\x64\xB0\x59\x20\xFC\x7A\x2C\xDC\x59\x04\xAF\x3F\x79\xC4\xB9\x35\xB8\x7D\xD3\xCE\x0A\x22\xA8\x74\xDB\x85\x70\xFE\xEB\x7C\x3B\x24\xB8\x29\xC1\x4D\x09\x6E\x4A\xB4\xFD\x8A\x16\x48\x34\x88\x05\xCD\xC7\x7B\x37\x1F\xDD\x82\xE6\xA3\xBD\x9B\x8F\x6F\x41\xF3\x21\xEE\xCB\x37\xF9\x27\x6D\xE6\x7B\x49\x9E\x50\x2F\x01\x87\xA4\xE1\x8E\x3E\x57\xAC\x14\x8B\xF6\x93\x75\xDF\x41\xAE\x16\x8C\x75\xBF\xD1\x09\x1E\x80\x9B\x0D\x8E\x95\x4C\xFB\x17\x76\x82\x25\xD9\x11\xA3\xF7\xD2\x55\xA1\x20\xEC\x6C\x08\xD7\x01\xB5\x7D\xE7\x6B\x06\x9E\x54\x20\x3B\x13\x37\x39\x34\x71\xEB\x8E\x31\xF3\x86\x54\x8C\x3C\x1A\x7E\x30\xE8\xE3\xE1\xB5\x9E\x7F\xAF\x44\xAA\x84\x12\xEE\xE1\x99\x77\x87\x41\xB2\x95\x70\xEF\x1D\xB0\xBF\xB7\xE2\xE2\xE4\xFA\xC2\xAD\x89\x2D\x4A\xDD\x5B\x36\x36\xDC\x22\xF3\xC2\xB2\x41\xAC\x6E\x6D\x7D\x51\x54\xA3\x5E\x82\x2F\x13\xAB\x4C\x02\x28\x54\x86\x2D\x1B\xA6\x02\x56\x54\x0B\x8C\xC5\x10\xF9\xCB\x51\x11\x88\x30\x49\x92\x44\x26\x49\x44\xD1\x96\x95\xFD\xEC\x1B\xBA\x05\x41\x11\xD0\x13\x83\xD8\xF4\x9E\x50\x7C\xF4\xD4\x20\x42\x81\x08\xFF\x9F\xF9\xB8\xEF\x72\x4D\xDE\x81\x67\xD3\xA1\x60\xB5\x36\x56\x4C\xEB\x74\xC2\x02\xC5\x41\x64\x2D\x1E\x82\x74\xD2\x17\x3B\x52\xB6\x35\x26\x79\x83\xCA\xF8\x7A\x0F\x61\x69\x67\x60\xDC\xEF\xDA\xFE\xCB\x47\x7B\xF6\xD1\xA1\xE1\x85\x6A\x07\x84\xA3\xF1\x37\x99\x73\x23\x49\x86\x23\x49\x06\x23\x49\x9E\xF0\x48\x7E\xB0\x3F\x12\x6D\x68\xF3\xDA\x81\x74\x0B\x7D\x1F\xB6\x74\xB4\x6B\xA9\xCE\x27\x9C\xB3\xF3\x4B\xF1\xC3\xE1\xBA\xC0\x0F\x05\x7E\x18\x4D\xEA\x02\x85\xA3\x0D\x97\x43\x5E\xAE\xC9\xE7\x81\x86\x11\xB9\x1C\x9D\x84\x10\xF2\x09\x8C\xCB\x7F\x8D\xC7\x6A\x9D\x17\xC2\x40\x01\x66\x83\xC3\x73\xE7\x76\xFB\xFE\x42\xCD\x0C\xF0\x37\x5F\xDD\x1B\x20\xA5\xAE\x83\x99\x95\x92\x06\x52\x1B\xAC\xA3\xEC\xB7\x3E\xED\x43\x0A\xF6\xF8\x2C\x1C\x1F\x74\xE3\x23\x07\x1A\x43\x09\xDE\x81\x1B\x9B\x3F\x22\x6D\xB0\x7E\x76\x6E\xA1\x7E\xF9\xBB\x87\x0B\x45\x82\x06\xD1\xC8\xA2\xA5\x08\x86\x29\x82\xB1\x8A\x45\x43\x4F\x0A\x4C\x4B\x0A\x0C\x8E\x9E\x76\xDB\x1F\x29\xC6\xDE\xD5\xD0\xAA\xB7\x2F\xC8\x2A\x1D\x29\x45\x01\xC6\xD1\x08\x67\xFC\xBA\xB0\x2F\x39\xDF\x17\xCD\x6B\xBE\x3D\xC3\x94\xCA\x30\xA5\xEA\x5A\xEF\x53\x1B\x9C\x74\x4E\xF9\xC8\x8C\x23\x26\x9E\x7D\x44\x48\xF4\x94\xC4\x14\x81\xFB\x0A\xDD\xD7\xB5\x8E\xB3\x44\x98\xE8\x3E\x3C\x6F\xF0\xE1\x68\xF7\xE1\x64\x47\x99\x3E\xE8\xEF\x10\xCD\xBF\x28\x44\xE6\x49\xD3\x8E\xE8\x78\xAD\x36\x96\x48\xF9\xA3\x44\x14\xC8\xDE\x43\xB8\x07\xD2\x14\xC8\xFE\x17\x0A\x76\x70\x86\x19\xAB\xAF\x63\x87\xD2\x36\x04\x4B\x78\xDC\x85\x8A\x47\xDA\xC4\xFE\xBC\xA2\xF5\x8E\x8E\x7D\x2B\x27\xDC\xB5\xAA\x32\x9C\x89\x20\x61\x0F\x41\x8A\xE0\xAF\x17\x8B\x66\xA9\xE3\x4F\x51\xA2\x4D\x98\x7B\x4B\x49\x2E\x23\x8D\xE4\x11\x59\xB2\x87\x20\xCB\x63\x09\x44\x55\xEC\xD4\xA3\x28\x32\x76\x72\x58\x3C\x90\xC3\xD4\x9A\x3C\xCD\x99\xAA\xCF\xD4\x09\x4A\x60\x19\xA8\x67\xAB\xE4\x6E\x72\x7B\x48\x1A\x9E\x48\xEB\xB0\xA1\x20\xB3\x79\x73\x44\x9E\xC6\xAE\xA8\xD6\xD7\xD5\x7A\xD3\xE6\x2F\x61\x6D\x75\x6C\x7F\xED\xED\x1F\x7E\xBB\xF0\x36\x1D\x31\x0A\x53\xF4\x15\xC7\x44\xCA\xD2\x78\x63\x62\x3F\x8C\x65\x20\xB1\x6F\xFF\xF0\xDB\x3F\x4C\xAA\x89\x36\x6D\x3D\x24\xBE\x81\x08\x12\x9B\x9F\x76\x13\x4E\x0D\xA4\x08\x84\xA7\x9B\x3A\x6E\x47\x17\xCF\x8F\x2E\xA5\x4A\x4D\x65\x48\x4E\xAD\x72\x1A\x44\x06\xB8\xD6\x90\x39\xB5\x01\xDD\x52\x27\xF8\x27\x71\xA6\xE0\x2C\xE7\xDA\xA4\xA9\x53\xD7\x4E\xC2\xC0\x9E\xF2\xF9\x93\x00\x07\xCB\x53\x60\x68\xCF\x21\x3F\x22\x4F\xF3\x8A\x7F\x1D\x64\xF6\x5C\x83\x2D\xE3\x6E\x75\x1E\x0D\x09\x8F\xA9\xD5\x8A\x81\x5E\xAF\xB8\x57\xA7\xF7\x02\xCD\xC1\x8F\x17\xB5\x69\x20\xC3\x36\xB2\x4E\x5E\x08\x3D\x64\xB1\xB8\x66\x45\xF9\x91\x1D\x17\x3F\xD8\xAE\x3A\x8C\xCD\x78\xCC\x19\x4A\x34\x7F\xAD\xF0\x90\xC3\xED\x63\xE7\x3F\x01\x59\xF9\xDB\x2E\x76\x7D\x0B\x6B\x2C\x48\xDD\x22\x58\xF3\xE0\xBE\x0F\xC4\x81\x42\x54\xA4\xD2\xAD\x10\x60\x1F\x0F\x7A\x62\x8A\xFC\x47\x0C\x91\x9A\x61\x2B\xDE\xAC\x13\x9B\xBF\x84\xB2\x64\x27\xB8\x2A\xA9\x07\x10\x85\xBD\x7B\x05\xE6\x11\xAC\x86\xE3\x7C\x11\xFE\x81\x14\xA1\xE0\x88\xFC\x5A\xCE\x92\x7F\x82\x83\xC8\x41\xCC\x5D\x56\xA9\xF7\x62\x21\x4D\x4B\xEC\x76\x37\xA5\x77\xC6\x35\xE3\x04\x90\xA4\x73\xE3\x2C\xFD\x76\x64\x3C\x89\xF2\x2F\xDC\x65\x94\xB2\xEF\xEB\x4F\xA0\xE7\x32\x91\x30\x21\xC0\xC3\x37\x41\x92\xE9\x18\x02\x12\xC6\x0F\x37\xAD\x2B\x3D\xD2\xFF\x4F\xD0\xAA\x20\xE1\xFE\x03\x1F\x79\xE0\x4B\x71\x00\x87\x11\x73\x1E\xC2\x65\xC6\xB3\x99\x04\xA5\x4F\x07\x4D\xF9\x5F\xB9\x10\x63\xC0\x07\xDB\xBA\x9F\xF6\x7A\xCA\xF2\x2F\xA9\x40\x52\xFE\xD5\x23\x5E\xD2\x3A\xDC\x4D\xE5\x79\xAD\x57\x02\x9E\x99\xCA\xB1\x51\x58\xE8\x68\x9D\xB0\xD7\x32\x7E\xBA\x0F\xA8\xEF\x78\x52\xFE\x02\x4B\x4B\xE5\x75\x9A\xB2\x66\xF9\x2C\x99\x69\xCD\xBA\x44\x8B\xAA\x23\xC2\xAA\x23\xC2\x4C\x69\xFE\x51\x9D\xFA\xD8\x07\x6A\x4D\xBE\xA8\x4E\x20\x5D\x67\xD5\x38\x23\xFF\x4D\xE1\x52\x3A\x8B\x4B\xD8\x30\x6F\x5D\x8A\x10\xD0\xFA\x12\xA9\x43\xC1\x03\xDD\x48\xFE\x41\x6F\xD6\x5F\x37\x18\xC7\xE9\x5B\x37\x8E\xAF\x6B\xC7\x71\xBA\x37\x0E\xBB\x23\x7C\x18\x04\xDD\x49\xFD\xDF\x2D\x45\xBC\x4D\x2A\x10\x10\xE5\x67\x38\x2C\x93\xED\xA2\x63\xBB\x10\x1F\xF2\x28\xE8\xAF\x47\xB9\xFB\x94\x8B\xD2\x01\x72\xCA\xBA\x54\x89\xEC\x2E\x85\xDB\xC6\x3F\x67\x2A\x27\x55\xB1\xAA\xC2\xE7\x73\x74\x09\x98\x8E\x3B\x5F\x1C\x76\x3E\x87\x90\x3C\x4E\xFC\x9D\x05\x27\x30\x95\x27\x38\x14\x0B\xB7\x18\x60\x8B\xEC\x7B\xB8\x5E\x84\xAC\x85\x8B\x59\xCA\xFD\x6B\xCE\x65\x60\xBA\xDC\x4E\x8A\x53\x65\xBA\x70\x32\xC3\x26\xB1\xA5\x29\x36\xA2\x17\x34\x22\xCD\x20\x41\x14\xC4\x14\x3E\xDB\x7C\x95\x10\xDB\x3D\x32\x17\xF4\xC8\x5C\x1B\xF5\xA1\x25\x73\x41\x8F\xCC\x05\x2D\x99\x33\xEF\xA7\xB0\xA3\x41\x27\x5D\x7B\x6F\x2E\xE2\x96\xBC\x65\x97\x73\x3E\xA7\x8B\x95\xFC\x25\x8E\x07\x70\xD1\x50\x1D\xA1\x11\x14\xAD\xC6\x5D\x26\xB3\x76\x47\xA1\x64\x77\x9A\x2D\x95\x49\xB7\xE5\xA2\xAB\xB1\x5B\x77\x59\x0B\x26\x39\x28\xBA\x3F\xF6\x4A\xAF\x73\xC9\x59\xFD\x96\x93\xFA\xCD\x45\xEF\xC8\x29\x5E\x38\xB5\x51\x1A\x52\x84\x82\x6C\xC7\xDB\x41\xCA\x2F\xC4\x2A\xDE\xD6\x17\xE4\x2B\x48\x00\x04\xE9\xA0\xC5\x93\x6E\xE1\xD2\x81\xB5\x37\x4E\x5F\x5F\x47\x0C\x2F\x21\xC2\x8B\x9E\xBA\x9C\x6D\xA4\x3A\x0E\x39\x11\x5A\x78\xA6\x8A\x0D\x9B\xA6\x0E\xC2\x6D\x72\x80\xCF\x94\x2C\xB6\x18\x46\xE8\xD6\x06\xBF\x49\x9F\x98\x33\x22\x9A\x4A\xF7\x17\xD4\x5A\x84\xAD\x51\x3E\x9D\x78\x1D\xE1\x22\x65\x33\x09\xC9\x9B\x7C\x9C\xAF\x79\x42\x8E\x52\xB2\xDD\x90\x38\xE6\xDB\x14\xC3\x36\x21\x82\x6C\x8A\xAD\xC8\x3D\x5B\x11\xC6\xDE\xE5\x95\xB5\xA0\x9E\x2D\x91\x8D\x56\xC7\xDA\xE0\x6E\xB5\xC4\x57\x39\x87\xF3\x63\xAF\x49\x7E\x55\x74\xAF\x80\xE3\xC1\x92\xB4\xC8\x74\xFB\xD9\xF2\x60\x35\xE2\x66\x0E\xE2\x69\xCD\x66\x1D\x7C\xFA\xDB\xB2\x81\xFC\x6E\xE2\x03\xCC\xDD\x32\x81\x04\x77\xA9\xA0\x17\xC8\x6E\x8F\xF8\x97\xB3\xDB\xE5\xFD\xF8\x3B\xB2\xAC\xC7\x10\xFE\xC8\xEB\x39\x0E\x9F\x68\x7E\xA4\x2E\xFF\x31\x6E\x47\xC1\xD9\x8F\xCF\x41\x68\xD5\x06\x84\xB4\x1E\x53\x1B\xAC\x6F\x6D\xB9\xC8\x8A\x25\x8C\xBF\xE3\x35\xF5\xF8\xAB\x82\x60\xE7\x85\xFF\x18\xF0\x6F\x10\xBC\x88\xC8\xCF\xF8\xFB\xDB\xA0\x39\x14\x95\xD4\x19\x62\x2A\x90\x1B\x93\x5A\x51\x20\x61\xB2\x56\x56\xD8\x24\x05\xA5\x62\x9B\x8E\xF2\x7B\xDC\x85\xBF\xB3\x1F\xE0\xF8\xB6\xA7\xD8\x04\xE1\x85\x74\xA2\xA8\x8D\x5E\x84\xC0\x1F\xA1\xE1\xDF\xC4\x20\xA8\x06\xE5\x74\xF9\x7A\x06\x36\x69\x85\x05\x0A\xF0\xFA\x7F\x6E\x9E\x03\x69\xE5\xC6\x84\xB5\xAC\xD2\x8D\x81\xF5\xAA\x10\xD2\x90\xB9\x77\xE6\x6E\xFE\xDB\x23\xEE\x88\x70\x79\xD4\xE8\xEE\x45\x4D\x39\x5F\xCC\xD4\xDF\x10\x6A\x08\xC9\xDA\xD1\x67\x20\x24\x10\xF6\xE3\x76\x60\xDC\x46\xF5\x07\x0E\xE0\xC4\xFA\xBE\x84\xD9\x16\x61\x93\xE6\x08\xA7\x31\x08\x38\x9E\xBF\xA6\xDC\xD0\x6E\x65\xBA\x5E\x35\xA8\x29\x5B\x3C\xF6\xF1\x45\x3B\x7C\xA1\xEB\x01\xBA\x18\x61\x8A\xA6\x1D\xD7\xA8\xE9\x9D\xA1\x70\x58\xC2\xEF\xFC\x11\x99\xD3\x8D\xE0\x3D\x81\xE6\x4B\x81\xC0\xFC\x10\x52\x26\x56\x9E\x1F\x6C\x95\xE7\x30\xA7\x3C\xCF\x29\x8A\xAD\xBB\x2B\x29\x9D\x9A\x5C\x74\x2E\xA3\x20\xF1\xE8\x91\xCA\xC7\xA4\x6D\xB5\x32\x83\x63\x0A\xAB\x90\xD6\x84\xCC\x31\x95\x8B\x6A\x9A\x73\x54\x43\x81\xE4\x4C\x0E\xD4\xF9\xBE\x99\xA0\x1F\xFE\xC5\x5C\x8C\xF0\xA8\xE2\x5C\xFD\x56\x34\xEC\x81\x2E\x39\xFA\xC3\x80\x0C\xB1\x23\xB1\x42\x88\xD4\x2F\xE4\xD0\x68\xAD\x21\xBD\x3D\xDF\xD4\xA1\xD5\x7C\x1E\xF3\x14\xF4\x5A\x10\xD8\xEF\xF8\xD0\x7B\x7F\x27\x3C\x5D\x4B\x7B\xF0\x21\x90\x67\xDD\x23\x28\xF7\xE3\x6C\x25\xF9\xE6\x45\x43\x88\xC4\x32\xB4\x7A\xA3\x10\xEA\x28\xBD\xEC\x55\x0F\xB1\x7A\xD8\x56\xD7\x7B\x7F\x92\xBE\xE5\x41\x0F\x14\x5C\x49\x51\x9E\x6E\x8B\xE8\x61\x0C\xA2\x50\x2B\xEF\x73\xCC\x5E\x25\x05\xC5\x46\x3B\x14\x48\x5B\x9E\x03\x79\x16\xD7\x91\x32\x03\x25\xFE\x49\x92\x67\xB3\x6F\x9B\xDE\xD8\xA2\x37\xB1\x5A\xDB\xF1\x43\xA0\xCF\xBA\x28\xCD\x65\xD3\x46\x28\xCE\x29\x75\x2C\x68\x88\x49\x64\xD2\x96\x92\x23\x9F\x22\x2B\x52\x8A\xA7\x1C\x23\xF9\x3F\x4D\xC4\xAB\x1F\x74\x85\x22\x01\x84\xDD\x3E\xC8\x36\x10\x40\xF8\xF4\x3E\xF4\xF6\x21\xB2\xD1\x39\x3C\xD5\x1E\x6A\xF6\xD8\x92\xC6\xFE\xF8\x9F\xFD\xF6\x9F\x7C\x63\x03\xD1\x59\x4A\xC1\xCC\xD9\xDF\xF8\x96\x35\x32\x3F\xA2\xF8\xCE\x6A\x18\x39\xA5\xD6\xE0\xD6\xBF\x8E\x7A\x11\x80\xBA\xA0\x2A\x6A\x42\x92\xFE\xD8\xF4\x62\xE5\xA8\x09\xE5\xC9\x14\x6D\x1C\x3D\x3C\xF9\x8E\xE3\x6A\xBA\xA4\x8A\x14\x42\xC6\xD7\x73\x96\x4E\x31\x95\xE7\x7B\x72\x97\xA5\x1E\x28\xDB\xB1\x70\x4E\xFD\x13\x50\x10\xB1\xEE\xEB\x38\x57\xED\x41\x98\xB3\x8D\x8B\x37\x49\xF3\xE9\x42\x19\xE6\xB5\xDE\x3C\xD1\x16\x8D\x91\x4B\x0B\xAC\x18\x1B\xE5\x79\x38\xCE\xF0\x48\xF3\x14\x7E\x3A\x2E\x1E\x35\x47\xCA\x6E\x83\x41\xD4\xB2\x9D\x27\xA7\xE3\x21\xF0\x9C\x54\xFE\x96\x59\xE1\x81\xC3\xB9\xE1\xC2\x36\xA7\xAA\x6A\xAC\x68\x70\xBA\x12\x14\xFE\x5D\x2F\x14\xF8\x8C\xAA\xEB\x6E\x8A\xA1\x63\x62\x13\x5A\x00\x8A\x92\x01\xE1\x60\x96\xC4\x54\x51\xF0\xB4\xD0\x8B\x74\xD1\x7A\x21\xB0\xAE\xF9\xBD\x58\xAC\xB0\xEF\x3D\x13\x53\x10\xED\x55\x21\x47\xEC\xF7\x96\x7F\x7C\x61\xC3\x57\x76\xC1\xDC\x95\x1D\x07\x71\x05\x41\x01\xCE\x59\x57\x44\xA1\xD6\x39\xC2\xBA\xE0\x38\xFD\x89\xBF\x97\xF5\x7E\x42\x6D\x44\x75\xFF\x7C\x7E\xE6\x79\xF0\xE0\xC3\x60\xD2\x5D\x9C\x1A\xC4\xE9\x6F\xFD\xF6\x29\x44\xBF\xEA\xF5\xD0\x77\x0E\xEF\x42\xAB\x83\x4F\xCF\x26\xBD\xD6\x9C\x92\x41\x31\xB0\x89\xAF\xC7\x5D\xC3\x03\x3B\xA6\x39\x38\xEE\x90\xC3\x5B\xE3\x1F\xE2\x0C\x83\xF6\x48\x65\x8B\xA8\xC8\x9D\x99\x92\x43\xCC\x13\x20\x9F\xAA\x52\x97\xD2\xD5\x71\x2F\x29\x9D\xD8\xC9\x94\xAD\x21\x38\x81\x1E\xFE\x39\x43\x51\x1F\x89\x3D\x89\x29\x20\x5E\x6B\xE8\xE5\x8D\x72\x02\xBA\x65\xAF\x33\xE6\xCC\x5C\x50\x88\xA0\x56\x78\x46\xB3\x12\x56\xBA\x34\xE9\x55\x04\xAA\x8A\xC9\xC1\xC3\xA6\x4D\x95\x73\x92\x35\x7D\x28\x48\x08\x18\xEB\x82\xC1\x71\x34\xA9\xC6\x58\x55\x1D\x85\x31\xA8\xBA\xC4\x42\x25\x16\xEA\x63\xE6\x7A\x21\xA1\x5C\x93\x25\x94\xA4\xBE\x60\xC4\xA4\x21\x8D\x8A\x00\xF2\x4A\x42\x51\x85\x30\x26\xCC\x24\xBC\x26\xBC\x5C\x2F\xC2\xD6\xF1\xC6\xD9\x7F\x3A\x5B\x11\x02\xD8\xB1\x1B\x7D\x09\x39\x8C\x10\x27\x9D\x3E\x1A\x39\x63\xBD\x26\x0F\xD6\x4B\xF8\x67\xB5\x96\x9B\xD8\xF7\x41\xFC\xCF\x6A\xBD\xBC\xB9\x4E\xFC\xB5\x84\x25\x96\x1D\x86\xA4\xA4\x5E\x69\xB1\x6F\x15\x96\xF1\xD3\x32\x67\x68\x77\xB7\xB1\x75\x38\x59\x2F\x34\x7E\x49\x60\x19\x27\x12\x4E\x9C\x59\x0D\xE2\xDD\x6A\x9B\x93\x91\x10\xCD\xE1\x5C\xE8\x71\x2E\x86\x95\x19\x9C\x5B\x75\x1A\xD4\x96\x5C\x41\x08\xAB\x6E\x16\xA1\x81\x65\xC4\xB7\x65\xBA\x19\xA8\x25\x2C\x31\x81\x2C\x2B\x83\x53\x09\x38\x54\x6B\x59\x51\x98\xC7\x12\x85\x82\x92\x54\xC8\xBC\xCB\x59\x43\x97\x4D\x1D\xE0\xD3\x0B\x77\x91\x1B\x39\xDE\x0D\xE5\xC9\x2E\x68\x81\x43\x4C\xEF\x03\x33\x83\x98\xA4\x40\x37\xE6\x33\xBD\x1B\xF4\x77\x88\x41\x3C\x83\xBE\x52\x97\x48\x54\xF9\x73\x8F\x70\x06\x52\xE1\x63\xAE\x92\x81\xAA\x0B\x72\xB8\xD8\x45\x4B\x3B\x42\x11\x70\x24\xE1\xB7\x08\x4A\xB7\x2F\xB6\x8E\xC8\xCB\x82\x2F\xF4\xDF\x26\x38\xAF\x04\xBB\x66\xE1\x5B\x7B\x59\x90\x4E\x2C\x70\xFE\x59\xC1\x8C\x33\x81\xF4\x11\x2C\x8E\xE1\x58\xCB\xBB\xD5\x77\x0B\xFE\xAD\xEF\x56\xAF\xC5\xFA\xE2\x9E\xE0\x5B\x05\x5F\xFD\x7D\x8B\xE0\x7B\xBE\x47\x05\xB3\x96\x6F\xEB\xDA\xC7\xB3\xE8\xB2\xA8\x03\x26\xC2\x1C\xD3\x59\xAC\xE1\x10\x03\x0E\xE3\xC6\xF1\x97\xCB\x86\xF8\x5F\x10\xF6\x51\xD1\x94\x9F\xBC\x44\xFE\x45\x6B\xF2\xF0\x71\xBF\x1C\x20\xCB\x3F\xBE\x44\x92\x27\x35\xE8\x03\x68\x60\x5F\x9D\x33\xCE\x3B\x5A\x45\x46\x4F\x3C\xFD\x2B\xD5\x2D\xFE\xE1\x99\x68\x98\x7B\xA9\xD4\xBD\x3F\xA5\xE6\xD0\x78\x33\xAB\xEF\x42\xE3\xE9\x99\xE8\x21\xEE\xCE\xFF\x8E\x36\x60\xC8\x41\x0E\x45\x0C\xEC\xC7\xAD\x69\xE9\x0F\x82\xB4\x07\xDD\xC2\xEB\x76\xE1\x3B\x5B\x28\x26\xF3\xBA\x35\x0F\x96\x6B\x12\x7C\x15\xC4\x38\xE4\xC9\x27\xE5\x0E\x5D\x21\x69\xB7\x04\xBA\x76\x31\x4E\x28\x64\x93\xB7\x65\xDE\x23\xC6\x89\xE2\x41\xE5\xED\x28\x5D\xA4\xE4\xB2\x1F\xDD\x84\x2D\x56\x71\x94\xCA\x05\x38\xC1\xA3\x2F\xE8\x19\x30\xFB\x00\x27\x72\x4D\x96\x34\x73\xBE\x77\x9B\xA5\x61\x47\xE4\x5D\xD8\xA8\x6F\x0E\xF0\xCD\x1A\x48\x14\x48\xB5\x45\x68\xFF\xB3\x4B\xBC\x74\xBA\x7C\xE4\x22\x6D\x9C\xDB\x70\x45\x6D\x75\xD7\x25\xAD\x82\x53\x76\x72\x09\xAE\x4D\x77\x8B\x72\xB8\x8D\xE5\xD9\xED\xFD\x5B\x7B\x71\x63\xA0\x87\x77\x36\x70\xCA\x1B\xD9\x53\xDE\x38\x09\x5C\x76\xCA\x1B\xD9\x2A\x6F\x10\xFE\x58\x79\x43\x20\x43\x21\x66\x6A\x85\xAB\xEE\xB1\x58\xF1\x72\xAB\x19\x94\x0C\x39\x4B\x40\xDE\x62\x24\x22\x40\xE8\x96\x9B\xB8\x09\x5A\x6E\xE1\x96\x3B\x44\x0A\xC2\xC6\xF6\x3D\xAB\xD6\x90\x0E\x53\x5A\x6E\xF6\xE4\xBA\x78\xD1\x67\xAA\x13\xDD\x2A\x89\x6E\x95\x3A\x45\x91\x5A\x93\xAB\x2E\xA9\x8C\x47\xE4\x83\x2D\x1E\x97\x1C\xE9\x17\x51\x38\x67\x0C\x4E\xF8\xA6\x7E\x64\x3F\x1D\xB4\xCC\x07\x11\x90\xC0\x07\xD2\x76\x93\xAF\x38\xE3\x3F\xB9\x23\x93\xAB\x60\xE2\xD1\x56\x21\xDA\x12\x15\x74\x08\x2B\x3B\x17\x0A\xF6\x78\x9C\xDF\xA7\xCB\xA2\xB3\x8C\xB9\x45\xFB\xC4\xF4\x24\x70\x2A\x2E\xC5\xB3\x58\x25\x0F\x4A\x52\x4D\xD0\x88\x5F\x79\x91\xC3\x3D\x2D\x5E\xC5\xDE\x60\xCD\x5B\xBD\x8C\xA9\x5B\xF1\xD2\xB1\x96\x6A\x52\x6B\x17\x4C\x98\xC5\x4F\x64\x42\x2B\xE4\x0C\x28\x10\x21\x0A\x22\x4E\x51\x8A\xA2\x4D\xE4\x44\x1B\x0A\x00\x4E\xA5\x48\xC4\x09\x7B\x82\x88\x40\x41\x44\x74\x32\x4A\x4F\x00\x09\x39\x4D\x4D\x84\x2B\x19\x51\xCB\xEA\x28\x07\xE2\xE9\xAA\x47\x58\x3D\x6A\xAB\x8B\xBD\x3F\x2D\x12\x71\x5C\x46\x8C\xBE\x88\x03\x8A\x92\xA6\x38\x38\x8F\x7A\x12\x8E\x18\x48\x38\x62\x46\xC2\x11\x43\x09\xC7\x06\xE4\xBB\xC6\x1E\x86\xAC\x0B\x18\x88\x39\x02\xC5\x1C\x71\xD6\x91\xE5\xA8\xE5\x03\x15\x75\x39\x75\x76\x91\x55\x82\x2F\x98\x73\x8B\xCE\x54\xCE\xD2\x08\x12\xAF\x25\x14\x6E\x0F\x83\x4E\x3F\x21\xFB\xEA\x64\xC1\x61\xCC\x15\x2B\xA4\x24\x37\x25\xB1\x29\x1A\x73\xE2\xD4\xC9\x1C\xFF\xB2\x65\xDB\x02\x2F\x45\xD0\x75\x0C\x56\x2C\x88\x7C\x6B\x97\xB4\x48\xE0\x71\x99\x3A\xC6\x45\x32\xE3\x22\x5D\x42\x06\x8A\x33\x94\x87\x86\x7C\x7E\x29\xAC\x5E\xE9\xD0\x1D\x45\x10\x4A\x8A\xE9\xE4\x05\x65\xDC\xC8\x7D\x92\x4B\xE4\x65\x17\x0C\x1C\x29\xEA\x14\x92\x13\x6C\x39\x30\x33\xD4\x68\x6E\xA8\xC7\x79\xAC\x37\xE8\x9A\x4D\x02\x90\x7F\xBE\xC1\x7C\x04\x44\xBD\xF9\x28\xD3\x1B\x2C\x72\x65\x82\x72\xF5\x9B\x7F\x63\x54\xBA\x2D\x2F\xC8\x57\xF4\x78\x44\x8A\x34\xAC\xBD\x1C\x5A\xE9\x1E\xF3\x16\x4E\xEA\xB8\x92\x10\x55\x09\x81\x56\x8B\x2B\x11\xE2\x4A\xE2\x70\x25\x42\xD1\x89\x4A\x91\xD6\x2B\xDE\x5B\x1D\x10\xB5\x90\xEC\x9C\xB4\x25\x24\x55\x48\x7E\x6F\x8C\x2B\x72\x88\x2B\x09\x56\x4F\x3A\x84\xD8\xFB\x93\x9E\xC3\x15\xED\xC3\x88\x87\x88\x2B\xA1\xC3\x95\xB0\x4A\x3A\x5C\x49\x7A\xB8\x22\x1D\xAE\xE8\xB3\x95\xA6\x27\xC2\x15\x7E\xE2\x75\x6A\x71\x85\xC2\x69\x17\x28\xF3\x77\xD3\x1C\xE3\x34\xDB\xE0\xBC\x0B\x74\xEA\x9A\xB1\x25\x61\xAD\xAD\xBB\xAF\xAE\x52\x7C\xB1\xC1\xEF\xCF\x54\xA9\xD3\x11\xA6\x1E\x5B\xA4\x3B\x46\xF9\x17\x47\x6C\x77\xB8\xC2\xC9\x04\x34\x82\x59\xC0\xEA\xCD\xBE\xAA\x51\x43\xBA\x5E\xB0\x3F\x69\x7B\xEE\xE3\x36\xD7\x19\x6F\xB4\x99\x90\xB8\xCE\xF6\xF6\xB8\x85\xC4\xE3\xEB\x3C\xA2\x8B\x38\x88\x8E\x17\x21\x9D\xEA\x8E\x31\x68\xC0\x4C\x50\xC2\xF7\x98\xC0\x7E\x0E\x9C\x55\x0E\x99\x21\x06\xC5\x18\xB4\x07\xC5\xCC\x99\x0A\xAF\xB3\xE2\x5D\x93\xF9\x8E\x9B\x07\x9B\x76\x92\x32\x3F\x9F\x9D\x09\x68\xC8\xA7\x38\x76\x79\x13\x63\x27\x3B\x17\x3F\x5E\x7D\xA3\xF1\xBA\x4B\xC7\xC8\x59\x4D\xE8\x27\x38\x78\xE5\x07\x4F\x11\xFC\x0F\x76\x0E\x7F\x7C\x35\xD0\xBF\x40\xD0\xF8\x6A\xE4\x13\xAA\x49\x48\x38\x05\x8A\x0B\x59\x57\x3A\x8D\xBF\x84\xE2\x6E\x97\x59\xF6\x49\xA9\xFB\xC3\xFD\xD5\xFD\x31\x6F\x51\x6C\xAC\x24\x85\x43\x4C\xBA\xF3\x78\x63\x82\x72\x1F\xC3\x8C\x6B\x12\xFB\x62\xCC\x0D\x9D\xAA\xDD\xD9\x77\x69\x07\xC7\xA7\x18\xB0\x90\xCF\x0E\x9F\x98\xBA\xBF\x1D\x44\xAB\xEE\x0F\x9D\xBA\x3F\x26\x75\x7F\xEC\xD4\xFD\x31\xA9\xFB\x59\x93\xE0\xC6\x10\xB3\x19\x80\xA6\x21\x73\xEF\x94\x65\xB7\x7C\xD5\xC5\x05\xEA\x7E\xCD\x8A\xF7\x84\x11\x8B\x4C\xFA\x49\xDD\x9F\x38\x44\x63\xEC\xEA\xD4\xFD\x84\x61\x75\xE2\xD5\xFD\xB2\xA7\xEE\xA7\x70\x8D\x49\xAB\xEE\x97\x37\x50\xF7\x27\x6D\xAF\xFB\xA8\xFB\xBD\x03\xA8\xEC\xA9\xFB\xA5\x53\xF7\xEB\x39\x75\xBF\x66\x75\x3F\x15\xF8\xFE\x48\x99\x6D\x71\x41\xF6\x0D\xBC\x2B\x0D\x9D\xE1\x51\xB8\x8F\x94\xE4\x0D\x8F\x50\x3C\x27\xDF\x12\x94\x74\x0E\x36\xE5\xFF\xE3\xD2\x23\xCF\xC9\xB0\xFB\x04\x84\x71\x43\xCF\xDB\x18\x23\x09\x68\x64\xDD\x4A\x5E\x01\xE9\x18\x66\xED\x18\xE6\x4E\x7C\xED\x85\x17\xA1\x71\x04\xBD\x10\x23\xA4\xF0\x70\x55\x48\x1B\xD3\x93\xA2\xE2\xE3\x45\x50\xFE\xD6\x23\xEE\x22\x94\xF8\x8B\x78\x4D\xFE\x83\x9A\xA2\x88\x66\x56\x9D\xB3\x63\x1C\x60\xCF\x7F\x10\x32\x48\x79\xE8\xB8\xD1\xEA\x1C\x72\x92\x97\x2F\xF6\x63\x60\xD7\x2E\x04\x2B\xA7\x79\x80\xA0\xA9\x0C\x45\x25\xAD\x72\x1B\x54\x85\x3A\x0A\x8A\x42\x41\xEA\x67\xAB\x83\xF5\xE8\xD8\xF6\xFF\x52\x04\x30\x82\xF0\xB9\x32\xF9\x5F\x0B\x4A\x21\xC7\xF5\xFC\x8D\x1D\xFB\x85\x44\x7C\x61\xC6\xDC\x37\xA7\xF2\x20\x9F\x17\x5C\x1E\x50\x9B\x9C\xA7\x01\x41\x2D\x2B\xFF\xF4\x12\x87\xB9\x75\x56\xE2\x50\xAC\xBB\x1C\x78\x8E\x22\xC7\x6B\xB2\xA4\x34\x1A\x40\xAA\x8F\xCD\x5A\x59\x0D\xCA\xEA\x93\x93\xF2\xDD\xAF\x62\xD5\x25\x0A\x82\xD8\x72\xF9\xAE\x57\x91\x61\x00\x42\xE9\xD8\x5E\xBF\xFE\xD3\xC1\xFD\x75\x39\xA9\x23\x52\xDB\x51\x56\x07\x24\x13\xF6\xBF\x5D\x57\x27\xB6\x78\x23\x48\xF7\x44\xB6\x43\x2F\xE0\xE4\x0F\x59\xF9\x11\x5A\xDD\x31\xC8\x4A\x41\x4E\x6C\x31\x9B\x8D\x92\x31\x3F\x45\x91\xAC\x28\xCD\x85\x15\x0D\x3B\x33\x71\xD8\x5A\x27\x6F\xD6\x85\x13\x7E\xC6\x50\x40\x50\xFE\x07\xF2\x22\xF9\xD5\xD6\xC6\xA4\x52\x83\x34\x30\x30\x76\xDF\xA4\xCF\x88\xD3\xD4\x0A\x89\x28\x8E\x07\x4B\xC4\x4E\xD2\x4C\x6E\x37\x1E\x14\xBC\x4E\xAC\x0D\xF3\xDA\x09\x29\x61\x27\xA4\xFC\x9B\x44\x8C\x9F\xD6\xD6\xFE\xCD\xD6\xD6\xEE\xA1\xAF\x75\x49\x4B\x6A\x45\x15\x0C\x72\x0B\x33\x8A\xD9\xDC\xB3\xCD\x55\xE1\xD8\xFB\x02\x59\x66\xD2\xE2\x8E\x58\x55\x2B\x8F\x42\x01\xAA\x1E\x63\xC5\xF1\xAC\xA2\x03\x79\x8B\x71\xA5\xF2\xD8\xC0\x18\x89\xCD\xB8\xA7\xB3\x85\x51\x25\x21\xAF\xC2\xF6\x32\xD4\x5F\xA0\x50\x8D\xC4\xEC\xA1\xAF\x35\x30\xAE\xA8\x35\xA7\xB5\x1C\xF3\xC0\x15\x85\x75\xF6\x03\xE7\x61\x1B\xBA\xCA\xC0\x21\x07\x34\x4A\x32\x6D\xC1\x89\x46\x2E\xA9\x4E\x08\xE6\x78\x35\x26\x20\xAF\x4A\xAF\xB4\xD6\x95\x01\x59\x68\x4A\x48\x63\x5C\xA0\x58\xBD\xA8\xFD\x49\x11\x12\x4F\x35\xA3\xE5\xD6\xBC\x28\x25\xF9\xD6\x81\xAA\x25\x4F\x5C\x4C\x27\x50\xB4\x77\x20\x88\xF4\x64\xD0\xCD\x27\x9D\xF2\xE6\xA7\x55\xC8\x9D\xE4\xA4\x0F\x3F\x4E\xD4\xC1\xE0\x02\x63\xF7\x39\xF6\x3C\x9A\xB0\x59\x8E\xC1\x2F\x6C\x22\xB2\x70\xF6\x13\x52\x80\x77\x53\x10\x8B\x0B\xF5\xB2\xE3\x46\x0C\x87\x4F\xB1\xD6\x77\x84\x14\xED\x7F\x9F\xDA\xF7\x6C\x37\xF6\xF6\x0D\x63\xF0\xD1\xFD\x3E\x2C\xA4\x8F\xB9\xFC\xB2\xA6\x16\xF6\x19\xA7\x28\xA5\x91\xB4\x02\xC4\x39\xFB\x3D\x3B\x3B\x7A\xCA\xB4\xD4\x9C\x20\xB9\xD4\xFC\xB9\x52\xE2\x82\xDA\x8E\xCE\x43\xF0\xEE\x5A\x5C\x6D\x4D\xB1\xC4\x31\x78\xF4\x6A\xAD\xED\xF5\xEB\xF7\xDE\xEF\x18\xF8\xEB\xD7\xAF\x5F\x8F\xA7\x20\xB7\x8A\xE0\xBE\x80\xFE\x3D\xF0\xD3\x36\x78\x8E\x7A\xC7\x5B\x5E\xBD\x13\xBC\x99\xAF\xDD\x8E\x6D\x7F\x0D\x11\xFF\xE0\x31\x57\x24\xC0\x2F\xD0\x56\xF8\xAA\x2B\x75\xF0\x6E\x6C\xBA\xD2\xF6\x7E\x4E\x8A\xA5\xED\xCE\xCE\xB5\xEB\xF1\x49\xAE\x77\xD9\x65\xFE\xE4\xB7\x78\xF6\xB7\x0D\x21\x0B\xC0\x69\x3F\xEC\xEA\x37\x80\x6A\x68\x48\xF7\x4E\x6B\x6D\x1F\xFF\xC9\x2F\x6E\x6A\x69\x57\x1F\x6A\xEC\x23\x0F\x37\xEF\xAC\xA3\xFB\x82\x17\x7D\xEF\x03\x2B\xF7\x7D\xE8\xDE\x2B\x75\x4C\xFA\x9C\x6B\xC1\x14\xF4\x16\xB7\x79\x76\xF7\x18\x7C\x13\x04\xEF\x3E\x86\x13\x1A\xBF\xF2\xD2\x7B\x5C\x0F\x9F\x7A\xCF\xE5\x9A\xC6\xE0\x9E\x3F\x7A\xEF\x95\x2B\x75\xF8\xD8\xBB\x49\xBD\x75\xE6\x95\xEF\xA9\x93\xFB\x82\x00\x0E\xBC\xFC\xE7\x7F\xEB\xDE\x2B\x75\x7A\xB9\xCE\x20\x85\x18\xB2\xC7\x2E\x43\x00\xC9\x63\x10\x3E\xD6\xAB\x1B\x1C\xBD\xFC\xE6\x3A\x20\x6D\x5B\x80\x63\x88\xAF\xB8\xA6\xBF\xF7\x67\x1F\xFE\x60\xFA\xC6\x9F\xBE\xF7\xCA\x7D\xFF\xF6\xE1\x6F\xFD\xA2\x9F\x99\xFE\xC2\xBD\x97\xAF\xDC\xA7\xFF\xEA\xBB\xDE\xF0\x86\x37\xFC\xE6\xBD\x97\xAF\x40\x0C\x5C\xEE\xBE\xBF\xF7\x95\xFF\x71\xF9\x4F\x7E\xEA\xDE\x2B\xF7\xFD\x8E\xFA\xF9\xEF\xFC\xDA\x13\xEF\xC5\x72\xFF\xF0\x75\xF5\xEB\x9F\xB5\xF1\x2B\xF8\xF3\x75\x5F\x83\xFF\x3E\x7E\xEF\xE5\x2B\x97\x2F\x5F\xB9\x5C\x07\xDD\xD0\x20\xBA\xEF\xC8\x73\xFE\x64\xE9\xCF\x3E\xFB\x0F\xBF\xE2\x0A\x8E\xEC\xF2\x7D\x1F\xDE\x7D\xE3\xFB\x9F\xF7\xFA\x77\x7D\xC5\x95\xCB\x97\x2F\x5F\xA6\x18\x18\xE6\xEF\x0A\xD1\x73\x18\xD3\xDE\x4A\x8F\x35\x8A\xC4\x4C\x82\x2A\xBF\xF9\xA2\x0F\xCF\xD6\x4B\x67\xF4\xA1\x2F\x0A\x33\x4A\xEC\xAC\x2E\xF4\x35\x6B\xAE\x09\xFB\x6A\xFA\xEF\xA3\xAC\x2B\x9F\xB8\x84\xA0\xDB\x20\xAD\x6C\xAA\x98\xB3\xC3\xB2\xEC\x9E\xB0\x71\xE1\x17\xAF\x93\x00\x97\xC8\xA3\xEA\x28\x02\x6A\x8F\xDE\x22\x8F\x5D\xA7\x77\x51\x58\xF9\x17\x35\xA3\x25\x19\x45\x2A\xE2\x7F\x22\x0A\xA2\x48\x47\x28\x22\x48\x4E\x66\x2C\x73\x69\x6C\xC2\x66\x3A\x25\x82\xB6\xBB\x37\x4E\x40\x6E\x55\x49\x4E\xC7\x13\xC5\xF6\x78\xA0\xB1\xD7\xC5\xD4\xA6\xA7\xD8\x1D\x92\x44\xF7\xEC\x34\x64\xF8\xB6\xB1\x0F\x20\x9D\xE4\x30\xEF\x1C\xCF\xDE\x15\xCF\x36\xC8\x58\xD6\xCB\x05\x1C\x26\xE4\xEF\x70\x7C\x83\x98\x8D\x27\xE5\x16\x24\x2E\x2A\xBC\x0D\x4E\x4E\xF0\xF4\x01\x69\xEF\xFC\x86\x3A\x6B\x20\x3B\x4B\x51\x56\xE8\x04\x8C\x0F\x05\x41\x95\x91\x23\x84\x1B\x12\x76\xF0\x25\xD8\x56\xCC\x4C\x3F\x0D\x3D\x85\xC4\xEE\x24\x5B\xA4\xBF\x88\x0F\x05\xA2\x9E\x1D\xBA\x51\x47\xC1\x0C\x87\x6E\xF6\x1A\x3A\x29\x52\xB8\x23\x4A\xE4\x40\xC3\xCC\x6C\x00\x99\x0D\xEE\x9F\x90\xE1\xB8\xC4\x2E\xBB\x21\x4A\x1C\xA2\xA8\x72\xBB\x23\xAB\xB8\xB7\x29\x19\xA7\x64\xB0\xDF\xBD\xDD\xD8\x3B\x5F\x36\xCA\x84\x0C\xB4\xD6\x0A\xFF\x87\xC8\x9C\xF2\x48\xFF\x88\x43\x3F\xB4\xF3\xA0\x3D\xD0\xB8\x5D\xB9\xDD\x89\x7C\x10\x08\x57\xF8\x13\x33\x85\x05\x16\x56\x5C\xF8\x9A\xE0\x25\xDF\xD1\x78\xDE\x27\x10\x6F\x55\x09\xA4\x90\xF3\x24\xAB\x6C\x71\x96\x2B\x37\xC8\x3A\xB6\xFF\x5B\x33\xFA\xFB\xD1\x7C\xAC\xEB\xC1\x3F\x29\xE4\xEC\xAB\xA0\xFB\xA5\x02\xFC\xCE\x2F\x82\x30\x8E\x35\xFE\x0E\x62\x04\xB6\xDC\x96\x33\x6B\x73\x5D\x4C\xEB\xCC\xBE\x6D\xBB\x19\x55\xC2\x93\xAD\x40\x06\xBD\x7F\xCA\xFF\x10\x94\x56\xE0\xA5\xD3\x2A\xB1\x19\xCD\xCD\x02\x02\x50\x66\x7F\x23\x38\x81\xC0\x13\x57\x39\x64\xF6\x4D\xDB\xCD\x28\x92\x42\x08\x21\x29\xE7\x76\x8E\x05\xCF\x4E\xAB\x04\x72\xAE\x74\x61\x0A\x09\x41\xCA\x74\xE2\x8C\x5C\x5D\xA5\x00\x2B\x05\x2C\x52\xDA\x1D\x3D\xF5\x4A\xCB\x84\x7A\x05\x65\xE3\xC6\x3E\x3C\xAD\x8B\x67\xAB\xA0\x1E\x1D\x0B\xFE\xDE\xA4\x1E\x63\x2B\xAC\x98\xC8\x60\xF4\x00\xD9\xB3\xEF\x56\x25\x8C\x60\x74\xEC\xDE\x57\xD7\x4B\xE7\x61\xE9\x55\xD5\x12\x8C\x2D\x4C\xED\xEF\x06\x67\xED\x5F\x38\xB1\x23\xA1\x9C\xB3\x49\x63\x0F\x43\x0E\x4B\xB0\x04\xE5\x4E\xBD\x0C\xE5\x85\x57\x5C\xAD\x33\x4A\x01\x97\x6D\x4C\x20\x6B\x90\xD1\x4E\x70\x93\x33\x2C\xF2\xF7\x0B\x01\x89\xBD\x73\xA3\x52\x90\x55\x09\x2C\x57\x4B\xC0\xD9\xEE\x0B\x97\xAA\x8F\x03\x1B\x81\x86\x0C\xBB\x7F\xF4\x2A\xC4\x80\x7C\xC2\xB8\xFC\x56\x96\x53\x18\x60\x64\x3E\x72\x33\x94\x73\x33\x44\x18\x5F\x99\xD4\xAB\xDD\xB4\x72\x58\xA1\x03\x64\x05\x56\x10\x31\x93\x06\x92\xB3\x55\x02\xAB\xFD\x19\x1D\x70\x9E\x73\x1A\x32\x9C\xD1\x01\x6C\x0E\xE2\x07\xEB\x31\xC4\xA7\x37\xEB\xBC\x9D\x50\x4E\x13\xCA\x48\xF6\xC2\x12\x1B\x85\x80\x0C\x27\x54\x40\x5E\x65\x30\xAE\x12\x28\xD8\x8C\x87\x71\xAA\x9B\x50\x8E\xBD\x3F\xE4\xE6\xB3\xBA\x68\x3E\x34\xA3\x6B\x01\x9B\x2A\xDD\x15\x04\x9C\xAC\x1C\x27\x54\xAD\x10\x86\xD0\x97\x15\x5B\x9E\xB3\xE5\x37\x54\x2B\xA6\x3F\xED\x2F\xBC\x19\x17\x37\x05\xA3\x4B\xF3\x30\xBA\xD4\xC1\xE8\xE7\x1D\x28\x6D\xB0\x17\x40\x16\x37\x04\xC8\x6C\x52\xAF\xF4\xB7\x27\xE3\xED\x99\xDD\x8F\xDE\xF0\x0F\x40\x06\xD9\xDE\xFB\x91\xF0\x7E\x64\xBC\x1F\x89\xDF\x8F\x04\xC6\x55\xB6\xE7\x7E\xB4\xA3\x5F\x59\x34\xFA\x01\xF8\x11\x99\xF5\xE0\x57\x67\xF6\xFA\x75\x35\x85\x8C\xA1\x70\xE2\x08\xB0\x9F\xE1\xDF\xF4\xC9\xE5\xA6\x85\xA9\x78\xB8\xA3\xF1\x73\x54\x40\x75\x5A\x32\x8C\x67\xE5\xCF\x06\x27\x26\xE5\xB7\x0D\xDB\x20\x39\x37\xB3\xB0\x45\x25\x7E\x9F\x12\x29\x25\x76\xA7\xDC\x72\xFA\x79\x57\xF5\x5D\x94\x45\x01\x09\x30\x35\x49\xFE\x11\xFB\x75\x5D\xBE\x76\xD8\x15\x9D\xC0\x31\xED\x86\x9C\xD6\x79\x11\x58\x59\xC5\xEE\xD6\x89\x05\xE9\x18\x65\xFD\x8C\x74\x5E\x17\x9C\x9C\x1F\xA3\x9C\x6F\x78\x95\x2D\x47\xB9\x4A\x28\xBF\x22\x84\x4B\xAF\xE4\xC4\xBB\xF8\x96\xAC\xCA\xE6\xDE\xC9\x05\xEF\x54\xFF\x1D\xB8\x50\x7F\x64\x1E\x48\x4D\x57\x91\x61\xF9\x32\x26\xFF\xAD\xB6\xD3\x5E\x9D\x88\x13\x9A\x64\x38\xE2\x8C\x37\xED\xAE\x20\xD8\xA3\x94\xDB\xCF\x9C\x14\xF9\x10\xBF\x80\xFF\xD8\xED\xB3\x0D\x4F\x14\xE2\x4D\xAB\xA6\xC4\x3E\xDE\xA0\x2F\xCF\xFD\x91\xB2\x93\xBA\xCC\x70\x79\x38\x8E\x5C\x15\x2F\xA8\xC9\x2F\x16\xAC\x4A\xB4\x60\x55\xA2\x99\x55\x69\x53\x59\x9E\x6F\xC8\xA9\x23\x32\x0E\x5F\x0A\x1A\x74\x81\x0C\x19\x45\x3F\x31\x76\x1B\xCC\xC4\x6E\x37\xD5\x18\x0A\xA2\xBA\x63\x5C\x38\xD2\x08\x1E\x0A\x02\x94\x31\x63\xFE\x1A\xF3\xA8\x29\x54\x12\x18\x84\xE8\x8D\x09\xC4\x7C\xD4\xD7\xE3\x09\xB5\x4B\x70\x71\xA0\x70\x44\x79\x85\x96\x27\x3B\x85\x47\x03\xA3\xAC\xC4\x35\x3F\x4B\x1E\x81\xF8\xBB\x5D\x38\xCE\xEA\xE9\x96\x20\x1E\xCC\x24\x71\xF9\x38\xDD\xFE\xC6\x38\xAD\x15\xFC\xB3\x42\xC1\x8E\x18\xC6\x24\xAC\x6C\xD2\x02\x3F\xC5\xEB\x68\x1C\x68\x65\xB4\x82\x2E\xDE\x12\xA2\x17\x2E\x65\xE1\x46\x20\x8F\xB2\xED\x99\x39\x5E\x48\x5C\xDF\xA6\x32\x06\x72\xE6\x00\xF7\x82\x0E\x6A\xCD\x03\x49\xDE\xA7\x23\x07\xE8\x42\x06\xB2\x17\xF0\x1F\x5A\xBE\xA4\xB7\x7C\xF1\x7E\x30\x4E\xC3\x25\xD1\x85\x27\x90\xE0\x92\x51\x5B\x9F\x07\xB8\x0B\x8D\x27\xDF\xD5\x16\x05\x84\xF3\xC7\x4E\x3C\xA9\xF3\x8E\x32\x27\x10\x1F\x67\x5F\x6C\x49\xA4\x58\x36\xF6\xB0\xBD\x1B\x62\x3B\x62\xE2\x1B\xDB\xF1\xB4\x69\x03\x3D\x22\x48\x94\x1B\x04\x99\xFA\x21\x22\x40\x77\x6E\x54\x19\x24\x95\x84\x6C\x31\x09\x4E\x6C\x60\x4B\x30\x36\x81\x7C\x96\x04\xC7\xC6\x7E\xF1\xCC\x0A\xB0\x62\x6F\xF8\xCE\xB8\x05\x35\xBE\x9E\xA2\xF0\x64\x58\xC6\x6E\x37\xF8\x67\x63\xD2\x5F\x88\x9E\x62\x35\x1A\xE3\xA2\x2F\xDC\x7E\xB7\xE1\x4E\x1D\xC6\x3D\xA0\x68\x68\x6E\xE3\x3B\x7B\x0A\xC5\x24\x40\x36\x10\xDC\x13\xA0\xC8\x18\x98\x4F\xC5\xEC\xB5\x90\x21\xAA\x19\x77\xA0\xA7\x9D\xE7\x6E\xE6\xB4\xBC\x29\xE0\xE6\xE7\x74\xCD\xB5\x49\x37\x36\x10\x02\x39\x16\x87\xF6\x4E\x77\x0F\x85\x07\x5E\x68\xB7\xCF\x22\x41\xCE\xED\x9D\x10\x6E\xD6\x05\xE4\xE4\xD5\x9F\x5B\xE1\x55\xFD\x90\x63\x49\x41\x49\x7F\x7C\xEB\x21\xA4\xF3\x4D\xA5\xD4\xD8\x0D\x9B\x0A\x21\x77\x4D\xB9\xA3\xA9\x74\xF1\x93\x33\x04\x06\xB6\x4D\xE6\x0B\x35\xBA\xC3\x0C\x21\x5D\xE7\xCB\xBB\x0B\x76\x1B\x77\xFD\xC4\xC4\x6E\x53\x8E\xE5\x49\xD3\x1A\xA3\xE3\xFB\x75\xD7\x08\x4C\x29\xF0\xF0\xA7\x11\xA4\x42\x0B\x1B\xE4\xD9\x76\xBC\x90\xF6\x37\xDA\x37\xA4\xA9\x8F\xAD\x44\x39\xFF\x63\xFC\xF6\xCE\x53\x9C\xBC\x35\x6C\x20\xE9\x22\x68\x19\xF7\xC5\x4D\x6F\xEE\x43\x60\x0F\xF1\x65\x7E\x11\xD8\xE7\x54\x78\x44\xEB\x69\x11\x58\xA0\x9F\x08\xD9\x0B\xDB\x94\x54\x27\xB3\xE4\xE0\x22\x49\x78\x0C\x5F\x40\xFB\x45\x8B\xE7\xAE\x04\x53\x08\x19\xBB\xE9\x4A\x31\xC3\x82\x74\xAD\xC8\x3F\x09\x6B\x05\x05\xDB\x0A\x3C\x28\x39\x44\x4F\x3C\x82\xB4\xF8\x1E\x31\x89\x4C\x11\xDF\xB3\x05\x35\xF9\x05\xE9\x35\x67\xDF\xC9\x05\xEF\x54\xFF\x9D\x0F\x21\x98\x21\xBE\x67\xBD\xB0\x6E\xCE\x45\x40\xF0\x95\x68\xE6\xEE\xF4\xE8\xBC\x6D\xEE\xA2\x6C\x61\xC9\xB0\x19\x41\x4E\xBF\x7C\xA1\xEA\x58\xFF\x1A\x27\xB1\xA0\x70\x6C\x2F\xF8\xD7\xC3\x51\xD3\x88\x69\x34\x17\xBA\xD1\x50\x04\x87\x04\xE4\x26\x43\x6C\x0A\xB2\xA9\x71\x99\x70\xAD\x63\xBA\xE1\x4A\x68\xA5\x5B\x5E\x61\xE1\xD0\xFA\xBC\x42\x4C\x0B\x6A\x03\x90\x4F\xC1\x82\x66\x8E\x9D\x69\xE8\xE2\x0D\xC9\xB5\xF9\xEC\x01\x15\x6F\xAB\xF3\xFD\x8B\xCB\xA4\x17\xE0\x53\x83\xFE\x27\xA4\xAA\x24\x1D\x99\xBD\xF8\xAD\x97\x76\x02\xAB\xF0\xA8\x2C\xBF\x9D\x48\x5D\xE8\xDC\xB6\xEE\xBB\x4E\xFF\xFE\xF3\xF5\x17\x93\x29\xBF\x95\xD3\x2A\x05\x49\xE7\x76\xEC\xC1\x31\x83\xC8\xB2\x2F\x21\x45\x04\x74\x9F\x23\xFB\xF2\xC6\x4F\x3A\xB2\xE7\x1B\x7B\x7E\xBA\x59\x21\x28\xFB\xE9\x67\xB3\xD3\xCA\x16\x4C\x3F\x5B\x30\xFD\x6C\x76\xFA\x08\x4F\x19\x85\xBE\x45\xE2\xE6\xFC\x0E\x16\xF4\xE2\xE1\xBD\x0B\x0F\x68\x0F\xCD\x0E\xE4\x0F\x82\x05\x23\xF9\xE3\x60\xC1\x50\x7E\x2F\xD8\x6B\x2C\x21\xA4\xC8\x03\x85\x08\x40\xD1\x0B\xD9\x34\xA8\x81\x6C\xB3\xCE\xEC\xCB\x39\x08\x63\x36\x03\x3D\x61\xBF\x0D\x17\x8B\xAF\x07\x3D\x06\x17\x52\x0C\xA1\x27\x9C\x5D\x96\x70\xC1\xF2\x85\x0B\x96\x2F\x9C\x83\x1E\x8A\xCE\x1B\x36\x64\xED\x95\x4B\x33\xD8\xF8\xED\xAF\xC5\xD3\x56\x5B\x85\x07\xE7\xB4\xCE\x27\x8B\xF6\x3F\xC5\x69\x06\x90\xE2\xB1\x10\x35\x55\xC1\xF6\x06\x90\x12\x92\x18\xD2\x2F\x12\x58\x20\x4A\xBB\x9F\x8B\xB6\xA7\x9D\xB8\xE0\xFD\xDC\xF6\xFB\x59\x30\x39\x8A\x20\xDC\xAC\xC2\xA7\x08\x7C\x42\x72\xAD\x22\xA0\xF8\x76\xCA\xB7\xFD\xA6\xD7\x5D\xDA\x09\x20\x27\x63\x7C\x3C\xC0\xFD\x19\x46\x0A\x5D\x47\x4C\x16\x40\x97\x61\x2C\x29\x38\xC4\x2C\xA4\x9B\x55\x8E\x94\xB9\xB1\x17\x28\x36\x30\x92\x92\x11\x92\xD5\x54\x11\xB1\x6B\x1B\x32\x14\x1F\xA3\xB7\x18\xFC\x21\x9E\x21\x57\x21\xB7\x93\xB2\xAA\x33\x07\x63\x65\x53\x9B\xA6\xA0\xE0\x06\x06\x3F\x15\xC7\xC9\x88\x46\x6E\x82\x69\x18\xFC\xB0\x3F\x30\x9B\x44\x1B\x89\x8D\xC6\xAD\xA3\x30\x15\x1C\x0F\x36\x45\xBC\x25\xB3\x80\x76\x6D\xC3\x41\xA7\x44\x64\x1B\x0A\xBA\x6A\xE8\x23\x1E\x4A\xF9\xE7\x1B\x28\x09\x2C\x83\x20\x08\xFE\xE9\x2F\xFE\xA2\xFD\x5A\x64\x21\xFC\xD3\xBF\x7F\xB1\xB3\x72\x10\x1C\xD3\x8F\x82\x57\xC6\x4E\x12\x45\x52\xD6\x5E\xBF\x80\x7E\x0C\x89\x5F\xFB\xFC\xE2\x49\x35\x76\xE7\x0E\x72\xA0\x3B\xC9\x14\x79\x8F\xEC\x85\x74\xE1\x54\xF8\x38\xDC\x89\x3D\x0C\xB9\x7D\xA8\xA9\x43\xE4\x52\xEC\x9D\x1B\x93\x9A\xA4\x9D\xB0\x65\x51\x6C\x3C\x6D\x35\xE1\xF1\x0B\x3B\xCE\x2A\xB4\xCF\x9B\x52\xB8\x61\xDC\x6E\x5C\xE9\xA4\xA9\x53\x5A\xC6\xDC\x81\x06\x79\x55\xB7\x87\x4E\xEA\x9D\xAC\x1D\x12\xE4\xB4\x65\x7D\x5C\x60\x65\x15\xF7\x35\x7E\x83\xBF\xE5\xB1\x44\x9F\xC7\x6F\xA7\x5E\xE9\xE2\xFD\x61\x6F\x71\x33\x86\xF4\x9D\x8F\xE1\x6E\x9D\xB3\x97\x1F\xBF\xB4\x13\x34\xCF\x51\x41\x5D\x5E\xA9\x97\xFC\x8D\x91\x7D\x31\x2C\xB5\x0B\xF2\xCF\x08\x30\x96\x7E\x80\x90\xD5\x7A\xE5\x34\x2C\x41\xF6\x63\x8F\xB5\x55\x3E\x7A\x2F\x52\x05\x28\x9D\xCC\xF7\x63\xFF\x14\x41\x8E\x75\xDF\xD6\x29\x5A\xBA\xA2\x2F\x66\xD1\x6E\x0A\xD9\xF1\x2D\xC8\x9A\x56\xDD\x8D\x22\x0B\xA4\x90\xF2\x00\xBB\xE2\x56\x9C\x9D\x36\x3C\x0F\xA8\x0C\x83\x73\x82\x30\xB7\xCC\xB7\x0D\xEA\x28\x2C\x23\x68\x67\x90\xD9\xEC\xC1\x7A\x64\xB3\xD3\x9B\xF6\xF0\x16\xF2\x4B\x86\x02\x17\xE3\xFB\x0D\xA2\xBF\xA6\xA9\x56\xA0\xA8\x0C\x8C\xAA\x0C\x56\xEC\x9D\x2E\xC9\x2D\x14\x4D\x4D\x2B\x4A\x6B\xB9\x42\x9B\xF6\x3F\x9C\xA2\xB8\xB2\x05\x73\x90\x45\x53\x8D\x60\x19\xFF\xAC\x40\x81\x54\x2D\x6F\x18\x1A\xA8\xFB\x95\x21\x6F\x97\xC1\x88\x99\x5D\x64\x62\x91\xF7\x2B\x20\x5C\xAF\x0C\x64\x55\x08\xBC\x67\xB9\x85\x8D\x6A\x05\xF1\x90\xE4\xBA\x15\x9E\x92\x69\xEC\x97\xB4\xA3\x0E\x79\xB6\xCC\x24\x52\xC8\x93\xB0\xC1\xD5\xB1\xD9\x4B\xEA\x8C\xA7\xD8\xB4\x3D\xA2\x40\x91\x36\x76\x85\x66\x89\x73\xC8\xC1\x20\xE1\x44\x8A\xE0\x58\x98\x98\x4E\x18\x32\x63\xF1\xD6\x5F\xDE\x4A\x26\xEA\xC3\x3F\xC4\x2D\xBF\x1F\x19\x30\x4C\xC8\x0D\xB3\xE8\x34\xC4\xC3\x10\x21\xF8\x9D\x65\xE0\x37\x9B\x75\x8A\x44\x90\x0D\xD9\x5A\x16\xDD\x38\x6E\x9F\xB9\x27\xCF\xE8\x86\x2D\xC6\x52\x57\xC4\xF2\x86\x74\x8E\x10\xCB\x4B\x51\x6A\x99\xE5\xC5\x9E\x20\x6C\x17\x83\x2E\xB5\xB1\x76\x46\x07\x0C\x62\xB3\x79\x01\x1D\x00\xC6\xAB\x02\xE8\x80\x31\x74\xC0\xA4\x38\x62\x62\x72\x09\x59\xDD\xCF\x45\x27\x6B\x7B\xFE\x0B\xA6\xA1\x48\xC9\xDD\x06\xB9\x03\x26\x7B\x0A\x68\x19\x87\x1D\xED\xF1\xBB\x8E\x8F\x72\x38\xCF\x56\xA0\xCC\xF6\x1A\x46\x9E\x84\x11\xDE\xB3\xBD\x0B\xB8\x03\xE4\x09\x84\x0F\x8E\x93\x31\x38\x2F\x28\x6C\x5A\xB6\x37\x9C\x3B\x47\x24\x0F\xCA\xB3\xBD\xC8\xB3\x04\x2D\xCF\xE2\x72\x94\x65\x4E\xC0\x20\x9E\xC5\x89\x17\x4F\x90\x6D\x09\x3E\x2F\x27\x04\xE9\x28\x3A\xB3\x24\x73\x31\x8B\xC5\xB6\x38\x2F\x2E\x88\xF3\xF8\x37\x5E\xCC\x07\x0F\x39\x5C\x12\x4A\xF5\xFF\x41\x07\x4B\x8F\xFB\x59\x70\xB0\xB0\x36\xB5\x8D\x12\xAF\xDF\x38\x73\xA6\xD4\xE9\xBB\xEB\xEC\x18\xB7\x71\x3D\x7E\x25\xDD\xBD\xEF\xEC\xEC\x7C\xFA\xDE\xF6\x9A\xFE\xD3\xEF\xB9\x7C\xDF\x3F\xF9\xAA\x97\xBE\xF8\xEA\xAB\x7F\xE5\xDE\x2B\x90\x1D\xFB\xD2\x47\xAF\xDA\xEB\xE3\x29\xDD\xFC\xDF\xF7\x99\x6F\x74\xF7\xFE\xF7\xFD\xC8\xFB\x5E\xF4\xCD\x6B\xFF\xF2\x67\xEF\xBD\x7C\xB9\x36\x33\xB4\xDE\xBC\xBD\xCA\x3B\x5A\x9F\x1B\xC8\xDF\x59\x9B\xFB\x6E\xFB\x9A\x1F\x7D\xD7\x0F\x6D\xCA\xA3\x57\x1C\x13\x66\xEE\x9B\xBE\xF9\x1B\xD3\x17\xBE\x2E\x3B\x7A\xA5\xA5\xB0\x0B\xDB\x1A\x75\x6D\x8D\x0C\x9F\x84\x78\x32\x42\x0A\x05\x8C\xDE\x79\xDF\x7F\xFE\xBB\x7F\xF5\xD9\x63\x5F\xF2\x7B\xEF\xB9\x72\xB9\x36\x60\x2E\xB7\x06\x02\x60\x1E\x03\x03\xE6\x0A\x45\x2F\xF7\xC7\xEC\x5D\x47\xDF\xEC\x7E\x3D\xEB\xE8\x65\xFF\xF3\xE0\xD1\xCB\x6F\x6E\x1F\x3E\x75\xEF\x65\x18\xD9\xEB\x71\xB3\x7B\xEC\x4B\xBF\xE9\x3D\x57\x6A\xC3\xC3\x68\x6B\xBD\xB9\x32\xEE\xE8\x33\x74\x32\x8F\x27\x55\x68\xB5\x0D\x21\xB7\xFF\x5F\xD0\xD8\xF7\x8A\x8D\x49\x55\x12\x95\xDB\xF1\xCA\x80\xF4\xBE\x43\x5F\x75\xC7\x27\x3F\x76\xE4\xF6\x7B\xFF\x19\x1D\xA4\xD4\xD6\xCE\xA5\x2F\xE1\x19\x86\x90\xDF\x5F\x8D\xB0\xCD\xB3\xD5\x98\xD4\xFA\x21\x8C\x1B\x1B\xC0\x68\x42\x74\x68\x27\xE1\xB0\x93\x25\xCB\x3D\x61\x97\x83\x23\x84\xB1\x0D\xD6\x37\x91\x40\x07\xD5\xA8\x05\x85\x14\xCC\x9B\x21\x85\x7C\x52\x9B\x37\x82\x19\xEC\x3E\x84\x96\x84\x1E\x28\x37\xEB\x31\x8C\x21\x3A\x35\xA9\xC7\x50\x4E\x60\x4C\x88\x5F\x2F\x59\xF1\xD0\x04\x62\x7B\xFD\xA5\x53\x07\x49\xCB\xF2\x28\x94\x45\x00\xCB\xCE\x40\xCC\xC2\xD4\x7E\x3C\x38\x0B\x02\x96\xFB\x20\x9F\x43\x6E\xEF\xFC\x86\x3A\x6E\x20\x3E\x5B\xC5\x50\xDA\x57\x34\xD5\xD8\xDE\x89\x27\x18\xD1\x81\x11\x85\x22\x87\x18\x8F\xC5\xD5\xDE\xB1\xB8\x82\x2B\xB9\x02\x63\xD6\xC9\x12\x9E\x8E\xB0\x14\x1E\x1F\xAB\x14\x2E\xFD\x4E\x1F\x4C\x3A\xE4\x83\x70\x04\xA5\xBD\xD0\x70\x4C\xEF\x11\x9E\x01\x25\x8C\x36\xED\x2B\x50\xBC\xBE\x13\x46\x9B\x75\x08\x31\xA5\x2F\xE9\x4E\x80\x31\xAC\xE0\xFF\x37\x26\x58\x1E\x3B\xC0\xD6\x46\xFE\x1C\x19\x35\xF6\x90\x7D\x0E\xE4\x36\x38\x39\xC1\x11\x8D\xB8\x08\xF9\xBB\x3A\x63\xB9\x11\x53\xC0\x91\xE3\xB8\x60\x44\x42\x9A\xA3\x5E\x03\xCE\x78\x99\xCC\x30\xB8\xFD\x96\x06\x86\x4D\x3D\x6A\x69\xE0\xA0\xF8\xA8\xA5\x81\xB3\xBC\x34\xD9\x0B\xC4\xC4\x52\xBB\x68\xF1\x4B\x24\xE5\x0E\x69\x60\xE8\x14\xF7\x23\xCE\x63\x45\x34\xB0\x27\x60\xC6\xC3\x16\xC5\x4C\xFC\xF9\x51\x8F\x06\x86\x0B\xAA\xF0\x8B\x05\x34\x30\x5E\x40\x03\xE3\xD9\x93\x85\xC3\xB8\xC7\x8B\x68\x60\x6C\xDE\xAD\x85\xDE\x76\xBA\xA3\xB8\xD3\x1D\x45\xAC\x3B\x0A\x9D\xB0\x96\xBA\xB3\xD4\x11\x77\xD2\xBD\x98\x4E\x77\x64\xF6\xD2\x1D\xF1\x41\xBD\x40\xD5\x91\xFA\xB3\xD4\x6C\x56\xE6\x29\xD2\x1D\x19\x3C\x4B\x4D\x77\x96\xC6\x0B\x55\x48\xA6\xA7\x42\x32\x37\xA9\x42\x32\xC8\x15\x2E\x54\x21\x65\xFB\xA9\x90\xF0\xB0\xE5\x41\xE1\x82\xB7\x2A\xA4\x0E\x92\x8C\x53\x21\x65\x94\x66\x81\x55\x48\xE6\xC6\x2A\xA4\x01\xE3\xDF\x41\xD2\x2D\x5F\x57\xD3\xAA\x90\x3C\x24\x99\x32\xB0\x97\xDF\x78\x69\x27\x28\xBF\xF3\x22\xD9\xBA\xE3\x8B\x1F\x9F\x7D\xF1\xCB\x83\x17\xE3\x80\xC3\xA5\x96\xAF\x62\xFD\xB7\x09\xA0\xFD\x9D\xE1\x6F\x32\x5F\xE6\x9F\x1C\x4E\x7F\x35\x00\xC1\x6F\x2D\xFE\x22\x62\x62\x72\xD2\x4C\x95\xBF\x45\x25\x1E\xD1\xC2\x6C\x03\x7B\xC0\xD3\x3D\xAE\x66\x87\xF7\xB0\x52\xA0\xCB\x47\xC9\x41\x8B\x1C\x45\x22\xD9\xE5\x89\xFF\x9A\x66\xA4\x44\xC0\x16\xFE\x11\x84\x64\xAB\x83\x3F\x59\x90\x45\x0C\xA5\x96\x3A\x5F\x24\x69\x5F\x4A\xDE\x04\x74\xED\x79\x98\x35\xC3\xF1\x21\x5C\xAC\x2D\xFB\x69\x8E\x6C\x7F\x9C\xC3\x60\xC5\x87\x02\x51\x69\x5B\x92\xC2\x14\xE9\x07\xF9\xA7\x1C\xAF\x42\x9B\x91\x95\x99\x6B\x40\x18\xB2\x52\x3E\x5C\x69\x4A\xE7\x09\xE1\xA4\x4A\x0C\x5B\x1F\xF2\x31\x0B\xD1\xA4\x4E\x21\x79\x90\x93\xD1\x78\x73\x41\x9F\x81\x86\x72\xFC\xA4\x90\x41\x72\x7A\xB3\x2A\xB8\x59\x32\x7A\xCE\x5B\x21\x28\x46\xF1\x82\x6D\xCD\x40\xD9\x07\x1A\x32\xA9\x02\x6D\x1F\x68\x9C\x51\x70\xF9\x4D\x6C\x64\x2A\xED\x86\xFD\xFE\x6D\x50\xE5\xAB\xF1\x79\x02\x9A\x2C\xD0\x13\x50\x64\xE9\xA3\x9F\x5F\x08\x22\xCC\x95\xB2\xDB\x8E\xB3\x56\x6C\x99\x6A\xF8\x30\xA4\x8B\xA4\xE1\xF3\x3A\x05\xE5\x34\x50\x80\x3E\xC9\xE9\xB3\x92\xD3\xD8\x2E\x59\x5B\x0D\x16\x99\xFD\x25\xDC\xC4\x38\x28\x48\x15\x63\x5B\xC6\x59\x8A\x03\x5F\x16\x42\x00\xF9\x84\x6C\x4A\x53\x30\x36\x00\xB3\x09\xD1\x84\x0F\x0C\x73\x77\xEB\xA8\x27\x68\x1D\x38\xD6\x4A\x70\x08\x91\x91\xCC\xF4\xC7\xCE\xB0\x3E\xE8\x59\x66\xDB\xC0\x6C\x90\x1D\x29\xC2\x12\x6F\xA0\x73\xDD\x0A\x68\x58\xEC\xA8\x4A\x81\xCB\x14\xE8\xCD\xB1\xE1\xA0\x2E\x62\x61\x4B\xEF\x12\x42\x6F\xFB\x41\x48\xAB\xA6\x3E\x53\x25\x15\x73\xF7\x89\x6C\x9B\xEE\xEC\xD2\x81\x72\x8D\x09\xFC\x13\x1D\x0A\x02\x7F\x0F\x45\xF7\x35\xED\xB3\x6C\xE8\x82\xAC\x7D\x56\x0D\x5D\x8E\xB9\xE7\x81\x5D\x3A\xEF\x09\xB6\x57\x49\x6A\x9D\xBA\x61\xD7\xB8\x5E\x36\x33\x8D\x7F\x34\x05\x12\xAB\xD9\xEA\x23\x30\x8F\x28\x11\x6F\x83\x74\x89\x50\xD4\x60\x1A\x9A\x85\x5A\x9E\x6D\xD4\x9B\x48\xCC\xEA\x20\xCA\x74\x57\x93\x57\x36\x5B\xAB\x79\x83\x06\x12\xF7\xDB\x67\xD9\x90\x46\xAA\x7D\x56\xF8\xDC\x4E\x24\x76\xAC\xBA\x37\xB0\x67\xCF\x7C\x14\xE9\x29\xF5\x47\x43\x1D\x85\x14\xAF\xF1\x9E\x8E\xF4\x39\x79\xC2\xC7\xCC\x41\x99\x70\x6E\xD8\xFD\x11\x0B\x37\xE2\x80\x47\xAC\x3E\x97\x11\x8B\xFE\x88\x79\xBC\x8A\xC7\x1B\x3C\x81\xF1\x2A\x26\x72\x10\x98\xFF\xDB\x85\x0D\x73\x33\xD8\x6E\x9C\x19\xBF\xB4\xF1\x94\x5C\xA0\x1C\xB3\x19\xB8\xA6\x45\x2F\x27\x0B\x15\x97\x83\x76\x6D\xCC\xFE\x9A\x6D\xC1\xB8\xF1\xBF\xA2\xF6\x57\xD8\xFE\xD2\xED\x2F\xD5\xFE\x92\x4D\xD7\x89\xFF\x95\x70\x77\x0F\xB7\xBE\x0D\xE6\x7F\x14\x6A\xDB\x05\xB0\x0A\x80\x00\x8C\x22\x2B\xA9\x4A\x10\xE0\x91\xFB\xB3\xDD\x6E\xCC\xAF\x2B\x11\xB1\x59\xAC\x5B\x2B\xFD\x42\xA4\x80\x17\x5F\xDB\x9E\x0E\xA6\x75\x96\xD2\x5D\x4E\x5A\x24\xCE\x41\x15\x83\xE6\xF8\xBE\x6C\x4E\x9E\x42\xD8\x90\x35\x3C\x9E\xC4\x3B\x7F\xF4\xFA\x37\x9F\x27\x5D\x0F\xA4\x1B\x90\xD9\xEB\xAF\xFB\xE4\x47\xD4\xA9\xAD\x3A\x9C\x90\xAB\x04\xDF\x1C\x4B\xF6\x65\x25\x97\xD6\x9E\xB9\x04\x22\x10\xA8\x8D\x29\xFB\xF2\x08\x72\xE9\xA8\xC3\xA6\xCA\x78\x13\xA5\x15\xB4\xDA\xD2\x06\x90\xB9\x54\x3E\x11\x44\x76\xE7\xF5\x9F\xFC\x88\x3A\xE1\x7A\x70\xF8\xE4\xDE\x92\x7B\x22\xAB\x8B\x5D\x9F\xD8\x81\x87\x6A\xBE\x79\xD5\xC8\x57\x29\xBE\x27\x42\xC2\x82\x87\x7C\x74\xBC\x1B\x80\xE4\x01\xC8\x86\xCD\x76\xA9\xEB\x23\x32\x98\x6D\xD7\xF7\xDB\xBA\xEE\xF0\x0D\x12\xFB\xA5\x44\xD4\x32\xE5\x3E\x0A\x20\x42\x31\x5D\xE0\x1F\xD9\x6B\x8A\x7D\x79\xFD\x73\xD2\xF8\xD8\x72\xFC\x9C\x37\x3E\x03\x80\x0F\xA6\x49\x84\xAF\x44\xC2\xD7\xB3\x5B\x08\xCD\xC7\xA4\x48\x7B\x80\x8B\x87\x90\x03\x5C\xC1\x1E\x6D\xBA\xCB\x55\x78\x61\xCA\x89\xD5\x41\x54\x31\x70\x04\xFF\x90\x78\xF3\xA4\xDD\xD6\xF8\xA6\xB6\x95\x58\x91\x8C\x56\xC6\x55\xE3\xB9\xCC\xD5\xAC\xD3\xB6\x6E\x6D\x7C\x6D\x97\x03\x8A\x12\x5E\x21\xB7\xAB\xAD\xEC\x39\xF7\xB8\xA4\x36\xB4\x15\x64\x52\xC4\x31\x4C\xB1\x17\x01\x09\xF7\x92\xB8\x5E\x70\x0C\x64\x15\xE6\xBB\x98\x90\xC5\x60\x9D\xD2\x41\xE5\x6E\x05\x48\x5D\x26\xA6\x4E\x4F\xC0\x79\x18\x11\x9A\x13\xCB\xC1\x0A\xC9\x10\x05\x25\x4C\x06\xAB\xB8\x9D\xA1\x8B\x3C\x6B\xCC\x9B\x05\x9F\x53\xA2\x08\xEC\x1B\xBF\xAF\xC3\x18\x97\xDE\xA1\x56\x9C\x2D\xDE\xCD\x5C\xA2\xF8\xB9\x01\xB2\x5D\x33\x3D\x61\x37\x0B\xC4\x3C\x2B\x29\x62\x85\xE4\xD0\x26\xDA\xC1\x98\x75\xEE\x5E\xBA\xA9\x29\xFA\x2A\x8F\x84\xEA\xF9\x47\x84\x6C\x07\x7A\x0A\xC4\x46\xD5\xE6\x68\x0C\x98\x02\x9A\xD3\x4C\xC0\xF0\xC4\xF6\x99\x5A\x89\x0B\x09\xDA\x1C\xFA\x0D\xA5\xCE\x77\x8D\x2B\x87\x3D\x6D\xE3\x34\x1A\x72\xF5\xDC\x20\xDF\x70\x3E\xAD\x98\x9C\x8D\x8D\x7D\x7D\x37\x6F\x2B\xCC\xC7\x35\x41\x5D\xC7\x8B\x68\xA4\x26\xDF\xDC\x95\xE9\x13\x11\x4A\xE9\xE5\xF3\x90\x39\xA8\x8C\x5C\x6E\x3A\xCE\xEA\x85\x90\x88\x84\x86\x6E\xAE\x40\xD0\x55\x07\xE2\x4E\x4A\xE9\xB9\xEA\x9C\xD1\x3F\x6E\xEA\x78\xB3\x2E\x78\xC8\x0D\x14\x90\xE3\x5E\xF3\x7E\xA5\x88\x51\x86\xA1\x10\xCB\xF3\xAF\x7D\xAA\x24\x4D\x95\x12\xEE\xD1\x65\x56\x42\xD2\x12\x9D\xF7\x33\xD9\xCC\x38\xEA\x6F\x9D\x22\x04\xA6\xD4\x43\x06\x92\x7F\x61\x0F\xE9\x66\x6D\x7C\x0F\x86\xC1\x3E\x66\x4E\x29\xF6\xCE\x79\xDE\xE6\x43\x73\x3B\x44\x60\x52\xE4\xC6\xB6\x69\x9E\x06\x62\x87\x4C\x0D\x43\x0D\xA4\x7D\xE2\x96\x16\xC2\x06\x76\x1B\xD2\x09\x59\x71\xB5\x97\xA9\x95\xB9\x49\xF2\x66\x7A\xE4\x2D\x6B\x28\x9D\x6B\xE6\x6E\x31\xC0\x87\x65\x4F\x29\xFF\xA2\x6F\xDE\xF5\xDC\x93\x6E\xFA\xE4\x2D\x63\xF2\x66\x88\xBC\x65\x9C\xF2\x2A\xEB\x22\xBC\x23\x57\x44\xED\xB5\xCF\x49\x43\xCB\xDD\x3E\xE7\x0D\x85\x04\x5F\x48\xDE\x3A\xD1\xD1\xFC\xB0\x14\xC9\xAD\x22\x6E\xB1\x4F\x06\xB7\x89\x44\x8A\x77\x2B\xA3\xE4\xCA\x7D\xA2\x96\xD2\x7A\x60\x71\x07\x41\x37\xAA\xF1\xE4\x08\x99\xA3\x62\xC4\x04\x6D\xD6\xDA\xB7\xAC\x21\x86\xC4\xB5\x4C\x0C\x3A\x47\x0A\xED\x13\x2C\x9E\x0A\x32\xE3\x0E\x4A\x86\xB4\xEA\xB8\x23\x56\x5A\x5C\x90\xE4\x3B\xC3\x6B\x55\x04\x03\xE4\x0D\x2A\x8A\xA5\xA5\x76\x69\xF1\x14\x90\xC3\x53\x15\x29\xCE\x69\x2C\x9E\x8B\xFB\x73\xA1\x8E\xAF\x36\x94\xB6\x5E\x6D\x90\x1B\x53\x7C\xDE\xB9\x27\x41\xB2\x7B\xE9\x18\x8D\x78\xA7\x8E\xAF\xD6\x8A\x89\xDE\xE9\x86\xB7\x91\xA2\xBB\x38\x6D\x7A\x7B\x32\x42\x7C\xB5\xEA\x48\x2F\x65\xC9\x31\x7F\x21\x54\xD2\x0E\xF1\x46\x54\x84\xDC\xE1\x8E\x17\xD2\xF9\x44\xD3\x38\x5D\xFA\xB0\x2A\x06\x81\xCB\x5E\xA5\xA0\x58\x46\x23\x8A\x91\x32\x05\x88\x71\x22\xC9\x73\x65\x70\xA1\x2E\xAE\x36\xF5\x08\x0C\x8A\xF7\xC5\x31\x78\x14\x46\x90\x6F\xEC\x9E\x07\x03\xA3\x8D\xDD\xF3\x38\x3B\xD3\xCD\xA9\xB8\x8A\xB8\xDC\xCD\xC9\xAD\x6D\x8B\x1C\x4E\x89\x02\x1C\xE1\x8D\x26\x54\x7C\x65\xC7\xE8\xF0\x2D\x2C\x07\xBD\x37\x24\xF4\x30\x00\x78\xFF\xEA\xC7\x84\x16\x34\x6F\x71\x61\xD1\xD6\x8C\x39\xC6\x99\x3C\x87\x28\x40\x8B\x49\x82\x44\x80\xBB\xD5\x09\x2E\x82\x43\x4E\xEA\x5D\x1E\x34\x6F\x96\x86\x68\xB0\x55\x9F\xA2\xD4\x94\xAF\x04\xBD\x7B\x09\xC2\x9D\xAB\x35\x7B\x3E\x42\x04\xEA\xF4\x26\x22\x89\x3D\xCF\x28\x77\x7E\x20\x4A\x81\x36\x3F\x5E\x84\xD1\xB6\xB8\xA0\xB6\xC5\x85\xC2\x2B\xE5\x5F\xAB\x37\xEB\xA8\x67\x9D\xC2\x11\xCE\x5D\xD0\x03\x63\xFF\xD3\xB7\x75\x67\x62\x1B\x6B\x03\x49\x1B\xC3\x36\x72\xE6\xC4\xF2\xB9\xB0\x1D\x56\x34\x0F\x56\x29\x44\xF6\xB5\x92\x54\x2D\x74\x2E\x5C\x0B\x98\xBA\x88\xC6\xEA\xF2\xDD\xE4\x37\xCD\x76\xDD\x11\x52\x05\x9F\xEC\x95\x4B\x85\xFD\x52\x86\xBB\x42\x69\xB7\x55\xF7\xCF\x81\x3C\x05\x29\x4A\x77\xD9\x7F\x1D\x32\x72\x89\x1D\xAB\xA3\x30\x82\x02\x24\x42\x4A\x7E\xA1\x2E\xAF\x12\x92\x42\xC1\xEB\x58\x0E\xD7\x31\xEE\x60\xA4\xBC\x8A\x27\x4A\x07\x23\xEE\x58\x1E\xE1\x9F\x11\x29\x71\xEB\x31\x85\xE9\xBB\x4A\x81\x53\x21\xDC\xC4\xE1\xF5\x42\x1D\x48\x17\x96\x74\x66\x90\xAA\x37\x48\x09\xA6\x2A\x20\xA4\x41\x16\xC8\x50\xF4\x06\x39\x82\x11\xC8\x85\x83\x1C\x0D\x06\x29\xFB\x83\x74\xD6\x93\x05\xFE\x29\x86\x83\xA4\x74\x3A\x88\x52\xF5\x12\x7B\x38\xB3\x57\xDC\x89\x82\xB5\x4E\xF2\x5C\xBD\xDC\x54\x2B\x54\xE6\x42\x53\xAF\x12\xEE\x1D\xC0\x87\x69\x75\x1B\x11\xAA\xEA\x19\x90\xB9\x2C\xFF\x07\xE9\x4C\x89\xEB\x67\x62\xB5\xDB\x9B\x7A\x02\x4B\x4D\x7D\x07\x33\x49\x2B\xE4\xB5\xF6\xA0\x3F\x8E\x4F\xC3\x04\x96\xE9\x34\x1D\x91\x02\xA6\x38\xBD\xD9\x42\x34\x4E\x53\x21\x7E\x2E\x98\xA4\xDA\xBD\x04\x72\x77\xE7\x6A\x5D\x70\xFA\x51\xAA\x59\x29\xC8\x60\x15\x9E\x49\x28\x59\x8D\x38\x65\x30\xC4\x8E\x07\x23\x89\xE7\x00\x2E\x61\xBC\x5B\xE3\x2A\xBA\x55\xBA\x5A\x17\xEC\x96\x5D\xAC\x17\x12\xFC\xA6\x42\x79\xB5\x81\xD1\x9A\x0C\x5E\x40\x86\x86\x28\xDD\xC5\x2E\x1B\x24\xB2\x7A\x1C\x26\x8D\x1A\xD8\xA0\x40\xA8\xF1\x6E\x95\x23\x7D\x1A\xD3\x9E\x1D\xE4\xA4\xCE\x24\x99\x73\xBC\x6C\x90\x90\x43\x41\xA4\x88\xB7\x6F\xB8\x79\x67\xFA\xFB\xD6\xED\x1A\x8C\x9A\x7A\xB4\x59\x7F\x91\x3F\x1F\xBE\x08\x34\x8C\x36\xEA\x11\x91\xFC\x11\x89\x27\x7E\x37\xDD\xDE\x8E\x91\xF8\x8C\x49\x79\x8C\x04\xEA\x76\x1C\x6D\xCC\xDB\x29\x59\xE8\xC3\xB9\xB4\xF0\xC7\xB7\x32\x9C\x1E\x26\xB6\x41\x35\xCA\x89\xCA\x2A\xFF\xA8\x8E\xC2\x41\x9C\x72\x01\x05\x45\x4F\x80\x11\xB2\x37\xAA\xA9\x47\xC4\x83\x0F\xF8\xDD\x71\x9F\xDF\x1D\xD3\x10\x0B\x4E\xE6\x4E\x5E\xD3\x05\x8C\x1B\xC2\x8F\xF9\x9A\x45\xBF\x26\x4F\xCE\x45\x45\xB8\x8D\xD4\x77\x64\x23\x9E\x4B\x33\x90\x55\x59\xCD\xF5\x0C\x9C\xD4\x84\x00\xAF\x56\x6D\x5F\xD2\x8F\x93\xFA\x52\xBE\x2F\x0E\xE6\x0E\x6A\xD8\x97\xA1\x5B\x0D\x6C\xE7\x0E\x1B\xC0\x1D\x4E\x1A\x50\x7E\x7F\x4F\x70\x99\x83\x48\x25\x0F\xC2\x33\x09\x12\xE0\x99\x85\x77\xF3\x14\x8E\x16\x55\x7C\x22\x41\xD6\x60\x41\x66\xBB\x99\xA4\x5A\x71\x92\xFD\xB9\x29\x0E\x0E\x7D\x25\x2F\x77\x32\x39\x0F\xEE\xF7\xC1\xD7\xE7\x69\x41\xCC\x29\x80\x42\x77\x16\xC8\x86\x42\xC0\xD1\xA6\x3B\x0A\x11\xBB\x50\x1C\x92\xA1\x6C\x04\xC5\x00\x7B\x8A\x3D\xB0\xA7\xD8\xBD\x04\xF9\xCE\xD5\x7A\x44\x7B\x04\x29\x8C\x48\x7F\xE9\xC6\x26\xF1\x4F\xCF\x85\x9F\xAF\x95\x89\x52\x53\x9A\x1F\x1B\xAC\xC9\x4F\x5C\x79\xF5\x4E\xB0\x74\xC9\x19\x50\x33\x81\x3E\x51\x50\xFA\xE1\xB9\xAF\x91\x7D\xAD\x6E\xEF\x3B\xC6\xF8\x7C\xE9\x3B\x28\xFF\x34\x9E\x98\x1F\xD1\x94\xB4\x6A\x4D\x5E\x7A\xEC\xD5\x3B\x81\x7D\xF4\xDF\x5E\xDA\x09\x20\x2A\xFF\x15\x1D\x95\xF6\x3B\x2F\x5E\xDA\x09\xEC\xB3\xAD\x68\x4B\x94\xAF\xE3\x2F\x59\xF7\xE6\xF5\xF4\xC6\x3B\xB6\x7F\x3C\x23\x76\xB1\xD5\x8F\x72\x10\x7C\xFB\xFA\x6F\xEF\x1F\x4F\x74\xDA\xA4\x0F\xD6\xA1\x7D\xD9\x69\x52\xFE\x21\x66\x70\x96\x4C\x87\xE8\x0A\x59\x66\xC6\x18\xDC\x55\x41\xBB\xAA\x49\xBF\x84\xCF\x01\x3D\x47\x94\x7C\xB8\x0D\x57\xC3\x45\x22\xFA\xC2\x2D\x0D\x82\x50\x10\xF3\x2C\xDA\xF6\xCE\xCC\xB4\x77\xC6\xB5\x87\x7C\xB1\xEE\xF8\xE2\x08\xF9\x66\xDD\xF1\xCD\xC8\x64\x70\x44\x70\xF7\xDC\xEB\xFF\x0C\xF7\x7F\x86\xFB\x3F\x4F\xCA\x45\xD2\xE6\x2A\x7B\x81\x63\x54\xE0\xEC\xEC\x3B\x70\x7B\xDA\x24\x26\xE4\x8B\xEB\xA7\x48\x39\x23\x3B\x3F\x53\x9B\x42\x84\x8C\x6F\x86\x67\xF9\xFF\xF5\xE6\x3E\xEE\xA1\x90\xC0\xDE\xC1\x51\xCF\x2B\x75\xBB\x19\xE9\x40\x48\x15\xA2\x50\x97\x3D\x58\x1B\xFB\x10\xA9\x9C\x23\x30\x1C\xE0\x4A\xDB\xC7\x03\x7C\xFD\xA6\xED\xDE\x7B\x0A\x35\xFC\x89\x18\xDF\x7F\xF7\xC3\xBD\xF7\x78\x72\xDA\xD7\xFE\x12\x55\xF8\xD4\x0F\xF5\x6B\xF4\x62\x1B\x67\xF6\xE5\xFD\x4E\x2F\xBF\x26\xC2\xE2\x1F\xFD\xD3\x6F\x9C\xED\xE0\xDA\xA5\xAF\xC0\x2F\xD7\x3E\x73\x6C\xB6\x8B\x9D\xDD\x8F\x69\xFC\xB4\xF3\xA1\x7F\xFD\xF2\x41\x27\xF8\xED\xA7\x7E\xFD\x7F\xA2\x6F\x3F\xF9\xB6\x07\xBA\x6F\xA6\x2F\xDB\xFA\xC0\x20\x9A\x19\x81\x0B\x8D\x0F\xE6\x94\xD8\xED\x75\x8A\xFD\xA6\x18\xC5\x79\x21\x19\x34\x0C\xDF\xF9\x9C\x63\x4D\x31\x1F\x9C\x82\x26\xC3\xD9\x14\x29\x91\x7B\x7F\xCD\x87\x6E\xC7\xA3\x10\x67\xAC\x43\x04\x2B\x9A\xB2\xE6\x29\xC7\x0D\x85\xAA\x80\x98\xA7\xAB\x79\xBA\xF4\x56\x91\xC5\x00\x4D\x55\xBB\xA9\xD2\x7B\x3A\xDF\x78\x9A\xDA\x4D\x33\x76\xB7\x13\x8E\x79\xA3\x0F\x7C\x20\xC5\x0D\x47\xCD\x44\x2E\x32\xF5\xAC\xCF\x79\xD6\x39\x12\x37\xA9\xAA\xA8\x95\x18\x9D\x99\x6A\xC6\x86\xC7\x37\x3F\xED\xD9\xA9\x6A\x25\x45\x10\xEE\x35\x1C\x9E\xEE\x82\x09\xDC\x70\xC2\x73\xCB\x23\x16\x2D\xE5\x13\x9A\x2A\xCF\x30\x6F\x67\x28\xDA\x19\xA2\x7C\x8F\x1C\x7B\x25\x91\x45\xDD\x6B\x7A\x92\x61\x5D\x33\xAC\x4B\x12\xED\xC9\x1F\x05\x24\x61\x87\x26\xEC\x68\xDF\x53\xE2\x83\xC7\xA9\xC2\x9B\xFA\xE5\x49\x12\xCE\xF0\xF5\x43\xDD\x5B\x03\x12\x45\x79\x9C\x51\xDC\x9F\x91\x9B\x4A\x8C\x7F\x62\x9A\x51\x8E\x33\x62\x47\xA5\x80\x85\x2C\x50\x90\x9E\xA6\x43\x30\x80\xB0\xFC\x45\xA2\xA1\xBF\x6A\x44\xB1\xED\xC4\x97\x3F\x7C\xCB\x90\xB6\xBA\xCB\xAE\x14\x87\xF0\xB2\xD3\xC8\x2D\x84\x9D\xEE\x81\x48\x4C\xC4\x34\xC8\x92\x33\x84\x7B\x47\xB7\xC3\xF3\x94\x86\x72\xDA\xE8\x53\x3E\x74\x83\x1C\x64\x29\x20\xB4\xD7\x78\xA4\x64\x0F\xD6\x29\xCE\x37\x6A\x58\x0C\xCC\x23\x7C\x8B\xAB\x93\xD2\xEA\x44\x4D\x4D\x85\x32\x57\xC8\xE4\x21\x16\xC0\x65\x4D\x69\x59\xA9\x00\x96\xCF\x5C\x79\x8E\x54\x13\xF1\x9E\xE4\xBC\x27\x51\x43\xEE\xF0\x11\xB2\xC4\x58\x35\xA3\xAA\xA4\x34\x8A\xAA\x94\x79\x27\x77\x53\x88\x0B\xE0\x75\x25\x20\xB0\xF1\xF3\x9C\x83\xDD\x41\x6A\xE6\x20\x55\x11\xF8\x40\x64\x1F\xA6\xEC\x45\x48\x4D\xD5\x51\x27\x38\xF7\x38\x87\xC4\x9F\xCE\xC9\x71\xE7\xFB\x11\xDC\x4F\x16\x78\xA0\x37\x6B\x14\x8E\xE9\xC4\x22\xA6\x20\xE5\x5C\xFF\x67\x41\xBA\x3B\x2F\x65\xCF\x4F\x29\x06\x2E\xAD\x3F\x99\x48\x2A\x67\xEC\x4A\x1A\xC1\x08\x0A\xFC\x93\xF4\x4F\x98\xA4\x77\xC2\xF8\xEC\x7D\xED\x73\xDE\x50\x82\x37\x9F\xB1\x90\x03\x34\x49\x12\x90\xD9\x6E\x96\x12\xB0\x23\x82\x53\x0C\x39\x62\x5B\x52\xD0\xCC\x97\x47\xC4\x15\x47\xBD\xCE\x7A\xAE\x98\x0E\xF6\x72\x65\x16\x65\x8C\x0D\x91\xD4\x87\x41\xE8\x37\x9D\xB0\x35\x65\x6C\xA5\xFD\xA3\xBD\xCA\xDA\xBD\x32\x79\x42\x19\xC8\x89\x04\xA4\x4C\x02\xAA\x9C\xB7\x95\x77\x21\xE5\x5D\xA8\xF2\x9C\xA2\xD1\x3A\xF2\x92\x3A\xF2\x52\xE5\x3E\xA5\x52\x9D\xD8\xEC\x85\x14\x5A\x94\xA1\x8B\x9A\x20\x12\x92\x12\x09\xE1\xFB\x54\x0A\x04\x95\x73\xCF\xC2\xF4\xC1\x18\x2B\x11\x70\xC8\x2E\xAE\xE6\xCB\x3B\xF8\x35\x3C\x95\x8C\xA7\x62\xB0\x3E\x0A\x64\xDC\x43\xC6\x44\x8A\xDE\x4A\x7C\xCB\xF3\xC9\x1C\x49\x33\xDC\x1B\xBE\x1F\x00\x96\xF1\x86\x9E\x04\x01\xA9\x97\xAC\xAB\xC8\x83\x64\xC7\xA7\xDE\x2C\xB4\xA1\x44\xC0\x0A\x5C\x12\x82\x58\x9D\xA9\xDA\xBC\xED\xD9\x06\x6B\x7E\x4E\x91\xFD\xD2\xCB\x9B\xAA\xF4\x70\x58\x8F\xAA\x1C\xD8\x1B\x3B\x45\x16\x85\x62\x0F\xB5\x4B\x21\xE6\x99\x8B\x31\x99\x86\xC8\x6E\xE7\x47\x21\x1D\x78\x14\x0B\xCA\xE3\xBA\x22\x4E\x98\x0F\x00\xD5\xC3\xF5\xF6\xBD\xC2\xF7\x1D\x8A\xB7\xEF\x25\xBE\x27\x58\x49\x19\x56\xDA\x2F\x83\x28\x5C\xA5\x3B\x64\xA9\xD3\x3E\xAC\x0D\x3B\xEE\x81\xC1\x4C\xD7\x03\xC0\x9B\xE9\x7E\x00\x7F\xFD\x01\x78\xB9\xA3\x03\xC3\xF6\xAB\x01\x45\xEA\xF1\xC2\x40\x02\x05\xC4\xC8\x8F\xF4\x14\x54\x0E\x97\x7A\x04\x3C\x65\xF5\xE1\x88\xA8\x78\xCA\xF7\x21\x74\xE6\xB4\x31\x3B\x20\x43\x96\x7D\x9D\x5C\x22\x32\x73\x51\x48\x49\x0A\xB9\x85\x6A\x29\xAF\x2E\x64\xBA\xD1\xAA\x09\xC3\xAB\x74\x25\xE7\xD5\x84\x61\x2B\xD1\x46\x9D\x26\x22\x5C\xAC\x26\xF4\xF7\x0D\xAD\xEA\x29\xBC\x6A\xBE\x3C\x80\xC0\x5E\x7B\xF5\x0F\xFE\xE8\x85\x23\x32\x81\xE0\xD8\xBF\xC2\xD6\x1E\x7D\xCF\xDB\x9F\x49\x59\xE9\x5C\x00\x70\x79\x87\xFB\x79\x2D\xB8\x5B\xAE\x42\x70\xEC\x27\xBF\x4B\xDC\x2D\xF3\xB9\xCA\xD7\x3E\xFD\xA1\xF7\x3F\xB2\x67\xE5\x8F\x0E\x2B\xFF\xA6\x0A\x49\x29\xB7\x2D\x2E\xE8\x6D\xFB\x28\x0A\x17\x0C\xDD\x24\xC1\xEF\x56\xA1\x15\x2E\x1E\xF6\x85\x86\x02\xD7\x93\xDD\x8A\x64\xCF\x60\xC6\x8B\x69\x11\x40\xBC\xCB\x66\x59\xF1\x6E\x9D\x40\xB4\x7B\x01\xC2\x8B\x57\x29\xD8\x21\x45\x88\x83\x04\x12\x7E\x15\xE3\x9B\x87\x48\xB5\x1F\x78\xE2\x00\x24\xB1\xD8\x7F\x41\x1A\xB7\x7F\xB9\xC3\x29\xBD\x77\xE0\x54\x1D\x4F\xEC\x37\xF3\xE3\xD7\x5B\x71\xA2\x4E\x27\xB5\x5E\x22\x91\x2A\xE0\xA3\x95\x0C\x5F\xBB\x37\xF6\x5B\xA8\x85\xD7\xB0\xD0\x35\xB1\xDF\x4A\x3F\xD2\x09\x9D\x4C\xB4\xDC\xB9\x73\xCE\x06\xD6\x28\xE2\x86\x3C\x97\x30\x5F\xBB\x8D\x7D\xAE\x0C\x70\x9C\x5F\xE9\x74\xF9\x35\x8E\x3C\x1D\xBC\xA4\x44\x42\x2E\xD8\x36\x8B\xE9\x90\x6F\x38\xC6\x01\xC7\x63\x7A\xE3\x11\x74\x6B\xFA\x8A\x86\xD2\x42\x42\xB4\x4B\xBA\x79\x56\x65\x06\xA4\xA5\xF5\xCD\x06\x9C\x5F\x26\x9E\x79\x99\x2C\x7A\xA9\x17\xBD\xA4\xFB\x03\xB6\xC9\xF0\x9E\x76\x99\xF9\xD1\x05\x5B\x2B\xFD\xD6\x12\x5C\x0B\x32\x55\x40\x41\x89\xE2\xBB\x71\x8E\x31\x4E\x2E\x4C\xEF\x69\x6B\xA3\x5D\x16\xCA\xA3\xDD\x3A\x86\x70\xF7\x02\xE8\x8B\x57\xAB\x90\x82\xDF\x9D\xAA\x12\x88\x21\xE6\x57\x11\x65\x87\x42\xFA\xE9\xB6\x36\x65\x0B\x17\x90\x83\xAD\x15\xB4\xB5\xCA\x6F\xAD\xA0\xAD\x8D\x26\x7B\x6C\xA3\xF2\xDB\x18\xB1\x4B\xBE\x20\xD6\x93\xB7\x31\xF2\xB6\x57\xA4\x24\xAC\xDD\x40\xBE\xD2\xE9\xFC\xEA\x84\xB4\xE0\xC3\xD7\x2E\xEA\xA4\x44\xE4\x35\x7E\xDB\x10\x90\xB2\xC1\xB6\xA5\x20\x70\xDB\x28\x86\x56\xB8\x4B\x1A\x07\xD5\x6D\x5B\xDB\x1E\x6D\x5B\x04\xD1\xCC\xCB\x64\xD1\x4B\xBD\xE8\x25\x6D\x1B\x0D\x56\x41\xC2\xC3\x49\xCD\x9F\x29\x8D\x5B\x16\x6D\xCB\x0B\x6D\xFE\x68\xC5\x9B\x43\x8E\x2F\xCF\x95\xB4\x95\x7C\xE9\xE3\xF4\x21\x89\xC7\x46\x56\x05\xA5\x7C\x3D\x9B\xB1\xC6\xDC\xD8\x80\x77\x9F\xE3\xE2\x1A\x08\x37\x26\x9B\xB5\xA1\x20\x2B\x21\x64\xFC\x98\x6F\x42\x68\x03\x30\x90\x6F\x4C\x9A\x5D\xEC\xE5\x32\x56\xB9\x00\xE1\xEE\x45\x88\xF8\x2E\x21\xF4\xCD\x24\xB5\xC1\x9A\x1B\x74\xAD\x70\xB5\xA9\x33\x30\x67\x98\x0E\xEE\x5E\x02\xB3\x5B\x17\x3B\x57\x4F\x53\x93\xE4\x67\xD6\x35\xF9\x16\x6E\xB2\xB8\x58\x17\xF8\xF8\x83\xF4\x58\x8F\xA8\x11\x18\x5D\xAD\x0D\xD0\xFB\xB7\xF1\x7B\xD7\xB8\x06\xB3\xD1\x90\x0B\x2D\x84\x80\xC3\xC0\x09\x34\x35\xF6\xDF\x50\x24\x92\x84\x06\xB0\xD1\x40\x88\xDF\x62\xFC\xC6\xA7\xC9\x0B\x3B\x71\x35\x9C\x93\xA2\xD8\x55\x0C\xF7\x77\xFE\x26\x24\xF6\x85\x4C\x43\x5F\xB3\xEE\xEB\xE8\xAA\xBF\x84\x3F\x0D\x59\x73\xC4\xED\x6D\xD5\x52\x77\x7F\x55\xED\xCD\x9C\x60\x04\xC5\xA5\xAB\xC7\xDB\x4C\x89\x2E\xA8\xBB\x30\xF6\xB7\x7B\x17\xEA\xE6\x67\x93\x98\xF1\x94\xF1\x95\xC4\x0C\x42\x8D\x22\xB0\xBF\x7A\x75\x70\xCB\x20\x28\x62\xB9\xFD\xB5\x37\xF7\xEE\xE3\x05\x61\x9E\xC7\xF1\x35\x19\xF4\x29\x38\xD5\x71\x41\x4D\x72\x66\x3C\xE5\xB9\xF6\xAE\x30\x25\x5E\x88\x09\x61\x06\x09\xC4\x0D\x05\x64\xAC\x72\x82\xD6\xAA\xE0\xEB\xC1\x11\x3B\xB1\x52\x50\xCB\x92\x74\xC4\xF5\xA8\xA9\x97\x40\xDB\x00\x96\xA0\x44\x08\x5A\xB2\x01\x68\x58\x02\xBD\x31\xD9\xE4\x04\xD7\x39\xDF\xDB\x2D\x35\xF5\x32\x15\x5C\x46\x48\xD8\xAC\x97\xA9\xE0\x72\x5B\x70\x0C\x19\x8C\xA0\xDC\xA4\x32\x23\x6C\xAC\xD9\xF5\xA4\xCD\x40\x01\x4B\x90\x6F\xBA\x8E\xF2\xFE\x37\xE4\xD1\xAB\xC8\xDF\x42\xA6\x88\xD5\x69\x77\x2B\x2C\xAC\x6E\x6D\x0D\x56\xAC\xAC\x56\x29\x68\xAD\x78\xA8\x3A\x20\x7D\x28\x58\xCA\xE2\x51\x2D\x73\xD0\xD1\x18\x0E\xE0\x9F\xDB\x9A\xEA\x19\x28\x56\x9C\xAB\x0E\xB2\x55\xBD\xBB\x81\x7C\x46\x53\x2F\xF1\xB2\xD0\x12\xA5\x9C\xE9\x33\x81\x04\x6E\x6B\x97\x85\x66\x6B\x9A\xFA\x99\x34\xDA\x67\xF2\xB2\x3C\x93\x66\xFB\xCC\x76\xB6\x29\x31\xCB\x6D\x99\x7C\x51\x99\x31\x18\xBF\x1E\x86\xD6\x03\x5F\x2E\x41\xE4\x17\x22\xA2\x85\xE0\x45\x3E\x88\x93\x1F\xC1\x32\xEE\x91\xD8\x60\xBB\x27\xF5\x82\x42\xE1\x0C\x05\xCE\x09\xFF\x7F\x6A\x52\xDD\x0E\x0A\x0E\xA0\x34\x52\x3D\xD3\x8A\xEA\x19\xB0\xE2\x32\x37\x3D\x03\x56\x4F\x23\xB4\xD8\x9C\x41\x80\x66\xAD\xFC\xBD\x2B\xCE\x6E\x89\x2F\x16\x69\x76\x25\x92\xB1\x76\x08\x9B\x75\x64\xF9\x72\xAD\xDB\xF4\x98\xAC\x89\xE1\x99\xBC\x46\x48\x67\xDD\xAA\x8C\xA9\xDE\x18\x22\xAC\x37\xA6\x7A\xE3\xB6\xDE\x12\x94\x7E\x72\x25\xEF\x32\x64\x7E\x9F\x53\x30\x10\xF9\xD5\x88\x86\xDF\x16\xCF\x9F\xE4\xAA\x67\x20\xC7\xFE\x0C\xB8\x7D\xBD\x55\x54\x28\xBA\xD7\x58\xC5\x4D\x5F\x85\x03\x95\xA2\xD8\xA8\xC2\x52\x12\xA4\x53\x93\x6A\x44\xA1\x3B\xD8\x90\x55\x73\x9E\x66\xD0\x2F\x28\x02\xC8\x79\xB3\x73\x70\xF9\x2A\xEA\xD2\x09\x72\x25\xC5\x55\x35\x94\x47\x65\x6C\xEF\x74\xE1\xE1\x29\x86\x53\x1D\x4D\xEA\xD2\x3E\xDC\x40\x09\x1A\xB4\x2D\x1F\x82\x68\x52\x47\x64\xD7\x6A\xF0\xCB\x79\xFC\x12\x21\x0C\x3F\x04\x06\xBF\x8C\xDD\x87\x0B\xFE\x83\x76\x1F\x14\x47\xB6\xA7\x1C\x83\x60\x26\x56\x4C\x37\xAD\x68\x1E\xDA\xE4\x50\x63\x39\x4E\x36\x87\x91\xBB\x7A\xDD\x11\x94\xE4\x78\x21\xED\x60\x92\xE1\x2C\x09\x18\x3D\xEC\x5F\x3D\x76\x89\xB3\xAC\xD8\x47\x76\x76\x0E\x6F\x50\xC6\xA0\x3E\xA9\xD9\x06\x59\xBE\x9E\x93\x14\xBB\xB4\x78\x24\xFE\x94\xFF\xEE\x22\x87\x71\x9E\x51\x7C\x5B\x71\xCB\x47\x20\x9E\xF0\x08\xDE\x17\x45\x9A\x2E\x72\xE5\x85\x78\x5B\x5E\xE8\xE2\x70\xBA\xE0\xE8\x48\x41\x1F\xAC\xB4\xBD\x4A\x83\x09\xDD\x60\x0E\x77\x83\x09\xFB\x83\x09\xFD\xD8\x87\x0C\x54\x15\xDA\x6D\x27\xC3\xD3\x22\x7D\xC7\x2B\xC9\xBD\x6B\xD7\x2B\xF2\x10\x5C\xEF\x27\xBB\x60\xFC\xFA\x3A\xCE\x2C\xCC\x8A\xD3\xCE\x16\x46\x3D\x57\xE6\x95\xC1\x3F\x49\x95\x83\x46\xAA\x53\x00\xDF\x40\x8D\x60\x04\xE9\x90\xE0\x46\xC8\xD3\xE0\x21\x9D\x30\x65\x49\xE8\xFC\x4E\xF8\xFC\x46\x1C\x28\x1C\xAE\x2E\x37\xF5\x0A\x15\x5C\x81\x64\xB6\xA0\xA7\x24\xE5\x6E\xBD\x0A\xE6\x02\xC4\x17\xAF\x36\x75\x49\xA5\x4B\x46\xCD\x92\x4A\x97\xDC\x6C\x09\xCB\xBB\xF5\x01\xC8\x5D\xC1\x41\xFF\xFD\x82\x4C\xB5\x22\x58\x6D\x8B\x0E\xDA\x8C\xA8\x68\xC4\x6D\x46\x70\x60\x71\xCF\xFD\x52\x74\xD9\xC7\x27\x0F\x07\x07\x1D\x2C\x24\xDB\x9C\x46\xCE\xB4\x8D\x55\x3C\x23\x8A\xE6\x5F\x87\xBB\xB5\xE1\xE6\x2B\x73\x4C\x10\x32\x97\xB4\x9E\xB9\x0B\x48\x5B\xE7\x30\x22\x96\x3E\x26\x0A\x42\x9D\x8C\x61\x8C\xAC\x59\xB8\x5B\xAF\x76\xEF\x13\x6C\x33\x77\x6D\xE5\x60\x80\xBE\x5D\xAD\x42\x28\xF1\xAC\x29\xD9\x21\x81\xFA\x57\xE7\x9A\x2A\x84\xA4\xED\xBA\x1D\x42\x7E\x4C\x54\xAB\xEE\x52\x31\x84\x55\x3F\x84\x55\x08\x7B\x43\x20\x8B\xDE\x12\x8C\xFF\x6A\xA0\x1C\x7C\x25\xEB\x57\x3F\x10\xE3\xD7\xB8\x5A\x45\x19\x17\x17\x88\x1D\x23\x9C\xA0\x26\xAC\xE0\x58\xBF\x82\x56\x25\x64\xC3\xA5\x11\xDB\x97\x23\x83\x58\x7E\xE7\x2B\xC9\x33\x81\x9C\x6E\x29\x61\xEF\xA8\xA9\x03\x32\x5D\x20\x2A\xD9\x06\xA3\x8D\xCC\x8F\x46\x49\xC4\xD6\x2A\x64\xB9\xA1\xB6\xC5\x4D\x61\xD3\xCD\x61\x14\x8F\x94\xEE\x1B\xC8\x70\xD4\x8F\x38\x66\x5B\x63\xF2\x69\xC2\xE1\x7E\x57\x3B\xDC\x24\x57\xC8\x77\x46\xE4\xF9\xC4\x73\x9E\x91\x6A\xC2\xDD\x2A\xB5\xA2\x4A\x1C\xDD\x60\x68\x61\xB9\x41\x6D\xB4\x56\x49\x14\x49\x0A\x61\x25\x76\x1B\x96\x76\xB0\x22\xE8\x7A\xAB\x07\x2B\x11\x6D\x45\x7A\xD1\x1B\xBB\xD4\x05\x5D\x69\xA3\xA4\x3B\xEA\xDE\x13\x27\x92\xBB\xB6\x10\x56\xE8\x1B\xCA\xBB\x02\xF7\x48\x50\xFA\xFB\x0E\x56\x22\xC8\xDA\xAE\xDB\x21\x20\xAC\x8C\x9C\x01\x60\x04\x23\x3F\x84\xD1\xFC\x10\x44\x1F\x56\xC4\xE0\xEB\x60\x20\xC8\x7D\xE6\xF4\x6B\xE4\x85\x55\xBD\xD1\xC6\x94\xA2\x25\x62\xBA\xA5\x3D\xD0\xF0\xF2\x8C\xE7\x28\x93\x3C\x47\x8A\x2D\x75\xAE\x5A\x62\x41\x08\xCA\xA6\x8E\xF9\x96\x3F\x70\xC4\x49\x10\x2E\x0B\x3A\xD3\x6B\x41\xB8\x2C\x18\xE3\x05\x04\xB0\xD4\xD4\x74\x9D\xEE\x7C\x77\x03\xC8\x40\xCC\x16\xE4\xAB\xD9\x08\xF4\x2E\x32\x9C\xB4\x82\xC8\xFF\x87\xCC\x64\x60\x69\x4D\xA5\x35\x37\xAB\x41\xEE\xD6\xCB\x3C\xC3\xAB\x33\xFD\xF7\x0B\x92\xC9\x1A\x2E\x69\x5B\x74\xD0\xE6\x0C\x71\x5A\x5E\xDC\xF3\x2C\x71\x22\x91\x2E\x80\xF1\x1C\xDA\x24\xE6\x03\x63\x91\xCD\x87\x7F\x66\x43\x2E\x1B\x94\x1F\xE6\x84\x79\x14\x6D\x9C\xFF\xE4\x1E\x67\x0E\x05\x41\x1D\xB1\x8F\x90\x0C\x10\x37\xCB\x5F\x61\xC3\x53\x94\x55\xC4\xA1\x40\xB8\x0C\xC2\x95\x58\x18\x99\x97\x1D\xE6\x88\xC9\x88\xED\x3B\xB7\x9B\xD1\x73\xC2\x28\x8A\xA2\x00\xFF\xA3\xA3\x45\xFF\x7A\xDF\x85\xCB\x04\x06\x5B\xF6\x77\x83\x13\x2E\xB0\x54\xA1\xF3\xC4\xE5\x0B\x83\x2D\xFB\xC7\x01\xE7\x9E\x08\xAD\x2E\x7F\x8B\x71\xF8\x50\xA0\x90\x0E\x23\xC6\x29\xC4\xDD\xC8\x1E\x3E\x51\xE8\x3C\x34\x33\x23\x31\x52\xF9\x7F\x22\xA0\x28\x56\x11\xC4\xF6\x0F\x82\xF5\x82\x53\x48\xB7\x7D\x73\x1F\xBE\xC7\xDF\xF5\x4F\x6A\xD8\x7F\xD2\xF5\x4F\x59\xC9\x62\x96\xBF\x62\xEA\xF5\xF0\x3A\xF9\x20\x70\xA4\x3D\x5E\xD2\xC3\x27\xF8\x3A\xBA\x12\x54\xC2\xFB\x24\xC5\x84\xEF\x9C\x93\x99\x0B\x3E\xD0\x70\x30\x6A\x03\x51\x11\x51\xA8\x4B\xE4\xDB\x28\x76\x20\x09\xDA\x20\x20\xDA\xAC\xD3\x36\x4B\xC5\x1D\x9B\x2F\x28\x42\xBE\x78\xA1\xC9\x1E\x2F\x42\x88\xED\xCF\x91\x63\x16\xCF\xAF\xB7\xAE\xC3\xB9\xFD\x5E\xC0\xDE\x5C\x7E\x0D\x03\x1A\x1A\x57\x8D\xA8\x6A\x44\x55\xB5\xAF\x1A\xE2\x53\x88\x4F\x7F\xE0\x9F\x22\x7C\xFA\x2F\xFE\x29\xC6\xA7\xBF\xF4\x4F\x49\x11\xBA\xEB\x2C\x5E\xAC\x9C\x1C\x85\x13\x97\x2B\xD9\x29\xF0\x06\x40\x93\xDA\x8F\xBB\xDE\x39\xC0\x73\x64\x7F\x86\x27\x10\xE2\x06\xB3\x72\x88\x27\x43\x20\xAA\x04\x5D\x20\x7C\xC9\x3A\xC7\xC1\xD8\xA2\x18\x1B\x21\x19\x27\x62\xC1\xF4\x78\x21\xF3\xD8\x79\x2F\x45\xC0\xFA\x0B\x48\xDD\xE2\x87\xDE\x15\x51\x50\xEC\x07\x97\x65\x2E\x1C\xBC\x61\xDE\x2C\xA5\x0D\xE5\xA6\x0F\x05\xA2\xDD\x24\xC1\x7D\xBB\x05\x75\xEF\xD8\x75\x83\xC3\x5A\xE3\x4F\xCA\x32\xC4\xB1\xE8\x04\x44\x54\xBD\x60\xF8\x53\xAE\x3B\x4B\x28\xF8\xB6\xB7\xE0\x41\x42\x66\x8C\x29\x99\x22\xD8\xAC\xFC\x1E\x8E\xAD\x27\x8F\x22\x93\x49\x05\x6A\xC1\x1E\xDD\x87\x02\x9F\x4D\x38\x27\x67\x13\x08\x38\xA9\x4B\x58\x09\xBA\x10\xB3\xB7\xF9\x16\xF9\x43\x4E\x3B\x11\x71\xFE\x90\x7A\xBD\xA0\x92\x10\xE1\x99\xB8\xF3\x70\x63\xC5\xA9\x42\xA3\x14\x11\x6F\x92\xD2\x2C\x61\xF5\x26\xB9\x64\xD6\xC2\x5E\xFB\xF4\x87\xDE\x8F\x25\x20\x70\xFE\x79\x9C\x8A\x0F\xC4\x26\xC5\xEB\xA3\x77\x5B\x27\x27\xD8\x78\x8A\x02\xD8\xA9\xCD\x2A\xC5\x75\x76\xF5\x94\x7D\xE4\xF5\x9F\xFC\x88\x22\xC8\xE7\x9A\xFC\xE2\xFE\x22\xE0\xFE\x69\x24\xD4\xCA\x7A\xA1\x7C\x9E\x93\x5C\x9A\x5B\x81\x42\x62\x88\x42\x72\x31\x0A\x51\xC0\xBA\xD4\xA6\x2F\xA9\x33\xC8\xEC\xCB\x4E\x23\xE7\x4B\x1A\x7A\x8F\x16\x28\x6F\xAA\xF2\x03\x17\x5D\x6A\x22\xBB\xA3\x41\x95\x1F\xF1\x77\xA5\xB4\xB2\x39\x4E\x6E\x95\x82\x08\x19\x3A\xCD\xF0\x30\x47\x59\x9B\xEF\x79\x50\x6C\x0B\x39\x55\x8A\xB1\x62\xEA\x82\x10\x60\x67\x56\x40\xBA\x89\x00\xF6\x00\x19\x08\x5B\x8A\x4D\x31\x75\x61\x90\x40\xD8\xEC\x34\xC4\x10\x43\x72\xA2\xA9\xC5\x5D\x41\xD0\xD8\x07\x9A\x3A\xC6\x76\x10\xD1\xB3\xD3\x3E\xA3\xBB\xA0\x12\xB1\x2F\x21\xD8\x4E\x9A\xBC\x0D\x59\x35\xC1\x4C\x6C\x76\xDA\x3E\x80\xE4\x9B\xAC\xA1\xE8\xE0\xF3\x8D\x53\xD5\xAE\x16\x11\x50\xD2\x1B\x5B\x79\x0E\x72\xBA\xB9\xE2\x95\x66\xD3\xF1\xB6\x5E\x78\x57\x10\xB4\x63\x12\x3C\x26\x67\x8E\x1B\xB6\x05\x9A\x5E\x89\xB9\x8F\x54\xE7\x6F\x5B\x09\xDA\x21\x5E\x7F\x77\xDF\xDF\x73\x59\x0A\xCA\x0F\xF2\x21\x4C\x89\x18\x7E\xC7\xDF\xC0\x05\x56\x7A\xB8\x32\xBD\xB4\x0C\xC6\x0E\xD2\x39\xB8\x7C\x52\xA4\x4D\x23\xAE\x35\x58\xB7\x72\xCB\x8A\xF2\x5D\x1C\x70\x40\xAE\xC9\xBC\xE2\x04\x59\x3E\xAB\xB1\xF9\xC3\x55\x61\xBC\xDF\x88\x28\xDF\xEF\xF3\xCC\x1B\xCE\x22\x54\xBE\xD7\x27\x5D\xE1\x84\x5F\x3E\x05\x95\x35\x0D\x27\x15\x14\xCE\xE6\x37\x44\x14\x3A\xE4\x13\xAD\x95\xEF\x63\x84\xB0\x40\xAE\x66\xD7\x82\x29\x05\xB2\x89\x20\x24\x23\xAC\x1D\xE7\xFB\x13\x42\xD4\x3A\xED\x51\x4A\x90\xAE\x3F\xD7\xCE\xCF\xF9\xEE\x95\xFD\xA5\x9F\xFA\x71\x79\x24\x08\xAC\x62\x1A\xAC\xBD\x8C\x1B\x1C\x0A\xFE\x12\x59\x79\x6C\x09\x85\xEE\xE0\xCF\xF1\xE9\x48\x10\x18\x50\x10\x11\x7D\x77\x43\x97\x20\xCA\x9F\x71\xA9\x2D\x22\x50\xA0\xCA\xFF\x97\x6F\xD3\xCA\x9F\xE8\xB2\xDF\xAE\xC9\x1C\x91\xF2\xAB\xB1\xCF\x2A\xE2\x8B\xE8\x6D\xE6\x1F\xE4\x39\xFB\x30\xDB\xFD\x8B\x35\xB9\xCA\x3E\x03\x64\xAF\x95\x22\x1E\xF3\x00\x20\x6D\x2F\x7E\x39\xF8\x90\x22\x37\x4A\xC8\xE8\xF5\xE9\xCD\x3A\x6D\x2F\x8C\x39\xD0\x84\x22\xB7\x4A\xFC\xBE\xFB\x31\x4D\xDF\xDD\x35\xB3\xAF\xAE\xF8\xF3\xB5\x4B\x5F\x41\x5F\xDD\xD5\xB4\xAF\xAC\xF9\xEB\xE5\xD7\x44\xF4\xD5\x5D\xB9\xFB\xAF\x21\x7F\x7D\xED\x2F\x05\xF4\xF5\xFA\x75\x35\x75\x96\x12\xBE\xF5\x88\x4B\x7C\x22\xEE\x0A\x3C\x3E\x68\x22\xE6\x02\x8F\xBB\x16\xC4\x94\xED\x36\xFC\xE7\x84\x3F\x77\x81\x33\x12\x04\xE9\x04\xA2\x4A\xB9\xE0\x98\xA4\x6D\x60\x13\x02\x10\xE5\xCF\x76\x1B\x9A\xD0\xAA\x1D\x0A\xBE\x65\xA7\xDB\xBD\x67\x07\x97\xF0\xE9\xEE\x00\xC9\xA7\x37\x22\x74\xDB\x42\x0D\xD9\xCF\xBE\xA5\xAF\x5D\x76\xE7\xCD\x9A\x4C\xBC\x45\xBA\xB4\xC9\x94\x43\x2E\x2B\x3E\x6A\x0D\x05\x7E\x2D\xA7\x8E\x8B\x21\xDD\x07\xBE\x33\xF6\x9F\xDF\x5F\xB0\x73\x36\x45\x57\x72\xE5\x69\x94\x3F\xDF\x83\x7A\x6D\x65\x63\xD5\x4B\x6B\xC5\xEE\xFF\xCA\x06\x27\x29\x8E\x33\x37\x4E\xAE\xCC\x2F\xAD\x29\x3E\x6F\x84\xCD\x9F\x9C\x90\xAB\x8F\xF1\x99\xDA\xE2\x4D\x0E\x25\x10\xE3\xFF\xEF\x9F\x20\x0D\x3D\x39\xB1\x4B\x4D\x3D\xC0\x1D\x61\x0F\x79\xE5\x53\x0F\x77\x54\x0F\x77\x14\x84\x84\x4B\x1D\xEE\xA8\x16\x77\x38\x99\x2E\xF0\x52\x43\x64\xDF\xF1\x91\xE0\xCB\x02\x0E\x24\x8C\xAD\xE2\x49\xB7\x59\x27\x8C\xBD\xCA\x1E\x86\xC4\x47\x20\x89\x78\x78\x64\x59\xE5\x50\x03\x11\x23\x82\xE8\x26\x50\x43\x3A\xD4\xC8\x9E\x28\x6A\x14\x43\xD4\x28\x66\x51\xA3\x18\xA2\x46\x31\x83\x1A\xC5\x10\x35\x8A\x21\x6A\x14\x43\xD4\x28\x86\xA8\x51\x0C\x51\xA3\x98\x47\x8D\x62\x88\x1A\xC5\x1C\x6A\x14\x43\xD4\x28\x66\x50\xA3\x18\xA2\x46\xB1\x18\x35\xD8\x6D\x85\x51\x83\xCD\xF2\x62\x02\x18\xA2\xAF\xFB\x11\xA9\x05\x7B\x21\x9E\xDE\x8B\x5B\xB1\x17\xC6\xB8\x7D\x78\x3E\x59\xB8\x29\xD2\x77\x28\x17\xB8\x4E\xB1\xA1\x0D\x17\xD8\x3C\xC9\x71\xD2\x55\xC3\xAB\xEF\x37\x2D\x01\xBA\x7B\xE1\x4D\xA3\x38\xEA\xB5\x72\x3B\xC7\x07\x51\x9D\x5A\xC9\x78\x28\xF8\x72\x68\xBB\xA9\x0A\xB2\xCC\xC2\x5D\x4B\xFD\xAE\x91\x2B\x23\x07\x85\x05\x45\x99\x7B\x38\xF0\x5D\x52\xA1\x44\xE7\x8E\x93\x07\xEB\x91\x5B\xA1\x0C\x46\x6E\xE7\x5C\x20\x59\xD1\xD4\x7B\x55\xA4\x7D\x1D\xB9\x75\xA0\x8A\xB4\xA5\xFB\x56\xA4\x0D\xEF\x77\x48\x7B\xBD\x6F\x35\x82\x84\x7E\x77\x04\x04\xFB\x56\x23\x10\xE9\x57\x23\xE8\xD8\xB7\x5A\x0B\x3B\xFD\x81\x22\xD8\xDC\x5C\xCD\xC7\x67\x3A\x7D\xFC\xA6\xFA\x64\x68\xEB\xD7\xCB\xF6\xAB\x86\x78\x28\xA8\x36\x57\x4A\x11\x1E\x53\x1F\x4D\xBE\x40\x90\x2C\x1C\x48\x52\xE4\x1B\xCA\x4C\x98\x3B\xF7\xAB\x9F\x09\xEC\xC7\xD9\xFB\x92\x22\x43\x45\x14\x28\xCA\x20\x91\x57\x1C\x2D\x4A\xD0\x41\x75\xE7\x37\xD4\xA2\xA1\x04\xE8\x1C\x53\x7E\x9E\x98\x44\x7C\xFC\x01\x27\xC5\xB6\x5F\x7C\x4F\x1B\xD0\xDE\x70\x80\x23\xD0\xCD\x46\x11\xD8\x0F\xBC\x76\x70\xD1\x1B\x41\x88\x5D\x51\x64\x16\x94\x4F\x3E\xF8\x58\xEF\x4C\x76\x24\x26\x80\x90\xCC\x11\xC7\x2E\x7D\x5D\x60\xB7\xCD\x67\x74\xB0\x50\x31\x24\xED\x85\x66\x94\x26\x71\xC4\xC6\xB7\x14\x44\x3D\xE0\x45\x0B\x78\xD1\x38\xC0\x08\x89\x80\xFE\xBE\x5C\xBA\x95\x73\xB1\x47\x04\x9E\x8D\xE4\xC4\xB4\x49\x3A\x28\xDF\x80\x5B\x75\xD5\x6F\x80\x50\x47\x3A\x00\xE9\xD7\x67\x0C\xDA\xAF\x3A\x21\xD0\xA2\xEA\x8C\x47\xFB\x54\x27\x34\x5A\x50\x9B\xB1\x69\x9F\xCA\x84\x4C\x0B\x2A\x33\x4E\xED\x53\x99\x50\x6A\x41\x65\xC6\xAC\x7D\x2A\x23\x4E\x2D\xA8\x4B\xA8\xB5\x4F\xD5\xC7\x17\x77\xFB\xF8\x4D\xF4\x9A\x2D\xAC\x99\xED\x53\x91\x91\x0B\xEB\x77\x55\xCD\xFF\x2C\xC5\xB6\xB8\xE0\x72\xB8\x89\x5D\x08\x76\x2F\xD4\xEA\x18\x3C\x7A\x15\x1B\xB8\x6A\x83\x35\xF9\x11\xBA\xAD\xE3\xD8\x4B\x9C\x24\xF1\xB0\x10\xDB\x83\x1A\xAE\x3C\xC9\x56\x6B\xF2\x71\xAA\xF0\x48\xAF\xC2\xBD\x42\x5C\x60\x90\xDE\x05\xB1\x7B\xA1\xD6\x6D\x05\x4D\x5D\x7C\xB4\x57\xA3\x96\x74\xA5\x68\x29\x92\x88\xAB\x6F\xFD\x20\xD5\x7C\x1B\xFA\x6A\x2D\x21\x40\x89\x5E\x72\x5E\xD7\xD9\x21\x2B\x03\xCA\xBC\x45\x70\xFC\x91\x00\x3F\xFF\x06\x7E\xAE\x84\x3C\x6A\x0F\xCE\x96\x76\xF9\xD9\xDD\x8D\x08\xA2\xB2\x9A\xBA\xA0\x3F\x88\xAE\xAB\x9C\x46\x44\xE0\xB9\x34\x5B\x13\x3F\x43\x40\x31\x8C\xE6\x2E\x39\x83\x31\x55\x3B\x42\x89\xCD\x83\x23\x32\x07\x79\x2C\xB8\x9B\x33\x8A\x53\xCC\x1D\xDA\x20\xF3\xC3\x42\xC8\x6D\xD6\x8B\x28\xD6\xFE\x94\x20\x9E\x5F\x04\xF6\xBF\x7C\xCF\x8C\x67\x50\x37\xB6\xF5\xF6\xF2\x68\x6E\x44\x1A\x47\xA4\xF9\xCE\x4D\xD1\xE8\xCA\x5F\x66\x31\xBA\xFC\x50\xA7\xD3\xCE\x21\x70\x63\x61\x4B\x94\x1D\xBD\x85\x74\x09\x87\xAC\x41\xD1\x20\xAC\x3C\xE7\x52\x3E\xB3\xB6\xA9\x84\xA0\xFC\x55\x17\x61\xE1\x75\xC3\x31\x0F\x07\xA6\xDA\x81\x0D\xB6\x58\xB3\xC9\x27\xB7\x04\x38\x4C\x2B\xC6\x7E\x94\x1C\x10\xC2\x4D\x1E\x39\xB9\x99\x51\xDB\x36\x25\xE8\xE2\x91\xE3\x76\x7F\xAF\x08\xE4\x51\xB6\xDF\xC7\x66\x82\x17\x14\xBD\xE7\x84\x9E\xAD\xB0\xD2\xBF\xD1\x10\x9C\x9A\x8C\x8D\x55\x56\xFB\x57\x39\xBF\xEA\x6A\x1D\xE4\x5A\xA1\x8D\xFC\x9B\xD5\xD9\x22\xC0\x45\x62\x9B\xF8\x37\x77\xB8\x86\x53\x9B\xF9\x57\xCF\xC2\x57\xE6\x6E\xBE\x74\xC7\x8E\x6A\x31\x27\xC2\xD1\xFB\x0A\xF9\x27\xF2\xBB\x5A\x93\xAB\xCE\xEF\x89\x02\x9E\x1D\x0A\x02\x2B\xA7\xED\xCF\xBC\xFB\x29\xBA\x9F\xBA\xFB\x39\x9A\x9A\x17\x30\xC6\x0A\x17\xB3\xCA\x8E\x58\xEE\x7B\xA2\xDD\x1F\xA7\x97\xFD\xEE\x6E\x0F\xBC\x6D\x0E\x85\x5D\xEB\x79\x90\x9B\x53\x6E\x8A\x87\x82\xA0\x16\x1C\xBE\x11\x9B\x9A\xC5\x0D\x2A\xC0\x74\x4B\x90\x3C\x0A\xC1\x9C\x7B\x9E\xF9\x49\x17\xF5\x28\xB8\x2B\x08\x6A\x65\xB7\x71\xC0\xEF\x78\x6B\x6F\xC0\x87\x08\xD1\xC9\x08\x92\x33\x75\xD9\xF7\xBF\xB5\x8F\x32\x2D\xDA\xD7\x0C\xD1\xB5\x46\xB4\xFA\x96\x37\x0D\xE6\x5C\x56\xBA\x23\x0F\xCA\x38\x73\x4B\xE2\x1A\x40\x9C\x9C\xD4\x0A\x34\x27\xDC\xA7\xD9\x33\x7B\xA0\x80\xC2\x9E\xE1\x0F\x59\xFE\x14\xB6\xC4\x80\x0C\xB2\xFC\x69\x0E\x1A\x6E\x20\x34\x7F\xEE\x02\xB3\xEC\xB3\xE0\xE4\xCE\xD6\x42\xED\xEC\xEA\xD7\x72\x06\x8C\xE5\x1C\x18\xCB\x53\x13\x4E\xFD\x39\x80\x64\x7E\x2B\x06\xC0\x2C\xE7\x80\x79\xBE\x14\x70\xA9\x01\x3C\xFB\x52\x43\x90\xA6\xB7\x6C\x52\x2B\x6C\x7A\xBA\xB1\x2F\x6B\x8E\xC8\xC4\xAC\x50\xF2\x56\x5C\xAC\x3F\x11\x53\xAB\x91\x31\x31\xE3\xF6\x9D\xA0\xE7\xB2\x7D\xFE\xAC\x98\x0E\x0B\xDC\x49\xCF\x5D\x23\x9F\x12\x53\xE0\x4A\xEF\xC3\xF5\xEC\x34\x7C\xA2\xCD\x18\x4F\x00\x22\xF7\x00\x10\xC9\x7C\xDD\x3C\x80\x0C\x3E\xC3\x94\x21\xB5\xEC\x60\x46\x3E\x9F\x4F\x96\x7B\x82\x31\x87\xD2\x59\x05\x69\x35\x08\x3B\x6E\xCA\xF7\x60\x03\x48\x5C\xC5\xA1\x60\xEC\xA1\x97\xAA\x1D\x91\xA5\x81\xE0\xD8\x23\x6C\xCF\x9D\x38\xE8\xF7\x71\x44\x9C\xAA\xF2\xAF\x95\xD0\xDB\x6C\xF7\xA9\x6C\xFA\x12\xEF\x8D\x44\x90\xB7\x0F\xB4\x43\x68\x83\x75\xB2\x21\x58\x08\xF4\xA0\x21\x72\x8A\x0E\x3F\x0F\xCD\xB0\x1F\x2D\x84\xFD\xA8\x83\x7D\xED\x33\x7F\x6B\x3C\x02\x4E\x4E\x6A\x0D\xD1\x09\xA7\x7D\xEC\xC1\xBE\x86\xE8\xF9\xE4\x41\xA3\x3D\xEC\x7B\x5F\x43\xDD\x82\xBF\x36\x6C\x3E\xC9\xC4\x3D\x77\x51\x21\x8E\xC8\x84\x1D\xD1\x04\x47\xD9\x3C\xE9\x88\x03\x9B\x14\x38\xE7\xCF\xD0\x87\x61\xD8\x76\x81\x01\x2E\x34\x4C\x94\x02\x5F\x9C\x2E\xAF\xF8\xD4\x11\x0D\x05\x40\x73\x4A\x7A\xBA\x15\x0F\xF0\x23\x05\x16\x22\x57\x6A\xBE\x04\x3E\xCF\x26\xBC\xE4\xF7\xDA\x0B\x84\x26\x28\x10\x1A\x5B\x1C\x6B\xF6\x35\x0D\x1B\xFB\xB5\x4D\x15\x60\x03\xC7\xB0\x81\x3F\x7B\xCF\x0F\xFF\xE2\x97\x91\x89\xBE\x3D\xD8\xCC\xBE\x29\xE7\xDE\x24\x73\x6F\x5E\xC4\x46\xDF\x0F\x37\xB5\xB6\x31\xC9\x30\xE6\x41\xE1\x72\x00\x3B\xC2\xD0\x2E\x44\xE5\xB2\xF1\xBA\x3C\xEB\x3E\x9D\x77\x76\xA6\x20\x2B\x40\xF6\xE5\x26\x2F\x13\x7C\x79\xBC\x08\x72\x6D\x4C\xE7\xAA\x90\xB2\x07\xE0\x76\x53\x0B\x67\xDC\x4D\x7C\xD4\x35\x21\xF4\x9E\xE7\x80\xDC\x83\x2C\x39\xB4\x70\xB4\xA8\xF6\x2E\xB8\x48\x7A\xC6\x5E\xB9\xE8\xEE\x6F\x94\x72\x5A\x18\x77\xC4\x6B\x3F\x60\x17\x02\x4A\xB3\xCB\x5F\x37\x60\x67\x84\xAB\x70\xC0\x6A\x10\x28\x11\xE1\x9B\x48\x9A\x6A\xEC\xF6\x59\x7B\xE7\x43\x7C\xF4\xFC\x80\x0B\x9F\xB8\x2F\x21\xED\xA8\x67\x2D\xE7\x47\x2C\x86\x23\x96\x37\x3F\xE2\xD6\x5E\x98\x46\x3C\x58\xE2\xD9\x11\x9B\x8F\xE0\xA9\x35\x67\x80\xE3\x11\x5B\xEF\x81\xD8\xDA\xB8\x38\x23\x8B\xA9\x94\xFB\xBC\x80\x4A\x85\xCF\x27\x18\xE9\x53\xA9\xD0\x22\x30\xF7\xA8\x94\xA4\xD0\xA7\x8B\xE8\x14\xB7\x3D\xC3\x58\x11\x41\xDF\x02\x8D\xA4\xD7\xE9\x8C\x5A\xCA\xF5\x7B\x7F\x0B\x66\x47\x13\xE3\x83\xC9\x31\x91\x20\xB6\xBC\xBF\xA0\x9B\xE7\xCF\x69\x8F\xA4\x82\xE6\xD4\x5A\x82\x20\x6C\x69\xE4\x7A\x36\xEB\xD6\x5C\x9D\xAE\x8C\x29\xCA\x1F\x9B\x9C\x0B\x67\x7B\x22\xAC\xF4\x36\xED\x1B\x93\x5A\x30\x25\xF4\x87\x1A\x9F\x84\xFD\xA3\x5F\x3B\x42\xC8\xE7\xBE\x9E\x39\xF7\xF5\xDC\xB9\xAF\x4F\x4D\xC8\x83\x7C\x78\xEE\xF3\x5B\x39\x38\xF7\xF5\xDC\xB9\x3F\x5F\x0A\xB8\xD4\xE0\xDC\xF7\xA5\x86\xE7\x3E\xBF\x15\x66\xB8\x26\x9D\x36\x22\xE4\xE5\x08\xDB\xE5\x68\xBA\xE5\x10\x7B\x2D\xC7\x2D\x59\x8B\x59\x1E\xC8\x8F\xF4\x66\xD6\x42\xDC\xD4\x5A\xCC\xF2\x40\xF4\x96\xE5\xF3\xF4\x34\x68\xC7\x03\xF5\xAF\x2C\xCD\x07\xC3\x96\xDE\x53\xE4\x58\x3B\x62\x1E\xD2\xE6\xD3\xE3\xC4\x44\x0A\x82\x40\x7B\x07\x4E\xB5\xFC\x8F\x2D\x37\x29\x40\x96\x1F\x6D\x91\x84\xA8\x36\x04\x7C\x91\xC2\xC4\xF6\xF9\x24\x4F\x29\xD0\x9B\xB5\x2E\x7F\xD7\xE9\xC9\x03\x26\x47\xBA\x39\x22\xF5\x9E\xD6\x47\xC2\x1E\xC4\xCE\x46\x49\x98\x4A\x15\x68\x11\xA5\x64\x6C\x76\xAA\xD2\x39\xB9\x14\x86\xA7\x8A\x08\xFF\xAC\x17\x31\xED\x2E\x45\x1E\x99\x56\x64\x21\xA1\x8E\xE3\x4B\x1C\xD1\xF6\x59\x7E\xA9\xE9\x65\xE4\xC5\x03\x8E\x6A\x80\x8D\x70\x14\x03\x85\x0C\x8B\x66\xFF\xC1\x60\xBD\x6D\x2F\x3B\x53\xEB\xE3\xA0\x6D\x78\x62\x6B\xDA\x32\x0C\xFE\x58\x5C\x25\x9A\x5E\x7E\x9F\xBF\x7B\xC2\xA7\x3E\x63\xDC\x6E\x14\x41\x16\x4D\xD9\x22\x98\xF1\x36\xF2\x41\xA1\xF7\x38\x28\xF4\x0C\xC7\xAD\x67\x39\xEE\x70\x06\xDA\xC2\x39\x68\x0B\x4F\x4D\x28\x9A\xF3\x10\xDA\xF8\xED\x10\xDA\xC2\x39\x68\x9B\x2F\x05\x5C\x6A\x00\x6D\xBE\xD4\x10\xDA\xE8\x2D\x1B\x28\xA6\xA7\x91\x57\x79\x59\x83\xF2\x3E\xB9\xC6\xAD\xC9\x80\x64\x12\x2B\x18\x16\x6E\xEF\x71\x5D\xFD\x15\x1A\x80\x96\x71\x04\xB0\x0E\xEC\x0E\x30\x0D\x24\xF7\x68\x7B\x0D\xB6\x2A\xBA\x6D\x67\x19\x1B\x9F\xB0\x2C\x32\xB2\xC6\x7C\x34\xDA\xEB\x10\x68\x2F\xC6\x69\xE1\x04\xC8\xCD\x5A\x3C\xDF\xDF\x60\x17\xED\x39\x21\xF6\x38\x27\xDC\x19\xBE\xD7\x39\x41\x7C\x20\x4C\x71\x96\x83\x53\x42\x3C\x9F\xA2\x97\xF4\x4F\x09\x31\x7B\x4A\xF0\x19\x21\x86\x67\x04\xDD\x56\x20\x03\x2E\x41\x22\x0B\x2E\x1A\x64\x88\xD6\x89\x43\x65\x57\x53\xE6\xCB\x38\xDA\x6C\xCB\x5B\xE8\x86\x92\x7D\x41\x88\x6B\xC3\x6C\xAB\x37\x6E\x3D\xC3\xAE\x06\xEC\xAE\x70\x72\x52\x31\x5B\x0A\x54\xA3\x92\xE4\xD3\x2C\xAC\x3C\x59\x08\xE2\x22\x2A\xC1\x71\x25\xB4\x8F\xDF\x80\xEC\x88\xB3\xC5\x22\xE3\xBC\x27\xB7\x5A\x03\xD1\x97\x17\x8B\xA4\x99\x79\xF6\x5F\xF6\x34\x63\x9C\xAE\x53\x80\x80\xF0\xE4\xA4\x16\x20\xE7\xC4\x5E\x01\x92\xC5\x5E\x41\x3C\x33\x73\xFF\xB7\xE7\x2E\xC3\x09\xBD\xFA\x69\xD6\x03\xB0\xCA\x3B\x27\x3E\x7F\x31\x16\x86\x33\x58\xC8\x4C\x1B\xBB\xB4\xFB\x68\x0C\xB5\x98\xC1\x44\x31\x87\x89\xE2\xD4\x84\xAD\xB9\x06\x98\xC8\x6F\x87\x98\x28\xE6\x30\x71\xBE\x14\x70\xA9\x01\x26\xFA\x52\x43\x4C\xA4\xB7\x6C\xB8\x97\x22\xDC\x30\xDD\xDF\x7B\xBE\xB8\xD9\x7C\x52\x12\x53\xDE\x06\x9C\x20\x21\x09\x19\x93\xA4\x6F\x9E\x6A\x7E\x2D\x13\x7A\x11\x86\x39\xFA\x4D\x16\x62\xD4\x5C\xD2\x50\x30\x24\x77\xFB\xFF\xFC\x8E\xB1\xBF\x11\xF5\x9B\x23\x79\xAC\x98\xA3\x46\xE9\xD8\xC0\xDF\x77\xB8\x00\x91\x44\x44\x36\x1D\xED\xF8\x9C\x11\x98\x3E\x2F\x60\xF4\x44\xCB\xE8\x15\x43\x14\x2E\x86\x28\x5C\xDC\x9C\xB0\x4D\x9C\x4D\x77\xDC\xAD\xC9\x83\xA3\x34\x10\x52\x05\x41\x20\x02\x9E\x06\x85\x6F\x77\x26\x94\x4F\x9E\x1E\x51\xF4\x58\xBD\xD7\xAC\x3A\xF6\x75\x34\x64\x5F\x47\xB3\xEC\xEB\x68\x0F\xF6\x55\x2C\x60\x5F\x41\x10\x67\x6E\x25\x32\x90\xF6\xDA\xE1\x2A\xC0\x53\x9B\xA9\xFA\xAF\x3B\x8B\x29\x41\xC8\xD1\xED\x1C\x17\x0B\xFD\xCC\xDB\xF8\x9D\x37\x53\x07\x47\xE0\xCD\x53\xF9\x9D\xEE\xB5\x73\x93\xCD\x10\x1F\x2E\x86\xCD\xA8\x19\xB3\x96\x3B\x9C\x7E\xA1\xAD\x4D\x10\x0D\x0C\x86\x11\x99\x92\x97\x9F\x60\x96\xA0\x63\x74\x36\x8F\x74\xBA\x60\x62\xE3\xF7\xC7\x80\x39\xF8\x3F\x5E\x28\xBB\x73\x2D\xA8\x48\x8D\x40\xA6\x07\x92\xBD\x18\xC8\x70\x45\x7F\x75\x21\x3D\x1E\xD0\x5E\x53\xF3\x7B\x13\x36\x17\x9E\x96\x23\x51\xCE\xE1\x19\x0D\xFA\xAB\x69\x02\x0E\xD9\x9E\x6A\x8C\xDA\xEF\x50\xBC\x39\x8C\x12\xA6\xD5\xB8\x5F\x0B\xB6\x38\x5E\x27\x3B\x2B\x7B\x75\x50\xB8\xE9\xB8\xD0\xD0\xCD\x0D\x58\x2D\x43\xBC\x26\xC8\xF2\x0F\x68\xD8\x8E\xE1\x10\x9E\xE1\x08\x91\xE1\x40\xC1\xE0\xDA\xE5\xC1\xF6\x3D\x01\x0D\x35\xF6\x5F\xFE\x1A\x3B\x80\x89\xAA\x65\x6A\x20\x70\x37\x03\x1D\x5D\xFD\x98\x16\xE1\xBC\x91\x60\xA7\x18\xC1\x35\x7E\xC3\x5B\x67\x3C\x79\x69\x63\x3E\xF0\x7D\x83\x58\xF4\xEA\x89\xD1\xD9\x3A\x6C\x21\x2D\x20\xCD\x39\x57\x27\x5E\xFE\xFE\x82\x9D\x20\x14\x88\x13\xE5\x7F\x6A\x4D\x8C\x9E\x42\xA8\x90\x73\x50\x21\x07\x50\x21\x9F\x10\x54\x84\x34\x7A\x51\xEE\xF6\x25\x90\x04\x05\x64\x71\x44\x26\x24\x2A\x53\x34\x86\xF4\x25\x14\xD5\x8E\x74\x97\xFB\x4C\x0E\x22\xD0\xEB\x8E\xAC\x2E\x52\x72\x86\xA0\x89\x47\x9B\x53\x72\x86\x0B\xB9\x9C\x70\xA8\xE4\xA4\xC8\x59\x9A\x94\xFB\x1A\xC2\x39\x2E\x47\x43\xC8\x5C\x8E\xA6\xA0\xF7\x03\x2E\xC7\xBD\xEA\xB8\x1C\x94\x62\x4D\xDF\x8A\x54\x99\xDF\x54\x03\xEE\x58\x0F\x55\x24\xE1\x1E\x93\x0E\x8D\x0B\x21\xBB\x78\x47\xDD\xE7\x05\x3B\x4A\x24\x52\x0F\x76\x34\xB2\x1A\xF4\x8C\x8A\x44\xEF\xAD\x22\x09\x9D\x9E\xD6\x49\xF2\xCA\x5B\x0F\x38\xC5\x86\xEA\x14\x1B\xB2\x27\xC9\xCB\x56\x92\x97\x2C\xC9\x87\x9D\xAA\x7E\xC8\xDA\x29\xE6\x8E\xBF\x70\xA4\x2B\xD9\x4A\x57\x24\x5B\x75\xE4\xE3\xDF\x0F\x6F\x20\x9E\xEA\xAD\x0D\xE7\xB7\x36\x9C\xDD\xDA\xBD\x36\x96\x23\xAF\x94\xBF\x7D\x91\x3D\xC7\xDA\x29\xFC\xB0\x92\x61\xE7\x21\x08\x03\xDE\x92\x5C\x9B\x48\xD1\xE0\xB2\xD3\x88\x3D\x0E\x3A\xE1\x0E\x3A\x3A\xDF\x04\x02\x4D\xE4\xB7\x37\x3E\x5E\x90\xA2\xF9\x38\xBB\x7C\x10\x87\x20\x9C\x75\xAA\xE6\xF3\x34\x68\x6C\x76\x3F\x61\x1D\xEB\x0B\x41\x94\xFF\x81\x4F\x75\x14\x22\x0E\xDF\x13\x24\x54\x9C\xB4\xE1\x08\x7E\xEA\x88\xBC\x03\x94\x95\x47\xE4\x41\x36\x5F\x13\x36\x80\x60\xB3\xFC\xF8\x45\xE7\x5C\x8C\xF0\xA7\x88\xCA\xE2\x78\xD4\x9A\xBC\xA3\x1D\x4F\x15\xE3\x73\xC9\xA6\x10\xE9\x19\x77\xF5\x40\x56\x67\x54\x38\x6A\xEC\xC3\x0D\xEB\x5F\x52\x10\x93\x9A\x6E\xAE\x09\x22\x5C\x03\x0F\xD6\x4E\xC4\xB3\x19\x08\x0A\x09\xC7\x7E\x85\x17\xEA\x84\x22\x32\x07\x1B\x85\x80\xE4\x18\x3C\x7A\xD5\x92\x17\x0E\x47\xDB\xF9\x6A\xF2\xB9\x20\x85\x14\xCB\x87\x7A\x50\xD3\x59\x31\x24\x57\x59\xB2\xF4\xE7\xAA\xA4\x3B\xF8\x6D\xCA\xF9\xD4\x65\x62\xD3\xE6\xBA\x5A\x2C\x0D\xF8\xFD\x42\xB0\xBA\xF4\xF6\xC1\xA9\x25\xE7\x4F\x2D\xBF\x9F\x0B\x4E\x2C\xD1\xE9\x45\xFC\xA9\x25\x66\x4F\x2D\xE9\x32\x59\x9E\x60\xF1\x7B\x15\xD8\x69\xB5\x67\x22\xB0\x26\xDD\xA1\x2F\x9E\x2D\x93\xBB\x59\xED\xD0\xDE\xE0\xE1\x8F\x31\xA9\x93\x7D\x1C\x6C\x82\x85\x08\xA4\x3F\xFF\x22\x90\xEE\xFC\xBB\xFD\xA9\xBD\xD3\x53\x73\x77\x7A\x9F\x0B\x53\x44\xC9\xD9\x36\x91\x54\x26\xED\xBA\x80\xE4\xD3\xF0\x76\x4A\xCE\xD0\xEA\x17\x6C\x00\xD1\xE6\x7A\x43\x47\x46\x7F\xBD\x16\xAD\xD2\x80\x6D\xF9\x46\xB2\x63\xA1\xAC\x66\x8F\xD3\x29\x2B\xD9\xE0\xCC\xDB\x22\xA3\x24\xB3\x23\x69\xBA\x4E\x07\x47\x61\x95\xE9\xCE\x8C\x32\xBE\x81\x84\x80\xAC\x16\x08\x31\x5D\xA2\x4E\x22\xF4\xF8\x8B\xE2\xE3\x0A\xD0\x1E\x02\x5D\xDB\x06\x94\x79\x93\x74\x77\xF0\x6C\x0F\x43\x80\xA2\x9E\xB4\xBA\xDF\x3C\xE9\x23\xBB\x7F\x2B\x19\x2E\xBC\x95\x0C\x6F\xEA\x56\xB2\x8B\x4E\xD0\xC2\xE8\x8D\xF7\xC1\xF1\x48\xEE\x3F\x79\x07\xF1\xC6\x0A\xF3\x2E\xC5\x17\x53\x4F\x01\x1E\x76\x58\xB8\x00\x07\x41\x95\xBF\xCF\x7A\x62\xCB\x3E\x28\x20\x9B\x5A\x6F\x92\x1F\x79\xFA\x92\x3A\x42\x36\x2B\xDE\x7F\x97\x20\x86\x70\x9D\xD3\x2C\x2F\x64\xB3\x22\x08\x49\x8E\xE9\x9D\x48\x37\x79\x97\x1C\xBA\x48\xFB\x88\x0C\x27\x27\x75\xB8\xF0\x26\x39\xF4\x37\xC9\x21\x28\xB7\x67\x9C\xC5\x8A\x5E\x74\x86\x14\x7E\xCF\xDA\x8D\xA0\x73\x24\x07\x59\xFE\xA0\x13\x39\x75\x7B\xC9\xEC\x29\x10\xAF\xC9\x91\x3D\x89\x90\x15\xE6\x4A\x48\xE6\x19\x8B\x6C\x30\x07\x8A\x71\x61\xF5\xA9\x4A\xA2\x88\x2D\x58\x31\x2E\xE6\x14\xE3\x12\xA5\x64\xD1\xEA\xC0\x59\x31\x2E\x51\xE6\x15\xC7\x8B\xB0\x45\x4A\x12\x2A\xB1\x11\x8E\x91\x2A\x6C\xB0\xCE\x51\x51\xC5\x8C\x62\x5C\x1E\x07\xC9\x8A\x71\x49\x69\xCB\x84\x99\x55\x8A\x57\xC2\xDD\xD3\x3A\x4B\x9C\x27\xAF\x2B\xE4\x7B\x65\xBD\x40\x65\xB8\x2F\x66\x0A\xCF\x4C\x0B\x90\xF7\x4F\x6A\xB9\x10\x33\xA5\xC7\x4C\xD9\xEE\xB2\xE4\x5D\x96\xED\x2E\xF3\x2C\x65\x6F\x96\xC4\x30\x8A\x56\xBB\x9F\x5B\xD1\x10\x9F\xC9\x38\xB5\xF7\x9D\x75\x9F\xEF\x94\xB3\x8A\xFD\x2F\x00\x75\xA2\xEC\xAB\x13\x8D\xF9\x1A\x21\x66\xAE\x8E\x3C\x5B\x35\x9A\xF2\x7D\x09\xDD\x23\xF9\x97\x39\xFB\xC3\x0C\xF9\x09\xA2\xE6\x4C\xF3\xCA\x4F\xD2\x96\xBF\x4D\xB5\x44\x7D\xDD\x21\x17\x79\x85\x4A\x2B\xA7\x4C\xDD\x59\x39\x26\x5D\x6E\x64\xE5\x13\xDC\x90\xBA\x6B\x73\x6C\xAC\xA0\x5F\x56\x9C\xDB\x1C\x9B\x7E\x01\x2B\xCE\xD9\xED\x66\x6C\xF6\x37\x2F\x68\x8F\x92\xBE\x9D\x41\x77\x67\x1F\xD0\x09\xA6\xF6\x20\x91\x74\xBF\x34\x4B\x1E\x85\xE7\x37\x7B\x83\xB8\x41\x03\x2D\x52\x52\xB1\x61\x63\x08\x16\x38\x1F\x37\x3D\xE9\x5B\xAE\x15\xE7\x7B\x94\x9B\x7E\x8A\x4E\x09\xCB\x77\x72\xCC\xDF\x36\x94\x15\xD2\x06\xDC\x80\x9B\xC7\x76\x43\x86\x27\x64\xD9\x68\xB7\xCF\x72\xC6\xC9\x2D\xB7\x2B\x7F\x44\x2A\x31\xE1\x87\x84\xD5\x4E\xB3\x6D\xE8\xFB\x3A\xCB\x07\x5D\x4B\xE6\xA2\x15\x4B\xAC\x12\xD4\x66\xED\x8D\x30\x98\x2E\x7A\x0A\x49\x5E\xB6\x48\x1F\xBF\xED\x95\x3B\x3B\xC1\xE6\x80\x76\x32\x40\x93\x0B\x19\xC8\x4D\x2E\xC1\x05\x70\xB7\x04\x5B\x64\x50\x58\x7F\x67\x89\xC2\x41\xD2\x88\xA5\x65\x48\x21\xA6\xC9\x6B\xB3\x41\x72\xD2\x3F\x0E\xCC\x4F\xF2\x1D\x85\x7D\xA7\x35\x30\x76\xDB\x0A\x3C\x02\x36\x26\x48\x6D\xF5\xB6\xD3\xDC\xF8\x00\xAF\xED\x25\x0B\x47\x4B\x70\xB7\x23\xDB\x0D\xDB\xC4\xB8\x88\xAF\xB2\x8D\xFE\x7F\xDC\x99\x5C\x68\xA4\x97\xF7\x3B\xB3\xB5\x81\xA0\x49\x46\x33\x33\x48\xAE\xE6\x90\x5C\x2D\x94\x2F\xD5\x02\xF9\x52\xCD\x21\xF9\x7C\x29\xE0\x52\x03\x24\x57\x0B\xE5\x4B\x7A\x6B\x64\x17\x25\xBA\x8E\x7C\xA4\x7A\x69\x7B\xD1\xED\x21\x9A\x89\x29\xAB\xDB\x98\xB2\xBA\x1F\x53\x56\x77\x31\x65\x75\x3F\xA6\xAC\xEE\x62\xCA\xEA\x41\x4C\x59\xDD\x8B\x29\xAB\x29\xA4\xAB\x8B\x4E\x28\x65\x3F\x94\xF1\x3F\x6F\xE6\xC2\xCB\xEA\x5E\x78\x59\x3D\x08\x2F\xAB\x7B\xE1\x65\xF5\x30\xBC\xAC\xEE\x87\x97\xD5\xC3\xF0\xB2\xBA\x1F\x5E\x56\xCF\x84\x97\xD5\x83\xF0\xB2\x9A\xC3\xCB\x6A\x67\xDF\x73\x92\xF8\xA0\xA0\x05\xA5\x87\x7D\xAC\xE0\xFD\x20\xC7\x41\x8D\x01\xDD\x0B\x32\x4B\xC6\xD8\xEB\xEC\xC0\x6B\x3E\x15\x0D\xB4\x83\xFA\x73\x93\xB3\xF6\x96\x9B\xDB\x95\x9E\x95\x9E\x3D\x8F\x47\xF9\xD4\x7A\x7C\xDE\xA4\x05\x76\x96\x9A\x62\x96\x9A\x22\x08\x21\x6E\xA5\x26\xA2\x64\x7F\xF3\x14\x11\x7B\x8A\x4D\x31\x8A\x4D\x31\xB2\x68\x31\x32\xAD\x09\x24\x2C\x22\xC5\x9B\xEB\x3E\xA9\xBB\x4B\xD0\x1B\xD1\x4C\x9D\x4C\xC5\x2C\xCF\x1E\xD3\x14\x37\x9C\xA6\x19\xB2\xB2\x62\x21\x2B\x2B\xE6\x58\x59\x64\x93\x71\xA5\x3D\x23\x2B\x3C\x44\x0C\xD8\x58\xD1\xB1\xB1\xB7\x13\x77\xD7\x71\xB1\xB7\x3B\xAB\x84\x27\xB5\x37\xE6\x49\xF3\xDF\x33\xC3\xFE\x5C\xF8\xEF\x6D\xE6\x5D\x87\xB2\x50\xB7\x23\xB4\x9D\x37\x29\xD6\x76\xBA\x28\x90\xE6\xAD\xCB\x62\xB9\x87\x71\x61\xFF\x9E\xD3\x07\xDA\x21\x9E\x47\xE2\xEF\x98\x7F\x47\x2E\x9A\x31\xFF\xC1\xFE\x0A\x0A\x51\xE1\x2C\x48\x9E\x22\xCD\x39\xA9\x0E\xC2\x39\xD5\x41\x38\xA3\x67\x0D\xF7\xD6\xB3\x2E\xB8\xCB\x13\x3D\x53\x34\x4D\x07\xA7\xE3\xDA\x9C\xCB\x09\x4E\x8B\xB4\x61\xC9\xFD\x75\x32\xA9\x53\xA4\x1F\xAD\x60\x17\xED\x31\x45\x76\x58\xDB\x0F\x94\x22\x06\xA5\x6C\x21\x28\x65\x1D\x28\x45\x0C\x4A\x88\x83\x08\x4A\x11\x64\x0B\x40\x29\x82\x8C\x41\x29\x02\xED\x40\x29\xF2\xCA\x11\xED\x40\x29\x32\x10\x79\x1B\x1F\x92\x1B\x24\x24\x93\xDA\x10\x73\x8F\xD4\x2C\xA5\xC8\x20\x60\xF0\x4F\x41\x19\x9F\x84\x99\x7B\x19\xB9\x30\x6D\x39\x44\x6C\x1D\x6F\xB3\x07\xAB\x18\x0A\xF7\x28\xD8\x4F\x8A\x9C\xF0\xEB\xC4\x8A\x53\x74\xB7\x14\xD3\xE3\x98\x1E\xA9\xEF\x31\x24\xAC\xE4\x2C\xE9\x0C\x5C\x93\x74\x46\x57\xA5\xCB\x5B\x41\xB9\xD9\xB3\x33\x14\x94\x02\x1F\xC7\x90\xA1\x54\x96\x71\x7B\x71\xD7\x4E\x0C\x19\xB7\xB3\x84\xED\x78\x06\xA3\x5A\x72\x79\x31\xAA\x0C\x12\xDF\x0E\x3E\x26\x30\xC6\x76\xC6\x0B\xDA\x19\x73\x3B\xCB\xBD\x76\x92\x6A\xD9\x67\xDE\x70\x23\xA0\x86\xE8\x99\x1B\xF6\x33\x1C\xB4\xE4\x66\xB6\xD2\x6B\x29\xAF\x56\x0C\x3B\xC6\xF9\x31\x50\x4B\xF4\xBC\xFF\xDC\x56\x7B\x2D\x95\xD5\xAA\x61\xFF\xBC\xC1\xE4\xE8\x79\xFF\xD9\x1D\xE8\xB5\xB4\x5A\x1D\x30\xCE\x4F\x70\x30\x3D\x7E\xB1\xFF\xFC\x6E\xEB\xB5\x75\xB0\xBA\xCD\x38\x97\x45\x04\x06\x9A\xA0\xF4\xCE\x8D\x55\xC2\x33\x14\x0E\x1A\xBA\xB6\x24\x08\x6E\x2B\xEB\xB5\x75\x47\x95\x19\xDE\x7D\xC1\x63\xE0\x0A\xE0\xAA\x67\x70\x1B\x1C\x80\x55\x58\x81\x65\x58\x82\xB2\xE1\x7F\xEE\xB8\x12\x20\xB9\x41\xD1\x6B\x10\x2A\x2F\xBD\x47\x0D\x08\x88\x9B\x5E\x72\x89\x11\x8A\xB5\x23\x30\x6C\x20\x72\x92\xF3\x53\x50\x50\xFC\x5A\x34\xCC\x95\x75\xDC\x6D\x01\xE2\x09\x48\xB1\xE5\x42\x29\xB6\xBC\x29\x29\xB6\xBC\x29\x29\xB6\x5C\x28\xC5\x96\x6C\x32\x56\x49\x28\xED\x76\x53\x2D\x51\xEC\x6B\x46\xAF\x08\x4A\x4E\xA5\x53\x27\x88\xB5\x12\x1C\xB6\x8E\x09\x3D\x69\x81\x05\x24\xEC\xD3\x58\x17\xF4\x48\xCB\x5A\xF8\x7D\xA2\xD4\xA0\x94\xA2\x81\x5C\x7F\x69\x09\x71\x45\x71\xCF\x29\x57\x4F\x9B\xFF\x12\x99\xAA\xB1\xCB\x2E\xE0\x92\xCC\x2D\xD9\xE4\x85\x24\xA5\x66\x67\xB8\xA7\xD2\x26\x28\xCD\x25\x88\xE2\xBD\x97\x31\xBF\xFC\x44\xDC\x7F\x19\xF1\x4B\xC4\xBD\xDE\xDB\x90\xDF\x22\x1E\xF5\xDE\x6A\x7E\x8B\x38\xD1\x7B\xAB\xF8\x2D\x81\x77\xEF\xB5\x74\xAF\x11\x52\xFD\x6B\x24\xB8\x23\x8A\x41\x40\xF7\xE0\x28\x33\x8E\xEC\xF6\x59\x10\x4D\x95\xD1\x9C\x20\x6F\xDA\xE4\xCC\x7C\x97\x9E\x3D\x58\x47\x0E\x83\x20\x22\xD7\xFE\x1E\xE2\x30\x15\x73\xAD\x4B\xA6\x45\xBD\xA7\xDE\xA4\xA4\xA3\x0A\xBD\xC7\xDE\x34\xA4\x47\xCF\xFE\x73\x6F\xE0\xD2\xDB\xC9\xB7\x1B\xD1\x99\x08\xD3\x89\x65\xCF\x4C\xEB\x0C\x8F\x3A\xE3\x1C\x1F\x9C\x91\x5C\xCC\xCC\xFC\xF8\xC6\x46\x72\x91\x17\x33\xA3\xCE\x48\x2E\xF6\x9C\x07\xE7\xF8\x5A\x78\x1C\x66\x9C\xE9\xE9\x56\x9A\xC9\xC5\x64\x26\x17\xEF\x6F\x26\xA7\x67\x6C\xE4\x74\x77\x75\xDC\x06\x37\x52\xE5\xA7\x2E\xB6\xA1\x8D\x76\x24\x68\x1F\xDB\x28\xEC\x2E\x00\x7E\x2B\x1C\xDC\x00\xC9\x81\xC5\x25\x4A\xE7\x9D\xCD\x25\x01\x04\xBB\x2E\xA5\x2F\xA1\x9C\xDA\x6C\xD8\xA8\x7B\x86\x8D\xBA\x75\xA5\x19\x1A\x36\xD2\x7D\x4F\xCD\x99\x21\xFB\x17\x57\x0B\x0C\x1B\xD9\xC9\x13\x42\x4E\x3F\xB7\x60\xCF\x34\xC5\xDF\xF5\xB9\xE7\xC2\x6E\xCF\xD4\xAD\x35\x6C\xDC\xD7\xA7\x8F\xB5\x64\xB4\x63\x8A\x76\x6C\xDE\x9F\x4F\x78\x7F\x3E\x41\xF1\xA2\x66\x36\x8D\x5E\x75\x57\xFE\x8A\xC1\x6D\x81\x6A\x29\x60\xD5\xD2\x8C\x52\xE2\x0B\x50\x0B\xA9\x06\x5A\xC8\x19\xBF\xA1\xBD\x27\x3E\x37\xD7\xA2\x73\xFB\x9B\x72\xA2\x77\x5D\x7E\xCC\x6B\x9D\x5B\xE0\xBE\xB2\xFA\xB4\x10\xF0\xB4\x10\xF0\xDF\x43\x08\xD8\xBA\x45\x42\xC0\xD6\x2D\x12\x02\xB6\x6E\x99\x10\xB0\x75\xCB\x84\x80\xAD\x5B\x26\x04\x6C\xDD\x42\x21\x60\xEB\x16\x0A\x01\x5B\xB7\x5A\x08\xD8\xFA\x9B\x22\x04\x14\x0B\x0F\x91\xE2\xA6\x0E\x91\xE2\xA6\x0E\x91\x62\xE1\x21\x52\xB4\x42\x40\x81\xCC\x40\x39\x10\x02\x8A\x9B\x17\x02\x4C\x27\x04\x18\xBF\x4F\x37\x2B\x04\xC8\xA3\xC8\xE1\x7B\x96\xDF\xB5\xEC\x98\x9A\xC2\x31\x35\x20\x4E\x73\xCD\x82\xC2\x83\xDA\xB4\x2F\x10\xF0\xA6\xFB\x2A\xC2\x57\x91\x6D\x15\xD9\xAB\xC2\x2C\xF5\xBE\x55\x54\xAF\x8A\xE3\xBB\xF7\xAD\xA3\x7B\x75\x1C\x73\xBE\x6F\x9D\xB0\x57\xC7\x71\xF0\xFB\xD6\x89\x7A\x75\x3C\x9B\xBF\x6F\xA5\xB8\x5F\xC9\xC9\x02\x37\xAE\x34\x90\xCE\x6A\xD9\xCA\x67\x84\x06\x71\x4F\xE6\x19\x93\x38\x51\x0B\x8E\x95\x2E\x66\xE5\x1D\x22\x16\x09\x08\x17\x71\x91\x10\x33\xF9\x3C\xCB\x3B\x11\x88\x1E\x22\x3B\xE9\x27\xEE\x4F\xE4\xE4\x0D\xA4\xA0\xA7\x05\xA0\x9B\x17\x80\xDE\x5B\x76\xD1\xDA\x3F\x28\x37\xEB\xC8\xA7\x3F\xA0\x74\x79\x3B\xC9\xDD\xF2\x4B\x21\x42\x86\xEA\x2E\x88\xEC\xE1\x7B\x82\x35\x8A\x40\x79\x2D\x68\x8E\xC8\x2F\x47\x5E\xF1\x88\x7C\x16\x44\xFE\x2A\x62\xD5\x55\x2B\xEF\x96\x39\x44\xF6\xB5\xE2\x9E\x20\x21\xDE\xB1\x8A\xD9\xF2\x30\xA1\xDA\xCF\x6A\x8E\xC8\x3B\x38\x45\x9A\xE4\x44\x33\x14\x79\x4C\x50\x8A\x10\x3E\x3A\xF0\x28\xBA\xBF\x4E\x27\x75\x8C\xBB\x9D\xCC\x73\xC6\xF8\xBE\x4A\x0C\x31\x3A\xE9\x84\xEE\x2E\x12\xDC\xEA\x78\xC0\x24\xC7\xAE\x49\xF6\x4A\xD7\x75\x46\x2C\x5F\x63\xB7\x1B\x2C\xA8\x6B\x63\xA9\xA5\xA4\x4E\xF8\x72\x35\xB7\xDB\x67\x1B\x48\x20\xBF\x7F\x42\xFE\x3F\x39\x8A\x7F\x59\x95\x13\x69\x74\xB3\x54\x25\x44\x90\x23\xA3\x19\x11\xE3\x6C\xC7\x53\xFB\x5A\xB1\x75\x4F\x90\x70\x2F\x7B\x0D\x57\x57\x86\x83\xE3\x26\x06\x22\x1E\xAC\xEC\x06\x6B\x83\x75\x5A\x4A\x9B\x34\x15\x05\xBD\x5C\xE3\xFC\x8E\x6B\x9C\xE5\x2B\xDF\x84\x90\x5D\x34\x8A\xA7\x2E\x30\x47\xFC\x7C\xE2\xFF\x83\x7B\x82\x0F\x48\xCF\x85\xC7\x56\x43\x64\x3F\x20\x07\x6C\x78\x74\x28\xF8\x80\xDC\x83\x0F\x97\x0B\xBC\xDE\x7B\x7C\xB8\x73\xEE\xE3\xE0\x42\xCE\x5C\xC5\x5D\x6C\x48\x48\x20\xA1\xEB\x8F\x35\x99\x9F\xAC\xE3\x09\x2E\x56\xEE\x67\x6B\xF6\x98\x2D\xC7\xAE\xDB\x0F\x09\xCD\x4D\xF2\xE3\xD4\x5A\x0E\x06\x97\xFC\xE4\xA4\x36\x0B\xF9\x71\xE3\xF9\x71\x03\xA1\xE3\xC7\x0D\x4F\x05\x5F\x30\x3F\x6E\x0C\x18\xB2\x19\x49\xEC\x6A\x83\x70\x87\x7F\x0A\x88\x27\x95\xA1\x4C\xBA\x75\x42\xE2\x6F\x2D\xF1\xD5\x08\x0A\xC8\xF0\x47\x46\x8B\x10\x4F\x38\x40\xB7\xBF\x0E\x13\x53\x0A\x41\xE6\x3D\x78\x33\x24\xFA\x06\xFF\x53\xE0\x7F\x46\x6B\x32\x28\xAF\x60\xA7\x94\x1A\x90\x5E\x76\x57\x96\xF1\x1E\xCB\x46\x11\xB6\xE3\xBD\xED\x96\xF2\xFE\xEA\xC5\xBC\x7A\xC9\xC2\xD5\x4B\xBA\xD5\xE3\x46\x21\x87\x98\xEC\x96\x62\x48\x16\xAC\x5E\x0C\x09\xAF\x5E\xDC\xAE\x5E\xEC\x2F\x2E\xFD\xEA\xC5\x14\xA3\x3E\x34\x5E\x12\x94\xE7\x1A\x30\x2E\x1F\x35\xA1\x1E\xB5\x77\x66\x0A\x6A\x8B\xE3\xE7\x48\x2B\x20\x9D\x40\x8A\x6B\x37\x66\xE5\x26\xA7\x02\xA4\xDC\x41\x12\xC9\x4B\xDA\x86\x23\x3F\xD7\x9D\x06\x2E\x7C\x30\xDD\x6D\xD3\x92\x4A\x44\xBB\xCC\xFD\xCA\xFD\xC1\x90\xF7\x92\x80\xE1\x9C\x90\x91\x24\x16\x68\x8C\x8D\x8E\x21\xA5\x34\xDF\x71\x2E\x0D\xFE\xE4\x78\xC0\x59\x65\xA0\xA8\x46\x6E\x0E\xD4\x78\xBC\x46\x09\xBD\x64\xF9\x56\x5A\x3E\x1E\x1B\x5B\xC4\x9C\x9C\x50\x6F\x3D\x31\x1F\xD9\x5E\x8A\x4E\x42\xE7\x15\x25\xF2\x97\xD4\x02\x09\x50\x94\xBC\x9B\x72\xF5\xBA\xFD\x32\x4F\x9A\x2C\x0C\x10\x45\xF2\x56\xC7\x0B\xB7\x3A\xEE\xB6\x5A\x32\xA2\xA0\x00\x88\x88\x22\x21\x9E\x3B\xAD\x24\xC4\x7C\x5A\x49\xBF\xD1\xB7\x7B\xFB\xB4\xB0\xA7\xF6\xC9\xF9\x90\x1D\x3B\x85\x83\xB3\x1E\xA2\x50\xFC\xED\x45\xA7\x62\xE7\x07\x98\x8F\xCF\xCC\x2A\x92\xDC\x40\x3E\xAB\x22\x11\xAE\x81\x5A\xBA\x58\x48\x48\xE2\x0F\x5A\xB5\x4E\xAB\x29\xB6\x5C\x88\xA5\x7B\xD8\x4B\x6E\xEF\x96\x87\x72\xC2\x6C\x37\x5F\x08\x61\xA4\x72\x9B\x9E\x06\xD9\xD3\x3A\x45\x87\x82\xB5\x5A\x72\x14\xB1\x68\x4D\x7E\xF9\x5C\x14\x31\x2C\xE0\x01\xA6\x64\xAD\xD0\x5A\x33\x17\x48\x0C\x22\xFB\x41\xE9\x19\x89\x9F\x51\x03\x17\x8E\x70\xC6\xFB\x81\x54\x4D\x3E\xCB\x47\xEE\x4C\x8B\xF1\xCC\xC5\x23\x6F\x6A\xC3\x73\x5B\x6C\x93\x88\x9C\xCC\xEF\x7B\x13\xEF\x61\xB0\x2B\xE4\x68\xD6\x28\x9A\x48\x2F\xA2\x03\x67\x40\xE8\x6A\x91\xBD\x0B\x31\x75\xCE\x02\x8E\xC2\x1B\x45\x40\x48\xC4\xCE\x1B\x9A\x0F\xD2\xDB\x7B\x1C\x92\x34\x64\x42\xCE\xD6\x1F\xBD\xE6\x38\x8E\x84\x2B\x05\x6C\xF2\xED\xA2\x00\xE4\x4F\xA1\x36\x4C\xCD\x6B\xC3\xD4\x53\xA7\x0D\x1B\xF0\x83\x9F\x5A\x79\x5A\x67\xF8\xB4\xCE\xF0\x96\xEB\x0C\x93\x9E\x92\xB0\xD5\x19\x96\xF4\xE8\xD2\x60\x8F\xA1\x24\xEB\x4E\xAF\x2E\xCC\xBC\xBA\x70\x3C\x54\x17\x96\xAC\x38\x5A\xEA\x2B\xAF\x6C\x50\x21\xF3\xBD\x44\x87\x50\xA7\x29\x4C\xBC\xA6\x70\xA9\x53\xCB\xE1\xE3\x32\x94\xD8\x44\x39\xD3\xC4\x58\x1E\x85\x72\xD0\x44\x52\x8D\x5B\x25\xE1\x0A\x2C\x0D\x94\x84\x4B\xB0\x8C\x8D\x2C\xCF\x34\x52\xCA\xA3\xB0\x3C\x68\x24\x67\xCD\x27\xE9\x03\x97\x61\x65\xA0\x1F\xE4\x46\xEB\xD5\x99\x46\x96\xE4\x51\x58\x1D\x34\x52\xB2\xDA\x93\x54\x81\xAB\xD4\x73\xA7\x1A\x3C\x40\x8D\xD6\x2B\x33\x8D\x2C\xCB\xA3\xB0\x32\x68\x64\x95\x75\x9E\xAC\x04\xBC\x0D\x56\x87\x5A\xC1\x55\x38\x80\xCD\x1C\x98\x69\x66\x45\x1E\x85\x03\x83\x66\x0E\xB2\xC2\x93\xF5\x7F\x07\xE0\xB6\xA1\x42\x90\xDB\x1D\x2A\x04\x5D\x7A\x23\x16\x69\x3B\x5D\x60\x6C\xA8\xCB\x4A\x50\x1B\x7E\x15\x5C\xCD\xD8\x6B\x01\x61\x0C\x09\x64\xAD\x2E\xD0\x85\xCA\x5D\x25\xBF\x99\x4E\x0D\x28\x3B\x35\xA0\xFC\xBC\xA9\x01\x97\x17\xAA\x01\x97\x6F\x4A\x0D\xB8\x7C\x53\x6A\xC0\xE5\x85\x6A\xC0\xE5\x56\x0D\xB8\x8C\x6C\xE1\xCA\x40\x0D\xB8\x7C\x03\x35\x60\x39\x54\x03\x2E\xB1\x3E\x89\x43\xD8\x95\xB0\x74\x82\xF9\x81\x9B\xD0\x00\xAE\xEC\xA5\x01\x5C\x9E\xD5\x00\x2E\x3F\x71\x0D\xE0\xF2\x13\xD7\x00\x2E\x3F\x09\x0D\xE0\xF2\x93\xD0\x00\x2E\x3F\x09\x0D\xE0\xF2\x93\xD1\x00\x2E\x3F\xAD\x01\x7C\x5A\x03\xF8\x79\xD1\x00\xBE\xD2\xA7\x2C\x66\xDF\x94\xDC\x05\xE2\x25\x56\x5B\xCF\xF0\xE6\xEA\x06\xBC\x39\x27\x34\x1B\x46\x5B\x63\xA7\xDD\x2E\x24\x0A\xA8\xF2\x2F\x2E\xF6\xE2\xB0\x81\x2A\xFF\x2B\x0D\xE8\x2F\x9F\xD2\x41\xD0\x30\xDC\x2D\xFB\x3E\x2E\x82\x73\xEE\x81\xCC\xC9\x1E\x64\x3B\xA3\x5E\xF7\xC7\x5B\x01\xD3\x8B\x97\xBD\xD8\x74\x63\x33\x2C\xDA\x5D\xD7\x9F\x25\x5B\xDD\x7E\x18\xBB\x3F\xFC\xDB\x3C\xF5\xDB\x67\xE6\x1A\x0D\xCC\x6D\xE6\x85\x44\x27\x56\x6C\xD5\x49\x2B\x2E\x26\xBC\x24\xEC\xC0\x61\x93\x9E\xE8\xD8\x71\xE6\xA4\xCB\x5A\xF0\x09\x24\x5F\x6B\x21\xA8\x45\x56\x4F\x9D\x19\xC6\x93\x11\x32\x63\xEF\xD3\x14\xB1\xFB\x16\x09\x3D\x95\x22\xBA\x43\xE7\x67\xEC\xDE\x2B\x8F\xB6\x5F\xE8\x02\x8F\x32\x4F\xC5\x24\x74\x3B\x89\x36\x74\x93\xA6\x49\x8C\x66\x27\xF1\x04\x42\x37\xA9\x41\xE8\x26\x0A\x52\x45\xFB\x51\x04\xF6\x77\xBF\x7F\xE0\x41\xF2\xD4\x88\xA2\xDD\xA4\x8A\xE1\xA4\x8A\xD9\x49\x2D\x8A\xB3\xB5\xC7\xA4\xE4\x70\x52\xDE\x68\x8A\x20\x11\x14\xE8\xF2\x4F\x3D\x95\x07\xED\xF0\xAB\x4F\xD7\xFF\xF8\x0B\x19\xD7\x9C\xA7\x6F\x0F\xDD\x5A\x3C\x8B\xC8\x5F\x90\x90\xCC\xE3\xDC\xD3\xC8\xF6\x34\xB2\x3D\x65\xC8\x46\xA0\x78\x13\xF8\xB6\x4A\x9E\xA6\xCC\xD5\x80\x2E\x3F\x43\x05\xBE\xFF\x80\xC8\x3D\x12\x5E\xD3\x8B\xAE\x57\xFF\x9D\xE4\xFB\xD5\x1F\x94\x7C\xC1\xFA\x36\x09\x8A\x2E\x68\x08\x3D\xDD\x8D\xE9\x35\xD9\x1C\x91\x3F\x21\xFD\x6D\xE9\x56\x9D\xB6\x28\x9B\xF6\x50\x36\x71\x78\xE9\x2F\xD9\x16\xA0\xEC\xE0\x13\x28\x76\x19\xC7\x11\x27\x84\xB2\x92\xF1\x37\xF4\x28\xEB\x7C\x7F\x3B\x94\x65\xFB\x53\xD0\xF3\x28\x2B\xA7\x9C\xBE\xB7\x45\xC9\xD4\x61\xE6\x1E\x71\xF7\xC5\x0D\xB3\x4C\x0C\x3E\x83\x53\x2B\xCE\xA2\x27\xDD\x55\x02\x0E\xD2\xA3\x67\x64\x61\xF6\xA6\x12\x7C\x1E\x9B\x0E\x08\x5C\xDB\x9C\x4D\x36\x07\xC1\x40\x20\x40\x2D\xB8\xA8\xEC\xA3\xE7\x8D\x67\x02\x31\x24\x67\x9D\xDB\xF3\xDE\x13\x22\x77\x6E\x57\x6A\xC1\xBC\x92\xF9\x79\x25\x9F\xF3\xBC\x24\xCF\x6C\x80\xB1\xE9\x10\x5D\x87\x40\x87\x50\x90\x40\x7C\x96\xCD\x4B\x35\x85\x86\x76\xA1\x03\xD9\xCE\xD4\x2F\x8F\xE8\x20\x4C\x71\xE2\xB9\x39\xEE\x12\xDF\x57\xA9\x63\xA0\x62\x14\x24\xA7\x55\xE6\x6E\xC2\x09\x7E\x39\xB4\x50\xC7\x74\x12\x10\xCD\x35\xE3\xAE\xCE\xFD\x3D\xBF\x9C\xA9\xF1\x85\x0D\x65\xE5\x96\x8F\xAD\xF9\x39\x4E\x04\x62\xC8\xCE\xFE\x77\x85\x2E\x3F\x25\xDD\x41\xC6\x8D\x76\x73\xE1\x7E\x92\xFD\x47\x76\xD6\xF2\x25\x02\x42\x87\xDE\x64\xB8\x73\xD7\x3B\x64\x55\x92\x6C\x12\x50\x36\x35\xDD\xEC\x1A\x48\xED\xF6\xD9\x2A\x67\x7B\x93\x82\x2A\x55\x23\xA7\xDA\x4E\xDD\x35\x90\x7D\x9B\x64\xB3\x8A\xB7\xC9\x06\x24\x5D\xD7\xB3\x51\x85\x00\x61\x77\x34\x84\x6D\x5A\x6D\xAA\xB4\xBD\x8E\x22\x10\xD9\x90\xB8\x3A\x0A\x72\x5F\xA7\x1A\xBB\xF6\x14\x16\x81\x31\x2D\x58\xAF\x0D\x61\xD8\x9A\x81\xF0\x62\x4C\xBA\xB7\xC6\x3B\x84\x53\x4C\xC2\x22\x80\x04\x04\x65\x98\x57\xF4\x17\x4A\x18\x9F\xE4\x1B\x13\x82\x82\xF1\x1E\x50\x40\x72\xF5\x78\x3F\xAD\xC3\x98\xFD\x70\x97\x16\xFA\xE1\x2E\x75\x5A\x87\xB1\x81\x31\x48\x18\x43\x79\xFF\xA4\x1E\x93\x12\xB0\x33\x3D\x60\xAD\xC3\x18\x96\xD8\x13\x77\xDC\x5E\xD9\x8F\x99\xD5\x1B\xB7\x57\xF6\xD8\xCC\xFC\x12\xF0\x64\x69\x05\x2C\xB9\x1C\x33\xCC\xB9\xFF\x70\x74\x31\x3C\x47\xDE\x4E\x6D\xB0\x7F\x14\x8C\xED\x4E\x72\xBF\x2B\x6B\x03\x2E\xAD\xDA\x2A\xE3\xF2\x07\xC8\xD8\xA1\xC4\xD5\x99\xDB\xB7\xDE\xC9\xBA\xCA\x07\x6B\xCE\xE7\x2A\x1F\xA4\xD0\x1C\x91\x77\x90\x2F\x4F\xD2\x40\x02\x0A\xC2\xF2\xAF\x2F\xCE\x45\x69\x43\x04\x48\x5C\x40\x79\x8E\x22\x63\x35\xC7\x57\x65\xE4\xC0\x6F\xFE\xF2\xF4\x8E\x45\x97\xA7\x09\x1B\xED\x97\xD3\x42\x52\x57\xB3\x65\x18\xC0\x22\x0A\xCD\x3C\x8F\x1C\xDC\x00\x7D\xA6\x08\x2E\x14\x4F\x6D\x4D\xDE\x51\x27\x3E\xF8\x1A\x23\xF8\x75\x31\xE5\x61\x24\x7B\x8D\x81\x8A\xB9\x0B\xDC\xF9\x51\x50\x6E\xB5\xF5\xAA\x64\x05\x80\x34\x50\x40\xDE\xC0\x68\x93\x2C\x11\xBC\x55\xC2\x98\xFB\xF7\x6A\x01\x17\x13\x60\x68\x95\xA0\xD8\x2A\xC1\x45\x05\xE8\x59\x25\x8C\x9F\x3C\x45\x9B\x49\x33\x25\x16\x6A\xD0\xC4\x8C\x06\x8D\x5A\x1B\x53\x8A\xA9\x31\xA5\x98\xF2\xA2\x78\x3F\xC5\x14\x69\xD0\x04\xED\x7D\xA7\x41\xE3\x67\xAF\x41\x13\x30\x26\x12\xC7\x15\x3F\x25\xA6\x44\x8F\x98\x01\x62\xE5\x1A\x01\x4C\x79\xDC\xEB\x55\x56\x6B\xE5\x2E\xBF\xCE\x8C\xA2\x40\x4A\x29\x03\x69\x28\x40\x23\x25\x69\x60\x05\x9C\x60\x33\x14\x22\x45\x02\x0C\x88\xF2\x37\x08\x92\x71\xB8\xE6\xFE\x49\xAD\x7C\x80\x79\x05\x79\x53\xE5\x38\x8C\xDC\x05\x87\x8E\x0E\x05\x6F\x93\x3D\xA8\xFB\x89\xB9\xAC\x78\x5C\xA4\x85\x3B\x47\xEC\xE6\xB7\x5C\x00\xA2\x8B\x63\x5F\x23\x7B\x4D\x7B\xFE\xF5\xC7\x9E\xD5\x59\x02\xBE\x23\x6E\x05\x46\xE4\xB4\x8F\xC8\x67\xF1\x9F\x1C\x42\x90\xCF\x96\xC9\xDD\x6A\x8D\x7F\xFD\xFF\xEC\xFD\x0B\xB0\x66\xD7\x55\x1F\x88\x9F\xFD\x38\xAF\xEF\x9C\xF3\x7D\xA7\x5B\x57\xD2\x95\xBE\xB6\xBD\xCE\x29\xFD\xAB\x6E\xA7\xDC\xE5\x4E\xFD\xC5\x6D\x95\xA3\x89\x7B\xF7\xA4\x1F\x57\x6D\xAB\x95\x4C\x31\x05\x55\x50\xE5\x9A\x78\x18\x7C\x6E\x23\xEB\xB6\xAE\x3B\x66\x68\xDF\x7B\x51\x37\xA2\x67\x12\x67\x04\xC8\xC4\x80\x8D\x05\x36\xD3\xB2\x71\x63\x63\x60\xCA\x21\x06\xDA\x06\x12\x01\x1E\x22\x6C\x19\x3C\x60\x40\x33\xC3\xC3\x06\x27\x38\xC1\x49\x4C\x00\xF7\xD4\x7A\xEC\xF3\xF8\xBE\xEF\xF6\xED\x46\x32\x83\x33\xB6\x4B\x7D\xBF\xF3\xDA\x8F\xB5\xD7\x5E\x7B\xED\xB5\xD7\xFA\xAD\xF2\x88\x39\x0C\x21\x18\xBA\x57\xF2\xAF\xF2\x88\x59\xC6\xA7\x48\xBA\x97\x6E\xB8\x6B\xC1\x39\x5C\x6E\x42\xD2\x30\xDA\x1B\x49\x87\xB9\xCD\x5B\xB9\xA0\x8A\x38\xD5\xA2\xCA\xCA\x8F\x10\x59\xA2\xDE\x0E\xB3\x5B\x4D\xBA\xC3\x4A\xC2\x4D\xA3\xC8\xBC\xFE\x57\x31\x39\x33\xC7\x6E\xC7\xB2\x6E\x1B\xBA\xFB\x9A\xF2\x03\xA2\xB3\x68\xF6\xF5\xC3\x15\x64\xBD\x36\xCE\x36\xAB\xFA\x3E\x97\x54\x1A\x12\x77\xED\x8B\x9F\xF8\x65\x3A\xF5\x34\xEE\xC3\xFE\x67\x88\x9B\x88\xA4\x4E\xDD\x64\xC3\x3D\x8B\xAD\xBE\x16\xF1\xF2\x96\xD7\xA3\x55\xFD\x0B\x72\x51\xD6\xD9\xAA\x7E\x46\x2E\x96\x6A\xBD\xAA\x3F\x26\x17\xCB\xAB\xFA\x59\xF9\x79\xA0\x4E\x56\xF5\x73\x11\x84\x9D\xE2\x6F\x21\x24\xC5\x1F\xB7\x2E\xA8\xF8\x13\xAC\x17\x0E\xC8\xAA\xFE\xB0\xED\xBD\xF8\xA4\xE6\x17\xFF\x89\xE6\x17\x2F\x6B\x7A\xF1\x29\xDC\x13\xBC\x1D\xEF\xE9\x55\x7D\x09\xDF\x11\x2F\xCC\x2F\x28\xF9\xB8\x3C\xA2\x3F\xA7\x20\x24\x3F\xCC\xCF\x28\x17\x40\xB6\x5E\x19\xFA\xF6\x12\x7E\xFB\x17\x8A\xB1\x0C\x5C\xD2\x9C\x2C\x02\xF7\xEF\x7E\x70\x00\x04\x6B\xB8\x83\xEE\xD9\x80\xBB\x8D\xC3\xFF\x5A\x08\xDB\xE3\xA9\x3A\x5F\xD5\xAF\x63\xD9\xD0\x8A\x25\x46\x1D\xCA\x5C\xFA\x48\x5D\xB8\x14\x8A\x29\xFE\xA1\x4C\x50\xE4\x19\xEE\xF1\x1F\xBF\x81\xF0\x1F\x19\x9B\x44\x0B\xF0\x88\x94\x43\x98\x91\x33\xCE\xE4\x0F\x0B\x4E\x64\x0A\xC5\xBA\x3F\xF1\x30\x67\x9B\xF9\x80\xB0\x45\x00\x35\xDB\x6E\x6B\x08\x50\x43\x9E\x38\x6F\x76\x17\x86\x00\x35\xB3\x9E\x38\xF4\xED\xB7\xB9\xFF\x71\x08\x50\x33\xEB\x89\x43\x6F\x7D\xAB\x7B\x13\x74\x85\xB8\x7F\xE4\xCE\x0F\x91\x69\x44\x82\xFB\xCE\xDF\x43\x94\xCA\x91\xF4\xBB\x78\x6B\xC5\x37\xEF\xAD\x95\x40\x8C\xFC\x57\xCC\xCA\x45\xBC\x5B\x15\x9D\x5C\x4C\x70\x1E\x3C\x13\x90\x2B\x1B\x98\xB3\x84\x45\x95\x38\x64\x7B\xFD\xC0\xB4\x4E\xA0\x60\x31\x19\x77\x62\x32\x81\x02\xC5\x64\x0C\x89\xC7\xFD\x23\x31\xC9\xD7\x5E\x4C\xC6\x0C\x58\xF2\x7C\xD0\x38\x0D\x86\x17\xF1\xBB\x21\x76\xDF\xBA\xAA\x2D\xEE\x08\x9D\xA2\x57\x50\x36\xDA\x29\x4E\x80\x1C\x62\xAE\xC4\x27\xA6\x19\xC2\xD9\x26\x3D\x38\xDB\x9B\x83\x1B\x32\x0B\xE1\x86\xCC\x4D\xC1\x0D\x99\x9B\x82\x1B\x32\x0B\xE1\x86\x78\xB7\xE7\xD2\xB3\x60\x3C\x9C\x6D\xE8\x76\xE2\x56\xD2\xB8\x4F\x45\xDD\xEF\x64\x55\xBF\x2D\xE6\xBF\x9F\xC7\x99\xF4\xE9\xC7\xDF\xF7\x53\x5B\xAB\xFA\x79\x94\x06\xEF\xFE\xB3\x4F\xFC\xF2\x47\xFE\xF8\xDA\x27\xBF\xE9\x88\xBE\x14\xFB\xEB\x27\x7F\xEE\xBD\x77\x1D\x31\x9F\x62\x17\xA6\x70\x45\xDF\x57\x1B\xB7\x85\x6C\xF2\x54\x07\x87\xC3\xC0\x3E\x21\xA7\xF9\xD6\x4D\x55\x60\xC1\x41\x53\x19\x17\x50\x5E\x76\x94\x6E\xB5\xA1\x00\x0C\xA7\xBF\x05\x67\x2B\xEA\x09\x94\x0F\xBC\x70\xF9\x03\xD5\x18\xF2\xAA\x80\x71\x11\x64\xE6\x28\x84\xEE\x69\xDB\xE0\xD8\x70\xCB\x41\x97\x8F\x5F\x64\x89\x49\x11\xAB\x6A\xD3\x85\x4D\x5D\xAC\xEA\x1D\x7C\xE1\x5A\x44\xEE\xA1\xD8\xB0\x67\xA2\x7A\x8C\x8C\x82\xBD\xBC\xAC\x1B\x7E\x08\x63\x28\xD6\x41\x97\xFF\x96\xD9\x21\x74\x01\xBE\xBA\x13\xAF\xAF\xB2\x08\xBB\xAC\x69\x71\xE3\x2F\x72\xA9\x9B\xA8\x07\xBA\xFC\x92\xD0\x8C\x4B\xE3\x7B\xBF\x23\xF7\xB8\x8D\xA1\xFB\x8C\x6A\x66\xDB\x89\xD2\xE9\x69\x6E\x3F\x97\xC6\xDF\xE3\x3B\xA1\x7B\x3A\x6E\xCA\x3F\xEB\x76\xAD\xA7\xF0\x9F\xA7\xE3\xBA\x38\xB7\xAA\x4F\xE9\xA3\x50\xB8\x8F\x80\x98\x2C\x68\x4A\x16\xBB\x4C\xC9\x22\xA3\x97\x17\x6C\xBE\xFA\x8F\x69\xD7\x15\x0F\x76\x5D\x79\x6B\xB0\x7A\x3F\x36\x91\x66\x4D\x4E\x16\xAB\xF7\xC7\xB3\x26\xAB\xF7\xC7\x64\xD6\x1C\x6C\xBC\xA4\x78\xBC\x4D\x53\x88\x37\x5E\x31\x14\xB3\x1B\xC9\x53\x6C\xAE\x38\x95\xB5\xAA\x98\xDB\x6E\xC4\x1D\x9D\x3C\x9B\xE3\x1E\xDA\x5D\xCC\xDB\x36\x48\x9D\x64\x23\xA3\x96\x59\xC8\x60\xD4\xAC\x8B\xDA\x11\x1E\x0A\x9E\xB6\x5E\xED\x08\x57\xF4\x87\xED\x9C\xDA\x41\xAF\x0C\x54\x4D\x1A\x82\x39\xC5\x83\x4B\xBB\xAC\x7B\xA5\xBD\x7D\x5E\x89\xA1\x57\x66\x4A\xBB\xBC\x48\x8D\x89\x69\x61\x6F\x99\x03\xA7\x15\x34\xDD\x5D\x62\x23\xA7\x19\xE1\x88\x6F\x5D\x62\x4E\x39\xF6\xF3\x5F\xFC\xC4\x2F\xFF\xDA\x7F\xA2\x19\x77\x3F\x93\x6D\xC5\x19\x54\x04\x64\x2D\xEC\xBE\xF0\xBC\x18\xAC\xEA\x6B\x96\x9E\x76\x55\xD0\x42\x3D\xC7\x55\xD7\x6C\x6D\xCE\xA1\x06\x70\x8A\x3A\x30\xE4\xAB\x5B\x16\xF5\x83\xC7\x0B\xF8\xAA\x73\x13\x7C\x4A\x7B\xBE\x62\x3F\xC1\xA7\xF4\x2C\x5F\x3D\xA5\x77\xE1\x2B\x3D\xCF\x57\x73\xFE\xFA\xC4\x57\xD4\x2D\xD1\x45\x65\x9B\x14\xB7\xB0\xD6\x56\x72\xFF\x31\xA8\x1D\x35\xEB\xDA\x3B\x2E\xED\x50\x7E\xCE\xD0\x3D\xFE\xCA\x55\x4D\x9C\x73\xE9\x87\x70\x21\xFF\x67\xEF\xBF\x84\x54\x2B\xBF\x9B\x04\x84\xFB\xDD\xC7\x2E\xED\x04\xEE\x7E\xA7\xDA\x37\xCA\xEF\xE1\x27\xA3\xEE\xCE\x93\x7C\x67\x47\x13\x60\x52\x07\xAB\xE6\x2E\xFF\x3E\x16\x16\x33\x02\x23\xAB\xD8\xDD\x88\xC9\x88\xF6\x46\xEC\x8B\xF3\x23\xA6\x45\x0E\xE8\x2F\xF7\x78\x25\x0B\xC6\x2B\xF9\xB2\x8D\x17\xC9\x01\x4F\x0A\xDC\xFB\x3E\x45\x94\xBA\x31\x2D\x12\x1E\x66\x48\x98\x16\x74\x00\xB3\x0B\x2D\xA2\x1B\xD3\x22\x9A\xA5\x45\xB4\x3B\x2D\x22\xDA\xEB\xDE\x98\x16\xD1\x90\x16\x52\x3C\xDE\xA6\x50\x1C\xA6\x45\x34\xA4\x85\x50\x82\x37\x02\xCC\x0B\xC2\x27\xDB\x27\x0A\x7D\x43\x3E\x59\x30\xB3\xFF\xBF\xC2\x27\x9F\xFF\xBD\x5B\xE6\x13\xF5\x5F\x08\x9F\xE0\xC6\x87\xC4\x9C\x17\x72\xC7\x0B\xC5\xB6\xFF\x08\x02\xF6\x82\x88\x3A\xDF\xF0\x50\x84\x61\xCF\x95\x7E\x3E\x21\xB7\x6F\x40\xEB\x48\xBF\x60\x61\x8C\x05\x04\x1C\xE2\x61\xE1\x71\xAF\xF0\x78\x51\xE1\xF1\xB0\xF0\x78\x7E\x9D\x14\x27\x7B\xEA\xE2\xCF\xC6\x1B\xB4\x8D\x8E\x32\x2F\xB9\x19\xBF\x51\xE3\x13\x02\x70\xE4\x97\xD9\x2B\x1C\x3B\xC7\x9E\xC4\x01\xEF\x0E\x5E\x78\x57\x21\xE6\x00\x20\xD5\x6D\x37\x5E\x58\xF7\xFE\xCA\x51\x4E\x83\xC7\x37\x3C\xFD\xBD\x66\x87\xC7\xBF\x38\x21\x86\x6C\x77\xCD\xEE\x72\xCC\xC8\x1E\x64\x3D\x46\x1B\x1C\x3A\x7A\xCB\xD8\x82\xB6\xAB\x1B\xB7\x5D\xCD\xB6\x7D\xC1\xA1\x85\x6F\x7B\xEF\xD4\x62\xD7\xB6\x2F\xB2\xF3\xC7\xF3\x76\xFE\x5E\x4F\x50\x60\x7C\x20\xF6\x96\xA3\xB7\xDC\xA6\xD2\xD6\x72\xB4\x30\x86\x74\xD6\x14\xAB\x25\xCA\x27\x66\xA3\x90\xDA\xA8\x38\x4E\xB4\x4E\xC9\x6E\x5F\x8F\x1E\xF0\xC1\x92\x79\x67\xB6\x5D\x1C\xA5\xA3\x25\xFE\x27\x81\xF8\xD1\xAA\x80\x11\xA4\x90\x4D\x7D\x68\x29\x7D\xB2\xE0\x74\xA1\x3D\xE1\x24\x74\x63\x48\xD9\x85\xB0\x70\x81\x0B\xE4\x9C\x81\x26\xBD\x96\x93\x06\x5B\xE7\x0D\x37\x8C\x6D\x47\xCD\x7A\xCD\x1E\x81\x5C\x03\x90\x90\xC2\x52\x73\x48\xDA\x40\xCD\xE3\x85\x91\xA2\xBD\x57\x44\x5B\x27\xC5\xAC\x0C\x6B\x73\x49\x45\xBE\x84\x34\x03\x7B\x25\xF3\x09\x44\xCE\x86\x71\x2C\xBE\x4E\x18\x94\x88\x22\x45\x73\x36\x64\x6B\x48\xBA\x93\x8C\xDC\x64\x7C\xD7\xB8\xA0\xD7\x1C\x3A\xF6\x48\xF8\xB4\x22\x87\x84\x1A\x91\x53\xB4\x2A\xDB\xE7\x47\xEB\xF5\xB8\x3D\xA1\x80\x1C\x02\xAA\x19\x12\x39\x90\x30\xFE\xC6\xE8\x54\x2F\xA6\x23\xD9\x65\xDE\xC9\x39\xDD\x9E\xD6\x8B\x9B\x0B\x78\x48\x70\x74\xB1\x8F\x63\x32\x56\x2C\x0A\x77\x48\x7C\xB8\x43\xD2\x9E\x50\x24\x7C\x88\x9F\xB4\x27\x14\x58\x0C\x75\xDB\xC8\xD9\x23\xF5\x93\x34\xC6\xA4\x1A\x41\x42\xE4\xE8\x9F\xBF\x2C\xF0\xFA\x54\x37\xEF\xF5\x39\x62\x9B\xF5\x78\xA1\xCD\x7A\xDC\xF5\x8E\x4A\x4B\x60\x04\x23\x48\x4E\x4D\xEB\x11\x8C\x17\x9C\xC0\x8C\x60\xCC\x27\x30\xA3\xB6\x7F\x23\x66\xAB\x51\xDB\xBF\x51\x06\xA3\xE3\x85\x91\xFE\x8D\xF8\xEC\x65\xC4\x2A\x31\x79\xFF\x8F\xB0\xC3\x38\xD6\x63\xBE\x39\xC2\xAB\x25\x0E\x6D\x94\x83\x98\xDA\x3C\x5C\x8F\x29\xA9\x3E\x6D\x52\x61\x0C\xE6\xEC\x3A\x9F\x4D\xCA\xEB\x78\x6B\x70\x68\x63\xF8\x5D\xD3\x1E\xF7\x8D\xE4\xF8\xA6\x03\x35\xCD\xDB\x13\x1C\xAC\x1E\x46\x7C\x82\xE3\xB6\x4F\x16\xCA\xED\xD8\x2A\xF3\x41\x5C\x79\x77\x20\xD3\x22\x23\xE9\xF6\x58\x46\xF3\x39\x45\xD2\x22\x9D\x77\x87\x14\xA3\x99\xD0\xC9\x5D\x0F\x29\x72\xBF\x33\xCE\xBB\x43\x8A\xD1\x8B\x1B\x3A\x69\x16\xB2\xB3\x99\x09\x9D\x1C\x51\xE8\xE4\x88\x42\x27\xCD\x82\xD0\x49\x73\x33\xA1\x93\x23\xEF\x9D\xBC\x38\x76\x71\x34\x63\x20\x1D\x7D\xE5\xC5\x2E\x7A\xEF\xEE\x51\x1B\xC2\x88\x7A\xE7\x20\xF9\x44\x21\xF9\xC6\x45\x70\x42\xE2\x14\xFE\x47\x50\x6A\xC9\xE0\xE8\x8A\xDD\xBF\x13\xE6\x8A\x04\xB9\x22\x61\xF7\xEF\xC0\xBB\x7F\x07\xC4\x15\x49\x8B\xA6\xA6\x76\xE1\x0A\x75\x2B\x5C\x41\x62\x20\x58\x28\x06\x82\x19\xAE\x30\xC4\x15\x86\xB8\x22\x98\x3B\xBA\xD2\x10\xF0\xD1\x95\x9E\x39\xBA\xD2\x83\xA3\x2B\x43\xAB\xB6\xF8\x80\xF7\x8F\xB0\xC4\xF9\x43\x4E\xE8\x21\xF3\x6A\xDE\x97\x07\x4D\xC0\xB4\xBE\x06\xEF\xD7\xFD\x28\xC6\xC8\xBD\x7F\x0E\x4D\xE0\xFD\x37\x42\x13\x18\xE8\xEA\x0B\x55\xA8\x5B\x1E\xA1\xC1\xE3\x05\x2A\x54\xD0\x6F\xBB\xA8\x50\xC1\x8D\xDA\xBE\x48\x85\xD2\x19\xEB\x4E\x94\xAF\xB9\xEB\x82\x0B\xAA\x34\x13\xCD\xC0\x1F\x27\xEB\xBD\x8E\x93\xFB\x5A\xF5\xA2\x83\xDC\xC8\x7D\xA0\x8D\xC4\xBD\x72\xBB\x89\xB7\xD5\x56\x9B\x2B\xEF\x09\xFD\x15\xE5\xFD\xD9\xF9\x7C\x32\x26\x95\xF7\x5E\x27\x68\x8D\x68\x17\x05\x30\x42\x12\x0C\x15\x3D\x54\xED\x34\x3B\x35\xA9\x36\x08\x33\xD8\xE5\x2C\x3A\x10\xDF\x96\x08\xB5\xC8\x68\x37\x27\x9B\xF6\x2D\xD4\x58\xA3\xCE\xC9\x66\x41\x2A\x48\xD2\xBF\xCB\x7E\x2A\xC8\xD0\x95\xB3\xDA\x77\xB9\x0B\xD7\x9B\x79\xFF\xC4\x68\xE8\xC2\xD5\x61\x3D\xDE\x72\xCC\xE9\xE0\xF1\x0D\x83\x8F\xCB\x61\xF0\xF1\x4D\x37\x3F\x5A\xD0\xFC\x5D\x1D\x5F\x77\x71\x1F\xED\xC6\x8C\xDA\xD9\x8F\x60\x58\xC0\x03\xAA\xCF\x03\x0B\x92\x97\xE0\x33\xE2\x05\x75\x13\x3C\xA0\x20\xFA\x8A\xE0\x01\xFD\x5F\x0C\x0F\x50\xB2\x53\xC5\xE9\x75\x38\x89\xCA\x03\x75\xC4\xC1\xD5\x75\x42\x51\xD7\xD1\xB4\x1E\x11\x22\x6B\x53\x47\xA4\xF4\x10\xE1\x5B\xD5\xB6\xE6\xB8\xE4\xAF\xD1\x01\x8C\xF8\xEF\x56\x9D\x1F\xA3\xAC\x17\x3B\x57\x6A\x4A\xC1\x49\x69\x30\x38\xFB\x45\x7E\xA5\x69\x85\x19\x52\xE7\xB4\x88\x1A\xE3\x34\x9B\x8F\xCB\x86\x8A\xA1\x32\x21\x29\xAF\x3F\xE6\x9D\x99\xD8\x22\x82\x2F\xB8\x80\x98\xAD\x7C\x8A\x97\x5D\x92\x9E\x6A\xF8\xB5\xA1\xEF\x51\xB4\x3D\xED\x0F\xEC\xE8\x61\x2A\x20\x2E\x4B\x90\x80\x29\xDF\xF3\x58\xDF\x59\x7A\x77\xB6\xBC\x39\x37\x99\x9B\xC3\x69\x11\x8B\x96\x01\x03\x11\xB9\xC9\x2C\xC2\x69\x31\x1E\xA7\xC5\xB4\x11\xE4\x86\x95\x50\xD3\x46\x90\xE3\x66\x83\xB6\x1C\xB4\xBF\xA0\x52\xC9\xFD\x22\x22\x4F\x2C\x67\x36\x38\x6E\x9C\xE8\xBC\xDD\x38\xB3\x46\x0B\xD8\x85\x8D\x6A\xE2\x02\x3A\xA9\x1C\x9B\xA3\x38\x32\x4D\x9D\x30\x6D\x4C\x53\xC7\x2B\x3A\xE0\xC3\x4A\xDB\x40\xEC\x6C\xD3\x5E\x27\x78\x9D\x74\xD7\x39\x5E\xE7\xFE\xDA\xB8\x12\x15\xFB\x09\x8C\x9D\x6D\xEA\xF1\x49\x32\x6D\x15\x28\x01\xC6\x4E\x6F\x56\x46\x6A\xF2\xF5\xB4\x5F\x59\xFC\xAA\x70\xDB\x4D\x5D\x14\x38\xA9\x3A\x1F\x93\xC4\xED\x70\x3A\x10\x67\xCB\x9F\x7D\x4C\x00\x76\xB4\x3E\xCA\x01\x88\x11\x8F\x77\xC6\x43\x89\xDC\x39\x33\xDE\x3A\x83\xAC\x37\xD8\x11\xBD\xC3\xE3\xED\x79\x6F\x27\xC1\x65\xBF\xE3\x35\x83\xDB\x55\x97\x37\xE5\xCE\xC5\x16\x55\x88\x02\xEE\x63\x7A\xA9\x92\xAF\x76\x76\x0E\x9F\xA6\xAA\xE3\xEE\x8B\x6F\xEF\xBE\x18\xDC\x7F\xEC\xA2\x87\x23\x2A\x38\x0A\xD0\xFA\x28\x40\x11\x20\x7C\x24\x48\xFA\x47\xB0\xC8\x08\xD7\x4A\x23\xC8\x56\x51\x0D\x62\xFF\x97\x68\x55\x97\x04\x68\xFB\xB2\x0D\x89\x68\x4B\x71\xBF\x30\x5A\xD1\xB6\xE9\x65\xA9\x78\x64\x83\xE4\x2E\x09\xE1\x47\x9D\xDA\x38\x57\x1B\x9C\xFE\x86\xF3\xC3\x57\x31\x8A\x51\xFC\xEF\xD4\x94\xEC\x13\x7A\x13\xB2\xDE\x06\x2E\x62\x55\x1D\x79\xA7\x8A\x78\x03\xA7\xFC\x06\x4E\x91\xAA\x1E\xD1\x89\x36\xED\xF3\x79\x2F\x74\xEB\x78\x47\x0B\x20\x18\xF6\xDC\xC2\x45\xBC\x67\x8F\x20\xA2\x3D\x7B\xB4\x60\x0B\x17\xF9\x2D\x5C\x34\x13\xA9\x19\x75\x91\x9A\x1E\x67\x7A\xC1\x8A\x16\xF0\x8A\x36\xDC\xC0\x45\xB3\x1B\xB8\xBF\xF1\x4E\x12\x94\xA6\xB3\x73\x92\x40\x09\xF8\x44\xAB\xA8\xBE\x5B\x6B\xB5\xA5\xB7\x51\xC3\x47\xC9\xAD\x48\x6C\xDB\x56\x6C\x87\x92\xE9\xDC\x8B\x6D\x7B\x85\xB3\x24\xB7\x6F\xDB\xAD\xDA\x5E\xA9\x23\x08\x9B\x3A\x84\x68\xED\xEA\x31\x78\x1C\xEC\x05\x32\x3A\x3E\x72\x11\xC2\xAB\x97\xDA\xA2\x38\x89\x68\x97\x6F\x19\xBF\x60\xF5\xDB\x38\xC5\x76\xFD\x10\x8B\xB5\xBE\x11\x57\x9A\xDA\x40\x38\x5B\xA4\xE9\x15\x19\xFA\xE4\xCC\x6D\x4A\x25\x6E\xAE\x2F\x05\x1B\x77\x0C\x2E\x83\xBD\x52\x2B\x88\x1A\xDC\x07\xAE\x5D\xBD\x80\x5B\xB5\xA6\x56\xA0\xD7\xAE\x5E\xC0\xB2\x55\xAF\x44\x49\xF7\xAC\x7D\x89\x8A\x53\xD5\xFD\xAA\x35\x7A\x5B\x6D\x29\xAF\xCE\x1F\xEE\x69\xF3\xA0\x9D\x39\x5D\x04\xEE\x13\xEF\x1A\xC4\xDD\x68\x30\x67\x8A\xC0\xBD\xF7\x2D\xFD\xBB\xB2\xC4\x6D\x3F\x0A\x9A\x5D\xB2\x74\x83\x8A\xC3\x09\x86\x43\x6A\x9C\x3A\xEF\x54\x73\xF5\x98\xD9\xAA\x63\x4A\xA3\xAB\xC1\x42\x7C\xA5\x4E\x68\x8E\xD5\x11\xB2\xFB\xE9\x29\x52\xD2\x6F\x10\xBC\x50\xB4\x9C\xD6\x58\xB0\xFB\x5C\x59\x3E\x71\xB1\x73\xDF\xA3\xAD\x11\x45\x80\x7E\xD7\xC5\x9E\xE0\xF3\xDF\x46\x9C\xFF\x68\x28\x72\xA4\x34\xFE\x1A\xA2\xEE\xEB\x45\x47\x0A\xA1\x3B\xDC\x26\xF4\x9D\xE0\xE5\xA5\xB7\xB6\xE7\xCE\x9F\xB0\x83\x73\xE7\xCB\xC3\x73\xE7\xCF\x7D\xFB\xA5\x9D\xC0\x1D\xBC\x99\x73\xE7\xF2\xAD\x8F\xF9\xDA\xA4\xF8\x52\x8A\x5F\x9A\x2B\x1E\x75\xA4\x17\xA3\x06\x80\xD0\x7D\xCE\xAE\xEA\x7B\xE6\x6B\x80\xB6\x86\x8F\xDE\x7A\x0D\x1F\x8C\x3B\x3F\xCF\x39\x84\x9F\x8E\x3F\xAC\x70\x08\x43\x08\x44\xCC\x67\xBF\xDD\xE3\x33\x88\xDC\xF5\xEB\xE6\xB4\xB3\x9B\x75\x8C\xEB\xF2\x39\xB2\xD4\x97\xE7\x69\x81\xB2\x6B\xD3\x3A\x76\x3B\x7A\x8D\x40\x72\x20\x75\xF6\x1C\xD0\x5B\xE7\x21\xC1\x27\x65\xFB\x40\xF3\x03\x2B\x0F\x2C\x3E\xA0\x3B\x5A\xEE\xA8\xD3\x35\x2E\xFC\x90\x9C\x6F\x70\x1D\x88\x85\xFB\x68\xE9\x74\x0A\x62\xAA\x1E\xA2\x93\x9B\x04\x23\xE7\x9F\xEE\xEC\x94\xBD\xE7\xD1\x49\x5C\x46\xAA\x38\xB7\x19\x3E\x5A\xA6\x85\xE7\xDB\x77\x76\x96\xD7\x0A\x43\xCB\x29\xE1\x8A\x24\xED\xDA\x6A\x73\x43\x01\x2B\xE7\xEB\xC4\x05\x90\x40\xDC\x60\x21\x6B\x53\x81\x59\x88\xD9\x74\x19\x54\x09\x99\x53\xBE\xAC\x2C\x47\xCB\x96\x9F\x2E\x23\x9A\x6A\x09\x27\x98\x4B\x5C\x50\xA5\xCE\x23\x5D\xC5\x7D\x25\x25\x18\xDC\xC0\x49\x96\xF1\xAB\x38\xAD\x46\xC0\xFE\x96\xB5\xE6\x8C\xBB\x55\x0E\x64\x65\x91\xAB\x14\xA2\xDE\x95\xEE\xF9\xE9\x92\xBE\x93\x43\xEC\x82\xF2\x83\xFE\x84\x23\x45\x32\x75\x97\x1A\x07\x8E\x2F\xF9\x24\x62\xD0\x2C\x4D\xD0\x8A\x8A\xE0\xEC\xEB\x80\x6B\xA8\xE9\x9C\xBA\xFC\x71\x7F\x1A\x83\xCF\xFD\xA3\x02\x82\x86\xB4\xC7\x18\xCC\x7A\x4D\xC9\x30\xC7\x8E\xFC\xDD\x5A\xC5\x1B\x52\x28\xA8\x09\x6D\x01\x45\x57\x40\xD5\x26\xD9\x9C\xFB\x8C\x00\x08\x9D\xE6\xCF\xAA\x98\xF7\x30\xC3\xC3\x47\x3E\x22\x81\x54\x6C\x8B\xE5\x4F\xD0\xC2\x9C\x56\x29\xE4\x55\x82\x7D\xCB\xE6\xA5\x51\x5A\xA5\x19\x8C\x16\x9D\x53\xA6\xC7\xD9\xF9\x7B\xCE\x72\x73\x7C\xF1\x51\x69\x17\x59\x08\x49\xF6\xD9\x17\x71\x0D\x78\x31\xD7\x81\xB9\x55\x40\xE7\x21\xAF\x02\x38\xCB\xBA\x85\x80\x58\xCF\x66\x03\xC1\x8F\xEA\xBF\x9E\x17\xFE\xF8\xDA\x57\x85\xF8\x4D\x48\x85\xFE\xEA\xF8\xDD\x17\x09\x19\x76\xF1\x52\xDA\x9D\xE4\x2F\x70\x0F\x20\x9F\x87\x6E\x09\xD5\xD9\x4B\x83\x5E\x10\x2B\x7B\xC0\xF5\x23\x5E\xAF\x69\x46\xAB\x20\xEB\x61\xD8\x9A\x0E\xC3\x21\xA6\x5F\xD8\xC3\xAD\x60\xC8\xBD\xDD\xE3\x49\xE7\x8D\x80\x93\x6C\xF0\xD1\x44\x8E\x62\xAD\x8F\x9C\x98\x7F\xB4\x30\x29\xA4\xF6\x49\x21\x39\x21\xE4\xDC\x19\xB0\x4F\x9F\xE1\x4F\x61\x39\x58\x05\xB4\x84\x9D\xAD\x83\x65\x17\xD6\x5A\xE3\xEC\x08\xF8\x3C\xDC\x42\xE0\xD4\x69\x69\x86\x71\xCB\xE4\x8D\x6E\xCB\xCF\x32\xA1\x5A\x0A\xD4\xCA\xED\x00\xDB\x45\x99\x2C\xCA\x5D\xA3\xEB\x2C\xFB\x8D\xA4\x8B\xCB\xFF\x80\xEE\xC5\x59\xF4\x4E\xCB\x43\x3E\x2D\x0F\xBB\xD3\xF2\x2A\x92\x73\x79\xF2\xDF\x97\xF3\x70\x01\xED\x8F\x76\x33\xE2\x2E\xB4\xD4\x7A\x15\x6C\x0F\x23\x6E\xFC\x02\x8C\xB8\x26\xEB\xF0\x30\xBE\x6C\xE7\x0B\x9D\x89\xAD\x3B\x5F\x60\x1B\xDB\xFB\xE7\x3C\x83\x6E\x78\xBE\x70\x03\xB4\xE2\x05\x31\xFA\x2D\x1E\x41\x1F\x74\x52\x67\xBD\x89\x49\xFC\x18\xEF\xC2\x8F\xB2\x0F\x5D\x14\xC1\xBA\xCB\x67\x8A\x3F\x63\x00\x04\xED\xBF\x56\x33\x5F\x7F\x25\x93\xB7\xEC\xA8\xFB\x95\xDC\x8D\xD6\xA0\xCD\xCE\x84\x9D\x14\xFD\xCB\xC7\x78\xA1\x0E\x7B\x47\x3D\xE1\xA2\xA3\x9E\x70\x78\xD4\x13\x2E\x3C\xEA\x99\xC1\x1F\xE8\xCE\x7D\x7E\x0C\x45\xF3\x9C\x62\xCF\xAE\x7D\x49\xDB\x1C\x72\x82\xEC\xE4\x3A\x5D\xB2\x0F\x75\xE7\x2A\x96\xD4\xE6\x1C\xAF\xEF\x3D\x1F\xE6\xBF\x9A\x7D\x72\xF0\xF8\x86\xE3\x33\x19\x0E\xCF\x4D\xC3\x6C\x2C\x30\x94\x9B\x9B\x82\x3A\xDD\xDF\x03\x6C\x60\xEF\x70\xBC\x15\xCC\xDF\xD2\x33\xB7\xBE\x5B\x75\x50\xB7\x87\xDB\x74\xC6\x60\x71\xB8\x96\x9B\x3A\x04\xED\x96\x9B\x83\x3A\x38\x62\xE4\x26\x34\xC8\x9C\xDE\xE4\x48\xB7\xCA\x86\x32\x35\xF5\xDF\x4A\xF0\x56\xE2\x6F\x85\xCE\xD0\xCB\xA0\xE9\x46\x22\x6D\xB5\xE4\xD9\x23\xCE\xEC\xB6\x5D\xB8\xFF\xBA\xDA\xA4\x6F\xA5\x4D\xFF\xD4\x68\xBB\xAD\x2E\x74\x86\x65\x39\xC5\xD9\x2B\x83\x7D\xEB\xCB\x51\x9B\xEE\x1C\x47\xE0\x7C\x39\x2D\x7B\x65\x8D\x04\x16\x07\x75\xC8\x26\xF9\xD1\x23\x85\x31\x04\x37\x46\x69\xD1\xDC\xE8\xE1\x1A\x6F\x1E\x2F\x50\xA3\xCC\xBC\x9D\xD7\xBA\x14\x1F\x6B\xB7\xDD\xD4\xDA\xA7\x3D\xCF\x38\xE1\xBC\x6D\x28\xA2\x38\x73\x7F\x3A\x44\x31\x21\x7C\xC3\xC0\x7D\xC7\xBB\x87\xD8\x26\xB3\x1D\xBA\xA5\xF4\x5D\x4E\xCD\x25\xBA\xC7\x79\xD3\xAB\x02\xCC\x87\x40\x7D\xA8\xFC\x9F\xF0\xE2\x5D\x75\xF4\xF7\x82\xE0\xDA\xF5\xEB\xD7\xFF\xC4\xBD\x4E\xA2\xEC\xF0\x0E\xFE\xEF\xF3\xEE\x1F\xB6\xBF\x83\xE0\x9B\xD8\xF6\xF8\xA3\x64\x40\xA1\xBE\xB9\x6D\xA2\x09\x98\x69\x76\x25\x56\xC9\xAD\x69\x1E\x3D\x5D\x03\xB4\xFB\xF6\x1D\xBB\x56\x04\xEE\xA7\xDF\x3B\x6F\x56\xEA\xA7\x31\xDB\xC5\xA2\x69\xFA\x67\x74\x06\x47\x37\x9E\x19\xDD\xC8\x8F\x6E\x42\xA3\x1B\x13\x82\x3B\x6F\x9B\x64\x74\x25\x80\x29\xC5\xD1\x4D\x7B\xA3\x1B\xE3\xE8\x52\x40\x16\x87\xF7\x6F\x37\xB8\xEB\xA0\xD1\x4D\xC8\x1F\xCA\x42\x22\xA3\x3B\x14\x08\x6E\xBB\x4A\xDB\xB0\x52\x1C\x09\xDE\x63\x6F\x13\x94\x89\x01\x5D\xFE\x13\xDA\xC2\x28\x86\x02\x18\x0A\x7A\x4B\xBA\x91\xD3\xE5\xCF\x90\x96\x18\xE3\xD4\x88\x05\xF1\x78\x8E\x20\x0B\x16\x76\x23\x0B\x7B\x3C\x87\x80\x41\x0D\x71\xC1\xDF\x51\x81\x53\x55\xDA\xC6\x3B\xB0\x0B\x62\xCC\x6A\x30\xAD\x31\x86\x3C\x86\xBD\xE3\x9F\x5F\x79\x0C\x25\xFB\xF6\xEA\x2B\x6D\xED\x32\x8A\x4D\x45\xED\x71\x45\x1F\x10\x0E\x52\xC7\xD9\xA7\x96\x78\xFE\x93\x43\xEE\x6E\x07\xFB\x7B\xDF\xDB\x67\x49\x77\xA1\xA9\x32\x09\x30\xA3\xB1\x20\xC4\x17\x50\x6B\xB3\x8B\x2C\x8E\x99\x1E\x2E\xB2\xB8\x75\x56\xB2\xCC\x52\x6C\x34\x36\x5E\x8B\x13\x59\xC4\x1F\x25\x4E\x6D\x36\x10\xFD\x1D\x45\x60\xD2\xF8\x1F\xE1\xE2\x33\x8D\x9A\x2A\xD1\x47\x99\x33\x18\x9F\x98\x5C\x3B\xBD\x88\xA8\x22\xC8\x19\x81\x3F\x26\x33\x11\x81\xF5\x57\x11\x24\xD8\xE6\x84\x81\x1D\x9D\xAA\x22\x02\x91\x44\xB2\xA7\x78\xB7\x08\x32\xC6\x79\xD0\x19\x8C\x5A\xC2\xCF\x9E\x80\x50\x72\xF5\x6C\x18\x8C\x8C\x3C\x92\xF9\x45\x3D\xDE\x6B\x51\xA7\x61\xBE\xC1\xA2\xDE\x5B\xC7\x21\xCD\x7E\x68\x9F\x51\xDB\xEA\x42\xD2\x5B\xCD\xE3\x6E\xBB\x5F\x04\xEE\x4F\x87\xA3\x65\xFD\x68\x7D\x6C\x26\x89\xFE\xA2\x57\x17\x08\x33\xBC\x2F\xA2\x46\x24\x49\x7D\xF4\x5D\xA0\x3F\xF4\xC3\x75\xF2\x7D\x72\xE3\x79\xF7\x0F\xC9\xCF\xF2\xBD\x34\x5F\x70\x7B\xBF\xF3\x26\x22\x00\x9F\x94\xE1\x5B\x9B\x0F\x5F\xFA\x5E\xF7\x4D\x92\x69\xA1\x1E\xB9\xEB\xD7\xAF\x5F\x8F\x4F\xF6\xBC\x45\x6F\x39\x85\xC7\xE0\xF1\x02\xE5\x81\x02\xEB\xE2\x01\xBC\x55\xEE\x2C\xC4\x33\xF0\x56\xF1\xEE\xF0\x56\xD9\xBC\xF2\x90\xF5\x94\x07\x0A\x38\x18\x6C\x9C\x5A\x0B\x8A\x76\x56\x26\x7E\xC1\xFA\x11\x6D\xF4\xB4\x33\x1B\xD5\x58\xC0\xA4\x75\x7B\xE6\xA9\x50\x3E\xE1\xC2\x76\x61\x83\x92\x76\x4D\xA0\xA8\x72\x73\x14\x72\x9C\xAE\xD9\x2B\x14\x4D\x8F\x1C\x32\x77\xA1\xF1\x17\xCE\x36\x90\xB9\xAD\xEE\x3A\xC7\x6B\xDD\x5D\x97\x0D\xCE\x0C\x97\x60\xC1\x25\x4C\x9C\x6D\xEA\x09\x1F\x7A\x8E\x19\x29\x74\x82\x93\xA9\xCA\xA0\x80\x09\xCD\x1E\xAA\xB0\xAD\x0D\xCB\xAA\x32\xAA\xA7\xCA\x61\x8C\x22\x73\x2C\x51\x82\x2F\x5E\xBE\x95\x7C\xE1\x41\x5A\xEE\x7A\xF9\x56\x20\x83\x11\x64\x90\x3E\x30\xAD\x33\xC8\x4F\x14\xA6\x3B\x44\x53\x90\x41\xFE\x6A\xBC\x03\x19\x44\x6D\xA6\x15\xFD\xD7\xC1\x50\x93\x21\x43\x4D\x66\x19\x6A\x37\x6D\xF4\x26\x19\xCA\x76\x2B\x08\x76\x6D\x98\x33\xA6\x0F\x59\xA3\x17\x08\xE7\x76\xBA\x3F\xDD\x17\xCE\xA4\x30\xE5\x50\xB0\x72\x93\x49\xA4\xBC\x3F\xAE\xC9\x50\xE7\xE0\xE3\x61\xED\xB6\x1A\x0F\xEB\x51\x34\xEE\x4D\x4D\x35\xC6\x47\x46\xFC\xE5\xC9\x46\x6B\xCB\x1F\x23\xDE\x2E\xBD\xEF\xB0\x86\x8C\xD3\x20\x65\x92\x1D\x06\xC7\xED\x54\xB7\x31\x78\x61\x08\xB9\xE2\x3A\x9B\x2E\x64\x97\xB4\xEF\x24\x09\x39\x25\xC9\xC9\x41\x13\x64\x86\x86\x74\x81\xEB\x02\xBF\x93\xB2\xFB\x82\x6E\x59\x47\x7B\x1F\x5A\x4F\x70\x5F\xDC\xF1\xC2\x0A\xFD\x34\x10\xE1\xA0\x64\xD7\x48\xB2\xA3\xE6\x9C\x62\xE7\xDD\x6C\x0B\xA2\xE2\x89\x8E\x39\xC1\x0E\x11\xC9\x5E\x3C\x4A\x94\x37\x49\x89\x32\x83\xAC\x1A\x41\x09\x38\xF9\xF3\x07\xA6\x75\xB9\x90\x12\xFC\x8E\x50\xA2\x6C\x29\x51\x32\x25\xCA\x96\x12\xBE\xB8\x96\x12\xB4\xB8\x36\xA0\xC5\x45\x74\xC2\x46\xF4\x0B\x4D\x35\x86\x09\xAE\xAF\x13\x66\x0B\x64\x09\xE4\x09\xD6\x31\xC8\x45\x20\xEF\x61\x32\xCF\xE6\x6F\xC9\x76\xD1\x91\xB3\x99\x83\xEE\x6C\xF6\xA0\x3B\x9F\x39\xE8\xCE\xE7\x0E\xBA\xF3\xD3\xD3\x2A\x9F\x3B\xE8\xE6\xBB\xC3\x83\xEE\x7C\xEE\xA0\x7B\xFE\x2D\xE0\xB7\x06\x07\xDD\xFE\xAD\xE1\x41\x37\xDD\xA5\xAE\xBB\xF4\x2C\xE4\x72\xD0\xED\x99\x21\xDF\x85\x19\x28\x5D\x4F\xBE\x17\x33\xE4\xCC\x0C\x73\x21\x04\xC1\x4C\x08\x01\x95\x96\xF1\xD4\xA5\xAC\x55\xE3\x39\x77\x84\x0C\xC6\xEC\x8E\xD0\x4A\xD2\xBB\x7D\xCA\xAA\xA8\xE7\x51\x1E\x42\xE4\x4D\x0B\xC5\x82\xF0\xAD\x6E\xF3\xFC\x4B\x89\x1A\x6D\xB3\x52\x2A\x0E\xEB\x74\x68\x88\xD2\xEA\xEA\xBB\xE7\xF6\x07\xD6\xA5\xB8\x1D\xFB\x47\x67\xC1\x36\xB5\x25\xC8\x02\x45\x3E\xF9\x64\xCA\xE2\x54\xD7\x60\x20\x6C\x68\xDC\xD9\x03\xC4\x1C\x95\x6D\xDA\xF1\x82\x0F\x60\xBC\x0B\x48\xEC\x73\x5F\x8B\x27\x48\x0C\xD1\x09\x54\xEE\xB6\x51\x63\x84\x68\x6D\x3A\xC9\x66\x72\x6C\x13\x36\x91\xAF\xA9\x4D\x0E\x23\x01\xC4\x29\xD8\x75\xDC\x4C\xA0\x40\xFD\xC1\x77\x0D\xF0\x15\x52\x67\x4F\x33\x9C\x4C\xBB\xEC\xA7\x6E\xBB\x19\xDB\x40\x69\x13\xE2\x56\x66\xF4\x70\x6D\xDD\xF9\xB3\x10\xA1\xC6\x69\xFD\x21\xC7\xEF\x05\x78\xFB\xED\xDB\xBD\xFB\xE4\x31\xF8\x99\x18\xEF\xBF\xED\x4D\xBD\xFB\xE4\x27\xF5\xC4\xAF\xD2\x07\x9F\xFF\xF1\xFE\x17\x3D\xB0\x9B\xD4\x7D\x5B\xBF\xD2\xA7\xDE\x12\xE1\xEB\xCF\xFF\xFB\x6F\x9D\xAD\xE0\xDA\xA5\xFF\x0A\x9F\x5C\xFB\xF3\x63\xB3\x55\x10\xDA\x7A\x6D\xDD\xCE\x27\xDE\xF1\x6D\x83\x4A\xF0\xD9\x47\x7F\xE3\x15\xF4\xEC\x67\x9F\x7E\xA8\x7B\x96\x71\xE0\x84\x1D\x3A\x75\x75\x44\xAD\x2D\xD1\x9D\xC8\x6E\x99\xEC\xBC\xB4\x58\x41\xFD\x72\x5B\x4D\xAD\x79\xE3\x14\xB3\x5D\xD9\x60\x81\x23\xB2\xB1\xA3\x42\x62\x7C\x69\x95\xC6\x2D\xDD\x5A\xC5\x14\xAF\x72\xEA\x70\x55\x18\xDE\xE1\x63\x13\x84\x10\xD9\xFC\x18\xE5\xA8\x70\x7A\xA0\x37\xA4\xD5\x38\x0C\x91\x52\x51\x06\x96\xD3\xF7\x9F\x47\xAE\xAB\xC6\x10\x55\x36\x27\x94\x6B\x1C\x9E\x88\x86\xA7\xBD\x6F\xF0\x3E\x0E\x4F\x44\xC3\xD3\xDE\xD7\x78\x9F\x86\x27\xE2\xE1\x69\x9F\x0C\x94\xC1\x62\x4C\x75\x5A\xAA\x94\x86\x27\xE2\xE1\x19\x56\x4C\xC3\x13\xF1\xF0\x0C\xAB\xE6\xE1\x89\x64\x78\x86\xD5\xF3\xF0\x44\x32\x3C\xBD\x06\xE0\xB3\x27\xFF\xF8\xD3\x86\x1E\x8A\x8F\x8C\x3C\xCD\x70\x04\x61\xCC\x28\xAA\x5A\x46\x0B\x62\x48\xCE\x36\x75\xDC\x8D\x5A\x4C\xA3\x26\xF3\x48\x62\x20\x54\x15\x77\xA9\x1E\x48\x26\x74\x03\xAE\x20\x39\x5B\x13\xDA\x12\x84\x14\x24\xC1\x47\x2B\x41\x2F\x59\x3D\x23\x00\x70\x46\x35\xCB\x49\x2A\x64\xBA\x6A\x8E\x36\x35\x2E\x78\x60\x92\x49\x03\xCC\xDA\x34\x7B\x5A\x77\xA6\x2B\x68\x73\xF0\x3B\x81\x13\xE2\x93\xE7\xD7\x90\xBA\xE5\x4F\x27\xEB\x00\x82\xA6\x36\x10\xAC\x15\x0A\x90\x8B\x94\x1C\x35\x06\x6E\xFB\xD1\xD3\xEC\xB4\xD8\x30\xD6\x26\xF8\x5D\x95\x21\xF6\x6C\xCF\x15\x35\x1D\xCB\xCD\x9D\x2B\xEA\x17\x70\x26\xD7\x15\x5F\x82\x5E\x7C\xAE\xA8\x5F\xD8\xB9\xE2\x2F\x59\x15\xB3\xD0\xB5\x10\xD2\x9E\x1B\xE5\xED\x87\x7F\x64\x46\x3B\xDC\x47\xE7\xBA\x94\x1D\x65\xE6\x49\x79\xBA\x45\xE2\xB7\x10\xB2\x2F\xA2\x47\xE9\x6F\xEA\xC8\xA9\xF3\x75\x4C\x25\x4B\x25\xF1\x7A\x6D\x21\x3C\x43\x02\x9B\x4C\x92\x9B\xCE\x9E\xAB\x13\x3E\x45\xAF\x53\xCE\xD4\x49\x8A\x42\x0A\x49\x03\x21\xD8\xD6\xCD\x87\xBE\xB0\x73\x5F\xF8\xF7\x35\x7F\x81\xAD\xA0\x2F\x38\x54\x16\x9F\x18\xBC\xD9\x94\xEF\x62\x1B\x89\xF8\xD9\xF6\x0B\xB1\xD2\x7E\xF2\x0A\x91\xCF\x51\x92\xE0\x95\x7C\x37\x21\x48\x45\x2A\xB0\x4E\xC4\x99\xC0\x00\x6E\xBC\xC9\xE7\x13\xF9\xB8\xA9\x47\xDC\xC1\xD4\x57\xC5\xE9\x25\xAC\x34\x4F\x3E\xAB\x32\x6A\x50\xDC\x10\x34\x11\xEF\x9C\xF0\x69\x53\xE7\x10\xE2\xE7\xA1\x7C\x1E\xE0\xBD\x04\x0C\xD0\x07\xCE\x6C\x62\xE7\x84\x1C\xD4\x8D\x54\x68\x91\xB5\xDD\x18\x41\x0E\x16\x32\x24\x5E\xD7\x8D\xC8\x6D\x6D\xD4\x86\x1A\x61\xF1\x41\xD3\xD6\x90\x10\xB5\xAE\xB0\xAD\xBE\x47\x97\x70\x86\x2E\x21\x21\xF9\xF4\xE8\x62\x7A\x05\x18\x2E\x20\xFB\x52\x38\xF0\x02\x8A\x7A\xF1\x21\x37\xE4\x2D\x76\x9A\x61\xF6\xFA\xCD\x1F\x5E\xB0\xD0\xEF\xEC\xEC\x2C\x9F\x96\x33\xE1\x6B\x8A\xF9\x0D\xD9\x04\x79\x83\xF8\xCD\xA9\x2A\xCE\x4D\xE6\x1B\x0B\x51\xEB\xBD\x5A\x27\x9C\x59\x20\xC1\x97\x91\xE0\x3C\x04\x77\xCF\xE3\x1B\x52\x19\xBA\x65\xDC\x94\x3C\x75\xFA\x8C\x9B\x78\xC6\xC5\xC5\x7B\xC0\xBC\x23\xA1\x16\x67\xF3\x4A\x38\xB5\x24\x8C\x98\x79\xF9\x1C\x5D\xE5\x36\x6B\xBF\xB3\x73\xDF\xF9\xAF\x34\x7F\x87\x83\x42\xDF\xB1\x23\xB4\x3C\xEC\x73\x71\x5B\x56\x2A\x23\x36\x9A\x19\xB1\x51\x43\x61\xAA\x52\x8A\xA6\x41\x4B\xBA\xCF\xB9\xB3\x5C\x8A\xD4\x5E\x8F\x5A\xAE\x1E\xF5\xB8\x7A\x84\x6C\x49\xBD\x2F\x5A\xAE\x4E\xFA\x5C\x3D\x1A\x70\x75\x32\xC3\xD5\xA3\xA6\x1E\x43\x88\x9F\x4F\x5A\xAE\x1E\xE1\x44\x99\xE5\x6A\x69\xA9\x6A\x7B\x56\x08\x95\xAC\xF4\x2C\xA4\xD8\x51\x1C\x46\x28\x60\xD2\xBE\x0F\x48\x83\xAD\x0D\x69\x4E\x88\xCF\x9A\xDA\xB3\xE7\x08\x42\xCF\x9E\x54\x6E\xD8\xA3\x98\x69\xCB\xA5\x68\x37\x30\x48\xB1\xA4\x5F\x2E\x0E\x5F\x57\x8E\x95\x72\x14\x39\x85\x39\x8A\x67\x8F\x3A\x8F\x9A\x38\xFB\x0F\x4A\x9B\x6D\xBD\xE5\x2E\xA3\x98\x65\x77\x28\xC3\x8B\x1C\xAE\x5F\x21\x04\x0F\x57\xDA\x3D\x49\x0F\x4D\x0F\xDB\x8E\x0D\xC1\xC8\xDC\xA0\xD7\x71\x33\xE8\xD4\x24\x03\x7B\xB5\x8A\x9C\xAA\x94\xF1\xA6\x21\xCA\x83\x67\xAE\x72\x62\x53\x73\xB5\x8E\x41\x5D\xDD\x82\xE8\xB1\x2B\x15\x43\xDE\xAD\x55\x04\xB7\xC4\xB7\x08\x25\xF2\xBC\x44\x1F\xE6\x3A\xCB\xB0\x28\x08\x71\x75\x0B\x6E\xB9\x68\x75\x7A\x71\xD1\xB4\xF4\xAA\xEC\xAD\x56\x47\xDB\x66\x0B\x17\xD5\xD7\x93\xE7\xAB\x7B\xEE\xCA\x8C\x25\x50\x39\xD5\x8C\x8D\x0A\x54\x90\xB9\xEF\x9E\xB5\x0A\xA3\x96\xEF\x3E\x33\xF0\x34\x72\xB8\xE1\x53\xE7\x6B\xE3\x6C\x79\xED\xB1\xD6\x15\x68\x92\xF5\x69\x1B\x12\x6D\xC9\x30\x1F\x23\x6D\x13\xA1\x2D\xDD\xAF\xC2\x1E\x6D\x21\xBA\x4A\x56\xE5\x98\x62\xDD\xC9\x6A\xA1\x90\xFB\x3B\x1A\x24\x4C\x83\xF0\x2A\x87\x40\x87\x57\xEB\x11\xD1\x20\x65\x1A\x24\x48\xDE\x18\x46\x30\xE2\x5B\x21\x7E\x70\xBE\x4A\x08\x7C\x02\xC9\x4B\x35\x10\x8D\x63\xA2\xF1\xAD\x14\xAD\x4E\x2F\x2E\x9A\xC8\x6B\x41\xAD\x6A\x14\xB2\x11\x6E\x7F\x2D\x6D\x9E\x91\xD2\x4E\x73\x2D\xB8\x0B\x51\xF8\xDF\xE9\x69\x1D\x12\x0A\xB8\x24\x46\x0B\x9C\x25\x2B\xE1\xD5\x6A\x84\x8B\x74\x53\x25\x10\xBA\x2F\x5D\xBF\x7E\x3D\xDE\xA8\x94\x53\x55\x88\xEF\x91\xC9\xFD\x42\x03\x21\x05\x3F\x40\x78\x95\x5B\x51\x67\x7F\x57\x82\x23\xAA\x04\x05\x12\xDF\xC3\xB6\xD1\xFA\xD7\xD4\xF1\xC9\x22\xC8\x20\xBB\x42\xE9\xF0\xC8\xE5\x83\x03\x27\x70\x06\x26\x6C\xD7\xCE\x9E\xD4\x37\x9C\x0D\xCE\xDC\x68\x36\x10\x8F\x0E\xA6\xC4\xEC\x88\xDD\xCC\x84\xB0\x0B\xB8\xD6\xFA\x11\xF3\xB3\x00\xA5\xF8\x70\xC4\x6E\x66\x42\x2C\x2C\x5A\x94\x5B\x25\x2E\x3D\x92\x61\x07\x3F\xA7\x57\xFF\xAE\x4E\xB2\xCF\x86\x36\xA5\x40\x4A\x9A\x30\x6A\xC1\x84\x71\xBF\x82\x53\xA1\xA5\x46\x89\x6D\x7B\xEB\x0F\x5F\xE2\xA8\x78\x5C\x2B\xCB\x35\x82\x77\xF6\xAF\xD3\x5E\x06\xAB\x03\xD5\x3A\xCF\x1A\x30\x2E\x39\x47\x79\xDA\xCA\xF3\xA0\xC4\x79\xD6\x90\xF3\x6C\x38\xAD\x23\x88\x9C\x3D\x07\xF4\xD6\x79\x08\xF1\x49\xD9\x3E\xD0\xFC\xC0\xCA\x03\x8B\x0F\xE8\x8E\x96\x3B\xEA\x74\x1D\xE2\x62\x14\x92\xF3\x2C\xD5\x85\x3A\xDE\x7A\x1D\x6E\xD6\x09\x7B\xC7\xA6\xE5\xDB\x2E\x4A\x92\xA2\xC4\x6D\x83\x2E\x9F\xA4\xB5\x78\x24\x4E\x2E\xB4\x3F\x93\x63\xD0\x53\x05\x2F\x6F\x95\x85\x80\x36\xE8\x06\x12\x18\x95\x3F\x29\x2B\x0D\x01\x0E\x66\x6B\xDE\x8A\x8C\x1A\x1E\x97\x5D\x04\xF3\x3E\x91\x33\x12\xA2\xF9\x1A\x8A\xCE\x57\xD4\x82\xEF\xA1\x98\x95\x02\x79\x8A\x9B\x45\x48\x95\xA7\xA7\x64\xA8\x9C\x98\xA3\x4E\x55\x16\x8A\x2A\x84\x09\xC3\x8E\xB1\x1F\xBF\xCC\xDC\x52\x66\x6E\x09\xF6\xEA\x16\xE4\x8F\x5D\xC1\xDD\x10\x52\xA2\x8A\xA0\x84\x92\x6F\x85\x44\x88\xCA\xF0\x09\x50\x00\x13\x88\x37\x65\x4F\x6A\xF1\x53\x7A\xAD\x2A\x8F\xA9\x6A\x9F\x0B\xAA\x90\xBA\xBA\xEF\xD8\xF5\xEB\xD7\xAF\x4F\x2E\xD6\xFB\xC0\x7C\x8D\x0E\xF0\x8D\xBF\x2B\x11\x44\x38\x18\x5F\xA3\x71\x26\xD4\xFB\xBB\xFB\x84\xB6\xB1\x4F\xCA\xDA\x07\x25\xEC\x97\xF6\xE0\xAE\x09\x17\x99\x14\x69\x45\x39\xA4\xEB\x09\x70\xC4\x12\xEE\x60\x29\x1C\x0B\x79\x65\x9E\x6A\x90\x0E\x87\x48\xE1\x34\xB3\xDD\xE8\xA4\x3E\xD4\x09\xA9\xEF\x47\x06\x27\x6D\x1D\xC8\xC8\x2C\xF0\x4E\x05\x9B\x7D\x21\x8C\x12\x62\x75\xB5\xA5\xB6\xF5\x8B\xC4\xEE\xFA\xE8\x4D\x33\xBA\x9D\xD6\x21\x84\x3D\x46\xB7\xC2\xE8\xF2\xA0\x63\x74\x2B\x8C\x6E\x5B\x46\xB7\xC2\xE8\x96\xB4\x2E\x61\x74\x64\xF3\x68\xB3\x8E\x9D\x6A\xC9\x95\xF0\xB2\x04\x06\xA2\xF5\x2A\x6C\x79\x3B\xED\xA8\x17\x0F\x79\x3B\x86\x64\xC8\xDB\x34\x5E\x99\x53\x10\x6E\x56\xA3\x59\xE6\xA5\xB4\x71\x2D\xF3\xE6\x9E\x79\x47\x4E\xE3\x7F\xA7\xA7\x55\xE1\x54\x35\x16\xE6\xCD\xAB\x10\xC6\x0B\x98\x77\x22\xCC\x3B\x21\xE6\xCD\x7A\xCC\x5B\xC2\x04\x26\x7C\xAB\x65\xDE\x92\x99\x77\x0C\x51\x8F\x79\xE5\xB5\x6A\x72\x63\xE6\xCD\x3A\xE6\x2D\xA1\xEC\x98\x37\x1B\x32\x2F\x97\xB5\x0F\x26\xB0\x5F\xDA\xE3\x99\x37\x26\xE6\x95\xEC\x9A\xC5\xC9\x76\xD6\x8F\x20\x6E\x67\xFD\xBC\xE2\xEE\x93\x92\x8D\x38\xA5\x68\x40\x20\x0D\xB3\x4C\x3E\xEA\x8F\x9A\x9C\xA8\xCB\x10\x8D\x86\x43\x34\x5A\x3C\x44\x0B\xCA\x44\xAD\x68\xC4\x0D\x73\xC1\x49\xF2\xCA\xC8\x7E\xC5\x1A\xB5\xAD\xB6\xFA\x8E\x33\xA6\x1F\x0E\xC1\x2B\xCC\x55\x5C\x50\x28\x98\xE7\x0A\x3B\xEA\xC9\xCC\xF8\xD2\xFF\x3A\x54\x8E\xE4\xF6\x2F\x5E\x99\xF1\xC3\xE6\x28\x55\xD0\xE5\xF7\xD1\xD6\xD8\xE2\x0E\x42\x21\x05\xD5\x5A\xA1\xC0\x5E\xF1\x8B\xE7\x49\xA2\x47\xC8\xFB\x73\x17\x94\xDF\xCF\x62\x99\x6E\xF0\xB7\xB4\x3D\x54\x9B\xEE\xC2\x46\xE3\x6F\x61\x71\x1A\x68\x40\xF4\x1A\x05\x27\x86\x33\x15\xF6\x3F\xEA\x6E\x51\x9E\x55\xAE\xA2\xB0\x58\x6D\x1E\xB6\x3E\x8C\x60\x3A\x3B\x89\x59\x6C\x27\x31\x2F\xC4\x4E\x62\x3A\x3B\x09\x17\x3F\x67\x27\x31\x3D\x3B\xC9\x0B\xAA\x01\xA4\x86\x39\xFF\x6B\xF3\xC2\xFC\xAF\x51\x29\xC3\x6A\x5A\xC7\xE9\x30\x7B\xCF\x58\xBD\xD4\x33\xD2\xCE\x4E\xD2\x1A\xB2\x40\x31\xAF\xD6\x96\x27\xAD\x75\x3B\xB8\x2D\xCE\xBC\x07\x06\xEE\x85\xAA\x90\x36\x57\x55\x04\xD6\xD9\x4D\x4A\x5A\x69\x37\xAB\x04\x6F\x6E\x54\x29\x58\xB7\xD5\x90\x6A\xB8\xDD\x54\x84\xE0\x7E\x16\xBF\xC9\x5D\x50\x15\xB4\x43\xBD\xB0\xB1\x5E\x8D\x49\x9C\xD5\x13\x67\x1B\xB0\x67\xAB\x12\x82\x6A\x9F\x39\x0A\x05\xA8\xB3\xD5\x7E\xD8\x57\xDD\x06\xFB\xAA\x25\x28\xAA\xDB\x69\x71\xB8\x1D\xF6\xD3\xDE\xAF\xBA\x03\xB7\xCB\x7C\x72\xBD\xB3\x63\x9B\xEA\x4E\xB8\x83\xD0\xA1\x33\x3E\xB8\x6E\xEF\x8E\xAB\x65\xB8\xAD\x32\x70\x47\x75\x97\x39\x0A\x77\x12\x50\x0F\x6F\x25\xEF\x86\xBB\x70\x17\x77\x27\xFE\x33\xC5\x7F\x0E\xC0\xA4\xA9\xEE\x42\xDD\x00\x45\xC8\xDD\x10\x36\x70\xA7\x7F\x1B\x2F\xA6\xFD\x8B\x03\xBD\x8B\xEA\x4E\x58\x46\x79\xB4\x8C\x82\xE5\x6E\x88\x9B\xEA\xCE\x2E\x72\xA2\x5A\x1E\xD6\xCB\xEF\x1B\xAC\xCB\xC0\x32\xCA\xFC\x65\x56\xC2\x8F\xD3\x34\x7A\x09\x39\x8A\xBC\x04\x17\x82\xA6\x7E\x09\x58\xB2\x61\x81\x3D\x0B\x2F\xC1\xCF\x5E\x82\xBF\x4C\x53\x2D\xB3\x55\x19\xCC\x3A\x4B\xE0\x97\x54\x77\xF1\x09\x29\xF6\x1A\xEE\x44\x0A\xD5\x77\xA1\xBE\x8B\x9D\x94\xBB\xCB\x74\x77\x2A\x8E\x73\x53\xB8\x9B\xA7\x89\xC6\x3E\x2F\x83\x6D\xAA\x65\xB8\xD3\x29\x6C\x3B\x8C\x28\x2E\x94\xBE\xA8\x96\xC1\xC2\x5D\xEB\xD5\x5D\x83\xC2\x49\x1F\x65\x0A\x7B\x63\xF7\xDD\x60\xE0\x4E\x58\x5E\x50\x83\x6D\xEA\xBB\xF1\x8D\x29\xDC\x8D\x04\x6D\xF0\xA5\x03\xF2\xD2\x01\x98\xE2\x9F\x3B\x21\x22\xAA\xB0\xF4\xBE\xCB\x6D\x35\xF5\x5D\x44\x95\x97\x80\x15\x09\xAD\x91\x27\x6E\x27\xB5\x9D\x10\x32\xA4\x7A\x41\x6E\x37\x48\xA7\x3B\x3C\x34\x3B\x3D\x5A\x86\x71\x75\x17\x8E\x3D\x09\xDC\xE5\xBA\x1D\x4B\xE4\x06\x98\xD0\xE0\x36\xF5\x72\x3B\x2E\x93\x06\x96\xF7\xBE\x53\xDD\xE1\x19\x04\x1F\x55\xCB\x70\x17\xF6\x0F\x1B\x0B\x77\xF2\xC8\xE3\xB8\x2B\x19\x77\xD3\xE3\x20\x1A\xF0\x3B\x3D\xBB\xF0\xB8\xF3\x26\x81\xD8\xB8\xBA\x93\xD8\x5B\x9D\x85\x82\x38\xBC\x7E\x69\xCB\xCF\xF5\xCB\xE8\x05\xEE\xD0\x52\x65\xE0\xA5\xD5\xDD\xE6\x28\x2C\xF7\x98\xF9\x4E\xB8\x1B\x59\x78\x79\xC8\xCC\x77\xFB\xB6\x32\xD9\x6F\x8E\x99\x17\x74\xE9\xFF\x1D\x66\x9E\xE3\xE3\x17\xCE\xC2\x1D\xF7\xFE\x35\x31\xEE\x50\x46\x51\x47\x5F\x86\x77\x5E\x3A\x60\x55\x16\x53\x4B\xC2\xAA\x77\xD6\x77\xF5\x58\xF5\xA5\xC8\x86\x77\x21\x1B\xF6\x59\x89\x99\xF7\xC6\x77\xAA\x97\xF6\x87\x7F\x20\xA4\xEE\x5A\x34\xAE\x7B\x31\x6B\x8F\x55\xBB\x8E\xF8\x39\xD7\x89\xDA\x05\x7D\xB8\xE3\xAF\xDE\x87\x3B\xF6\xEA\xC3\x2D\xF6\x60\x09\xF2\xA6\x5A\x82\xDB\xF0\xC1\x6D\x70\x3B\x2A\x5F\xB7\x93\x1F\x60\x86\x3B\x9E\xA6\xDA\x07\x05\xDE\x2B\xF8\x1E\x8F\x50\xE2\x0F\x8F\x9F\x89\x62\xDE\x69\xA8\xAD\x78\x5B\x6D\x65\x2D\x52\xD1\x8E\x6E\x5D\x97\x29\x16\xC7\xAB\x49\x62\x71\x43\xF5\xEA\x2D\xF3\xF6\x2A\x7D\xDE\xA9\xA6\x1F\xED\x48\xFA\x8C\x6A\x57\x58\xCD\x06\x18\x0D\x0A\xCC\xF4\x6A\x15\xF3\xB9\x72\x72\xB5\x4A\xC9\x1A\x50\x93\x4E\x49\xB6\x82\xCD\x2A\x07\xED\x15\x75\x7D\xD4\x7D\x80\x54\xFC\xB1\x7B\xD9\x06\xCD\x98\xC9\x8A\x0E\x20\x82\xB1\x33\xE7\xDD\x5F\x5C\xBF\x7E\x5D\x6D\x34\x75\xB9\xA2\x03\x9C\x6D\xE3\xAB\x10\x6F\xD5\xFB\xAE\x40\x02\xFB\x20\xDD\xB9\x72\x76\xBD\xDE\x0F\xFA\x2C\xAE\xF9\xB7\xE9\xA3\xB0\x9F\xCE\xE4\x61\x2C\xB7\x96\x5C\x50\xDD\x0E\xD6\x3D\xB1\xA3\xD8\x2D\xB9\x6C\xAA\x3B\x88\xF2\x28\xBC\x96\xD8\x2E\xBB\xB3\xA3\x60\x04\xB7\xAF\x3B\xBD\x09\xB7\xBB\x1D\xC0\x21\x1D\xAD\xD5\xCB\xD3\xFA\x2E\x6F\x19\xBF\xAD\xBA\x1B\xF6\x57\x21\xDC\x51\x4D\xCD\x51\x98\xD6\x77\xD0\xF7\x77\xB3\xE4\xBB\x1B\xE4\xBD\xEA\x00\xDC\x0D\xA6\x9E\x42\xFB\x5D\x04\x61\xBF\x13\x9C\x6E\x58\xBA\x1B\x4A\x77\x57\x74\x40\x41\x41\xE1\x7C\xCF\x42\xE9\xC6\xDD\x70\x80\x32\x8A\xD0\xDE\xB5\x57\x35\x1C\xE8\xAA\x2A\x01\x49\x04\x13\x5F\xDC\x9D\xD5\xED\x30\x45\x31\x27\x62\x3B\x80\x42\xD4\x53\xB2\x73\x19\xB7\x03\xD5\x5D\x9E\x34\x53\x26\xCD\x01\xC8\x48\x54\x1F\xC0\x6F\x21\x24\xCA\xA0\x28\x1B\xAD\xD5\x77\x4C\xEB\x03\xD2\xCB\x29\x45\x34\xDC\xD6\x88\x71\xF9\x40\xF9\xCE\xC5\x2D\xB8\x0B\x69\x59\xDD\x05\x77\xBB\x9D\x9D\x6D\x94\xEE\x54\x60\x43\xFB\x2A\x7C\x62\x60\x5A\x1D\x80\xDB\xAB\x29\xDC\xC1\x20\x22\xC8\xEB\x05\x8C\x71\xDF\x33\xA6\xB3\xC8\x2C\x9B\x0F\xA6\x63\x95\x10\xDB\xDD\x7A\x72\x86\x13\xBC\xD1\xEA\xD2\x76\xB1\x2E\x6D\x5F\x80\x1E\xFA\x8F\x27\x2A\xE2\x13\xC1\xC0\x7D\x82\x3C\x3B\x0F\x57\xCA\x5D\xFB\x7D\xDC\x93\x53\x52\x77\x54\xF3\xCF\xF3\x79\x52\x6D\x51\x37\x64\x5F\x0A\x08\x70\x07\x5D\xD1\x94\x70\xDB\x8F\xD2\xCE\x79\x5A\x2B\xC2\x44\xA4\x43\x5B\x9E\x3E\x81\xBB\x70\x86\x93\x57\xBB\x8F\xAB\x87\xEB\xD0\x7D\x5C\x9D\xAD\xA3\xF5\x8A\xAB\xD9\xF9\xC3\x17\x52\x0D\x7F\xBE\xEE\xF4\x1B\xAB\xC0\x90\x95\x74\x93\x6F\x51\x48\x5D\xD4\x54\xDA\x7D\x6A\xAB\x52\x8C\xE4\x08\xCA\xFD\xC5\x1F\xC8\xB3\xDA\x3C\x5C\x5B\x30\x6B\x85\xA5\xC3\x13\x73\x96\x42\x57\x41\x75\x5F\x2F\x7C\x43\xB9\xA4\xA1\xAA\xDD\xBB\x4C\xA5\xA4\xD0\x0F\x6C\x35\xBB\x16\xF7\xD3\x5B\x4D\x6D\x1F\xAE\x23\x30\x6E\xD4\xD4\xF2\x42\x04\xB6\x7D\xE1\xC3\xDD\x0B\x7A\xE1\x0B\xD7\xBA\x17\xEC\xC2\x17\x3E\xBA\x57\x09\xFF\x72\xAF\x12\x3E\xD6\xBD\x10\x2D\x7C\xE1\x57\xF7\xAA\xE2\xB9\xBD\x4A\xF8\xD4\x5E\x6D\xF8\xCD\xBD\xAA\xF8\x9D\xBD\x4A\xF8\xBD\xBD\xDA\xF0\x47\x7B\xBD\xF0\xB9\xBD\xDA\xF0\xEF\xF7\x2A\xE1\x3F\xEE\xD5\xC8\x2F\xEE\x55\xC5\x97\xF6\xAA\xE2\xB1\xED\x3D\xAA\xB8\xBC\xBD\x47\x09\x4F\x74\x2F\x24\x0B\x5F\x78\x72\xAF\x2A\xBE\x77\x7B\x8F\x5E\x7C\xFF\x5E\x25\xBC\x7D\xAF\x12\x9E\xDA\xAB\x84\xA7\xF7\xEA\xC5\x8F\xED\x45\x87\x0F\xEE\x55\xC5\x87\xF6\x2A\xE1\xA7\xF7\xEA\xC5\xCF\xEE\x55\xC5\xDF\xDB\xA3\x86\x13\x7B\x54\xF0\xEA\x3D\xBE\xFF\x07\x7B\x3C\xFF\x6F\xF7\x68\xDF\xD7\xED\x51\xFF\x37\xEC\xF1\xFD\x7F\xB7\x47\xFD\xAF\xDB\xA3\xFC\x66\x8F\xEF\x1F\xDE\xA3\xFE\x47\xF6\x28\x7F\x73\x8F\xEF\xCF\xDF\xF8\x7B\x92\xFA\x6E\xD4\xE0\x03\x0B\xEA\x2C\xD9\xB0\x3F\xAE\x9A\x8A\x42\xFD\x4E\x72\xF8\x10\x39\x1D\xD5\x1A\xB4\x3B\x7C\xA2\xD6\xD3\x8A\x70\x82\x1A\x5E\x08\xAB\x28\x0F\xB2\xCE\x68\xA6\xB3\x51\x00\x41\xF9\x7F\x90\x86\x4B\x3F\x7F\xB3\xFB\x39\x77\x77\x1C\xB8\xFF\xFB\xBB\x3A\x4D\xF6\x69\xA5\xCC\xB6\x47\x7B\xA5\xDD\x5B\x00\xEA\xA0\x0E\x8E\xE0\x1A\xEA\x92\x86\x56\x2F\xD9\x32\x51\x7C\xFC\x8A\x4E\xDA\x64\xA2\x6A\x45\xDB\x5A\xB9\x91\x38\xF7\x38\xB5\xC9\x48\xA9\xA8\x5A\x52\x38\xB7\x7B\xE4\x0C\x2D\x80\x65\xE3\x1E\xD9\xA8\x2D\xB7\x86\xDC\xEF\xF8\xF8\x0B\x2C\x27\xF2\x45\xFD\x89\x50\x9A\x0C\xAF\xE8\x8D\xD7\xE1\x82\x49\xDB\xCF\xCF\x19\x65\xB7\xBD\x5B\x16\x85\xC5\x33\x8A\xBF\xDA\xA0\x7D\x76\xE0\x3B\xA0\x8F\x82\xB9\xB9\x6E\x98\xDD\x3B\xE1\x35\xF8\x80\x22\xD8\x40\x49\x7B\xE8\xE0\xAE\xC1\xFE\x71\xB8\xBA\x3B\xDF\xB8\xFF\xFE\x4C\x81\x03\xB8\x84\xFF\x9D\x9E\x4A\x57\x35\x77\xB5\xFB\xB4\xDF\x69\x3D\xE8\x34\xBE\xD3\x54\x9C\xED\x93\xC8\x27\xCD\x23\x3A\x80\x99\xF2\xC9\x1D\x6A\xC9\x5B\x1B\xB8\x29\x21\x74\x64\xA7\xCE\x83\x99\xD6\xEC\x9E\x26\xE5\x4A\x97\xB8\xE1\x06\x59\xAB\x6D\x35\x56\xD1\x96\xEE\xDE\x50\x69\xF7\x06\x08\xD7\x41\x41\xB8\xEE\x54\xB3\x46\x69\x34\x42\x3E\xBE\xF8\x83\xEB\xD7\xAF\xC7\xA8\xCF\x91\xF9\x7F\xB3\x36\x40\x39\x2B\xE9\xC0\x84\xFA\x06\xC6\x65\x6B\x0C\x92\x8C\x35\x86\xA0\xFD\xB0\x2E\xAE\x3B\x64\xC6\xBB\x61\xF7\xA9\x6D\x59\x7F\xB4\x1B\xA5\xB7\xDB\x38\x45\x46\x8A\xC3\x2E\x53\x0A\xFE\x69\x6D\x41\x9D\xF1\xA1\xF0\xEB\xA0\xCB\x77\x5F\x6C\x41\x19\xC8\x2B\x98\x5C\xB5\xDB\x2E\x53\xC5\x93\x76\xEC\xDA\xBB\x3C\xD4\xD9\xAF\x68\xEF\xDC\xA6\x3C\x7E\x01\xC5\xE3\x19\xB7\xB5\xE1\xB6\x1B\x37\xE2\x06\x58\xAC\x9B\x8E\x3F\x39\x68\x13\xDB\x13\x4E\x6B\xBB\xCE\x51\x68\x16\x47\x31\x66\x9A\x44\xB9\x42\x2A\x1B\xF7\x06\x88\xD6\xE9\x49\x1D\x43\xB4\xBE\x56\x68\x9E\x1C\x09\xBF\x96\x52\xE8\x61\x8F\xE0\x11\x12\x3C\xA2\xB4\x30\xD1\xDA\xB4\x8E\x84\xE0\x11\x11\xDC\x64\x90\x42\x02\xE1\xB4\x4A\xC0\x08\xC9\x23\x6F\x44\x8D\x20\x01\xEB\x49\xDE\x1E\x5F\xCA\x94\x87\x00\x04\x87\xCC\x38\x75\xAE\x36\x74\xF6\x1E\x09\x6F\x3B\xB5\x31\xAD\xC9\x5F\x46\x83\x12\x94\x9A\xCE\x99\x5F\x09\xE4\x40\x2C\xD9\x04\x28\x3C\x6E\x93\x13\x31\x40\xBC\x60\xDC\xFE\xC0\x13\xB3\xA5\xE3\x90\x8A\x34\x82\x74\xB6\xC6\x28\x71\x48\x45\x3B\xAD\xCD\x3A\x83\xC2\x90\xCF\x35\x8D\x1E\xB5\xCE\xE2\x50\x9B\x06\x94\x87\x05\xBA\x9B\x60\x63\x08\x8D\x60\xD0\x3E\x2D\x8E\x0F\x0D\x81\x86\x33\x93\x6D\x72\x14\x36\xDD\x53\x99\x7B\x18\x39\x9E\x41\x67\x20\xEC\xC6\x42\x86\x2C\x71\x6F\xA0\xF1\x08\xFB\xE3\x11\xE2\x78\x84\x9C\xCB\x6B\x6D\x5A\x87\x32\x1E\x21\x8D\x47\x4A\xA9\x1F\xC0\x4E\xAB\x18\x52\x19\x8F\xD0\x33\x60\x08\x31\x98\x6E\x0A\x0C\xFB\xE0\x07\x2D\x1E\x8C\x0F\x41\xCE\xF1\x30\xA5\x32\x43\x42\xDA\x8D\x50\xB6\xDC\x59\x3A\x2F\x31\xF7\xAB\xF2\x87\xF0\xBA\xFC\x11\x72\x65\xFC\x21\x2D\x9B\xB2\xFE\xFF\x99\xEE\xDA\x93\x0A\xA5\x9D\xE9\x0D\x4C\xAD\x71\x59\x22\x1A\xD8\xDC\x66\x6E\x24\xAE\x0A\x75\xE8\x46\x27\x8B\x90\x89\x14\xB9\x11\x13\xF1\x0D\x38\xDF\xDD\x1B\x28\x5A\x8A\x38\x1D\xB4\x27\x98\xCA\x1C\x9B\x32\x6A\x92\x4F\x4D\x1D\x42\x08\x76\x6D\x5A\x5B\x21\x9B\x25\xB2\x85\x28\x11\x88\x5A\xE4\x19\xEC\xE1\x9E\x62\x72\xA9\x18\xA1\x5C\x8F\x3A\xB2\xC4\x4C\x8F\x90\xE9\x91\x31\x07\x37\xC8\x8B\x1D\xF0\x05\x83\x4A\x35\xEE\x4B\x94\xD9\x91\xDB\x6B\x21\xC4\xF9\x62\xE9\xFC\x19\x89\x47\xF1\xE2\x48\xC3\x1F\xD0\xB4\x8A\xCC\x73\x66\x2B\x59\x88\x0F\x3B\x16\x14\x01\xEC\xD1\x54\x40\xCB\xE6\x3E\x23\xD6\xA2\x81\x59\xC8\x8C\xC1\xA4\xCF\x86\x74\xFA\xE8\x1E\x26\x69\x05\x76\x7D\xAD\x08\x90\x94\xE4\x46\xD8\xE3\x37\x2B\xD4\x03\xC2\xBF\x9F\x27\x9C\x90\x5C\xE8\xD7\x71\x14\xB5\xD0\x48\xCE\x1C\x6E\x74\x47\x43\x2D\x2B\xED\x80\x94\x5D\xE3\xFB\x4C\x35\x16\x1F\xDD\xF2\x69\x62\xA7\x19\x1E\xE3\x9B\x8F\x8B\xC2\xD0\x49\xE6\x90\x24\x21\xD2\x2F\x9A\x8A\xD5\x0C\x2C\x21\xD7\x78\xAF\x4E\x50\x64\xB9\x41\x15\x66\x6D\x8A\x4A\x0D\xD8\xD3\x75\x3C\xE5\x08\xFA\x8E\xD4\xD1\x14\x68\xB5\x0D\xCB\x77\x90\x41\x2D\x40\x9E\xDA\x86\x18\xDF\x5F\x9B\x56\xC3\x09\xF0\x9E\x42\x8D\xB6\xCD\x51\x50\xEE\x4D\x84\xCA\xD9\x3A\x82\xF7\x38\x5F\x41\xB0\x5E\x87\x4E\xBF\xB1\x8E\xC6\x51\x14\x91\xA7\x7C\x46\xAF\xD6\x21\xB9\x85\xB1\xA0\x8F\x1F\xA4\xB3\xCE\x88\xA7\x5C\xBC\xAA\x69\x09\xC6\xB7\x62\x71\x12\xE3\x69\x23\xAF\x27\x14\x05\x0D\x11\x7F\x45\x60\x91\x31\x7F\x4C\x6F\x85\xFC\x56\xD4\x16\x2A\x13\x9B\xFD\x9D\xE9\xBB\x40\xEE\x40\x4C\x8E\x50\x13\x81\x11\xB5\xBE\x11\x14\x80\x1E\x42\xF2\x60\x61\xBB\x1A\x92\xAE\x5D\x55\x4A\x4A\x4D\x85\xA3\x6D\x9B\x6A\x24\x2B\x71\x5F\xEC\x5A\x8E\x84\x0C\xEA\x98\x5B\x9B\x9D\x2A\x14\x84\x0C\x1F\x16\x54\x39\x44\x95\x82\xB8\x4A\xFC\x13\x96\xF0\x1C\x47\x15\xC0\x68\xF0\x42\xEC\x3F\xD5\xB8\x6A\xF0\x0B\xDC\x7C\x2C\x68\x54\x29\xC8\xF8\xBD\x07\x29\x67\x39\x11\x10\x72\x7A\x83\x40\x64\xA5\x2B\x06\xA5\x28\x45\x63\x53\x57\x28\x1E\xB1\x0E\x61\x24\xC4\x32\xDD\x2A\x45\x1D\xEE\xC8\x68\x66\xC9\x48\x66\x21\xDB\xB0\x62\x17\xB8\xBC\x21\xA2\x80\x2E\xAF\x5E\x14\x50\x1C\x12\xE4\x7F\x4E\x59\x19\x88\x42\x09\xD3\x2B\x5B\x48\xA8\x68\x40\xA8\x88\x7B\x1B\x31\x2C\x8C\x1F\xAE\x2A\xA5\x4E\xC6\x3E\x16\x25\xEE\x8D\x56\xD2\x3D\xEC\xA8\x15\x31\x0A\x6F\x24\x3E\x73\x4C\xAD\xB4\xA5\x52\xC2\x54\x4A\x85\x4A\xD8\x89\x3A\x43\xB5\x56\xBB\xA4\x4A\xE9\xD4\x25\x23\x5E\xE3\xFA\x53\x6C\x33\x9D\x9F\x86\x4D\x1D\x63\xDF\x05\x06\x2D\xA4\xC6\x85\x1C\x81\x7F\xA1\xA9\x42\x2C\x59\x42\xD7\xA2\x53\x6C\x1B\x0D\x1B\x24\x01\xCA\x5D\x71\xBB\xCB\x70\xDF\x12\x62\x6D\x06\x52\x7C\x96\x52\x65\xD8\xFC\x2A\xCB\xD9\x3D\x2D\x72\xFA\x5B\xBC\x97\x0A\x51\xF3\x1D\x2F\x3B\xC5\x06\x39\x88\x9C\xFD\x16\x76\xAD\x84\x94\x1C\xC5\x51\x98\x30\xED\x39\x52\xD3\x76\xAE\x95\x69\x7F\x9A\x31\xC6\x42\xCE\x44\xD5\xD2\x79\xF6\x5A\x0C\x9C\xAA\x72\xCF\x2C\x7E\xE0\xB5\x9F\x26\x29\xBD\xC2\x96\xC1\xE1\x04\xB2\x44\x47\xFC\x98\xC7\x25\xE2\xD7\xFD\x4D\x86\xC4\x62\xA2\x77\xF5\xE1\xD3\xCC\x69\x8A\x5E\xEB\x03\x30\x08\xB7\xC6\x58\xBE\xC4\x28\x99\x2A\xA4\xC1\x08\x39\x0B\x51\x20\xD4\xE2\x72\xA3\x53\x84\x43\x68\x4F\x14\x11\x8F\x98\x74\x80\x46\x8C\x9B\x62\xA5\x03\xF4\x5D\x1E\x52\x9C\x51\x53\xD3\x37\x31\x51\x3A\x14\xC2\xA3\xB4\x41\x65\x3A\xC2\xB6\x06\x6D\x57\x12\xF6\xF1\xCA\x38\xF2\x5A\x35\x1C\x71\x87\xCD\xA8\x47\x10\x9D\x29\x02\x73\x14\x46\x3C\xF0\x29\x85\xA8\x51\x4E\x5D\x2B\xF2\x2D\x41\x6E\xA4\x20\x7F\x94\x12\x84\x6B\x7C\xA1\xA9\x23\xEE\x2B\xD5\xC3\x04\xC5\x5D\x44\x58\x8D\x38\x45\x07\xD5\x15\xC9\x1C\x45\xA2\x41\x7A\x62\x5A\xA5\x52\x7D\xCE\xDE\x36\x23\x0A\xFA\xF4\x03\xE6\xE7\x70\x2B\x18\xA4\xF7\x5D\x8B\x73\x1A\xB2\x11\xE8\xF2\xC7\x2E\xFA\xE5\x7E\x24\x6C\xE8\x6F\x92\xD3\x12\x4A\x9D\xE3\x45\x94\x87\x38\x40\x11\xC4\x85\x24\xAD\x09\x70\xB7\x32\x5A\x7F\xB0\x2D\xE7\x7D\xE4\x4F\x82\x65\x54\x01\x0A\x35\xFC\x45\x65\xC9\x83\x4A\xE5\x36\xE3\xD1\xD1\x38\x9A\xE6\x28\xA4\x4C\xA8\xC0\x13\x2A\xEE\x93\xEA\xC1\x96\x50\x29\x3D\xEA\x91\xEA\x54\x4B\xA8\x35\xCA\x38\x1A\xCD\x11\x8A\x87\xB2\xA2\xB9\x71\x82\x88\x49\x33\xE8\x04\xEE\xE5\xB3\xEC\x0F\x70\xE7\xCA\xE6\xED\xC8\x6F\x39\x83\x3A\xEE\x56\x13\x17\xF0\x5E\x80\x86\x8B\xE0\x42\x35\x65\x03\xA0\x78\x21\xCF\xCC\x29\x7F\xE6\x17\xA1\x07\x19\xE3\x99\x81\xEE\x68\xB0\xA8\x82\xD8\xE9\x16\x58\x44\x98\xC9\xBF\xA6\xE9\x35\x02\xF6\x50\x22\xC3\xFC\xB7\x54\xBC\xE6\x56\x51\xC9\x22\xC5\xE5\x13\x4D\xCA\x2E\x43\x88\x19\x6E\x04\x15\xAC\xB9\x60\x43\x2F\xFB\x78\xAF\x58\xDA\x79\x4A\x00\x19\xB8\xBA\xB6\x40\x11\x5C\x7D\x12\xC8\xAB\xBA\x99\x69\x19\x4F\x66\x67\x1A\x06\xBC\xB0\xFC\x89\x69\x99\x8D\xEB\x15\x01\xE2\x9B\x16\x73\x2B\xDA\xEA\x27\x99\x90\x10\x8C\x7F\xAF\xDF\x4C\xE3\xEB\x9E\xF4\xDA\x68\x7A\x6D\xD4\xBD\xD5\x01\xDB\x32\x11\x9D\x2A\xF0\xAF\xF3\xDC\xCE\x20\xCD\xFE\x24\x52\xC9\xB6\x53\x3E\x7C\xAC\xA7\x84\x38\xFD\xC6\x71\xC4\xE1\x81\x5E\x01\x91\xD8\xB8\x40\x24\x86\xA8\x68\x94\x68\x89\x06\x64\xD2\x53\x54\x78\xBF\x83\x6B\x95\xED\x51\x4C\x83\xE5\xEF\x34\xAF\x97\xB6\x55\x41\xA4\xCD\xB6\x2D\x56\xDE\xF2\xE8\x3F\xF4\x65\xE0\xFB\x11\x8A\x00\x9E\xF4\x36\x22\x54\x56\xC8\xFD\x97\x25\x58\x6A\x89\xFA\xAD\x23\xFC\xD2\x04\xF5\x6D\xBF\xCC\xF6\xCC\x43\xBE\xCD\x4A\x3A\xC9\x6D\xA6\xF8\x3C\xCD\x71\xA3\xB8\x22\xDA\x6A\x84\xCB\xA5\x7F\x22\xAB\x3F\xED\xD9\x79\x69\xED\x5E\x08\xFD\xA7\x5A\x5C\xDE\x71\x69\xD5\x7E\x69\x4D\xAA\x11\x28\x7E\x8F\x96\xD8\x56\x14\x51\x28\x39\xEB\x54\x9A\x57\x1E\xF6\x90\xF7\x5A\x9B\xE5\x27\x89\x90\x4C\x77\xCA\x32\x75\xB9\x23\xA6\x9E\x27\xE6\x1E\xAA\x08\xBF\x82\xA4\x89\x98\x50\xB2\xDB\xB4\x43\x0A\xE9\x1E\x85\x2C\x10\xF4\xBC\x95\x74\xC3\x78\xCD\x1A\x88\x28\x18\x7E\x83\x1B\x76\x0D\x81\xA8\x7B\x88\x2B\x8E\xFF\x5E\x67\xA8\x3B\xB3\x94\x62\xE6\xA8\x28\x94\x8B\xC8\x13\x79\x7F\x7E\xDB\x69\x20\x11\x19\xD6\x08\xE1\x37\x41\x11\x4D\xAB\x9A\xE8\x69\xD4\x8A\x58\xF0\xF1\xC9\xB1\x97\x8F\xDB\x64\xEA\x51\x13\x0D\x56\x29\x41\x9D\x21\x45\x5F\x12\x69\x45\x0F\x31\xA4\x87\x98\x0C\x35\x7C\x9F\x14\xBE\x4E\x5C\x72\xA2\x90\xDC\x3B\xE4\x47\xA9\x4E\x4A\x6C\x8B\x40\x2B\x45\x1C\xCA\x69\xD8\xE8\x49\x6F\x81\x3A\xC1\x6E\x94\xFF\x3F\xC1\xC0\x0A\xD8\x52\x27\xA0\x9F\x1D\x7A\xCE\x36\xE1\x5E\xEE\x68\x31\x1C\xF2\xCB\xF7\xE1\xF6\xD7\xDD\xCB\x51\x06\x44\x71\xDA\xE5\xBB\x5F\x7F\xEF\x27\xDF\xAB\xCE\xF0\x20\xBB\xE5\xB3\xE5\xDB\x2F\x4A\xC4\xFB\x61\xB2\xD7\xE6\x67\x79\xE3\xB7\xAA\x0F\xFB\xFD\xBB\x2D\xDF\xE1\xDF\xB9\x8F\x31\xE7\xC3\x55\x7D\x2F\x76\xEF\x41\xEF\x3A\x1A\x70\x9A\x7C\xF2\x20\xC5\xA2\x28\x4A\xD5\xE5\x67\x9B\x9A\x4E\x29\xC9\x61\xD2\xE2\x06\xC8\xD2\xD6\x81\x82\x21\x1B\x90\x6C\x20\x84\x63\x73\x82\x5A\x28\x1A\xDF\xA4\xDB\x02\xDF\xE7\xCB\xC3\xC2\x88\x0F\x92\x0E\xFF\x00\x49\x91\x75\x7B\xA5\x8B\x99\x09\xB7\xF5\x96\xD9\x26\xB7\x57\x7A\xEB\x3E\x0E\x09\x38\xA6\x98\xBF\x6D\x1D\xAE\x13\x87\x72\xB4\x8C\x6A\x88\xAC\x2B\xFA\x14\x59\xC9\x9A\x57\xA8\xE0\x2C\x2A\x9E\xAF\xDA\xB8\xFA\x38\x67\x4F\x04\xCA\x69\xF0\x2D\xCE\x6C\x36\x75\x7C\xD0\x04\x75\x7A\xF1\x58\xF0\x0F\x28\xC9\x6A\x02\xE9\xA5\x23\xC6\x6F\x25\x3F\xD0\x91\xC8\x4A\xDC\x9C\x28\xDA\xB4\x71\x71\x39\x03\x3B\xB1\xEB\x38\x45\x7D\xD8\xB6\x73\x96\x3A\xA7\xDB\x19\xC3\x92\x5B\x31\xB4\x41\x00\x6A\xDD\x0B\xC6\x60\x45\x3F\x84\x12\xDD\x99\xCD\x75\xDC\xD8\xC7\x1B\xCE\x20\x27\xFF\xE3\xF7\x0C\x71\x4A\x56\x28\x19\xB1\xD9\x6C\x56\x34\x47\x82\xD9\x56\xAF\xD4\xC8\x7E\x2C\x24\xC4\x27\x9B\xBC\x35\xD8\x50\x89\xCD\x17\xAA\x93\x41\x09\x7B\x62\x8E\xE2\x24\x9B\xB5\xB6\xF4\xDA\x13\x81\xAA\x43\x26\x10\x47\xB4\xC4\x1B\x6E\xBB\x19\xC7\xB8\x0A\xC4\x36\x1B\xF9\xD0\x45\xF2\xBD\x3C\x5F\x2B\x6E\x57\x50\xE7\xF8\xE2\x56\x33\x0E\x95\xD6\x3A\xA0\x4D\x57\xEE\xD4\x79\xF7\x45\x8A\xCB\x69\xA4\x7F\x5A\x00\x8E\xAC\x8B\x08\xCA\xE1\x99\x27\x7B\x5D\x8D\x57\xB4\xAD\x14\x8C\x2A\x9B\xA7\x19\xFE\xC1\x9D\xFA\xED\x45\x92\xC7\x99\xDB\x66\x7B\x06\x8C\xCE\xA0\x3C\x3E\x14\x10\x0E\x9D\x75\x40\x01\x76\xEE\xB7\x82\x53\x53\xFA\x71\x2D\x78\x00\x7F\xC4\x2E\xC2\xD1\xAF\x15\xD9\x09\x2D\xC4\x87\x02\x7B\xAA\x88\xE8\x49\xD8\x1C\x0A\x82\x07\x8A\x88\x7D\x61\xD6\x24\x9B\x81\x72\xEA\x7C\xE3\x07\x5E\x32\x1D\x7A\x59\xE6\x12\x3E\x61\x40\x46\x6F\x79\x23\xDE\x75\xC4\x62\x5A\xF2\x70\x12\x69\x16\x07\x4A\xD0\xD9\x16\xBD\x4E\xCB\x35\xF5\x05\x47\xE0\xE5\x0F\x16\xDA\x05\x10\xAE\xD3\xF6\x7D\xCF\x6F\x02\xDE\x6F\xAE\xB4\x20\x03\xDC\xB6\x1B\x7C\x2A\xFC\x84\x05\xB4\x80\x25\x24\x71\x57\x28\xE7\x8F\x2E\x7F\xE3\x12\x61\xA0\x3D\xBA\x51\x58\x10\x06\x48\x36\x78\x4F\xD3\xEB\x3D\x47\xD3\x27\x1B\xBC\xF4\xF5\x1E\xF0\x8A\xC5\x63\x63\xCF\x53\xCF\x2D\x05\x3E\x10\x97\x7B\x26\x6B\xDC\x9B\x70\x10\x88\xAE\xEE\x3F\x5E\xED\x5A\xD7\x4E\xEC\x75\x5A\x71\x76\x99\xD1\x76\x30\xA3\x6D\x3B\xA3\x6D\x37\xA3\x2D\x85\x84\xF2\x8C\xD6\x84\x0C\xC0\xE6\xAE\x3B\x18\x7D\x63\xE4\x71\xB6\x56\xF4\x01\x8A\x8F\x32\x79\x24\xA4\xD1\x94\xE8\xC2\x4F\x89\xE5\x9A\xE1\x89\xC9\xE4\x73\x28\xB8\xFD\x78\xA1\x29\x10\x16\x05\x5F\xF3\x1A\xB6\xFA\x1E\x20\x4C\xB8\x9C\x8F\x16\x0E\x50\x28\x3F\x97\x55\x2B\x64\x68\xE4\x3A\xD2\xC9\x94\xA8\x7D\xA0\xD6\xE5\xE4\xE3\x80\xC3\x55\xC6\xA9\x4A\xB9\xBC\xB2\xE6\xA8\x0C\x26\x6B\xF0\x15\xC1\x99\x2C\x93\x9F\x3E\xAE\xCE\x91\xDF\x20\xB3\x56\x98\x43\xB4\xEE\xA1\x28\x2B\x4B\xC7\x2E\xAC\xB3\x1C\x38\xC5\x51\x17\x95\x91\x16\x87\xD8\xE1\x13\x45\x28\x69\xAC\x69\x38\x09\x8E\x40\x11\x69\x18\xF7\xCF\x48\xFB\xEF\x43\xCE\x7D\x40\x90\x07\xD4\x46\xF6\xB9\x50\xAB\x2D\xBB\xDD\x1E\x4C\xD1\xEA\x7D\x10\x55\xCB\x23\x9C\x40\xE6\x0A\x61\xFC\xF0\x63\xBA\x09\xF1\x15\x02\x98\xED\x76\xA8\xC9\xA1\x80\x91\x09\x5D\x50\x51\x9A\xD7\x00\xBC\x1C\x48\x0E\x05\x8A\x10\xF8\xE9\x91\x06\xC2\x50\x6F\xF8\x1D\xA4\x5E\x73\x92\x81\x72\xA2\x55\x7D\x80\x57\xAA\x65\x08\x20\x7B\x65\x70\x07\x45\xFD\x66\x2E\x38\x79\xEE\x95\xC1\xED\xE4\xC9\x94\xF2\xC5\x52\x0B\x8D\xC8\xE8\x8C\xB2\xDA\x0A\xAD\xF3\x4A\x3B\x45\x80\x7C\x9C\x86\xB3\x71\x17\x1A\x7A\x8B\x80\x3B\x2B\x2D\xEB\x7B\x47\x48\xDE\x78\x25\x74\xBA\x45\xC8\x90\xC9\x8A\xBE\xE7\xAC\x7B\x55\x83\x6C\x57\x27\x04\xED\xA6\xCF\x50\x86\x4C\xB3\x59\x87\xBC\x00\x57\xC4\x18\x50\x8D\x58\xD7\x01\xDC\x0C\xD2\x46\xB1\x5B\x7B\x81\x61\x72\xEF\x69\x64\x1B\x84\xD3\x5E\x0E\x32\xF8\x44\x45\x3B\x0D\xDA\xE9\x07\xA6\x52\x89\xE5\x4A\xF4\x66\x1D\xF5\x2B\x59\xF1\x95\xAC\x2C\xA8\x64\x85\x2B\x79\x79\xAF\x92\x48\x2A\x71\x3B\xBE\x18\xEA\xEC\xE1\x4A\xB3\x39\xEF\x70\x8B\xCD\xC6\xC5\x64\xA4\x76\xB8\x6B\x81\xA4\xB9\x0D\xFC\x31\xC6\x92\x98\x41\x57\x44\x99\x95\x99\x9A\xD0\x4B\xE5\x8F\x5F\x24\x5C\x01\x3F\xE2\xB6\x4E\x04\xA0\x0B\x92\x46\x00\x2B\x35\x24\x67\x04\x43\x10\x67\xCA\x32\xC7\xD0\x25\xA0\xD7\x65\xC3\x29\xC9\x78\x29\x25\xE8\x01\x88\xCA\xB7\x3F\xD6\xC2\x52\x51\xDE\xC4\xF2\xF7\x2F\x09\x2C\x95\xC6\xCF\xFC\xE9\xDE\x5C\x83\x34\xB7\xA6\x1E\xA1\x22\x7E\x5C\x80\xFB\x51\x77\x66\xD7\xB4\x51\xA7\x72\x7C\xB2\x87\xAD\xF1\x6C\xD0\xC2\x08\x42\x7C\x2C\x38\x62\x4E\x41\xEC\x96\x84\x59\x62\x37\x6D\x5C\xB0\x1A\xE0\xAF\x15\xBC\x67\xF0\xD7\xE1\xF6\xD7\x7D\xED\xAF\x6B\x81\x44\x8C\xC7\x0C\x15\x1A\x03\xE1\x0B\x53\x89\xE0\x38\xE7\x42\x4C\x67\xBD\xB8\xC1\x71\xFA\x44\x4D\x71\xAF\x27\xCE\x81\x75\xC1\x49\x88\xDD\x33\x41\x03\x21\x50\x3E\x81\x53\x1B\xB5\x99\x3A\x05\x11\x98\x69\xF9\x41\xB1\x3E\x90\x88\x95\x68\x98\xA0\x7D\x9F\x92\x44\xB1\xBA\x46\xB2\x6A\x66\x4A\x39\x55\xB1\xAC\xBE\xAF\xC2\x35\x11\x55\xAB\x19\xFD\x90\x9E\x1E\xF6\x4F\x67\xB5\x47\x7A\xBA\xE2\x9F\xAE\xCC\x3F\x75\xCF\x06\xDE\xAD\x2F\xC8\x2E\x6A\x15\x6E\xFB\x53\x9D\x15\x1E\xF3\xFB\x6B\x0D\xAA\xE1\x43\x9D\xA3\xAF\xE6\x64\x5B\xEA\x95\xC1\x8A\xDB\x96\x90\x27\x54\xF9\xEE\x63\x7D\xF7\xD5\xA4\x4D\x26\x60\xA7\x14\x9E\x15\xD6\xA8\xFC\x56\x21\x18\xB0\x0F\x90\xB7\x73\x37\x0B\xAD\x33\x9B\xC2\xD7\xA1\x97\x40\xF7\x52\x4A\x8F\x90\x75\x28\x47\x9B\x95\x75\x7C\x4D\x66\x1B\x2B\xC0\x9C\x55\x2F\xA2\x97\x6A\xC3\x5A\x73\x1D\xA2\xFA\xBF\xCB\x07\x10\x75\xF3\x8C\x3B\xC3\xDA\x34\x2A\x82\x19\x1F\x13\xDF\x0F\xBA\xE3\xAD\xB7\x2B\xA5\xB7\x21\x70\xCF\x7C\x86\xFD\x1B\xB9\x6D\xF7\xF0\xDA\xA2\xCA\x4F\x22\x2F\x0B\xF9\xC8\x58\xF6\x29\xF6\xCD\xC3\xC7\xE6\xA8\xCC\x3C\x50\x52\xA7\x46\xE1\xED\xC5\xD3\x43\x95\x62\xF5\xF8\x21\x7F\x96\xAD\xDA\xB9\xBB\xA2\xEF\xF5\x4F\xEF\x9D\x7F\x5A\xFE\x6F\x17\xB9\x47\xD9\xBF\x51\x2A\xC2\xD6\x7D\xA6\xD7\x3A\x86\xEB\x97\x16\x70\x73\x13\xEC\x28\x87\xC6\x12\x00\x9C\x6D\xEA\xD0\x7D\x92\xF0\x7E\x58\x03\x21\xD6\xFB\x64\x70\x76\xDD\x1D\x3E\xDB\xAC\xE8\x97\x0B\xDE\x70\xF9\x4E\x4F\xAA\x65\xD4\x84\x09\xCC\x88\x36\xDD\x8C\x8B\x66\x85\x60\xCB\x1E\x1D\x98\x83\xA3\x9D\x7E\x23\x96\x85\x9B\x8F\x00\xF4\x94\xCE\xA4\xD7\x9D\x6A\xDC\x7B\x05\x42\x95\x56\x44\x3F\x0C\xF4\xA5\xBB\xD0\x10\xA6\xDA\x8A\x5E\xA6\xDD\x56\xA5\x70\x97\x97\xF1\x7E\xB6\xFC\x94\xEF\x6E\x19\x40\x50\xFE\xD4\xC5\x8E\xE4\xDF\x6F\x3D\x8B\xB2\x89\x83\x3C\x02\x68\x0F\x49\xAC\x29\xE0\x84\x34\x93\x39\xF1\x01\xB1\x15\x63\x3C\x90\xA5\x05\xB5\xAD\x88\xC4\x95\x9C\xB3\x6F\x12\x92\x93\x9B\x9C\xF6\xE9\x58\x2D\x0F\x75\xE8\xEC\x26\x2D\xF6\x11\x29\x9D\xA4\xB9\x10\xF8\x6A\xEC\xF1\x04\x36\x7C\x46\x9A\x05\x3A\x72\xC2\x3A\x32\x19\xE7\xE9\x6C\x3B\xA1\xB4\x05\x60\xCF\x01\x36\x81\x4E\xC9\x62\x52\xBD\x32\x50\x12\xB4\x6D\xFA\x2E\x25\xA6\x75\x29\xE1\x7E\x1C\x0A\x0E\x1F\xF7\xB8\xBB\xD2\x42\x0A\xAA\x32\x2F\xB8\x85\x76\xD8\x42\x73\x83\x16\x86\x92\xA9\xEC\xA0\xB6\x47\xB4\x6D\x33\x58\xF4\x08\xA6\x5F\x6C\x82\xE9\x1B\x34\xC7\x8A\xA3\x07\x37\x27\xFB\xE7\x7C\x40\x4D\x21\x80\x3F\x49\x2B\x89\xA5\xCD\xDC\x2C\x5B\x10\x53\x30\x4B\x50\x9A\x8C\xFF\xFD\x92\x40\xD0\xD2\x4E\xB5\xA5\x6C\x44\xEF\xB5\xB6\xDC\x7E\x8F\xFC\xB1\x85\xEF\x51\xBA\x6B\x8F\xC2\x01\x81\x53\xEC\x51\x74\x0E\x52\xDF\xA3\x14\x7B\x14\x66\x10\x4A\x9C\x29\xFB\x97\x70\xC7\xBA\x56\x12\xAF\xB7\x2E\x4D\x9B\x57\x8F\x7D\xFB\xCE\xCE\x4E\xB9\x75\x44\xFC\x4E\x84\x0C\xF8\xC8\xA9\x73\xBD\xA7\xBD\x22\x6A\xCE\x8E\x5D\x87\x44\x53\xDC\x56\xD8\xCD\x73\x10\x4A\x3B\xB2\xCF\x32\x52\x42\x3B\xAD\x38\x22\xB4\xA5\xA3\xBE\x01\x1D\x45\x5A\x97\x1F\xBF\x24\x21\x98\xC3\x16\x8A\x15\x69\x92\x0D\x0A\x0C\x6F\x50\x20\xED\x6A\x45\xB5\x30\x92\xE0\x05\x2C\x8F\x12\x39\x4D\xB0\xFB\x17\x4E\xEA\xA0\x77\xD7\xA9\x73\x84\x8D\x75\x50\x5B\xE1\x38\xE3\x9B\x51\x27\x57\xC8\xBC\x94\x48\xA6\x67\xD6\x85\x13\xC6\x56\x50\x2C\x7A\x93\x63\xD5\xE5\x2B\xE4\x65\x94\x10\x6E\x78\x79\xF1\x21\x82\x9B\x35\x3C\x00\x32\x84\x72\x5D\x2B\xD9\x2A\x10\x35\x89\x94\xCA\x93\xB2\x9B\x26\xB1\x4F\xB0\x97\x48\x94\xF2\x25\xE4\xD2\xFF\x86\x33\xCD\xBC\x98\xC4\xCD\xBE\x7E\xB6\xCC\x3D\xE9\xCB\x39\x1C\x5C\x50\xFE\xAA\x94\x1C\xCE\x95\x4C\x87\x87\x4E\xE9\x6D\x27\xDE\x57\xBD\x69\xC5\x29\x6B\x7A\x0C\x5A\xFE\x1A\x95\x63\x69\x6D\x6D\x13\x16\x60\x09\x5F\x3B\xDB\x36\x73\x83\xB6\x09\xB5\x5D\x50\x3E\x2B\xED\x32\x73\xED\xA2\x52\xDF\x2A\x9E\x16\x9E\x5D\x17\x89\x7E\xA7\x6F\x9A\x1A\xE2\x8C\x66\xCB\x7F\x7D\x49\x36\xEE\xF7\x02\xDB\xCB\xFC\xE3\x73\x2E\xE0\x87\x5E\x50\x68\x66\x4D\x8E\x6B\x45\x1E\xC2\xB5\x82\x9D\xE7\x88\x45\x38\xAF\x67\x8B\xE6\x71\x5C\xA6\xD4\x02\x41\x81\x12\xBD\xD2\x7C\x82\x40\x5C\x66\x80\x18\x2B\xEE\xB8\x0A\x14\x0A\x0A\xD4\xA0\x87\x92\x38\x1A\x52\xA7\x25\x4B\xE0\x3E\x1D\x30\x06\x7D\x70\x0B\x4A\x4A\x67\xC0\xFB\x3A\xFA\xEA\xB9\xEE\x2B\x50\xCE\x6C\xD0\x96\x42\xF5\xE0\x9F\x2F\x6C\x54\xD6\x25\x95\xF2\x26\xB6\x15\xFD\x10\xA8\xC6\xBD\x49\x54\x69\x7F\x7D\x61\xE6\x7A\x70\x61\xE5\x5A\xB9\x12\xB7\xBA\x56\x1C\x5F\x19\xF3\x99\x88\x6C\x9C\xDE\xA4\xD0\x86\xC1\xF7\x04\x3E\xAD\x7A\x39\x0F\xB8\xE5\xDF\x20\x7F\x5F\x87\x4A\x52\xC0\xEA\x7D\xF6\x47\x5A\xEB\x6D\xBD\xE5\x5A\x65\xB3\x65\xC6\xC8\x05\x0F\x16\xC1\xB1\x80\x65\x01\xAA\x61\xC0\x43\xCE\x89\x90\xFC\xFC\xE8\xD3\xB9\x8A\x21\x12\x7F\x99\x7B\xD9\x44\x27\x0E\x57\x57\x6A\x02\x6D\x6A\x97\x65\xF2\x01\x63\x66\x50\x72\x3C\x07\x41\xCB\x0A\xD1\x22\x56\x20\x4B\x04\x39\x1E\x92\xB8\xE7\x64\x73\x8A\xD2\x1A\xE1\xC2\x1C\x79\x76\x88\x9C\x3D\x5F\x07\x5D\x62\x5F\xEE\xEC\x41\x13\xCC\x8B\xBA\xA8\x13\x75\x2C\xE3\xD4\x8C\x8C\xD3\x64\xF3\x69\x5A\xEE\x93\xEB\x81\x8C\x8B\xE6\x64\x9C\x97\x69\xAD\x94\xBB\x74\xC4\x7C\x73\x06\x61\xF6\x9B\x9A\x92\xBA\x91\xD8\x68\x4F\x96\x48\x8E\x2B\x6C\x9F\xBD\xE2\xDE\xFA\x89\x7F\xF5\x3B\xE1\xD9\x5A\xB9\xE5\xF3\xA0\x1E\x95\x4B\xF7\xF4\xF7\xFD\xCC\x17\xDE\xEC\xAF\x80\x70\x09\x16\xBF\xF9\x68\xAD\x5C\xD1\xBB\xAE\x95\x9B\xE0\xA5\x4C\x36\xF3\xFA\x3A\xF4\xB9\xAC\x8C\xDB\x6E\x36\xF8\xAC\x5D\xE1\x7A\xCE\x70\xAE\x0A\xCC\x23\x15\xDB\x91\x18\x6E\x8E\xB8\x3C\x38\xEE\x37\xB9\x81\x77\xD4\x0C\x9D\x66\x90\xC0\xED\xA6\x05\xB5\x10\x2F\x1C\xF2\x11\x0E\xC1\x6C\x40\x74\xB2\x08\x89\x77\x0E\x9A\x04\xEC\xDF\x97\xB4\xD1\x01\x8D\x8E\xC9\xB2\xF9\xCF\xE8\xB0\x97\x1A\x12\x62\x43\xC8\x75\x22\x3A\x59\xD8\x7E\x21\xBA\x57\x08\x45\x93\xD1\xDC\xCF\xDE\x1B\x1B\xB5\x15\x6E\xEB\x37\x83\x46\x5A\x46\x2D\x85\x34\x52\x48\xEF\x42\xCB\xA8\x4F\xCB\xE1\x9B\x8F\xD6\x1A\x69\xD9\x5E\xD7\x1A\x69\xA9\x1F\xF5\xD0\xB0\x62\x13\x28\x3C\x4A\x3C\x24\xAF\xAF\x53\xB6\xC6\xC6\xD5\x88\xF1\xAC\xD6\xD8\x2E\xFF\x48\x35\xCA\x78\xF7\x3A\xF2\x24\x25\x7F\x4B\x7A\x9D\x02\x93\xE4\x96\xF1\x12\x34\x25\xB0\x0C\x86\x86\x59\x23\xAF\x86\xE4\x11\x3A\xEB\x84\xD1\xC9\xC2\xE2\x52\x6E\x12\x88\xFE\x41\x41\x08\x43\xE4\x5B\x44\xC5\xC7\x90\xE0\xB0\xD6\xFD\x6A\x08\xC5\x6A\xBB\xA9\xB2\x5E\x3D\x6A\x58\x0F\xA4\x90\x6D\x60\xC9\x7A\xAE\x64\x95\xB9\x65\xDE\x8C\xD6\x9A\x9D\x6F\xB4\x13\x38\xCD\x52\xA4\x0D\x07\x7E\x1D\x34\xC1\x11\x93\x30\x61\xFE\x96\x2E\xEB\x1C\x92\x9F\x78\x92\x52\x59\x39\xD5\xFC\x44\x5D\x7C\x23\xCE\x35\x6E\x8C\xDA\x84\xC4\x99\x35\x48\xB8\xB5\x2E\x38\x79\xEE\x9C\xE8\x8D\x05\xE4\x6F\x7D\x4B\x9D\xFF\xD7\x41\xB0\x73\xE6\x1B\x01\xFF\x06\xC1\x6B\x19\xE1\xFB\x3D\x84\xC1\xED\x82\x2A\xCB\x9C\x26\x02\x67\xE4\x06\x93\xAD\x4D\xEB\xD4\x29\x4F\xAE\x14\x8B\x24\xCF\x1D\x02\xF4\x4A\xCB\x1F\xA0\x9D\x34\x9B\x70\xAA\xC4\x63\x91\x70\x1A\x6E\x48\xC4\xEC\xB5\x56\x65\x7E\x4C\xF3\x9F\xA0\x0E\xEC\xDE\x8C\x94\x9B\xC1\xA9\x10\x20\xC3\x7E\xBD\x9E\x67\x55\xEA\x94\x03\x6A\xC1\xFF\xB0\xBE\x49\xA3\x3B\xE5\x93\x75\xDF\x0A\x1E\x41\x18\x41\xBA\x36\xAD\xB9\x76\x94\x61\xA3\xF2\xB7\xFD\xA6\x99\x99\x8A\x08\x53\x8F\x36\xA8\x8D\xF1\x06\x75\x85\x39\xEA\x0C\xB3\x17\x16\x2A\x8C\xD5\x3A\x36\x30\x73\xD5\xE4\x13\x12\x7B\xA3\x53\x22\x67\x73\xAB\x3A\x81\x11\x39\xE3\x7B\x8F\x85\x3A\x46\x89\x4C\x07\x16\xAC\x13\xF6\x6B\x8D\x61\xB4\xC1\x82\xB1\xCF\xC5\xB1\x67\xB3\xCE\xF0\x4E\x42\xD1\x27\x70\xE7\x43\xCF\x8C\x80\xCF\x94\x1F\xFB\x55\x4D\x60\xD9\xC9\x2B\x83\x56\x27\x7B\x3C\x31\xF1\xB6\xDE\x0A\x7B\x18\xB0\x3D\x14\x16\x5E\x6A\xFC\xA9\x8C\x8B\x37\x3C\x36\xDD\x1F\xBE\x6D\x90\x99\xC4\x67\xDC\x2A\x7F\xF1\xA2\xFF\xD2\xE0\xF6\xDE\x7A\x54\x95\xFE\x4A\x76\x8A\x7D\x6D\x7B\x4B\x98\xE8\xC7\x11\x05\x0E\xA3\x0C\x6C\x13\xB7\x88\xB4\xBB\xB7\x0E\xDB\xCD\x31\x3D\x8D\x6F\xBC\x40\xB5\x6F\xB7\x30\xCB\xF6\xBC\xC8\x52\xAE\xB6\x4E\x9A\x9A\x1D\x5D\x20\xFC\xDB\x01\x1D\xAF\xBB\xB0\xF9\xDB\x41\x40\x68\x5B\xB2\x07\x89\x37\xC8\x7D\xC5\x7D\xF4\x3D\x03\xF4\xEF\x7B\x21\x61\xE0\x3C\x3A\xC5\x75\xF6\xFC\xD5\x63\xFB\x1E\x87\xD1\xB1\xF4\xF1\x4B\x90\x1D\x53\x8F\x5F\x82\xF4\xFF\xAF\xF4\x31\x85\x6B\xCD\xB2\x2C\x7B\xCF\x06\x84\x4F\xEA\x96\x9B\xF2\xA3\x02\xCF\x13\x79\xEE\xED\x93\x27\xED\xC8\x53\x87\x84\x5A\x4A\x78\xFF\x22\xE6\x42\x6E\x7B\xD1\xD4\x63\x6C\xE1\x44\xCE\xC0\x02\x18\x1F\x0A\x6C\x35\xC6\xE2\xFA\x6D\xE5\xBB\xBE\xB0\xA2\xE9\x7F\xA3\x20\x14\x99\x4E\xBA\x3B\x1D\x63\x55\xC5\x6C\x09\x52\x61\xD9\xBE\xE0\x0B\x2B\x7B\x85\x2D\x24\x11\xBD\x50\x4D\x32\x82\xA2\x2C\xDC\x75\xB5\xD1\xD6\x45\x87\x69\x4E\x6D\xC0\xC4\xD9\xF3\xAC\xB9\xE0\x88\x92\xA0\x50\x5E\xBF\xE9\x8F\x32\x88\x65\x04\x52\x59\xFB\xE5\xBA\xE6\x84\x0E\x35\xE9\x1D\x29\xAD\xFD\x51\xB7\xF6\xFB\x8D\x49\x3D\x41\xFD\xD8\x36\x55\x08\x11\x0F\x75\xE4\x87\x9A\x3F\xA7\x93\xC7\x98\xE3\x08\x76\xEF\xD0\x84\x66\x00\xB3\x91\x81\x68\x6E\xCC\xC3\xFE\x98\xC3\x70\xAC\x79\x22\xDD\xEF\xB6\x1B\x0E\xA3\x7F\xEF\x8F\x0E\x8F\xD0\x64\x07\x55\x6B\x3E\xBB\x16\xB1\x1E\xF8\x8F\x56\xF5\xFD\x8C\xEE\x44\xAF\x22\x53\xD4\x5A\x5A\x03\x71\xF9\x0B\x17\x87\x81\xF7\x52\x20\xE1\x34\xD3\xE2\xDC\x69\x46\x41\xAB\xA7\x33\x80\x0E\x74\x00\x3A\x1F\xD2\x12\x03\x85\xE3\x6B\xFD\x01\x71\x6F\x16\x70\x28\x30\xB9\xE9\xC5\x1B\x12\x2C\x33\x3F\xF3\xC8\xD7\xC7\x32\xD7\x84\x0D\x45\x37\x94\xFF\xD2\x3B\xB6\x89\x0F\x96\x40\x22\x0D\x6C\x00\xAD\xF5\x68\x97\x0D\x50\x6D\x09\xD2\x95\x3B\x5D\x0F\x5A\x77\x5C\xE0\x7C\x28\x47\xDC\xE2\xBA\x66\x4C\x11\x33\xE5\x2F\xD8\x93\xCD\x94\xD5\xE6\x1D\x5C\xF0\x41\xF6\x7B\xBA\x75\x06\xBF\xB7\x36\xA2\x95\x5A\x7F\x42\xBD\x88\x97\xC8\x2D\x1F\xE7\x66\xC8\x79\xAE\x3A\xF1\x3A\xF7\xBE\x47\x2D\x0C\x69\x06\xC9\xBB\x87\x02\x7B\xD2\xCB\x45\x23\x85\xA8\x1B\x14\xC2\x95\x86\x99\x7F\x4D\x26\xB3\x61\x5C\x07\xB6\xBB\xEC\x56\x3F\x18\x9E\xBB\xFC\xCD\x49\xCF\xCB\x5D\x4F\xD5\x5E\x3D\x55\x3C\xE1\x39\x1D\xF4\x8D\x3B\xEB\x9F\xF3\x17\x10\x3C\x8A\xF3\x9F\x8E\x7F\xB3\x97\x4B\xC5\x2F\xA7\xAF\xDA\x9C\x80\xCF\xF0\xC1\x50\x7B\xFD\x4B\xB4\x78\xFF\xCA\x58\xE7\xDB\x7A\x6B\x2E\x23\x9F\xF8\xA2\xE9\x07\x04\xCE\xE4\x91\x0D\x0A\xE5\xD8\x79\xED\x06\x05\x71\x1C\xDD\x20\x78\x8A\xEB\xD7\x7F\x3E\x38\x51\x25\xF8\x2B\x78\x80\x60\x2A\x76\xCA\x53\xD5\x88\x9E\x98\x07\x08\xAB\xE2\xFA\xF5\xC9\x06\xFE\x73\xB2\xCA\xF1\xEA\x65\x1B\xEE\xFA\xCB\x4E\x56\x05\x68\xF7\xAA\x0D\xF7\xAA\x13\x0C\x60\xB1\xE8\xFF\xA0\xDC\x0E\x1D\x43\xA1\xFE\xE5\xAE\x73\xAA\x22\x85\x15\xA1\xF8\x2B\xD2\xCC\x5D\x9F\xF8\x7B\x13\xBC\x37\x3A\x4E\x37\xAF\x1B\x7F\xF7\xBA\xC1\xDB\x59\x91\x52\x14\xBF\x3A\x55\xC4\xA0\xDC\xD1\x0D\x54\x1E\xB5\x7B\x55\x35\x01\xE5\x5E\xB5\x71\x9C\xB7\x64\x96\xFA\x42\xB0\xA8\x6E\x47\x91\x25\xBB\xFC\x65\x3A\x71\xCF\x32\x88\xAB\x09\x8C\x8F\x17\x96\x8D\x10\x3B\xAF\xDD\xE0\x8C\x02\xD7\x5F\x46\x65\x5C\x7F\xD9\x06\x65\x6D\x9F\x40\xD1\xBE\xB2\xF3\xC8\x06\x84\xF4\x0E\x36\x92\x1A\x33\xD9\xA0\x95\x64\x02\x79\x41\x5E\x07\x9A\x31\x58\xA9\x2E\xF2\xD8\xB1\x2E\x6F\xCA\x0F\x7F\x07\x8A\xA8\x09\xF9\x54\x24\x72\x79\x52\x0E\xB9\x29\xB4\x6E\x72\xAE\x9E\xF8\xD1\x24\x48\xC6\x4A\xE5\x61\xE6\x9E\x9F\x85\x1B\x09\x5A\xFB\xF2\xC4\xA9\x07\x49\xE4\x29\xF6\x84\xC3\xF9\x54\xEB\xAB\xC7\xAE\xAB\x8B\xC7\xD2\xC7\xF1\x32\xA8\x0D\x5F\xAA\xC7\x2F\x1D\x31\x65\x6F\x21\x66\x9C\x57\x16\xCE\x62\x36\x58\xD1\xA5\xAC\x3D\x38\x9E\x38\xD5\xF0\x5F\xB2\xBB\x7E\x84\xD4\x45\x45\x8D\x2F\x9B\x55\xF2\x11\x58\x69\x80\x0B\x71\x3B\x9F\x25\x14\x09\x72\x1D\x41\x89\xFF\xF3\x9C\x87\x71\x45\xAF\xB8\xB2\x61\x1F\x24\x6C\xEB\x89\x42\x71\xEA\x16\x43\x64\xC1\x1E\x4E\x38\x5D\xA2\x96\xAD\xD7\x84\x8E\xB5\x39\xB5\x73\x72\x92\x3D\xA6\xC8\xEF\x09\xD2\xAB\x75\x89\x7D\x52\xEF\x3B\xB6\xEF\xF1\x4B\x50\xB6\x1D\x82\x41\x47\x12\x3A\xEF\xE4\x8E\x10\x61\x51\x01\x0F\x40\xBD\xE0\x0E\x30\x59\x52\x48\xFA\xA5\x31\x72\x26\x65\x32\x92\x64\x22\xDB\x94\x46\x91\xB7\xFA\xFD\xE1\xE1\xAC\x7F\xC8\x0B\x48\x80\x5A\x35\x14\x8A\xD5\x8D\x14\xF0\x9D\xDD\x06\x2B\x79\x31\x06\x2B\xB9\x95\xC1\xD2\xA7\x0A\x64\xC2\xAD\x6E\x6C\xD8\xCB\xAF\x9E\x70\xD3\xC1\x02\xFF\x22\xBC\xAE\x09\x07\x45\xE9\xAB\xBD\x21\x32\x57\xBB\x31\x1A\xB4\x9F\x61\x7E\x7A\x63\x44\xD6\x9B\xDD\xC7\xE8\xA6\xDB\xED\x93\x2B\x99\xD9\x31\xDA\xC6\xC1\x99\x78\x27\x0C\x3A\x30\x8D\xBC\x66\xFF\xBA\x7A\x82\xE4\x24\xDC\x4B\x2F\x5D\x05\xBE\x70\x02\x8A\x15\x0B\x7A\x48\x49\xD3\x82\x9E\x29\x38\x10\x5F\x3E\x51\x5C\xA8\x30\x50\xA2\xC6\xEB\x8A\x04\xF2\x64\x55\xBF\x8E\xFC\x60\x80\x59\xBF\x9B\xDF\x7E\xCE\xF6\x84\x85\xF3\xA7\x15\x5D\x2D\x13\x6F\xF3\xDA\xD7\x2A\x35\x13\xDE\x30\xD0\x98\xFB\x91\xF7\xB6\xAF\x49\xDF\xF6\xB5\x0F\x26\x7E\x53\xD8\x9A\x9C\xCA\x2B\x45\x00\xFB\x08\xA1\x56\x1F\x85\x7D\x57\x6A\x55\x04\x50\x76\xA5\x57\x13\x28\xC9\xF2\xA4\xF5\x51\x28\x7B\x96\xA7\x89\x74\x52\x89\xF6\x29\xD7\xF5\x84\x40\x6D\xEA\x09\xEA\x57\x8A\xF4\xAB\x49\xCF\xF2\x54\x7A\xCB\xD3\xBE\x9E\xE5\x89\x07\xC0\x7A\xF2\x2B\xA2\x75\x8A\x84\x0A\x67\x47\x00\xC9\x96\xF2\x18\xF0\x73\x47\x76\xDA\x21\x81\xDA\x61\x98\x74\xC3\x90\xCA\x30\x4C\x24\x7F\x38\x7E\x6D\x65\x18\x66\x64\x2C\x79\x3E\xBC\x5A\xE0\xB3\x7B\xA3\x91\xF5\x13\xC9\xFE\x67\xB6\x4A\xF2\xEA\x43\xA9\x1F\x79\x05\x75\x92\x1B\xA6\xD7\xA2\xD0\x0F\x59\xD4\x0E\x59\x38\x33\x64\xF4\xCF\xA9\xE9\xCC\xE0\xF5\xCD\xFA\xE4\xC3\x6B\x67\x06\x2F\xBE\x22\x41\x01\x9C\x3F\x9D\x07\x2F\xEE\xEA\xA9\x34\xC4\x34\x78\xE4\xE9\x36\x30\x1B\x86\x83\xC1\x93\xEB\x5A\x8B\x01\xA6\x1B\x3C\xDD\x1B\xBC\xD8\x0F\x5E\x34\x37\x78\x41\x37\x7B\x14\xC5\xC8\xA8\xF9\xD9\xA3\xC0\xCA\xEC\x51\x3C\x7B\xD4\xDE\xB3\xC7\xB6\xB3\xA7\x1D\xB6\x2C\xFB\x5F\xC6\x86\x68\xBF\x4D\x2B\x5F\xBF\x18\x31\xE9\x2A\x76\x76\xF2\x06\x7C\x21\xBF\xE9\xB6\x01\xA4\x4E\x7D\x54\xB9\xDF\x55\x24\x1F\x99\xDC\xFD\xF3\x19\xC2\x91\x53\x33\xE4\xB6\x48\x6E\x23\x89\x76\xC0\xF0\x49\x94\xED\xCA\xAD\x42\xB0\x44\x6E\xDA\x34\xF5\xC8\x2D\x7B\xBC\xF6\x24\x4A\xAE\x87\xE7\x7A\xC3\x43\x3D\x08\xC0\x7A\x72\x9B\x1E\xB9\x65\x67\xBC\xA3\xDC\xCF\xA9\x9E\x54\xD3\xAD\xDF\xC7\x70\x20\xF4\xE2\x81\xD0\xBB\x0F\x44\xD4\x1B\x88\xA8\x1B\x88\xCE\x1A\xD1\x0D\xC4\x0C\x9F\xEF\x4A\xE8\xE7\x95\xFB\x93\x1B\x12\x3A\x94\xA2\x0E\x9A\x6F\xDE\x9B\xC8\x6A\x17\x22\xAB\x19\x22\xCB\xF5\x1E\xC7\x7D\xBB\x11\x59\xF5\x88\xDC\xFD\x0A\x99\xDC\x37\x20\x7A\x48\x44\x0F\xE7\x89\x1E\xB6\x44\x0F\xE7\x0D\x1C\x7B\x12\x3D\x94\x73\xA3\x9B\x27\xFA\xE7\x95\xFB\xC2\x4D\x71\xF7\x4D\x11\xFD\xAF\x81\xB3\x17\x12\x7C\x8E\xD3\x87\x03\xC0\xE4\x57\xC3\x95\x43\xAF\xEA\xD7\x4D\x16\x2D\x1C\xBA\x5D\x38\x26\xF3\xEB\x46\xD8\x1B\x82\x70\x7E\x08\xC2\x8E\xEF\xFD\xC7\x37\x92\x2D\xD2\xEA\xAF\x2C\xD9\x02\x81\x97\x89\x37\x94\x2B\x5F\x76\x16\x87\xC0\x8B\x8C\xF9\xF1\xAF\xB5\xD7\xE3\x76\x69\x99\x5A\xDC\x32\x75\xCB\x2D\x53\x8B\x5B\x26\xF3\xAA\xD7\x32\xBD\x88\x27\x5F\xB0\x1C\xBE\x79\x7E\xCC\xB2\x3B\x67\xCC\x43\xA0\xC8\xCD\x6F\xD7\xFB\xD3\x80\xED\x4A\x2B\xEC\x2F\xA3\x18\x42\x82\xBC\x2E\xB3\xC7\x42\x6D\xB6\x95\xB7\x12\x3C\x1F\xF4\x12\xF7\x23\x8F\x7D\x43\xB7\x67\xC7\x9E\x6C\x37\x63\xAD\x02\xBD\xB7\x83\x88\x9C\x00\x75\x1E\x0C\xFD\x13\xDA\xDD\x3C\x18\xD8\xA5\x5C\xB3\x21\xE0\xCB\x56\x05\x58\xF7\xA9\x00\xF7\x10\x9F\x0E\x64\x17\x8E\x93\xED\x50\xF0\x75\x55\xE4\xBA\xCC\x02\x73\x02\xB7\x75\x40\x11\x7D\x25\x82\x68\xB7\x09\x1F\x83\xAE\x38\x68\x9C\xB7\x51\x21\x6E\x40\x22\xAE\x57\x76\xBB\x60\x21\x3E\xA2\xEF\x07\x0B\x66\x55\xDF\xC7\xD1\x17\xE1\x4C\xD5\xE2\x7C\x61\xC9\x1D\x74\xD0\x02\x76\xBD\x08\x21\xEC\xB5\x80\x4A\x5A\x19\xF8\x60\x54\x71\x46\x5B\x25\x97\x60\x77\xEF\x6B\x3A\xE4\x23\xAA\xFD\xE5\x64\x88\xE0\x87\x2B\x83\x87\xF6\xA0\xB9\xEF\x88\x59\xE6\x5F\x2B\x47\x0C\xEE\xC0\x9E\x09\x1A\x54\xE9\xDC\x72\xC3\xF6\x8B\x0F\x5D\x94\x7E\xE3\xDE\xCA\x1E\xD4\xAF\xAE\x62\xDC\x8C\x9D\xAA\x4C\x2F\xFB\x3D\x6A\x91\x9D\x27\x48\xAE\x5A\xB6\x03\xEB\x9E\xF7\xDE\xAC\xD9\xCF\x97\xA4\xE0\x19\x7F\xFC\xF2\xBC\xEA\x65\x1B\xEF\xB1\xEE\x70\xF8\xFB\xE9\x0F\xF7\xFE\x3F\x91\x18\xF9\xF7\xF6\x40\x25\xA9\xB1\x61\x14\x2F\x15\xA3\x6C\x3C\xD9\x7F\x5B\xB9\x2F\xD7\x4B\x33\x06\xCD\xFC\x76\x9A\x6E\xE5\xBF\x20\x43\xD1\x92\xD8\xD6\x56\x74\x59\xFE\xCC\x45\xF1\x3A\x79\xC4\xE9\x93\xC5\x6D\x10\xBB\x0F\x28\x42\x95\x2A\x7F\x96\x83\x58\x70\xAF\x9A\x34\x7C\xBF\x23\x6A\x0C\xF1\x41\xF3\x01\x75\xC4\x94\xEC\x9A\x1C\xB8\x84\x43\x3F\xF9\x61\x70\x50\x8B\x73\xF3\xB3\x58\x1A\x95\x02\xB1\x27\x32\xDF\xEE\x01\x57\x61\x61\xCF\xAA\x23\x3A\xC8\x6F\x23\x61\xD5\xF3\x46\xAC\x13\x4A\xF0\x2E\x6F\x46\xC8\x81\xFB\x21\x74\x39\x0E\x13\x0E\x0C\xB9\xFB\xC6\xEE\x30\x35\x32\x99\x69\xC0\x61\x88\x1D\xD0\x13\xDD\x6F\xBA\xA6\x87\xD0\x6B\xDD\xE1\x86\xDE\x94\x06\x26\xBB\x37\x10\x34\x57\x4D\xA0\x52\x61\x81\x0D\x9E\x6D\xF1\x5C\x7B\xF7\x41\xE8\xCA\xB6\xBD\x09\xB6\xCD\x56\x29\x47\x0D\x84\x94\x94\xD5\xE7\xDB\xC7\x86\x55\x23\x3E\x16\xAA\x33\x0A\x1B\x4F\xE5\x04\x8D\x93\x86\x42\xD6\xE2\xF0\xCE\x6F\xD0\x52\x7F\x58\xD0\x13\x23\x19\x44\xDE\xC9\x50\x1F\x85\xF4\x0A\x97\xC9\x45\x8C\xD8\x43\x43\x4E\x72\xC9\x91\x28\x83\x94\x96\xE9\x1C\x5F\xEE\x2D\xD3\x19\xE4\x03\x47\x0D\xB9\xAE\x33\x46\x73\xC8\xBA\xE3\x88\xAC\x5D\xA6\x47\xDE\x07\x2D\xF5\x3E\x68\x55\x4A\x46\xA7\x4E\x7C\x79\x6A\xD2\xC1\x68\x52\x04\xF9\xFE\x2C\x03\xFD\x0A\xA5\x71\xEB\x47\x7E\xF1\x43\x72\xC6\x4C\x68\x0A\xA9\x53\xFC\x06\x68\x77\x2D\xD8\x70\xD1\xF9\xF2\x9F\x13\x6F\x97\xFD\x02\x54\x5B\x00\xC5\xCB\xF4\x8B\xB8\xA6\xD8\x03\x1D\x5F\x98\x7D\xF6\x50\x8F\x33\x9E\x9D\xAB\x82\xB8\x41\x3F\xA7\xB8\x1B\xF1\x8A\xBE\x46\xC6\x9A\x49\xAF\xE2\xE7\x77\x6D\xF9\x6B\xA5\xE5\xCF\xF7\x8B\xFD\x69\x6A\xF9\xB8\x9B\x95\x54\x92\x53\x1B\xE5\xAF\xD2\x93\x82\x30\x64\xCB\xE3\x05\xAD\xFD\xCE\x52\x60\xA1\x24\x2B\xA8\x04\xA2\x81\xAC\x65\xF6\x44\x11\xB8\x3F\x7E\x7A\x70\x74\x13\xD1\xB7\x8C\x71\x91\xB3\x27\x67\xB1\xEA\x59\x95\x62\xFC\xC7\x99\x97\x27\x73\xDF\xB7\x84\xD0\x2B\x7A\x89\xCF\x6A\x86\x6D\x63\x5E\x2D\x88\x0E\x9F\x52\x55\x82\x7F\x9F\x53\x55\x8E\x7F\x9F\x55\x74\xF8\x08\xB1\x7B\x8A\x07\x22\x61\x4F\xDA\x9D\xC1\xD5\xE7\x03\x91\x2F\xCF\x0E\xE5\x4B\xBE\xAA\xDF\xAD\x20\xC6\xC6\x3E\xA5\x58\xDE\x3C\xAB\x8E\x98\x47\xF8\xE7\x53\xEA\x88\xD9\x51\x3C\xC6\x44\xD0\x1D\xD5\x70\x61\x32\x87\xE3\x15\xFD\x8C\xB4\xE7\x17\xA4\x3D\xD7\x54\x55\x08\x14\x02\x45\x3F\xE0\x22\x46\x2B\x15\x63\x83\xA8\xF2\x1A\xCB\x42\x17\x90\x89\xDD\x05\xE5\xBF\xE2\xF1\x66\x9C\xBD\xCF\x05\x9D\x38\x50\x07\xF5\xEB\x50\x5C\x71\xC6\x2B\xFC\x40\xB9\xA0\xFC\x30\x8D\xD6\x48\x7C\x93\xCB\x3A\x59\xD1\x73\xFE\x81\xA2\xC4\xE4\xA3\x0C\x12\xCA\x5B\xED\xEB\xD5\x47\x51\x7C\xE5\xE4\x04\xE1\x82\x93\x55\x81\x97\x09\x83\x01\xB7\xB8\x23\xAD\xF8\xCF\xD9\xED\xF9\x23\x74\xB2\xAC\x9D\xA5\xE3\x97\xDF\x0A\x4E\x15\x21\xF6\xF0\xA3\x1C\x48\xC3\xB7\xD9\xA0\x22\x17\x75\xE4\xFE\x2C\x78\x80\xE2\xA4\xE9\x63\x2C\xE4\x3F\x8B\x7E\x12\xBA\xDF\x42\x49\x02\x45\x95\xA1\x5E\x51\x54\xF8\xD2\xF3\xF4\xB6\x93\x1C\xFF\xAA\x3D\xAC\x51\x14\x0A\x91\xB1\xE9\x89\x37\xAC\x7C\xE3\x97\x7C\x78\x64\xD2\x70\xF0\x79\x42\xFD\x42\x71\x71\xD2\xA7\x08\x63\xEA\xCD\x11\x36\x4F\x5B\xD2\x79\xCF\xD4\xDD\x24\x6B\x9E\x22\x5F\x3D\xA1\x76\x9B\x67\x4F\x28\x99\x68\x4F\x50\xE4\xE1\xCF\x51\x2B\x13\x1E\x2D\xED\x14\x1D\x4B\x4E\x79\x89\xCB\x63\xBE\xAD\x9C\x1E\xDC\x66\x33\xA7\x4B\x9C\x1D\xDC\x0E\xF9\xB6\x75\xC9\xE0\x36\x5B\xE3\x5C\x29\x97\x86\x2F\x41\x2E\x75\xB6\x98\x0B\x54\x6F\x59\xC6\x0E\x3D\xAF\xBC\x12\xF1\xCF\x46\x2A\xF1\x0A\xC4\x53\xBA\x75\xE0\x00\xE3\x7E\xD2\x23\xFD\x19\xF7\x3E\x25\xC1\x46\xC6\xBD\xBB\xFB\xF9\x79\x25\xA1\x45\xC6\xED\x68\x9F\xCB\xD2\x7D\x51\x35\x28\x90\xFF\xC2\xD0\x13\x7C\xF7\xED\x8A\xFE\x9A\xCF\x28\x30\x3E\xE6\xE4\x6D\xCA\x7D\x51\x8B\x43\x92\x2D\x9F\xBB\xC4\xD0\x74\x4E\xAD\xEA\x5F\xC0\xB7\x82\x57\x06\xD7\x14\x18\xB0\xAB\xFA\xC3\xCA\x5D\x0B\xFC\x9B\xBE\xEA\xB7\xA9\xC6\xDD\xD7\x1C\xBB\x8C\x0B\x07\xD6\x63\x39\xCC\x89\x82\x10\x0F\xB7\xBF\x56\xDA\x5F\xD0\xFE\x5A\x6E\x7F\x95\xED\xAF\xC4\x77\x09\xEB\xFB\x98\x02\x5B\x7E\x42\x1C\x0B\x82\x43\x81\x96\x3C\xEC\xC6\x69\x6C\x1C\x9D\x33\x1E\x31\xCF\xCB\xCE\xF6\x98\xFA\xFA\x22\x70\x4F\xBD\x25\xEA\xB6\x4C\xFA\x8A\xFB\x33\x82\xAD\xDF\xD9\xD9\x49\x40\x1F\xBB\xFC\xE6\x0B\xC7\x92\xED\x2B\xA0\x8F\xFD\xA7\xC7\x76\xAE\x05\x5F\x3B\xAD\xC8\x1E\x6B\xB1\x9B\xAB\xFA\xB2\xE6\x70\xFE\x4F\x61\x77\xD5\x2B\x83\x67\x28\x2F\x72\xF9\x4E\x3A\xF5\x0A\x57\xF5\x93\x5C\x51\x78\x5C\x12\x8B\x19\xF7\x24\x8E\x4B\xF9\x6B\xB4\x23\x8A\xDA\xEB\x8F\xD3\x35\xA1\x7B\x3C\xA9\xAA\x10\xFF\x7E\x8A\x13\x17\x3E\xC3\x82\xCF\xB8\x27\x74\x4F\xF0\x19\x8E\x23\xA3\x3F\x4F\x68\xCE\x36\xBA\xB9\xAA\x3F\x84\xD4\xBF\x4C\xB8\x86\x6F\x53\xA8\x2B\xF3\xF7\x41\xF9\x09\x12\x51\x61\xF9\x97\xA4\x35\x9B\x43\xC1\x35\x45\x44\xF9\xA2\x2F\xFD\x72\x4F\xAC\xFA\xEE\x49\x77\xCC\x41\x73\x59\x1D\x31\x6F\xF2\xBE\x3D\xCF\xA9\x3A\xE1\x06\x4A\x74\x3F\x8E\x76\xCE\x3E\x65\x9C\xD6\x8B\x1E\x3F\xAB\x3C\xBE\xC0\x7C\xBC\x95\x85\xE4\xD5\x83\x78\x2B\x33\x1F\x6F\x65\x56\xF4\xC7\x54\x3D\xA2\xDD\x7F\xD2\x06\x5C\x25\xFD\xF8\x29\x22\xFD\xC7\x14\x5B\x04\x46\x1C\x38\x99\xB6\x11\x57\xE9\xEE\x5F\xB4\xA1\x8D\xD2\x1D\x6A\x78\xC6\x1C\xF4\x2C\x31\x49\xBA\xAA\x9F\x53\x44\x08\x89\x37\xA4\xE6\xF8\x86\x88\x33\x79\xC2\x59\xC2\xEE\x13\x1A\x32\x31\x7B\x23\xA4\xCF\x82\xE1\x4D\x06\x18\xF7\x99\xC1\x53\xAC\xE9\x9B\x99\xB8\x6F\x3A\x62\xEE\xE3\x5F\xDF\x7C\xC4\xAC\xF0\x48\xCB\x00\x62\xD9\x54\x84\x2C\x57\x7E\x9C\xC8\xA1\x88\x19\xA2\x4E\xDB\x22\xCD\x41\xF3\x84\x3E\x62\x2E\x2B\x6C\xEC\x87\x95\xA4\x1E\x57\xD3\x57\x06\x94\x73\x1C\x22\x50\xD3\x3A\x62\x58\x48\xF5\xCA\x80\x50\xD7\x57\xF4\x65\x85\x8C\x9A\x88\xE6\x68\x68\x9B\x48\xB5\x97\xCF\x49\x9D\xCB\xD2\x83\x4F\x0D\x7A\x20\xBB\xA8\xA4\x5F\xFD\xD7\xE1\xFE\xA9\xE5\x98\x64\xA6\x2F\xCB\x1C\x34\xD4\xAA\xCF\x33\x2D\xAF\xB9\xED\xFA\x0A\xF9\x2B\x60\x0F\x70\xB0\x72\xF9\x3C\x28\x3F\x49\xCC\x41\x14\xC7\xD6\xFD\xD4\x45\x91\x3A\x4F\xB5\xB8\xFF\xB6\x0B\x92\xFB\x43\xA3\xA2\x5E\x9E\xE5\x41\x72\x73\x76\x8E\xB1\xE2\x34\xFE\x40\xD1\xCF\x3A\x66\x51\xC1\xF9\xB6\x66\x3C\x0E\x42\xFC\x9F\xA2\x7F\x43\x1D\x66\x60\x5F\xA1\xA2\x3A\x3C\xCE\x70\xF5\xA4\x2B\x32\x74\x45\xE8\xD4\x89\xE9\x8A\x0E\xCA\x3F\xBF\x44\x76\xB7\x9C\x76\xF3\x04\x85\xA2\x1A\x9E\xE8\x64\x81\x78\x85\x8A\x9C\x16\x16\x67\x98\x9E\x9C\xC3\xE3\xA8\x60\x46\x10\x70\xB6\x32\x4E\xF1\x66\x05\x0C\x6E\xB7\x2D\x43\xE1\x5E\xBF\x6E\x36\x9C\x5E\x9B\x72\x1A\x50\x5F\x15\x27\x79\xA1\x6C\xF0\x04\x38\xC9\x25\xAD\xF9\x3D\x28\xEE\x5C\x5F\xA1\x74\xF9\xF8\x77\xC8\xDB\x56\xE6\x7F\xCC\xDC\x4F\x39\xCC\x56\x74\xCE\x77\x39\xA3\x99\xA5\x9D\x31\x41\x72\x50\x83\xD7\x06\x5D\x9B\xED\x58\x45\x2A\x5F\x4E\x47\x2F\x4A\x7A\xA9\x89\xC4\x44\xC7\x0C\x15\xC5\xA7\x07\xA0\x01\x40\xC1\x96\xB8\xA5\xBC\x2C\xAD\x22\x34\x29\x69\x82\x21\xDF\x78\xC7\xE5\x94\xFF\xB3\xBC\x91\xF9\x86\x53\x30\x7A\x46\x4E\xAF\x7E\x9D\xCB\xFE\xF2\x05\x0E\xF4\x64\x30\xD0\x73\x23\x4D\x79\xF6\xE7\x86\xBB\x91\x80\xC1\x85\x43\xFE\xF1\xDD\x86\xDC\x0F\xF4\x66\x43\x20\x39\x2D\x03\xC8\xD8\x07\x95\x91\xB4\x81\x7F\xF5\xB1\x77\xDB\xCD\xA9\xBF\x96\xE1\xFF\xF8\xDF\x90\xE1\xFF\xE0\x44\xB5\x69\x6C\x9E\x0F\xD6\xEB\xB0\xCB\x62\x23\x51\xA7\x91\x8F\x3A\x55\x2B\x3A\xA9\x63\xB6\xD3\x50\x70\xF9\x52\x9D\x36\xF5\x88\xA2\x4E\x33\x1F\x75\x3A\x82\x6C\x10\x75\x2A\x81\xF0\xEF\xF4\xF6\x8F\xE5\x5A\xB6\xE7\x04\xA1\x4D\x81\x35\x4B\xB8\x74\x10\x90\xD8\xB2\xF8\x2E\xD7\x23\x88\xD7\xEB\x78\x10\x75\x1A\x4F\x21\x85\xA8\x1F\x75\x3A\x1A\x44\x9D\x8A\xA7\x2E\x47\x9D\x62\x45\x55\x92\x41\x52\x45\x90\xB4\xF9\xC3\x18\x1F\x61\x97\x20\x4C\x06\xE4\xB6\xE4\x33\x0A\xCA\xE5\xD4\x1B\xE5\x21\x6D\xD6\xEB\x64\xD0\x9A\x64\x0A\xCA\x2D\xD1\x33\xF2\x38\xC9\xCB\xDF\xF1\x3D\x5C\xA2\xDA\xDB\x2E\x46\x19\x44\x90\x10\x91\x62\x4F\xA4\x04\xE2\x96\x48\x75\xE2\xB6\x71\xBF\x28\x50\x54\x89\xB4\x2F\xC1\x75\x73\x55\xBF\x1C\x12\x5C\x07\x81\xFF\x2C\x41\xE2\xCA\x7E\xA3\xB9\xD7\xCB\x84\x94\xB0\xAA\x97\x39\xF2\x27\x01\x1E\x1D\x4A\x28\x9A\x0C\x83\x82\x93\x5E\x50\x70\x9D\xB0\x17\x6F\x17\xC2\xC9\x9B\x23\xCB\xD9\x91\xA9\x68\xA0\xE1\x62\xE0\x84\x88\x5C\x21\x58\xA7\x3E\x40\xE7\x65\x38\x76\x11\xED\x0B\xA8\x61\xD2\xA6\x10\x92\x83\x3A\x41\xBD\x34\xA4\x7D\x3D\x6E\x2B\x43\x88\xF6\x5D\xF4\xB8\x4A\xB4\x97\x78\x36\x18\x58\x7E\xC2\x83\xE6\xA1\x23\xC2\xCD\xA8\x39\x9D\x91\x0A\x08\xB0\xD6\xDB\x28\x16\x54\x72\xB4\x5F\x09\x59\x2A\x42\xF7\xAF\x83\x06\xA2\x7D\x8F\x73\x75\x89\x3B\x30\x57\x5D\x22\xD5\x2D\xA1\x6A\x79\x28\x38\xC3\x5E\x8F\x89\x5B\x6A\xEA\xA4\xCF\x1D\x49\xC7\x1D\x3A\x83\x94\x75\x7C\x3F\x2E\xEE\xE5\x8D\x9C\x24\xC7\x28\x3A\xA2\xB1\xD6\x2A\xC0\xBD\x5C\x7E\xB6\x2E\xCA\x1F\xA4\xF9\x3D\xE6\x70\x00\xB7\xD9\xD4\x13\x97\x3F\x8C\xBC\x1B\x6F\x90\x39\x68\xCC\x28\xDF\x01\x44\xEB\xD5\x08\xAF\xCC\x51\x88\x58\x0F\x8F\x5C\x8E\x7B\xE6\x11\x8E\x6A\x06\xA3\x33\x95\x81\xAC\x1A\x71\xF4\x34\xF9\x07\x3D\x47\x19\xD6\xC7\x50\x08\x2A\xA3\xFF\xEC\xB9\xA0\xF1\x3F\x9F\xE9\x7E\xDE\xDF\xFE\x3A\xDC\xFE\xBA\xA7\xFD\xB5\xDC\x74\x95\xFA\x5F\xCF\x07\x28\xAD\x47\x12\xA0\x05\xE3\xD6\x43\x3E\x71\xD0\xA0\x40\x69\xFB\x7C\xDC\x83\x38\x8E\x18\x56\x93\x8C\x94\x31\xB2\x60\x84\x62\x5E\x62\x3C\x2D\xE7\x93\x95\x0D\x71\x1B\x3D\x52\xC5\x28\xB0\xF0\x7A\xBB\xE9\x04\x48\xF7\xBC\x8E\x4F\x16\x5A\xD8\xA0\x24\x36\x68\xCD\x8F\x2F\x87\x8C\xB0\x9D\xE2\x6E\x4C\x43\x88\xC5\x18\xE9\x19\x02\xDF\x87\x68\xDF\x77\x0A\x1F\xE0\x57\xD8\x56\xFA\xD0\x73\x00\x67\x20\x9F\xE5\x8F\x94\x5F\x54\x8D\x44\xE9\x2E\x73\x70\x0A\x16\x51\x45\x8C\x94\x99\x4A\xC3\x72\xF2\x7F\x87\xDD\xD8\x13\xC8\x28\xC9\x0F\x0F\x0C\x1E\x2E\x1D\x31\xE5\x80\x77\xA1\xA1\x77\x09\x34\x73\xB4\xEF\xF2\x4D\x4C\x16\x1A\x04\x8E\xD5\x41\x6E\x24\x61\xC4\xC0\xED\xEF\xE4\x9D\xAE\xEA\x24\x6E\xDD\x17\xAF\xD4\xF8\x64\x81\x84\x8D\x58\x70\xF4\x84\x6C\x7A\x13\x42\xF6\x78\xA1\xFD\x61\xCF\xCD\xCB\x1E\x5C\x1C\x22\xC1\x9E\x47\x99\x93\x9F\x6D\xD8\x84\x82\x4C\xC6\x52\x33\x90\xB9\x18\x79\xD3\x77\x3B\x7D\x13\x86\x80\x59\xE5\x08\xE8\x65\x8A\xF0\x69\xC9\x54\xA7\x8B\x88\xFD\x90\x74\x9E\x97\xB1\x85\x02\x7D\x45\x2F\x73\xE3\x6F\x41\xA6\x8F\xBA\xDE\x8D\x16\xC8\xF4\xB8\x2F\x6E\x6E\x24\xD3\xD3\x59\x81\xBE\xCA\x8D\xC1\xBE\x11\x3A\x68\xEF\xE4\xE3\x8A\x56\x09\x05\xDF\xED\x1D\xB7\x2B\x36\xB3\xEE\xBC\x07\xF7\x8F\x9A\xA2\x78\x6F\x29\x9C\xB8\x8E\x7A\x91\xF4\x12\x36\xEF\xF1\x69\x08\x16\x20\x21\xB5\xAB\x8D\x8D\xA5\x39\x62\x67\x62\x63\x17\x62\x8D\xA5\x1C\x1B\x9B\xB6\xB0\x00\x29\x01\x5A\x41\x72\x8E\x42\x0A\x18\x6D\xBC\x17\x1B\x2B\x95\x67\x1D\x06\x76\xD4\x80\x11\x7C\xE7\x99\x60\xD9\x9F\x36\xDA\x76\xC7\x94\x30\xA7\xDD\xCA\x19\x2C\xC3\x7F\x98\x15\x9D\xF0\x3E\x35\xF7\x78\x80\xF7\xCB\xB1\xBA\x50\xE4\x5E\x7C\xF8\x10\xC3\x9D\x30\xC6\x87\x60\x6A\x08\x8A\x87\x41\x09\xEC\xC1\x52\x24\x02\xF6\x1E\x06\xE8\xB8\x9F\x92\x06\x97\xBF\x7E\x49\xDC\xBB\xEF\x29\x7F\xBB\xFB\xF9\x69\x6F\x6E\x59\xD1\xF7\x1C\x0A\x34\xD9\x14\x97\x7D\x58\xDA\xB1\x00\x05\x85\x22\xAD\x0D\xB4\x9B\x34\xE5\x93\xDF\x21\xAA\xE3\xA1\xA0\xAC\x43\x9F\xC0\x84\x33\x06\x1E\x0A\x26\xEE\x3A\xE7\x61\x21\x48\x13\xB1\x28\x4B\xBE\x0F\x42\x6D\xA2\x7B\xCB\x1C\xF9\xA6\xCE\xE1\x6D\xFB\xF2\x20\x58\xD5\x07\xC8\xB6\x68\x71\xE6\xD2\x35\x64\x1C\x9C\x8F\x6F\xF7\x50\x4C\xEE\x91\xC3\xCE\xE0\xA0\x39\x55\x47\x64\x00\x3A\xB6\xB3\x73\x4D\xEC\x9F\x11\x92\xA8\xAC\x15\x0D\x04\x11\xF3\xD4\x94\xBC\xDD\xAF\x5E\x78\xF3\xB1\xCB\x6F\xBE\x50\x47\xC7\x02\x5C\xB0\xBE\x76\x4A\x31\x45\xD8\xC7\x7B\xC0\x40\x74\xC4\xBC\x16\x15\x53\xE8\x14\xD3\x31\x8E\x41\x00\xAA\x7C\xF6\x22\xE7\x9D\xC1\x21\x69\x2F\x9F\x57\xCA\x6E\xF3\xF7\x4B\x1E\x10\x45\x71\x0C\xA6\x5E\x77\xFA\x8D\xB5\xA1\x08\x66\xB1\xEA\xF6\x81\x49\xAC\xB3\x4D\xAD\x7D\xC6\x13\xA4\x90\xBC\xAF\x4F\x53\x84\xF4\xDF\xF2\xE8\x9A\xE4\x97\xCB\x07\xD7\xEE\x93\x82\xF2\x66\x57\x75\x29\xC9\xB1\x4F\xCC\x96\x4E\x08\x80\x9C\x35\x3B\xF3\x08\x2D\x0C\x60\xCA\x5E\xF7\x84\x57\xB6\xEE\x2E\x34\x4E\x9F\x77\xDB\x8F\xA2\xDC\x58\xD5\x49\xE6\x77\x59\x33\x50\x4F\x99\x0D\x82\x20\x3B\x70\x03\x85\x95\x1F\xDA\xFE\x43\x3B\xF3\x50\xF7\x1F\x76\x67\x73\xD9\xA7\xC3\x7E\x96\x0D\x89\xFE\x6E\xA1\xAD\x54\xBB\xE7\x62\x54\x18\x4E\x3E\xE3\x91\x3A\xC9\x18\xD8\x36\x93\x93\xB2\x70\xE8\xB8\xC4\x5B\xCC\x84\x8F\x87\x2E\xA1\x7D\x99\xA0\xAA\x11\x8E\x67\x2F\x7C\x9C\xAE\x2F\xCC\x5C\x0F\x2E\x6C\x6B\x5B\x2D\x3D\x30\xA8\x6D\x6A\x9F\x16\x58\x32\x2B\x38\xBD\xE9\x03\xD4\xBB\x12\x64\x83\xD7\x0F\x20\xC7\x61\x09\x8F\x17\x86\xE1\x54\x0D\x84\x14\xE7\xC6\x64\x78\x7D\xAD\x39\x00\x32\xA2\x3E\x6C\x78\x8E\xA6\x75\x30\x02\x35\x13\x52\x4C\x87\xD3\xEC\x3F\x63\xB0\x44\xCA\xD6\x49\x6E\xCF\x8C\xE0\x62\xDB\xC4\x2E\x8F\x78\x9F\x4C\x0B\x04\x4C\x63\xF8\xFC\xD0\xB4\xF0\x93\xD4\xFE\x3A\x66\xEC\x91\xA2\x45\x7C\x37\x55\x28\x40\xE2\x1C\x41\xC4\x10\x78\x04\x9C\x65\x0E\x9A\x84\x82\x5C\xFF\x3E\xF5\x2B\xE4\xCD\x03\xC3\xFB\x42\xDC\x90\x6A\xD2\x51\x91\x6E\x30\xAC\x82\x34\xB8\xC0\xFD\x6A\xD6\x8E\xAE\x0B\x16\x8E\x2E\xC5\x54\xF5\x73\x65\x10\x62\x50\xEC\xB3\xFF\xAF\x68\x0B\xF1\x46\x7D\xE3\x5E\x29\xEE\xD5\x71\x1F\x06\x75\xB3\x9D\x50\xB3\x9D\x50\x6D\x27\x18\x6A\xA5\x4F\xA2\xAE\x5B\x59\x96\x5D\xCE\x55\xD1\x8B\xC9\x54\x73\x99\xF1\x03\x3E\xFA\xF8\x64\xC0\x98\x93\xDA\xFD\x28\x6E\x71\x4A\x11\x1F\x2D\xC3\x7B\x41\xE1\xE5\x0B\x35\x2C\xE7\x38\x1E\x0F\xB9\xFB\x9A\x82\x27\x76\x88\x42\x04\xB5\xA6\x2D\xC2\xB6\x8F\x1A\x5A\x1E\x2D\xE8\x75\x36\x9F\x50\x12\x0A\x9F\xD7\x33\xA0\x25\x8F\x6D\x0C\x96\x00\x4B\x09\x2D\x18\xA2\x46\x96\x40\x0B\x21\x2A\x67\xEA\x8D\x4E\x41\x3C\xAD\x63\x3F\x37\x35\xC5\xAC\x7A\xD8\xB9\x3A\x81\x08\xF7\x6F\x58\x4B\x35\x62\xE4\xFE\x8D\xA6\xCE\x68\x16\x8E\x7C\x88\x3B\xEE\x92\xEB\x5C\x72\xCD\xC6\x1B\x14\x53\x9B\x91\x57\x08\xAA\x7C\xEB\x95\xC5\x2B\x5E\xBC\x85\xDA\xD8\xF1\x4A\x7B\x00\x5D\x4B\x20\x19\x60\xCF\x54\x05\x44\x95\x85\x82\xC6\x2A\x83\x51\x53\x59\xC8\xDD\x01\x96\xB2\xDD\xD7\x94\x5E\x40\x8E\xC1\xF8\x7A\xC6\x2F\xC0\xE5\x0D\xD9\x41\xDA\x6B\x5A\xD7\xCA\xEE\x7A\x09\xAF\x97\xBA\xEB\xE5\x86\xD6\xC3\xF6\xFA\x00\x5E\xB7\xFA\x1D\x2E\x1A\xD4\x5A\x68\xEA\x58\xB2\x85\xB3\x67\x78\xDE\x21\x31\x67\xAC\x1F\xD0\x59\x68\x7F\xCD\x06\x86\x53\xCE\xD8\x0F\x46\x56\x08\xDC\x16\xAF\xEA\x24\xB7\x99\x1F\x79\x82\x5D\xB6\xB4\x4E\x30\x13\x08\x8C\x3A\x23\x63\x9D\xE1\x60\x5A\x8E\x41\x57\xEE\xF3\x2F\xE3\xE1\xE1\x9A\x95\x4B\x9A\xF2\x77\xFD\x91\xD6\xC2\xA7\xCF\xFB\x96\x7C\x95\xF9\xBE\xCA\x7C\x1D\xF3\x91\x01\x8E\x10\x8E\xB9\x26\x1A\x10\x3D\xAD\xB5\x1F\x10\x72\x63\xA0\x85\xC3\x43\x1B\x60\x7D\x82\x5E\xB8\x49\x43\x4B\x8A\x23\xC4\xCD\x2A\xEE\x60\xB0\x3D\xF9\x80\x05\x2D\x31\x20\x99\xBD\xCA\xFF\xF3\x62\xAB\x84\x7A\xCD\x85\x40\xD4\x68\x97\x4E\x89\x72\x88\x94\x17\x9A\x5A\x97\xFF\x97\x38\x18\xC9\x4B\xB9\xEE\xA5\x13\xA4\x4E\xC5\x72\xFE\x9C\x1F\x61\x3D\x2B\xA8\x08\x56\x5A\x02\xF9\x69\xDB\x1C\xF1\x8D\x25\x6A\x13\x93\x8A\x20\xFB\x97\x18\x18\xBF\xE4\x70\x7F\x42\x36\x05\x39\x0A\x8F\xF1\xE7\xAC\x06\x55\x66\x33\x88\x87\xAA\xD5\x22\xB3\x7F\x6A\x55\xB6\xAD\x8F\x0E\x7A\x95\xB7\x81\x98\x3E\x91\x55\x6D\x5F\x43\xD9\x00\x0C\xD8\xE1\x44\x0A\x09\xA4\x0A\x34\x98\x75\x01\x03\x02\xD3\x26\x70\xE2\x89\x44\x07\xD0\x11\xE8\x86\x89\x8F\xC5\x84\x4D\x1B\x2F\x61\xD7\xEB\x88\xC7\x2D\x9A\xD6\x51\xA7\x61\x45\x54\x7A\x6B\xD7\x0E\x71\x63\x89\xB5\x54\x29\x67\x8E\xD8\x68\xEA\x11\x2D\xA3\xA9\x9F\x48\xB8\x0D\xAF\xB3\x6E\x22\x45\xE4\x61\xD4\x99\x90\x34\x5E\x11\x32\x9B\x69\xCF\x97\x48\xF9\x89\x18\x3D\x5E\x32\x6A\xE9\x33\x55\x4E\x30\xEF\x39\x03\x6E\x42\x8A\xEC\x9B\xF9\x89\xD4\x7D\x4D\x38\xE9\xA6\x9B\x48\xD1\x50\xF7\x24\x73\x11\x98\x6E\x22\x45\x6C\x97\x2C\xBB\xEB\x25\xBC\x5E\xEA\xAE\x97\xF9\x04\xAB\xBD\x3E\x80\xD7\x07\xBA\xD3\x30\xE0\xD6\x02\xCE\x77\xED\x27\x52\xC2\x13\x49\x68\x3B\xE2\x89\xC4\x59\x1B\x66\x10\xD4\x13\x46\x1F\x55\xAD\x8D\xA4\x9B\x4B\xDD\xA1\xD6\x3B\x3D\x3F\x0C\x91\xF3\x4F\x10\x9F\x9B\xD6\xF7\xD6\x4B\xDB\x3A\x3C\x43\x0A\x5C\xD8\x31\x86\xF6\x12\xD6\x78\x09\x1B\x72\xCA\xC2\x70\xBD\x36\x28\x8A\x7B\xDC\x91\xB4\x68\x8A\x84\xEB\x46\x12\x96\x0D\x15\xF4\x1D\x33\x46\x38\xAD\xC3\x8E\x31\xC2\xBE\x84\xC5\x9D\x6D\x95\x90\x6D\x08\x75\xE4\x8D\xA6\x36\xEC\xD8\x8D\x9F\x53\x38\x9C\xA9\x2C\x68\x3A\x3D\x20\xFE\x18\xF5\xF8\x83\x0E\xCC\x91\x33\x22\xFF\x16\xB9\x8F\x86\xDE\x2A\xC2\xEE\x39\xA8\x4A\x57\x96\xBC\x74\xEA\x0C\x22\xE4\x8E\xAC\x8A\x84\x3B\x0C\x72\x47\x04\x23\xCF\x1D\xDD\xD7\x14\xD4\x17\x76\xDC\x61\x67\xDC\x43\x2C\x72\x47\xD8\x71\x87\x45\xEE\x08\x3B\xEE\xB0\xC8\x1D\x61\xC7\x1D\xE4\x1F\x1A\x76\xDC\x61\xD9\x66\xDB\xD9\x70\x90\x3B\x42\x72\x64\x10\x6D\xB3\xCF\x1D\x3E\x39\x61\x8F\x3B\xFA\x9B\xA9\x76\xA2\x12\x78\xF3\x80\x41\x2C\x59\xB2\x7A\xA7\x9E\x5F\x15\x18\x5F\x15\x18\x5F\x15\x18\x5F\x15\x18\x37\x2D\x30\xDE\x61\x09\xAD\x97\x11\xA5\xF1\x8F\x39\x0C\x41\x0B\x4E\x1D\xB4\xF0\xD3\x81\xFB\x85\xA0\xBD\xF9\xB1\xEE\xE7\x73\xDD\xCF\x4F\x77\x3F\x83\x57\x06\x49\x97\x7A\x51\xB1\x1F\xE9\xB5\xA0\x71\xE4\x59\xC6\xAD\x31\x14\xCC\x5D\x5B\x47\xBA\xE1\x89\x29\xE1\x01\x53\x5A\x12\x54\xE8\xEE\xF3\x3F\xAE\x91\xD7\x5F\x43\xB3\x99\x78\xE5\x2A\xA5\xA0\xA8\x63\x67\xC9\x26\xF7\xB6\xED\x0B\xB5\x39\x62\x00\xF7\xF3\xC7\xB6\xBF\x96\x2D\x9F\x54\x3D\x27\x0B\x39\x62\x56\x18\x24\x63\x45\x2F\x81\x26\x72\xDC\xD3\x38\xBD\xE9\xCA\xE6\xEA\xB1\xA5\xAD\xAF\x1F\x7C\xE0\xEE\x11\x4E\x54\xDE\xA0\x10\x77\xCA\x2A\xED\x1E\x12\x31\x2C\xA5\xAC\x4F\x4A\xA6\xA6\x94\x8C\xE5\xBD\x47\xEA\x95\x01\xB0\x56\x78\x80\xF5\xC0\x65\xBE\x5A\x22\x10\xC1\xDE\x67\x39\xD3\x27\x9F\xA9\x19\x6E\xA2\x9E\xE4\xA6\xEB\x49\xDA\x7A\x4A\x08\xBB\x0E\x79\xD8\x76\xD5\xE1\xE1\x2B\xFC\xB4\x97\x65\xF2\x23\x61\x9B\x2E\x34\xAF\x95\x20\x44\x72\xF0\x2C\x25\x21\xF9\xCB\x1F\x9C\xC9\x25\xA1\xC5\xBA\xC8\xAC\xEF\x02\x3E\xAF\x31\xD3\x57\x17\x81\xFB\x81\x1F\x1B\xBC\x9C\x33\x42\x88\xB8\x4F\x0C\x2D\x37\x84\x12\x66\xF8\xA9\x9D\x7F\xDA\x43\xB7\x26\x6B\xED\xDF\xC4\x86\x09\xE4\x78\x1F\xD0\x92\xB8\x30\xF1\x9E\x67\x1E\x50\x9A\x5E\xBE\xBF\xE9\xB0\x30\x57\xF4\x7D\xB3\x9F\xCD\x7E\xE3\xC8\xB4\xE9\xD1\xD6\x7D\x3A\x9F\x0E\x06\xD3\xEC\x0E\x83\xA9\xD9\xCA\x78\x78\x06\x06\xF3\xF0\x0C\x0C\xE6\xE1\x3E\x0C\xE6\xE1\x85\x30\x98\xBC\xC7\x67\xD9\xC4\x96\xF6\x1E\x0C\xE6\xE1\x59\x18\x4C\x8F\x12\xE1\x61\x30\xEF\xCF\x6E\xB2\xAF\x54\x9A\x87\x1E\x3F\xBC\x08\x98\x7C\xC9\x3F\x5D\x9A\x7F\x9A\x7D\x6F\xA2\xF6\xF7\xFC\x83\x6C\xDF\x9A\xD6\x1D\x99\xD5\x61\xD3\xE9\x2B\xEC\x12\xBB\x51\xC5\xD0\xB9\xEF\x54\x49\x8B\xF0\x56\xA7\x9C\x65\xC2\x05\x84\x85\xC7\xFB\xC3\x9C\xA5\x6E\xC1\x0C\x34\x86\x14\x69\x31\x71\x1C\x6E\x91\xC9\xEB\xA5\x0B\xAA\x7D\xE6\x28\x8C\x21\x6D\x04\x7C\x15\x4A\x01\xBA\x2A\x8E\x17\x1A\x42\x50\xAF\x2E\x38\xE7\x05\xFE\x26\xB8\x17\xCA\x03\x02\x99\xD3\x9B\x90\x34\xB5\xE2\xF3\x4C\xB7\x7D\x82\x2C\xDD\xDB\x9C\xBD\x6E\x1F\xBE\xD6\x50\x73\xF4\x51\xC8\x57\xF4\x43\xF5\x7E\x01\xE9\xB9\x0D\x5B\xBC\xE4\xE2\x0D\x0A\xC8\xE3\xF3\xFC\xB1\x0D\x94\xD2\x2A\x83\x94\xC2\x4A\xF7\xC3\xD2\x30\xC1\x4E\x7D\xFB\x6C\x0E\x9E\xFD\x70\xFB\x2E\x39\x78\xA0\x70\xEA\x04\xEC\x23\xB8\x3A\x5C\x83\x75\xA5\xF2\x08\x5B\x71\x5B\x9B\x9B\x85\xB3\x31\xD0\x09\x39\x2F\x06\xDC\x58\xC0\xA6\x55\x4B\xDC\x5B\x6A\xE1\x85\x66\xAC\x03\xA5\xE9\x89\x85\xD1\x39\x0E\x39\xC9\x0F\x05\xEA\x38\x94\xE7\xAA\x32\x83\x89\x64\x90\xF4\xD6\xF3\x0C\x22\x9C\xC1\x3F\xD3\x4B\xFD\xE2\x14\xD1\x35\xE3\xCC\x1D\x19\xB2\x5D\x0A\x8C\x6D\xCD\x47\x9D\x6E\xFB\xC4\xB4\x22\x33\x8C\x81\x18\x46\xD3\xFA\xB6\x73\xFE\x04\x98\xCF\x3B\x38\x74\x51\x5A\x5C\xA7\x90\xBC\xA6\x50\x90\xA0\x42\x81\x63\x28\x83\x93\x12\x3E\x26\x83\x91\x84\xC8\xF5\xC2\x1D\xE4\x7C\xE2\x54\x55\x52\x12\x43\x6C\xAC\xA2\x77\xC9\x40\xAC\xC0\x4A\x82\x46\x05\x25\x95\xCE\x05\x65\xCC\x3D\xE4\xC2\x63\xB8\x72\x4F\x30\x0A\x3B\x7B\x9F\x17\x2B\x9A\x0C\x50\x29\x96\x2D\x2F\xB8\xAD\x96\xA2\x92\xC3\x50\x51\x66\x0A\x64\x43\x32\x09\x13\xAE\xBF\x8F\xB2\xD3\x9B\xF5\x18\x35\xA7\x25\x67\x69\x58\x4A\x1F\xB2\x19\x42\x89\x8D\xEF\xC3\x3A\x85\xDC\x90\x94\xC7\x6D\x45\x5B\xAE\x48\x93\xAB\x6C\x49\xB3\x3A\x83\x25\xAC\xAC\x84\x71\x53\x95\x92\x29\x54\xF9\x72\xB2\x45\xE5\x64\x52\x0E\xA7\x46\xE4\x9A\x95\x04\x26\xDC\xEC\x47\xB4\x26\xD6\x0A\xCA\x13\x85\xCE\x83\xBE\xED\x06\xBF\x80\xDB\x60\xE4\xCA\xCD\x73\xE5\x1F\x5C\x14\x28\x14\x2F\x72\x6D\xE7\x38\xA6\xB2\x3F\x33\xD6\x6C\xAB\xAD\xE1\xF9\x6A\x17\x05\x6A\x70\xB1\xA4\x60\xD0\x9C\x91\x31\x12\x81\x30\x22\x37\x04\x12\xBA\x87\xC9\xED\xCE\x2D\x37\xE5\x1F\x72\x88\x2D\xED\x98\x50\x0D\x96\x9C\x5A\xC1\x41\x03\x75\xCC\x36\xAE\x44\x86\x62\xA9\xC6\x4E\xD0\xB1\x7B\x9D\xA1\xCC\xB9\xB7\x79\x5F\x9D\x93\x1A\x72\x6C\x5B\x12\x79\x48\x16\x48\xC8\xDF\x7C\xC4\x00\x41\xAA\xE5\x8D\xA8\x07\xB2\xEC\x90\x57\x5A\xE0\xF2\xC6\x05\x90\x79\xDF\xE6\x54\x82\x69\xFD\xCA\xC5\x88\x33\xB8\x1C\x44\x8E\xC1\x0D\xD3\x76\x49\x48\x71\x49\x48\xDD\x81\x0A\x25\x59\x66\x28\xBB\xA4\x85\xA0\x71\x9B\x22\xB5\xFD\xF5\x9B\x66\xAE\x2F\xCC\x5C\xF3\x12\x81\x4B\x02\xE1\x97\x22\xBF\x67\xBC\x24\x10\x39\x32\x56\xB5\xAA\xA0\x57\x85\x7C\x62\x1B\xCA\x12\xDA\x06\x4D\x5A\x42\x16\xA6\x94\x82\xEC\x42\x23\x96\x43\xBC\xBF\xC2\x7A\x8C\xDF\x6B\x50\xB6\x38\x52\x80\x5B\x35\x93\x35\x6A\x0F\x3C\x65\x9B\xF2\x33\x17\x05\x28\x87\x0A\xB5\xDD\xD1\x6C\x90\x7D\xC1\xAA\x62\xBB\x55\x18\x8A\xC0\xFD\xDA\x77\xF5\x0E\xF4\x35\xEE\xA3\x3E\x11\xF0\x8E\x47\x73\x0A\x2E\x25\x89\xF2\x08\x34\xAC\x4B\x64\x6A\x49\x52\xB8\xA0\x62\xC3\xB3\xDB\xDA\xA8\x62\xF2\x19\x8F\x38\x20\x87\xF9\x9A\xBE\xC0\x25\x65\xE7\xF9\x77\x47\x67\x71\x17\xF8\x3D\xDF\x77\xF5\xF8\x59\x57\x9C\x6F\xEA\x85\x77\xE9\xCC\x8F\x72\x08\xC5\x60\x9D\xF6\x87\x85\x6C\x7C\x8D\xDA\x1C\x5D\x8B\x8B\xAC\x0C\x41\xC6\x7A\x61\xC6\xDB\xA6\xF6\x71\xFB\xCD\x39\x09\x20\x4C\x5F\x5F\x67\x7C\x6A\x38\x62\xD0\x56\xCE\x61\x3C\xAA\x72\xC2\x47\xE5\xFB\x8F\x54\xB9\x9C\x1C\xE6\x3D\x48\xDB\xA0\x05\xB2\x0D\xE8\xD3\xAA\x90\x83\x35\xCD\xA6\x86\x91\x4F\x81\x2E\x15\x18\x28\x36\x38\xD6\xDF\x70\xC9\x06\x4B\xA6\x84\x88\xF9\xC9\xC2\x48\x74\x61\xE0\xE9\xCE\x10\xA2\xFD\xA1\xE1\x44\x8B\xEE\x17\x87\xB7\x2C\x98\x13\x3E\x8D\xA0\x59\xD1\xC9\xC9\x82\xB3\x6A\xD5\x63\x3A\xE1\xC3\xCB\xB1\x0C\x20\xD9\x3E\x88\x72\x74\xDA\xB6\xA2\x03\x7C\x2A\x86\x03\xDE\x1A\x3A\xC5\x99\x1F\x61\x0C\x71\x1D\xE0\x22\x4B\x19\xF5\x39\xEF\x2A\x6E\xC9\x38\x04\x73\xF0\x0D\x44\x84\x8D\x46\x5B\xB4\x31\x25\xED\xD2\x9C\x2E\x5C\x0B\x83\x65\x19\x24\xD9\x07\x53\x3D\xDE\xD6\x6F\xEE\x41\xC2\x0F\xF9\xCE\x7A\xBE\x0B\x39\xD9\xAB\xAD\x25\xF9\x54\x84\x7C\x17\x73\xCD\xAC\x85\x44\x9E\xEF\x04\xE1\x74\x6B\x83\xD4\x8C\x08\x12\xC6\x78\x17\xBE\xA3\xAE\x86\x2D\x93\x84\x3D\xBE\x5B\x78\x97\x76\x9C\x94\x04\x2C\x45\xCD\xA7\xA9\x65\xE3\x19\xF3\x02\xDE\x65\x66\x59\xFC\x71\xD6\xBB\x6C\xDF\x11\x3E\xF3\x98\xBA\x59\x07\xD4\x9C\xBD\xBE\xCE\x3D\xE7\x65\xC4\x79\x05\x73\x5E\x81\x37\x98\xF3\xB2\x47\xAA\x42\xD0\x6E\x0B\xCF\x79\x6C\x08\x69\x7F\xE1\xA7\x0C\xBA\x27\x49\x24\x3B\xBE\x93\xE2\x43\x18\x6F\xF0\xD9\x6E\xC8\xE5\x86\x58\x2E\x21\xA0\x16\x0C\xB6\xEC\xD3\xDC\x5B\x72\xAD\x9A\xE7\xBA\x70\x86\xEB\x08\x14\x8B\x06\x5D\x73\x86\xD0\x04\x42\xE2\x3C\x92\xFF\xB6\x2E\xE9\x5C\x1E\x2F\x4B\x1C\x42\x23\x58\xA5\x2B\xFE\xBC\x3D\x24\xCE\x0B\xBA\x20\x43\xDB\x54\x51\xCB\x79\x25\xC4\x75\x8A\x9C\x97\x32\xE7\x51\x0A\xDC\x84\x33\x81\xEE\x1B\x7E\x03\xFB\x20\x61\xCE\x4B\xA1\x7C\xB0\x30\x92\x99\xDB\xD2\x1E\x21\xF7\x61\x57\x6C\x47\xB2\x22\xB8\xC5\xE4\xD5\x03\x89\xDE\x0F\x59\x1F\x24\xFA\xB6\x6F\x24\x94\x41\x95\xE1\xC4\xDD\x84\xCC\x99\x35\xC8\x78\x88\x18\x24\x9A\x85\xDF\x6D\xB0\xFF\xAD\x6F\xA9\xF7\x0B\x3A\xF3\xFE\x1E\x3A\xF3\xFE\x1E\x3A\xB3\x13\x73\x0D\x44\x10\x11\xDE\x72\xE4\x41\xA2\x23\x12\xFE\x1B\x45\x0B\xFE\x1B\x09\x3C\x73\xD8\xEA\xBF\x34\x09\x78\x08\x23\x36\x23\x9A\x35\x9F\xAC\xA5\x0F\x11\xBD\x57\x23\xD8\xCB\x15\x6B\xBA\x25\x88\xE8\x90\xD4\xBC\x69\xCD\xB5\x53\xEC\x62\xF9\x9D\x97\xFA\x10\xD1\x4C\x96\x3A\xDC\xA0\x36\x8E\x98\x85\x09\x9D\x2F\x3B\x43\x06\x4B\x66\x69\xE6\xE3\x76\x03\xC1\xBC\xCC\xD9\xB5\x43\x76\xB1\x68\x21\xA2\xE9\x20\x29\x6A\x21\xA2\x39\x81\x3B\xEA\x58\xE4\xE9\xC1\x6A\x9C\xD4\x1A\x71\xAD\xD1\x62\xE6\x56\x3C\x53\x7B\x59\x3E\xFB\xD6\x30\x4E\x7B\xEB\x54\x35\x99\x83\x88\x9E\x30\x44\x34\xBD\x70\xC9\xB6\x00\xA7\x1E\x99\x61\xB0\xB5\x66\xFF\x8B\x4C\x92\x6C\x31\x32\xB0\x75\x81\x18\x13\xE7\x36\xD7\xE4\x08\xCB\xA6\x46\x1A\x65\xE4\x49\x08\xA7\xA0\x4F\x17\x81\xFB\x93\xE1\x8B\xF9\xC4\xA7\xFE\x53\x7C\x68\xCC\xB5\xFF\xFA\x8F\xDC\x44\x71\xF8\xE2\xFB\x77\xAD\xB7\x7D\xED\x35\x0C\xB7\x6C\x49\x43\x6B\xBA\x84\xD4\x64\x98\xEC\x96\x59\xDC\xE6\xB8\xA7\xBA\x70\x67\x9F\x30\xF5\xCB\xD0\x9E\x60\x97\xF6\x04\x83\xF6\x9C\xE0\x54\x86\xBA\x7C\xFC\x52\x67\x06\xF1\xB9\x84\xE6\xC7\x87\xF3\xCE\x52\xA5\x37\x1C\x9A\x9C\x65\xE2\xA0\x05\xF8\x27\xFB\x79\x36\x12\x52\x9A\x2E\xF6\x71\x31\x6E\xFB\xD1\x2A\x74\x8A\x61\x12\xF9\x24\x75\x19\x8B\x7C\xF6\x6A\x7F\x45\x13\xD7\x25\xEF\x2F\x6D\x38\xA5\x32\xB9\xA6\xA9\xA6\x8E\xC6\x3A\x30\x38\xAD\x0E\x05\x50\x68\x50\x2E\x60\x6B\x96\x5A\xD1\x07\x6A\xCE\xF2\x7B\x00\xEF\x82\x29\x3F\x4B\xB1\x5E\x8C\x94\x62\xE6\x08\x2F\xD3\xB1\xAC\x63\x6F\xD7\x49\xA6\x60\x66\x89\x4E\xD8\x91\x31\xEF\xCF\xC9\x6D\x2D\x66\xB7\x59\xF3\x1A\xF6\x38\xB5\x94\xC6\xA3\xE9\x92\xA3\x9F\x29\x38\xD2\xC1\x6B\x5B\x27\x0A\xDA\xD1\x71\x2B\x7B\xCD\x1A\x7A\xE6\x04\x75\xBA\xA2\x1F\x92\x2C\x3C\x94\xF8\x34\x91\xCC\xB7\x91\x0E\x94\x09\x54\x94\x41\xE2\x92\x8D\x3A\x71\xE6\xBC\x53\x8F\x82\x41\x45\x22\x39\x5E\x58\x06\x02\x4C\x38\x49\xAC\x0C\x81\xA4\xAE\x67\xBC\x91\x58\x12\x48\x26\x92\x22\x4C\x8B\xAF\xED\xA9\x42\xA3\x2E\x45\x47\x3D\xF2\x2B\x39\xC1\x79\xBE\x49\x8F\xDE\xEE\xCA\x60\x12\x26\xBC\x29\x77\x7F\xD4\x1F\xAD\x88\x52\x77\xEE\xDD\x02\xAC\x9C\x10\x00\xAE\x3E\x3D\x93\xAB\x16\xC2\x8D\x82\x53\xC0\x12\x5A\xB6\x21\xC7\x5F\x2D\x00\x94\xCF\x1A\x15\x6F\xB7\x69\x0B\xC8\xAF\x73\x55\x1F\x10\x89\x28\xD8\x9E\x39\x69\x56\xD0\x80\x01\xC3\xA9\x78\xF1\xC7\xB5\xE0\x81\x29\x2A\x60\x6E\x47\x9F\xE0\x8C\x8F\xFC\xD5\x12\x2F\xAB\xE9\x03\x38\x3D\x0E\x05\x41\x61\x40\x23\x2B\x29\xAC\x05\x07\x89\x6A\x39\x40\x1B\x2E\x64\x25\xED\x02\x48\x3D\x2B\x79\x68\x2A\x72\xFE\xC7\x01\x4B\xDB\xED\x67\x9D\x8A\xF5\xC3\x06\x5A\x2B\xCA\x22\x6E\x20\x73\x11\x59\x33\xD4\xC6\xB4\x1E\x41\x46\x69\x80\x15\x69\x2E\x92\x06\x58\xD1\xE6\x87\x73\x68\x52\xD6\x2C\xB0\xE5\x1F\x89\x4F\x42\xB6\x1B\xEA\x31\x2E\x15\xAF\x50\x1A\x9B\x78\x9A\x14\xBA\xD4\x50\x42\x05\x2C\x27\xC5\x4D\x6E\x5E\xA5\x30\x6A\xAD\x28\x9C\x0B\xF9\x4D\x8D\x53\x6F\x6C\x28\x0F\xFC\x0D\xCA\xA6\x82\x69\x1D\x73\x5F\xBA\x6E\x36\x70\x25\x8F\x70\xFB\xD6\x02\x27\x5A\x0A\xD5\x1D\x1D\x0A\xC8\x62\x96\xF0\x5E\x91\x2C\x3A\x21\x81\x25\xE1\xFE\x51\x73\xC6\xF1\x3A\x6A\x5D\xCF\x49\x22\x90\x5A\x98\xFD\xDB\x50\xA5\x3D\xCB\x9A\x84\x9E\x3B\x2D\x7E\xD3\x8A\x3C\x52\xF1\xC7\xD8\x04\x81\x66\x3C\xEB\xD6\x5E\xE6\x7D\xAB\x77\xB4\xE4\x00\x5F\x92\x4D\x14\xE8\xC6\x7D\x3E\x68\x24\x31\xB3\x62\xB1\xC7\x27\x2B\x5D\x4E\x01\x72\x61\x62\x56\x22\x20\xF8\x8E\x95\x78\xAF\x86\x55\xB9\xC0\x7B\xFA\xF9\x65\x97\xA1\x0D\x12\xA7\x9B\xB1\xB6\x01\x6A\x1C\xCC\x2E\x11\xB3\x0B\x95\x74\x80\x82\x81\x91\x5D\x22\x14\x0F\x1D\xBB\xD0\xB1\x27\x05\xF3\x27\x3C\xAE\x99\xD7\xFE\x63\x14\x66\x24\x32\x2C\xC7\xDE\xB1\x87\xBE\xBB\xAE\x36\xEA\xD8\xE9\x73\xE4\x57\x3E\x3A\x51\xA7\xD3\xDA\xBA\xD1\x39\x9A\x58\x3B\xFA\x44\x9D\x4C\x09\x90\xBE\x3C\x4F\x7B\xF3\x84\xAC\x46\x21\x68\xF7\xBB\x6A\xC3\xFD\xDC\x76\x43\x50\x1D\x77\xAF\x15\x82\x74\xEB\x7E\x37\x38\x01\xDA\x3D\xC4\xF7\x47\x6B\xE7\x24\xC3\x4F\xC9\x69\x57\x73\x72\x5C\xCA\x20\x76\xDB\x8F\x6E\x38\x70\x25\x44\x6E\xC7\x6E\x1C\x87\xF0\xD1\x29\x8C\xCE\xD5\x23\xB6\xC2\xB9\x2E\x00\x41\x68\x66\x65\xFA\x59\x9A\x7E\x84\x99\xC2\xBA\x66\x1D\xBB\x6D\x1A\x18\x88\x1F\x98\xC2\x88\x3B\xED\xD9\x01\xDF\xA6\x4F\x5B\x76\x68\x17\x35\x1E\x9D\x84\x1C\x12\xC9\x65\x27\xE3\x58\xF2\x09\x9D\x09\x26\x38\x3B\xC0\xB8\x49\x53\xFE\xF1\xC5\xFE\x5A\x28\x83\xDA\x7D\x06\xA9\xDB\xD1\xE7\xF0\xE5\x43\xC1\xA4\x8E\x29\x7B\xAD\x3D\x07\x09\x84\x53\xD9\x06\xF9\x14\x11\xD6\x99\x13\x8C\x3E\x41\x60\x0C\xB8\x1F\x4B\xCA\xDF\xE7\x68\x61\xE6\xB6\x68\xC0\x6D\x11\xC5\x7D\xB1\x73\x6C\xC7\x6D\x11\xA3\x29\x32\xB7\xB1\xFA\x43\x07\xBC\xDE\x24\x94\xFD\xA4\x51\x76\x3E\xD2\x94\x91\x9A\xAC\x0F\x2C\xB3\x59\xF9\x1F\x2E\x89\x7F\x3B\x04\x84\x23\x45\x9E\x46\x6C\xE1\x47\xE5\x4E\x83\xEA\x52\xF4\x7F\x7A\xE1\x51\x5A\xEF\x80\xCD\x1F\xBB\xDD\xF2\x91\xC2\x8A\xBE\xFF\x96\xCE\x13\xEC\x2D\x9E\x27\x84\x72\x9E\x60\x5F\xA4\xF3\x84\x81\x5F\x34\x9F\x27\xD8\x5B\x3B\x4F\x40\x1E\x22\x82\x43\x2F\x44\x34\xCB\x09\x88\xA5\xFC\x13\x9A\xB2\x5F\x18\xA9\xFD\xDB\x62\xF7\x61\x0D\xDE\x34\x9E\x47\x42\xE4\x1F\x47\x91\x84\x7C\x24\x90\xB8\x54\x44\xBE\x77\x4A\x50\x2B\xFA\x9E\xE3\x34\x4F\x23\xB0\x55\x2C\xBC\x77\xB2\xD0\xED\x9D\x84\x6C\x2E\xB4\x24\xD3\x3D\xDA\xC2\x8D\xC8\x8C\x89\xB2\x17\xC5\x7B\x82\xA2\x88\xEC\xFF\x75\xE6\xB6\x4E\xF0\x2A\x7E\x8F\x53\x90\x89\xB0\xA1\xB3\x6D\x94\xFB\x94\x07\xC6\xA9\x07\x24\x0D\x77\x94\x61\x15\x2D\xFE\x17\x8E\x1C\x44\x64\x22\x77\x41\x95\x43\x82\xA5\xB3\x4D\x07\x89\x59\x13\x81\x08\x0C\x3D\xF3\x3C\xDE\x13\xA5\x1C\xD7\xEA\x31\x9E\xC8\x76\x57\x7E\x8E\xB6\x8B\x83\x27\x2E\x38\x14\x7C\xFE\x87\xBF\x93\xE3\x99\x73\xC6\xBB\xCE\x29\xA7\xFA\xFD\x6E\x74\xF6\x4C\x6F\xFE\xF9\xC0\x4A\xB2\x4A\x10\x44\x46\xC1\x01\x99\xCE\xAC\x51\x99\x2E\xA8\xC6\xB8\x71\x76\x01\x01\x31\x33\x6E\x4D\xE9\xB6\x99\x66\x63\xA7\x37\xEB\x1C\xC6\xEE\xFA\xF5\xEB\xD7\xCD\x06\x8C\x4F\x4E\x39\x06\xB1\x9E\x30\xC5\x72\x31\xFE\xB9\xC0\xA7\x0F\x4A\xB1\xE4\x7D\x5C\xF2\x7E\x24\x01\x6D\xE0\xB0\xD4\xC2\x6D\x53\x5C\x4E\xBD\x1F\x4B\xBD\x0D\xF6\xFB\x52\xF7\xB7\xA5\xE6\x14\xB1\x04\xB7\x75\xA5\x12\xEA\x14\x2A\x54\xE5\xBF\x91\x65\x7B\x24\x78\xD8\x63\x28\xC5\x98\x2C\x35\x27\x3C\xB3\xF6\x31\x5A\x12\xEC\x87\x42\x9E\xE3\x72\xCB\x0C\xB2\x7F\x40\xD3\x94\xB6\x8F\x9E\x3E\xFB\x65\x53\x0A\xFB\x20\x3F\x7E\x8E\x22\xB5\xDE\xE9\x0D\xFC\x23\x8F\x2B\x3E\xF1\x41\x70\xB6\xCA\x61\xBF\x58\x7B\x2C\x5B\x71\xB5\xEC\x61\x4F\x14\x86\x8F\x7D\x01\xD9\xC3\x3F\xED\x71\xAD\xE3\x84\x3A\xC2\xB7\x2D\xBF\x32\x07\x13\x85\x46\x6C\x63\x1A\xF1\x89\xCC\x48\xF8\x74\x64\x8E\x32\x7D\x46\xAC\x0F\x23\x9F\x2A\xCF\xA7\xB1\xE7\xD3\x11\xBD\xCA\x9B\xD1\x8C\xF9\x94\xD9\xD4\x10\x06\x1F\x7B\xB5\x80\x26\x7B\xE8\x08\x52\x4F\x3D\x3E\x43\x23\xC1\x25\xB9\x2C\xE8\x0C\x0F\xC5\x7D\x77\xA0\x46\x9B\xF3\x01\xF3\x32\xC9\xE4\x48\x40\xB8\x38\xE5\xC0\xD7\x1E\x9D\x5B\x3F\x79\x48\x98\x4C\xFD\x31\x15\x1B\x59\xDA\xB2\x9F\xF1\xEC\x97\x92\x83\x12\xA4\x9E\x51\xD2\x96\x51\x22\x66\x3F\xF1\x73\x41\xE5\xA3\x57\x20\x05\xB4\xA7\x60\x78\xF8\xEB\xF4\x38\x67\xC4\x36\x37\x18\x7C\xE3\x07\x3F\x61\x61\xCA\xA7\x06\x32\xD8\xD4\x40\x73\x9C\x8E\x00\x65\x9C\x65\x08\x53\x3E\x94\x22\xDF\x52\x20\x4F\x28\xCA\xBB\x87\x63\x9F\xEB\x0C\x6F\x0C\x47\xBE\x2F\xAF\x76\x1D\x77\x45\xD7\x37\x31\xF6\xFA\x66\xC6\xBE\x27\xA3\xA4\x9D\x19\xD9\x2E\xBD\x08\xFE\x77\xD4\xC5\x7F\x91\x89\x08\xB6\x95\xF6\xEB\x11\x6B\x84\x4E\x64\xB0\x97\xBF\xF1\xB0\x3F\x6D\x6F\x0C\xED\x3C\xA3\xB6\x57\xDC\x0B\x53\x85\xB4\x09\x41\xFD\xCD\xF7\x8C\x71\x5D\x28\xD7\x3D\x36\x97\xB3\xC5\xC7\x14\xED\xC8\x42\xD0\x6D\xF9\xC8\x0F\x86\xD5\xC4\x2E\x8E\x7C\x17\x13\xFA\x86\x18\x11\x45\x3A\xC3\xE8\x4B\xE5\x84\x23\xD4\x54\x9D\x30\xD6\xFD\xB4\x4E\x39\xC4\x14\x50\x68\x6A\xDA\x37\x71\xC0\x59\x82\x33\x02\x39\x7A\xB4\x40\x1C\xF3\x59\x8D\x4F\x8A\x48\xD1\x98\x9E\x7B\xFA\x4F\xE6\xC5\xB1\xB9\x45\x71\x1C\xA3\xD0\x4C\x17\x88\xE3\xD8\x8B\x63\x22\xE2\x84\xC5\xF1\xC4\xCF\x87\x49\x3B\x1F\xC6\x4C\xC0\x9E\x38\x8E\xB3\x4E\xD0\xCF\x8A\xE3\xC4\x8B\xE3\x64\x2F\x71\x9C\xCC\x88\xE3\x64\x4E\x1C\xF3\x64\x1F\xC3\xA4\x15\xC7\x52\x73\xBC\x9B\x38\xE6\xE0\x5E\x2F\x8E\x3B\x9A\x26\x73\xE2\x38\xD9\x45\x1C\x63\x9D\x63\xA4\xF3\xD8\x8B\x63\x33\x2B\x8E\x45\xCE\x24\x5E\x1C\x67\x03\x71\x9C\xEC\x22\x8E\x67\xE0\x85\x65\xCC\xF8\x96\xBC\xE6\x75\x89\x07\xDB\x07\x0B\xD4\x09\x3D\x50\x28\x42\x66\x68\x7C\x5B\xB7\x0C\x1D\xDE\x40\xAF\xC0\x57\x3D\x3F\xD3\xB9\x1D\x32\x31\x24\x55\x0C\x19\xD9\xF5\xFC\xD6\x89\xF2\xFD\x33\x6A\x46\xE2\xD1\x25\x9A\xBE\xF4\x4E\x50\x7A\x8F\x5A\xE9\x9D\x0C\x18\x5D\xEC\x2D\x34\x11\x3D\xC7\x27\x34\x5A\x71\x6F\x4C\x3A\xD1\x1D\xEF\x22\xBA\x39\x6B\x28\xB3\xAA\xF6\xAC\x8A\x85\xD6\x16\x12\xCF\x54\x49\x4F\x74\x13\xAB\x4A\xE8\x88\x77\xE1\xEF\x44\x37\xED\x46\x44\x74\x27\x22\xBA\xF5\x0D\x18\x45\x7B\x46\x89\x77\x13\xDD\x5A\x44\x77\xD2\x17\xDD\x89\x17\xDD\x21\x8A\xEE\xD4\x8B\xEE\x4C\x44\x77\x4A\x82\x6A\xB1\xA8\x1B\x08\xEE\x19\x96\x30\x34\xC8\xBB\x68\x99\x0B\xD9\x62\x28\xE7\x6E\xC4\x16\x3D\x39\xC7\xCD\x1E\x4A\xF2\x3F\xA5\x1E\xFF\xE1\x48\xDD\x36\x23\xC9\xBD\xE8\xB6\x6D\x7F\x84\x45\xAA\x90\x93\x94\x92\x82\x0D\xA6\x4A\x70\x15\x04\x8B\x8B\x4D\x1F\xAD\xD6\xE7\xDB\xB4\x10\xE2\xD8\x50\x14\x7E\x86\xAC\x94\x2F\x90\x99\x05\x4B\xC6\x0C\x8A\xCA\xB2\x2D\x84\xB4\x53\x19\xBA\xB9\xA7\x03\xD9\x49\x52\x55\xC3\xFF\xC3\xDE\xBB\x07\x5B\x76\x9D\xF5\x81\xEB\xB1\x9F\x67\xEF\x7D\xEE\xEE\xEE\x2B\xBB\xC3\xED\x4A\xD6\xD9\xA3\xC9\x5C\x55\xA9\xCB\xFA\x43\x75\x5B\x23\x67\xE2\x5E\x2A\xF7\xED\x87\x64\x49\x35\x43\xD5\x68\x6A\xA8\x1A\xFD\xE1\x3F\x5C\xFB\xCA\x13\xDF\xAB\x2B\x95\x92\x32\x7D\x1B\x68\x19\x07\xC2\xC4\x63\x0C\x81\xE0\x60\x41\x9C\x92\x1F\x22\x78\x82\x99\x31\xE1\x61\x85\x31\x55\x22\x81\x29\xA7\x06\x03\x13\x08\x60\x12\x88\x19\x48\x10\x13\x66\x30\x36\xA6\xA7\xBE\xDF\xF7\xAD\xB5\xF7\x3E\xE7\xDC\xEE\x36\x7E\x91\xA4\xDB\x65\xDD\x73\xCE\xDE\x7B\xED\xF5\xF8\xD6\xB7\xBE\xE7\xEF\xCB\x7A\x57\xDE\x82\x7D\x02\x08\x7D\x2E\xD9\x40\x03\xEF\xDC\x08\xE1\x3C\x30\xF0\x13\x41\x9E\xF0\x47\xF4\x41\x78\x67\xB3\x8E\x77\xB6\x74\x83\x72\xCD\xC0\xE5\xA4\x70\x64\x42\x2D\x9F\xE4\x96\x4F\x79\xAE\xC6\x07\x6F\xD4\x62\xCE\x4A\x78\xD2\x9D\xA2\x56\x37\xDD\xA9\xD0\xEA\xA9\xD8\x6A\xC3\x8E\xBE\xCD\xA1\xD5\xE4\x18\xDE\xD9\xBA\x0D\x77\x42\x78\xA3\xBC\x99\x53\x9D\x88\xA3\x39\xE5\x1A\x77\xCA\xCD\xE5\x7A\xC2\x79\x29\x5E\x2F\x4E\x4D\xE6\x35\x99\x6C\x89\x53\x5E\xEC\xCB\x27\x5D\x03\xDE\xD9\x4C\x79\x67\x8B\x20\x93\xC0\x3B\x4F\x35\xDA\xAB\x8D\x91\x14\x98\xAF\x48\x81\x39\xF1\x91\x22\xF2\x91\x5C\xA4\xC0\x22\xB2\x92\x3C\xA4\xA2\xE6\x23\x52\xC8\x1B\xD9\xCF\xD2\xC9\x7C\xC4\x4C\x92\xB5\xCC\xC4\x90\x9C\x12\x67\xB9\x0A\xB3\x8C\x20\xCF\xD2\xE5\x61\x96\xF3\x38\xCB\x5C\x5F\xD5\x09\xBC\xDC\xA2\x5C\x66\x26\xA5\xCB\x5D\x25\xCC\x24\x17\x66\xC2\x61\x1C\x43\xA7\xC6\x33\x57\xC9\xC9\x89\xB5\x6F\x62\x04\x6F\x64\x26\xC6\x55\xD4\xCA\xC0\x0D\xF3\x70\xC0\x58\x1C\x30\xA0\xCB\x90\xB3\x9C\xDF\x96\xE8\x67\x98\x61\x26\x41\x40\x32\x5C\x86\xAA\x80\xA2\x4D\x4C\x80\x74\xF9\x7C\xB0\x94\x1D\xA3\x9E\x26\xB8\x55\xD0\x78\x07\xD1\xCF\x10\xD3\x60\x93\x1D\x18\x86\xD9\xA8\x78\xEE\x0D\xC7\x49\x68\x09\xAA\x74\x1E\xC8\x0C\x71\x18\xB0\x9F\x2D\x18\x97\x09\xAC\x10\xF0\x4A\x15\xB1\x87\xF1\xA8\x2A\x66\x08\xBA\xE7\x1A\xDF\xCB\xC3\x0B\xCB\x07\x6D\x26\xE7\x61\xE5\x32\xAC\x9C\x83\x4F\x24\x8F\x98\x87\xA5\x57\x86\x95\xE3\xD6\xE9\xB0\xA2\x76\xB5\x21\x02\xDF\xD7\xE2\xDD\x81\xEB\xFE\xBF\x20\x8D\xDF\x8E\x26\x8C\xD0\x99\xCD\xCE\x8E\xD9\xAD\x18\x36\xE0\x25\x56\x02\x18\xCD\x33\x6C\x30\xB5\xE0\xBC\x36\xB8\xE5\x2D\xE0\x6D\xA4\x30\xB5\x63\x91\xB9\x33\xC7\x59\x0D\x18\x77\x36\x66\x0E\x90\xD0\x20\xC4\x3D\xB9\xB2\xCA\x6A\xF3\xDB\xE3\xB3\xCD\xC0\x67\xEB\x35\x32\xEA\x2C\xC8\xA8\x03\x9F\x2D\xD7\xCB\xA8\xB3\xD1\x5E\x1D\xF1\x59\xBB\x24\xA3\x42\x13\xB4\x41\x46\xB5\x37\x97\x51\x4B\x78\xCE\xC7\x32\xAA\x5D\xE1\xB3\xF5\x8A\x8C\x3A\xE5\xB3\x2C\xA3\x96\x23\x19\xD5\xB2\xB1\x4B\x64\xD4\x61\x42\xED\x8A\x8C\x6A\x59\x46\x2D\xC1\x67\xCB\x81\xCF\xD6\x2B\x32\xEA\xC9\x0B\x8D\x25\x19\x44\x22\xEC\x18\x1B\x82\xAB\xCA\x25\x72\xA0\x83\xA4\x13\x22\x88\x24\x6C\x40\x03\xBC\xD1\xF1\x96\x5B\x4B\xEC\x76\x1A\x12\xB5\xCC\x3F\xEC\xC0\x3F\x8E\x31\x1B\xAC\xE3\x1F\xA3\x8D\xB6\x2C\x74\x26\xC7\x0B\x9D\xC9\xF4\xB0\x48\xA6\x27\x85\x80\x16\xD8\x65\x17\x02\xA6\xB9\x88\xC7\xD9\x70\x52\xD8\x63\x4E\x0A\xAE\x58\xC2\x74\x92\x07\x3A\x49\x58\x43\x4A\x02\x9D\x24\xA3\x93\xC2\x8E\x34\xA4\x45\xBD\x7C\x52\xD4\x74\x78\xC9\x49\x91\xC8\x49\x91\x4F\x8E\xAF\xE9\x19\x9B\x87\x33\xD6\x1E\x77\x52\xE4\x72\x52\xC8\x51\x97\x84\x93\x22\x1D\x4E\x0A\x2B\xC5\xEE\x93\xDB\x90\x34\x4D\x88\xA5\xB1\x2C\x69\xA6\xF0\x0C\xE2\xC0\xB0\xBC\xE0\x56\x16\xDC\x86\xB8\x1B\xC5\xE0\x84\xB4\xE4\x17\xE5\x89\x41\xD2\x8C\x2B\x6F\xF1\xCC\xD2\xCA\xA7\x80\xBC\x34\xE1\x48\xC0\xC1\xC1\x47\x44\x3C\x2D\xBE\x06\x74\x18\x98\xEE\x1F\x61\x96\x7F\xB7\xFA\x8F\xDA\x6E\x1C\xFD\xF2\x25\xC2\x89\xA4\x24\x4F\x70\x8F\x56\xDB\xE6\x74\x37\x6B\x94\xFF\xC9\xA1\x76\xB1\xA3\x1F\x17\x33\xCE\xF9\x13\x03\xC7\x8C\xF5\x9C\x84\xAE\x9D\x71\xB6\x7D\x2F\xC8\xDF\x72\x21\x80\xC4\xD9\xF6\xB7\xAE\x07\x86\x27\x3D\x90\x48\xF7\xD2\xB1\x76\xF0\xB5\xB2\x56\x67\x77\xAC\xD5\x5F\x11\x6B\x75\xBA\x6C\xAD\x4E\xC6\xD6\xEA\xF4\xCF\xA5\xB5\x3A\x58\xAA\x9D\x09\xE7\xC7\x1D\x73\xF5\x7F\x9A\xE6\xEA\x3F\xC6\x10\xFF\x51\xFD\x9F\x88\xB9\x3A\x9E\x01\x05\x03\xC1\xD8\x85\x19\x9D\x01\x33\x3A\x03\xCA\xA5\x33\x80\x7E\x24\x26\x52\x39\xBB\x27\x50\xA2\xB2\xEE\x96\xAE\x9D\x71\x46\xCE\x00\xC3\x51\x90\xD6\x19\x39\x03\x18\x0B\x0F\x3D\xA0\x8F\x7F\x2E\x8C\xE4\xD9\x1D\x23\xF9\x57\xC4\x48\x9E\x2E\x1B\xC9\x93\xB1\x91\x3C\xFD\x0F\xD9\x48\x5E\xDD\xB1\x92\xDF\xB1\x92\x1F\x6B\x25\xFF\x3C\x46\xFC\xDD\x95\x6E\x27\xAA\x43\x7F\x53\x33\x39\x26\x28\x19\x22\xD7\x44\x85\x35\x21\xA7\x21\x09\xDC\xB9\x18\x71\xE7\x92\xB8\x73\xB1\xC4\x9D\x4B\x81\xB7\x05\xCE\xC6\x4C\xB8\x8F\x70\xE7\x92\xB8\xF3\x4C\xB8\x73\xC1\xA8\xA5\xD6\xCD\x84\x3B\x33\x47\x60\x8D\x12\xC5\x37\x8E\xFA\xCE\xDE\x82\x29\x8F\x2C\xF0\x4B\x4C\x79\x8D\xF5\x3D\x63\xA6\x9C\xDD\x8E\x49\xA8\x5A\xE1\xC8\xA1\xA4\xD5\x3C\x52\xF9\x3C\x50\x79\x0D\x88\x00\x57\x07\x2A\xAF\x47\x46\x72\x50\x79\xB6\xC2\x91\x61\x7A\xDF\xE0\x96\xDB\x90\x52\x33\x35\x0A\xB7\xD4\xEA\x09\xD7\x86\x56\xDB\xD1\xDE\x81\xEE\x7B\xE2\xD6\xA6\xF7\xC6\xD5\xD1\xB4\x3E\xE5\xC8\x1B\xB8\x9E\xB9\x36\x98\x8F\x47\xA6\xF7\x76\x32\xA1\x53\xB3\x40\x1B\xCC\x02\x1B\x2E\x03\x47\xCE\xA6\x1C\x19\x0E\x90\x66\xE0\xC8\x99\x6B\x97\xCD\xEF\x76\x45\xAA\xB5\x53\x8B\x8A\x5D\x31\xBF\xDB\x60\x7E\xB7\x23\x1A\xB0\x4B\xE6\x77\x7B\x4B\xF3\x7B\x1E\xCA\xBE\x62\xA6\xCB\x30\xD3\xD4\x68\x37\x73\x36\xCC\xB4\x8D\x33\xCD\xF9\x4F\x6E\x26\x5C\x6A\xB6\xCC\xA5\x48\xBA\x28\x85\x4B\x59\xE1\x52\xE5\xC4\xD2\x33\x35\xA8\x95\xC1\xA0\x26\xE6\x77\xC4\x12\xE6\x03\x97\xCA\x5D\x29\x5B\x60\x6C\x3D\xB3\x23\x3B\x55\xB0\xCC\xE0\x96\xDB\x11\x65\xF3\x91\xC9\x42\xE3\xFB\xCD\xCD\x16\x6A\x22\xCA\xAA\x5B\x98\xDF\x73\x56\x59\xA2\xF9\x3D\xBF\x85\xF9\xDD\x0C\x46\xC0\x5B\x9A\xFC\x82\x31\x48\xEA\xFD\xF4\x30\x04\x45\x0B\xD0\xD8\xFE\xA3\x26\xBA\x97\x5A\x32\xFC\x48\x3A\x40\x34\xB5\x14\xD1\xE4\x77\xDC\x4B\x0B\x7E\x69\xC1\x2F\x1D\x49\xC6\xFC\xD2\xE2\x66\x2F\x15\x81\x78\xF9\xA5\x76\x62\xDF\xF9\x02\x16\xFC\x47\x23\x93\xBE\xA9\x51\x7D\x6A\xF3\x21\xC5\x71\xC9\x7D\x99\x04\x36\x6D\x99\x4D\x9B\xDB\x67\xD3\x0C\x44\x38\x66\xD3\x66\x2D\x9B\x36\x2B\x6C\x5A\x66\x07\x6F\x4D\x97\x64\x8A\x55\x46\x6D\x03\xA3\xB6\xCB\x8C\xDA\x1E\xC3\xA8\xCD\x57\x80\x51\x37\xCC\xA8\x9B\xB0\xD1\x9B\x91\x90\x7B\x0C\xA3\xB6\xEB\x18\xB5\x0D\x8C\xDA\xDE\x8A\x51\xDB\x25\x46\xBD\x6A\xBB\x2F\x45\x34\x6E\x6E\x9B\x51\x0F\xB6\xFB\x76\x32\xA1\x76\x85\x51\xDB\x9B\x30\x6A\x76\x9D\x04\x46\x6D\xD6\x30\x6A\x11\x2C\xCD\xF1\x82\xA5\x99\xF2\x6D\x33\x65\xDA\x26\x58\xC2\xCD\x88\x1E\xCC\x92\x25\xDC\xDC\xD2\x12\x9E\x8B\x52\xC8\xB3\x5E\x86\x59\x37\xCC\xB4\x4D\x98\x75\x33\x62\xDA\xF6\x16\x4C\xDB\x44\xA6\x6D\x46\x4C\xDB\x8C\x3A\xB5\xCC\xB4\xCD\xC8\x12\x7E\x3C\xD3\x36\x3C\x00\x13\x98\x76\x32\xB5\x84\x03\xFB\xFD\x36\xA4\xC9\x3C\x66\x33\xB3\x34\x99\x04\x4B\x78\xBE\xDE\x12\x6E\x26\x9C\xF0\xA2\x3C\x31\x48\x93\xCB\x0C\x71\x6A\x7B\xE6\xE2\x4F\xF9\x60\x09\x27\x1E\x6E\x99\x87\xDB\x75\x3C\x3C\x89\xBE\xC9\xDB\x74\xDB\x18\x66\xA7\x50\xB8\x19\xAF\x85\xFA\x61\xC2\xB9\x63\x6E\x66\x3F\x33\xB8\xD5\x29\xEA\xB2\xBA\x4D\x1E\x6E\x97\x79\xB8\x9A\xF0\x70\x3B\xF0\xF0\x63\x5E\x2A\x3C\x7C\xF9\xA5\x62\xA3\xFF\x9E\x04\xF0\x44\x31\x95\x9F\x33\x2F\xD8\x49\x12\xBD\x47\x36\x6E\x98\x2E\x7D\x93\x90\xD3\x8E\xD9\x0C\xAE\xA6\x48\xF3\xFA\xAC\x3A\xC5\xEC\xCE\x92\x8E\x6A\x44\xF4\x31\x7E\xB6\x0B\x74\xEC\xA5\xB4\x0C\x4D\x3F\x21\x41\x3C\x26\x65\x94\x6F\x6A\x34\x4A\x6D\xBB\x94\x1E\x49\xE2\xE3\x66\x7D\x56\xC7\xAF\x33\x10\x5B\x7C\xFC\x61\xDA\x01\xD7\x88\x0A\x92\x85\xA9\x2D\x13\x6B\x86\xA4\x2D\xAF\x17\xD6\x1B\xEC\x3E\x9F\xB2\x33\x87\xB9\x38\x8A\x51\xB6\x5D\x7A\x80\x9D\xFD\x09\xD5\x0B\x3E\x75\x23\x09\x15\xCB\x09\x57\x4E\x70\x4F\x95\x2F\x7C\xED\xD2\x2D\x4E\x6D\x8B\x99\x4A\xFC\x14\x92\xDB\x7E\x2B\x70\x85\xE9\xAE\x95\xE6\xF5\xED\x34\x4F\x5D\x9B\xBC\x61\x7D\xFB\xC3\x31\xC2\x88\x1B\x4A\x96\x78\x93\x46\x2D\x87\xE7\xB6\x39\x8D\x6C\x84\xB3\xAA\xC5\x9C\xF1\x22\x6F\x1B\x52\x68\x2D\x6B\x91\x96\x1B\x3A\x51\x39\x5B\x7D\xDA\x68\x13\x72\x35\xEE\xDB\xEB\xCA\x58\x37\x61\xE6\x72\xC1\xF6\xE2\x02\x0A\x28\x38\x92\x21\xE1\x89\x8D\x61\x33\x4E\x07\x52\xAE\xA4\x3F\xF7\xBA\xD2\x29\x7F\xA6\xDF\x31\xDB\xAE\xE4\x52\x06\x28\xA3\xE3\xF8\xF2\x19\x57\xD2\x06\xB8\xDB\x95\x4E\x4B\x2D\xEB\x52\xEA\x46\x17\xA4\x52\x3D\xA8\x5A\x57\xBA\xE4\x41\x75\xD2\x95\x2E\x7D\x50\x9D\x72\xA5\x2B\xFD\x36\x83\x88\x38\xE5\xCA\xF6\x4F\x69\x16\xC6\x5D\x60\x33\xDA\xB6\xD9\x04\x7A\xBF\x51\x74\x74\x0E\x87\x00\x2A\x1A\xA6\x4F\x6F\xF5\x8C\xFB\xC3\xA6\x04\x57\x9E\x55\x27\xBD\x3E\xA4\xBF\x27\xBC\x39\x3C\xA0\x0F\xA7\x0E\x40\x54\xF6\xB0\xD3\xFE\x3D\xBF\x73\xFD\x1A\x70\xD4\xBC\x7E\xA6\xA7\xBE\xFA\x77\x85\x5F\x4E\x7C\x13\x52\x51\x16\x38\x00\xCF\xAA\x33\x93\xE9\x70\x39\xF5\xC4\x21\x6B\x63\x31\xAB\x5C\xE9\xEF\x0B\xE9\x30\xB3\xEA\x63\x46\xEB\xA3\x95\xD2\x14\xC3\x76\x8A\xE9\x73\xC6\xA9\x08\x04\xFD\xAA\xF1\x47\xBB\x0C\xFB\x0C\xAC\xC2\x33\xF8\x04\x54\x15\x7C\x3A\xCD\x40\xE6\xFE\x55\xC3\xB8\xC6\xFE\x7D\xD4\x68\xFB\xC3\xDF\x22\x80\xEE\x9F\x51\xFD\x02\xC5\x8C\xBE\xA4\x46\x5F\x58\xD7\xA8\x9E\x36\xFA\x87\xB7\xD9\xE8\x1F\x86\x46\x7F\x60\xDA\xE8\xEF\x51\xA3\x92\xEC\xA6\xC6\xA5\x21\x54\xF5\x5F\x6A\x7D\xC4\x9B\x24\x11\x90\xF8\x05\x75\x22\x20\xC0\xA1\x6E\x84\xBA\xC7\x3E\xF9\x90\x39\x6A\xFF\x15\xF3\x41\x40\xB3\xBD\xA5\xDA\xB9\xE5\x93\x16\x4F\x2E\x3D\x76\xD1\x98\x23\x7D\x75\xFA\x60\xA7\xCE\x2A\xB3\xD0\xC3\xD3\x06\x05\x2A\xE8\xE9\xCE\x3A\xFB\x90\x39\x72\x7A\x6B\x68\xC7\xEC\x98\xA7\xAA\xD3\x6A\x00\xD8\xFA\xD7\x74\x45\x60\xB6\xAA\xCF\x5A\xAB\x01\xA6\x33\xA4\x47\x15\x32\x5C\xD9\x4D\x05\xB6\x45\x57\x9E\xB3\x05\x2A\x8C\xA3\xB0\x67\x81\x5D\x83\x1F\x5D\xF9\x62\x08\x95\x50\x67\x95\xEE\x2A\xFA\x43\xD2\x20\x75\xB3\x6B\xB6\x88\x98\x59\xF6\x63\x33\xBC\x9B\x5D\xE4\x93\xBE\x76\x95\x9B\x5E\x9E\xB9\xF2\x21\xF7\xCE\x17\xFB\x91\xE8\x74\x51\xCC\x56\x9C\x7A\xBC\x8B\xE2\xC9\xB4\x3B\xD4\x45\xD2\x91\x76\x0F\x44\x42\x8A\x2A\xEB\xD5\x5E\xAA\x61\xA1\x60\x36\x49\x63\x8B\x9A\xA8\x2E\xBF\xB0\x98\x79\x0B\xDB\x7F\xCD\x5C\xB9\xF6\x7A\x91\x30\xDF\x48\xD9\x11\xED\xEA\x2D\x80\x7B\xEB\xA7\x69\xCB\x28\x97\xB4\xD7\xB8\xE8\x0C\x97\x6A\xA2\xFE\xD1\x98\x2E\x20\xD8\x26\xE9\x5D\xDE\xDE\x60\xAD\xDF\x9C\x77\xD9\x59\x05\xB9\xBB\x68\xFF\x88\x9E\x91\x3C\x31\xCF\x16\x74\x8E\x5A\x63\x61\xA5\x0E\x83\x2D\xB6\xB9\x86\x90\x40\xDA\x25\x68\x7B\x2B\x0E\x18\x98\x31\x52\x09\x57\x92\x6C\x93\x85\x40\xE2\x69\xD2\x16\x00\xB1\xC0\x90\x00\xDA\xA9\xDE\x65\x7B\x02\xE6\xCA\x40\x4E\x5D\xA8\x41\x84\xA6\xE9\x5A\x85\x7A\x3F\xB2\xB8\xA6\xFA\x13\xCD\x08\x01\x7A\x6D\x56\x76\xC0\x04\x30\x21\xF1\xDC\x6E\x01\xD5\x6C\x35\x1B\x9E\xA1\x6A\x59\x4E\xA6\xBB\xE9\xC6\x98\x0F\xAF\x57\xF3\xE1\xED\x90\x0F\x0F\x2C\x94\x51\x7E\xFE\x6A\x83\x9C\x1E\x2F\xED\x5E\x6A\x94\xFF\xA9\xE5\xCC\xF7\xE0\xD4\xA9\x46\x2F\x44\x0E\xA0\xE3\x2B\x03\x76\x19\xCB\xDC\x5C\x5B\xAB\xFA\x70\xA6\xE7\x01\x5F\x75\x5A\x86\xC4\xFC\x47\x5F\x86\x24\xBB\x00\x70\x1C\x94\x21\xC9\x88\xAC\x04\x98\x23\x96\x21\xC9\xA5\x0C\x49\xCE\x27\x09\xAA\x89\xE4\x4E\xBF\x0D\xE8\xEA\x1C\xDC\x19\xCB\x90\x64\x52\xFC\x4B\x73\x89\x8E\x92\x44\xD4\x87\xE1\xDB\x2B\x80\xB8\xC8\x01\x06\x6C\xBB\x8C\x20\x42\x9A\x41\x84\xF4\xDB\x82\x55\xCA\xBA\x1C\xB0\x61\x52\x73\xC4\x84\x0C\x6F\x48\xE9\x5D\xD5\xA3\x3E\x01\xF6\x17\x5B\xAD\x68\x5F\x5B\x57\xF0\xDE\x36\x9C\xE9\x5E\xB0\x62\x0F\x84\xB5\x45\xCD\x42\x6A\x01\x31\xA4\x00\xA4\xF2\xED\x60\x12\x31\x86\xF2\xAE\x48\x28\x11\x95\x08\x9A\x44\x13\x51\x89\x1A\x11\xC6\xB1\x41\x8E\x43\x25\xE2\x7A\x02\xC4\x64\x10\xAB\xD7\xB8\xA4\x4B\x77\x01\x9C\xAB\xFB\x05\x63\xB9\x2C\xE6\x8C\x60\xB3\x31\x7D\xC6\x6D\xB8\xF9\xAE\xE0\xCE\x35\x40\x25\xAA\x11\xC2\x27\x43\xAF\xF8\xFB\x02\x25\xEC\xB0\x06\x35\xCF\x55\xD5\x2F\x15\x2E\xC1\x0F\x80\xFF\xAF\x64\x9D\x18\x32\xDF\xDD\x4E\xF5\x95\x6A\x00\x13\xFD\x4D\xAD\xCD\x91\x53\x28\x0D\xE5\x5F\x51\x0C\x5B\x06\x57\x02\x6F\xAC\x87\xD4\x39\x9C\xA8\xEA\xF5\xBA\xE6\xA7\x8B\x08\xBA\x9F\xB0\x40\xDF\xB6\x7F\x93\xCB\x4E\xF9\x7B\x7B\x11\x0B\xE5\xA7\xB0\xCB\x05\x3A\x36\x60\xC9\x7A\x7D\xB8\x84\x22\xAB\x43\xC9\x69\xA4\xB6\x86\x1D\x8A\x5D\x7E\x1E\x9D\xB1\x3B\xBC\x80\x17\x26\xF5\x8C\xA8\x73\x4F\x72\x3E\x6E\x00\xB4\xFD\x8C\x92\x0A\xEB\x23\xC0\xD4\x2F\x18\x9D\x46\xC0\x94\x35\x58\xA4\x6A\x84\x45\x1A\x98\xA3\x0D\xCC\x31\x59\x0F\x98\x62\x25\xBF\x3B\x98\x41\x94\x4B\x80\x96\xFB\xE8\xB1\x65\x8F\x52\x17\x47\xA9\xC7\xA3\x64\x27\x34\x03\x6C\xA1\x90\x83\x98\x04\x02\xAE\x5A\x86\xDC\x74\x1B\xBE\x4B\x8D\xB2\x65\x9C\xD3\xD0\x1D\x46\x22\xAE\xA5\x3E\xBF\xD4\x32\x9E\x94\x32\x10\xCC\x59\x3D\xE1\xC2\xFA\x11\x56\x29\x90\xCB\x2B\xE0\x24\x6B\x27\xAA\x1D\xB8\xB2\x58\x9C\xF9\xD5\xCB\xB3\x34\x22\xB1\xBF\x9D\x44\xBC\xDA\x37\x4B\xFA\xF2\x60\x6B\x51\xFE\x93\x20\x03\x4E\xB0\xC6\x02\xB1\x80\xA7\xFC\x2F\x2B\x59\x89\xFC\x59\xFF\x27\x37\x6E\x6C\xEC\xF7\x82\x13\xE6\x6F\xD8\x7D\x06\x45\xC4\x7D\xBF\xCA\x0D\xD8\x7E\x7C\xEB\xF4\x46\xBB\x3B\x94\x9B\x4D\xBA\x49\xAC\x9E\x39\x1F\x90\x0B\xA9\xF1\x3D\x7F\xCD\x05\x3B\xE7\x42\x0B\x98\x00\x74\x7F\xC9\xA7\xF7\xAF\xA8\xBE\xFD\xF6\xEB\x01\x5D\xFB\x7C\x40\x39\x3D\xBF\x8C\x72\xCA\x03\xBE\xF7\x18\x00\x98\x7B\x23\x00\x0C\xE7\xA0\x7B\x35\x64\x92\xAF\x23\xB8\x07\x16\xBA\x1A\x72\xC1\x97\xE0\x54\xFD\x7D\x3C\x33\xE8\xB0\x6C\xFB\x35\x7D\x39\x7D\x4C\x5F\x4E\x0F\x7D\xD9\x8E\x7D\xB9\xFB\xF8\xBE\xDC\x1D\xFB\x72\xF7\x9A\xBE\x9C\xB9\x45\x5F\x9C\xAA\x3E\xFD\x15\xAB\xB3\x76\xCD\xAC\x2B\xB4\x76\x2D\xF9\xAA\x17\x5A\xFB\x77\x27\x74\x79\x14\x76\x34\xD1\x49\x07\x0D\x37\x71\x39\x63\xF5\x80\x8E\x60\xA9\x5A\x0B\xAC\x52\x92\x1A\x09\x82\x9D\xD1\xD3\x6F\x5C\x41\xF1\xC9\x83\xCF\x93\xA8\x9B\x8E\xB0\x85\x20\xA6\x31\xF6\x4F\xC5\x02\x66\xD6\x28\xFF\x63\x63\x38\xBB\x8C\xCF\x8A\x9A\x9E\x6D\x78\x71\x6A\x18\x70\xCC\x79\xB8\xA6\xFC\xFF\xF7\xE1\xC9\xDD\x6A\xDB\x3C\xC9\xF2\x55\xB6\x63\x9E\xAC\x5C\xB5\xC8\x82\x5C\x22\x68\x4B\x5D\x73\x0C\xEC\x4F\x1D\x60\x7F\xAA\x35\x58\x4B\x80\xFD\x01\x74\x49\xBD\x90\x3A\xCA\x74\x7F\xC5\xA8\x53\x2C\x09\xCF\x05\x7B\x70\x84\xBC\xB4\xC1\xE2\xEF\x9C\xCE\x89\xCA\x6D\xD0\xE0\x23\xF6\x92\x39\xEF\xE6\x6B\x61\xA6\x87\x17\x49\xBB\xB8\x55\xDE\xB7\x06\x69\x7A\xD4\x31\x86\xBF\x92\x44\xA3\xD0\xC1\x37\xF1\x6C\x2D\xC1\x4C\x2D\x8F\x28\xDE\x7E\xAB\xA1\x8F\xDB\x5D\x1A\x2E\x63\xBB\xF0\x60\xEB\x38\x58\x12\xAC\xDF\xFD\x81\xC9\x84\x7F\x59\xD6\xE0\x26\x1D\x99\x8F\x3A\x32\x3F\xAE\x23\xAB\x0D\xF2\x62\x49\xBB\xCB\x12\xFE\x74\x49\x86\x17\x76\x1B\x3D\x8A\x03\x33\xE6\x14\xD1\xAA\xEE\x3B\xFC\x20\x93\x10\xA4\xF5\xDA\x5F\x5D\xB3\xDA\xCB\xCB\xAC\x19\x60\x6B\x65\xA5\xBF\x6C\xD3\xB6\x6E\xE2\x36\x46\xB3\xD6\xD5\xCB\xF3\x06\x58\x94\xEF\x9C\x4E\x9C\x3C\xB3\xE0\x67\x86\xF6\xC3\x83\x01\xDA\xBC\x3E\x06\x49\xBD\xAE\x78\x70\x8B\xB9\xAB\xC3\x98\x35\x1F\xCB\xBA\x72\xF3\xAF\xC2\x98\x6F\x41\xB5\xEB\x07\x1D\x51\xC8\xC6\x03\xC6\x43\x74\xF0\x36\x63\x53\xB0\x19\xC3\x79\x6D\x9B\xC7\x21\x13\x09\x36\x54\x1D\xC0\xBC\x94\x35\x59\xA2\x67\xD5\xF1\xBC\x4F\xEA\x2F\x37\xAC\x57\x54\x01\xC5\xA9\x18\xC3\x6E\xD5\x05\x75\xA2\xD8\x6F\x32\xF0\x59\x7A\x4D\xCD\xDC\xBB\x86\x95\x9E\x9E\x44\xC0\xE3\x8C\x23\xA5\x6A\xC6\xE6\xBA\xC5\xBD\x11\x14\x8A\x84\x11\xFF\xED\xE3\x0A\xFC\x15\xC2\x24\x70\xEC\x3C\xDA\x24\xB1\x21\x28\x3F\x89\xC0\x73\x85\xB1\x3F\xC9\x67\x8D\x92\xF2\x64\x4F\xD2\x5B\x44\x2B\x28\x02\xD2\x21\xE6\x47\x6D\x9B\x47\xBA\x66\xAF\x33\x3E\x3F\xE4\x72\xC4\x66\x8B\x7E\xFC\x06\xFA\xCF\x13\x0C\x12\xEF\x8A\xF6\x3B\x42\xC9\xD8\x27\x70\xB1\x67\xBD\xF8\x91\x45\x53\xB9\xC6\x99\xA9\x40\x65\x82\x40\x25\x40\x38\x78\xA2\xE3\x61\xA2\xD9\x47\xD8\xD0\x4C\xAD\x1F\xF3\x64\x27\xDD\x35\xFE\x88\x8F\xD9\xC6\x9B\xC3\x35\x2E\x47\x18\x35\x64\xE2\xE8\x7C\x83\x36\x0B\x8C\xC6\x20\x17\xE3\x57\x40\xDF\x72\xD5\x46\x83\xEC\x6E\xC4\xCC\x8C\x71\xC1\xE6\x11\x17\x6C\x3E\xC2\x05\x5B\x47\x20\x80\x0A\xAB\xD7\xD1\xC9\xCD\xD0\xC2\x5C\xFD\x3A\x6D\x68\x14\x97\x9B\xEC\x66\xC8\x5F\x74\x1B\x4D\xAA\x71\x8D\x60\x7F\x3D\xDB\xC3\xAE\x50\xD5\xF9\xED\xD3\x2B\xEF\x75\xA6\xB8\xA4\xB6\x7F\x06\xEA\x5B\x45\x89\x23\xEA\x73\x89\x4B\xDB\x5F\x22\x52\xF0\x47\x4F\xEF\x37\xB6\x1A\xBD\x2A\x79\x56\xFA\x19\xA2\x42\x8D\x80\xEC\x63\x81\x1E\xFD\xA2\xCA\x63\x5E\x4B\x42\x79\xCC\x6B\x6F\xBF\x53\x1E\xF3\x4E\x85\xC2\x2F\x53\x79\xCC\x6B\x6E\xA5\x00\xE6\x77\x5D\x8F\xE5\x31\xD7\x5C\xFD\xEE\xEB\x77\xCA\x63\xDE\x21\xBE\xAF\x5E\x79\xCC\x81\x04\xC7\xE5\x31\xFF\xCE\xF5\xDB\x2F\x8F\xF9\x3D\xD7\xFF\xC3\x28\x8F\xF9\x6B\xA5\xAE\x8E\x38\xA2\x78\x90\x5B\x02\x9C\x68\x8A\x52\xCF\x2C\x68\x0A\x34\x1A\x6A\x87\xC4\x74\x32\x75\x56\xBD\x65\x11\xAD\xC4\x67\x55\xCD\x39\xCC\x95\xB8\x92\x60\x43\x82\xA3\x89\xEB\x0F\x00\xCF\x19\x79\xC5\x1D\xA2\x36\x2A\x18\x8B\x49\xA3\xB5\xAE\x82\xDF\x7F\x59\x6C\xEC\x1A\x24\xA6\x74\x90\xAA\xBB\x0D\xFA\xD8\x8A\xF4\x98\x2A\x63\xAD\xB6\x24\x42\x14\x17\x1B\x13\xA2\xA7\x48\x68\x76\xED\x52\x2D\x96\x36\xD4\x62\x31\x5A\x6B\xA5\xAB\xD5\x5B\x58\x08\x80\x44\xCB\xE1\x44\x16\xEB\xF4\x66\x74\x9D\xA9\x6E\x90\x65\x50\xED\xEA\x2D\x61\x28\x65\x18\x0A\xE7\x6E\xB0\x7A\x5E\xAD\xA8\xE7\xB3\x91\x7A\x3E\x23\xF5\x9C\x61\x62\xE1\x62\x13\x58\x50\x7B\xD8\xEF\x22\xCA\xA3\x0E\xA1\xD5\xE5\x90\xB9\xB7\x76\xDE\x6E\xF3\x55\x58\x35\xC1\x43\x5D\x94\x6E\x36\x7D\x27\xEF\x49\x3D\xF8\x3A\x16\x33\x73\xDE\x6D\x44\x18\x57\x4D\x32\xC8\xCF\x8F\x94\x01\xBA\x14\x20\xC1\x67\xB4\x4A\x33\xA0\xB5\x76\x99\xE8\x0F\x9D\x8A\xB5\x6F\x66\xB3\x99\x9A\xA1\x47\x6B\x6B\xDF\x94\x9C\xF5\x94\xC1\xF1\xF8\xCA\x7B\x46\x63\x68\x82\x0A\x96\x6C\x08\x7C\xFC\x2D\xD7\x83\xAE\x87\x72\xE5\x4C\x8F\x0D\xF3\x76\xB5\x6D\x36\x2F\x72\xBC\x7C\x7D\x21\x16\x54\xB8\x81\xA8\xC4\x19\x5B\x63\xE8\x59\x0E\xFE\x42\x5E\x3C\xEB\x29\x6C\x03\x85\xA3\xA3\x25\x41\xA8\x76\xED\xC5\xC6\x3A\x4E\x35\x7D\x13\x91\x48\xE1\xAA\x9B\x74\x86\x76\x29\x75\xE6\xB6\x7A\x8E\x9B\x57\x96\x99\x43\x19\xBF\x18\x9A\x2A\xD7\xD2\x54\xF9\xB5\xA0\x29\xC9\x09\xC2\xE1\xD9\xFA\xE4\x59\x67\x11\x0C\x3F\x6B\xFF\xD6\x75\x69\x73\xDD\xFB\x05\xB5\xE3\xD6\xAF\x2F\x7A\xCE\x29\xAC\x39\x5D\x64\x4C\xCB\x90\xBF\x2B\x97\x55\xFF\xBE\xB2\xE6\x48\x5F\x4D\x47\x5E\xF5\x7C\x90\x76\xC1\x9E\xE1\x05\xE6\x74\x8F\xB1\x83\x7D\x70\xB1\xE7\x13\x17\x7B\x67\x42\xDE\x6F\x11\x42\xA6\x66\x97\x1B\xF9\x21\x84\x4C\x55\xD1\x83\xDE\xD5\x0F\x37\xD1\xB3\xD2\x9C\x55\xAA\x9B\xF3\xD4\x1B\x71\xBE\x37\x82\x1E\x5C\xBB\x8A\x13\xDB\x12\xA6\x8A\x86\x5B\xCC\xB8\x22\xCD\xDC\xAB\x8B\x48\x83\x4F\x1E\x54\xA8\x5F\xE8\xF5\x61\x57\xA0\x04\x01\x74\x11\x83\x08\x2F\x3A\xDE\x2A\x57\x3B\x7B\x40\xF7\xFA\x23\x77\x7C\xCA\x4A\xB1\x63\xDE\xEC\x0A\x4F\xA7\xC2\x0B\x81\x16\xD5\x8E\xB9\x97\xFF\x9C\xE6\x43\x79\x93\x67\xBB\xE5\x7A\x84\x6F\x11\x3B\x68\x03\xDF\xD1\x62\x06\x3F\x14\xE3\xC8\xC2\xB1\x36\x5B\x24\xFE\x08\xBC\xC3\x30\x76\xBF\x39\xAB\x14\x3B\xF0\x0B\x47\xA4\xCD\xB2\x46\xFB\x3F\x21\xF4\x76\x26\x99\x23\x44\x69\x2F\xBD\x34\x5A\xEA\x22\x32\xA2\x4D\xA0\xFA\xCE\xE4\x24\x29\xB8\x42\xD4\xC8\xA6\xC0\x2A\x54\xC1\x7E\xA2\x7C\xDF\x7F\x63\x3F\x4F\x54\x96\x02\x1E\x1C\x51\x73\xEC\xE0\xC1\xEE\x2A\xB8\x0A\x12\xC8\xC2\x90\x92\x34\x63\x68\x6D\x57\x3D\x8C\x0A\x14\x74\x94\x85\x9A\x17\x17\x42\xF0\x9B\xDA\x95\x68\xB4\x6D\xA3\xF0\x5B\x25\xC1\xBB\x9C\xCD\x27\x0B\xE9\x2A\x67\xF6\xDA\x77\x5D\x97\x08\x10\x54\x49\xDC\x0A\x01\xA3\xB0\x01\xF0\x10\x40\xED\x3B\xE6\x49\xD2\xC3\xA2\x0D\xA0\x5A\xB6\x01\x54\x7B\xDD\x2C\xDA\x00\x66\x13\x1B\xC0\xEC\x22\xD2\x11\x56\x6C\x00\x33\xB6\x01\xB0\xDC\x37\xD5\xE4\x67\x6B\x6D\x00\xB3\xB1\x0D\x60\x16\x5A\x3F\xE6\xC9\x4E\xBA\x3B\x0B\x36\x00\xD4\x51\xAB\x02\x49\x55\x53\x1B\x80\x30\xB9\x37\xA3\x6C\x51\x60\x73\xF8\xD5\xC8\x98\x93\x85\xF5\x47\x8B\x2A\x48\xE9\x9B\x91\x52\x2A\x46\x1E\xB6\x80\x60\xE6\x30\x0F\xE6\x14\x81\x51\x20\xF5\xD4\x4A\xAD\x2E\x66\x54\xF6\x36\x18\x15\xC3\x33\x9B\xE8\x90\xE4\x40\x78\xA6\x2F\x57\x2C\x66\x2E\xC1\xF6\xE7\xB7\x17\x5E\xF1\x38\x37\xB7\xD0\x0B\xAF\x98\x5E\xBF\x6E\xC0\x79\x9E\x92\x2B\xFB\x4D\xC0\xE5\x47\xCC\xBC\xC2\x6C\x24\x9C\xC5\x55\x0C\x2E\xF3\x3A\xC0\xFD\x8B\x18\xC5\xCE\x16\x66\x24\xF0\x11\x24\x62\x04\xE0\xE1\x25\x2B\xC3\xD3\xA3\xE1\x69\x0C\x8F\x84\xF7\x0E\x51\x2C\x9C\xDC\x38\xB0\x41\xA2\xE9\xCE\x10\xE7\x28\x78\xE8\xE3\x6A\x38\xD6\xF3\x1E\xE4\x62\x22\x5C\x01\x87\x8B\xEC\xD5\xA1\xC0\xE1\x9B\x9D\xE9\xBB\xC4\x3F\xD7\x33\x04\x4E\x4A\x5F\xB5\x47\x41\x35\x89\x25\x61\x30\xE1\xE7\x7A\x97\xB9\x64\x4F\xD6\x97\x9F\xE2\x47\xB4\x7F\x7B\x8F\x6C\x8B\xE1\xEE\x78\x2B\x29\x15\x0B\x03\x92\x35\x4C\xDB\x01\xB4\x98\x8F\x4B\x6F\x0F\x3B\xD5\x77\x32\x53\x08\xB5\x51\x7D\x17\x9C\xFD\x32\x03\xCE\x70\x40\x4D\x3E\x04\xD4\x14\xD5\x65\xAD\x8F\x82\x13\xCD\x2E\xE5\x0A\xC0\x83\xC6\xE5\x36\x42\xB6\x9C\x1A\xE5\xD9\xA2\xE2\xA9\x0E\x49\xB6\xD5\x8F\x68\x6D\xC3\xD1\x71\x4D\xC7\xB3\x83\xB5\x22\xD5\xBE\x13\x9C\x4C\xB9\xFC\x21\x75\xCE\x16\xA1\x66\x54\x50\x30\x76\xBB\x12\xA1\x52\x07\x44\x23\x17\x99\xA6\x53\x57\xBA\xEC\xC2\x7E\x67\xB7\x3C\xCD\x84\xDD\x6A\xFF\x36\xB5\xC1\x1E\x9A\xC4\xC3\xCA\x97\xF2\xAD\xD9\x16\x3B\x50\x13\x3A\x51\x56\x03\xB1\x8A\xCA\xA9\xF6\xDB\xAE\x0B\xF6\xD1\x35\x3D\x0C\xFF\x23\x76\x54\x71\xE4\x56\x15\x2D\xC4\x81\x9E\x06\x07\x7A\xB6\xDE\x81\x9E\x8A\x4A\x02\x59\xCC\x2B\x97\x7D\x69\x15\x47\xD6\x35\xB7\x36\xA6\x29\xDE\x18\x6F\x5B\xAD\x38\xC2\x18\x4D\x28\x6E\xE3\xD2\x91\xBF\x43\x8F\xFD\x1D\x37\xAF\x38\xF2\x25\xF5\x67\xB9\xE2\x48\xE8\x8F\x9A\xF4\x27\x56\x1C\xF9\xCE\xEB\xE2\xD8\x37\xCE\xB6\xEF\xC1\x97\xCF\x7F\x0D\x23\xC0\xEC\x28\x02\xEC\xDD\x5F\xA9\x08\x30\x39\x34\x83\x91\xD3\xB2\x3B\x00\x01\x0C\xD5\x6F\x9B\x2F\xBA\x40\xCE\x71\x51\x0C\xC7\x4F\x61\xA8\x93\x7C\xF3\xF9\x4B\x86\xE1\x4A\xE9\x99\xF5\x33\x98\xCA\x64\xF0\x1C\xC6\x35\xD6\x2B\x6E\x93\xC9\x63\x01\x4C\x2C\xBE\x25\x3C\x2B\xC6\x99\xE4\xD8\x30\x8E\x7A\x29\x8C\x43\xEA\xD0\xAC\x96\xDE\x1D\x1A\x9E\xBC\x99\x5E\x72\xA7\xB0\xFD\x9D\xC2\xF6\x77\x0A\xDB\xDF\x29\x6C\x7F\xFB\x85\xED\xEF\x30\x8C\x3B\x0C\xE3\x0E\xC3\xB8\xC3\x30\x6E\x9B\x61\xFC\x6A\x69\x93\x23\x7D\x35\xBA\x96\x5F\x55\x53\x6B\x1B\xE7\x8D\xC0\xAC\x95\xF8\x60\xFF\x82\x25\xAB\xF4\x86\x3E\x5C\xD9\xEA\x66\x5E\x3D\x02\x08\xD8\x97\x15\x0C\x20\xB4\xA4\x36\x98\xA8\x48\xED\x42\xE2\xDB\x0B\x12\x5C\x2B\x16\xBB\x8A\x53\x62\xAA\x17\x01\xA3\x20\xF6\x3A\xFA\xD1\x55\x2F\x8A\xF0\xCA\x59\x21\xD1\x6A\xE7\x2C\xBF\x5E\x72\x65\x42\x5A\x87\x95\xF8\x9C\x60\xE0\xB3\xAE\x1A\xF2\x60\x38\x56\x26\xE9\x2C\xA9\x60\x1E\x48\x01\xB9\xAB\x77\x4C\xCB\x7F\x68\x28\xE6\xD2\xA2\x81\x39\x85\xA3\xD0\xC2\xA8\x1F\x65\xBC\x6B\x37\x67\x20\xB2\xDA\xD9\x9E\x0B\x8B\x6F\x7A\x73\x48\x44\x74\xC2\x95\xDE\xD2\xFF\xAF\x6C\xD1\x2F\xCF\xF5\x8B\x93\x32\x05\xF5\xC1\xE2\x14\x8F\xBB\x3E\x58\x6C\xCA\x8F\xC5\xC1\xE2\x2E\xFE\xB1\x38\x58\xBC\xC6\xCD\xA1\x1F\x87\x37\x44\x1C\x42\xE5\xEA\xB3\x4A\x4D\xD2\xD8\x13\x51\x24\xDE\xB0\xDF\x71\x38\x32\xAB\x9A\xF5\x10\x4C\x40\x93\x74\xDF\xE2\xB5\x01\xD8\x75\x4E\x6A\xFA\xB3\xEE\xC4\x53\x7D\x57\xD2\x5B\x5E\x2B\x18\x99\xEC\xC8\xBD\x8F\x93\x2F\x5F\x56\xFB\xDD\x69\xF7\x5A\xFF\x86\xFD\x03\xC9\x03\xBE\xC9\x0B\xC4\xEA\xD9\x28\xF7\x1A\x77\x97\x3B\xD9\x7E\x3F\x2F\x6D\xC3\x9E\x61\xC9\x41\x32\x8B\xC2\x6D\x4A\x42\xB0\x76\xC5\xB3\xDE\xC1\x9F\x04\x1A\xAC\x49\x69\xE0\xDF\x69\xE3\x0D\x17\xF9\x0A\xF8\x2C\xF2\x82\xE9\xA2\x13\xBB\x02\x56\x68\x13\xC5\x3A\x93\x46\xB9\xD3\x4D\x02\xE0\x3E\xC3\x25\xAD\xE8\xAD\xAE\x39\x70\xCD\x01\xDE\x6D\xDD\x29\xA9\x46\x6F\x9C\x0D\xCD\xEB\xF1\xBB\x19\xC5\xB3\x1B\x2E\xCA\xBB\x35\xBD\x5B\xCB\xC5\xF0\xEE\xD7\x2E\xB4\x43\x5C\x5C\xED\x5A\xAE\x62\x5F\xF2\x2A\xDC\xB7\x2B\x4A\xF2\x1B\xF6\x83\x21\x68\xCD\x6C\x51\x87\x84\xB8\x42\x5F\xB9\xA7\x03\x85\x7C\xF9\x3A\xCB\xD0\xC1\x0C\xC2\x0C\xF2\x8A\x14\xC7\x68\x71\xCF\xF5\xB2\x60\x61\x4A\xC3\x4E\xD9\x40\xE2\x34\x5B\x26\xC4\xCE\x14\x3E\xEE\x03\xC1\xA0\x5E\xB2\x35\x11\x89\x21\x29\xCC\xA2\x4E\x2D\x75\x01\x56\x15\xCD\x29\x26\xDD\x8C\x0F\xB6\xC1\x3B\xDD\x95\xE1\x60\x8E\x55\xC1\x67\x7B\xFC\xD3\xE0\xB2\x76\xE5\xDE\xCE\x50\x10\xEB\xB5\xAE\xF6\xA6\xEF\x6A\x5E\x87\x86\xE3\xF5\x6B\x24\x74\x8E\xDF\xA7\xA3\x45\xC9\x4A\xA0\x5B\x30\x2A\xBD\xAA\x06\xB3\xCA\xCB\xAF\xD5\xAF\x09\x9C\xED\x93\x6A\x29\x6A\x86\xB8\x48\x22\xB6\xAB\x27\xBC\xBA\xC8\x49\xFE\xB0\x08\x26\x5E\xB2\x97\x0A\xDE\x12\xCA\xDF\xC7\x6A\xB8\xFF\xAD\xCF\xDB\x2B\x8D\xE4\x29\x26\x47\x23\x7E\xD0\xA5\x2F\x1D\x71\xD9\xF2\xEC\xA5\xFF\x06\x8F\xDC\x2F\x51\xF7\x08\x3A\xEF\x9D\x1A\xA7\x63\x98\x21\x51\xC1\x0C\x89\x0A\xB9\x0B\x41\x3C\xE1\x78\x84\x27\x3C\xE1\xCB\xFE\x86\x0E\x96\x7F\xE2\x88\x5E\xED\x98\x4B\x4E\x3F\xA4\xCE\xC1\x8F\xAD\x76\xCC\xB6\x93\x3B\x17\xA5\xD3\x7E\xBB\xC7\x42\xB0\x01\xAF\xDC\x31\xF7\x2E\x5F\x6D\xC7\x57\xEF\xE7\x26\xDA\xC9\x4D\xED\x52\x13\x9B\xF1\x6A\xC2\x57\x87\x26\x12\x2E\xAB\xB7\xBE\x4A\xA9\x59\x29\x4B\xCA\x90\x07\xE8\x86\x8E\x15\x45\x5D\x11\x7C\x6D\xA1\xE6\xD9\xB6\x79\xE3\x42\x4A\xA4\x6E\x70\x3E\x0B\x49\x1C\x4E\xFB\x8D\xBE\xFD\xBE\xEB\xA1\x0A\x29\xDD\xB7\xCD\xCB\xF4\x07\x8A\x4E\x77\xBD\x6D\xEE\x5F\xCC\xE8\x15\x5C\xFC\xED\xBC\x2B\x82\x35\x3A\xD4\xF2\x7E\xAA\xAB\x11\xFE\xC8\x2B\x08\x13\x6B\xC3\x39\x47\x0A\xA9\x60\xC6\xDF\x37\x0A\xDC\x90\xFD\x46\xF7\x72\xAD\xC3\x06\x97\xBD\x5A\xCC\x5D\x0C\x3B\xAF\x98\x8A\x69\x16\x89\x69\x21\x84\x63\xC3\x1F\xF5\xDD\xC6\x10\x43\xDE\x72\x0C\x79\x2C\x09\x1B\x04\xB3\xC7\x9D\xED\x4E\xE0\xB6\x93\xD1\x35\x82\x70\x4B\x6D\xAD\xB1\x29\x1B\x87\x56\xAB\x97\x22\x91\xEB\x04\x08\xCA\x9D\xB8\xDC\x10\xCF\x39\xC1\x41\xBB\xAB\x11\xA8\xAE\xAB\x38\x02\xF5\x4C\x67\xD7\x47\xA0\x3A\x8E\xEC\x3D\xC3\xC7\x6F\xC5\xA6\x94\x10\x7F\xBA\x39\x18\xCB\x70\x57\xCB\x86\x94\x6A\x54\x2E\x98\x33\x36\xCB\x8A\x69\x43\x3B\x13\x31\x14\x80\x06\x21\x48\x87\xC6\x27\xCF\x62\x21\xBA\x53\x92\x7D\x82\x84\xE3\x72\xB1\xE9\x6C\x80\xBA\x2D\x5D\x76\x29\x36\x87\xB1\x13\x17\xE6\x71\x3C\x80\xCC\x21\x5A\xE2\xCE\x6C\xB9\x6A\xCD\x38\x1E\x60\x99\xE0\xFE\x00\x9A\x81\xB8\xCC\x4A\xC6\x71\xAF\x93\xD7\x82\x6A\x1E\x63\xE4\x83\x30\x88\x6A\x97\x91\x12\x40\x96\x01\x60\xB8\x72\x85\x3B\x89\x64\x48\xF8\x5A\x6E\xD8\xFD\xDE\x95\xB7\x49\x9E\x8C\xBE\xB6\xC9\x44\x48\x14\xC7\x2D\xAD\x0D\x73\x2C\x7D\x7B\xE8\xDA\x83\x8E\x0E\xE7\x03\x67\xDC\xDC\xEB\xFD\xAD\xC5\x5D\xEE\x24\xD1\x60\xE5\x4E\x9E\x55\x89\x1C\xAF\xC7\xB6\x21\xE9\x80\xD5\x59\xA5\xAF\x30\xCE\xC9\x80\xCB\x0A\x7F\x9F\xED\x8C\xBF\xA1\x19\xBE\x41\xA3\x94\xBF\x33\x74\xD9\x95\xBB\x72\x94\x34\x02\xDD\x9A\xF4\xC8\xC9\xEB\x4A\x7F\xDF\xBE\xBF\x8F\xF1\xD3\xEE\x12\xAF\x49\xE9\xEE\xDA\x45\x16\xA1\xA9\x11\x09\x7C\x93\xFE\x50\x5F\xCC\x08\x21\xEA\xE4\x10\x48\xC0\xC1\xC1\x9D\xF1\x9F\xA6\x7D\xF6\x69\xF5\xF0\x16\xE7\xFB\x8E\x63\x3A\x57\xEF\xB8\xC9\x04\x1A\x0C\xBC\xB3\xFE\x73\x74\xC0\x7C\x4E\x5D\xDE\xEA\xC0\xAC\xAD\x7F\xB2\x5F\xBC\xC6\x19\xFA\x63\xB0\xB7\x6F\x68\xFC\xE7\xCA\x16\xE3\xAE\xC4\x49\xAA\x30\x49\x25\x2C\xBF\x98\x1A\xA4\x15\xEA\xBE\x3B\x45\x4C\x80\xE6\xA9\xA2\x79\x72\xA7\x90\xAD\x59\x49\xE8\x4A\x15\x26\xCC\x30\x3D\x87\x09\x33\x3C\x61\x48\xDA\xBC\x6B\x57\xE8\xB9\x16\xA4\x46\xE3\x4A\xF7\x9A\x4B\x32\xE9\x27\x63\xAD\x5B\xB7\xB9\x28\x19\x93\xD3\xD2\xC7\xD1\xE6\x4E\x6E\xBE\xB9\xCB\x5B\x6E\xEE\x72\xB4\xB9\xCB\x63\x37\x77\x39\xDA\xDC\xE5\xCA\xE6\x9E\xAD\xDD\xDC\xE8\x74\x12\x10\x4D\x6B\x1E\xD3\xDC\xC7\xDC\xF4\x3A\x47\xF8\x0E\x83\x37\xCC\xC3\x68\x11\x00\x34\x66\x5F\xFA\xCF\x03\xFB\xCA\x2B\xF4\x71\x1D\x0B\xD3\x81\x0D\x33\xA4\x28\x9F\x09\x27\x2E\xCB\xB7\x5B\xF7\xDB\xAC\x2F\xCF\x1D\xFB\x6D\x42\xBF\xCD\x50\x9E\x7B\x73\x28\xCF\x3D\xF4\xDB\x0C\xFD\xB6\xA1\xDF\x45\x5C\x99\x78\x9C\x9E\xE4\xAC\x4B\x1C\x43\x88\xE4\xA7\x25\x48\x5C\xC5\x81\x79\x1B\xB4\x58\xD1\xF7\x13\x9C\xF1\x46\x69\x53\xD1\x93\x49\x67\x7D\x79\xA5\xD1\xDE\x39\x4B\x6C\x89\x91\x60\xE9\x02\xBE\x88\x47\xDE\x4E\x58\xF3\x57\x6B\xAC\xD9\xFA\xB1\x0E\xD5\xB7\x37\x38\x6A\xDA\xCD\x68\x57\x22\xCC\xE5\x7E\xB0\xFB\x51\xD9\x5E\x88\x70\xAE\x24\x49\xC0\xBC\xF4\x8D\xE7\xEC\x93\xCE\x8C\x93\x62\x9F\x40\x6C\x9F\xDD\x31\x4F\x2C\xA5\x77\xF2\x45\x5B\x71\x5A\xA4\x1B\x44\xD0\xF3\xD1\x50\x06\x5C\xFF\x90\xF6\xB1\xC6\x3B\x80\x64\x0F\x71\x0B\xD0\x9C\x0C\x74\xBD\x26\x59\x8A\xC8\x99\xA5\x8A\x33\x2B\xB9\xC6\x1A\xD8\x85\x7C\x75\x73\xF9\xAA\xC8\x45\xC7\x74\x60\x7B\xD4\x81\x07\xC2\x79\xBA\xB6\x03\xF7\xC7\x0E\xDC\xBF\xAE\x03\xF7\x86\x0E\xDC\xBB\xBE\x03\xE7\x59\x82\x06\xB6\xD3\x1B\x87\x27\x43\xF1\xDE\x7C\x08\x70\xFC\x64\x14\xDC\xCD\x60\x9F\xF8\xC1\x4A\x17\x47\x82\xDC\xCC\x70\x4B\x83\xB9\x2A\xE9\xD2\x3D\x67\x2E\xA3\xF0\x42\xE6\x0C\xE2\x83\x53\xA7\xF7\xBA\xC2\x21\xD1\x39\x15\x63\x66\xB1\xD7\xE5\xDC\x07\xD3\xBB\xD4\x1F\x3D\xDD\x2F\x4A\x97\x4A\x05\xA1\x7C\x9F\x43\x3D\x52\x60\x0C\x24\xE0\xF2\x0F\x2A\xA9\xAD\x90\x48\x12\xA0\xE4\x93\xE7\x3D\xF0\x54\xF3\x60\x1D\x1A\xDF\x3B\xFA\x62\xC6\x5F\xEC\xF8\x4B\x32\xFE\x92\x8E\xBF\x64\xE3\x2F\xF9\xF8\x4B\x31\xB1\x00\x65\xEC\x73\x2A\x78\x3A\x18\x8D\x37\xC3\xC8\xF6\x16\xA5\xCB\x18\x4E\xC3\x99\xBD\x6E\xE6\x60\xCB\x43\x00\x40\xE2\x32\x8C\x3A\x80\x58\x66\x7B\x34\xEC\x02\x50\xD4\x30\xDC\x66\x32\xF8\xD9\xF2\xE0\x91\xD8\xC0\x70\x5D\x1C\xA4\x3C\x0C\x7E\x7C\x2F\xEE\x1C\x8F\x7F\xE6\x4D\x3F\x9E\x82\x99\xB7\xFD\x78\x16\x68\x63\x8E\x27\x62\xC6\x79\x23\x71\x2E\x66\x9C\x5F\x12\xA7\x63\xE6\xF3\x7E\x34\x23\xE8\x5B\x41\x5B\x3B\x0D\x13\x93\xF0\xC4\x94\x6C\xB5\x2E\xF7\x48\x63\x99\x18\xAE\x13\xA0\x81\x03\x49\xC8\x3E\xBC\xE5\x41\x92\x6F\xDA\xA2\x13\x3E\xA7\xD3\x3C\x15\x1C\x3F\xA2\x87\x06\x29\xDA\xC8\x8E\x5C\xA2\x8A\x9C\xC3\x6B\x57\x88\x03\x3E\xE8\xFC\x31\x5E\x95\xAF\x26\x6D\xA0\x8E\xCD\x73\x7D\x97\x13\xBB\xE3\x0D\x22\x9E\xCE\xD9\x1E\x2D\x7D\xEE\x8F\x1E\x01\xE9\xCF\xF6\x48\x04\x44\xE0\x71\xE2\x66\x7B\x5D\xE2\xF5\x61\x37\x73\x33\x97\x5F\xDE\x42\x5C\x50\x1E\x8A\xBA\x5C\xDE\xEA\xB2\x21\xC0\x4E\x6C\xB3\x80\xD4\xA5\xC1\x93\x34\x5C\xF6\x5D\x0E\x4A\x33\x71\xFB\xE4\xB7\xDC\x3E\x39\x89\x53\x33\x70\xFD\xAF\xC1\x06\x8A\x36\x25\x2D\x1E\x90\xBD\x8E\xD1\xDB\x9D\x00\x7C\x1A\xD4\xEA\x08\x86\x12\xE4\x01\x14\xCC\x4F\xD8\x6E\x76\xF4\xB4\x13\xEF\x08\xB1\x18\x1A\xB7\x65\xE9\x86\xC6\x3D\x73\x49\x4F\x4A\x70\xA0\x51\xCB\x31\xFE\x08\x9D\xC7\xEF\x0B\x8D\x7B\x16\x9C\x38\x17\x26\x60\xE6\xB4\x3C\x40\xFB\xC7\xE9\x61\x13\xD1\xFE\x71\x7A\xD8\x44\xB4\x7F\x9C\x1E\x36\x11\xED\x1F\x44\xB8\xC7\xEF\x29\x7D\x4F\x87\xEF\x19\x7D\xCF\x86\xEF\x39\x7D\xCF\x87\xEF\x05\xED\x6C\xC9\xCB\x4F\x63\x29\x7B\x41\xF2\x2D\x26\xEE\x07\xB3\x0E\xE4\x20\x1F\x78\xF4\xBD\x31\x05\x9D\x73\x25\xDB\xF7\x72\xB4\x1B\x0C\x33\xF2\x25\x24\xA7\x57\x7F\x2C\x70\x18\x47\x5E\xB5\xDF\x1F\xA2\xB4\x68\x1A\xFC\x26\x42\x7B\xD8\x46\x2E\xE9\xDE\x08\xFD\x69\xBF\xF7\xF9\xC1\x66\x03\x8C\x52\x00\xA0\x1D\x41\x58\xB1\xC8\xCA\xAE\x19\x36\x41\xB7\x5F\x80\x11\xEE\xB4\x18\x4C\xD8\xDB\xD5\x7E\x96\x2B\xD3\xC3\x43\xC5\xF7\x76\x16\xA0\x74\x7F\xE1\x61\x52\x84\xE0\x8A\xE9\x6C\xFB\x3D\xCF\x23\xAA\x47\x1F\xD3\x8C\xD7\x07\xA1\x25\x81\x2A\x44\xAA\x3B\xBA\x7E\xD7\xC3\x1C\x12\xD6\xFE\x20\xB5\xF1\xA0\xD2\x12\xF4\xD5\xB2\xBD\x5E\xC3\x24\x50\xFD\x1F\x46\xA7\x47\x5E\x05\x70\xB7\x96\xF3\x58\x12\xF8\x5D\xF0\x89\x94\x9D\xCC\x7F\x23\xE3\xE2\x01\x4C\x26\xA5\x9F\xF2\xE9\x4F\xAF\xD3\xC6\x25\xAF\xD3\xE6\x69\x60\xB4\x82\x69\x01\xF2\x0C\x71\x48\x3E\xA1\xE5\x4D\xFD\xA6\x98\x50\x52\x9E\x99\x94\x8D\x3B\xA9\xB3\xED\xB5\xE7\x81\x67\xC7\xB5\xFF\x31\xA7\x19\x3F\x2F\x35\x28\xB7\xA9\xAF\xAC\x7D\x5B\xAF\xE0\x63\x4B\xE9\xD7\x4D\x46\x78\x94\xE7\xA3\xB3\x8D\x53\x59\xB6\x4D\xCB\x86\x11\x84\xD6\xB5\xDC\xCE\xA6\xD8\x93\x63\x91\x14\xC8\x29\xEC\x0F\xA2\x27\xB8\xA9\x4A\x66\x44\x73\xD4\x6C\xC2\xE0\x19\x89\xB7\xD8\xBD\xCE\xBC\x5E\x1B\x80\x3D\x02\x94\xA3\x75\x09\x2F\x4E\xE5\x6C\xF5\x41\x03\xD9\xC0\xAB\x6D\xF3\xD9\xF7\x3C\x7F\x4D\x75\x9A\x3B\xFE\x91\x7F\x7F\xFD\x9A\x5A\x18\xFE\xEB\xB4\xAF\x9F\xEA\x83\xC5\xC5\x99\x2E\xA1\x7D\xF6\x6C\x97\xD2\xCF\xAC\xB1\x65\x4E\x5D\x41\x12\x0F\x12\x1E\xD1\x8B\x74\x51\x20\x88\xD4\x2B\x40\xCE\x74\x99\xCB\xB7\xBA\x02\x2D\x41\xC3\x77\xD9\x1E\xFD\xA2\xBD\x62\xA4\x7C\x67\x61\xAD\xBB\xC0\x70\x37\x34\xB3\x8E\x5A\x6D\xD4\x24\xB1\x36\x71\x68\x42\x6C\xA3\xB4\xDD\xE7\x36\xD5\x16\xD2\x0F\x49\x3E\x6F\xEF\xE7\x86\xBF\x3A\xD5\x63\xC8\xD9\x1E\x2D\x32\xBB\x43\x04\x2F\xDD\x6B\x5C\xF2\x47\xE3\x3B\xD8\x91\xE7\xCD\x5F\xE3\x00\xB9\xD1\x4D\x40\x63\xAB\xBE\x99\x03\x76\x62\x28\x6E\x27\x3A\xB9\xD3\xED\xFB\x78\x7F\x92\x7E\x1A\x08\x9E\x94\xA5\x3F\xBD\xA1\xF6\x85\xDE\x6D\xB8\xD0\x99\xF6\xBB\x9E\x17\x20\xCF\xCD\x3E\xE0\x2C\x80\xD1\x6B\x0E\x92\xA4\x41\x0E\xBA\x2C\x50\x20\x19\x38\x75\xD3\x19\x67\xDA\xEF\x7E\x9E\xD9\xAB\xE1\xB7\x0A\x1C\x44\x6D\x18\x8B\xC9\x5F\x6B\xC5\x12\x4E\xDC\x3A\xF6\x86\xB4\x30\xE9\x08\xE3\xFE\xD1\x0F\x60\x1D\x8D\x5E\xBB\xD5\x3B\x0B\x1A\x82\x2B\xB2\xFD\x7B\xE0\x31\x95\xD7\x55\x47\x9D\x95\xFB\xE5\x18\xE0\xFB\xE5\x1E\xAF\xAB\xEF\x65\xDC\x0C\xF6\xCD\x9C\x0E\x30\x56\x20\xCE\x77\xB6\xFB\x9E\x31\xC3\x9D\xF1\xB5\xD3\x4B\x8F\x6E\xD0\xF1\xB8\xE9\xF4\x78\xDE\xF4\xF2\xBC\x41\x54\xDE\xA4\xE3\x88\xD5\x06\xAF\x7C\x29\x73\x82\x40\x55\x5F\x31\xC3\x93\x9F\xC0\x35\x43\xD7\x7E\xC6\x68\x73\x34\x24\x7A\x26\x91\x61\xA4\xFE\x35\x97\x1B\x46\xC0\xAD\x3B\xE5\x4B\xF6\xF5\x01\x8D\xD1\x24\x40\x97\x94\x05\x7F\x59\x3D\x4D\x84\xBA\x25\x35\x4B\x52\x8E\x1F\xD3\xA4\xC3\x81\x9F\xEC\x06\xEF\x5A\xEA\xFF\x7A\x2F\x40\xB2\xB8\x00\xB8\x16\x9F\x0B\x56\x64\x18\x13\x40\xC1\x65\xC2\x13\x31\x5C\x6F\x9B\x9A\x38\x5E\xE2\xF5\xEB\xB5\xA4\x27\xD5\xAC\x5E\x46\x06\x2F\x79\x68\xEC\x3C\xC1\x1D\x44\x1A\xBA\xFD\xBE\xE7\x11\x8F\xDC\x02\xB6\x14\xEA\xDD\x70\x92\xC8\xE1\x41\x9D\xC9\x3C\x62\xBB\x80\x9F\x0E\x8E\x0E\x21\x48\x0B\x74\x26\x9D\x4F\x60\xC1\x01\xD2\xA8\x25\x0E\x42\xFC\x98\xC6\x49\x3C\xE3\x41\x8E\x33\x05\xD7\x9E\x5D\x0C\x8A\x6B\x50\xD3\xC2\xF1\x83\xFC\xB5\x28\x17\xBC\x15\x9E\xF6\xEA\xE7\xB4\xCE\x8F\x30\x3D\xB8\x90\x7A\xBB\x4F\xF2\x00\x7E\x39\xEA\xBD\x7D\x0C\x3C\x13\xB0\x5F\xA9\x7F\xFB\x3E\x4A\x4E\xE6\x2E\x81\x1F\x3A\x0D\x68\xD4\xC1\x69\x4C\xA2\xAB\xB8\xF6\x2A\x78\x99\xBB\xDA\x59\x57\x3B\x7B\x69\xAB\xB3\xA4\xA2\xE3\xEF\x4C\xFE\x96\xF4\x17\x6C\xBC\xED\x17\x29\xF2\x12\x93\xBE\xCB\xF9\xD8\xCE\xB8\xE2\x17\x60\xC5\xF8\x65\x40\x96\x75\xD6\xE5\xF1\x31\xF8\xBE\x33\x12\xC6\x32\xF6\x74\x57\x89\xF2\xAA\x7A\x6F\xA2\x93\x11\x5E\x68\xBA\x04\x66\x19\xB0\x20\xFD\x67\xF2\x2B\x80\x54\xE2\xBF\xC6\xE9\x4B\x1C\x64\xAF\xFD\xEF\xE4\x97\x22\xF1\x19\x0F\x07\x90\x71\x36\x10\x1F\x84\xE8\x81\x68\xB9\x26\x06\x4D\xFD\x88\x64\x0D\x93\xEC\xF2\x53\xBC\x0B\xB3\x48\x6E\xB9\x9F\xB9\x0C\x34\x9C\xB9\x64\x4B\xE8\x2E\xA3\x45\x6F\x39\xF6\x02\x01\xEF\x99\xEC\x0E\x21\xC4\x8C\x09\x31\x19\x08\x31\x89\x84\x98\x08\x21\x16\x0C\x01\x07\x42\x4C\x5C\x31\x10\x62\x41\x84\x58\x4C\x08\x31\x68\xAD\x59\x24\xC4\x8C\x08\x91\x05\x65\x22\xC4\x82\x44\x47\x84\x4F\x66\x42\x88\xE2\xA1\x6F\xA9\xDB\x0F\x2A\xCD\x90\x66\x74\x04\x5D\xC1\xCC\x9A\x4B\x9C\x87\x92\xB8\xD4\xDF\x2D\xC0\x5A\xA9\x3F\x23\xB1\xFF\x74\x8E\xCB\x27\xFA\x53\xBB\xD4\xBF\x7C\xFD\xBF\xDA\x31\xDB\x2E\xF5\xBF\xFC\x3B\xD7\x91\x70\x0B\xFA\x2B\x7A\x97\xF9\xCF\xE4\x5E\xB7\x1F\xB8\xCE\xC8\xAA\x6B\x57\x04\x77\xB6\x1F\x04\x4F\x61\x3A\x8D\x5F\x01\xE8\xE8\xD2\x01\xEA\x34\xA9\xDE\x35\xD3\xF9\x28\x09\x27\x09\xD4\x81\x24\x12\xC4\x54\x5E\x10\xD0\x27\x46\xC6\xC3\x3E\xE0\xB4\xE2\xCD\xAE\xE8\xBB\xD2\x7F\x48\xFF\x8F\xDD\x2C\x98\x6F\x4A\x37\xF3\x1F\xD2\x4F\xED\xF9\xD3\x4F\xF5\x74\x07\xE3\xAF\xB3\x70\xC0\x81\xDF\x92\xBC\x88\x48\x92\x8C\xA3\xFD\x51\xEA\x27\x17\xE0\x58\x1A\x52\xE9\xB2\xBD\x2E\xF3\xE6\x19\x6A\x8B\x33\x2E\xB2\x2D\x57\xB8\xB4\xDF\xF3\xBA\xF7\xBF\x00\x7B\x40\xC9\xE1\xEB\x63\x60\xA3\x1A\x91\x0F\x35\x07\x93\xE7\x95\xCB\x17\xA9\xCB\x45\x75\x63\xD9\x95\x36\x49\x89\xD1\xD5\xE2\xB9\xE3\x9D\x8A\xC4\xE1\xF1\xFB\xF2\x2D\xA7\x84\x78\x00\xBC\x57\xB6\xAF\x84\xF4\x8E\x4D\xB4\x1F\x07\x91\xC2\x6A\x80\x69\xC8\xC2\x34\xE4\x2E\x8B\xD3\xC0\x98\x45\x9B\x8C\x59\x44\xBB\xFA\x88\x5D\x49\x03\xE4\xE2\x69\x92\x3E\x68\xFC\x1C\xDE\x5F\x48\x6A\x1C\x4A\x84\x73\xC3\xE9\xD0\x70\x3A\x6E\x18\x51\x50\x39\x62\x58\x8D\x54\xD4\x4A\xFC\x11\x9F\x60\x6A\xDB\x38\xE4\x70\xED\x18\x87\x74\x35\x75\x85\xA5\x86\x07\xD5\x19\x54\x4B\xCB\xE9\x55\x82\xB2\x7E\xE2\x1A\x63\xE5\x1A\x91\x88\x1E\x54\x1B\xD4\x8B\x70\x57\xE2\x37\x02\x28\x5D\x71\xE2\x6F\xE2\xD6\xCE\xE0\x14\x38\xAB\x36\x10\xFA\x25\x49\xC6\x3C\x52\xBF\x29\xF9\x47\x24\xC8\x66\x9D\x11\xB0\xDF\x02\x9E\x5D\x31\x97\x17\x0C\xCF\x26\x29\x2A\xE2\xD6\xE0\xDB\x79\xCB\x05\xDD\x14\xFB\xD7\x19\x54\xEB\x44\x44\x0E\x75\xC8\x78\xBD\xBB\xC5\x83\x0E\x26\x29\x0C\x10\x72\x67\x11\x02\x7B\x25\x2E\x89\x26\xC1\x9B\xC3\xCE\xB0\x5B\x30\x5E\xEF\xE4\xAC\xCF\x21\xBA\x30\x3D\x30\xF0\x5E\x80\x0E\x4E\xF9\x5A\x16\x60\x32\x53\xE6\x3F\x23\x48\x09\x26\xB5\xDA\xA2\x0A\x40\x0D\xDD\x91\x08\x91\x77\xCB\x5A\xAA\xDA\x36\xA7\x79\x79\xBF\x08\xC2\x32\xC3\xFA\x9B\x35\x84\x95\x4D\x08\xAB\x58\x25\x2C\xA4\xC1\x84\xC5\xA4\x8E\xF2\xB8\xC2\xF2\xD7\x27\xBE\x4D\x96\x1F\x2B\xE0\xB1\xBF\xF2\x40\xFE\x63\x24\x7A\x37\xDA\xC5\xDD\x78\xCB\xB2\x5B\x77\xCD\xAE\x4D\x79\xB0\xA3\x8D\x5B\xDC\xC6\xC6\x0D\x75\x9B\x97\xB7\x43\x3A\xDD\x67\xE9\x78\x9F\xA5\xC4\x70\xF8\xF8\x21\x22\x41\x2A\x39\x97\x52\xC0\x91\x0A\x71\x1D\xF2\x43\xCA\x97\x35\x42\x1A\x59\xA1\x19\xC1\xBE\xFE\x5B\xAD\x33\x52\x5E\x3F\xCB\xBC\x77\xCC\x0B\xF5\x84\x17\x1A\x00\x7A\x8A\xCB\xB8\x4B\x6E\xDA\x39\xBA\x43\x32\xE8\x47\xBC\x10\x80\xA0\x56\x02\xFD\x98\x17\x86\xE0\xBA\x38\xB1\x08\xAB\x33\x93\x59\x35\x5B\xA4\x41\x8D\xA7\x34\xBD\xC5\x94\x92\x58\xB4\xD0\xCE\x0A\xFA\x5E\xD2\xB7\x1F\x12\xF0\xBD\xAF\x20\x8C\xDC\xA7\xD4\x3A\x18\xB9\x0F\xE9\xAF\x3A\x8C\xDC\xBF\xD4\xA4\x93\x07\x2D\xE2\x35\x63\x2D\xC2\x48\x6C\x00\xF2\xD7\x5A\x9E\xFA\xCD\x01\x76\x1F\x79\x5E\x75\xF4\x19\x20\xCC\x4C\xFB\xB7\x3D\xC6\xD4\x89\x33\xBE\x62\x1C\xDB\xB6\xF7\x6F\xDB\x0F\xD1\xAF\x8B\xCC\x59\xC6\xBD\x25\xB9\x4C\x1F\xB0\xFF\x20\x93\x40\x49\x80\xB9\x1F\xC2\x84\x44\x7A\x55\x46\x47\x13\x31\x3A\x5A\xD4\x68\x9C\xCA\x9C\x16\x49\xC1\x20\xE8\x14\x91\x41\x18\x55\x54\x46\x82\x69\xE6\x67\x8C\x4E\x97\xAD\x2D\x41\x1A\xD3\x82\x2C\x1F\xA5\xB1\x38\x0D\x66\x55\x39\xA2\xCB\x69\xFB\x43\xD7\xD7\x60\xCE\x68\x41\x58\xAE\x65\x72\xA9\xC5\xD7\x5E\x64\x75\xBA\xEE\xEC\x85\x86\x5D\xA7\xA4\x24\xB2\xBE\x52\xC3\xBB\xEA\xEB\x8B\x8D\x71\xA9\xFF\xC6\x7E\x7E\x32\x31\x4B\xFF\x12\x33\xEE\xCF\x6B\xD7\x29\x6B\xA8\xE1\x43\x0F\x6B\x25\xFF\x8C\xFC\xD5\x4A\x0C\x34\x12\x49\xDB\x12\x49\xEC\x98\x1A\xCF\x68\xD2\x46\xEB\x95\x6B\xCE\x8A\xD1\x87\x4D\x0D\xB8\x12\xEC\x40\xBA\xFA\xF9\x04\x60\x83\xCB\x28\x59\x2D\x46\x2C\x86\x12\xA9\xE0\x98\xFA\xBB\x1E\x86\x67\x33\x01\x5E\x12\x0B\x77\x5E\x85\x08\x9F\xDA\x54\xF6\x3C\x1D\x12\x35\xAA\x5E\xB0\xF4\x76\x43\xEF\x83\xA0\x21\x57\x8A\x0C\xD9\x2F\x06\x00\x52\xA2\x99\xA1\xED\x5A\x23\x4D\x5B\x02\x2A\x78\xCE\x89\x6D\xA3\x7E\x43\xC2\xB1\x43\x97\x1B\x5D\xA7\x55\xB0\xE8\x3E\xDC\x24\x28\x7F\x9F\x0C\x58\xA9\x39\x1F\xE6\xC3\x77\xEE\xBD\x65\x53\x56\xCE\x75\x43\x13\x94\xAB\xA0\xC6\xC5\x76\x6C\xA9\xAB\xA4\xF9\x09\x64\x0B\x1D\xC9\x99\x64\x29\x92\x86\x31\xE3\xF3\x13\x6E\xC1\xA2\x67\x18\x60\x3A\x32\xB6\x10\x00\xDE\x53\xAB\x99\x45\xC5\x7D\x5A\xCF\xCF\x5F\xE7\xF3\x3D\x94\x05\xC9\x70\x1D\x47\xF2\xA2\x60\x61\x32\x0D\xC5\x2F\x0C\xAF\x05\x6C\x78\x22\xCD\x3B\x3B\xD5\x28\xED\x44\xA3\xB4\x83\x20\x6F\x39\x74\x7B\x2C\xC8\x47\x3F\x54\x1E\x8A\x91\x18\xC1\xBD\x43\x8C\xA1\x76\xE5\xC2\xB8\x32\xCE\x88\xA9\x39\x35\xBE\x44\xCD\x54\xCE\xE6\x51\x5E\xB7\xCF\x3F\x0F\x4D\x58\x44\x7D\xDA\xD9\x00\x49\xB1\x2E\x7B\x50\xE9\xA0\x81\x06\x17\xD7\xB0\x2B\x7F\xCC\x68\x7B\xE4\x54\xFB\x92\x30\x5A\xD9\x47\xA0\xCB\x48\x76\x4E\xF9\x6C\xD8\x70\x03\xB9\xF1\xB6\x1A\x6D\xDE\xDD\xA8\x4A\x69\x9F\x73\x34\x96\xBA\xC7\x24\x51\xFB\x77\xA1\x61\xE5\xF4\xEB\xB4\x19\x21\x86\xB0\xA1\x8F\x18\x15\x31\x13\x7D\x1B\x5A\xFB\x9F\x61\x8E\xE9\xA5\xA2\x2C\x69\x60\x4A\xC3\xAF\x82\x80\x4F\x78\x64\xB8\xC7\x83\xE9\x14\x97\x65\x0A\xCC\x82\x0D\x3E\x83\x19\xE4\x13\x1A\xB5\x00\x6E\x7F\xEA\xE4\x06\x88\xB7\x08\xCD\x1A\x55\x18\xAA\x1B\x09\xD8\x12\xE0\xE5\x36\x1A\xA2\x75\x34\x44\xB3\x45\x98\x47\xAC\x9D\x1A\x46\x4C\x6A\x42\x48\x6E\x0B\x16\xEF\x30\xE2\x8D\xB8\x20\xD9\xD2\x82\xC0\x54\x51\x5D\x1F\xE9\xDD\x6D\x2C\x22\xC1\x86\x50\xCE\xCC\x97\xF8\xE7\x2E\xF1\x29\x4A\x93\xC3\x37\x6F\x6C\x28\x88\xFF\xF0\x10\xB7\xE4\x64\x0B\xE9\x9E\xBD\xD2\x45\xCF\x92\x91\x62\x69\x6D\x0B\xD5\x12\x6A\x3E\xDC\x0D\x70\x9B\x31\xD4\x18\xFB\xDF\x76\x89\x93\xD4\xE5\xA0\xD5\x88\xA5\xFF\x7D\x23\x10\xE8\x56\x9E\xAE\xA5\xE2\xFE\x51\xEF\xC4\xD2\xB8\x29\xB5\x57\xD2\x21\x69\x4E\x0F\xA2\x4C\xB0\xE6\x72\x08\x3A\x8C\xBA\x6C\xC4\x4B\x9C\xDE\x63\x30\x9F\xF0\x56\xDC\xCD\xF9\x87\x43\x1E\x7A\x15\x98\xA0\x8C\x36\x16\x8C\x83\x78\x24\x09\xDE\xDD\xEA\xA8\x39\x40\x34\x45\x10\x35\x0C\xE3\x42\x19\x3A\x42\xE6\xD0\x78\x89\x7F\xA5\x73\xAB\x10\xF4\x30\x9E\x9E\x42\x6A\xA4\x26\x00\x95\x56\xEC\x07\x49\xDB\xBF\x05\xDB\x77\x3A\xF0\x22\x25\x6C\x8A\x36\x8E\x86\x43\x86\xD8\xF9\x0D\xBB\x3F\x1A\x41\x28\x5D\xC7\x12\x96\x89\xF2\x64\x75\xA3\x9C\x40\xBA\x59\x11\x33\x21\x1E\x9C\x76\x96\x16\xF6\x8C\xB3\x4C\x98\x16\xE9\xCA\x1C\x5C\x6B\x39\x95\xE4\x74\xDF\xFE\x23\xEA\x07\x83\xB5\xD7\x9D\xE2\xD7\x28\xFF\x14\xB1\x62\xE5\xDF\x42\xCC\x54\xF9\x37\x33\x70\x96\xF2\x6F\x13\xDC\x85\xD2\x29\x7F\xD8\x77\x08\x88\xD4\xBB\xC1\x44\xCD\x13\x49\xFC\xAD\xDC\xF3\xC9\x33\x97\x44\xD1\xF2\xFA\xD1\xA1\x0E\x44\xE9\x2A\x9F\x1C\xF6\x1D\x76\x6F\x85\xE1\x2E\x34\x57\x55\x93\x68\x0D\xA9\xB0\x00\x68\x3B\xA1\x9C\x11\x58\x94\x20\x88\xB7\x62\xD3\xB0\x52\x63\xA6\xAE\x13\x0E\x01\x77\xDA\x55\x3C\x26\x2E\x03\x54\x71\xAC\xB1\x94\x83\xEE\x14\x42\x41\xBA\xA6\xEF\xE6\x5C\x1E\x7B\xE3\x31\x94\xB4\x30\x87\x0B\x6D\x91\xC4\xDD\x23\xAE\x9E\xE3\x30\xE1\x74\xAB\x69\x5B\xD4\xC3\xE3\xBA\xE7\x18\xD0\xC5\xDC\x71\xC0\x67\xD7\xF4\xD2\xD6\x65\x36\x06\xB0\xAE\x49\x23\x3F\xEA\x03\x65\xA0\x72\x28\x6F\x33\xB7\x41\x9F\x6B\x67\xDA\x6F\x7D\x5E\xD0\xA6\xAC\x37\xCE\xB4\xEF\x78\x5E\xF2\xCA\x51\x01\x79\xD8\x98\xD2\x83\x92\x88\xB3\xEA\xA5\x74\xBA\xDC\x10\x5A\x5D\xBE\x87\x35\xDB\x45\x48\x84\x6A\xFB\xAE\x44\xFC\x2A\xAB\xA4\x15\x23\x89\x57\x70\xE9\x6F\x60\xFA\xAB\x3D\x6F\xF7\x19\xA5\x84\xA7\x03\x33\xB5\x68\xAC\xCC\x5C\x49\x53\x53\xBA\xA6\x9F\xCE\x0E\x22\x5E\x9B\xF0\xFA\x8A\x2B\x8C\xCF\x89\xA0\xE7\x0D\x87\xD1\xD8\xCB\x8D\x9E\x36\x89\xB7\x61\x65\x54\x68\x7D\x4E\xAD\xAB\xDE\xCD\x5D\x43\x7F\x03\xAA\xD8\xE8\x12\xED\x0B\xB9\x3A\x82\x31\x1B\xDF\x50\x0C\x37\x14\x6B\x6F\xA8\x87\x1B\x62\x96\x0D\x91\xD2\x82\x86\x9C\x70\xFF\xDF\xDE\x77\x55\x00\xE9\x77\xD5\x15\xA0\xFD\x17\x0C\x7C\xE4\x32\x44\xD0\xBB\x19\x42\x37\x2B\xD6\xF3\x88\x71\x40\x55\x9F\x0A\x76\xF3\xC4\x2A\x8D\x60\xF2\x6D\xA9\xD1\xE0\x94\x2B\xDA\x1F\x11\xC2\x1C\xFD\x6C\x5C\xD1\x7E\x14\x3F\x1B\x59\xB3\x4A\xF6\x22\xAA\x1F\x74\x7A\xCF\x9F\xD9\x65\xAC\x12\xFF\xB6\xC1\xEC\x4D\xFB\x4F\x90\xE0\x07\x58\xEF\x27\xFB\x4E\x40\x25\x74\xB0\x1F\xF8\xAF\xE7\x1C\x65\xDA\x74\xB4\x87\x6A\x6C\x99\x37\x63\xCB\xD8\xE5\xEE\xFD\x28\xBB\xC5\x12\x8E\x74\x96\x62\x0F\x9A\xF6\x72\x57\x21\xD2\x0A\x44\x6D\x77\xB7\x68\xA3\xEC\x6E\xB1\xA9\x25\x58\xBF\x78\xE0\xCC\xD2\x35\x57\x05\xA2\x1E\x2F\xA2\x4A\xAB\x18\x26\x8C\xF4\x43\x9E\xC8\x7A\xE8\xF9\x61\x0F\xE8\x9A\x37\x63\x2D\x44\x33\x43\xF9\x08\x55\x31\x33\xAB\xC3\xC9\x58\xC1\xD7\x2A\x7C\xAE\xAA\xBE\x6D\x10\xAB\xA7\xE9\x89\x7B\xFE\xCC\x5B\x3B\x76\xBC\xF9\x7F\xFA\xFE\x77\xFF\xAC\x7A\x4C\xBA\x98\x31\x97\xE7\x1B\x72\x00\xEA\xB8\xC2\x65\x97\xB7\xFC\xCF\xD2\x6D\x2E\xF7\xEF\x7F\xF7\xCF\xDE\x1D\x23\x18\x72\x36\xCC\x4A\x13\xDA\x65\xFE\xCC\x53\x41\xE0\xAD\x60\x30\x3B\xF3\x54\xDF\x95\xCC\xA6\xCA\x90\xD3\xED\xCA\x98\x34\x55\x46\x66\x95\xE3\x61\x46\x4F\x3A\x43\xFB\xC4\xBA\x84\x83\x60\x35\x42\x46\xD1\xD7\x37\x93\x6A\x6A\xE9\x73\x4E\x9F\x53\x58\x5D\x55\xB0\xF9\xC3\x04\x7F\xD8\x77\x05\xB7\x08\x11\x99\xAD\x46\xC4\x9B\xA5\xA6\x58\xE1\x72\x46\xEA\x60\xC3\x6D\x4E\x97\xD2\x98\x43\xF8\x5C\xCF\xE1\x1E\x26\xFE\xF2\xF6\x1E\xD2\x7A\x84\xEE\x63\xAF\x6F\x60\xB0\x9A\xBA\xA1\x7B\x06\xF1\x63\x8F\x7E\xC1\xE1\x30\x59\x64\xCB\x01\x9D\x91\xC6\xC0\x36\x2C\xBB\xDB\x18\x74\x8A\xA4\x75\x73\x9E\xED\x6C\x39\xCB\xF9\x49\xCF\x09\x67\x31\x70\x8B\x86\xB4\xA0\xD9\x79\x33\x0C\x8E\x88\xD2\xAB\xED\x18\x3B\x4F\xDE\x1A\x4E\x81\x52\x72\x5C\x26\xC1\x62\xD5\x3F\xCB\x74\x2D\x45\xED\xE0\x08\x66\xB5\x89\x4B\x8E\xDA\xF3\x2E\x95\x3A\x2B\x89\x93\xCC\xAC\x3C\xD0\x4D\xAC\x88\xD3\x15\x74\xDA\x5F\xED\xE7\x89\x61\xC4\xB8\x22\xD8\xAB\x3C\x76\x43\xC1\xD6\x42\xDC\x65\x76\xB7\x98\x38\x10\x40\x25\xD5\x82\xE0\x6C\x2F\xFC\x61\xEF\x4B\xB6\x45\xCD\x2E\x22\x43\x71\x93\x4D\x36\xDB\xA6\xE5\xEF\xC1\xD0\x98\x0E\x12\x16\x3F\xB9\xF9\x70\xA3\xBD\x76\xC5\xA1\x7F\xE7\x6F\xBC\xAC\x90\x5B\xE3\x52\x97\xB7\xDF\xF1\x3C\x9C\x95\x1C\x70\x97\x41\x9E\x83\xC6\xB4\x0B\xC1\x04\x6A\x10\x2C\xCF\x89\xCB\x98\xB7\x43\x5B\x5A\x14\x2E\x63\x8C\x7A\x8E\x9C\x8F\x02\xCD\x68\x88\x83\x49\x2E\x05\xCC\x0F\x36\x2A\x0F\x90\x18\xB7\x08\x3D\x55\xA8\x13\x00\x1B\x6A\x2D\xC1\xD7\x44\x05\x35\x5F\x4B\xA2\x7C\x54\x73\x79\xF8\xD4\x01\xC1\xBE\x26\xB9\x86\x71\x7B\x51\x5D\x7A\x2C\xD7\xD4\xFC\x7A\xB3\xA8\xC2\xC3\x23\xFB\x41\x25\x2E\xD0\x94\x74\xD9\x94\x71\xC2\x12\x14\xFD\x45\xA8\x4B\x80\x8E\x49\x5C\xB6\xB7\xA8\xE2\x80\xA1\xD4\x21\xAE\x8F\xD9\x67\xF1\x18\x23\x1A\x12\x07\x49\x89\x7C\x0A\xEA\x18\xBA\xCB\x11\xF8\xEC\x11\x8B\xA5\x9B\x2C\xC7\x3D\x75\x0D\x31\xBD\x39\x12\x80\xA1\x0E\xEB\xC7\x9A\x44\x7C\x9E\x85\x9B\xED\x75\x85\xB7\xCF\x74\x1B\x6E\xC3\xA5\x12\xFB\x84\xEE\x3D\x43\x1F\xDF\x70\x79\xAB\xDB\x18\xA2\x9F\x36\xC2\xF3\xA9\xDB\xF0\xC9\xE1\x10\x0B\x55\xB8\x39\x44\xA1\x9B\xF5\xAE\xC0\x33\xC4\x2B\xD8\x6F\x67\xCE\x87\xE2\x37\x85\x9B\xB9\x26\xDA\x73\x2C\x3F\x6C\x51\x75\xC4\x59\x36\xD2\x32\xDA\x5B\xD8\x64\x09\x9B\x80\x2E\x6D\xA1\xD0\x3D\x69\xF6\x4E\xEF\x22\x27\x13\xDE\xF7\xE2\x02\x4D\xE7\xA0\x60\xA2\x40\xD3\x20\x59\x26\xA3\x84\xCE\x10\x66\xE1\x37\x7B\xDE\x5C\x52\xD0\xB9\x18\x08\xCD\xA1\x8F\x39\x67\x13\xAA\xC1\x64\x5E\x86\x92\xB0\x39\x51\xD8\x37\xF6\xF3\x93\x4A\x2F\xFD\x23\x9A\x2C\xC3\xD5\x68\x98\x51\x83\x61\x06\xED\x63\x0B\x14\x71\x0B\x24\x40\x93\x7D\x9E\x6B\xA5\x66\x0B\xB9\xE7\x11\xB6\xA7\xCA\x26\xAF\x22\xF8\x79\x6C\x7D\xD9\x66\xA4\x0C\x6A\xFB\xB4\x5D\xB3\x6D\x6A\xD2\xDA\x1A\x80\x76\x22\xCE\x2A\xDF\x16\x1C\x55\x52\xD6\xC4\xBD\x9E\xFB\xA2\xEF\x72\xDC\x45\xFF\xA9\x05\x84\x09\xE2\x03\x02\x8A\x48\x6C\x16\xEC\xE9\xEB\x2D\xE3\x3A\xBB\xDC\xCF\xAE\xE0\x4F\x79\x45\xB2\x22\x25\x90\x62\xE6\x72\x97\x7B\xD5\xFE\xAF\x41\x35\xC0\x0F\x31\xE2\x61\xD8\x8A\x48\x3C\x2E\x98\xFE\x0B\xA2\xFF\x3A\xD2\x7F\xE5\x66\xAC\x5F\x3A\xD3\x7E\x3B\xE6\x83\x0B\x23\x60\x4F\xE4\xBC\x27\x9A\xC7\xF0\x7A\x50\x1D\xD2\xF9\x0A\x4E\x3D\x4A\x5C\xCE\xA8\xFE\x56\x3C\x29\x09\xC3\xE3\x85\x3D\xB1\x41\x7B\xA2\x85\xD1\x3F\xD2\xB4\xC4\x03\x36\x0E\x7B\xC6\x3E\xD3\x9D\x70\x27\x62\x3C\x20\xE4\xE3\xB8\x27\x4E\x60\x4F\x34\xD4\xE6\x89\xF0\x7C\xE6\x4E\x0C\x7B\xA2\x61\x84\x4E\xDA\x13\x37\xEB\x5D\x83\x67\x20\x93\xD3\x9E\xA0\x29\xDE\x90\xF4\x2B\x37\x73\x1B\xA3\x3D\x51\xF3\x9E\xC8\x79\x4F\x34\xAB\x7B\x02\x41\x44\xA7\x59\xF5\xC6\x54\x82\x72\x69\x87\x94\xC4\xD4\x0B\xDA\x21\x0D\xD7\x5F\x2D\x90\x48\x08\x6B\x7E\x10\x3F\x86\xCD\xF2\xAD\x99\x3E\x79\xE4\xC5\x20\x3D\x9C\x3E\x4B\x9B\x43\xEA\xF0\xCE\x62\xC9\xE7\xA4\xAB\x9D\x7E\xB4\x6B\xA2\xF0\x54\x47\xB0\x1E\x40\x06\xFA\xF4\xA2\x78\x04\x12\xE2\xBA\xF3\x06\xF1\x0B\x66\x68\x85\x44\x38\xE4\x91\x27\x44\x81\x73\xAF\x77\xB7\xC6\x4F\x73\x11\xAB\xE9\x23\x78\xD3\x7C\x88\xD3\x4F\x68\x0E\xAB\x50\x11\x53\x3C\xDA\x0B\xC0\xA7\xB5\x8B\x19\x4D\x51\xE9\xF5\x7E\x57\xB2\xB9\x4E\xA0\xC4\x66\x21\xBC\x8B\xCE\xD7\x39\x32\x5E\x72\xAE\xBD\xEE\x2A\xFE\x7B\x11\xA9\x18\x60\xA0\x73\x64\x62\xB9\xF9\x22\x63\x7B\x22\xB2\xE7\xE7\x6E\x76\x89\xE4\x09\xFA\xD5\x56\x2E\x5D\x94\x2E\xA1\x0D\x47\xDF\x0D\xBE\x73\x89\xC2\x02\x25\x07\xB1\x04\xD8\x3B\x2E\x73\x69\xFB\x6D\x20\xEC\x0D\xBE\x81\x8F\x3B\x36\x1A\x26\xAE\x58\xCC\x43\x89\xF3\xAC\xFD\x08\xF6\x8B\xBC\xB1\x96\xDE\xF0\xB6\xA9\x5D\xB1\x87\xDA\xD3\xFC\xE0\x89\xB0\x45\x12\xDE\x22\xA8\x5E\x92\x30\x11\x26\xAE\x25\x3A\x4A\xDC\x89\x1D\x60\xB0\x6F\xF0\x16\x49\x86\x2D\xC2\xD5\xC4\xBA\x93\x7B\xDD\x29\xDA\x22\x30\xF8\x27\xD3\x63\x83\x46\x7C\x72\xAF\x9B\xD3\x16\xB9\xCB\xDD\xE5\x92\xB8\x45\xE6\xA3\x2D\x72\x17\x26\x15\x45\x55\xEE\x1A\x8E\x8D\xBB\x86\x2D\x32\xAF\xDC\xDC\x6D\x62\x8B\xDC\xAC\x77\x73\x3C\xD3\x2F\x36\x78\x79\x13\x73\xDE\x9D\xE2\x95\x9B\xBB\x93\xEE\xD4\x68\x8B\x6C\xF0\x16\x49\x78\x8B\x60\x7F\x9D\xA4\x2D\x72\x32\x6C\x91\x86\xE6\x7F\x51\xB8\x6A\x91\xBB\x19\x36\x46\x5A\xB1\xBA\x97\x04\x49\x7D\x74\x6A\xFC\x92\xE1\x92\x3A\x96\x79\x1C\x07\x5A\x7D\x37\x47\x45\xE2\xCB\x15\x1A\x61\x08\xB6\x4B\x46\x85\xF1\x53\x89\xB5\x4B\x10\x21\x47\xF2\x9A\x5E\xE4\x12\x87\x67\x5D\xB2\x28\xC4\xA4\x85\x14\xEF\x67\xBB\xD2\xD7\x4F\x75\x33\x29\x2D\xA1\x5D\x7E\xA5\xAB\x91\x8D\x33\xEB\xB7\x89\x53\x3C\xDA\x90\xCE\xED\x15\x12\xA5\xBA\x99\xAB\xB7\x38\x84\xCF\x95\x38\xBE\x5D\xBD\xD5\x15\x12\x78\x67\x5D\xF6\x58\x63\x1C\x3B\x19\xD8\xF6\x60\xB9\x34\x4A\xCE\xA7\x58\xC5\x6A\x05\x49\xE5\xE6\xD2\x16\x4D\x05\x92\x7A\xD9\x9E\xCD\xD1\xBF\xFD\xDC\x6A\x3A\x3C\xFC\x91\x47\x71\xBE\xBF\x06\xFF\x31\xAA\x54\x3F\xED\xF5\x3E\x1F\x83\x05\xFF\x4E\x67\xA0\xD7\xFB\x30\x3E\xC1\x1B\xE7\x72\x52\xAC\x49\x22\x64\xCE\x8F\x17\xD2\x94\x70\x48\xD2\x23\xC0\xEE\xFF\xA0\x0E\x60\x0A\xF1\x1C\xD1\x3C\xC7\xC6\xD3\x4E\x02\xDB\x29\x05\x1D\x51\xCE\x13\xBA\xA5\x1C\x9F\x27\xF8\x61\x88\xA0\xF3\xD5\x25\xEC\xD2\x78\xBF\x87\x65\x9C\xEF\x47\xEC\x5F\x15\x03\xEE\x36\xAA\x69\xC3\xD3\x1B\x87\x65\xFE\xBA\xAA\xFA\xAE\xDC\x66\x47\xFA\xAA\x5E\x95\x1E\x3C\xE3\x6E\x68\x7F\xAD\x90\x7C\xFF\x94\x95\x36\x11\x40\xA1\xB9\x41\x32\xF8\x55\x18\x2A\x12\x12\x13\x74\x57\xF8\xD3\x87\xFE\xF4\x33\x5D\xB9\x63\xCE\xE0\x99\x02\x35\xA1\x3F\x29\xF7\xA8\x7B\xE8\x78\x3A\x67\x5B\x1C\x82\x57\xFB\x2E\x15\x50\x0B\x64\x52\x07\x5B\x75\xE6\x52\x9F\xD0\xFF\x2F\x6F\xB5\x3F\x19\x08\x12\x98\x19\x70\x0C\xFD\x38\x9B\xDF\xD9\x88\x75\x06\x07\xDC\x27\xC0\xC5\xBF\x70\xC3\xC2\xED\xF1\x85\x1B\x76\xC7\x9C\xA9\x11\x06\x96\x32\x27\xBC\x18\x2F\xE7\xF4\x97\xC4\xE5\x71\x8D\xCB\xE0\xC6\x3C\xEA\x71\x78\x2D\x4A\x76\x93\xA3\x32\x60\x36\xBA\x81\x2B\x87\xD2\x0D\x2E\xBB\xD0\x70\x99\xF8\x87\x9A\x73\x92\x07\x0F\xB9\xBB\x72\xB3\x17\x45\x7B\x29\xFD\xE7\xD5\x6E\x88\xD6\xEA\x69\x4E\x0A\x71\x26\x74\xA9\x2B\xF7\x68\xC0\xB5\x18\x66\xFD\x17\x94\x57\xED\x3F\x86\x83\x21\x43\xDC\x06\xF5\x9D\x81\x45\x6C\xFB\x63\x8C\x40\x48\x64\x3D\x3B\x07\x1B\x54\x22\xB9\x99\x09\xAA\xD3\xE1\x5E\x59\x93\x52\xD0\x4E\xC6\xBD\xAE\xF6\x98\xE1\x70\x0F\x60\x54\x0C\x6D\x72\x64\xE4\x22\xA5\xFD\xA0\x5D\xE1\x3F\xA9\x76\xB7\xE8\x48\xE1\x28\x26\x12\x6E\xF3\x3D\x71\x84\xCA\x1E\xCA\x58\x2A\x84\xDC\xF7\x6B\xB2\xAE\x1C\x6B\xC6\x3C\x0E\xDB\x98\xDE\xE4\xD9\x76\xC4\x31\xF0\xC8\xE6\xFA\xC3\x1F\xFC\x56\x14\xEE\x2B\xFD\x77\x5C\xBB\x7E\x4D\xED\x36\xA9\xB7\xC8\xB1\xC1\x57\x6F\xDB\xBF\xF7\xCD\xE8\x54\x4A\x7C\xE7\x93\xEF\xFB\xD6\x6B\xAA\x53\x5E\x3D\x82\xCA\x1C\x4B\x3F\x59\x97\x35\x5C\x2F\x76\xF2\xB3\xA9\xFC\x27\x3F\xFF\x0E\x7A\x49\x88\x8C\xCB\x2E\xD0\xBD\x00\x17\xB1\xC0\x60\x81\xB5\xF4\xA8\xEF\x14\x0B\x7F\xC3\xFD\xD1\x16\xD8\xE5\x23\xC4\xE1\xF6\x07\x38\x05\x5F\xE4\xD4\x8B\x38\xD9\xCA\x5D\xA0\x4C\x97\x2E\x0B\x5D\xA6\x83\x71\xDA\x78\x55\xC9\x3A\x11\xCF\xBE\xC7\xB6\xE7\x88\x53\x2C\x52\x46\x4E\x4A\x5D\xE5\x74\xFB\x8F\xAF\xB3\x1F\x20\x1D\x8B\x2A\x2E\xAD\x5E\xD1\x3A\x5D\x75\x18\x08\xD0\x0B\x2F\x98\xE2\x04\xDF\x0B\x8B\x84\xD8\xA1\x98\x65\x3D\x48\x63\xBC\x6B\x0C\xDC\x16\xB2\x65\x52\x76\x26\x65\xFE\xC6\x8D\xFF\x3D\x44\x2B\xBB\x94\x11\x1F\xAC\x4B\xF7\x60\x17\x48\xA5\x4C\x74\x16\xF6\x89\x95\xD0\x04\x54\xB6\xA7\x51\x1E\x5D\x41\x38\x60\xE5\x34\x48\xDF\x33\x3E\x40\x24\xFF\xC1\xE4\xED\x92\xEA\xE7\x53\x9D\x1D\xA3\x9A\x18\xAF\x63\x44\x66\x1D\x84\x94\xD4\x1B\xC6\x36\x85\xF7\xC8\x11\x57\xF0\x57\xFB\xD7\x69\x38\x4B\x95\xB3\xED\xFF\x76\x3D\x84\x05\x2C\x0C\x07\x6B\xA3\x86\x6F\xCD\xF1\xE8\x5A\xD2\xFB\x35\x9A\x4A\x86\xA6\x5A\x6E\x0A\x96\x13\x36\xA2\x0D\x4D\x15\xD3\xA6\x36\x2F\x85\x18\xE5\x70\x64\xEC\x07\x8F\x77\x1E\xEC\xE3\x12\x1C\x15\x15\x99\x74\x69\x18\x26\x86\xD2\x87\x61\x14\xD4\xFD\xC2\x9B\x7E\xCD\x30\x24\xAC\x3E\x0E\x43\x12\x1F\xD6\x36\x86\x81\x14\xDB\xA2\x62\xAF\x19\xC8\xD0\xD8\x66\x68\x6C\x74\x10\xA5\x7E\xE6\x67\xF1\x58\x49\xDB\xBF\xFB\x3C\xFB\x6D\xD2\xF6\xEF\xB0\xB3\x83\x91\x8D\x46\x6E\x2E\x71\x8D\x14\x90\x8E\x49\x4A\x2E\x7C\xA8\x8A\x67\x88\x53\xD3\x80\xBA\xF2\x31\x89\xC6\x28\xFD\x51\x1F\x7B\x84\x12\x54\xB9\x3C\x5D\xE1\x66\x9E\x00\xCE\x88\x96\x38\x55\x81\x81\xC9\xC7\x43\xA7\x99\xBE\xD4\x48\xEE\x64\x7C\xF1\xE8\xA5\xE0\x74\x8F\x1E\xF3\xD2\x4D\x7E\x69\x1B\x5F\xCA\x91\xB7\x65\x78\x6D\xB1\xFC\x5A\x57\x62\x4B\x3C\x2A\x00\xA0\xF8\x12\x1A\x1C\x6F\xCD\xEA\x23\x1A\x61\xD9\x03\xD6\x6B\xB7\x66\xCF\x49\x5A\x0C\xEF\x39\x3B\xC0\xBC\xCA\x31\xE5\x14\x1F\x50\xE2\x5C\x93\x52\xDD\x5A\x40\x8F\xFA\x9D\x50\xAA\x2D\x71\x16\x6C\xDB\x22\xF4\x01\x9B\x31\xDC\x26\x87\x16\xDF\xC6\x87\x16\xC7\xA4\xF3\x59\x64\xC2\x66\xF4\x47\xD5\xAB\x5A\x42\xA0\x61\x68\xC1\x39\x0A\x03\x4D\x32\x04\xE0\x6E\xC4\xDA\x72\x76\x38\xCF\x81\x90\xCB\xF1\xE9\x3F\xC9\xA2\x03\x04\xBA\x35\x07\xB3\x19\x0F\x97\x36\x82\xE4\xC5\x87\xE1\x9A\xC9\x70\xD3\xC9\x70\xD3\xF1\x70\x53\x0C\x97\xF8\xCB\x06\xB1\xCE\xDB\x1A\x71\xC2\x23\x4E\xE2\x88\x55\xF5\x9B\x33\x5D\xAC\x8B\x63\x1D\xEC\xCE\xE9\x85\x20\xC0\x64\x3C\x60\xF8\xD4\xCD\x43\x9A\xB9\x59\x3A\x4C\x60\xEC\x33\xBC\xE2\x0F\x15\xEB\x6E\x88\x5E\x41\x16\x65\x32\x89\xB4\x83\xE5\x64\xBD\x20\x53\x4F\x04\x99\x9C\x05\x99\xBA\x2B\xE2\x94\x89\x08\x53\x87\x70\xEA\x9C\x3B\x50\xF0\x9C\x98\x41\x74\xE9\xCA\x89\xF0\xD2\xCD\x44\xCB\x11\xD1\x25\x9F\xDC\xE0\xF2\x3D\xDC\x80\x11\xE4\xA4\xF8\x8E\x0B\xC2\x2F\xFF\xCF\x15\xFE\xF1\x7E\x7E\xD9\x04\x53\xCF\xED\xFE\x2B\xE9\x3F\x33\xF9\x92\x29\xA5\x72\x55\xA8\x4A\xA5\xC0\x8D\xBF\xA1\xAE\x34\x95\x2B\xDA\x77\xD0\xA1\x49\x9F\xFC\xE3\xBD\x9F\x5D\x6E\x48\x7A\x2E\x18\x30\x20\xAD\x9B\x8A\x54\xFF\x19\x4B\x18\x9F\xD5\x80\x57\x98\xE1\x56\x9C\x9E\xE8\xFE\x0C\xD6\xD6\x84\x83\x36\x72\x11\x3A\x81\x97\x96\x87\x47\x00\x2F\x2E\xC1\x89\x30\x44\x43\x2C\x62\xB8\x35\x7B\xE8\x8A\x1E\x0D\xC2\xC8\xE5\x12\x86\x9E\xB8\x59\x43\x57\xFB\xD0\x8C\xE9\x97\x1B\x81\x35\xF2\x4A\x93\x3B\x0D\xEB\x27\x46\x51\x62\x14\xE0\x5D\xB2\xE4\xCE\xB4\x3F\xCE\x07\xFE\xA5\xA6\x90\x45\xF7\x9F\x57\x9C\xE6\xBE\xEE\xA6\x60\x08\x08\xFA\x54\xC2\x67\xD0\xE3\x7D\x97\xD2\xA4\x71\x32\xD8\xC7\x89\xF4\xE6\x7F\x99\xE7\xDB\x1E\xFB\x4F\xAE\x33\x39\xA7\x1C\xF5\x73\x99\x27\xD4\xE5\xFE\x83\x1C\x93\xEA\xDF\x77\xD4\xFB\xEC\xF2\x16\x8F\xA8\x80\xEE\x5A\xD0\xC8\x4A\x10\xD0\x85\xA6\xC4\x8C\xA2\x53\xAB\x1D\x7E\xB4\x51\x75\x55\x51\x73\x5F\x50\x17\x9B\x9C\x86\x9F\x8F\xD7\xB5\xAC\xA4\x95\x30\x29\xC5\xF2\xA4\x84\x11\x7E\xFC\x88\x73\x47\x47\x63\x75\xB3\x30\xC6\xFC\xD8\x7F\x72\xBD\xE2\xA9\x2A\x3A\x22\x99\xCE\xF2\xE3\x05\x35\xFA\x45\x35\x11\xE7\x78\x76\x7B\xF3\xE5\x93\xC3\x05\xAB\x74\xF2\x1C\x5E\xC9\xCF\x16\xF4\xAC\x65\xCD\x92\x9F\x2D\xE2\xB3\x16\x51\x6A\x4E\xF5\x61\x9E\x80\xED\x30\x0B\xDF\x72\xFA\xD6\x84\x6F\xC0\x42\x28\xC3\x37\x54\xE3\xCA\xC3\x37\x94\x5B\xAA\xC3\x37\xA4\xF1\x57\xE1\x1B\x4A\xCD\x05\x7C\x7F\xB5\x50\x2E\xD6\x42\xC4\x22\x90\x5A\x03\x2A\x84\x74\x1A\xC4\xBA\x11\x9F\x9B\x44\xC6\xA6\xD5\x3F\xAF\xAC\x3E\xD2\x57\xDB\x55\x3E\xEB\x12\xC6\xF6\x4B\xA5\x44\x07\x4D\xE4\x43\x74\xA5\xFD\x7A\x12\x3F\x5E\xA4\x3D\xFD\x2B\xCC\x70\x6D\x60\xB8\xD9\xA0\x3D\x8D\x53\x8E\x72\xBF\xC9\x7C\x6E\x94\x64\x55\xC4\xAB\x65\xB4\x21\x17\x7E\x33\x40\xF4\xF0\x31\x96\x45\xCE\xC8\xE7\x05\x97\x25\xA8\x91\x32\x6E\xCE\xBB\x1A\x1B\xAE\x16\x03\x27\xFD\x19\x2B\x33\xDA\x55\xA4\x98\xC2\xB2\x0F\x24\x53\x59\x97\x86\xFB\x12\x25\xA9\xD9\x48\x92\xE2\xD8\x91\x5A\x2A\x79\x32\x93\x4F\x0E\x16\x73\x52\x4E\x5E\xFE\xDD\xEB\xD7\xD4\x62\x83\x3E\xFE\xC6\x1F\xD0\xC7\xD6\x6B\xB6\x99\x40\x26\x11\x1C\x9A\x09\xE7\xCD\x58\xE4\x44\x01\x1C\xFF\x2F\x55\xCC\xD4\xCA\xF8\xA7\xAE\xF2\x67\x2F\x1E\x0C\xF9\x81\x70\x00\xED\x43\x07\xA8\xF9\x74\x2D\xCF\x2A\x8D\x07\x47\xF5\x9A\x4A\x96\x0B\xE1\xE8\x29\x3D\xA9\x69\xD3\x53\xC9\x65\x72\x1E\x95\x21\x3A\x24\x5B\x3E\x8F\x70\xDA\x95\x74\x16\x69\x56\x0F\x2E\xB2\xE9\xA6\x61\xE2\xD1\x80\x24\x34\x74\x8B\xBA\xD4\xD4\x51\xFD\x81\xDC\x55\x4A\xB0\x48\xC5\x51\x73\x85\x84\xC5\x02\x51\x08\x1D\xB7\x97\xBB\x13\x18\xF4\xAF\x60\x19\x30\x84\x07\x2F\x72\x88\xC3\x25\xFC\x49\x2E\x71\x85\x86\xCD\xC5\x49\xA6\xE2\x53\xAE\x76\x65\xEF\xAF\xF6\x8B\x4D\x9F\x88\xF3\xB2\x74\x59\xDF\x11\xB7\xA6\x59\xC1\xF3\xC4\xB1\xE8\x0B\xCD\x07\x8A\x51\xB1\x84\x4C\xCF\xB9\xCD\x37\x41\x6D\x38\xD1\xD0\xAA\x93\xE0\xA3\xF9\xC5\xEE\xC0\xBF\x8A\x6F\xAC\x1E\xE3\xAD\xCE\xB6\x1F\x23\x1A\x9C\x9B\x7C\xA6\x2B\x87\x68\xBA\x84\xB4\x61\xEB\x5F\xFE\x3D\x5A\xD8\xCA\xB5\x8B\x4D\x51\x23\x61\xEB\x62\x2B\xFA\x5D\xB0\xA2\x8B\x1A\x79\x17\x71\x96\xBB\x58\x8D\x3C\x41\x6A\x64\x4E\x7F\x06\x35\x32\xAF\x5C\x45\x6A\x64\xE5\x36\x69\xA1\x36\x1B\x55\xA7\x55\x0C\x67\x9E\xF9\x6B\x92\xC7\x49\x54\x7D\x82\x2D\x31\xA0\xA3\x6A\x90\xDC\x36\x24\x2D\xF6\xFF\xBE\x0E\x5C\xB7\x0D\x89\xA9\x29\x45\x19\x56\x2E\xE3\xBE\xB0\x5F\x30\x73\xD5\x2E\x0D\xE3\x75\x4A\xB9\xEA\x75\x4A\xED\x02\x77\x91\xBA\x50\x8A\x53\x90\x0D\xD3\x0F\xA9\x73\x36\xD4\xC6\x85\x11\x29\x09\x86\x84\x9F\x60\x43\x42\x11\xC2\xEC\x92\xAE\x1C\x75\x0C\x12\xD6\xD9\x8B\x9C\x09\xC6\x44\x89\x12\x13\x51\xA0\x2B\xC5\xE8\x42\x9F\xE7\x4B\x6D\xCE\x50\x5A\xA1\x2B\x85\x1E\x1F\x85\x36\x9A\x08\x18\x17\xAC\x7F\xEC\x81\xB3\x4E\xBB\x7A\x6F\xC7\x14\xF5\x8C\x84\x82\x44\xB0\x08\x13\x57\x2E\xB8\xD2\xCC\xA6\xCB\x5C\xC9\x1B\x38\x39\x60\xB9\x1E\xCC\x13\x43\x84\x2C\xB9\xC8\xD0\x17\x89\x0F\x61\x24\x42\x7A\x90\x35\x24\x52\xD9\x89\xCF\x82\x37\xBB\x13\x22\x0C\x20\x48\x06\xC3\xC0\x2B\xCA\x50\x7A\xCD\xCD\x58\xB3\xAF\x6D\x35\x22\x8D\x13\x56\xC8\x62\xF3\x42\x63\xDC\x26\x2C\x12\x26\xAC\xF6\x09\x9A\xEA\x13\x0D\xD2\xA3\xB3\x4A\x42\xC4\x79\xAB\x44\x73\x0F\xEA\xE6\xD7\x62\x82\xD2\x0E\xB9\x5E\x27\x9D\xD4\x1B\x3D\xC5\xAF\xC4\xF3\xBC\x0A\x01\xE4\x36\x65\x8B\x51\x0A\x2E\x5E\x88\xEF\x16\x6F\x19\xB1\xF4\xAC\xFA\xB5\x1C\x2C\xBD\x5C\x63\x77\x30\xCC\xD2\xED\xC0\xD2\x83\x46\x93\x30\xAE\xE7\x43\xEE\x9D\x2F\x76\x29\xAF\xBA\x7D\x91\xD7\x7C\x3B\xAE\xF9\x1B\x62\xC8\x2A\xCA\xD4\xC4\x63\xC6\xA5\xC4\x68\x73\x92\xB5\x0C\x18\x6D\x28\x97\x94\x23\xCF\x80\x5D\xF7\x6C\x39\xCC\xCE\x2A\xEB\xFF\x8A\x84\x21\xD3\x59\xF1\x57\xBD\x11\x6B\x19\x67\x39\xE1\xB4\xC2\x66\xAD\x07\x6B\x99\xE4\xC8\xCE\x59\x04\xD6\xAC\xEF\xE0\x28\x84\xAB\x10\x35\x40\xAE\xE2\xAD\x09\x62\x2F\xC4\x06\xC9\x27\xDF\xA4\x2D\x0D\x64\x0A\xA0\x6F\x5C\x65\x37\x56\xE6\x8A\xF6\xE5\x71\xF2\x9D\x62\xDF\x55\x78\x1D\x3C\x3D\xA1\xEC\x8C\x33\xED\x87\xAF\x87\x83\x82\x4E\x5F\xCD\x85\x2B\xD3\x91\xF6\xA3\x5C\xBA\x27\xA4\x98\xB2\xDA\x91\x73\xB4\x39\xA4\x40\xCE\x07\xAB\x63\x05\x3D\xE4\x84\xD2\xFF\xA7\x9A\x58\x11\x8C\xA3\xAC\x84\xD5\xC1\xD2\xC3\x26\x50\x22\x77\xD1\xAA\x0C\xB0\x4D\xC2\xE4\xB3\x2E\x2C\x71\x90\xD5\x70\xFC\x57\xF1\x58\x4E\xB0\x03\x5C\xB6\x17\x1F\x21\x3E\x5B\xC0\xCA\x87\x40\x17\x75\xA9\xB1\xC7\x6A\x13\xAE\xF6\x1F\x3E\xEA\xE7\x7F\x29\x69\x56\xFF\xE9\xA6\x69\x0C\x3E\x59\x55\xB9\xDA\xFF\xB7\xFD\xFC\x64\x51\xD7\x75\x52\x8F\xFE\xE5\x8C\xA3\xB6\x7F\x01\x31\xEF\x24\x0F\x2C\x92\xBA\x60\x68\x33\xE6\x88\x89\x6F\x0F\xF0\x9B\x4B\xFC\xEF\xE3\x3A\xFC\x64\xC9\xF4\x9E\x3F\xE2\x27\x73\x18\x6F\x0E\xE4\x1E\x3B\xBD\xE7\xE3\x7C\x0F\xF6\xC7\xCB\x2A\xDC\x64\xB8\x03\x30\x44\x00\x6B\xD2\xA5\x17\x9A\x94\xF1\xC3\x66\x2E\xF1\xFA\x60\x91\x20\x66\x9C\x36\xFD\xDA\x04\x13\xC4\x0E\xB4\xB0\xB5\xC4\x2C\x75\x04\xD1\x97\xFE\xE8\xE9\x03\xBC\xDE\x72\x36\x0C\xB4\xEF\x7B\x2C\x69\x9B\x75\x02\xEC\x4C\x7A\x49\xC5\x09\x8C\x81\xAA\x6B\xA1\x6A\x58\x93\x99\xF2\x8A\xDE\x15\xD0\xA2\xC5\xA6\x68\xC6\x22\xDB\x75\x2D\x98\x00\x4E\x0B\xAE\x16\x64\x1F\xE3\x75\x44\x4B\x50\x8C\x0E\xD8\x25\xA4\x95\x91\xEC\x01\xC9\x39\x81\xA4\x4A\x5F\xBF\x8E\xBF\xD2\xC7\x5F\x67\xC3\x36\x24\x5C\xFA\xFE\x17\x1E\x6E\xA4\x9A\x71\x67\x9C\xBE\xBC\x40\xBA\xB8\x14\x0C\xF2\xFA\xE9\x05\x5F\xDC\xAF\xFE\xED\x19\x6B\x8F\xF4\xD5\x62\x0C\xD3\x1A\x23\x61\xFD\x03\xBD\x2F\x7A\x3E\x69\x10\x01\xFB\x40\x8C\x8E\xF5\x37\x34\x5B\x65\xB4\x77\xFB\x5D\xB2\xB5\x63\xEE\xE3\x5B\xB6\x1D\xDC\x3D\xFE\x81\xDE\x99\x2D\x9A\x88\xBB\x49\x96\x26\x76\x95\x9D\x33\x67\x88\x91\x9E\xB3\xAD\xCB\x88\x43\x2D\x94\xCB\x5E\x64\x75\x7B\x94\xCC\xD3\x25\x1C\x3D\x15\x4D\x08\x2E\xF1\xEA\xF5\x1A\x01\xD2\x82\x8F\x7C\x56\x29\x31\x99\xF8\x57\x54\x1F\xFE\xDB\xFE\x04\x6D\xB5\xF6\x1F\x30\x22\x99\x05\x34\xEC\x65\xD8\xED\x11\xA3\x8B\x20\x59\x07\xA0\xC1\x5B\x3E\x0C\x55\x1E\x41\x84\x9B\x97\x25\x6C\x18\xF1\xF5\x0F\x20\x10\xD2\x6C\x31\xD6\x15\xD6\x9D\x8D\xEF\x8A\xAB\x71\x87\xBA\x6B\x31\x38\x6F\xC8\x73\x08\x86\x67\xD4\xA2\xA6\x45\x55\x5E\x3F\x0B\x54\x89\x2E\xDF\xEA\x10\xDF\x6E\xD9\x15\x0D\x34\x9E\x7C\x8B\x36\x76\x3B\x88\x8B\xDA\x9B\x7D\x79\x97\x94\x68\xF4\x44\x62\x23\x7E\x63\x10\x97\x8D\xD1\x6C\x9B\x56\x18\x0F\x06\x71\xA9\xCB\x85\xF1\x58\x3E\x2D\xD8\x7C\x21\x9E\x97\x54\xE6\x06\x67\x6D\xCB\x5F\x36\x71\x0C\x6F\x72\x4E\x94\x4C\xDC\x70\xD1\xA9\x3D\x54\xBA\x23\x26\x93\x56\x88\x85\x76\x79\xFB\xC2\xF5\x38\xF1\x1C\xB6\xCF\xCF\x4D\xBB\xC9\xC1\x69\x1C\x3E\xFD\xC3\x02\x0C\xC3\xF8\x1A\x17\xD8\xBB\xC9\xE2\x79\x31\xF0\x59\x9F\x87\x5C\xB5\x2F\xF6\x7F\x2E\xBF\x57\x29\xFF\x0D\xFD\xFC\xBF\x4B\x95\x52\xDA\xCE\x0A\x36\x80\xAC\x9A\x47\x8E\xF9\x97\xD7\x2A\xB9\x99\x69\xA5\x32\x58\xDA\x95\x95\xA8\x6F\xB9\x0E\xF5\x9A\x75\x98\x7F\x49\xEB\xF0\x17\x97\xD6\xA1\x06\xEE\xB8\xDD\x36\xA7\x07\xB9\x40\x49\xFA\x28\x84\x00\x3E\xA4\xA1\x23\xF0\x93\xD6\xB7\x7D\xFB\x71\x1C\xA1\xA7\xEB\xBF\x18\x0A\xFA\x5C\x03\x02\xF6\x88\x83\x26\xC2\x3E\x15\xB3\x4F\x3C\x1A\xD2\xC2\xD8\xFF\xA6\xCE\xD9\x4B\xA1\xC9\x57\x54\x2F\x47\x6D\xB5\x34\x3C\x1E\xDC\x42\xD5\x9B\x42\x0E\x92\x53\x72\xCC\x5D\xA7\xE4\xAE\x7F\xC8\x02\xC0\xB1\xF7\x9D\x44\xF0\xEB\x69\x9A\x70\x7F\xBA\x8F\xFD\x3E\xE9\xE7\x4E\xF9\x76\x7F\xCB\x4D\x13\xD9\x8E\x6B\xE6\x44\x15\x42\x7F\x4E\x77\xF4\x1C\x73\x1E\xE5\xAF\xBD\xAC\x0E\x90\x5D\x3B\x6D\xFE\x94\xCB\x97\x9B\x55\x3B\xE6\x34\xE6\x63\x7C\xDF\x86\x5B\xCE\xEB\x3B\xE6\xFD\xAD\x0C\xF7\x07\x6F\xDA\xCB\x8D\x6A\x58\x3A\xA1\xB9\xB4\xFD\x29\x9E\xA1\xD7\xC8\xA5\x6D\x73\x29\xAC\x4E\x5B\xF9\x02\xE8\x0E\xBE\x64\x3C\x36\x7B\x8F\x6D\xBB\x0C\xCB\xE5\x72\x31\xC7\xB5\x18\xE7\x51\xDF\xE9\x1D\xB3\x29\x28\xE4\xF6\xAC\xFA\x0B\xCC\x7F\xE4\x07\xC4\x51\x43\xC7\x79\xC3\x45\x99\x99\xAB\x3D\xF3\x30\x88\x6D\x15\xA3\x30\xD7\x15\x23\xC2\x19\xAE\xEE\xF1\x40\xEF\x93\x03\x67\xB6\x58\x54\x4E\x3D\x1B\xFF\x5C\x05\x54\xF2\x02\x84\xE0\xB2\x17\xE9\xC5\x97\x50\xD3\x50\xEF\x91\xD6\xF0\x08\x0F\x24\x30\xEC\xBA\xFD\xFB\xA0\xD1\x39\x48\x62\x34\x04\x9C\x29\x5D\x75\xA1\x29\xD0\x08\x75\xEE\xF3\xEA\x62\x53\x38\x58\x1B\x48\x23\x72\xFA\xAC\xD2\x6C\xB0\x4B\x7C\xF9\x30\x8C\x7E\x61\x4F\xB0\x61\x51\x9F\x55\x46\x4C\xA0\x65\x05\xC3\x03\x90\xD8\x11\xF1\x6A\x01\xF5\xF4\x4F\x7E\xE9\x75\x57\x9A\xD2\x35\x4E\xED\x36\x25\x90\x82\x10\x4C\x35\x7B\xCA\xA1\xB8\x38\x7B\xE5\x00\xA2\x4E\x4C\xBE\xE6\x77\xD1\xB9\xCD\xB0\x42\x88\x27\xA9\xFD\xBD\x7C\x5A\x6B\x64\x42\xF8\xA3\xA7\x91\x77\xE0\xF2\x85\x22\x51\x88\x86\xAE\xF6\x48\x12\xAE\x30\x7A\xC6\x35\x9F\xD3\xAC\xE0\xB5\x85\xFF\x82\x5A\xD4\xFE\x68\xC1\x60\x6D\x34\xD0\x2F\xA8\xDD\x26\x8B\x27\x5A\xDD\xB7\x3F\x2D\x4A\x20\xAD\xFF\x23\x3C\x29\x60\x40\xF3\xB3\x4A\x2D\xEA\x3A\x1F\x91\x77\xE2\xAF\x99\x81\x3D\x68\x8E\x1B\x80\xCA\xFD\x9B\x90\xF5\x27\x04\xDC\xFA\x13\x4E\xFB\xDF\x54\xBB\x5B\x81\x11\x2C\x6D\x24\x33\x90\xE8\x55\x22\xD1\x2A\x32\x92\x11\x0F\x32\x03\x8B\xE6\x97\x7D\xFC\xA8\x9F\x9F\x50\xA9\x49\xC7\xFF\xEC\xF2\xEE\x39\x79\x9B\xEF\xDC\xA8\x9C\xF6\x9F\x53\x17\x1B\xBB\xBC\x4B\x6F\xEF\xF9\xB9\xBC\xB7\x69\x7F\xE0\xA6\xF7\x35\x95\x50\x3E\xF6\x8A\xD0\xFE\x8E\x69\x63\xE0\xA4\x9E\x9B\x52\x73\x38\x7E\x3B\x04\x67\xF0\x14\xFB\x9F\xC1\x47\xEB\x34\x35\xA0\x26\x0D\x60\xFB\x23\x06\x12\x98\x79\x3E\xA1\xFF\x0F\xE7\x4B\x7A\xCB\xF3\x25\x5D\x73\xBE\xE4\x5F\xD2\xF9\xF2\x9A\x75\xE7\xFC\x42\x87\xCC\xB6\x40\x38\x08\xBC\x21\xDA\x21\x21\xF4\x55\x35\x02\x92\xEA\xB4\xDF\x14\x5D\x41\x6C\x87\xC3\xA5\x5C\x6C\x87\x09\xB2\xD1\x5C\x1E\x02\x4F\x98\xAF\x6D\x9B\xD3\x2E\x71\xA9\x58\x7B\xAC\xCA\x90\x14\x34\x39\x77\x1A\x92\xB5\x18\x99\x2C\x90\x75\x19\x6C\x33\x10\x48\x66\xA2\x0C\x23\x1B\x71\xB0\xCD\x14\x63\xDB\x8C\x72\xC5\x22\x19\x32\x2E\x04\x5D\x61\x64\x9B\xC9\x9D\xA6\x69\xCC\x5F\xA7\x94\xD3\xB0\xCD\xE4\x15\x97\xF4\x45\xC0\x5B\xDF\x25\x22\x60\x0F\xDC\x98\xDE\x78\x1F\x7B\xEC\x2E\x34\x27\x46\x2C\x59\x4E\x69\xC8\x96\x13\xE6\x28\x29\x04\x30\xE2\x31\x70\x5C\x51\xB7\xB7\x75\x17\x91\xFD\xA2\x04\x02\xBC\xA4\xC8\x81\x40\x67\x91\x35\x6B\xA6\xAE\x11\xD0\x58\x39\x5E\x8F\x3C\xAE\x47\x22\xEB\xA1\x01\x35\x11\x54\x20\xCE\x3F\x81\x57\x74\xDB\x9C\x66\xFB\x46\x20\x0A\x59\x07\x98\x4A\xD4\x8E\xA9\x49\x73\x24\x12\xF8\x82\x98\x96\x69\x3B\x7D\x5E\x45\xDA\x39\x96\x10\x91\x94\x5A\xE1\x0B\x71\xBE\x39\xEE\xB9\xC4\x7C\xFE\x52\xC7\xBA\xC3\x8B\x08\x9D\x54\xCB\xE7\x12\x62\x2C\x71\x06\xD4\x01\x9D\x2B\x9C\x4E\x6C\x55\xDA\x24\x55\x20\x82\x43\x02\xA2\xEC\x36\x4E\xA7\x44\x4E\xA7\x52\xB2\x69\xD0\x82\x1C\x4D\x89\x0C\x1F\x59\x47\x34\xF2\xF6\x45\x9C\x4D\x96\xF4\x5E\x4C\xC4\x66\x35\x32\xC4\x86\x8F\x0C\xC9\x06\x68\xB8\x20\x27\x2F\x72\xA7\xA5\xC0\xF2\x44\x36\x36\xB7\xB3\xF2\x68\x58\x08\x7F\x1B\x74\x4B\x67\x55\xA7\xD8\xDD\xAF\x42\xC2\x36\x94\x94\xD7\x84\xB4\xD1\x90\x69\x1A\x50\x99\x44\xDD\x1E\x69\x2F\xC9\xA0\xC6\x70\xF4\x74\x00\xEC\xB1\xDB\xE6\x8D\xB7\xAB\xF9\xD8\x11\xB0\xB0\xAE\x5E\xD1\x3A\x3B\xF2\x0C\xEA\xC6\x29\x47\xD1\xF5\x2B\x68\x6A\x8F\xF7\x74\x32\xB3\xDF\x46\x00\x34\x70\x32\x27\xC4\x32\xF9\x64\x86\x9E\x0C\x28\x50\x3A\x5D\x73\xDC\xCF\x87\x69\x2A\xC1\x31\xA3\x20\x8D\x4C\x8E\x6C\x33\x54\x6C\x82\x95\x27\x77\x99\x9F\x3D\xD5\x2F\x32\x84\xEB\x84\xA0\x9A\x94\x8E\xC7\x1C\xC1\x27\xF0\xC9\xA2\xFD\xD9\xE5\x51\x89\x74\xA0\x4B\x3B\x53\x9D\x52\x4E\x09\x76\x53\x1C\x6C\x42\x43\x24\x8D\x81\x3E\x19\xFA\x8F\xAD\xDE\x59\x4F\xB2\x4C\xF5\x14\xE3\x2B\x66\x96\xFB\x0F\xE9\xCB\x82\x6E\xF7\x8B\x57\xB1\x2D\x18\xE7\x23\x64\x07\x99\x80\xE8\x31\xAD\x18\x26\x16\x38\x89\x61\x7B\x53\xC3\x00\x1C\xE9\xB4\xAC\x5C\xC6\x16\x32\x94\xF1\x4A\x58\x79\x34\x2E\x99\xA0\xF3\xC2\x28\x06\x13\x58\xC2\xC9\x42\x06\x61\x76\x12\xA1\x0B\xB0\x1C\x54\x0F\x23\xB5\x34\x94\x47\x81\x85\x6E\xA8\x1E\x06\xCC\xD6\x12\x6F\x59\xCC\x60\x0F\x7B\xFB\x7E\xDF\x55\x10\x78\x66\x21\xC7\x6D\xE6\xDF\xDE\x77\xF5\x50\x36\x2C\x27\x31\x12\xA9\x40\x5E\xB9\x7C\x6F\x91\xD0\x37\x44\x44\x9B\x98\x52\x85\x90\x1C\x00\x2A\x31\xF2\x6F\x97\xB9\xE4\xB1\x45\xE3\xB2\x45\xE2\x1A\x86\x45\x76\x33\x84\xD4\x86\xC2\x61\xC3\xD3\xF9\xA8\xF6\x0D\x7F\x1F\xAA\xE3\xF0\xF7\xA1\x10\x0E\x7F\x6F\x7B\x98\x69\xE2\xF7\xCD\x3E\x64\xDF\xF3\xF7\xD3\xF4\xFD\xF4\xF0\xFD\x0C\x7D\x3F\x33\x24\x80\x39\xEE\xAD\xEB\xBB\x9C\xD3\xBE\x2A\xCE\xB9\xAA\xD9\x94\x34\x4A\xC4\x42\x9E\x89\x20\xCF\x8C\xCA\xCA\x25\x88\x89\xDE\x59\xAA\x29\xC7\x86\x33\xAC\x3C\x30\x77\x50\x20\x2E\x94\x8D\x43\x5A\x9E\x62\x2C\x9A\xC7\x38\xD6\xDC\xE5\x1C\xE6\xFA\xEA\x5F\xE2\xE5\xE1\x37\x6B\x5F\xF4\xED\xCF\x46\x6B\xFB\xBA\xAB\xFF\x34\x40\x1E\xDD\x21\xBE\x3B\xC4\x37\x10\x9F\xD3\x90\xB7\x7C\xDD\xF3\x9B\xB0\x20\x66\x6B\x28\x96\xC4\x71\xCD\xE6\xB0\xCB\x03\xE8\x26\xBD\x4F\xEA\x7A\x1F\xF2\x79\x8B\xAF\x79\xCF\x58\x0E\x09\x75\x6B\x4C\x82\x1C\xB4\x88\xE4\xE8\xF6\x9F\x0D\xE8\x11\x01\x61\x08\x11\x3F\xC0\xF3\x42\xC1\x25\x4C\xE5\xDB\xFB\xCE\xB4\x3F\xC7\xD1\x76\xE1\xA6\xDA\x8C\x93\x1F\xE1\x58\xA8\xD8\xDB\x50\x9F\x0B\xF9\x96\x89\xC4\x33\x8A\xFF\xBD\x46\x3A\xA6\xDE\x36\x9B\xE8\x13\x4F\x15\xD8\x3D\x75\x34\x97\x9E\xF3\x21\x9A\x3B\xAE\xE4\xE6\x72\xFA\xB8\x8C\x74\xD4\x56\x4B\x38\xCD\x7A\x88\x55\xBB\x53\x1C\xF4\x4E\x71\xD0\x3B\xC5\x41\xEF\x14\x07\xBD\x53\x4D\xF8\x0E\xC3\xB8\xC3\x30\xEE\x30\x8C\x2F\x3F\xC3\xF0\x08\x2D\x17\xFB\x15\x23\xE3\xD1\x16\x7B\xC7\x4B\x43\x45\x21\x84\xDC\x14\xFB\xBC\xF3\x19\x21\x12\x86\xA1\xE4\xF0\x60\xC7\xA8\xEA\xBF\x5F\x69\x21\x3D\xAE\x85\x14\x52\xD5\x83\x2A\x71\x0A\xD5\x3F\xBC\xDE\x7F\xBD\x56\x4E\xF9\xB4\x77\x8C\xE4\xE4\xB0\x42\x87\x07\xDE\xA0\xED\x37\x7E\x91\xBD\xD3\xA1\x77\x06\x8D\xD8\x2F\x47\x23\x09\x1A\xF9\xAB\x4B\x8D\xA4\xEC\x7C\x5F\x69\xE4\xB9\x7D\x9F\x1E\xB0\x7A\xA2\x87\x5A\x1C\xD5\x7D\x2A\x3E\x4A\x63\xDF\x31\x6A\x63\xE9\xE1\xF8\x78\x86\xF7\x9D\x9D\x3C\x90\x1F\xFB\x40\x8E\xBB\xFF\xBA\x80\x60\x29\x00\x38\xD9\x82\xFF\xBC\x6A\xF8\xD7\xBB\x9D\x7A\xE8\xE8\x9C\x39\xC3\xBF\x5E\x82\x49\x8E\xD1\x28\x05\xD9\xC9\x02\x67\xBF\x0F\x90\x5F\x9F\x54\xF1\xC7\x5F\x1E\x3E\xFE\xC6\xF0\xF1\x33\xF1\x63\xF5\x37\x00\xEE\x46\x7D\x79\xAA\xBD\x7E\x3D\x76\xEC\x2D\xE1\xCB\xB6\x79\x62\xA1\xB9\x1B\x4F\x04\x27\xD0\x00\xA1\xB3\x6D\x1E\x0F\x57\x1F\x5F\x77\xF5\x52\xB8\x7A\x69\xF5\x6A\xF5\x8E\x52\x97\xEB\xCD\x24\x6E\x5A\x94\x55\xB1\x96\x61\x27\x95\x50\x47\x65\x50\x6B\x17\x2E\xA5\xAC\xB3\xC8\xA5\x14\xC2\xBC\x17\xC8\x6E\x3E\x50\x1B\x99\x20\xCE\x9B\x46\xCA\xD4\xA8\xE6\x9C\x5C\xE2\x04\x16\x4E\x1C\xBA\xD0\x68\x2B\x78\x2C\x89\x00\xFA\x1C\xF6\x1D\x80\x92\x0A\x3A\xF1\xA4\xC6\x85\xF5\x1B\x08\x50\x76\x85\xDF\xD8\x47\x99\x5C\x4E\x0D\xB4\x5E\x3F\xEB\x3F\x7B\xE3\xC6\x8D\x7C\xBF\x8F\x75\x46\x33\x94\x34\xE4\x7B\x9D\xF5\x6F\xDB\x3F\xE0\xC2\xBA\x5C\xD9\x2D\xF4\xCA\xFA\xE4\xD9\xD0\xB1\x9C\x0B\xCB\x49\x01\x20\x89\xC1\xCA\x60\x05\x7D\xE5\x3D\xA3\x82\x8C\x50\x5B\x43\x84\x58\x1E\xAD\x6D\x00\x4A\x4A\x38\xDA\x84\x36\x0F\x97\x02\xA2\x91\x0C\x7D\xBA\x88\x12\x95\xA1\xC2\xDC\x63\x82\x96\x60\x25\x8B\xE2\xAC\x52\xAB\xEF\x63\xA8\x27\x62\x54\x17\x46\xDD\x8E\x45\x3C\x53\x14\xE9\x95\x02\x32\x98\xCC\x50\x0D\x8E\xB8\x7B\xAD\x07\x9D\x8D\x17\x69\x30\x12\x4A\x11\xE0\xA5\x16\x25\xAF\x88\x43\x1C\x0B\x00\x66\xA4\x12\x45\x13\x41\x73\x18\xC2\xAA\xC0\x22\x25\x3C\xB8\xF1\x9C\x9A\xD1\x9C\x26\xB7\x39\xA7\x88\x7F\x34\xA3\xFE\x04\x72\xA9\x42\x94\x2E\x3F\xCF\xB3\x97\x32\xAD\x70\xD6\xD1\x9A\xF6\x78\x2D\x2C\x37\x2B\x20\x9F\x9C\xC9\xC0\xA8\x43\xBC\x1A\xE6\xA2\xB4\x07\xDC\x0A\x97\x9E\x55\x09\xD7\x9A\xE9\xCF\x2A\xED\x6F\xC0\xC7\x53\x72\xC2\x24\x32\x82\x64\x25\x53\xA9\x29\x98\x28\x63\x95\xC1\x0A\x3C\xBB\x48\xB9\x3C\x24\x13\x6A\xBC\x21\x31\x4A\xEA\x73\x4D\x08\x74\x61\x6A\x39\x49\xD1\x23\x3B\xEE\x91\x8E\x3D\xD2\xCE\x52\x8F\x40\xA3\xB1\x47\x3A\xC4\x7D\x12\x17\x04\xCC\x6E\x25\x55\x13\x98\x04\x78\x3F\x31\x7A\x36\x13\xC1\xDA\x3A\x79\x61\xE3\xC9\xD2\xD0\xE7\xD3\x5E\xB9\x74\x5D\xA5\xBC\xCD\x45\x5A\x0D\xC5\xF8\xC6\xB6\x8D\x8A\x33\xDF\x8C\xA0\x4D\x46\xD8\xD6\x35\x4A\xFA\x0D\xA3\x93\xA3\xA9\x75\x96\xEB\xA4\x74\x76\x10\xBC\x13\x4E\xC8\x4F\xFA\x05\x9D\x1D\x49\x6D\x2A\xBF\xB1\x48\xD8\x4A\x82\xD8\xE6\x2E\xF5\xB3\x5D\x92\x5B\x70\x7F\xDF\xA1\xC0\x0B\xCB\x67\x8E\x0B\x68\x78\x73\xE0\x12\xA9\x83\xB1\xBB\x05\x6F\x3C\x02\xC6\x81\xD1\xAB\x91\xA3\xFD\xF1\x71\x94\x97\x96\x08\x2F\x1D\x42\xC1\xCA\x87\x1B\x00\x58\x20\xCA\xCE\x70\x7E\xC4\xCA\x53\xA9\x3C\x95\xC6\xA7\xF8\xE5\x33\x7E\xC8\x25\xFE\xEB\x0E\x98\x15\xE8\x05\x2C\xDB\x08\x34\xE2\x21\xD8\xD5\xE6\x2C\x35\xE7\xEC\x10\x8D\x76\x20\x95\x16\x6F\xD6\xE9\xA1\xCB\x74\x3F\x27\x2E\xD0\xD8\x8D\x53\x4F\x6F\x49\xA5\x8F\x43\x7F\xF4\xB4\x77\xFB\x2E\x39\xA8\xBE\xD7\xEA\xEC\x88\xBB\x41\xF4\xF0\x7F\x7E\x78\x5A\xCA\x9A\x7F\xFD\xF5\xE9\xAF\xDA\x5F\x33\x97\x1A\xE5\x7F\xE3\x43\x4B\x37\xF3\xCF\xFF\x6E\xFA\x33\xE3\x02\xF8\xE7\x3F\x30\x2D\xDC\xCB\x81\x78\x4E\x0A\xF4\x72\x12\xF8\x47\x0B\x29\x56\xAC\xFC\xC7\xF8\x63\xC2\x85\x4B\x8D\x44\x39\xB3\x2C\x46\x9A\xC5\x8E\xF9\x68\xB1\x01\xA7\xCF\xB6\xF9\x08\xF0\xE5\x33\x6F\x9F\xE9\x10\x55\x61\x63\xB9\x71\xF6\x6A\xB2\x55\x33\xF1\xE6\x19\xE0\xA7\x59\xC1\xB3\x40\x50\xEB\x33\x0C\x6D\x71\x79\xAB\x4B\xA2\x78\x23\xB0\xC2\xE6\xB1\xC6\xD0\x2D\x01\xD0\x02\xBE\xB2\x9C\x8F\x20\xEE\x51\x1E\x1D\x54\xF4\xEB\x82\x8B\xE3\x63\x55\x05\x7C\xC4\xBA\xD4\x65\x11\xC2\x02\x62\xCD\xC7\x0A\x96\x6B\x3E\x2A\x52\xE8\x47\x8A\xE9\xDE\x19\xC4\xCB\xDF\x9B\xD9\xFC\xC8\x5C\x45\x15\x16\x1E\xA5\x8E\x73\x04\xF4\x6F\xD8\x1F\x91\x07\x8D\xFC\x0B\x1B\xB1\xBA\xF9\xE5\x89\x14\xFA\x73\x89\xFF\x26\x73\x09\x18\x71\x7F\x32\x5A\x1A\xE6\x7C\xCC\x7F\xF4\x7E\xDF\xE5\xF7\x58\xD5\x15\x0F\x91\x5A\xF1\x86\xFD\x1F\x7A\x47\x57\x7E\xCB\x43\xEA\xBF\x8E\x78\x99\x4B\x4F\xBB\xDC\xD1\xFD\xAE\xBC\x7E\xCE\x2A\x86\xDE\x29\xF8\x0B\x4D\x1E\xF1\xAB\x6A\xF2\x32\x13\xC0\xCB\xB2\x67\x69\xA1\x0E\xFB\x7B\xAC\x7A\xE8\xC8\x25\x78\xD5\xB7\x74\xE5\xE3\xE8\xEF\xF9\x7D\x57\xFE\x8D\x17\x0F\xD8\x46\x7C\x14\x41\xCF\xB2\xB9\x55\xDA\x90\x7C\x79\x8F\x2D\xF8\xDE\xF2\x6F\xBC\x88\xE0\x54\x00\x3B\xAB\x7B\x6C\x3B\xFC\x7C\x4D\x73\x03\xF4\xF3\xE9\xD1\xDD\xF4\x33\x47\x5D\x03\xFC\x98\x2B\xC8\xB0\x7C\x3F\xE4\x3A\xAB\x4B\xAC\x0C\xFD\x2E\x51\x3C\x0B\xE3\x18\x06\x5C\x5D\xD7\x50\xBE\xC4\xA5\x03\x6D\x13\x2D\xBE\xAC\xDE\xDA\xF1\x11\x12\x67\x90\x4E\x85\xEC\x70\x6F\x98\x44\x69\x6F\x16\xDA\x1D\x52\x5D\xC2\xB4\xE2\x05\x37\x50\x26\x65\x98\x4A\x46\x1B\xA9\x87\x77\x4E\xA6\x5D\x2E\x3B\xEB\x8F\x2E\xC5\xF9\x5E\xCC\x5C\x1A\x3A\x3A\x1B\x4D\xFE\x43\x47\x2E\x75\x29\x3A\x9B\xFB\xEC\xB0\xCB\xF6\x7E\xE8\x1D\x00\x9E\xE0\xB5\x90\x25\xC8\x78\xFE\x53\x9E\xFF\x50\x54\x22\x3F\x7E\xFE\xD3\xF5\xF3\x9F\xAE\x9F\xFF\xB4\x8A\x30\xBE\xB2\x0E\xA3\xF9\x47\x64\xED\x95\x00\x22\xC4\x8C\xE7\x77\x3F\xBC\x86\x84\xE2\x84\x0B\x0D\x61\xB2\xBB\x7C\x2F\x12\x52\xCE\xA3\x98\xF1\x28\x66\x21\x0D\xEB\xF8\x51\xCC\xD6\x8F\x62\xB6\x7E\x14\x33\x80\xFC\x01\x9E\xD7\x31\xAA\x0B\xD7\xA9\x78\x7F\xD1\xD5\x71\xCE\x50\x6E\xE0\x03\x45\x07\xF0\xAF\x3A\xC2\x31\xB8\x7C\x17\x8A\x3B\xFB\xF5\xE5\x4B\x41\x92\x5A\x16\xC0\xB8\x52\x06\x0C\x4B\x01\xC7\x1B\x93\x59\x5F\x28\x80\x10\xB9\x63\x5E\x28\x02\x30\xE4\x4B\x02\x8F\x95\x31\x92\x5A\xC6\xCA\x2C\x5B\x5F\x76\xCC\x07\xB8\x66\x88\xAB\x05\xC9\x67\x83\xF1\x25\x85\xB1\x59\x49\xA8\x6A\xE8\x7A\x43\x9C\xB1\x75\x2D\x90\x25\x99\x33\x36\x63\xCE\xD8\x80\x33\xA2\x7E\x44\x13\x1A\x48\xE8\x96\xC0\x19\x91\x67\xCE\x55\x97\x37\xB8\x33\x1B\xDC\x99\x0C\x77\xF5\x8B\xDC\x6D\x10\x67\x6C\x18\x67\x09\x68\xBB\xAE\x76\xF3\x11\x67\xCC\x77\xCC\x4B\xC4\x11\x1B\xEA\x39\xAB\xE9\xEF\x2F\x38\x97\x2C\x62\x6F\x56\xEC\x4A\x9E\x39\xBB\xDB\xA5\x28\x59\x8E\x42\x92\xE9\x16\x62\xAA\x04\x01\x81\x6B\x3C\xAA\x6D\xF3\xD1\x62\x97\xD1\xE3\xC7\xDE\x89\x8F\x14\x82\xB1\x2D\x0C\x18\xCB\x46\xAF\xAC\xFE\xD8\xE8\xFC\x48\x96\xD1\x30\x32\x35\x9D\x99\xFE\x03\x72\x10\x89\xD9\x87\xE1\x87\x59\xC0\x08\x26\xAF\xF0\xA5\xE8\x3B\xEB\x82\x4F\x84\x11\x33\x11\xBB\x91\x6C\x0C\x8B\x68\x87\x45\x24\x4D\xFE\x25\x29\xBA\xC3\xA6\x23\xCB\x30\x25\x72\xCC\x25\xB2\x8E\xCE\x6E\x54\xC1\x90\x33\x1C\x72\xF1\x8C\x62\xAD\x2D\x09\x76\x22\x39\xE4\x92\xB8\x94\xE9\x78\x29\x53\x86\xC2\xE2\x52\x20\xE1\x94\x84\x0C\x1B\x14\x3C\x60\x9F\xC8\x21\x67\x47\x87\x5C\xC2\x92\x6E\x38\xE4\xD2\xE1\x90\x23\xD9\x62\xE9\x90\xC3\x52\xA6\xB2\x94\x09\x2D\xE5\x8A\x10\x68\x47\x90\x64\xC6\xD8\x23\x7D\x15\xF5\xD1\xA1\xB1\xBE\xC0\x50\x07\xF6\x58\x26\xC0\x5C\x37\x19\x98\x00\x1D\xC1\xD9\x61\x97\x82\x09\x64\x8F\xA3\x14\x70\xB6\x86\x95\x25\x13\x26\x80\x1B\xB3\x15\x56\x16\x7F\x9E\xB2\xB2\xE1\x6E\x66\x65\x30\xB9\x08\x1B\xD8\x36\xAA\xFD\x17\x88\x21\x49\x58\x2B\xB8\xC2\x55\xA0\x7B\xCC\x1B\xAC\xA9\x76\xAF\x7D\x81\xAD\x7D\x29\xDB\x97\x6E\xE8\x47\xB8\x9A\x22\x28\x83\xD8\xE0\x87\x4E\xEA\x3A\x68\xE5\x2F\x17\x51\x2D\x87\xCE\x2D\x09\x68\x31\x41\x40\x13\xED\x70\xE8\xBE\x26\xA1\x20\x7C\x7C\x6F\xD1\x3F\x74\xED\x9B\xAF\x5D\x7B\x97\xC3\x77\xB9\x99\xAE\xBE\xBF\x70\x1A\xA9\xA8\xF8\x77\x74\xCE\x4A\x65\x95\x76\x0C\xB4\x61\xB9\xC6\xEB\x58\x20\x7F\x1C\xE2\x4D\x97\xF6\x1D\x20\xB2\xF3\x7D\x7F\xB5\x9F\x5B\x65\x35\x32\x77\xB3\xB3\x2A\xE9\x72\x97\x71\x35\xD6\xAE\x68\x7F\x19\xC6\x8B\x2C\xD4\x97\x0A\xCF\x98\xDD\x46\xF9\x9F\xFE\xE0\x44\x34\x2C\xCF\xAA\x50\x99\x31\xF7\x9F\x53\x9C\x29\xFC\x69\xC5\x01\xA2\x9F\xE6\xFF\x3C\xBC\xD5\x91\xB2\xFE\x39\x85\xFF\x5C\xDE\xEA\x0A\x9E\xDA\xDC\x3F\xD9\xBB\xC2\x3F\xD9\xF3\x0B\xA5\x3C\x52\xE6\x8B\x7D\x22\x68\xF4\x3B\x94\x9E\x4C\xA3\x0E\x95\xBA\xEC\x69\xD2\x49\x4D\x40\x16\xA0\xDE\x7B\xBB\x7F\xC1\x99\x03\xA9\x2F\xCA\x42\xD4\x47\x0B\x56\x99\x3E\x16\x60\x39\x1F\xFA\xB6\x6B\xD7\xAE\xBD\xA0\xCF\xB1\x71\x43\x04\x50\x36\xCF\x6A\xC8\x66\xF9\x5E\x57\xD2\xE6\x9C\x0D\x40\x9C\x24\x40\x9A\x00\xC4\x99\xEF\x75\x05\x6D\x4E\xB0\xB1\xB8\x39\x8B\xF1\xE6\x2C\xA2\x05\xD0\x15\x91\x51\xD3\x2D\x61\x73\x22\x2D\x7D\x86\xCD\x39\x1B\x75\x28\xC5\x3D\xFD\xA2\x90\x42\xB7\x9C\xCE\x83\xAA\x7D\x74\x16\xC6\xAD\x89\x5C\xF4\x8F\x15\x9C\x3B\xF7\xD1\x82\xAD\x34\x1F\x29\x9C\x44\xDE\x05\xED\xAC\xFD\xBF\xB8\x0E\xC8\xC2\xC4\xC5\x23\x4E\xE6\x7F\x68\x24\xC1\x63\x7D\x49\x9B\xBA\x6F\xFF\x82\xB3\x98\x3B\xC3\x39\x68\xE0\xB1\x15\xCF\xDD\xA2\xE6\xB9\x44\xFE\x56\xDA\xE5\x21\x75\x06\x91\x3E\x51\x47\x21\xA1\xFA\x7F\x80\x2A\x13\x6A\x36\xFB\x3F\x15\x88\xFE\x32\x24\xD7\x28\x97\x07\x74\x5E\x9E\xDF\x35\xBA\xD7\x90\x97\x93\x8E\xB4\xB6\xA4\x92\x84\x57\x28\x1A\xAC\x61\xE7\x52\xC9\xD0\x0A\x86\x80\xEE\xBB\x62\xDA\xE2\x63\xFC\x46\x57\x70\x5E\x79\xF8\x18\x8C\x1D\x5F\x7A\x1F\x20\xB6\x17\xF1\x7E\xCD\x50\x10\x92\x74\xA4\x5D\xE6\xFF\x54\x33\x2C\xB1\xE1\xEA\x20\x9A\x4B\x1A\x6A\x2E\x19\xB2\xF4\x66\x8B\x51\x65\x41\x5A\xF4\x37\xF4\x22\xF3\xD7\xCC\x22\xAD\xE2\x1C\x7E\xDD\x65\x2C\xD3\x45\x19\xF1\x3A\x65\x2F\x5B\xAB\xEC\xE5\xC0\x26\x5D\x52\xDF\x5C\x8E\x9C\xBA\xA8\xBA\xCD\x5C\xFD\x18\x50\x41\x91\x54\x3F\xE3\xC3\x9F\x89\x91\xC1\x41\xAB\xBD\x6E\x46\xBB\xC3\xF2\x5C\x4F\x76\x47\xE1\x6A\xBA\x5E\xD2\xEE\xA8\x5D\xED\x8A\xB8\x3B\xCA\xF1\xEE\x28\x07\x50\xAA\x72\xD8\x1D\xE5\xB0\x3B\x80\x53\x6B\xD9\x74\xC2\x1D\xB1\xDC\x11\xBA\x07\xBE\x90\x59\xC0\xA3\x25\xD9\x78\x52\xFC\xBA\xC6\xF1\xCD\x05\xBB\x11\x6B\x1F\x2B\xF7\x2F\xAA\x4A\x24\x6A\x84\x94\xD7\x4B\x3B\xA8\xC2\x0E\x0A\x7B\xE6\xCB\xBC\x03\x22\xA1\xCA\x3E\x78\x53\xF8\xE9\x36\x29\x37\x8D\x94\x3B\x21\x36\x3B\x21\x36\x2B\xC4\x96\xDC\x26\xB1\xC1\xA4\xF5\x67\xDA\x06\xE9\x64\x1B\xD8\x3F\x73\x3B\xEB\xB6\xF4\xAD\x89\xFF\x2B\x46\xF9\x40\xE7\x58\xA6\x70\x10\xF0\xC3\x22\x67\xFF\x59\x28\x5C\x1A\x48\xBE\x12\x14\x4E\x77\x2C\x2A\x24\xBB\x8A\xD7\xAB\xE0\x52\x0D\xEB\x36\xF0\xAD\xC9\xDF\x6B\x98\x4D\x06\x8B\x32\xEF\x86\x4A\x22\x7F\x21\x63\x88\x74\xEE\x94\x7F\x55\xF5\x08\x88\x6D\x7F\x45\x82\x8E\xF8\xFC\x94\xD4\x7C\xDE\x76\xFC\xEB\xFB\x97\x7E\x75\xDA\xBF\x5C\x04\xCB\xE1\xF7\x9C\xD2\x11\x7E\xE3\x9A\x5E\x72\x61\x9C\x55\x36\x54\x78\xB4\x6C\x85\x7C\xCB\x92\x5F\x43\xAF\xFA\x35\x3E\xC3\x85\x19\x06\xDF\xC6\x53\x6E\x7C\xF9\xB3\x6A\x7C\xED\x90\x9B\x7D\x92\x25\xAC\x27\x82\x78\xAB\xC3\x23\x5D\xB2\x63\xBE\x81\x6F\x7D\x02\x68\x01\xB6\xDF\x31\x4F\x8A\xC1\xF4\xFC\xA4\xE1\x97\x97\xDF\xFB\x46\x67\x06\xD9\xDC\x90\xEC\x95\x8C\x7C\x2C\x74\x7B\xDB\x77\xD9\xC8\xD3\xF2\x08\x37\xBB\x1D\x9B\xA5\x41\xDF\x17\x9B\xB4\x3B\xE6\x5E\x97\x8C\x2E\x6D\xF7\x23\x27\x8E\xDD\x31\xF7\xF3\xF3\x8E\x07\x73\x1A\xB1\x89\x2F\xAB\x1E\x77\xB2\x57\x07\xB3\xE3\xB4\xFF\x65\xD5\xB7\xBF\x76\x3D\x98\xD2\x3B\xEB\x61\x0A\x7E\xA4\x33\xA3\x58\x41\x92\xCD\x62\xF7\xF5\xB6\x39\xBF\x28\xA0\x51\x48\x07\xCA\x30\x57\x70\x33\x94\xB0\x08\xD8\xA1\x7C\x8C\xF6\x3F\x72\x4D\x42\x24\x4B\x57\xB2\x64\x83\x02\x83\x45\xDF\xFE\x86\x50\x8C\xF1\xCF\xF5\xAE\x04\xAC\xB6\xB7\x87\xFE\xB9\x5E\x00\xD9\x4A\xAE\x5D\xC9\x91\x1A\x35\xC3\xFA\xA9\x6D\x53\x70\xAD\x3C\x7B\xD3\x51\x7D\x9A\x49\xD6\xA0\xDE\x40\xC3\x05\xB0\x8C\x00\x10\x25\x4C\x8F\xE7\xA5\xC2\xFB\x10\x14\x69\xBD\x39\x5C\xA4\x2E\x75\x21\x3C\x03\x25\xD4\x59\xCE\xC7\xD6\xD1\x52\x68\x09\x9E\xC5\xE0\x5C\x63\x7B\xDF\x60\x63\x64\x05\x2C\x86\x78\x18\x8E\x8A\x8B\x01\x7D\x69\x1F\x32\x53\x89\xED\x8B\xFB\x06\xD3\xBE\xB0\xD1\x55\x4F\xAA\x68\x37\x54\xE7\x74\x66\xCF\xDB\x67\xC4\x5C\xCA\x55\x35\x78\xE8\xC5\x74\xD4\xBF\x19\x81\xEA\x3A\x8C\x9C\x0B\x85\xF8\xE2\x40\x32\xF3\x91\xC6\x1F\xAB\x7E\x72\x11\x10\xBD\x6D\x0A\x6A\xBD\xFD\x57\xE1\xE1\x62\x61\xD8\xE4\x9F\x06\x2D\x98\x7A\x87\x29\x13\x98\x71\x64\xAD\x6C\x47\x27\xE8\x7D\xF1\xD3\x03\xF1\xD3\xCB\xD1\x31\xEA\x66\x1C\x42\xB1\x63\xEE\x66\xAD\xB6\xF2\xF6\x19\x2C\x84\x1D\x21\x55\x03\xC0\xC6\xAB\xC5\xDC\x03\x5E\x66\xD1\x12\x67\xF3\xEA\x11\xC6\x7C\xA6\xFF\x5F\xD9\x12\x2F\x0D\xD7\xBD\xF7\xF9\x7E\x27\x68\x73\x06\x98\x9D\xE6\x59\xFF\x27\x37\x6E\xDC\xB0\x70\xBE\xF8\xE4\xD0\x91\xF6\x70\xE0\xD8\xF3\x0D\x2D\x93\x94\xBB\x6D\x1A\x8F\x1D\x30\xD8\x22\x60\xE2\x51\x4F\xEA\x34\x0D\xEC\x13\x28\x03\x03\xED\xF5\x65\x25\xB5\x2F\x94\xFF\x2B\xF8\x94\x61\x90\x82\x1B\xE2\xEF\xC7\xA7\x06\x53\x40\x9F\xE6\x4E\xF9\x7B\xF1\x69\x03\x13\x44\x9F\x5A\x81\x9A\x83\x8D\xA4\x71\x73\xB7\xE1\xDA\x9E\xFF\xB9\x19\x09\xE2\x9F\x79\x69\x6C\xC5\xC5\xF4\x07\x5D\xAD\xEE\x40\x22\x17\x1A\xED\xBF\x6F\xB8\x4B\x6A\x50\x01\xFB\x4D\x8A\xC9\x21\x6A\x86\x95\xF4\x22\x5E\x2E\x76\x4C\x11\x4A\xA8\xB1\xC3\xC8\xEE\x75\x86\x4E\xB5\x0C\xB5\x03\xFD\xD1\x23\xC4\x91\x42\xA5\x78\xF6\x41\xA7\x6C\xCF\x7E\x9C\x35\x1D\x62\xD7\xC1\x64\x19\x03\x2A\x5A\x06\xCF\x7D\x9C\xC9\xA4\xE0\xCA\x5D\xBF\xA0\x7A\x44\xF8\x03\xAD\x38\x1D\x58\xC0\xC8\x05\x9D\x30\x68\xF0\xD1\x23\x8D\x71\xC6\xEB\xB8\x45\x18\x04\xF4\x12\xC2\x48\x2D\x3B\xB8\xC7\x07\x85\x28\xB5\x87\xFC\xD6\x4B\xFC\xBA\x9F\x0B\xEC\x9B\x14\xEF\xB0\x47\x11\x79\x73\x77\x67\xF8\xD4\xE4\x9B\x89\xE3\x32\xCE\x01\xB3\x9A\xA3\x1E\x70\x1C\x9E\x7A\xF2\xF6\xFD\xBD\x45\xE6\x12\xA0\x06\x70\x96\xBB\x01\x61\x1C\xF6\xC8\xE1\x7C\xBD\x96\xD0\x87\x3E\xFE\x1A\x7E\x2A\x86\x9F\xBC\x8E\x37\xD6\xA3\x5F\x0D\x7E\x35\xBE\x05\x8A\x25\x02\x0D\x3A\xD5\x7B\x29\xFD\x89\x04\x54\xAE\xDD\x01\xC3\x91\x39\x04\xD3\x9B\xF4\x80\x9F\x47\x08\xB5\xE4\x45\x46\x22\x95\x13\xB5\xE8\x04\xDC\x5A\x8D\xCB\xD4\x08\x22\x40\x8C\xA2\x05\x76\xD7\xB2\x7B\x90\xF1\x5F\xF3\x25\x47\x60\xE5\x4A\x7E\x88\xDB\x3F\x0D\x0E\xC9\x9F\xCF\x74\xCA\xD9\x5D\xA9\xFF\x33\x2E\xAD\x63\x62\x69\x1D\x76\xBC\xC7\xA8\x04\x67\x2F\xA2\xEB\x74\xBB\x33\x1C\x55\x34\x11\x08\xB6\xBB\xF5\x6E\xCB\xED\x91\xBF\xF2\x01\x3E\x92\xEE\xEF\xD4\x3A\xA7\xE5\xFD\x8B\xE0\xF1\xBE\x3F\xA4\xF6\xAB\x91\xD3\xF2\xDE\x85\xD4\x4C\xBB\x77\xF9\x6A\x38\x01\xD6\x77\xE0\xFC\xA8\x03\x8F\x87\x33\x71\x6D\x07\x1E\x89\x1D\x78\x64\x5D\x07\xDE\x18\x3A\xF0\xC6\xF5\x1D\x78\xA2\x5B\xF3\xE3\x5B\x8E\xE9\xD5\x5B\x46\xBD\x7A\x8E\x7B\x75\xB8\xBE\x57\x87\xB1\x57\x87\xEB\x7A\xF5\x54\xE8\xD5\x53\xCB\xD7\xFC\x35\xBD\x52\x61\xE0\x0F\xFE\xA2\xDE\x18\xA1\xE8\x64\x01\x62\x47\xA2\xA9\x0A\x6A\xCA\xB3\x99\xBA\x7D\xF7\xF5\x61\xCB\xD7\x9D\x48\xE7\xBF\xF8\x0F\x96\xDC\x8D\x2D\x17\xBC\xF1\x72\x52\x95\x5B\x2E\xA7\xFB\x7E\x78\x3A\x88\x76\x51\xF0\x29\x54\x46\x94\x18\xAF\x1C\xDD\xFC\x70\xA8\xD5\x56\xC8\xC9\x99\x44\xEF\x4F\x37\x63\x9F\x19\xAC\x3C\x25\x1B\x57\x82\x69\x9E\xB1\x46\x16\xB9\xCB\x5C\x21\xFD\x46\xC4\x65\xEC\xF7\x90\x23\xB2\xBE\xE3\xC4\xC6\xBE\x36\xBD\x2E\x6E\xD6\xEB\x50\xA0\x50\xAA\x82\x95\x52\x17\xA9\x0E\x0E\x37\x48\x52\x69\x2C\x49\x53\xB2\x36\x90\x02\xDE\x89\xD1\x24\x4B\xEE\x53\xE1\xCA\xBD\xAE\xE0\xD0\xC4\x82\x55\x1A\xD2\x2D\x0D\xAB\x34\xFA\xF2\x96\xA8\xFA\x1C\xAD\x88\xFA\x05\xDD\x2C\xA4\x90\xEC\x70\xF9\x27\x6E\xBA\x70\xB3\x9E\x4B\xCA\x14\xC3\x4B\xB8\x62\x3E\xB3\x74\xB8\x43\x72\x74\x98\xD5\x20\xF8\x4E\x68\x54\xA8\xCE\x07\x78\xE0\xDC\xBF\x5D\x1E\x45\xC8\x0E\xE3\x7E\xD5\x6B\x97\x26\xA1\xA5\x29\xBD\x12\xC4\xC5\x2D\x57\x2F\x2F\x0D\x8A\x4A\x00\x79\xAB\x5E\xE4\x23\x64\x1F\x57\x72\xF5\xF2\xFA\x4D\x9C\xBE\x97\x0C\x95\x30\x18\xA7\xEB\x31\xF6\x40\x05\x79\xA9\xC6\x92\x90\x5C\x58\x4B\xE8\xCA\x60\x57\x45\x11\x96\xAE\x61\x70\xDE\x7C\x7F\x9E\x2B\xA5\xB5\x36\xA9\xA9\x5C\x46\xB3\xFD\x2C\x47\x4F\x35\x6B\x03\x80\xE8\x9E\x66\xDB\x20\x06\x0B\xA5\xED\xE6\x98\x8E\xC5\x06\x86\xBD\x41\x8D\xB6\xFE\xE8\x0A\x50\xD8\x96\x46\x6F\xC2\xE8\x8D\x8C\xBE\x5D\x1E\xBD\xE1\xD1\x9B\xF1\xE8\xC3\xC8\xDB\x87\xB9\x2C\xAB\x10\xA6\xA1\x39\x68\x07\xC2\x34\x91\x30\xF3\x81\x30\x5B\x4E\x4B\x05\xAA\xEF\xB6\x51\x7B\xBE\x7E\x2B\x54\xD1\xD5\x73\x66\xFA\x6A\x79\x0F\x1F\x3E\xFC\xFE\x65\xF6\x35\xED\xAA\xFF\xFD\x7F\x08\x40\x93\x49\x87\xDF\x84\x33\x69\x69\x22\x56\xC6\x18\x6F\xBF\xD5\x64\x8C\xDB\x5D\x1A\x3E\x57\x55\x0B\x24\x10\x06\x7F\xB1\x51\xFE\xDD\x53\x33\xC2\x97\x65\x55\x6E\xD2\x91\x62\xD4\x91\xE2\xB8\x8E\xAC\x36\xC8\x8B\x27\xED\x5E\x6A\x94\xFF\xA9\xE9\xEB\x47\x4B\x32\xBC\xB0\x3B\xD9\x73\x82\x93\xE1\xE7\x4B\x81\xD3\x8A\x93\xE0\x5A\x8E\x64\x2B\xFD\xD5\x35\xAB\xBD\xBC\xCC\xD4\xA9\x75\x2B\xFD\x65\x9B\xB6\x75\x13\x77\x72\x34\x6B\x0C\xE7\x3D\x9E\x37\xA2\x5E\xFF\x9D\xD3\x89\x93\x67\x16\xFC\xCC\xD0\x7E\x78\x10\x01\xE8\x78\xD5\x3A\x12\x8F\x83\x2E\x2B\x1E\xE3\x42\x4A\xF2\x85\xA1\x13\x7B\x1D\x06\x39\xAE\x89\xC4\xBD\x43\xD4\x9E\xCB\x43\xE9\xBB\x1C\xA9\x62\xE3\x2A\x48\x27\x69\x93\xB1\x3B\xA2\xF0\xBF\xF8\xA1\x4F\x7D\x48\x3F\xD6\x94\xB4\x8B\x1C\x5F\x29\x49\xA0\x3E\xE5\x4E\xC1\x1A\xF5\x29\xBA\xEE\x4A\xFF\xA1\x4F\x7D\xE8\x53\x30\x45\x0D\x4F\xA0\x7C\x49\x29\x5C\xFB\x94\x9B\xF9\xFA\x29\x6A\x11\xAF\x97\xDA\x7E\xA7\x5C\xD9\x2F\x4E\xB9\xC2\xD7\xFD\x82\x6B\xC6\xE7\xEE\x24\xA7\x82\x9F\x62\x1C\xBF\xCD\xA1\xD6\x17\x87\x40\x16\x28\xCA\xCA\x8F\x17\xC4\xB7\x0B\x3E\x93\x66\x28\x08\xD8\x95\x52\x4B\xB0\x10\xA6\x1E\x53\xF0\xB8\x4E\x21\x3F\x37\xA3\x39\x5B\xE4\x34\x22\xAE\xC2\x28\x6F\x43\x70\x5F\x7C\x29\xFB\xE7\x2D\xA7\x99\x97\x9C\xA8\x59\x92\x8C\x59\x0E\xB8\x90\x05\x8E\x1A\x12\x49\xF9\x58\x0A\xF2\x4D\xE9\xF2\x8B\xF4\x12\xDC\xEE\x0A\x88\xA4\x13\x6F\xC9\x57\x92\x24\x6F\xC1\x54\xD6\xD3\x64\x3B\x94\x2F\x0B\xF4\x08\xFD\x3F\x3E\x89\xA2\x35\x1D\x62\x55\x67\x43\xA4\x6A\x49\x7F\x8A\xAE\xE5\xF3\x3E\xD4\x0E\x45\x46\x04\xA3\x9E\x4A\x6C\xE7\x5E\x77\xD2\x9B\x67\xBA\x13\x40\x98\x0D\x67\x7B\x29\x22\x43\xEB\x8A\xBD\xAE\xF5\xFA\x19\x90\x55\x79\x39\x48\x04\xED\x58\x22\x68\x19\xD4\xAF\x06\x84\xA8\x34\x30\x43\xB7\x83\x7D\xA8\x62\x0F\xF3\x29\x57\xBA\x13\xC8\x65\x3C\xC1\x1D\x2A\x79\x70\x8B\xD6\x9D\xA0\xDE\x9F\x30\xE7\xDD\x49\x56\xDC\x88\xAC\x4F\x46\x73\xE7\xCC\x9D\x10\xDE\xB3\x03\xDA\x6C\xB1\xA4\x00\xB4\x0C\xEB\xBA\x41\x42\xC6\x86\xAB\xC4\x0A\xBD\x22\xAB\xE5\x5B\x38\x7E\xBB\xF2\x66\x02\x9B\xAC\x5B\xC1\xEB\x56\x46\x71\x6D\x16\xC5\xB5\x20\x04\x74\x2D\x8B\x6B\x33\x5E\xC3\x36\xAC\x61\x09\x44\xD6\x86\x97\x6B\x06\x0B\xC6\x6C\xAD\x24\x3F\xBC\x93\x5F\xD1\x56\xE8\x1F\x6C\x69\xAD\xF4\x60\x8D\x58\x3F\xEA\xAA\x9C\x8B\x6D\xEC\x70\x77\xD2\x95\x7C\x2E\x9E\x74\xB7\x1C\x66\x1C\xA4\xD4\x35\x5F\x19\xDA\xF2\xC0\x96\x0F\x9A\x63\x1B\xFC\xEA\xBF\x59\xF3\xB3\xC5\xDA\x23\x6E\x69\x92\xE3\x0B\xE9\x88\x6B\x77\x06\x88\x65\x20\x46\xB6\x4C\x66\xCC\xED\x4B\x3E\xE2\x8A\xD5\x23\x6E\xDA\x26\x56\x2C\x52\xDB\xAD\x96\xED\x8B\x98\xA9\x75\x73\x75\x72\x4C\x7D\xC5\x2A\xF5\x2D\x71\x90\xE1\x99\x45\x2B\xAA\x47\x68\x3F\x3C\x18\x4E\xB5\x35\x82\xDB\x78\x9C\x45\x38\xD5\x5A\x87\x26\xC2\xA9\x66\x2B\xD7\x8A\x4D\xCA\x54\x1C\x11\x95\xE2\x68\xBB\xD8\x24\x75\x56\xDD\x44\xD8\x9D\xA3\x59\xC6\x01\xA4\xA3\x10\xFA\x89\x6C\x28\xCE\xAE\x46\xC0\x73\x3C\x03\x59\xB5\x21\x7E\x64\x9E\xE9\x4E\x0E\x95\x00\x93\x50\xE6\x2C\x27\x0E\xB7\x47\x07\xE4\xB4\x12\xA0\x75\xB3\x31\xBF\x9A\x0D\xA5\x75\x67\xA1\x81\x14\xE1\x5F\xE3\xD2\xBA\x27\xC1\xA8\x4E\x3A\x91\x7A\x66\x52\xD1\xF4\xA4\xB0\x59\x46\x4B\x06\x4F\x72\x6D\x64\x54\x29\x2B\x68\x29\xA7\xAA\xA5\xE2\xCA\x19\x69\x3D\x81\x37\xC8\x6E\xAF\x89\x57\x0D\x6E\x96\x6C\xA8\xAD\x2E\xA0\x1E\x98\xFB\x4B\xD5\xB7\xCC\x53\x7B\xA4\xAF\x1A\x54\xB6\x42\xF6\xF6\x73\x7D\x00\xB9\x8C\x6A\x8C\x76\x6A\xAF\x4B\xBC\x7D\xA6\x4B\xE7\x59\x96\x29\x6D\x2C\x02\xAA\x9F\x83\xC7\x06\x27\xCE\x89\x6F\xA2\xC6\x01\x80\xA4\xEE\x61\x53\xA6\x4B\xE4\x60\x4E\x5C\x76\xCE\x70\xE5\xAD\x62\xE5\x09\xA2\x02\xB6\x26\x27\x93\x9F\x0B\xF1\x9F\x15\xD2\x64\x81\x26\xB9\x6E\x30\xCD\x43\xC6\x2D\xAF\x7B\x3B\xDF\xAA\xE4\x56\x15\xDF\x8E\x7E\x11\x3D\x72\xE5\x91\x58\x83\x78\x74\x43\xB1\x28\xF9\x01\x7A\xAE\x3C\x27\x28\x33\x93\x57\x24\x18\xD7\x22\x73\xC9\x9A\x16\x18\x59\x54\xF9\xB6\xC7\xA8\x9C\x69\x7F\xFB\x3A\xD7\x19\x93\x6B\x5D\x81\xAB\x5D\x8A\x4C\xA0\x2E\x89\x77\xF0\x2C\xE4\xCB\x6F\x33\xC3\xDB\x72\x79\x5B\x1E\xC6\x9E\x2E\xDF\x9A\xF2\xAD\xE9\xD2\xCC\x43\x67\x5E\xBA\x55\xDD\x63\xDA\x30\x4D\xC5\x39\x60\xAD\x62\xD4\xC5\xF2\x8C\x9A\xE3\x66\x94\x23\xDB\x13\xFF\x47\x7A\x28\x89\xF7\x6F\x62\xE5\xB4\xD4\x9B\xB7\x72\x38\x9F\xC4\x6B\xFE\xD8\x5F\x85\xD9\x14\x97\x92\xB7\xC2\x51\x45\xB3\x90\x0B\xBA\x44\x98\x87\xC5\x0C\x31\xD8\x93\x2E\xE8\x61\x0E\xAC\x0C\xCC\xA2\x0B\x88\x65\xCB\xC7\x37\xB3\x41\xBF\xE7\x28\xCD\x9C\x9F\xCA\x97\xA6\x83\x8B\x8D\xAC\x3C\x65\xE4\xA9\x38\x89\xD3\xF9\x46\x68\x8E\x5A\x79\xCA\xF6\x21\x22\x54\x88\x7E\x4A\x13\x6E\x86\x5D\xCD\x6C\x66\x4A\xF7\x70\xC8\x4E\x7E\x61\x64\x30\x8E\x2D\x9F\xB9\x94\xE6\x3E\x0E\x61\x3A\x70\x00\xE9\xE6\x6B\x97\x8A\x7B\x31\xED\x3B\x87\xA5\x09\xF5\x73\x45\x97\x35\x0D\x32\xFC\xF2\xF4\x49\xBC\x68\xA9\x9F\xB4\x20\xC7\xF4\xAB\xF2\x66\x31\x1B\x80\x51\xD5\xF2\x83\x48\x2C\x05\xA2\x99\xA0\x79\x25\xC2\x43\x18\xF9\xA8\x00\x4C\x9F\xD8\xA9\xC3\x2C\xD3\xDB\xED\x6E\x93\xD9\x63\x5A\x3C\xA6\x2B\x2C\x18\xE6\x88\x6B\xE0\x72\xB6\x76\xB7\xC9\x6B\xAE\xAD\xBC\xDC\x48\x75\x1C\xD3\x12\x6A\xE2\x94\x03\xEC\xDE\xE4\x31\xA4\xAC\x16\xC4\x27\x11\xA9\xCB\x21\xB9\x4B\x54\x58\x71\xE9\xA4\xC2\xC6\x21\xAE\xBC\x12\x04\x85\x04\xF1\x74\x51\x4C\x08\x6F\xA9\x0B\x09\xC8\x35\xDF\xDD\x5A\xE4\x81\xB8\x59\x0E\x2D\x58\x41\xC9\x97\x77\xB7\x72\x05\x37\x55\x2C\xF3\x8C\x61\x34\x33\x2E\x2F\xEA\x4C\xFB\x99\xEB\xA1\xB2\x4E\xC1\xEC\x59\x87\x1F\xD1\xD2\x42\x93\x16\x93\xD5\x69\x85\x1A\x85\x39\xEA\x24\x10\x9B\xD8\x73\xDA\x15\x7B\x8F\xC6\x76\xE0\x70\x45\x1B\x0B\x55\xDB\x0A\x9F\xD0\x96\x5C\x58\x68\x44\x47\x71\x09\x75\x8B\xB9\xCB\x69\x12\x13\x7B\x5E\x02\x58\xBB\x64\x65\x59\x12\x2C\xA2\xE5\xD4\x2B\x9C\x12\xD3\x69\xE6\x68\xD6\xCB\x8D\xF2\x09\x07\x65\x1E\xC3\x9B\x79\xDD\xE1\xE5\x40\xA4\x2D\x7F\x34\xA8\x24\x5B\x55\xFF\xA6\xD1\x63\x2C\xF7\x3C\x00\x41\x4B\x71\xF0\x50\xE7\xBD\xA0\x3F\x05\x57\x68\xD6\x1C\xB2\x0C\x0B\x24\x8C\xA2\x05\xEF\xF2\xCC\x8B\xF3\xBC\x9B\xB1\x62\x24\x37\xCE\xE4\xC6\xC2\x65\x0C\x54\x3F\x56\x32\xA8\x5D\x52\x43\x91\x72\x26\x55\x1E\x66\x70\x5A\x23\x75\x1E\x29\xD8\x8C\xD1\x8F\x23\x9E\x75\x51\x7F\x74\xA5\xAB\x56\xF5\xC9\x94\xC4\xC5\x99\x57\x8C\x81\x5C\xAC\x11\x06\x53\x94\xC0\xC5\x75\x98\x51\xA0\x49\x28\x57\x88\x30\x98\x4E\x04\x67\xDC\x07\xC1\xB9\x60\x49\x25\x4A\x83\x34\x3D\xAE\x8C\x76\x4A\x57\x3A\xBD\x0B\x1D\x61\x4D\xA7\x4C\xE8\x94\x39\xAE\x53\x86\x3B\x65\xB8\x53\x93\x0E\x99\xA5\x0E\x99\x51\x87\xC6\xE2\xE9\x60\x2B\xE5\x9C\xB8\x62\x6C\x2B\x35\xC6\x18\x6D\xD8\x0E\xB6\x46\x62\x2C\x68\x01\xAA\xB0\xD4\xA4\xE3\xD2\x52\x77\x35\xEB\xB8\x15\xAF\x61\x39\xC8\x8C\xBC\x56\x5C\x51\xDD\x48\x45\xF5\xA8\xE3\x4A\x1C\x96\xAB\xDD\x6C\xAF\xAB\x49\x66\xDC\x70\x1B\x23\x1D\xB7\x1E\xCB\x8C\xF5\xA0\xE3\xD6\xA1\x81\x1C\x76\xE2\x41\xC7\x2D\xDD\x1C\x32\xE3\x9C\x7B\x52\xB2\x19\x79\x51\xB9\x39\x9D\x2E\xF5\x50\x3D\xBD\x9C\x54\x4F\xCF\xB8\x7A\x7A\xC6\x85\x1F\x32\xD6\x71\xA7\x95\xA2\x8B\xB1\xCC\x48\x9D\xE0\xD8\x58\x99\x84\x8B\xE2\x0D\x24\x12\xB8\xC0\xA4\x51\x2F\xB4\x3D\x1F\x34\x23\xD0\x60\xBD\xBA\xDC\x36\xA8\x2C\x16\x10\x58\x6B\x96\xDB\xB2\x42\x62\x01\xF0\x13\x34\x97\x4C\x16\xDC\x46\x4D\xAA\x91\x25\xB7\xB2\xE4\xD9\x68\xC9\x39\x3C\x71\xE9\xED\xC7\x37\x7C\xAB\x1E\xAC\xEB\x43\x23\x86\xBE\xE1\xF5\xC5\xF1\x0A\xD1\xF0\xC4\xD0\xAC\x54\x6C\x0E\x8F\xB1\x3D\x05\x94\x53\xF1\x14\xAB\x5D\x49\x62\xA9\x38\xEC\x60\x9D\xFF\x61\x65\x47\x67\xB7\xBF\xA3\xB3\xB8\xA3\xB3\x9B\xEE\xE8\x4C\x42\xD4\xD6\x2C\xA6\x09\x8B\x69\x8E\x5B\x4C\xC3\x53\x69\xD6\x4C\x65\x34\xFF\xCF\xA2\xF9\x9F\x27\x75\xCE\xD6\xD3\x19\x4F\xEB\x7C\x32\xAD\xA4\x14\x71\xE8\xC1\x1A\x0B\xC7\xE4\x65\xD2\x36\x07\xD7\xF1\x3B\xD7\x9A\xFC\x47\x9D\x1B\x77\x8D\x0D\x1D\x31\xEE\x61\x0D\x39\x1D\xFB\xF4\xAD\xE6\x60\x42\x4C\x66\x4C\x4C\x66\x0D\x31\x2D\x1B\xFD\x9D\x71\x65\xFB\x9D\x02\x9E\x55\xC2\x60\xDB\xBE\x47\x1C\x72\x15\x9D\x24\x15\x44\x26\xE4\xB0\x24\x7D\xA7\xB1\xF4\x1C\x8E\x63\x42\x0C\x4D\x1E\x55\xBE\xC1\xDD\xFA\xF9\x13\xBA\x3C\x12\x97\x1E\xC7\x60\x80\xEC\xA0\x20\xEF\xA2\x4F\x9A\xD5\xE3\x2C\x7A\xD6\x32\xF6\xDC\xA4\xF1\x5C\x13\x67\x9C\xDC\x18\x5C\x70\x05\xEB\xA4\xC9\x34\xB8\x62\xEA\x89\x93\xF8\xC6\x9C\xCF\xB5\x15\x1F\x1C\x34\x05\x7B\x5E\x22\x2F\x11\xE1\xB8\xCE\x13\x57\x04\x4F\x5C\xB6\xC6\x49\x9A\xF0\x32\x24\xCB\xCB\x90\x47\x0F\x5C\x3E\x78\x3D\x92\xD1\x4A\x98\xE8\x14\xB5\xEC\x81\xCB\x87\x93\x2D\x0F\x27\xDB\x9A\x2E\xAD\xEC\x8E\x95\x2E\x4D\x29\x63\xD2\x21\x13\x3A\xD4\xCD\x8E\x27\x8E\xFC\xA2\xA4\x1A\xC0\xAE\x9F\xB1\x5D\x1F\xF6\x99\x2C\x18\xF4\x33\x12\x9E\x32\x39\x9A\x18\x6A\xA7\xAB\xE8\xE8\xB1\xCF\x74\xCD\x24\x97\x2B\x97\x5C\xAE\xC2\x55\x92\x63\x30\x77\xF3\x51\x2E\xD7\x24\xC7\x20\x1B\x62\x4C\x63\x03\x85\xCB\x96\x62\x4C\x1B\x76\x4C\x8A\x79\x1E\x97\xFB\xC5\xDC\x35\x24\xB9\x65\x5C\x67\x2B\x84\x99\xD6\xA3\x0A\xED\x73\x36\xF6\x67\x83\xB1\x7F\x12\x4D\xCD\x96\xF9\x8C\xDE\xFC\x1C\x89\xDD\xC5\x18\xB5\xE3\x26\x4B\x51\xDE\x72\x29\xCA\x09\xA3\x2A\x97\x97\x63\xB6\x64\x46\xCF\x06\x33\x7A\xBE\xDE\x8C\x3E\x1B\xDB\xD0\x69\x85\xD1\xD9\x2E\xF7\xC5\xFE\xF0\x34\xD0\x56\x5C\xEE\xF3\x7D\x1A\x40\x6D\xAB\x69\xD6\x0C\x67\x51\x65\xFC\x5C\xBE\x3F\x2F\xB4\xB2\xD6\xA6\x26\xD5\xD5\xB0\xF6\xF9\x68\xED\x47\xCE\x9C\x82\x37\xD1\xC8\x9D\x53\x4A\x7C\x71\x35\x98\xB2\x4C\x30\xBD\xE7\xC1\x13\x2F\xF1\xC5\xF9\xFA\xB5\x1F\x97\x75\x0E\x0D\xCC\xC6\xA1\x99\x58\xFB\x0A\xDD\xAE\x1E\x8A\xFE\x17\x5A\xFB\x1A\xB5\x70\xA0\x51\xC4\x10\xE3\x72\x14\x62\x8C\x42\x54\x05\x87\x23\xC7\xB5\x9F\x38\xF0\xB1\xF6\x79\x05\xFD\xA0\x2B\x64\x54\x7B\x5D\x4E\x1D\xD4\x28\x27\x25\x04\x51\xEC\xBB\xDC\xEB\xC3\x03\x86\x31\x60\xF9\x19\x86\xD0\x7C\xAD\x0F\x7C\x40\x72\x58\x63\x18\x2C\x8F\x33\x0C\x16\x7B\x5D\x85\xD9\x9A\x18\x06\x8B\x68\x18\x2C\xC4\x30\xD8\xB8\xE6\x8B\x31\x0C\x96\x2B\x86\x41\x48\x75\x5D\xCD\x3D\x09\x86\x41\x12\xF6\xC4\x30\x58\x05\xC3\x60\xE1\xAA\x91\x61\xB0\x5C\x35\x0C\x4E\x3D\x18\x32\x2D\x34\xB4\x67\xD1\x89\x2F\x1F\x4D\xA5\x5F\x2A\x4D\x65\x5F\x0B\x9A\x72\x59\x30\x15\x08\x45\x2C\xB1\xF1\x7C\x8B\xC3\x58\x93\x67\xBB\x15\x59\x6B\xCC\xCE\xA7\xA6\xF4\x2C\xF2\x90\xEC\xA6\x9E\xF5\x6C\x3D\x1B\x09\x0F\x0D\x8D\xBB\x92\xE8\x3F\x1B\xB2\xD5\x86\x56\x98\x9D\x64\x81\x9D\x7C\x19\xB9\x44\xF2\xA5\xAE\x68\xFA\xB5\x5C\xD1\x25\x3F\x81\xAA\x46\x18\x09\xFF\x3C\xB1\x19\xCC\xDC\xA3\xAC\xC9\x58\x89\xD3\x7F\x2C\x24\x4B\x26\xFE\x23\x45\xDF\xA5\xE1\xCB\x07\xE8\x68\x95\x2F\xF4\xE7\x85\x82\x93\x6F\xCF\x70\x64\x2C\x67\x43\xE6\xCE\x72\xC9\x6D\x11\x71\xB8\x1A\x33\x57\x1E\xA7\xB1\x97\x43\x62\xE3\x50\xA6\x78\x9F\x44\x92\x7B\xEC\xC7\xD8\x72\xF0\x10\x6C\x7E\xFC\xA2\xE9\xFB\x2A\x97\xD0\xE8\xDF\x4B\x2F\x9E\x8D\x92\x32\xAF\x9F\xB3\x1F\x2B\xF0\xF2\x82\xF5\x47\xF4\xEB\x34\x1B\x09\x0A\x14\x1D\x3C\x03\x40\xD7\x04\x30\x68\x3B\xA6\xC5\x27\xD7\xEF\x98\x1A\x9F\x5E\x28\x7A\x04\x0D\x81\x84\x03\x98\x4D\x09\x64\x22\x57\x9E\x55\x09\x63\x2B\x8C\xF2\x30\xA7\x17\x5D\x79\x5C\xBA\x26\xC3\xB6\x21\x81\xA9\xCB\x5D\x71\x56\xE9\xAE\x6A\xFF\xB5\x04\xE4\x1F\xFB\x14\x75\x63\x35\xC9\xB3\x5A\x93\xE4\x59\xBA\xCA\x7F\x4E\xE1\x3F\x44\x84\x57\x1A\x01\x8A\x2B\xFD\x93\x3D\x92\x3D\xF9\x65\x22\x32\x9E\xEE\x4A\x5E\x10\x22\xB0\x64\xDB\x9C\xD9\xEB\x4A\xCF\x09\x35\x37\x6E\xE8\x4B\x5B\x5E\x1F\xBA\xE2\x75\xDA\x1C\xBC\x5E\x9B\xCA\xEB\x45\x8E\x9B\x48\x2B\x4E\x50\x71\x91\x24\xDE\x33\xAE\x74\xE6\x0A\x97\x77\x1C\x25\x96\x7C\x77\x81\x12\x5D\x5C\x35\x5C\xCA\xB0\x33\x5A\x39\x63\xD9\xE2\xE7\x6F\x02\x91\x4C\xE0\x34\x12\x67\x7D\xF6\xEC\x80\x4C\xA0\x9D\x65\x20\x8D\x8C\x31\x20\xE2\xE3\xEC\x7E\x59\x6D\x22\xA4\x58\x77\x09\x91\x90\x72\x19\x40\x1E\x54\x80\x8F\x8A\x97\x53\x97\x0E\x97\xF5\xB1\x68\x1B\x89\xD3\xD4\x9F\x7C\x48\x92\xD6\x82\xB6\x81\x44\x66\xED\xCF\xEF\xAF\x4D\x91\x8E\x68\x0F\xC9\xFA\x14\xE9\x64\x7D\x8A\x74\xB2\x36\x45\x1A\xF0\x3D\x02\xF9\x90\x4C\x20\x1F\xA6\xD3\x61\x50\x3E\x59\xE0\x31\xEC\x00\xBB\x61\x05\x55\x46\x57\xCE\x8E\x74\x06\x67\x90\x00\x6E\x87\xB9\x36\x34\xFB\x80\xDC\x98\x4E\x77\x40\xDC\x90\xBF\x43\x84\xA3\xF1\x2A\xCE\xFF\x22\x07\x0E\xCD\x15\x4E\xC2\x97\xFC\x2A\x3B\x4C\xB2\xF1\x5A\xCA\x17\xE7\x71\xA9\x8E\xBD\x2D\x40\x1E\x9B\x09\x2C\x47\x8A\x37\x2C\xE1\x87\x18\xCE\xDC\x58\x02\xB4\xB0\x63\x40\x8B\x8C\x01\x2D\xB0\x50\x96\x17\xCA\x2E\x03\x5A\xAC\x2E\x94\x5D\xBF\x50\x76\xFD\x42\xD9\xB8\x50\x46\x16\xCA\x4E\xB0\x39\x9C\x0E\xC8\x1C\xE3\x2D\xB0\x3E\x37\x9F\x13\x9A\x26\xE3\xD1\x3C\x1E\x3B\x8C\xC7\x7E\x35\xC6\x23\xB8\x9C\x32\x18\xC6\xF7\xCA\x19\x33\x2C\x71\xB9\x4B\x2E\x6D\xB1\x70\xA8\xB6\x4D\x1D\x70\x45\x9D\xD9\xED\x12\x94\xBD\xB5\x5E\xF7\x2E\xD9\xEA\x74\x40\x96\xA8\x7E\x80\x0B\x08\xD2\xFD\xA7\x3B\xEB\xF5\xC3\x8D\xF2\xEF\xF9\xF0\x04\xC2\xF0\xF4\xC2\x56\x3E\x40\x00\x18\xD8\x49\x9C\xDD\x23\xAA\x30\x4E\x2F\x2C\xE7\xA2\x38\xEB\xF4\x1E\x10\x1F\xFC\x4F\x8F\x13\x39\x40\x38\x67\x80\xE8\xDE\xBE\x97\x7E\xEA\xEC\x05\x60\x54\xA9\x6D\xE3\xD0\xBA\xB3\xBD\x3F\xEA\xEF\x55\x5C\x26\x88\x21\xA5\x2C\x93\x26\x98\x58\xF5\x2E\x6B\xCC\x91\xBE\x8A\xDA\x21\xC9\xD8\x09\xE5\x8C\xD3\x93\x5F\xB2\x30\xE7\x52\x3A\x34\xBB\xD0\x70\x95\xD3\x45\x8E\xE2\x8B\x2C\x4A\x19\x97\xB3\xBF\x28\x5D\x6E\x32\x3A\xA6\xF2\x58\x59\x14\x49\xF9\x63\xC7\x54\x1E\xD2\x32\xD4\x9A\x06\xD9\x31\x35\x7D\x12\x2F\x5A\xEA\xE9\x05\x5A\x8D\xF5\xFD\x82\xF3\x82\x73\x65\xCC\xF4\x11\x7E\x21\x2A\x9C\x0E\x0E\xA4\x9C\x1D\x7E\x4B\xED\x23\x13\x4B\xF7\x1B\xD5\x71\x6F\x61\x57\xF9\xCA\x23\xA6\x0F\xDE\xE7\xD5\x19\x00\x04\x2B\x92\xF5\xAB\x1F\xB7\x26\x3D\x32\x57\x07\x3F\x44\x28\xF6\xB8\xD6\x6D\x9E\xB2\x1F\x39\x5B\x72\x9B\xE7\x31\xB5\x26\xA7\xE9\x80\x9B\xB7\x98\xB8\x79\x8B\x73\xA6\x58\x75\x9C\x6B\x76\xF3\x16\x13\x37\x6F\x31\x5E\xA2\x42\xCA\xAC\x72\x9B\x25\x5F\x6E\xC5\x4D\x2E\x77\x15\xE1\xF7\xE8\x3E\xCF\x96\x9D\xB9\xEC\x61\x83\x7C\x12\xA4\xF0\x0C\x2F\x62\x1F\x1B\x7C\xE2\x7A\xB7\x91\xB2\x5E\xC4\x25\xBA\xD4\x65\x2B\x6E\x44\xF6\xEF\xD8\x82\x73\x9C\xE1\x20\xCA\xFA\x0E\x70\xAF\x89\x6F\x7B\x6E\x19\xE9\xBA\x6F\x83\x6F\x91\x71\x24\x33\xFF\x5C\x2F\x25\xD4\x59\x62\x58\x72\xDA\x51\x23\xFE\x74\xCF\x20\x35\x96\xCB\x8A\x4A\x88\x59\xEE\xD2\x45\xC6\xDE\xBE\xC4\xE9\x8B\xB1\x9A\x50\x40\x90\x2F\x46\x71\x14\x2B\x4B\xE8\xF5\xB4\x50\x22\x22\x2A\xBC\x7D\x66\x9E\xA5\xE9\x38\x9A\x22\x5D\x5E\x96\xE4\x38\xC7\x72\x9D\x48\x38\x45\xBA\x3E\x9C\xA2\x98\x7A\xB8\x17\x5A\xC2\x29\xB4\xB4\xC9\x1E\xFD\x42\x96\xBA\x08\x71\x02\xEB\x5E\xBF\xD6\xF9\x9F\x8C\x3C\xCA\x9A\x37\x6E\x11\xE3\x30\x47\x77\xAC\x06\x54\x14\xCB\x71\x0B\x76\xF0\x7C\x2E\x35\x61\x6F\x12\x51\xC1\x4C\x41\xB1\x6B\x17\x5B\xA1\x60\xF2\x49\x97\x42\x2A\xF4\xF2\xFB\xF4\xF0\xBE\x61\x2F\xAE\x8F\x93\xD0\x83\x27\x75\xC9\xAF\x5F\xAC\xFA\xC4\x8F\x0B\xA9\x58\xF1\xD3\xEB\x63\x67\x55\x57\x31\x48\x64\x69\x51\x25\x68\x64\xD2\xBF\x10\x94\xD9\xC0\xAB\x73\x7C\x07\x56\x62\x3A\x6E\xDE\x81\xB0\xAC\xB3\xC9\x66\xCF\x8E\xD9\xEC\xD3\x58\x99\xE3\x3D\xD1\xD3\xF9\xE3\xDA\x5C\x5D\x11\xB7\xBA\xC7\x68\xB0\xE1\xE9\x30\x4E\x57\x37\x7C\x12\x37\x7C\x2A\x1B\x3E\xED\xBB\x82\x36\x7C\x31\xDE\xF0\x69\xD8\xF0\x38\x36\x52\xB8\xE9\x79\xC3\x17\x6B\x37\x7C\x8A\x0D\x0F\xB4\x14\xD9\xF0\xF0\x92\x77\x33\x5F\xEC\x36\x26\x6E\xFE\x04\x28\xD1\x58\x1A\x6C\x7E\xDA\xD1\x0C\xFF\x57\xF4\x4E\xEF\x02\x25\x72\x28\x21\x9C\x54\x7F\xBF\xD4\x49\xE0\x03\xF7\x45\x3E\x60\xCE\x7B\xB5\x6D\x7E\xE2\xFD\xDF\x7A\x4D\xF9\x23\xCE\x39\xB6\x7E\xBB\x47\xCD\x4A\x80\x98\xEF\x98\xC2\xE3\x32\xBA\xEC\x3F\x4A\x0F\xB5\x3F\x1C\x72\xDE\xB4\xC4\x6D\x25\xED\x07\x03\x0E\xAE\xE3\xB2\x24\xF7\xD8\x88\x2E\x8C\x04\xC2\xF0\xDD\x19\x7C\x6A\x19\x2B\x69\x9B\xFF\xDC\xE7\xB8\x9C\x2B\xBD\x67\xC7\xDC\x8B\x54\xD3\x90\x16\x7A\x56\xDD\x1F\x4A\x46\x71\x52\x6A\x00\x30\x7E\x40\xAA\x5B\xEB\x8A\xC4\xE1\xD7\x6B\x2C\xE1\xCB\x5C\x35\x91\x1F\x3C\x1F\x1F\x7C\x45\x8D\x9F\xFC\x84\x1A\x1E\x4D\xE2\xA3\x9C\x67\x4F\x1A\xEE\x93\x0E\xB7\x7E\x83\x43\xD9\xD2\x27\xF8\xC7\xC7\xF9\x47\x2E\x30\x39\xAD\x74\x7C\x5A\x32\xB5\x91\xB8\x88\x22\x57\x9D\xF6\x7A\xBF\x4B\xB7\xD8\x60\x00\x41\x4E\x7B\xFD\xAC\xA3\x5F\x6A\x01\x81\xF4\x77\x7B\x46\x17\xF0\x66\x77\xAB\xD3\x4E\xFB\xF7\x6C\x1C\x00\x5C\xE0\xAC\x52\x24\xBB\x99\x03\x40\x0E\xB8\xF0\xFD\x5A\xCB\x3F\x2C\xC2\x0F\x05\x7F\xEF\xE2\x0D\x8E\x7F\xF8\xCF\xF0\x83\x53\xFE\x6E\xA0\xDC\xE8\xA7\x0F\xE8\xA2\xE6\x8B\xFF\x79\xBC\x5B\xDA\xFF\xCB\xE1\x07\x69\xFE\xBF\xE0\xA7\x51\x59\xEC\x9F\xA0\x7A\xAB\xDE\x31\xDB\x26\xA2\x5E\xC6\xDA\xCB\x9A\xEF\x79\xD7\xF3\x0C\xFE\xAC\xB1\x72\x5C\x4F\x4F\x7B\x1A\xCF\xC6\xE5\xAD\x1D\x73\x5F\x30\x04\x9D\xE9\x52\xE4\x9F\xB7\x02\x37\x8D\xA9\x2B\xB8\xEA\x29\x30\xA8\x3A\xED\xDF\x26\xE0\x6E\x90\xD1\x2B\x86\x05\x6D\x7B\xFF\xB6\xFD\x80\x14\xBF\x10\x24\xCD\x1A\x4E\x06\x7D\xC0\x13\x9C\xEC\x98\x96\xCB\x12\x6A\xAF\x0F\x1F\x54\x05\x4F\xB0\x3E\x58\x24\x74\x16\xB0\xFA\xE9\x74\x34\xE5\x24\x4E\x33\x0E\x36\xAC\xF4\xA3\xA2\x64\x4C\x61\xC1\x4C\x69\xE2\x06\x34\xA1\x26\x40\x04\x7E\x0F\x16\x2E\x8D\x84\xD6\x9F\x53\x7C\xB9\xFD\xCE\x50\xCE\xDD\x22\xB3\xB9\xE6\x84\xB6\xED\xF6\xF3\x88\x69\x31\x02\xA4\xFE\x38\xEA\x16\xD1\xDE\x80\xFE\xF7\x8E\x87\xEC\x51\xFB\x8B\x21\x7D\xFE\xEB\xA9\x4F\xC3\x34\x2F\xB4\xBF\x2F\xE6\x2C\xEB\x87\xFE\x97\x6B\xD7\xAE\xBD\x90\xF0\x11\x81\x24\x7B\x43\x03\xE9\x9C\xF2\x1B\xD8\x47\xBC\x0C\xC6\xAB\x7B\xD4\xFB\xBF\xEF\xFA\x35\x75\x4E\x71\xEF\xE9\x87\x97\x86\x1F\x5A\xFC\xF0\xD1\xE1\x87\xD3\xF8\xE1\x27\x86\x1F\x5C\xEF\xD5\xEB\xD4\x27\xE8\x87\xD7\x2B\x15\xA1\xFC\x1E\x6F\xFF\xE7\xE7\x81\x70\xF0\x04\x7F\xFF\xFA\xF6\xEF\xD2\xD0\x1E\x54\xDF\x20\x7E\xF6\xBB\x6F\xBF\xD8\xDB\x7D\x2B\xEE\xC0\xB7\x69\x7D\x34\xD0\x17\xEC\x73\x1E\x43\x02\x6B\x60\x18\x65\x9A\xFD\xE9\x3D\x09\xDF\xF3\x09\xBE\x07\x1E\xC8\xE3\xEE\x79\x65\x7A\x4F\xF5\x4D\x5A\x90\xD3\x05\x47\xFD\x10\xDF\xCC\x53\xB4\x95\x51\x58\x8D\xBE\xDD\x8F\x33\x30\x80\xA4\xBB\x1E\x26\xAF\x36\xB0\x37\xFA\x96\xF0\x97\xD7\xF4\x5E\x33\x8B\xD9\xEC\xC1\x57\x40\x51\x6C\x66\x04\xDB\x91\x8F\x88\x42\xFA\x1D\xC6\x49\xAF\x0E\xB4\x5E\x05\x33\xE7\x2C\x67\xA9\xA1\xF5\x14\xB3\xE6\xD0\xEB\x33\x82\x6D\x40\x4C\x5A\x03\x0B\x02\x3D\xF5\xBF\x07\x60\x84\xB6\xF7\x3F\x3A\x61\xD2\xDB\xE6\x49\x64\xC0\x06\xD8\x62\xA7\xAA\xB3\x82\x21\x8F\x54\x6F\x52\x8A\x05\xBC\xFD\xF1\x48\x93\xF6\xA8\xFD\x25\xAE\xB4\xF4\x64\xF5\x0B\x06\x10\xF8\xB8\xA1\x13\x2C\xF6\x58\x3E\x93\xA8\x7B\xDD\x8F\x5F\x1F\x7E\xFC\x14\xD7\x50\xFD\x96\xE1\xD2\x93\xC7\x5F\xBA\x57\x2E\xC5\xD3\x80\xDD\x9C\x7A\x20\x25\x3D\x22\x25\x3D\x79\xF6\xCD\xA3\x67\x3F\x21\xCF\x72\xC4\x7B\xD2\xFE\x3E\x83\x3D\x4C\xEE\x7F\xCB\xE8\xFE\x57\x96\xEE\x7F\x75\xE5\xFE\xD5\xD3\x64\x72\xF1\xFE\xB5\x1C\xE4\x26\xBB\xA0\xBA\xA4\x06\x41\x95\x8E\x51\xE9\xA3\x1A\x75\x98\x47\xDE\x86\x91\x73\xD1\x88\xE5\xC1\x57\x0F\x2F\x37\xF4\xEA\xB8\xA1\x57\x43\x43\x7C\xAE\xC4\xB6\xCE\xAC\x6D\xEB\x5F\x98\x41\x48\x78\x97\x8E\xE9\xCF\x9E\xF5\x61\xB5\x6D\xEE\xE3\x71\x3C\xE1\x52\xE0\x60\x7A\xE5\xDF\xA9\x19\x8D\x80\xCB\xB0\x43\xDE\x21\xE6\x4A\xD2\xCB\xC1\x56\x97\xFB\x93\x97\x1A\xE5\x8F\x60\x63\xEF\x4A\x97\xB3\xD1\x7E\xDF\xE5\x17\xB7\xDA\xEF\x67\xF7\x3D\xA2\xB6\xB8\x8D\x32\x60\x24\x6A\x2F\x01\xFD\x86\x04\xDB\xF6\xFF\x91\xE8\x3F\x62\x31\x2C\x7C\x15\xB0\x65\x6F\x9B\x84\xEB\x5C\xEC\x89\xB1\x87\x21\x06\x20\xE7\x21\xEE\x0F\x31\x61\xA9\x57\x57\xB6\x00\xB1\x84\xF0\xC8\xA3\xBE\x4B\x21\x72\xFA\xA4\x5F\x68\xB6\x0A\x59\x06\x75\xCF\x43\x99\x10\xC5\x4F\x5B\x96\x70\x72\xF9\x4F\x72\xE2\x5A\x30\x20\xD4\xAA\x62\xD3\x6C\x79\x01\x80\x7C\x30\x7B\x97\x38\x6B\xCA\xF6\x7D\xE2\xE6\xF2\xEF\x0A\x09\xE6\x2E\xAF\x3E\xC3\xA6\xD9\x7C\xB4\xD9\xF3\x38\xBD\xC5\x30\xBD\xC6\x59\xC9\xC6\x49\xBA\xD2\xD9\xCB\x5C\x4B\xF6\x1E\xA3\xBA\xFF\x9F\xBD\x77\x81\x93\xA2\xBA\x12\x87\xCF\xA9\xAA\xAE\x7E\x54\xF7\xD0\x32\x18\x51\x06\xBB\xBA\x17\x75\x50\x06\x06\x54\x54\x14\xA5\x40\x5E\xF2\x8C\xE0\xFB\x31\x14\x33\xCD\x4C\xF7\xF4\x74\x0F\xDD\x3D\x28\xC6\x30\x93\x28\x6A\x8C\x6B\x4C\xB2\x49\x36\x89\x89\xE4\xEF\x46\x44\x62\x56\xF3\x36\xD1\x85\x3C\x75\x93\x35\x81\x28\x6A\xDE\x98\x64\xB3\xEB\x26\x21\x68\x7C\xF0\x50\xEA\xFB\xDD\x73\xEE\xAD\xAE\x2E\x06\x93\xFD\xFF\xF7\xFB\xBE\xDF\xEF\xFF\x4B\xC1\x99\x5B\x75\xFA\xBE\x9F\xE7\xDE\x73\xEE\x39\x89\xAD\x36\x9B\xCB\x1E\x6F\x47\xED\xC4\x5C\xFB\x8E\xAD\xB6\xD6\x6F\xEB\x76\xBC\xBF\x98\x4B\x8A\x79\x21\x3A\x97\x49\xAB\x76\xED\xCA\x6C\xAA\xC1\xCD\x5D\x4D\x96\x9C\xF9\x9E\x98\xA6\xCE\xFB\x5B\x3A\x00\x73\x63\x58\x8B\x8D\xBE\x30\xA5\x39\x5A\x36\x6D\xC7\x52\x68\x39\x48\x67\xF1\xDA\xC2\x09\xB6\x61\xC7\x26\xF0\xAD\x0F\xA5\x26\x84\x06\x42\x4E\x17\xB4\x8C\x2E\x48\x1C\x7D\x02\x2B\x38\xD7\x52\x68\x27\x6D\x63\x31\x55\xEB\x53\xD2\x2B\x69\xFB\xC1\xA2\xAD\x4F\x90\x2A\x49\xA1\x03\x96\xCE\x67\xFB\x9B\x8B\xC8\x9C\xB1\x51\xCC\x1E\x67\x5B\x24\x95\xAA\x77\x00\xD8\x31\x91\x4F\x8D\x68\x48\x5B\x0B\xE8\xD3\x9A\x20\xDC\x85\x29\xC3\x8E\x89\x66\x8C\xD1\xC1\x51\x4E\xB7\x8F\x5B\xEC\xAB\x2B\x25\x0B\xA4\xA2\x0F\x11\x89\x2C\xF2\xD2\x4F\x35\x22\x7E\x10\x33\xA5\xA3\x39\x68\x13\xA9\x95\x4D\x5B\x0E\x88\xD9\xF5\xDF\x68\x2D\x00\xE7\x78\x31\x30\xB2\x9A\xDD\xD2\xAE\x2D\xCA\xD2\x5D\x8D\x49\xD9\x98\xC8\xEB\xB5\xA4\xDB\x0C\x66\xC1\x18\x5B\x77\x60\xA1\x9D\x12\x85\x31\xED\x94\x6D\x2E\x9A\x90\xD3\x1D\x73\xD1\xFA\x6C\xD2\x19\x19\x49\xDB\xB1\x81\xEC\x71\x0E\x28\xF3\x09\xFE\x9E\x3A\xDD\xA2\x83\x86\x71\x96\x7C\x21\xB6\x3E\x0D\x90\x31\x72\x7D\x4C\xFF\x81\xA8\xAB\x16\xDA\x6F\xA4\x92\x86\x45\x3A\x61\x30\x1B\xB3\x5B\xD8\x7C\x1F\x3A\x82\x8E\x12\x83\x66\x4C\xD1\x81\xF4\x66\x32\x17\x6B\xD8\xD1\x0E\x18\x93\xD2\x93\x51\x4B\xF6\x1B\xDF\xB7\x92\xCD\x09\xF8\x26\x03\xFE\x63\xE6\xA7\xE2\x94\x84\x6E\x49\xAD\xCD\x98\x8D\x25\xE3\x56\x20\x13\x71\x99\x09\x3A\x02\x8C\x51\x2C\x69\x95\x01\x7A\x1D\x25\x0F\xF3\x53\x31\x0A\xA3\x59\x8E\x96\x25\x46\x45\x8A\x74\x07\xC7\x9D\xF7\xD8\x8B\xD7\xB3\x39\x69\xDD\xC1\x4B\x52\x68\xC7\x9D\xF8\x62\x96\xE7\xB2\xB8\xC7\x1D\xB7\x84\x72\x91\x92\x1E\x28\x55\xFA\x6E\x14\x9D\xB4\x5C\x87\x53\xD6\x64\xCA\xA6\x45\xFE\x58\x4E\xD9\x9C\x9F\x32\x54\xF2\x14\x1F\xA5\xBB\x50\xEC\xE3\xED\xE8\x64\x3D\xCD\x3B\x77\x91\x49\x8A\x31\x1B\xA3\x6B\xAE\xBC\x23\x61\x79\x90\x9A\x9D\x9A\x5F\xE5\xA9\x6D\x35\x19\xD1\x26\xD5\x45\xB6\x9E\xFE\x38\xA7\x2A\xE9\xB8\xD5\xA4\x55\x85\x8A\x20\x66\x03\x2A\x05\x49\x80\xA3\x9D\x92\x01\xBE\x78\x4B\x23\x40\xD3\x0F\x1F\xA4\x98\xB8\xA4\x2C\xDA\x45\x03\xC7\x54\xBA\x2E\x49\x26\x8E\x38\xFD\x52\xB5\x01\xF5\x60\xDA\xD7\x91\x58\xCA\x72\xF2\x45\x07\xF7\x0E\x99\xE7\xEA\x77\xF4\xBA\x54\x04\x63\x91\x1D\x47\x39\xD3\xC4\xAC\x71\x44\x3E\x90\x9E\x0A\xCD\xD6\xD3\x2F\xF3\xB2\x0C\x96\x45\xBB\xAC\x7F\xBC\xFF\xF6\xE0\xE7\xBD\xF4\x79\x3E\x93\x41\x68\x0A\xC2\xE7\x7C\xA4\xD3\xB8\xF3\x51\xE3\x3D\x0F\xB2\xF1\x1E\x31\x56\x5A\x04\x09\x83\x92\xD6\x61\x0A\xC7\xFA\x12\x4A\x13\x39\x6A\x8C\x83\x73\x73\xB1\xE5\x38\xD4\xE9\x01\x5D\x3E\xA4\xB8\x56\xDA\x12\xC6\xF9\x29\xED\x2F\xAE\xEB\x36\xDF\xD0\xF1\x43\xA0\x8D\x3C\xB3\x8A\x1F\x94\xA2\x1D\x49\x4A\xC8\x19\x37\xE0\x5B\xAC\x21\xC9\xA2\xD2\xBF\x91\xBE\xB7\x69\x45\xB7\xC1\x7A\xDC\x40\x7D\x58\xAC\xF7\x36\x2C\xC8\x69\xB3\x20\xC9\x64\xB7\x2E\x55\xCD\xE8\x1D\x90\x64\x15\xFC\x4C\xD6\xC6\xC8\xC2\x69\xCE\x10\x73\x25\x30\x7F\x5E\xDA\x76\x30\x84\x4F\x36\x04\x95\x35\xA4\x1C\x8D\x54\x5B\xB5\x30\x70\xBF\x80\x79\xE1\x74\x82\x8F\x4C\xD9\xA1\xD4\xB9\x26\xAF\xFD\x92\x59\x2F\x83\x89\x7A\x4E\x90\xCE\x06\x6C\x7D\x21\xF3\x35\xA4\xB2\xFE\x99\x5A\x4C\x99\xDD\x34\x54\x20\x41\x75\x26\x79\x27\x90\x54\xF7\xD8\x39\x39\x96\x4D\xE2\x7C\x52\xD4\x16\x5F\x66\xA0\xBB\x5B\x1A\xF3\x19\x48\x24\x54\xE3\x54\x74\xC9\x49\x10\xDD\x90\x75\x45\x05\xFA\x6A\x73\x49\xB5\x60\x49\xE9\xE8\x20\x8B\x72\x13\x76\x74\xD2\x4D\x85\xC4\x63\x16\x52\x5A\x24\x40\xD6\x1E\x46\x1A\x9C\x51\x05\x3A\xBA\x90\x06\x5B\x19\x31\xB8\x3A\xFF\x77\x0A\xA9\x14\x62\xD1\x5E\x4F\x9A\xDE\xD1\x65\x42\x62\x87\xB6\x20\xA7\x8B\x77\x83\xC9\x7A\x2D\x8B\x44\x7E\x58\x96\x75\x67\x04\x23\xC7\x32\x41\x34\x0D\x8D\x9C\xE6\x1C\xF1\xC4\xD2\xCD\xE7\x2B\x2F\x34\xCE\x57\x46\x21\xDD\x9D\x17\xE4\xE6\x26\x5D\x74\xBE\xD6\x20\xDD\x29\x22\x31\xAF\x42\xBB\xF6\xAC\xF0\x91\xD3\x99\x36\x21\x31\x83\x08\x71\x01\x3D\x4F\x5F\x2F\xFE\x28\x89\x3C\xB6\xD0\x46\xDE\xF4\x39\x4A\xAE\x28\x9D\xD3\x6C\x90\x62\xB7\x90\x13\x2B\x75\x84\x0D\x90\x68\x36\xAC\x10\x93\x24\x47\x1B\x91\xD6\x6A\x74\xDB\xA0\xD3\x27\x36\xA5\x26\xEF\x29\x44\x1A\x26\x6B\x02\x56\xFA\x35\xD2\xC0\x65\x6B\x73\x25\xCB\x01\x98\xA5\x2F\x2B\x76\x5C\xB1\x21\xD6\xA9\xF3\x46\x83\x8E\x00\xA4\x0E\x38\x9D\x15\xA2\x13\xDD\x15\x91\xEA\x35\x22\xE9\x43\x6C\x28\x9E\x3A\x07\x89\x56\xC5\x2C\x8A\x4A\x29\x77\x23\x35\x66\xAA\xEF\xFD\xDF\x59\x5C\x10\x2D\x2A\xE6\x60\x8A\xD3\xF3\x74\x79\x54\xC5\x3C\xDA\xF3\xD1\xB0\x9A\x77\x73\xE7\x89\x4D\x33\x38\xA8\x6C\xE3\xEE\xA2\xDE\x36\xDA\xEE\xF7\xD9\xD0\xEE\xF7\xE3\x26\xD9\x4F\x51\x9D\x16\x03\x5D\x89\x3B\x2E\xF7\x3B\x1A\x6A\xA8\xAC\xDF\x33\x2B\x50\xDD\x3A\x37\xB8\xAE\xD9\x74\x47\x8A\x6E\xD4\x18\xA2\x9E\x0D\x4A\x24\xAB\x26\x83\xF9\x29\x43\xCC\x0F\x96\x8D\x82\xD4\xCE\x22\xD5\x33\xC5\xC6\x38\x79\x78\x00\x64\xC8\x44\xD5\xB7\xC1\x13\x81\x31\x57\x5E\x3E\xA2\xFA\xD6\x79\x26\x34\x9A\xEA\x5B\x93\xF5\x8D\x74\x93\x06\xD9\x24\x25\x4F\x26\xA2\xBE\x95\xA4\x37\xCA\xFA\xA6\xD9\xC3\xAF\x6F\x43\x2D\x1A\xCE\x70\x51\x5A\xB5\xF1\x3C\x9D\xEF\x10\x6A\xE7\xA3\x11\xAA\x08\xBA\x06\x47\x56\xAE\xCD\xAC\x6E\x47\xB2\x48\x8A\xCC\x68\xE3\x98\xB6\x61\x71\xCE\x98\x20\x4A\xCD\x96\x8C\x65\x9E\x44\xEB\x9B\x74\xCB\x88\xFC\x2C\x91\xBA\x83\x69\x8A\x46\x79\x58\x94\xA3\x3B\xB0\xBA\x52\xC6\x4F\xB7\x90\x78\xB5\xA0\x2F\x4E\x80\x7D\x49\x25\xFD\x62\xA9\xCA\xA9\x62\x5B\x4A\x01\x0A\x2D\x50\xDC\x39\xE9\x08\x8B\x4A\x3A\x2C\xF5\x13\xE9\xE9\x23\xDC\xE1\x78\x1D\x1C\x63\x85\x9B\x3A\xE2\x37\x75\xE4\x7F\xB4\xA9\x23\xFF\x63\x4D\x1D\x39\xBA\xA9\x23\xCD\x4D\x1D\x09\x36\x35\x4D\x1B\x4D\x4D\x4D\xE3\x95\xAE\x84\x8A\x06\xCF\x11\xB5\x43\x2D\xBE\x9E\x94\x83\x79\x44\x2D\x59\xFB\xE3\x68\x0E\xDB\xC8\xA7\xD0\x9A\x7F\xC1\x44\x9F\xAF\xB6\x66\x06\xDB\x77\x33\x24\x5D\x40\xEC\x37\x5E\x93\x52\x48\xCA\x1E\x25\x9F\x18\xC5\xEA\xE2\x2F\x47\x0E\xD0\xDC\xC2\x6D\x15\xCB\x49\xEB\x70\x24\xE7\x64\x2A\x93\x69\x1A\x89\xD2\x3B\x74\xD6\x99\xA5\x25\x89\xE4\x44\x95\x2C\xF5\x4C\x4D\xCC\x65\x1D\x90\x94\x16\x5B\xA5\x4E\x48\xEA\x1D\x24\xF7\x4E\xB7\xD8\x63\x4C\x80\xB0\xB9\x92\x99\xF2\x76\x35\x9B\xFE\x89\x09\xCA\x7A\xA6\xD2\x86\x69\x90\x14\x9E\x62\xA5\x46\xE5\xFA\xA9\x09\xDA\x2C\xC9\xEB\x20\x9D\x25\x2C\x98\x90\x95\xDA\x86\x3D\x79\x9A\xC1\xBB\x18\x31\x05\x8B\x6A\x10\xF9\x61\x8B\x45\x86\x22\x96\xC8\x84\x84\x14\xC0\x58\xCF\xBB\xBE\xA3\x57\x70\xAE\x55\xCE\x81\xDC\xDA\x73\x0E\xB0\x79\x05\x67\x2D\x96\x6A\x05\xC7\x99\xEA\x00\x14\x78\xA0\x33\x03\x19\xE4\x9C\x2D\x32\x4B\xD1\x67\x8D\xA0\xF2\x2B\xC3\x57\xE0\x27\xAA\x6F\x3E\x6F\x48\x0C\xBF\xC7\x23\x21\x89\x9E\x02\xA2\x07\x55\x61\xD4\x75\x50\xC9\x60\x11\x54\x90\xA9\xA8\xAE\x31\x96\x1F\x83\xE8\xDE\x92\x42\xE2\x93\x5F\x8D\x0B\xCC\xD1\x31\xDB\xD9\xC8\x71\x47\x96\xAA\xE8\x88\x9C\xB3\x58\x02\x80\xCA\x4A\xE9\xC9\xFB\xF0\x46\xA0\xF5\x0C\xD5\x70\x06\x59\xD9\xA0\x19\x4D\x92\x36\xB2\x9D\x64\xED\x1A\xD2\xEA\x1C\x99\x92\x63\x3B\xE9\xC8\xC4\xED\xFF\x41\x8A\x4C\x6F\x1F\xAB\x1D\xB9\xFC\x5C\x2C\x49\xD3\x72\x13\x1A\xCD\x4D\xC8\xDA\x27\x54\x13\x1A\xDC\x76\x92\x9E\xD5\xD8\x78\x08\x4D\xB9\xAA\xF5\x40\xB6\x9E\x9C\x06\x55\x13\xF9\xF5\xAB\x7E\x33\x9A\xDB\xD4\x6F\x3E\xD9\x78\x84\xA6\x39\x65\x21\x5F\xA5\xE0\x66\x93\x99\xE5\x5D\x88\x8A\x54\x93\xD6\xF5\xB4\x51\x1B\x8F\x6B\xF4\xFF\xB0\x2A\xFF\x62\xE3\xC9\x65\x86\x4E\x9B\xFF\x27\x1A\xCF\xE0\x2E\xEA\x93\xDA\xC6\x02\x25\xD7\x6B\xB0\xC5\x3E\x4A\xC3\xB2\xB6\x6B\x01\x62\xC0\xA4\xF9\x15\xD2\x07\x1B\x47\xB4\xCC\x0B\x86\x99\x5A\xD2\x01\x56\x59\xAC\x96\xCA\xF5\xE4\x99\xCD\x0A\xD3\xB4\xC5\x1C\xF2\x76\xD2\xDF\x87\x0B\x26\xE4\x22\x59\x93\xCD\xCF\xA8\x8B\xAB\xA8\x8C\x3D\x23\x13\xCB\x4C\x77\xA8\x23\x64\xC2\xF2\x10\x42\xD2\x61\x9A\xD3\x88\xEA\xD1\xE4\x2C\xCD\xE1\x4C\x0A\x10\xCA\x20\x6D\x1A\x92\xB4\xCC\x28\x7B\xD0\x4A\x8F\x37\x15\x6A\x71\x4A\xCA\x84\x69\xA4\xA3\xBD\xB1\x6F\x84\xF3\xD1\xB4\x9A\x0B\x2C\xE6\x23\xB4\xF5\x69\x68\x0A\xBA\x9F\x8D\x02\x8A\x09\x59\xD0\xFF\x87\x94\x3C\x95\xD8\xC1\x92\x81\x89\x9F\x7F\x24\x6C\xE8\x85\x48\xD1\x94\xE4\x06\xD8\x7C\xDA\x39\x53\x03\x69\x78\x52\xEC\x3C\xC4\xCE\x5F\x2B\x90\x34\x16\xF9\xA2\xFD\x8C\x98\xF6\x9D\x61\x52\x9D\xCD\x7A\x63\xC5\x5E\x42\x14\xDC\xD6\xFC\x33\x53\x39\xF1\x19\xCC\x8B\x52\x96\x50\x48\x26\x93\xF5\xBB\xEA\x2C\xEE\x36\xCE\xD6\x8A\xA2\x33\x28\xEB\xC2\x52\xC1\xAA\x3C\x3E\x6B\xA8\x36\x37\xD4\xC6\xBA\x5D\x4B\x4B\x2E\x15\x2D\x99\x69\xB5\xA3\x96\xC2\x75\x62\x94\x5B\x2F\xF3\x7E\x9F\x2D\xF9\xE9\x3C\x37\xE2\x34\xD4\x78\x19\xC3\x31\x72\x5F\xE8\xD0\x5D\x70\xDD\x39\x82\xEB\x1D\xB2\xFC\xA8\xFC\xA0\xBF\xEA\xEB\x7C\xC8\xA0\x3B\x51\xD2\x29\xEB\xBC\x67\x84\xA6\x37\xDD\x49\xCA\x63\x06\x9D\x97\x7E\x64\xFD\xD7\x31\x3A\xEB\x8D\x51\xE3\x3B\xDA\xE2\x09\x62\x5F\x4F\x8C\x35\xDA\x2E\xCA\xBE\xA3\x8F\xB1\x9A\x73\x73\x74\x6A\x74\xA4\xA1\xDB\xC4\x0D\x96\x9B\xCD\xB7\x4F\xDB\x6A\x01\x27\x26\x56\xC1\xF4\x7B\x36\x8B\xF6\x6D\x01\x27\xDA\xFC\x19\x0F\x7C\x1E\x34\x31\xA6\x4C\x6F\x6B\x2D\x1A\x82\x46\xBA\x78\xA0\x91\x15\x12\xB2\x03\xC7\x14\x39\x34\xF8\xD8\xC5\xE0\x63\x17\xC3\xD6\x45\x1E\xE9\x4B\xCC\xB4\xEA\xBC\xC5\xF0\xCF\x5B\x68\x38\x8F\x1E\x46\x7B\x9B\x30\x4E\xD0\x70\x9B\x4A\x9A\x49\x7A\x79\xFD\x43\x67\xA5\xF2\xEA\x56\x5D\x44\xD0\xB0\xCC\x31\xCD\x19\x59\x24\x43\x49\x7A\xFA\x51\x3A\x2D\xD4\xF8\x70\x8F\xF6\x32\xA6\x9A\x89\x45\x44\xC6\xA2\x00\x4D\x67\x72\xF6\x4C\xCE\x9E\xC9\xD5\x6E\x72\xB5\x9B\x7E\x35\x9B\x7E\x35\xD3\x9E\x99\x36\x3C\xE5\x9C\xA0\x9E\xB9\xAF\x93\x2E\x50\xE3\x7C\x34\xD9\x46\x58\xCC\x8E\x66\x63\x64\x1D\x30\x17\xCF\x26\x6C\xCC\x5A\xFA\x9C\x06\x15\x19\xAA\x10\x4A\xD1\xE0\x14\x0D\x3F\x45\x43\xA5\xE8\xBC\xE9\x8D\x91\x89\x88\x29\x98\x76\x5D\xA6\x6D\xB0\x9C\x7F\xD2\xB6\xF8\x77\x56\xB5\x1C\xE3\xC9\x3B\xC6\xF2\xE6\xF4\x53\x31\x6B\xD9\x09\x91\x91\x04\x1B\xE5\xA5\x1B\xE1\xC4\xDC\x8B\x8B\xB8\x4A\x54\x88\x22\xD9\xC6\x17\x9D\x2E\xBD\x99\x19\xA7\x62\x9E\x6A\xD4\x11\xB7\xE4\xDB\x57\x55\x8B\x5F\x55\xB2\x41\x55\x6D\x2F\x4E\x81\xF3\xA1\x7B\x02\x16\x62\x45\x0F\x0F\x34\x88\xB6\x48\x9A\x40\x96\xA5\x54\x3E\x44\xD4\x52\x75\xB6\x44\x50\x8B\xB2\x4A\xD2\x2C\x29\x92\xD7\xD7\x67\xD5\xF9\xB2\xAF\x45\x9A\x74\x48\xB3\x91\x25\xF1\x13\xAB\x49\x28\xE6\x92\x64\xDE\x34\x97\x90\xF2\x34\x49\x31\x4E\x13\x4E\xAC\xA8\x94\x46\x38\x69\x3A\xB5\xE0\x9B\x99\x31\x16\x54\xB3\x98\x4E\x20\xA5\xAA\xB4\xBD\x01\x5B\x6A\xA7\x88\xA9\x59\x4D\xB1\x62\xFC\x2A\x1F\x2E\xE6\x2C\x0A\x1C\x99\x9F\x6A\xA8\xE8\xB6\x6C\xB3\xC1\x3F\x1E\x0B\x4E\x84\xD9\xAB\xAA\xBE\x05\xCA\x0C\xA1\x16\x8A\x0D\xB3\x3F\x2B\x34\x06\xBC\x8D\x3C\x3F\x58\xAC\xE1\x1A\x42\x13\x43\x63\x2A\xD2\x68\x77\xD2\x1C\x8F\xC1\xA1\x0D\x27\x41\x23\x8E\x43\x1B\x2A\x34\x9D\x80\xA5\xA9\x13\x72\x24\xFE\x4E\xC4\x36\xAC\xD7\xD5\x84\xEA\xC0\x25\x81\xF1\x73\x8C\x03\x59\x8D\xF9\xCA\xA3\x9C\xC4\x4A\x2A\x8D\xC6\x26\x71\xB5\x1A\x33\x6D\xC0\xFF\xE8\xB1\xEA\xE7\x37\xC7\x65\x33\xFB\x3B\x38\x7D\x72\x40\x9D\x03\xFA\x93\xA9\xC1\x53\x77\x8B\x3F\x6B\xCA\x28\x68\x57\x2C\x72\x91\x1E\x61\x01\x11\xD1\x8C\x6A\xCF\x45\x5B\x62\x6B\x7E\xB0\xFE\x34\x3F\x0A\x8D\x13\xD2\x38\x21\x8D\xEB\x55\x73\xC6\xB1\x86\x70\x95\x43\x79\xFC\xA2\x59\x37\x9B\xC8\x37\x1E\x36\x09\xD7\x3F\x93\x3B\x89\x0E\xD4\x72\xDA\x64\x60\xB3\x11\xB1\x62\xCE\x98\xCC\x62\xAA\x4A\x80\x11\x27\x03\x09\x3B\xB0\xCC\x1E\x0B\x81\x46\xED\xC8\x39\xD2\x3E\x7A\x8E\x3C\x8A\xFE\xA4\x0B\x14\xDA\xA6\x70\x0C\x3B\x46\x02\x12\xE4\xCF\xFA\x63\xB4\x61\xF4\x2C\xDD\x9F\xD3\x1A\xE7\x80\x8A\x68\xC3\x14\xDB\xEC\x5B\xA0\xC8\x23\x9D\x96\x8A\xBB\xA5\xA0\x88\x58\xAB\x59\x64\xC8\x70\xDE\x55\x74\x4C\x92\xBB\x66\xFD\x94\xDA\x5C\x66\x50\xF0\xBF\x80\x59\x33\x83\xB8\x62\xEF\x2A\x2A\x35\x12\x9A\x48\x41\x90\x37\x86\xB2\x35\x4D\xC2\xB8\xB1\x22\x99\x1A\xD0\x59\x70\x95\xCD\x2E\xC6\x78\x9E\x5B\x2F\x96\x4F\x5B\x97\x57\xC2\x22\xCD\xCB\xAA\xC4\xC7\x55\x1E\xA3\x6A\xBF\x6A\xB0\x3E\x14\xBA\x98\x6D\x28\xA5\xA9\x62\x94\xD2\x11\x2C\xDF\xF2\xD0\xA4\x16\x8A\xC6\x05\x45\xC3\x57\x22\x21\x65\xEC\x2D\x75\x37\xD1\x08\x28\xA0\x50\x17\x13\x0D\xC9\x88\xF2\x0D\xD9\x1A\x2B\x52\x31\x3A\x4C\x91\x4B\x67\xCA\xB2\x53\xFE\xC5\x44\xCA\x43\x8A\xCF\x5A\xB2\x31\xBE\x98\x68\x34\x2E\x26\xA6\xEC\x44\xE0\x62\xA2\xC6\x1A\xCB\xE4\x1E\x4A\xB3\x53\x61\xED\x13\xC1\x2D\xA3\x1D\x2C\x6A\xBC\x51\xD4\xE6\x82\xA6\xE4\x8D\xB4\xC4\xA8\x05\x8D\x89\x82\x5A\xB6\x15\x28\x68\xC2\x2F\xA8\xD5\x30\x66\x68\x35\x0A\x6A\x35\x0A\x4A\xC7\x2B\x74\xE5\x32\x97\xE0\x1C\x18\xF4\x73\x31\x1B\x67\x25\x4E\x7C\x9B\x5C\x5A\x31\x4C\xD9\xB1\x40\x41\xE3\x5C\x50\x8B\x0B\x4A\x2B\x57\x4A\x14\x34\xD5\x10\xBE\xA1\x13\x1B\xB2\xC1\x14\x95\x9D\x67\xB1\x7F\x86\x21\x37\x3B\xFD\xA2\x68\x86\xE4\xDD\x1A\x2D\x5A\x04\xD0\x72\xDE\xB7\x3D\x74\x4D\xB0\xE1\x97\x8C\x93\x3B\x5A\xDD\x86\xA2\x73\x33\xED\x30\x48\xF8\xDC\xD9\x54\xE4\xA5\x90\x34\xA1\x66\x59\x49\x96\xBE\x4C\x5A\xB2\xD0\xFB\xD3\x1F\xE0\x09\xA2\x03\x80\x18\x28\x02\x49\xC6\x63\xF8\xA6\x9D\xC4\x6B\x56\x60\x5B\x01\x0D\x0A\x30\x07\xA3\x53\x86\x39\x7F\x8C\x19\x96\x8A\x83\x79\xF6\x7A\x52\xF3\x17\x0C\x1A\x8F\xB4\xD7\x14\xBB\x80\xF9\xEA\xE4\xCC\x68\x88\x29\x69\x0D\x21\x4A\xDD\xFA\x90\x8E\xDA\xB0\xCF\xE6\xCA\xD1\x29\x09\x1D\xD6\xA0\xBA\x08\xD2\xC4\x00\x16\x2B\x67\xCB\xF1\x89\x04\x6A\xBA\x6E\x18\x46\xC4\x4C\xF0\x03\xB1\x68\x5C\x52\xA9\x30\x0D\xB5\x9A\x33\xB2\x13\xD6\xCF\x1F\x23\x67\x87\x76\x8D\x75\x2C\x2E\x4C\xC5\x82\x7E\x50\xF8\x90\xBB\x0A\xF9\x3B\x0D\xEF\x66\x3F\xA9\xA8\x4F\xB2\x63\x80\x74\x67\xDE\x38\x45\x30\x0D\x4D\xAA\xA7\x05\x63\x9A\xB2\xC0\x39\x08\x61\x52\xBA\xB4\x6B\xD9\x14\xE3\xC2\x94\xD1\xC8\xE5\x82\xA3\xB3\x7D\x54\x80\x05\x32\xE3\x2A\x00\x97\x82\xC5\x22\x89\x89\xD7\xAE\x19\x7C\xE4\xC5\xC6\x6F\x1C\xBD\xCE\x65\x14\x6F\x82\x2E\xE1\xBD\xA4\xF8\xA3\xF9\x25\x71\xD8\xA8\xB5\x75\x9F\x86\xCA\xCA\xB4\xD8\xE3\xD2\x65\x20\x5D\x74\x47\x1B\x17\xA7\x1A\x67\x12\x8E\x56\xCF\xF2\xDD\x18\x31\xFD\x4A\x23\x37\x24\xCF\xAD\xD3\x1E\x59\x53\xEB\x55\xCE\x90\x96\x48\x4D\x3B\x78\xBD\x9A\x72\x47\x17\x51\x05\x41\x27\x48\x60\xB6\x55\xDA\xB8\x60\xAD\x05\x2F\x58\x6B\x4D\xB4\x35\x1B\x6C\x6A\x98\xE2\x6E\xE4\x86\x6F\xD9\x06\xF3\x84\x36\x53\xE2\x7C\x16\x1D\x51\xB6\xB8\x0D\x3B\x12\x30\x53\x4A\xDB\x7B\x5A\xA4\x0D\x2E\x01\x84\xED\x60\x04\x2C\x94\x1A\x4D\xBC\xAA\xE6\x35\x0A\x59\xA2\x51\x74\x61\x3E\x22\x44\x1B\x9D\x96\x25\x13\x1C\xAD\xEE\xEC\x7F\xF9\xD6\x11\x60\x9B\x44\xE9\xCF\xD0\x46\x80\x24\x34\xB5\xD1\x25\x34\x1B\x66\x9E\xFC\x7C\x19\xF2\x1A\x7C\xD0\x52\x0B\x4B\x68\x02\x1B\xF9\x88\x48\xF1\xCC\x88\x20\xBE\x02\x37\xC8\xA5\x78\x26\x6F\x93\xD3\x45\x5F\xFA\x8E\x96\xC4\xB4\xCA\x6A\xC4\x8E\x1C\x9D\x55\x43\x66\x35\x22\xB2\xAA\x18\xB8\x11\xCE\x6A\x24\x24\x4C\xAA\xDB\x1A\x5B\xFC\xD5\xA4\x30\x29\x30\x1F\x90\x55\xC3\x8B\xBA\xA7\xAC\x6A\x0E\x56\xB3\xBA\x24\x3B\x9B\x5B\x42\xB7\x23\x9C\x55\xCA\x27\x2D\x19\xE4\x9D\xA4\x4A\x73\x11\x16\x91\xE1\x79\x32\x42\x42\x33\x13\xD2\x9F\x55\x5A\x3C\x02\x82\xA7\x5A\x80\xFB\x4D\x05\xFE\xEC\x3F\x8B\x59\xF6\x01\x16\xAA\xA6\xA8\x51\x8E\x96\x98\x8C\x80\x67\x43\x59\x3B\x92\x62\x26\xE9\xEA\x34\x49\xB3\x6B\x3E\x87\xA8\xD1\x11\x46\xB0\x21\x6A\xD8\x19\x34\xD0\x3A\x49\x96\x01\x9D\x36\xDF\x14\xEB\x38\xF5\x26\x9C\xA4\x8D\xCE\x97\x5E\xB9\x55\xEE\xB4\xE7\xEE\xBC\x75\xF6\x39\x7A\x3B\xCF\xBB\x40\xAC\xA7\x07\xE5\xFD\x06\x65\x85\x24\x56\x4C\x6F\x53\x57\x1E\x3A\x1B\xAC\xAA\x4F\x18\x74\x6C\x22\x69\x23\x07\xB2\x6C\x49\x06\xD2\xF7\xF3\xFA\x2B\xE5\x82\x44\xD7\x30\xC4\xD7\x7A\xC9\xE7\xE4\x33\x6E\x67\xD3\x7A\xA2\x68\xC0\x46\x36\x1F\xD3\xCE\xBC\x70\xAE\xC7\x09\x36\xB0\x34\x3D\x6D\xEB\x8F\xFA\xC9\xC1\xA2\x32\x6E\x33\x0B\x7C\x43\x48\x26\xD9\xAE\xC9\x49\xDD\x35\x24\x16\x36\x4A\xAC\x68\x2B\x0B\x52\xB3\x00\xC8\x70\xBF\x18\xA0\x7E\xB6\x89\x7D\x63\xF2\xA9\x8A\x1F\x30\x17\x99\x20\x6D\xF0\x88\x01\x4C\x7B\x25\x31\xC8\x1F\xDF\x4C\x83\x1C\x98\x67\xA2\x8E\xB7\x88\x1E\xE6\x70\xD9\x08\xF3\xC9\x4D\xBA\x54\x47\xD7\x95\x22\x13\xFA\x83\x69\x71\x12\x39\x12\x28\x27\x39\x73\x96\xDE\x16\x3D\x5B\xEC\xD4\x12\x36\x3A\x89\x25\x13\x72\x1A\xDB\x7D\xDA\x24\x91\x30\x61\x81\x3C\x73\xD3\x9C\xC4\x42\x25\xA1\x6C\x3A\x98\x95\xC3\x0C\x8B\xFE\x00\x96\x55\x00\x13\xC8\x00\x14\x2C\xCC\x82\x1D\x21\xBA\x48\xA4\x57\xCC\x45\x68\x66\xB5\xC5\x7E\xD5\x1F\x09\x8A\xDC\x0D\x18\xC1\xD3\x9C\x04\xDF\x28\x30\x58\x4A\x1B\xF9\xDE\xBC\x48\x89\x07\xDB\x98\x86\xC4\x36\x58\xD6\x5D\x06\xC6\xB9\x6F\x84\xC4\x42\x54\xDF\x04\x49\x7B\x21\x8B\xEC\x33\x83\x44\xCE\x0B\xE4\x47\x9D\xC7\x45\x68\x03\x24\x68\x1A\x56\x2D\x49\xCA\x68\xA4\xAD\x13\x3B\x22\xE8\x83\x2C\x5F\xE7\x65\x99\x45\x07\xF9\xA6\x87\x20\xF4\x68\x0A\xB1\x44\xF5\x26\x9D\xB1\x62\xE5\x61\x05\x85\x64\xBF\xC1\x28\xB2\x02\x32\x07\xC6\x70\xE2\x74\xF9\x3F\xE9\xBC\x9B\xA5\xD4\x70\x89\xAC\xDF\x84\x1D\x61\x3D\x2D\x32\x31\x3B\x5E\xE4\x7E\xC7\x87\x55\x27\xB1\x82\x01\x41\xBE\x52\x1E\xD9\x4C\x89\x62\xF9\x49\x2A\x3D\xD6\xA0\xD2\x83\x41\xD5\x9E\x38\x19\x0C\xBF\x58\x5E\xFD\x8A\x34\x02\xFB\xA9\xDA\x31\x3B\xD1\x4F\xEB\xF9\x31\x0E\x20\xE2\xBC\xD9\x8A\xF3\x66\x2B\xCE\xBB\x4B\x9D\xE9\xCD\x38\x05\x14\xBB\xAF\xB8\xBF\x39\x8B\xAB\x0D\x98\x65\x5B\x22\x56\x7B\xBD\x54\x60\x22\x2A\xCD\xA1\x53\x5C\x4B\x6C\xF8\x13\xEA\x3E\x8A\xC5\x24\x40\x36\x61\x5B\xED\x5A\x5A\x50\xB5\x71\x07\x16\xDA\x09\xDB\xB0\x31\xFD\xA1\xCD\xF2\x2E\x9A\x20\x56\xF9\xAA\xBD\xE8\x0B\xB6\x61\xFD\x0C\xD5\x4C\xF1\x57\xF7\x06\xDD\xEF\x06\x8A\x3E\x33\x02\x5D\xC1\x60\x91\x4F\x45\x41\x1A\x0E\x2E\x60\xF2\x91\x48\x43\x41\xB8\x06\xDA\x91\x6B\xCC\x5E\xAF\x68\x5F\xFA\x41\x67\x9D\xA0\xE9\x62\xD6\x20\xE6\x1E\xFA\xF2\x46\x59\x43\x1A\x0A\xB3\x58\x50\x90\x6F\x1B\xA8\xC2\x89\x55\x06\x02\x85\xD3\xAD\xD3\x10\x87\xF9\x2C\x3D\xC6\x12\xE6\x72\xC3\x89\xBE\x7C\x38\x5D\x13\xB1\xBE\x98\xC0\xE8\xB0\xA4\x3B\x34\x5B\xCC\x23\x34\x04\x0F\x01\x9B\x16\xFB\x26\x30\x46\x73\x7E\x0A\xF4\x67\xD1\x84\x9C\x61\xA3\xF3\x4D\xA0\x3F\x97\x4C\xC8\x45\xA4\x8A\x5B\x31\x11\x0C\x8B\x6C\xF5\x3B\x58\xAC\x3B\xC3\x35\x3B\xE2\xEC\x18\x2E\xD6\xD9\x58\x9C\xE6\x1C\x02\xFA\x23\xC3\xBF\x08\xF4\xA7\x29\xBC\x71\x54\xF8\xFB\x38\xBC\xC1\xE2\xA1\x69\xA5\x2C\x88\x26\x81\xAC\x61\x47\xA5\x6D\x3F\xB1\x48\x4B\x62\x70\x51\x2E\x46\x23\xCF\xD7\x6E\x45\xD6\xBD\x68\x3B\x60\x2E\xE0\xF3\xC4\xB4\xCD\x17\xD9\xE9\xE8\x7A\x79\x8A\xE5\x7E\x16\xF1\xE5\x51\x5F\x95\xC1\xE8\xD1\xEB\xA3\x47\xDF\x1C\x39\x6B\x07\x25\xB5\x5D\x74\x93\x76\x5C\x2E\xEE\x0C\xD7\xB8\x4F\xF8\xBA\x27\xD8\xD6\x1C\x87\x89\x91\xD6\x46\xDB\x58\x90\x42\xD6\xFD\xA9\x8C\xE8\xE7\x48\x9F\xAA\x2E\x19\xCC\xE4\xAB\xC1\x97\x8F\xB0\xAE\x47\x32\x5E\x46\x8A\x1A\x68\x1D\x13\xA4\x2D\x99\xD8\xB7\x41\x6C\x45\x93\xF4\x93\x1D\x99\x40\x76\xC3\x1A\x22\x2E\xC3\x45\x56\xB7\x63\x48\x66\xBD\x11\x9C\x52\x8F\x78\xDF\xA6\xD6\x6F\x54\xB6\xBC\xB8\xDF\xA8\x94\x08\x29\x2B\x18\xA7\xF4\xBD\x5C\x22\x2B\xC5\xD6\x96\x8E\x56\x2F\x24\xF9\x9C\x5E\xE4\xD7\x0B\xD7\x8A\xF6\xFF\x76\xAD\x8C\x5A\x1D\x7F\x5D\x2D\xF8\x87\xD8\x69\xD6\x63\xC2\xD5\x10\x9D\x4F\x99\xF8\x6F\x75\x3D\xAE\x03\x8D\x39\x29\xE9\x45\x29\x2D\x58\x07\xFF\xFF\xD4\x00\x71\x6D\xA8\xF0\xE3\x8E\x2E\x3C\xEF\x9B\x44\x49\xED\x28\x9B\xA5\x14\xD3\x42\x3F\xE7\x92\x65\x1C\xA4\xA0\x50\x08\x4B\xC2\xDA\xBC\x3C\x49\xC6\x9D\x66\x47\xDB\xB5\xF4\x72\x96\x64\x15\x39\x42\x3B\x9A\x35\x93\x24\x57\x2B\xEB\x6C\x79\x4A\x27\xE5\x53\x59\x83\xCD\x6C\x90\xE0\x0B\x2B\xB8\xC9\xC6\xED\x48\x36\xAA\xCC\x91\x0B\x24\xA9\xB7\xCE\x9A\x76\x9C\x95\x4A\x99\xD9\x28\x6D\xB5\x29\xF6\xC0\x99\x66\xC4\x36\x49\x12\x65\xAE\xBC\x7A\xA9\x3D\x3C\xD7\xBE\xCD\xC6\x87\x6F\x3D\x47\x4B\x8B\xF5\xD2\xE7\x66\x25\x89\x96\xF7\xB9\x9E\x49\xD6\xC8\x17\x91\xFB\x48\x29\x38\xC5\x87\x67\x5C\x65\xA2\x9C\xE3\xC4\xE4\x7A\xA7\x64\xCD\xB1\x5D\x50\x3E\x05\xD4\x17\x48\x12\x4C\xCB\xF9\xED\x9B\xFE\xE8\x66\xDF\xA0\x95\xA1\x24\x71\x9A\x84\x17\x64\xCB\x1E\x43\x12\x47\x0B\x49\xE2\x68\xBE\x24\x8E\xD8\x44\x10\x41\x69\x5D\xE7\x93\xBB\x74\x16\x49\xEB\xCD\x7C\xE6\x95\xF2\xF9\x68\x63\xA7\xBC\x88\x05\x87\xA0\x21\x38\x04\x2C\x38\xA4\x2D\x60\x62\x2D\x6D\x23\x19\xD6\xD1\x68\xDB\x6B\x6B\x0B\xAD\x4F\x46\xE8\x8A\xB7\x4E\x47\x9F\xB4\x32\xC8\xD9\x81\xDE\x0F\xFA\x6F\x2F\x42\x60\x1A\xF7\x3C\x2F\x23\xE6\x7B\xEC\xDF\xB0\x5E\x2E\xA1\xB4\x12\x89\x89\xFF\x27\x7E\x90\x9D\x34\xC9\x80\x7F\x4D\x52\xAE\x21\x14\xF8\xA7\x2A\x30\x31\x4E\x55\xB9\x58\x32\x86\x29\x77\x14\xAD\x4A\x83\x50\xCB\x9A\xB6\x4E\xC7\x8F\x51\x41\x10\xF9\xB3\x12\x5E\x32\x81\x54\x64\xC7\x6C\xF4\x67\x25\xD1\xA5\x6D\x53\x94\x17\xC5\xF4\x96\x76\x3C\x2A\x0D\xDA\xE6\x64\x2D\x9D\x8B\xCF\xB5\xEF\x90\xF7\x8B\x49\x51\x69\x94\x27\x41\x39\x0E\x4D\x65\x50\x8D\xAC\xA9\x59\x52\xDD\x5D\x31\x6B\xD9\x51\x36\x0E\xD3\x3C\x0E\xA3\x3C\x0E\x79\x10\x25\xB6\x66\xA3\x76\x7C\x6B\xD6\x1F\x91\xE6\x31\x47\xA4\x6E\x9B\x72\x44\xFA\xD6\x3A\xED\xA8\x58\x0E\x2D\x39\x55\xE4\x74\xB6\x7A\x66\xD1\x3A\xE6\xCB\xA6\x51\x6E\xB5\x6C\x54\xD4\x07\x51\xDB\x59\x53\x8E\xC3\xA8\x6C\x58\x39\x0E\x63\x72\x1C\xEA\x8D\x71\x68\x2D\x4F\x69\x3C\x0E\x4D\x1E\x87\x31\x31\x0E\x4D\x1E\x87\xA6\x1D\xE3\x71\x28\x39\xEB\x81\xA1\x16\x63\x76\x59\x4C\x69\xB9\x8A\xCF\xF5\x3C\xCF\x1B\x73\x8B\x1D\xA1\xC1\x46\x06\xF6\x82\x83\x8D\x55\x29\x88\x40\x49\x56\xD0\x16\x93\x83\x2D\x26\x45\xA9\x82\x03\x8D\xCD\xF4\xD1\xF1\x89\x61\x5B\xFD\x64\x20\x9A\xAD\x6A\x41\x4E\xA7\xAE\xFF\xBB\x84\x66\x0D\xE3\xA6\xA6\xF3\x30\x5F\x3C\x5B\x90\xAF\x1A\xEF\x22\x7C\x6B\xD1\x06\x5B\x8B\x46\x9F\x5B\x8B\xC5\x2C\x2A\xE3\x8A\x1A\xF7\x34\x52\x95\x8D\x62\x87\xC2\xEA\x13\x74\x52\xEE\xA1\x2F\x4B\x69\xB6\x4E\xB7\xF2\xF6\x7B\x3B\xE8\xC2\x26\x89\xD0\x99\x7C\x9D\x21\xE2\x40\x16\x1D\x3A\xF2\x6F\x8E\x8A\x6F\xED\x48\x65\x6B\xBC\x0B\x17\xC5\xCB\x06\x0E\xBC\x51\x6E\xCE\x73\x52\x74\x25\x85\x62\xEF\x99\xD4\x9A\x87\x3C\xDD\x41\xD1\x94\x7E\xF5\x98\x3F\x57\xC4\x42\x73\x85\xD4\x73\xA7\xF1\x9C\x6F\xB2\x0A\x18\x93\xE6\x02\x4D\xF4\xC8\xB4\x94\x40\x24\x8D\x98\x72\x13\xC1\x33\x64\x54\xCC\x90\x6C\x76\x92\x7B\x90\x69\xEB\xFD\xB9\xB8\xA3\xCB\xB3\x67\x6C\xD6\x7F\x87\x76\x44\xFC\x6E\x3A\xDA\x90\x58\x4F\x6D\xF4\xB5\xA5\x99\x47\xEB\xD3\xC4\x26\x7D\x9A\x66\x50\x9F\x26\x49\x54\x90\x86\xE3\x5C\x22\x9C\x13\xBD\x4E\x07\x54\x71\xA9\x65\x4A\x6C\x1A\xFC\x63\x11\xB4\xB5\x62\x36\xC2\x36\xD8\x4D\x5F\xB2\xCB\x97\xEF\x17\x1D\x27\xD6\xAE\x8D\x93\x35\x1D\x53\x35\x1D\xF3\x67\xDF\x98\x18\xCF\xB1\xBF\x5C\xA3\xA3\xCD\xBE\x51\x6A\x43\x59\xCB\xF4\x2D\x27\x92\x60\xB5\x46\x1F\xF6\xFB\x4B\xB0\x66\x35\x59\xB3\x64\x37\x38\xD6\x54\xB3\xDA\x7F\xA7\x66\xB1\x51\xB3\x7E\x04\x62\xAF\xDB\x5C\xB3\xAC\x8B\x2D\x16\xCE\x8C\x7F\x6A\x40\xB5\xAA\x35\x4C\xE1\x1F\xB3\x3A\x59\xF4\x25\xD2\x9F\x43\x47\x1B\x72\xD0\x46\x95\x29\x6C\xCA\x54\xC0\xB2\xB9\x9F\x93\xDC\x68\x03\x87\x25\xAB\x7D\x0E\x2B\x38\x49\xD2\xF0\x4B\xE3\xA6\x71\x9B\x42\xAE\xCC\x52\x68\x93\xEA\x8D\xAF\x69\x88\xCA\xE3\x2B\xE0\xB8\x64\x02\xD9\xBE\x46\xC9\x84\xD2\xA5\xB2\xBD\xB8\xBA\x4D\x6E\x93\x6C\x01\x4B\xC4\xC4\x49\xC3\x2A\xAB\x8B\x8B\x08\xA4\x9C\x23\xC9\xF8\x2A\xD1\x2A\xA8\xE6\x48\x31\x08\xE9\x1C\x95\xE7\xC8\x88\x98\x23\x91\x85\x5B\xD0\x9F\x23\x35\x96\x38\xD5\x48\xD5\x7F\x43\x42\xB6\x31\x47\x66\x63\x76\x7C\x32\x19\xB7\x8E\xB1\x18\x88\x3F\x59\x26\xC5\x14\xA9\xA9\x29\xD2\x52\x32\xA7\xA4\x0F\x96\xA5\x0F\xB5\xBF\x34\x45\x5A\x8A\xBB\x92\x10\x24\x26\xC9\x4A\x50\x2E\x7C\x13\x38\xE0\x8C\x8C\x7C\x87\xAF\x85\xF6\xF3\x71\x2E\xEF\xD3\x6A\xB4\xCC\xAE\xE7\x03\x3D\xDE\x7A\xF9\x28\x23\xDC\xFA\xD6\xFF\xD2\xD0\x14\x7B\x44\xF4\x67\x57\x31\x8D\x3A\xE9\xA2\x14\x15\xA2\x13\x92\x24\x1F\x94\x18\x2C\xFC\xC2\x5A\x1E\xD2\xC4\xB6\x8B\xA9\x19\x56\xA9\xEF\x61\xF5\x17\x24\xDA\x4A\xBD\xA4\x68\x47\x26\x6B\xE9\x73\x1A\xD2\xB7\x46\xCE\x60\xE1\x7A\x39\x54\x23\x62\xA8\x12\xD1\x4B\x9B\x0F\xA2\xDB\x98\x05\x69\x34\xC8\xA9\xC6\x50\x6D\x12\x98\x25\x7A\x01\x69\x4A\x34\x17\xA8\x33\x29\x43\x8A\xCC\x53\xDD\x8E\x63\xF1\x59\x64\xF5\xDB\x68\x6B\x4B\x53\xE0\x7C\xF8\xE1\xB0\xEA\xA9\x18\x9F\x2D\x11\x4B\x41\xD4\xA2\xAC\xB0\xF9\x54\x58\xAB\x15\x6C\x50\xE7\x91\x8A\xB1\x73\xB2\x7F\xB1\xD8\x5A\xA8\xA8\xA3\x49\xD4\x72\x93\x2C\x5B\xB3\x0C\xB0\x35\xEB\x13\x1A\xEA\xEA\xC0\x74\x17\xF8\x27\xA6\x0E\x5D\xC5\x67\x35\x12\x7C\x18\x71\x81\x7F\x6C\x7A\x56\xF0\xD8\x74\x8A\xA0\xA4\x6E\x9D\x3D\x53\x5B\x64\xA3\xF3\x2C\x9F\x9F\xB6\x13\x7D\xD5\xAE\x0E\x4E\xB7\x6C\x26\xF6\xBC\x2F\x89\x78\x31\x9B\x28\x26\x59\x26\xE6\x17\x6B\x74\x2C\xAB\x52\xF2\x8F\x65\x59\x6C\x41\xA5\x64\xA8\x94\x6C\x1B\x9D\x57\x39\x25\x75\x0B\x4C\x26\x22\x88\x4B\x8E\x57\xF6\x21\x4E\xB0\x8D\x6E\xEE\xD0\xF8\xD3\xE5\x0D\x6E\x1B\xD3\x9F\xD9\x2C\x27\x72\xA7\xBD\x28\x3F\x6C\x74\x76\x81\x3A\xCA\xD5\xAC\x0F\x26\x30\x1A\xE0\x7C\x1B\x8A\xAB\x40\x57\xC8\x69\x4B\x31\x3F\xE5\xDB\xCD\x8E\xE5\x4C\xBE\x78\x1B\xA5\x36\xCD\xC5\x8A\xB9\xB8\xF3\x10\x96\x73\x0D\x0B\x0E\x76\xC2\x79\x08\x4B\xFD\xCE\xF8\x52\x91\xB6\xE5\xAC\x58\xD3\x17\x4B\x1B\x9F\x13\xE1\xB3\x51\x26\x3C\x4C\xEE\x1A\x64\xEF\x3D\x3A\x53\x1B\xAF\x3A\x71\xDC\x36\x79\x6E\x16\x71\x39\xC3\x7C\xE9\xD2\x8E\xD9\x91\x62\xBF\x83\x45\xE7\x59\x92\xFA\x8B\x33\x73\xD2\x27\xE0\x88\xE3\x74\x33\x29\xA1\x24\x35\x68\x44\x58\x45\x78\x87\xE5\x5F\xAD\x16\x73\x4B\x9C\x4A\x97\x94\x97\x5B\x63\x5C\xCA\xFE\x5C\xB4\x29\xBD\xE8\x04\x7F\x20\x46\x49\x7D\x65\xFA\xCB\x9B\x65\x19\x88\x6A\x1A\xEF\x17\x82\x54\xBC\x44\xA9\x1A\x4C\x5F\xDD\xB7\x6D\xFA\xD5\x90\x23\x83\x91\xE3\xA4\x4C\xB6\xA0\x33\x49\x33\x39\x4F\x3F\x5C\x29\x34\xC1\xCC\xD4\xC6\xB3\xB4\x1F\x69\xF6\x69\xD7\xC6\x15\xA9\xA6\x29\xE2\x48\x23\xE2\x48\x30\x62\x45\x4D\x2A\x1B\x3F\x51\x9A\x66\xFC\x81\x60\xE7\xF8\xCE\x80\xCD\xCA\x64\x79\x2F\x80\xB3\xA0\xCD\x26\x8D\xC4\xC4\x82\xA5\x39\x71\x9C\xBC\x11\x2D\x65\x67\xF9\xB2\x6C\xB4\xE1\xCB\x70\xC6\x14\xA5\xD7\xD8\x71\xEF\x23\xAF\x39\x8D\x16\x8D\x0E\x18\x43\x24\x7B\xC3\x34\xE6\x38\x22\x79\xA5\xF5\xDB\xC8\x34\x34\x73\x5A\x8B\xA6\x01\x99\x6A\x8E\x3A\x69\x65\xCD\x31\x16\x90\x68\xE2\x5B\xDF\xBE\x77\x79\x2E\x22\x35\x84\xCB\xF3\x0F\xBA\x38\xE9\xC4\x8A\x34\x65\x11\x2F\x61\x82\xAF\x2E\x52\xE7\x9B\xE2\x82\x38\xB0\x02\x87\x03\x52\xC3\xBC\x2D\x2A\xC1\xD1\xEA\x39\x8D\x17\x99\xC6\xE1\x41\x8C\x6B\x2A\xCA\x12\x96\xD4\x1F\xA8\x8C\xE9\x86\x32\x34\xA9\x25\x97\xAF\x1E\x48\xAF\x14\x9E\xEF\x01\x71\x4F\x23\x3A\x9F\xB4\x74\xAB\xBD\x07\x0D\x96\x51\x3B\x55\xBB\x36\x9E\x5B\xF7\xBF\xD1\xAF\xB4\x46\xF3\x6B\xA3\xF4\x2B\xB3\xA9\x5F\xC5\x8E\xEE\x57\xAC\xD3\x4A\xB6\xA5\xC8\xA8\x2C\x87\x2C\x6E\xF2\xB8\x3B\x65\xEB\x53\x03\xB0\x50\xAD\xDA\xE4\x49\x53\xA3\x4A\xC2\xB6\x31\x88\x73\xC1\x11\x1B\xE3\xF5\xEF\xE8\x41\x1B\xE1\xC2\x06\xC6\x6D\xEC\xAF\x18\xB7\xF3\x99\xE5\x77\xD4\x68\x88\x34\x0F\xB3\x48\x70\x98\x45\x68\x46\xA2\x9E\xC0\xEB\x9B\xE9\x0B\x52\x73\x37\x20\x71\xAF\x88\x54\xFF\x2E\x82\x92\xCA\x6F\xBE\x2A\xD6\x90\x0E\xD0\xAC\x3F\x22\x9A\xC3\x36\x38\xFF\xF0\x67\x9E\x7F\x03\x93\x21\x36\x4D\x86\x1A\x8B\x41\xE9\x3C\x19\x1A\x6F\x9B\x3D\xE1\x43\x52\x63\x81\xC9\xB0\xA1\xD4\x35\x96\xD5\x78\x32\x34\x24\x2F\x7B\x7C\x63\x45\xD7\x04\x21\x1B\xAC\x57\x6D\x02\xE9\xBE\x09\x54\x6A\xE4\x2F\x54\xAA\x2E\x36\x59\x68\xEB\x7C\xDC\xE4\x18\xC5\xF4\xA3\xBC\x1A\x80\xF5\x75\x0D\x8D\x61\x07\xB3\x18\x92\x7A\x20\x1E\xCC\x70\xB1\xE5\x78\xD3\x34\x4D\x00\x5D\xD7\x0D\xCD\x94\x0F\x9A\x86\x94\x55\xD2\xE6\xA7\x22\xD2\x88\x15\xB7\x96\xE6\x68\x8B\x27\xD0\x7D\xA6\x0E\x20\xF2\x79\x8C\x25\x85\xC6\x38\x0D\x9D\xCE\xC0\x73\xF2\x86\x14\xFA\xAA\x1B\x0C\x25\x5A\x66\x50\x38\xD2\x06\xA0\xAD\xE0\x1B\xC2\x4D\xD7\x81\x49\x56\xDE\x81\x85\x92\xF0\x5A\xB8\x5E\x0A\x22\x90\xB4\x7D\x18\x3F\x9A\xA8\x49\x07\xA0\x03\x0B\x8F\xF5\x6B\x8E\xF2\x8D\x16\x67\x9C\xEB\x80\x14\x0E\x83\x13\x5F\x91\x02\xE7\x02\x1B\x36\x38\xB8\x9E\x68\x52\x2B\x01\x36\x32\x4F\x27\xF8\xFA\x9B\x24\xB1\xC0\x48\x3C\x23\xC7\xF2\xED\x87\x6F\xF5\xEF\x69\x31\xB5\x12\x59\x28\x15\x3C\x1B\xD9\x98\x6D\x32\x21\xDD\x90\x10\x53\x34\x27\x99\x59\x66\x1D\x19\x92\xF7\xF4\x00\xD1\x00\x71\x8A\x5B\x10\xD9\x74\x4E\x9F\xD2\xE8\x3A\x3A\x19\xC7\x22\x29\x4B\xB1\xFE\x90\x44\x7F\x7C\x31\x9D\xC6\xF0\x47\x94\xAE\x49\xB2\x40\x87\x12\xB5\x89\x92\x64\x3F\xED\xCC\x12\x0B\x53\x86\xF4\x9A\xB5\x98\x45\x15\x95\x11\x3A\xC8\x27\x1D\xB6\x65\x9B\xB9\xE4\x42\x92\xB5\xF2\x65\xFC\x69\xC2\x30\x64\x00\x07\x69\xAB\x27\xD6\x24\xE9\x23\xAA\x22\x42\x52\x72\x36\x0D\xCD\xC5\x29\x48\x46\x2C\x8B\xD8\x5F\x59\x93\x6D\x05\xA5\x78\x19\x6B\x69\x78\xA7\x82\xB4\xD8\xA9\x05\x29\x48\xEA\x16\x2B\x34\x30\xA4\xEE\x46\x19\x99\xB9\x22\xA5\xB4\x7A\xD0\x45\x3E\xBF\x82\xF8\x0E\x94\x99\xDE\x4A\x82\xA2\x7E\x9C\xB2\x45\xE4\xED\x81\x26\x33\x14\xBA\x62\x80\x99\x24\x26\xCB\x82\x22\x66\xE3\xF7\x37\x45\xAD\x24\xE9\xF8\xC1\x20\xAD\x72\x2C\xE4\x1A\xA5\xFE\x04\x8E\xB6\xA0\x48\xA9\x34\x7D\xC6\x9A\x3F\x93\x81\xCF\x2C\xAF\x85\xD9\xA8\x9D\x64\x9B\x28\x26\x6F\x72\xE3\x7C\xC7\x85\xA6\x10\xDA\x69\x35\x25\x41\xA1\xA8\x28\xA4\x3F\x3A\x2E\x82\xA8\x4D\x10\x75\x2B\x92\xFB\x49\x1F\x20\x2D\x55\x96\xEC\x81\x76\xAC\x9F\xC4\x5F\xC2\x62\xC6\xBE\xBE\x70\x41\x16\x58\x62\x06\x8A\x31\xA3\x6F\x98\x4C\x87\x48\x1D\x2F\x7A\x93\x8E\x17\x3A\x1D\xB2\xEC\xE8\xF9\x52\xF8\x28\x67\xCE\x4F\x19\xA4\x70\x9D\x16\x77\xC5\xEC\xA4\x8A\x8A\x2B\x4D\x9B\xEA\xEA\x4F\x8C\x4A\xB1\x90\x2C\x08\x34\x3A\x0F\x9F\x83\x24\xA5\xD8\x35\x6B\x63\x17\xFD\x22\x6E\x29\x2E\xA9\x1C\x0B\xDC\x33\xFD\xBE\x63\x93\x2A\x8A\xC6\xD0\x7A\xBB\x7A\x50\x63\x31\xAA\x96\x36\x5F\x94\x3A\xA9\x4A\x63\x91\xB8\x38\x55\x00\x23\xA2\x62\xEB\x49\xAA\xE4\x93\xA2\x03\xE9\x2C\xB3\x10\x63\x2B\x32\x52\x90\x3A\xCA\xDD\x22\x25\x15\x0D\x8A\x2E\x91\xB4\xA3\xC5\x1C\xDA\x86\x70\x94\x01\x71\x64\x1D\xE2\xBE\x28\xB5\x6C\xF9\x54\x93\x28\xB5\x49\xF6\x28\x7D\x51\x6A\xD1\x31\x7C\x51\xEA\x86\xE9\xF1\xE0\xD8\x10\x3D\x55\x4C\x2F\x62\x47\x26\xEB\x38\x30\xB0\xCD\xC6\xC0\xA6\x9D\x51\x78\x3C\xC7\xE5\x4C\x63\xDA\x71\x1E\xD1\x56\x68\x44\x6B\xFF\xCD\x11\xAD\xAB\x11\x1D\xFF\x2B\x46\x34\xB5\xA0\xB9\x38\x38\x37\xBE\x7D\x03\x1E\xBB\x09\x53\x8D\x26\x4C\x1D\xD5\x84\xC8\x4D\x98\xFA\x4B\x4D\xD8\x12\x68\xC2\x14\xB5\x1D\x37\x61\x52\x36\x99\x68\x3D\x12\x89\x6F\x6E\xC2\x96\x70\x13\x42\x53\x13\xA6\xFE\xDA\x26\xB4\x02\x2B\xC3\x86\x62\x8B\x06\xA8\x89\x52\x20\x6D\xD9\xD3\xB6\x35\x86\xBE\xDA\xB5\xF4\x4C\xF5\x65\xDD\xA6\xF9\x4C\x09\x11\x6A\x63\xD1\xD1\x97\x28\x29\x09\xED\xE8\x55\x4D\x67\x46\xFA\x38\x52\xD8\x62\x1B\x75\xE7\x8E\xBD\x3B\x61\xFD\xFC\x86\x5E\x2E\x15\xC1\xB1\x6E\xE8\xFC\x7D\x43\xD2\xD9\x08\x0B\x1C\xA6\x48\x6D\x8C\x61\x1F\x9D\xAE\x8D\x1C\x90\xE4\x40\x47\x5B\x6A\xA5\xB0\x01\x65\x81\x8E\x53\xE5\x59\x90\x24\xC6\xD3\x0B\x8F\x4E\x11\xD7\xCF\xE7\x33\x5F\xC3\xB2\x0D\xEB\xB9\x28\xDD\xF0\x15\x5D\x4D\x63\xC1\x3F\x9D\x75\xE9\x92\xF4\x8B\x1E\xBE\x63\x24\x30\x9A\x03\x0E\xA4\xDF\x47\x2B\x69\xA4\x89\xE6\xD1\xB9\x26\x5A\x0C\x40\x4D\x17\x04\xB8\xBA\xB3\xEC\x0C\xF3\x69\x3E\x21\x50\x21\xB4\xA0\x0F\xE4\x99\x4C\x20\xE8\x96\x81\x7F\xA9\x8B\x22\x4D\x2C\x48\x81\xF3\xEF\x0F\x04\x0F\x3C\xDA\xB5\xB4\xA4\x44\x67\x6A\xE9\xA3\x85\x4E\x45\xA8\x77\x17\x5B\xC6\x1A\x91\x58\x4C\x64\x27\x66\x18\x31\x7A\x8C\x98\x48\x65\x78\xA6\x96\x4E\x46\xD4\x1B\x25\x2B\x4D\x7A\x48\xE5\x0A\xC4\xD1\x8F\x8C\x96\x36\xDB\x75\x23\x1B\x48\xC3\x0B\x48\xBD\x0E\xD3\xC2\x69\x55\x42\x19\x11\x69\x2E\x68\xD7\xA0\x28\x2F\x34\x8F\x5A\x0E\x79\xEE\xCE\xBB\x4E\x99\x2D\xFD\x98\x49\x5B\xAC\xDA\x81\x12\xD6\x8E\x15\xA7\x9F\x3F\xA9\xA0\x39\x4D\x79\x93\x95\x2D\x95\x80\x70\x21\xFF\xBA\x12\x62\xA0\x84\x7C\x59\xF8\xA8\xCE\x91\x33\xA9\x2B\x8A\x68\x7E\xBF\xB5\xD9\xB4\x4F\x3A\x1B\x75\x0C\x92\x15\x61\x12\x48\xF3\xEF\x0A\xFE\xFE\xC1\xA6\x6A\xE0\xBC\x2E\x4F\xC1\x31\x3D\x11\x6B\x82\x25\x81\xC8\xFC\x96\x5A\xE1\x4C\x22\x84\xE8\xAC\x8C\x8A\x18\x65\xA2\xC6\x7C\xBB\x22\xFA\x16\xFC\xA2\xAC\x91\x24\x22\x08\x1E\x51\x44\xEB\x6D\x93\x4F\xF3\x09\x1D\x48\x1E\xAD\xEF\x77\xA1\x64\x9C\xD8\x46\x91\x8D\x31\xFC\xBE\xD9\x8E\x0C\xFF\x70\xAC\x88\x23\x32\xEA\x7E\x5B\x13\xF3\xF6\x9D\x3C\xB4\x9A\x6E\x48\x39\x9A\x03\x3C\xA1\x9F\x64\x83\xE8\xF2\x40\x67\xDD\x74\xE3\x5F\xAE\x2B\x1A\xED\x79\xE5\xE9\xD5\xE8\xA4\xB5\x3A\xEE\x53\x3B\x0B\x9D\x55\xBA\xA8\x9F\xEF\x8B\xFA\x77\xF6\x42\xC3\x89\x6E\x39\x9A\xB4\x9F\x41\x44\x04\x88\x44\x22\x86\x89\xF2\xD1\xC5\x7C\xCB\xEB\xCB\xFC\x54\xF4\x6D\x68\xE4\xF8\x42\xBA\xC6\x40\xE7\x2E\xFE\x52\x9A\x60\x92\xDD\x4E\xF0\x4A\x1B\x0F\x2D\xA4\x2A\x80\x5C\x48\xCD\xC6\x42\x6A\xAA\x88\x9A\x16\xD2\xB8\x25\xA8\x30\x96\x57\x6B\xA7\x63\x6D\x93\xD7\x53\x33\x98\xAE\x9D\xB2\x93\x82\x34\x8E\x5A\x52\xA9\x4E\x90\x18\xB2\x31\xFD\xB9\xCD\x2C\xE9\x2B\x7E\xE1\xDB\x9C\xD2\xEE\x80\x4F\x85\xD1\x15\x44\xFF\xEC\x9D\xF4\x5E\xD0\xAA\xFB\x30\xB5\x1E\xDA\x86\x2A\xC7\xA8\x71\xE4\x62\x7E\x2C\xD2\xE4\xC2\x28\xAB\x06\x79\x08\x45\xD3\xD8\x0B\xD9\xA6\x1D\xF3\x6F\x9D\x1A\xD3\x50\xC5\xE7\x53\x02\x66\x80\x0C\x10\x44\xD5\x4C\xBE\x8A\x4C\xD4\xE7\x51\x99\x32\x8F\x9D\x05\xF3\x18\x59\x50\x09\x39\x56\x30\x21\x83\x0F\x54\xE8\x16\xA2\xE9\x9F\x2E\x8B\x3E\xFF\x60\xB3\x2D\x1A\xE4\x7B\x89\x6A\x5D\x36\x68\x95\x8E\x32\x59\x7A\x8C\x94\x8E\x22\x6E\xA2\x41\xE2\x46\xDE\x67\xC6\x20\x7D\x1A\xB5\xA9\xBC\x51\x9F\x3E\x95\x08\x62\xD1\x44\x25\x7D\x8A\x41\xE2\x06\x7D\xE2\x06\x1B\xF4\x29\xFA\xF4\x29\x16\x73\x24\xB5\x57\xCC\x59\x92\x98\x49\x88\x7A\xB3\x1A\xC4\x0D\x0A\xE2\x06\x8F\xA6\x4F\x83\xC4\x0D\x32\x7D\x6A\x30\x83\x41\x12\x37\x52\x18\xD6\x9F\xD5\x14\x71\x23\x46\x90\xA8\x8E\xA8\x35\x20\x75\x90\x8A\x2A\x1D\x27\xAA\x74\xDB\xD6\x51\xEE\x5B\x13\x23\x91\x76\xF1\x0B\xF8\xEA\x95\xDF\x50\x48\x34\x00\xD7\x5D\xA0\xFD\x92\x4D\x68\x5B\xAE\xF5\xD6\x8B\x06\x1A\xC3\x8D\x4B\x1C\xBE\x4E\x1C\x1B\x25\x89\xE5\x13\x05\xE9\x5B\x36\x37\x05\xB5\xF5\xF4\x7B\x25\x06\x1B\xD2\xFF\x74\x5C\x63\x70\xFB\xC8\xEB\x6E\xD9\x08\x2F\x0C\x76\xC4\xF6\x6F\x31\x4B\xAD\x2B\x11\x1B\x69\x66\xE7\xBD\x1D\x5D\x3C\xA2\x9B\xA5\xCD\x29\x48\xDD\x17\xB6\x9E\xBE\x8D\x67\x4D\x5B\x53\x46\x2E\xF9\x22\xA0\xAD\x55\x1B\xAC\x2E\x47\xB3\x75\x55\xCC\xC6\x2D\x41\x79\x91\x17\x9A\x8F\x19\x1C\x5C\xDE\x30\x0F\xD7\x54\x0A\x3F\xFB\x86\xC8\xE4\x31\xB3\xEF\x37\xA0\x68\x62\x3F\x83\xD1\x26\x19\x2E\x91\x77\xB1\x81\x8C\x4A\x49\x4A\xBF\x70\xB7\x92\x67\x53\x0A\x1B\xF3\xF9\x2D\x3A\xC3\x35\x5B\x2B\x92\xD4\xB7\xF4\x99\x35\xFC\xA2\x45\x54\xD1\xB2\x11\x27\x26\xFB\x3F\xDA\x11\x87\x6E\xB6\x46\x94\x36\x31\x32\xEC\x88\x7C\x19\x2A\xD2\x48\xA7\x21\x81\x6D\xD9\x51\x12\xAC\x20\xE5\x4F\x91\x40\xD5\x61\x23\x7E\x4B\x54\xAE\x7F\x79\xE8\xED\x32\x00\x8D\x0C\xB0\x9C\x27\x67\x00\x58\xF5\x4A\x84\x79\x52\x92\xB3\xA2\x78\x52\xC4\x74\x1A\xED\x87\x17\x11\x0D\x29\x31\x3A\x4E\x1D\x22\x22\x5F\x75\xD1\xE8\x96\x96\x4E\xE3\x56\x67\x06\x67\xF0\x30\x8F\xAE\xE7\x91\x76\x0D\x19\x48\xF9\xD7\x96\xD0\x21\xC0\x9E\x86\x19\x23\x5D\x2C\x72\x74\x5D\xDD\x79\x08\xF9\x38\xD1\x50\x64\x01\x69\x99\x69\x8E\x9E\x54\x20\x69\xAC\x7C\xBE\xE9\xF2\x0D\x32\x9D\xA0\x91\x14\x9A\x73\x73\xD1\xD1\x36\x38\xC3\x35\xD1\x85\x67\x6A\x31\x29\x64\xA3\xF4\xB0\x34\xF4\xF1\x19\x00\x54\x01\xBA\xF8\x83\xD6\x1D\x49\x4C\x8D\xAE\x60\x4D\xF3\xF5\x9A\xA6\x73\x9A\xF3\x10\x09\x91\x88\xDC\x3C\xB7\x89\x15\x16\x28\xF5\x27\x24\x4C\xA5\xA9\xE2\xAB\x5A\xA3\x7A\x48\x4A\x1D\x7E\x9C\x93\x08\x5F\x45\xD3\xEC\x48\x3F\x71\xCF\x8B\xCE\xA6\x01\x31\x59\x99\x6C\x0C\xCE\xB0\xB5\x7E\x96\x44\x23\x99\x82\xF4\x7D\x7C\x21\x1C\xA4\x9E\x10\x92\x51\x64\x56\x2A\x15\xDA\x6C\x5C\x11\xA4\x63\x7C\x1C\x72\xD0\x8E\x4E\x68\x5C\x24\xD2\xD8\xC0\xAA\xE9\x8B\xE9\x98\x62\xE7\x2B\x52\x21\x39\xF3\xA8\x73\xF3\xFA\x62\xCE\xA2\x6B\x8C\x09\xA5\x26\x23\xE1\xDC\x5C\xCC\x25\x45\x45\x62\xD1\x89\xAE\x27\x71\x22\x8B\xAD\xA1\x81\x1D\xED\xCF\x1A\xE2\x8B\x84\xAC\xB4\x66\x4D\x0F\x72\xD7\x48\x3A\x30\x4C\xDB\x58\x21\x88\x81\xAC\x61\xA7\xF8\x12\xB6\x9D\x60\x61\xF2\x36\xEE\x3B\x8D\xD0\x22\x98\xBA\x2D\xC1\xDF\x31\x56\xE3\xEF\x7F\x27\xC5\x77\xB2\xF1\x9D\xE6\xBB\x28\xFE\xF7\xB8\xA2\xDA\x04\xF0\x37\x99\x01\x18\xDF\xF8\x6E\x13\xDF\x6D\x0D\xBD\x14\x36\xE7\xD6\x16\x34\xAC\x94\xDB\x00\x3B\xCE\xDA\xF2\x65\xDD\x5A\x3C\x7E\xE8\xA4\x27\x28\x9F\xC0\xA7\xE0\x86\xA5\xD4\x22\x32\x82\xEE\x6A\xC6\x78\x33\x23\x45\xFF\x45\x93\xF1\x9D\x46\x3E\x20\x27\x2E\x22\xF0\x19\xF9\x0A\xEA\x11\x26\xD9\x02\xB6\xD1\xD9\x9F\x51\xDA\xD4\xE8\x9A\x95\x13\x2B\xA6\xBF\x22\x86\xA4\xD8\x38\x8D\xFA\xEB\x57\x15\x33\xE6\x6F\x9D\xEF\x6F\x9D\xAF\xD1\xF9\x48\xC7\x23\x31\x8D\x39\x25\x6A\x10\x6D\x42\xCE\xBF\x19\xC8\x9A\x09\xB4\x7A\x2E\xAA\x08\x03\x5E\xAD\xE4\xA2\x2F\x9A\x36\x4D\x9F\x51\x12\xBB\x43\x16\xB9\x09\x76\x41\x45\x41\x3B\x46\x31\xFD\xB5\xCD\x92\x07\xDF\x98\x8E\x17\xB2\xE4\x61\x52\x27\x39\x29\xAE\xCA\x9B\x8B\x39\x2D\xFD\xD8\x66\x56\x85\x27\x3D\x25\xB5\xA0\x68\x8A\x28\x54\x54\x1A\xB9\x4A\x9E\x23\x85\x2A\x48\x4D\x03\x92\xB0\x05\x33\x8A\xF8\x16\x7D\xBB\x36\x8E\xF2\x24\xC9\x6C\xB1\xCE\x89\x8C\x46\x65\xCE\x59\x72\x23\xCA\x72\xAC\x68\x47\xC5\x6B\x78\x59\x48\x5B\xCD\x97\x17\x03\x0A\x0E\xAD\xBB\x0D\xB4\x86\xB5\x39\x4D\xA5\x92\x63\x88\xEF\x7A\xF2\x40\x32\x96\xB1\xF5\x12\xDB\x68\x1E\x48\x91\xA2\xD4\x93\xA6\xF7\xF3\x9A\x63\x92\x4C\x55\x70\x20\x91\x18\xA3\x69\x6B\xC5\x9C\xBC\x56\xA9\xDB\x11\x35\x90\xE8\xB2\xA7\xC9\xED\x66\x4E\x90\x76\xAB\x89\x1A\x32\x29\x76\xD9\x6A\x51\x3B\x22\xB6\x56\x22\x95\x6C\x9C\xB8\x28\x62\x20\x25\x58\x42\x4E\x0D\xA4\xB8\xA8\x78\xAB\x31\x90\xCC\x14\xD8\x09\xDA\xBE\x3B\x60\x9B\xFD\x59\x4D\x7C\xD1\x1D\x20\x5D\xD9\x74\xE3\x6B\xDA\x72\xFF\xA6\xF1\xE5\x2B\x6D\x45\x36\x69\x47\xB2\x9A\x9D\x64\xF1\x55\x3B\x2E\xBA\xAF\xA5\x06\x52\x23\xB4\x08\x46\x51\xF8\xDF\xD2\x8A\x85\xFF\x9D\x2C\x92\xC2\x1B\xFF\x3B\x5D\x24\xC1\x20\xFF\xBB\xE9\x44\xCA\x36\x45\x5F\xD6\x1B\x03\xC9\x14\x03\x49\x6F\x0C\x24\x5D\x0C\x24\x91\x5B\x5B\x8C\x77\x4D\x0D\xA4\x18\x0F\x24\x59\xB7\x89\x00\x15\x14\x6C\x73\xC9\xCB\xB4\x58\x2B\x1D\x84\xC7\x52\x83\xEA\xBA\x4F\xF5\x07\xD5\x01\x78\x26\x5D\xE0\xAB\xA0\x6A\x9E\x6D\xC9\x58\x1D\x5D\x06\xF0\x3B\x86\xA6\x66\x58\x5D\xCD\xB0\x11\x69\xC2\xAB\x3F\x47\x57\x42\x03\xBD\x23\xE6\xF3\x55\xE9\x42\x2B\xCD\xB0\x52\x3A\x40\x84\xE3\x8E\x11\x99\x90\x8B\x34\x3A\x46\x24\x38\xC3\x46\xC5\x0C\x4B\x6A\x25\x1C\xBD\x48\x9D\x42\xA7\xFE\x48\xC9\x12\xD3\x46\x67\x81\x27\xD5\x3F\x12\x81\xFE\x41\x77\x24\x45\xCF\x30\x95\x2F\xBE\xD7\x20\xEB\x47\x52\xDB\x81\xB3\xA0\x9C\x65\x9B\xA2\x77\x58\x59\x53\xF6\x0E\x5D\xF4\x0E\xD3\x4E\xA8\xDE\xD1\x08\x2D\x82\x51\x14\xFE\x77\xAC\x48\x82\x12\xFE\x77\x52\x7C\x27\x1B\xDF\xE9\x22\xDD\xA0\xF5\xBF\xC7\x15\xD5\x05\x0B\xFE\x1E\x2F\xBE\xC7\x37\xBE\xDB\xC4\xB7\xDF\x3B\x22\xA2\x77\x44\xC8\xDE\xAB\xD8\x29\x87\x7A\x87\x29\x47\x5E\xA0\x77\x04\x29\x44\x7F\xA0\xEA\xCE\xCD\xC5\xE6\x0E\x62\x10\xBB\xDA\x68\x74\x90\xBF\x4D\x18\x7F\x9B\x30\xFE\x36\x61\xFC\x6D\xC2\xF8\xAB\x27\x8C\x1A\xEA\xC3\x6C\xCA\x95\xFB\x87\xD2\x24\xBE\x8C\x9D\x7E\x69\x16\x37\x07\x7C\x94\x0D\x74\x12\xC1\xEA\x30\x05\xA5\xAC\xF3\x51\x38\xDF\x18\xD2\xD8\x60\xAA\x4E\xF2\x3D\x62\x67\x1C\xC9\x6A\x96\x33\x2C\xE8\xAC\x7E\xD1\x53\x16\x4C\x60\xAD\x15\x47\x10\x23\xC3\x36\x4E\x21\x7B\x17\x22\x22\x9D\x4D\x32\xDA\x1A\xD9\xA6\x95\x87\xD5\xCE\x08\xD6\x72\x11\x67\x27\x5F\xDE\x22\x2B\xBC\xEB\x73\x86\xF3\x2B\xBA\xA3\x6A\x38\x3B\x71\x71\x0A\x1D\x8D\xEE\x41\x9B\x75\x67\xE7\x98\xF5\x76\xA4\x9A\xD3\x9D\x11\x81\x27\xE5\xB3\x1D\xA0\x51\x1C\x66\x73\x1C\xCE\x9F\xC8\x02\xAC\x4E\xF7\xA0\xCD\xBA\x18\xD3\x75\x67\x64\xAF\xBE\xBE\x6A\x9B\x14\x41\xBA\x11\x81\x4E\x11\x20\x47\x20\x83\xBF\x41\xC1\x8D\x60\x70\x11\x92\x32\xE1\x79\xB0\xDE\x46\x11\x8B\xE7\xE9\x4B\x52\x68\x39\x6F\x79\x3A\x17\x4F\x0E\x0C\xB0\x0D\x6B\x37\x69\x31\x63\x8D\x13\x8E\x07\x6C\x48\x01\x67\x01\x38\xA8\x94\x8D\x8E\x91\x48\xE7\xA2\xF5\xCE\x08\x56\xA5\x1D\x02\xC7\xDC\xE0\xEC\x14\x5F\xE0\x68\xC2\xA7\x88\x9D\x54\x5A\x7B\xDE\xB7\x61\xC9\x04\x56\x62\x1D\x0E\xA9\xD1\x57\x72\x83\xB3\x97\x42\xAA\x78\xFC\x88\x1D\x5D\x2A\x04\x56\x18\x52\xD2\xE4\x8C\xDD\xE0\xEC\x1F\x35\x80\x1F\x5F\x23\x06\xC3\x5A\x8D\x38\xEC\x48\x8D\xC1\xEA\xDA\x2A\x88\x06\x06\xD1\xA4\x74\x7C\x46\x28\xAA\xC1\xBD\xA2\x89\x6C\x74\xF4\x06\x6A\xBF\x42\x2D\x64\x2E\xA5\x98\xDA\x2C\xEB\x81\xFB\x75\x0D\x9C\x91\x11\xC3\x7A\xF6\x3E\x7D\x13\xDC\x0C\x8B\xCB\xEB\x0A\xE5\x42\x7D\x23\x38\xE5\x8D\xE0\x56\xAB\xEE\x46\xA8\x0C\xBA\x5D\x03\x6E\xA9\x54\xE9\x9E\x65\x17\x4A\xA5\x7C\xAF\x5B\xB2\xD7\x0E\x15\x4A\xF5\x42\xD9\xEE\x76\xBB\xFB\xF2\x76\xA1\xDC\x93\xBF\x11\x4E\xE9\x9C\x71\x23\x9C\x72\x23\xF9\x2F\x0F\x0D\xAC\xCD\x57\xBB\xEA\x95\xAE\xB5\xEB\x66\xD9\x85\xF2\x06\xB7\x54\xE8\xB1\x19\x6B\x8F\xE6\xA5\xB2\x21\x5F\x5D\x57\xAA\xDC\x00\x8B\xF2\x6B\xAB\xF9\x1B\xE0\xDA\x1B\x60\x69\xA1\x36\x04\x17\x17\x36\xE4\x6B\x5D\x4E\xFF\x50\x75\x08\x96\x0F\xD5\xFA\x86\x60\x75\xBE\x34\xD4\x3B\x04\x4B\x0B\x03\x6B\x87\x60\xA5\x3B\xD4\x35\xAF\x50\xEE\x5A\xE4\x0E\xC1\xD2\x21\x18\x18\x2A\x75\xD5\x0A\x37\xE5\xBB\xEA\xED\x93\x1B\x71\xCE\xB2\xBB\xFB\xF2\xDD\xFD\x76\xBD\x2F\x6F\x77\x57\xCA\xF5\xFC\x8D\x75\x70\x7B\x7A\xFE\x4A\x9F\xB5\xA1\xB5\x7F\xA5\xCF\xA1\x72\xFE\xC6\xC1\x7C\x77\x3D\xDF\x63\xE7\xCB\x3D\x76\x65\x9D\x5D\x28\x0F\x0E\xD5\x61\xB5\x5B\xEE\x1D\xAA\x43\xA9\xB0\x76\x60\xB0\x27\x2F\x6A\xB1\x5C\xCF\x57\xCB\x6E\xC9\xCE\x57\xAB\x95\xAA\x5D\x28\xDB\x5D\x03\x83\x3D\x5D\x6B\xDD\x5A\xBE\xAB\xDC\x53\xD8\x30\x50\xE9\x99\x65\x0F\x96\xF2\x6E\x2D\x6F\x57\xF3\x83\x95\x6A\x1D\x96\xF4\x15\xEA\x6E\xB9\x6B\x95\x68\x88\xAE\x55\xDD\xD5\xC2\x60\x9D\x6A\x71\x6D\xA1\x5E\xEB\x72\xCB\x3D\x5D\xE5\x4A\x1D\xE6\x6D\x1C\xAC\x16\x2A\x75\x70\x7B\x7B\xAB\xF9\x5E\xB7\x9E\xAF\x51\x5A\xA0\xEA\xBF\x9A\xAF\xE5\xAB\x1B\xF2\x3D\xB6\x5B\xED\x1D\x1A\xC8\x97\xEB\x30\x30\xD8\x23\xBC\x14\x02\x91\xD5\xFA\x0A\xEB\xEA\x50\xCB\xD7\xE1\x62\xE1\x3D\x2F\xF2\x5F\xE8\xBA\xBC\x90\xAF\x43\x65\x6D\x31\xDF\x5D\x87\xA5\x8D\x18\xBB\xFB\xDC\xAA\xDB\x5D\xCF\x57\xED\xEE\x92\x5B\xAB\xC1\xFC\xDE\x8D\x83\xF5\x82\x5B\xEE\x5A\x54\xC8\x57\x2B\xBD\xA5\x8D\x83\x7D\x35\x70\xCA\x6E\xBD\x52\x0A\x63\x97\xE5\xAB\x95\x42\xBD\xD0\xDD\x84\xBC\xB6\x06\x57\xD7\x60\x65\x0D\xE6\xD5\xE0\x94\xDA\x2C\xFE\x6F\x9F\x52\x6B\xEA\x31\xD5\x8D\x5D\x85\x72\xBD\xD1\x19\xAB\xF9\xC1\x6A\xD0\x83\x5B\xEB\x5A\x57\xAA\xB8\xA3\xF9\xD8\xE0\x96\x86\xF2\x5D\xB5\x3E\xD1\x9D\x6F\xE8\xEA\xAE\x0C\x6E\x94\x81\x46\xF1\x5B\xAC\x55\xCA\x5D\x37\x54\x0B\xF5\x7C\xB5\x2B\x3F\x50\xA8\x1F\xDB\x67\xA8\xB7\x87\x7F\xA6\x6A\xBD\xB1\x52\x85\x72\xC5\xAF\x79\x7B\x5D\xA5\x2A\x7C\xE4\xEB\x85\x7A\xA1\x52\xB6\x2B\x83\xF9\xAA\x5B\xAF\x54\x61\xAD\xDB\x33\x2A\x5E\xD5\xF8\x60\xBE\x5A\x6A\x60\xCB\x15\xEE\x43\x4D\x7D\x8F\x10\x7E\xC2\x95\x2A\xF4\x15\x02\x0D\x55\xEB\xAB\x0C\x95\x7A\xEC\xB5\x79\xBB\xB7\x9A\x77\x05\xA6\xDE\xE7\x96\xED\x52\xA5\xE1\x27\x18\x5B\x29\x5F\xEE\xAD\xF7\x89\xCE\xFC\x97\x7D\x04\xD3\x01\x55\x0D\xB5\x7A\xB5\x50\xEE\xB5\xF3\xB5\x6E\x77\x30\x1F\xF8\xBD\x27\x5F\x2A\x0C\x88\xCA\xB5\x0B\x35\xBB\x5C\xA9\xDB\xAE\x5D\x2B\x94\x7B\x4B\x41\x3F\x7E\x1A\xDD\xA5\x8A\x18\x0B\x6E\xB9\x37\x3F\x4A\x12\x43\x65\x8E\x3D\xD0\x1F\x61\x49\xDF\x40\x3E\xD8\xE6\xDD\x95\x81\x41\xB7\x9A\x97\x6D\x45\x3F\xB8\xD5\x42\xBD\xAF\x6B\x60\xA8\x54\x2F\x0C\x96\x36\x86\xA7\xA9\xA0\x97\x42\x79\xA8\xF6\x36\xBF\x0F\x96\x8E\xF1\xF3\xDB\x4D\x83\x81\xF0\x3D\x85\x0D\x85\x9E\xFC\xDB\x78\xA8\x56\x86\xCA\x3D\x47\xFD\xBE\xCC\xED\x77\x6B\x6E\x15\x96\x6D\x74\xCB\x03\x6E\xA3\x8B\x94\xDD\x01\x51\x15\xEE\x60\x7D\xA8\x2A\x5A\xB9\x32\x34\x08\x57\x0F\xC2\xB2\x6A\x05\x16\xB9\xE5\xA1\x72\xA5\x02\xCB\xFB\x2B\x70\x85\x5B\xEE\xEE\xAB\xC0\xDC\xCA\x60\x65\xA0\xB2\xAE\x02\xAB\x2A\x1B\x2B\x03\x6B\x2B\xB0\xAC\xE0\x56\x60\xA9\x2B\x10\xB0\xB2\x02\xCB\x2B\xB0\xB4\x02\xF3\x2A\x70\x4A\x25\x50\xA6\x5A\x17\x35\xC6\x2C\x9B\xEB\xB5\x50\xAB\x94\x61\x5E\x65\x60\xA0\x52\xF6\x7B\xDE\x2C\x5B\xFC\x15\xB3\xE3\x86\x7C\xB5\x56\x68\xFA\xC5\x5D\x7B\xD4\x0F\x47\xC5\xEB\xFF\xBA\xD4\xAD\x17\xCA\xB0\x2C\xDF\x93\x5F\xE7\x16\x7A\xAA\x85\x32\x0C\x95\xF2\x65\xFB\xC2\xD9\xF6\x86\x52\xBE\x1C\xEC\x8A\xF5\x4A\x7F\xBE\x0C\xCB\xDC\x6A\x77\x5F\xBE\x0C\x17\x0F\x0D\x96\x2A\x1B\xDD\x32\x38\x1B\xF2\xB5\xBA\x5B\x86\x55\xEE\x80\xA8\x4F\xB7\x0C\xAB\x0B\x6B\xF3\xC2\x9D\x5F\x5A\xEB\xD6\xDC\x32\x2C\x72\xEB\x55\xB7\x0C\x65\xE1\xA9\xCF\xDD\x50\x70\xCB\xB0\xA2\xD4\xD3\xB5\x52\xE4\x40\xBE\x2F\x1A\x2A\xF7\xBA\x55\xF1\x35\x8F\x1D\xA7\x3A\x90\x2F\xF3\xF7\x50\xB7\x2B\xFC\x75\x39\xA5\xB5\x2E\xA3\xFA\x2A\x55\xB7\x36\x20\x5E\x97\x55\xCA\xBD\x34\xFB\xC1\xE2\x72\x8D\x26\xEE\x42\xA5\xEC\x96\xBA\x56\xBA\xD5\x7A\x9F\x40\x2F\xCC\x57\xAA\xBD\xE2\x65\xE9\xC6\x1E\x95\xD8\xAA\x4A\x6F\x0F\xA3\xBA\x85\xB3\xB2\xAF\x92\x2F\x17\xBA\x1B\xBF\x0E\xD5\xFB\xBA\x9C\xAA\xBB\x56\x61\x96\x57\xAA\x01\xCC\xDA\x4A\xA5\x94\x77\xCB\xB0\xDC\x5D\xEB\xD6\x5D\xF1\xB6\xCC\x2D\x17\xBA\xFB\xE8\x75\x91\xF8\x2C\x43\xD9\xBE\xD0\x9E\x6E\x9F\x7A\xAA\x5D\x16\xBD\x77\x40\xD4\x26\x23\x3B\x05\xB2\xF1\x3D\x9B\x11\xB5\xBA\x2B\xE6\xEA\x6E\xB7\x56\xBF\x60\xA8\x50\xAE\x4F\x9F\xD9\x55\xBF\xB0\xBD\x3C\xD9\x9E\x2D\xFC\xCD\x75\x07\x86\x06\x60\xDE\x50\x39\x5F\x58\x57\xA9\x0E\x80\xD3\x57\x19\x80\xEE\x41\xFB\x02\xBB\x27\xDF\xCD\x13\xC0\x19\x03\xF9\x01\x70\x7A\x4A\xEE\x00\x2C\x73\x4B\xEE\x46\x57\xBC\xAD\xE8\xED\x73\x07\x60\x55\xA1\xA7\x47\xB8\x62\x9D\x59\x2D\x5E\xE6\x11\x76\x00\x96\x0E\xC0\x80\xCA\x11\x65\x65\x20\xD0\x85\x06\x86\x4A\xA2\x2C\xBD\x43\x25\xA8\xDB\xD9\xD9\x76\x57\x57\x79\xA8\x54\x82\xB5\x81\x77\x37\xF0\xBE\xDA\x1D\x28\x94\x60\x50\x8C\x88\x0D\x85\x6A\x7D\xC8\x2D\xC1\x3C\xB7\xEC\xF6\x50\xB3\xAD\xAD\x54\x0B\xBD\x85\xB2\x5B\x82\xAB\x4B\xB0\xBC\x04\x4B\x4B\xB0\x2C\x9F\xAF\xE7\x0B\x5D\xCB\xDC\x8D\xF9\x7E\x58\x58\xCD\xE7\xFB\x61\xAE\x5B\x77\xFB\x61\x55\x3F\xAC\xAC\xB9\x25\xB1\x2A\xAC\x74\xFB\x4A\xEE\x86\xC2\x51\x0D\xCB\xD8\x65\xF9\x72\x4F\xBE\x6B\x49\xA1\xDF\xED\x1F\x2A\xC0\x15\xAE\xE8\xD8\x5D\xF3\x0A\xF5\x02\x2C\x1C\x2A\xBA\x55\xB7\x5E\x80\xD5\x6E\x7F\xB5\x00\xAB\x36\x96\x2A\xF5\x42\xD7\x72\xB7\xB7\x5A\x80\xE5\x6E\xB9\xA7\x50\x76\x45\x57\x83\x8B\xF3\x1B\x5C\xF9\xBA\xAE\x5C\x80\x65\x43\xA5\xBA\x2B\x5C\xB7\xCF\x2D\x8A\x97\xB9\x55\xB7\x6F\xA0\x00\x73\xF3\xE5\x5E\xB7\x54\x80\xB9\x7D\x6E\xA1\xBF\x36\xD4\x5F\x80\x25\x7D\x95\x62\x7F\x01\x56\x94\xBA\xE6\xF5\x15\xC4\xB7\x5B\xA8\xF7\x09\xB4\x5B\xAD\xD4\xFA\xC4\xEB\xC2\xA1\xEA\xC0\x50\x7F\x5F\x01\x96\x55\x7A\x44\x8C\x35\xB7\xEA\x0E\x74\x2D\xAC\x94\x7B\xC4\x6F\xE5\xA2\x5B\x72\xE5\xD7\x55\xF9\x9B\x0A\x3D\x22\xEC\x50\x8F\x7B\x83\xDB\x53\x80\xD5\x7D\x6E\x01\x2E\x77\x0B\x70\x55\x01\x56\x8A\xB8\x37\xBA\x7D\x5D\x4B\x0B\xB0\xBA\xB0\x4E\x64\xBC\x0F\x2A\x83\xDD\x95\x9E\x7C\x3B\xF5\x8D\xFE\xC5\xE5\x5A\xDD\x29\xD5\xED\x9B\x6F\xB6\x47\xC3\x2F\x73\xEB\xDD\x7D\x30\xD7\xAD\xD5\xDC\xAE\xCB\xDD\x3E\xA8\x55\xBB\xA7\x29\x42\x69\x5A\x7D\xE3\x60\x9E\x26\xC0\xA9\x7D\xB0\xDA\xED\x75\x4B\x95\x5E\x58\xE9\xF6\xB9\x37\xF4\x75\x2D\x1A\xA8\x94\x7B\x61\xF9\xC6\x82\xDB\x9F\x2F\xF7\x76\xAD\x1C\x72\xBB\xFB\x86\xF2\x12\xBD\xAA\xD0\x5B\xBE\xA2\x5A\xA8\x17\xCA\xBD\xC0\xBD\x0F\x2A\x83\x5D\x94\xEC\xA5\xF9\xDE\xFC\x8D\x83\x4B\xC5\xBA\xE3\x96\x56\xF1\x6F\xAB\x2A\x55\xB7\x6B\x55\x65\x60\x30\x5F\xEE\x85\x4B\xF3\x45\x97\xFC\xCB\xE5\xB3\x9C\xEF\xFD\x4B\xB4\xC5\x3A\xE8\x9C\x3E\xE3\xCC\xB3\xCE\x9E\x79\xCE\xB9\xE7\xB9\x6B\xBB\x7B\xF2\xEB\x60\xE5\x3A\x98\xB7\x2E\x40\x77\xF9\xEB\xB9\xA0\x24\x1B\xA4\xCF\xBC\xA1\x6A\xAD\xB0\x21\x0F\xF5\xEA\x50\xDE\x5F\xCD\x68\xBD\x82\xE5\xF9\x1B\xBA\xC4\x80\x58\x3A\x94\x6F\xCA\x8C\x5B\xCF\xC3\x3A\xB7\x54\xCB\xC3\x5C\xB7\x54\x28\xE7\xC5\xCB\x50\x2F\xBF\x5C\xE2\x6E\x70\xE9\x65\xD5\x50\xB9\x87\xDF\xAE\x76\xCB\xEE\x5A\xF7\x26\xB7\xDA\xB5\x6A\xFD\x90\x5B\xCD\x43\x37\xAD\xFE\x62\xBD\x1D\x2A\x57\xF3\x82\xD8\x18\x2A\xE7\x83\xC3\x4A\xAC\x75\x76\xA5\x09\xE7\xF6\xF4\x10\x66\xA5\x5B\x1A\xD8\x58\xCD\x97\xF3\xA2\xE7\x89\xEC\x1E\x8B\x0E\xE4\x95\x3A\x90\xD6\xA0\x5B\x55\xEB\x37\x14\xCA\xB5\x7A\xD7\x35\xD5\x4A\xA5\x7E\xDD\xD4\x63\x75\x95\xB7\xF5\x33\x77\x63\x3D\x7F\x29\x45\xB5\xA2\xE6\xF6\xE6\x61\x5E\x5F\xBE\x5A\xE9\xCF\xE7\x65\x5B\x37\xD6\xC9\xA1\x72\x41\x04\xF5\x73\x29\x69\x90\x5A\x7E\xFD\x50\xBE\xDC\x9D\x87\x95\xA2\x29\x68\xDA\x59\x9A\x0F\x51\xEF\x4D\xC4\x35\xCC\x1D\xEA\x2B\xF4\xC0\xBA\xC1\x6A\xA1\x5C\x5F\x37\x8B\x4A\x54\x18\x18\x2C\xE5\x05\x25\x97\xEF\x81\x75\x44\x25\x8E\x82\x1F\x1C\xAA\x77\x1F\x8D\x5E\x5C\xEE\xCB\x8B\x00\x3D\x32\xC3\xB5\x59\x76\x6F\xBE\x6E\x8B\x69\xD5\x5E\xE7\x16\x4A\xC1\x1F\xEA\xD5\xA1\x72\xB7\x5B\xCF\xAB\x1F\x06\xDD\xBA\xD8\x3B\xD8\xF5\x4A\xC5\x2E\xB9\xD5\xDE\xBC\xDD\x41\x4B\x70\xA1\xE4\x7B\x09\xEE\xD7\x02\x28\xBF\x29\xE1\xDA\x1E\x58\xD9\x03\xCB\x7B\xE0\x94\x1E\x28\xDF\x50\xA9\xF6\xD4\xC4\xDC\x5A\xCD\xD7\x86\x4A\xF5\x8E\x0B\x29\x24\x2C\x5B\x79\x71\xD7\xB2\xC5\xCB\x9D\xA5\x4B\x57\xCC\xB3\x2F\x08\xFF\x3A\xAF\x32\x58\x2F\x74\xC3\x65\x62\x6A\x12\x2F\x0B\x4B\xAE\x58\xE3\xC4\xEB\xFC\x7A\x5F\xA1\x32\x58\xE8\x86\x4B\x45\xED\xAB\x55\x74\xA0\xD0\x0D\xF3\x36\x56\x0B\xA5\x92\xC4\x2D\xAE\xBB\xEA\x75\xF5\x50\xB5\x5F\xC4\x51\xA9\xF7\x15\xBA\x81\xD6\xB0\x6E\x98\x5F\xDA\x38\xE0\x16\xBA\x61\xF1\xC0\x60\xBE\x5A\x70\x4B\x62\x6D\x23\xC4\x32\xB7\xDC\x23\xDC\x55\x1B\xAB\x05\xB7\x1B\x56\x75\xC3\xCA\x6E\x58\xD6\x0D\xF3\xBA\x9B\xE7\x0E\xFA\x5B\x18\x70\x4B\x53\xBB\xE1\x94\xB5\xB0\xA2\x36\xE0\x96\x37\xBA\xB0\xA2\x5A\xD8\xE8\x8A\x75\xA3\xB0\xAE\xD0\x75\x69\xA5\xAF\x50\xEE\xDD\xE8\x8A\xC9\x65\xAD\x5B\xBE\xC1\x15\xA3\xCE\x85\xD5\x85\x6A\xDF\x50\xDD\x85\x55\xEE\x50\xD5\xAD\xF5\xD5\xAB\x2E\x5C\x5C\xE9\xAD\xBA\xB0\x44\x2C\x00\x6E\xD9\x85\x45\x85\xAA\xDB\x2B\x5E\x56\xF7\xB9\xC2\x99\xD7\xE7\xF6\x0F\xB8\xB0\xAA\x50\xEE\x73\x4B\x2E\x2C\xAC\xBA\xE5\x7A\x9F\x0B\x4B\xF3\x83\xDD\x7D\xAE\xA0\x28\xAA\x6E\x8F\x08\x5E\x2E\x0B\x77\x65\x9F\xDB\x5B\xEB\x5A\xE9\x02\x57\xAA\x58\xA4\x5C\xE8\x14\xB5\x4C\xFB\x4C\xB8\xE6\x7A\xB8\x66\xD6\xF5\xD4\xD9\x66\x5D\x07\xD7\xCC\x6A\xBC\x5D\x7F\x63\x4F\xA1\xB7\xC0\xEF\x81\xD7\xEB\x1B\xAF\x01\xE4\xE0\x50\xB9\x5B\x46\xE0\xBF\x5D\x5F\xAA\xDC\x90\xAF\xD2\x6B\xE3\xED\xFA\xA1\xC1\x41\xF9\xDA\x78\xBB\xDE\x2D\x95\x87\x06\xE8\xB5\xF1\x76\x7D\x77\xB9\x5E\x2D\xD1\x6B\xE3\xED\xFA\xB5\x25\xB7\xDC\x4F\xAF\x8D\xB7\xEB\xDD\x5A\x77\xA1\xC0\xC1\xFD\xB7\xEB\x7B\xAB\xEE\x60\x1F\xBD\x36\xDE\xAE\xAF\x0D\xBA\xDD\x79\x7A\x6D\xBC\x5D\x2F\xBA\x25\xBD\xF9\x2F\xD7\xBB\xA5\xC1\x3E\x57\x66\x48\xBE\x5D\x7F\xED\xD4\xEB\x60\xA0\x50\x13\x1B\x07\xFB\x3A\xA8\x8B\x59\x49\xBC\x5E\x0B\xD7\xC0\xD5\xD0\x3D\xD8\xE1\x93\x1F\xF6\x05\xB6\xE8\xD2\xAB\x56\x2D\xBE\x7A\x7E\xD7\x32\xE7\x4A\xB8\xF6\x0A\xB8\x76\x15\xAC\x84\xDA\x72\x77\x39\x94\x45\xE5\x9F\x79\x3A\x75\x7A\xE7\xCA\xD5\x97\x3A\xCB\x57\x2D\x58\x71\xE9\xB2\xAE\x19\xF2\xA7\xD1\x7E\x58\x06\x25\x57\xFD\xB6\xC4\xB9\xD4\x59\xBD\xEA\xB2\xB9\x4E\xD7\x5C\x67\xD5\xFC\x79\xCE\xAA\xF9\x62\x12\xBB\x41\xB4\xEC\xF2\xCB\x96\x2E\x95\x3E\xCF\x3C\xBD\x7D\x94\x88\xA6\xCD\x98\xDC\xEC\xF9\xDA\x8B\x61\x1E\x2C\x2D\x94\xF3\x6E\xB5\x6B\xAE\x7A\x71\xC4\x5E\x48\x44\x72\x1E\x13\x65\x67\x9D\xDB\xD8\xFD\xD4\xD7\x75\x9C\xEB\xCF\x71\x97\xAD\x5E\xD0\x71\x2E\xFB\x99\x3E\xD3\x27\x94\xCF\x82\xAA\xE8\x5B\xFC\xBA\x72\xBA\x88\x68\xA0\xD2\x53\x1E\x1A\x20\x62\x8F\xDF\x2E\x98\x6D\xAF\x3C\x13\xD6\xBA\x35\xF2\x36\x03\x36\x4D\x87\x5A\xA1\xB7\x2C\xA6\xDD\x8E\xE9\x22\x8B\xEA\x6B\x3A\x6C\xEA\x0C\x13\x66\x44\x23\xF6\x15\xD6\xD5\x05\x1A\xA8\x17\xD6\xE8\xF5\x06\xCA\x01\x7B\x1D\x92\xEF\x50\xF3\x5F\xAA\xDD\xEA\xB5\x27\xDF\xDD\x71\xA1\xFA\x08\xC6\xAE\x3E\x61\xC0\xBD\x71\xB0\x9A\xEF\x56\x3F\x15\xCA\x85\xBA\xFA\x16\x35\x7C\xE1\x6C\xBB\xB4\x56\xFC\x50\x5A\x4B\x28\x26\x5E\xA1\xC0\x4E\xB7\x5B\xAD\x6E\xBC\xA6\xF3\x3A\x51\x00\x0A\xCE\x88\xE9\x61\xC4\x0C\x46\xC0\x34\xE8\x10\x44\xD7\x19\x30\xF5\x74\xC8\x0E\x0C\xF6\x74\x15\x6A\xB5\xC1\x7C\x77\xC1\x2D\xB5\xF3\xB0\x9D\xAC\xD0\xDD\x15\xB1\x58\xF5\xB8\x75\x37\xFC\x4B\xAD\xCF\xAD\xE6\x7B\x9A\x7F\x6A\xAF\x0C\x76\xD1\x52\xD6\x2E\x69\x91\x4B\xF3\x83\x79\xB7\x3E\x19\x0A\xB5\x41\x31\x22\x67\xB4\x57\x2B\x37\xD4\x26\x43\x77\xA5\x54\x13\xDE\x02\x47\x5A\x33\xA6\xD8\xFC\xDB\x51\x71\xCC\xEB\x73\xAB\xF3\xC4\xC2\x1B\x88\x46\x44\x10\xF8\x2C\x4F\x06\x49\xAE\xB7\x0F\x8C\x12\x83\xA4\x88\x26\x43\x7B\xA9\x32\x59\x74\x85\xF6\xBE\xC2\x64\xC8\xF6\xB9\xB5\x2E\x3A\xDD\x6B\x17\x5F\xDD\x95\x72\xDD\x2D\x94\x6B\xE2\xA3\x5D\x2D\xCB\x32\x1A\xB1\x30\xCF\x1F\x18\xAC\x6F\xBC\xA2\xD0\x53\xEF\x1B\xFD\x77\x45\xEF\x1D\x3B\x7D\xA6\xC8\x46\x0F\x3D\x8F\x37\xB4\xA3\x15\x3F\xF0\xCB\x51\xC1\x7C\x6A\x61\x32\x70\xAB\x14\xF8\x88\x33\xDF\xBE\xB6\x81\xA9\xE7\x7B\xF3\xD5\x76\x77\x72\xB8\xA9\xDD\xC9\x70\x03\x8D\x08\x51\x23\xD3\x2F\xBB\xE0\x82\xE9\x33\x27\xC3\xD0\x51\x98\xEA\x51\x98\x5A\xB5\xFB\x28\x9C\x5F\xB5\x33\xCE\x3E\x7B\x32\x88\x0A\x0E\x7C\x75\x07\x3F\xCA\x93\x45\xA7\x6D\x9F\x31\x19\xDA\x07\xAB\xF9\x75\x85\x1B\xB9\x0B\x34\xB0\x85\x1E\x2E\x61\x99\x08\xA5\x8E\xE9\xA3\x7A\x9C\x2E\x93\x14\xEF\x9D\x32\x41\xF5\xDE\xDD\x78\xAD\x77\x5C\x58\xCD\xAF\xE3\xF8\xC4\x77\x65\xA8\xDE\xC5\xF5\xD8\xD5\x40\xD6\xEA\x6E\xB5\xAE\xEA\x55\x20\xAA\x43\xE5\xF5\x1D\x17\x8A\xD4\x82\x58\xF1\xCD\x31\x07\x37\x8C\xDC\x7F\x2F\x6C\x1F\xEC\x18\xEC\x9C\x2C\x52\xAC\xD4\xDD\x12\x07\x0A\x7A\x2B\x94\xEB\x17\xB6\xAF\x2B\xB9\xF5\xA9\x32\x5A\xAE\x89\x0B\x46\xF3\x46\xA4\x64\xAD\xEB\x68\x9F\x8D\x4C\x41\x7B\xB9\x56\xEF\xF7\xC3\x77\xF7\xFB\xBE\x21\x5B\x18\xEC\xB8\xB0\xE4\xD6\xEA\xED\x93\xA1\xD0\xC3\x33\xC0\xCD\x37\xDB\x83\xD5\x4A\x6F\x57\xC7\x85\xA2\x4E\xDB\x0B\x3D\x1D\xD3\x27\xFB\x9E\xA8\x04\x34\x6B\xBA\x5C\xC5\x22\xA0\x7D\x41\xF0\x53\x7A\x09\xA2\xD4\xCA\x34\x39\x48\x5C\x4E\x86\x5A\xBE\xDE\x0E\x93\x98\x8A\x17\x13\x4F\x4E\xCD\xE4\x34\xD2\x72\x74\xF0\xD9\x33\x8B\xCF\xF1\x66\xD9\xEA\xF3\x06\xB7\x5A\x26\xEA\x36\xB0\xDB\x80\xA6\xE7\xA8\x5D\x08\x1C\xEB\x77\x67\xEE\xBC\x8B\xE7\x2F\x08\xFF\xCE\xD8\x85\x8B\x16\x5F\xB2\x64\xE9\xB2\xE5\x2B\x56\xBE\xF3\xD2\x55\xAB\x2F\xBB\xFC\x8A\x2B\xAF\xBA\x9A\xE3\xEB\xED\x2B\x14\xFB\x4B\x03\xE5\xCA\xE0\xFA\x6A\xAD\x3E\xB4\xE1\x86\x1B\x37\xDE\xD4\x88\xF3\x8C\x69\xFF\xB3\xF1\x75\x74\x89\x38\x7E\xD2\x82\x3E\x64\x4C\x84\x97\x53\x8D\xEF\x8B\x01\x9B\xD2\x8B\x43\x0A\x10\x6C\xD8\x0C\x79\xD8\x02\x23\xEF\xD8\x7B\x3C\xD8\x09\x1B\xA1\xDD\x9E\x6C\x23\x4C\xB3\xBB\xEC\x4E\x80\x4E\xE8\x54\xA1\x1C\x00\xB8\x5A\xBE\xDB\xD2\x75\x01\xE0\x26\x00\xD8\xEB\x79\x1E\x48\xF7\xCB\x00\x20\xE0\x3F\x34\xF6\x23\xDC\x9D\x00\xF0\x5C\x28\xEC\x0B\x00\xF0\xCB\x10\x6E\x2F\x00\xBC\x1E\x8A\xEF\x00\x00\x1C\x09\xE1\xC4\x8B\x80\x8D\x32\x1C\xB9\x08\x30\x8D\x32\x7B\x9C\x0F\x22\xF3\xAA\x00\xE7\xC9\xF0\xD3\x91\x61\xBC\xFC\x16\xEE\x0C\x04\x38\x27\x14\xF6\x3C\x04\x58\x14\xC2\x5D\x82\x00\x37\x84\x70\x37\x22\x10\x88\xE7\x76\x19\xE7\x46\x04\xD8\x14\xF2\x37\x8C\x40\xF0\xEC\x11\xF6\x23\xDC\x11\x04\x10\xF0\x6D\x59\x0E\xE1\xBE\x07\x19\xC4\xF3\x63\x89\x7F\x2F\x02\x6C\x0E\xC5\x77\x1B\x02\x81\x78\x7E\x28\xFD\xDD\x8E\x00\x77\x84\xFC\xBD\x0F\x01\xEE\x94\xFE\x9E\x96\xFE\xDE\x8F\x00\x77\x85\xFC\xDD\x8D\x40\x20\x9E\x15\xD2\xDF\x07\x90\x41\x3C\xDF\x97\xB8\x7B\x90\x41\x3C\x3F\x90\xB8\x0F\x22\xC0\x87\x42\xF1\x7D\x18\x81\x20\x98\xEE\x3F\x20\x83\x78\x7E\x24\x71\x1F\x41\x06\x37\xD0\xA7\x3E\x8A\x0C\xE2\x79\x46\xE2\x3F\x86\x0C\xE2\xD9\x2D\x71\xFF\x88\x00\x1F\x0F\xA5\xFB\x09\x04\x82\xFB\xA5\x1F\xE1\xDE\x8B\x0C\xC1\xF8\x3E\x85\x0C\xE2\xD9\x23\x71\x9F\x46\x86\xF7\xAA\x7A\x07\x80\xFB\x90\x01\x02\x7D\x78\x0B\x02\x7C\x36\x94\xEE\x03\x08\x04\xE2\xF9\xA9\xF4\xB7\x15\x01\x1E\x0C\xF9\xDB\x86\x40\x10\xF4\xF7\x39\x04\x78\x38\xE4\xEF\xF3\x08\x04\x41\x7F\xFF\x8C\x00\x8F\x84\xFC\x3D\x8A\x00\x5F\x90\xFE\x7E\x22\xFD\x7D\x11\x01\xBE\x12\xF2\xF7\x55\x04\x02\xF1\xFC\x4C\xFA\xFB\x1A\x02\x3C\x16\xF2\xF7\x38\x02\x3C\x11\xC2\xED\x40\x20\x38\x57\x86\x13\xEE\x77\x90\x41\x3C\x1A\x0F\x3F\xF8\x2E\x32\x90\xC2\x33\xE9\xF7\x7B\xC8\x70\x44\x8E\x0D\xF1\xDB\x93\x08\xF0\x64\x28\xEC\x53\xC8\x10\x0C\xFB\xAF\xC8\x10\x0C\xFB\x7D\x04\xF8\x7E\x28\xEC\x0F\x90\x21\x18\xF6\xDF\x90\x21\x18\xF6\x69\x04\xF8\x79\xA8\x6C\xBF\x40\x20\x78\x54\xFA\x13\xEE\x2F\x11\xE0\x4F\x21\x7F\x2F\x23\x10\x04\xD3\x7D\x05\x19\x82\xE9\xFE\x19\x19\x82\xE9\xBE\x8A\x00\xAF\x85\xE2\x7B\x1D\x81\x40\x3C\xF7\x49\xBF\x6F\x20\x83\x78\x9E\x92\xB8\x03\x08\x90\xD1\x9A\xC3\xDA\x1A\x10\x88\x67\x93\xF4\x97\xD3\x00\xCE\x0C\xF9\x9B\xA5\x01\x81\x78\xCE\x38\x9D\xDD\xF3\x35\x80\x0B\x42\xFE\x66\x6B\x40\x20\x9E\xEB\x64\x7C\x17\x6A\x0C\xE2\x69\x97\x61\x2F\xD2\x00\xE6\x68\x00\x17\xA9\xEF\xD3\x01\x1C\x0D\x60\x6E\x28\xBE\x79\x1A\x10\x88\x67\xB6\x8C\xEF\x62\x8D\x41\x3C\xF3\x65\x5D\xCD\xD7\x18\xC4\xB3\x50\xE2\x16\x68\x00\x2B\x42\xF1\xAD\xD4\x80\x20\x23\xD3\x15\xEE\x3B\x35\x86\x36\x89\x13\xEE\xA5\x1A\xC3\xC9\x12\x27\xDC\x55\x1A\xC3\x54\x99\x0F\xE1\xAE\xD6\x00\x04\xCC\x90\x38\xE1\x5E\xAE\x01\x5C\x21\xEA\x50\xE2\x84\x7B\x95\x06\x20\x60\xA6\xC4\x09\xF7\x1A\x0D\x40\xC0\xD9\x12\x27\xDC\x6B\x35\x00\x01\x2B\x3E\xCB\xE9\x0A\x77\x8D\xC6\x10\x8C\xCF\xD5\x00\x04\x2C\x91\xFE\x84\xDB\xAD\x31\x4C\x57\xEB\x93\xE7\x79\x79\x0D\x40\x40\xBB\xF4\x27\xDC\x75\x1A\xC3\xC5\x12\x27\xDC\x3E\x8D\x61\x9A\x0C\x2B\xDC\x82\x06\x20\xA0\x43\xE2\x84\x5B\xD4\x00\x8A\xA1\xB0\xFD\x1A\xC3\x1B\x93\x19\x27\xDC\x92\xC6\xE0\x48\x7F\xC2\xAD\x68\x0C\xC1\xF8\xD6\x6B\x00\x02\xDE\x92\x61\x85\x5B\xD5\x18\xCE\x90\xFE\x84\x3B\xA4\x01\x08\x38\x5D\xE2\x84\xFB\x6E\x0D\x40\xC0\x7F\xC8\xB0\xC2\x1D\xD1\x18\x4E\x95\xFE\x84\xFB\x5E\x0D\x40\xC0\x3C\x99\x17\xE1\xDE\xA2\x31\x04\xFD\xDD\xAE\x01\x08\x38\x5D\xFA\x13\xEE\x1D\x1A\x43\xD0\xDF\xFB\x34\x00\x01\xDF\x90\x38\xE1\xDE\xA9\x01\xBC\x5F\x03\x38\x4D\xE2\x84\x7B\x97\x06\x20\xE0\x31\x89\x13\xEE\x87\x34\x00\x01\xA7\x48\x9C\x70\x3F\xA5\x01\x08\x38\x5E\xA6\x2B\xDC\x4F\x6B\x0C\x63\x25\x4E\xB8\xF3\x75\x86\xD5\xB2\x7F\x0B\x77\x50\x07\xA8\xE9\xCD\x7D\x7C\x83\x0E\x70\x43\x08\xF7\x2E\x1D\xE0\xDD\x7A\xF3\x7A\x34\xAC\x33\x88\xA7\x2E\xF1\xB7\xE9\x0C\x54\x56\x89\xBB\x43\x07\xB8\x53\xE2\x4E\x91\xB8\xBB\x74\x06\xF1\xCC\x91\xB8\xBB\x75\x80\x0F\x48\xDC\x45\x12\xF7\x41\x1D\xE0\x33\x12\xA7\x68\xB3\xFB\x75\x80\xED\x21\xDC\xE7\x74\x86\x9F\xCA\x7A\x11\xEE\xC3\x3A\xC0\x3F\xEB\x00\x3F\x93\x38\xE1\x3E\xAA\x03\x7C\x53\x6F\xA6\xE1\xBE\xA5\x03\x08\x78\x51\xE2\x84\xFB\x6D\x1D\xE0\x07\x21\x7F\xFF\xA6\x03\x08\xD8\x29\x71\xC2\x7D\x5A\x07\xF8\xA1\x88\x53\xE2\x84\xFB\x23\x1D\xE0\x47\x32\x7F\x31\x99\xBF\x5D\x3A\xC3\xB7\xA4\x3F\xE1\xEE\xD6\x01\x04\x3C\x29\x71\xC2\xDD\xA3\x03\x08\xD8\x2D\x71\xC2\x7D\x4E\x07\x10\xF0\x7D\x89\x13\xEE\xF3\x3A\x80\x80\x03\x6A\x6E\xF6\x3C\xEF\x05\x1D\xE0\x4F\xA1\x76\xDB\xAF\x03\xC1\x43\xD2\x9F\x70\x5F\xD6\x01\x04\x3C\x22\x71\xC2\x7D\x45\x07\x10\x10\x95\xF9\x15\xEE\x9F\x75\x86\xBB\xA4\x3F\xE1\xBE\xAA\x03\xBC\x2A\xCB\xF6\x1D\x89\x7F\x4D\x67\xD8\x22\xBF\x85\xFB\x86\xC8\x5B\x28\x2F\x07\x75\x20\x10\xCF\x41\xE9\xF7\x90\x0E\x70\x38\xE4\xEF\x2D\x1D\xC0\x93\xFE\xD4\x9A\x02\x06\xC0\x18\x83\x71\x2B\x65\x1E\xD3\x06\xC0\x34\xA3\xB9\x1F\x74\x1A\x00\x2B\x8C\xE6\x76\x5B\x69\x00\x74\x19\xCD\xE5\x5D\x63\x00\xBC\xC7\x68\x4E\xF7\x4E\x03\x60\x47\x08\xB7\xD3\x00\x02\xF1\x8C\x91\x69\x7C\xD3\x00\xF8\x61\xC8\xDF\x8F\x0C\x20\x78\x59\xA6\x21\xDC\x5D\x22\x7F\x91\x66\x7F\xD3\x23\x00\x97\x47\x64\x5E\x65\x7C\x6E\x04\xE0\xB6\x08\xC0\x2E\x19\x56\xB8\x5B\xD2\x00\xDF\x4D\xF3\xEF\x6B\xDA\xD8\x7D\x32\xCD\x10\xC4\x3D\x9D\x66\x08\xE2\x76\xA5\x01\x0E\xA5\x01\x76\x5A\xE0\x3F\x6F\xA5\x01\xBC\x10\x6E\x4B\x2B\xC0\x9F\x5A\x65\x98\x8F\xB1\xBB\xBF\x15\xE0\xB5\xD6\xE6\xBE\x7B\xA0\x15\xE0\xAD\xD6\xE6\xBE\x36\xD2\x06\x20\xE0\x43\xFF\xC1\x38\xE1\xBE\xA7\x0D\x40\xC0\x87\x25\x4E\xB8\xEF\x6D\x03\x10\x70\xAF\xC4\x09\xF7\x96\x36\x80\x5B\xDB\x00\x3E\x2D\x71\xC2\xDD\xDC\x06\x20\xE0\x53\x12\x27\xDC\xDB\xDA\x00\x04\xFC\x93\xC4\x09\xF7\xF6\x36\x00\x01\x3F\x93\x38\xE1\xDE\xD1\x06\x20\xE0\x5B\xEF\xE3\xFC\x0A\xF7\x9E\x36\x80\xAF\xCB\xFA\x98\xF3\x2A\xFB\x7D\xA2\x0D\x60\x47\x08\xB7\x71\x22\x83\x71\x27\xE3\x85\xFB\xEE\x89\x0C\xBF\x6B\x61\x9C\x70\xEF\x9E\xC8\x70\xAE\xF4\x47\xEE\xC9\x00\x1F\x39\xB9\xB9\x7D\x3F\x79\x32\x10\x7C\x57\xD6\x95\x70\x3F\x7D\x32\xC0\xA7\x4F\xE6\x70\xCE\xAF\xE5\x18\x39\x19\xC0\x0B\x85\x85\x0C\x40\x34\xD3\xA8\x77\xE1\xC6\x32\x00\x63\x32\xB2\x1D\x64\x9C\xE9\x0C\xC0\xF1\x21\x7F\xE3\x33\x00\x13\x43\xFE\xEC\x0C\xC0\x69\x21\x7F\xED\x19\x80\x69\x21\x7F\x9D\x19\x80\x73\x42\xFE\xCE\xCD\x00\x5C\x14\xF2\x37\x27\x03\x30\x3F\xE4\x6F\x51\x06\x60\x59\xC8\xDF\x3B\x33\x0C\x41\x7F\xAB\x32\x0C\x41\xDC\x65\x19\x86\x20\xEE\x8A\x0C\x43\x10\x77\x55\x86\x21\x98\xC6\x35\x19\x86\x20\xEE\xBA\x0C\x43\x10\xD7\x95\x61\x08\xE2\xD6\x64\x00\x7A\x43\x69\xF4\x65\x00\x2A\x21\x7F\x83\x19\x80\xF5\x19\x80\x4B\xA4\x3F\xE1\x56\x33\x00\x43\x19\x80\xCB\x25\x4E\xB8\x1B\x32\x00\x37\x64\x00\x7A\x24\x4E\xB8\x37\x66\x00\x36\x66\x00\x46\x24\x4E\xB8\x37\x65\x00\xDE\x95\x01\x18\x94\x38\xE1\xDE\x9C\x01\x78\x77\x06\x60\x93\xC4\x09\x77\x24\x03\x70\x7B\x28\x7F\x77\x64\x00\x3E\x10\xCA\xDF\x3D\x19\x80\x8F\x85\xFC\xFD\x63\x06\xE0\xBE\x90\xBF\x2D\x19\x80\xAD\x21\x7F\x0F\x66\x00\xFE\x39\xE4\xEF\x91\x0C\xC0\xA3\x21\x7F\x5F\xCC\x30\xC4\x25\x4E\xB8\x5F\xCB\x00\x3C\x16\x0A\xFB\xF5\x0C\xC0\x37\x24\xEE\x2B\x12\xF7\x78\x86\x41\x3C\x6F\x48\xDC\xBF\x64\x18\x7E\xF6\x1B\x39\x7E\x7F\xE3\x79\xDF\xCE\x00\x7C\x3B\x94\xC6\x53\x19\x80\x1F\xC8\xB0\x6A\xED\xFA\xB7\x0C\x43\x30\xBE\x5D\x19\x80\xDD\xA1\x3C\xBF\x90\x01\xF8\x49\x28\x7F\x3F\x15\x69\x4A\xDC\xBD\x6A\x6D\xC8\x00\xBC\x18\x0A\xFB\xEF\x19\x86\xE0\x3A\xF8\x9F\x19\x80\x97\x42\xF1\xFD\x57\x06\xE0\xF7\x12\x77\x8F\xC4\xFD\x21\xC3\x00\x81\xF5\xED\xCF\x19\x86\x60\xD9\x0E\x64\x00\x0E\x4A\x7F\x23\x6A\x1D\xCC\x00\x1C\x96\xB8\xF7\x4A\xDC\x9B\x19\x86\x60\x79\x4F\xCD\x32\x88\xE7\x7E\x39\x97\x9C\x9E\x65\x10\x8F\xF3\x2B\x49\xF3\x66\x19\xA8\x6D\x24\x6E\x46\x96\x41\x3C\x72\x1A\x84\xE5\x59\x86\xDF\xCA\xF8\x85\xBB\x26\x0B\x50\x91\xFE\xE4\xD2\x02\x83\x59\x80\xE1\x2C\xC0\x7E\xE9\x4F\xB8\xB7\x64\x01\x6E\x25\x7F\x8D\x39\xEC\x2B\x93\x00\x7E\x34\x89\xC3\x9C\x24\xC3\xEE\x9A\x04\xF0\xD2\x24\x80\xDF\xC9\xB0\xE4\x4E\x01\x98\x3A\x85\x7F\x57\xEB\x60\xE7\x14\x80\xEB\xA7\x34\xAF\x83\x6B\xA6\x00\xB8\x53\x9A\xD3\x58\x3B\x05\x08\xA8\x3E\x9F\x63\xBF\xDD\x53\x18\x28\xDD\x97\x19\xD7\x33\x85\x41\x3C\x27\x4A\x7F\xF9\x29\x0C\x7B\xF6\xF0\xB7\x70\xD7\x4D\x01\x10\xF0\x82\xC4\x09\xB7\x77\x0A\x40\x29\x94\xEE\xC0\x14\x20\xA0\x7A\x92\x7E\xCB\x53\x18\xC4\xA3\xCB\x34\x2A\x53\x18\xC4\xF3\xA2\xF4\x37\x38\x85\x41\x3C\xBF\x96\xB8\xEA\x14\x80\x5A\x28\x8D\xA1\x29\x00\x1B\x42\xB8\x4D\x53\x00\x86\x65\xD8\x6F\xCA\xB0\x23\x53\x00\x7E\x13\xF2\xF7\xFB\x29\x00\xFB\x42\xB8\x57\xA6\x00\xFC\x39\x84\x83\x0E\x80\x53\x3A\x00\xB6\xC8\x31\x28\xDC\xD3\x3A\x00\x4E\x0B\xE1\x3A\x3A\x00\x3A\x42\xB8\x39\x0F\x00\x0C\x3C\xD0\x1C\xDF\xC8\x03\x00\x9F\x0C\xE1\x72\x5B\x01\xA6\x6D\x6D\xC6\xCD\xD8\x0A\x50\x09\xE1\x36\x6E\x05\xB8\x39\x84\x7B\xF7\x56\x20\xA0\x31\x30\xC4\x69\x6F\xDA\x0A\x70\x7B\xC8\xDF\xFB\xB7\x02\xDC\x15\xC2\xFD\xFD\x56\x20\xA0\x76\xBC\x5A\xCE\x93\x5B\x01\x3E\x1C\xF2\xF7\x0F\x5B\x81\xC0\xEF\x7B\x00\xF0\xD1\xAD\x00\xDB\x42\xFE\x1E\xDA\x0A\x04\xE2\x79\x5C\xC6\xB7\x7D\x2B\x83\x78\x1E\x95\xB8\xCF\x6D\x65\x10\xCF\x97\x25\xEE\xE1\xAD\x0C\xE2\xD9\x21\x71\x9F\xDF\xCA\x10\x8C\xEF\x91\xAD\x0C\xE2\xD9\x27\x71\x8F\x6E\x65\x10\xCF\x73\x12\xF7\x85\xAD\x0C\xE2\xF9\xBD\xC4\x7D\x71\x2B\x03\xB5\x8F\xA4\x99\xBF\xB4\x15\x60\x47\xA8\x1C\xDF\xDA\x0A\xF0\xED\x10\xEE\x3B\x5B\x81\x40\x8D\x35\xE1\x7E\x77\x2B\x83\x78\x9E\x90\x69\x7C\x6F\x2B\x83\x78\x9E\x92\xED\xB1\x6A\x3B\xC3\x9A\x37\xE5\x9A\xFA\xA6\xE7\x0D\x6E\x07\xD8\xB1\x1D\xA0\xB3\x4F\xD2\x15\x7D\x9E\x97\xF5\x00\x66\x31\x05\xEF\xD3\xE8\x8E\x07\x70\xB5\xD7\x4C\xA3\x83\x81\x70\x9A\xC1\x07\x4E\xED\xD2\x5F\xBB\x81\xB0\xC2\x40\x78\x41\xFA\x13\xEE\x23\x06\xC2\x33\x21\x7F\x2F\x18\x08\x87\x43\xFE\x46\x92\x08\x5F\x48\xB2\x3F\xB5\xBF\xDC\x99\x44\x78\x25\x89\x4D\xFB\xB7\x2D\xE3\x11\x76\x8C\xC7\xA6\xFC\xED\x1C\x8F\xF0\xAB\xF1\xD8\x94\xBF\x39\x65\x84\xAE\x72\xB3\xBF\x35\x65\x84\xE1\x72\xB3\x3F\x78\x09\x21\xFB\x12\xFB\xCB\x49\x7F\xB9\x97\x10\xE6\xBD\x84\xF0\x4B\xE9\x4F\xB8\x86\x5C\x6F\x92\x12\xE4\x16\x83\x1E\xB1\xF7\x3D\x5D\xD0\x38\x00\xF0\x2E\x38\xFA\x09\xFF\xFE\x6E\x00\x90\xD3\x03\x5C\x37\x8A\x7F\x7E\xF6\xCB\x2D\xF5\x24\x59\x1D\x57\xB1\x3B\xE7\x03\xEC\xDE\xF1\x6D\x76\x8B\x07\xC8\x1D\xB9\x75\xAA\x43\xFD\x62\x6C\x37\xB9\xB7\x3E\xFF\x31\x72\xF3\x4F\x3F\xED\xDC\x23\x63\xC4\x00\x94\x76\x23\x8C\x86\x3F\xF7\xC7\x08\x1F\x3C\x86\xFF\xE0\xA3\x89\xF9\x13\x00\x22\x72\xCD\x15\xDB\x94\x94\xEC\xA9\x62\x3B\x72\x02\x00\x4C\x04\x22\x91\xE9\x0C\x61\xB2\x58\xE3\x00\x60\x1A\x00\x9C\x0D\x00\xE7\x03\xC0\x6C\x00\x98\x27\xCF\xEF\x16\x4B\xDE\xC1\x2A\x51\x52\x79\x96\x9F\x07\x80\x5E\x00\xE8\x07\x80\x01\x00\x58\x0F\x00\xC3\x00\x70\x0B\x00\x88\xAD\xC3\xFB\x01\xE0\x23\x00\x20\xB6\x42\x9F\x92\xE7\xF5\xA2\xDB\x3F\x2C\xC6\x99\xE4\x25\xED\x10\x73\xB0\x18\x27\x62\x3F\x26\xCF\xF0\xF1\xFF\xC3\xFC\x6E\x1C\x25\xCF\x1F\x38\x46\xBE\xB7\x85\xF2\xFE\x8D\x51\xF2\xBF\x3B\xD0\x06\x22\xDF\xE3\x00\xE0\x78\x00\x78\x87\x2C\xC3\xF8\xC0\xEF\x27\xCA\x35\x7D\x82\xA4\x1F\x26\xF2\x16\xE8\xA8\x47\xF5\xFB\xBF\x93\xAE\x24\x07\xFC\xB3\x1F\x75\x2E\x74\x9A\x74\xD5\x38\x9E\x1C\x8A\xE7\x74\x59\x67\xA2\x6F\x77\x00\xC0\x54\x59\x7F\xC1\xDF\xC5\xBC\x3D\x5D\xD0\x36\xF2\xF7\x93\x09\x77\x1E\xC4\x21\x01\x49\x60\xCD\xF5\xE2\xDB\x81\xAB\xA1\x0B\xBA\xC0\x85\x9B\xE0\x3F\x53\x48\x6D\x96\x3E\x1B\x9B\x38\x91\xB7\x1F\x87\x74\x46\x1E\xC6\xDF\xA2\xF1\xF7\xB8\xB3\x11\xF4\x00\x7E\x7F\x9A\xFD\x87\xF1\x77\x01\xFB\xB7\xCF\xC6\xA6\x71\xFD\x47\xE9\x3F\x8C\x37\x65\x9E\x45\x1E\x45\xFE\x94\x2B\x5A\x3A\x0E\x71\x2A\x83\xE8\x49\xC3\x30\x4C\xFE\xB2\xB0\x89\x7E\xB7\xE5\xAE\x21\x0B\xD3\x60\x16\xCC\x81\x6B\x60\x0D\xBC\x0B\x36\x11\x3F\x95\x8C\x88\x49\x2E\x69\xB8\xFC\xFC\xBD\x00\x5C\x60\x56\xF3\x3B\xD2\x9C\xDF\xEF\x84\xCA\x61\xC9\xFC\x86\xF1\x5B\xA5\xFF\x5D\x67\x23\x68\x01\xFC\xBD\xD2\x7F\x18\x7F\x99\xF4\xFF\x42\xA8\x5E\x2F\x91\xFE\xC3\xF8\x39\xD2\xFF\xCF\x43\xF1\x9C\x2D\xFD\x87\xF1\x67\x48\xFF\xBF\x0D\xE1\x6D\xE9\x3F\x8C\xFF\xD6\x18\xF6\xFF\x87\x50\xBA\x5F\x1D\xC3\xFE\xC3\xF8\xA2\x8C\x7F\x7F\x08\xDF\x25\xE3\x0F\xE3\xFF\x20\xE3\x7F\x35\x84\x7F\x51\xC6\x1F\xC6\x7F\x54\xFA\x3F\x10\xC2\xBF\x5F\xFA\x0F\xE3\x9F\x97\xFE\x61\x66\x73\x3F\xFA\x37\xE9\x3F\x8C\x1F\x96\xF9\x4F\xCF\x6C\xAE\x87\xBA\xCC\x7F\x18\x8F\xD2\xFF\xF8\x99\xCD\xE9\xBE\x2E\xE3\x0F\xE3\x3F\x2C\xFD\xDB\xA1\x74\xDF\xA7\xFA\x7B\x08\xFF\x39\x99\xFF\xCE\x99\xCD\xFD\x6A\x8B\x8C\x3F\x8C\x6F\x83\xE6\xE7\x8B\x3A\xC2\xE3\x3A\xC2\x4F\x92\x08\x67\x68\x08\x67\x26\x11\xBE\x9A\x46\x78\xE7\x09\x08\xD7\x9C\x80\xF0\xCD\x34\x82\xAB\x23\xBC\xD3\x42\xD8\xAC\x23\x7C\x4A\x47\x78\xE2\x38\x84\x9D\x91\x46\xAE\xC5\xDC\x71\x1E\x00\xCC\x0A\xCC\x1D\xE7\x07\xD2\xB8\x40\xCE\xCF\x17\xCA\xF5\x76\x8E\x94\x3B\x08\xFE\x3E\x57\xCE\xDF\x17\x4B\x3E\xD2\x82\xD0\xEF\x0B\x65\xF8\x45\x72\x7E\x57\x73\xA5\x92\x5F\xB0\xE5\x3C\x5F\x94\xB4\x43\xBF\x9C\xEB\x77\xD8\x00\x25\xB1\x3F\x90\xF8\x1A\x00\x08\x10\xE4\x4C\x3D\x20\xE7\x10\x94\x6F\x50\x72\x0D\x76\x40\x9E\x41\xBC\xFF\x4A\xC2\x8E\x89\x2C\xD3\xF0\x5B\x19\xF6\xDF\xC5\x5E\x57\xE4\xD9\x06\xF8\x5D\x40\xD6\x21\x28\xE3\xE0\x05\x65\x1B\xA4\x4C\x83\xA8\x41\x25\xA3\x20\xDE\x95\x6C\x82\xF0\xAE\x64\x12\x04\x5E\xC9\x22\xDC\xEE\x79\x9E\x92\x41\x10\x7E\x82\xB2\x07\x41\x99\x03\x25\x6B\xF0\xE3\x80\x8C\x81\x88\x47\xC9\x16\xFC\x30\x20\x53\x20\xE2\x51\xB2\x04\x4F\x07\x64\x08\x04\x5E\xC9\x0E\xAC\x08\xC8\x0C\x7C\x3F\x20\x2B\xF0\x83\x80\x8C\x80\xF0\xAF\x64\x03\x9E\x0E\xC8\x04\xFC\x28\x24\x0B\xA0\x64\x00\x9E\x09\xF0\xFE\x77\x07\x78\xFE\x22\x9F\x41\x5E\xBF\xE2\xF1\x3F\x13\xE0\xED\xEF\x09\xF1\xF4\x15\x2F\xFF\xB9\x00\x0F\x5F\xC4\xA3\x78\xF7\x3F\x0D\xF0\xEC\x45\x3E\x15\xAF\xFE\xA7\x01\x1E\xBD\xF0\xAF\x78\xF3\x3F\x0D\xF0\xE4\x85\x7F\xC5\x8B\xFF\x49\x80\x07\x2F\xF0\x8A\xF7\xFE\xB3\x00\xCF\x5D\xC4\xA3\x78\xED\xE2\x3D\xC8\x63\x57\xBC\xF5\x30\x4F\x3D\xC8\x4B\x57\x3C\x74\x81\x57\xBC\x73\x0C\xF1\xCC\x15\xAF\x3C\xCC\x23\x0F\xF2\xC6\x15\x4F\x5C\xE0\x83\xBC\x70\xC5\x03\x17\x61\x15\xEF\x5B\xF8\x51\x3C\x6F\x0C\xF1\xBA\x15\x8F\x5B\xE0\x15\x6F\xFB\x3E\xCF\xF3\x14\x4F\xFB\x29\xD1\xCF\x25\x2F\x9B\xE6\x2D\xC9\xC3\xDE\xE4\x79\x9E\xE2\x5D\x0B\xBC\xE2\x59\x9F\x71\x7A\x83\x57\x2D\xD2\x55\x3C\xEA\xEB\x3C\xCF\x53\xBC\xE9\xF6\xD3\x9B\x79\xD2\x8A\x17\x2D\xFC\x2B\x1E\xF4\x6C\xCF\xF3\x14\xEF\x79\x7E\x80\xE7\xBC\x30\xC0\x6B\x16\xE9\x06\x79\xCC\x41\xDE\x72\x90\xA7\x1C\xE4\x25\x07\x79\xC8\x41\xDE\x71\x90\x67\x1C\xE4\x15\x07\x79\xC4\x41\xDE\x70\x90\x27\x1C\xE4\x05\x07\x79\xC0\x41\xDE\x6F\x90\xE7\x1B\xE4\xF5\x06\x79\xBC\x41\xDE\x6E\x90\xA7\x1B\xE4\xE5\x06\x79\xB8\x41\xDE\x6D\x90\x67\x1B\xE4\xD5\x06\x79\xB4\x41\xDE\x6C\x90\x27\x1B\xE4\xC5\x06\x79\xB0\x41\xDE\x6B\x90\xE7\x1A\xE4\xB5\x06\x79\xAC\x41\xDE\x6A\x90\xA7\x1A\xE4\xA5\x2A\x1E\xAA\x68\x3B\xC5\x3B\x15\xEF\x41\x9E\xA9\xE2\x95\xD6\x03\x3C\xD2\x53\x03\xBC\xD1\x53\x02\x3C\xD1\x39\x01\x5E\xE8\x45\x01\x1E\xA8\x2D\x79\x9F\x02\xC4\xBE\xE2\x9F\x24\x1F\xD4\x0E\xF1\x3F\x83\x7C\xCF\x47\x75\x80\x47\x25\x1F\xF3\x0B\x3A\x80\x80\x93\x69\x0D\x05\xF8\x92\xC4\x7F\x59\x07\xF8\xB2\x4C\xF7\x2B\x3A\xC0\x57\x25\xFE\x6B\x3A\x80\x00\x41\xF7\x3F\xA6\x33\x44\xDA\x00\xBE\xAE\x33\xCC\x14\x7B\x0A\x9D\x41\xF8\x7F\x5C\x07\x10\x70\xF0\x4D\xCF\x7B\x42\x07\xD8\x21\xF1\x3B\x75\x00\x01\x62\x4F\xF1\x4D\xC9\x7F\x15\x74\xBB\xE2\xBB\x8A\xBA\xFA\xB6\x0E\xF0\x5D\xE9\xFF\x7B\x3A\x80\x80\x31\x62\x8F\xA2\x03\x3C\x25\xF1\xFF\xAA\x03\x08\xB8\x6E\x22\xC0\xF7\x03\xFC\xD9\x20\x5F\x36\xC8\x8F\x55\x7C\xD8\x58\x88\xFF\xAA\xF8\xAE\x62\x7F\x12\xE4\xB7\x06\xF9\xAC\x41\xFE\xAA\xE2\xAB\x12\x0D\x18\xE0\xA7\x06\xF9\xA8\x41\xFE\x69\x90\x6F\xAA\xF8\xA5\xFF\xE4\x79\x5E\x90\x4F\xAA\xF8\xA3\x62\x7E\x50\x7C\xD1\x83\x9E\xE7\x29\x7E\xA8\x48\x4B\xF1\x41\x89\x07\x2A\xF9\x9F\x2B\x03\x7C\x4F\x5B\xF2\x3B\xA7\x4B\x7E\xE7\x0C\x03\x40\xC0\xF2\xF1\x00\x67\x1A\x0C\x02\x7F\x96\x01\x20\x60\xD9\x78\x80\xB3\x0D\x80\xD9\x12\x7F\xA1\x01\x20\xE0\xE2\xF1\x00\x17\x19\x00\x73\x24\xDE\x31\x00\xE6\x0A\x18\x0F\x30\xCF\x00\x58\x2C\xF1\x97\x18\x00\x02\x2E\x18\x0F\xB0\x24\xC0\x67\x0D\xF2\x57\xD7\x18\x00\x6B\x0D\xCE\x7F\xB7\xC1\x30\x69\x3C\x40\x8F\xE4\xB7\x0A\xBC\xE2\xB3\x8A\x77\xC5\x5F\x1D\x13\xE0\xAB\x8A\x3A\x09\xF2\x53\x15\x1F\x55\xF8\x57\xFC\xD3\xCE\x10\xDF\x54\xF1\x4B\xD7\xB4\x35\xF8\xA4\xE2\x5D\xF1\x47\xC5\x7B\x90\x2F\x1A\xE4\x87\x2A\x3E\xE8\xAE\x8F\x35\xF8\x9F\xB1\x10\xDF\x33\xC8\xEF\x0C\xF2\x39\x83\xFC\xCD\x5B\xDA\x00\x6E\x91\x7C\xCD\x5B\x25\x8F\x53\xE4\x39\xC8\xDB\x0C\xF2\x34\x83\xBC\xCC\x20\x0F\x53\xF1\x2E\xE7\xBC\xEA\x79\x8A\x67\x29\xDE\x83\xBC\xCA\x20\x8F\xB2\x89\x37\x79\x32\xC0\x9A\x93\x39\x5D\xF7\x64\x06\x41\xE3\xAE\x95\xBC\x4A\x81\x0F\xF2\x28\x15\x6F\xD2\xF9\xB5\xE7\x29\x9E\x24\x51\xC9\x01\x5E\xA4\xE2\x41\x8A\x7A\x08\xF2\x1E\x15\xCF\x51\xE0\x83\xBC\x46\xC5\x63\x14\xF8\x20\x6F\x51\xF1\x14\x05\x3E\xC8\x4B\x54\x3C\x44\x81\x0F\xF2\x0E\x83\x3C\xC3\x20\xAF\x30\xC8\x23\x54\xBC\x41\x11\x56\xF1\x04\xC5\xBB\xE2\x05\x8A\x77\xC5\x03\x14\xEF\x41\xDE\x9F\xE2\xF9\x09\x7C\x90\xD7\x17\xE4\xF1\x05\x79\x7B\x41\x9E\x5E\x90\x97\x17\xE4\xE1\x05\x79\x77\x8A\x67\x27\xE2\x0F\xF2\xEA\x14\x8F\x4E\xE0\x83\xBC\x39\xC5\x93\x13\xF8\x20\x2F\x2E\xC8\x83\x53\xBC\x37\xE1\x47\xF1\xDC\xBE\x22\xE6\x5E\xC9\x6B\x7B\xC3\xF3\x3C\xC5\x63\xBB\x3D\xC4\x5B\x53\x3C\x35\x31\x77\x29\x5E\x9A\xF0\x1F\xE4\xA1\x29\xDE\x99\x88\x5F\xF1\xCC\xEE\xF5\x3C\x2F\xC8\x2B\x0B\xF2\xC8\x14\x6F\x4C\xF8\x57\x3C\xB1\x7B\xC4\x1E\x5A\xF2\xC2\xC4\x9C\x16\xE4\x81\x29\xDE\xD7\x88\x98\xEB\x24\xCF\xEB\xBD\x9E\xE7\x29\x5E\x97\xC8\x8F\xE2\x71\xDD\xF2\x6B\xCF\x53\xBC\xAD\xEC\xAF\x3C\x4F\xF1\xB4\x3E\xF1\x2B\xCF\x53\xBC\xAC\xB6\x10\x0F\x4B\xF1\xAE\xD2\x21\x9E\x95\xE2\x55\x89\xB9\x45\xF1\xA8\x4E\x0A\xF3\xA6\x24\x4F\xAA\x33\xC4\x8B\x52\x3C\x28\x31\x2E\x14\xEF\x29\xFE\x9C\xE7\x29\x9E\xD3\x49\x2F\x7B\x9E\xE2\x35\x9D\xF8\x9C\xE7\x05\x79\x4C\x41\xDE\x92\xE2\x29\x89\x3C\x28\x5E\xD2\x6F\xF7\x78\x9E\xE2\x21\xE9\xCF\x79\x9E\xE2\x1D\xBD\xB8\xC7\xF3\x14\xCF\xE8\xD7\x7B\x3C\x4F\xF1\x8A\x44\x1E\x14\x8F\x48\xC4\xA3\x78\x43\xDF\xDC\xE3\x79\x8A\x27\x24\xFC\x28\x5E\x10\xD1\xC9\x92\x07\x44\xE3\x3A\xC0\xFB\x09\xF2\x7C\x82\xBC\x9E\x39\x0F\x00\x5C\xF2\x00\xFB\x5F\xF2\x00\xC3\xEC\x0D\x9E\xB7\x54\xF2\x7E\x04\x5E\xF1\x7C\xC4\xBB\xE2\xF5\xD0\xDE\x50\xF2\x78\xC4\xBB\xE2\xED\x88\x3C\x28\x9E\xCE\x9B\x43\x9E\xA7\x78\x39\xC2\x8F\xE2\xE1\x08\x3F\x8A\x77\xF3\xC2\xD5\x9E\xA7\x78\x36\xC2\x4F\x90\x57\xA3\x78\x34\x02\xAF\x78\x33\x8F\x5F\xED\x79\x8A\x27\xF3\xE8\xD5\x9E\xA7\x78\x31\x5F\xBE\xDA\xF3\x14\x0F\x66\xC7\xD5\x9E\xA7\x78\x2F\xC2\xBF\xE2\xB9\xEC\xBB\xDA\xF3\x14\xAF\xE5\xB9\xAB\x3D\x4F\xF1\x58\x7E\x7F\xB5\xE7\x29\xDE\xCA\x16\xBD\xC1\x53\x11\xE9\x2A\x5E\x0A\x9D\x99\x49\x1E\x8A\xE8\x27\x8A\x77\xF2\xC4\xD5\x9E\xA7\x78\x26\x4F\x0D\x79\xDE\x93\x5B\x01\xBE\x2F\xCB\xF8\xDA\x56\x80\xD7\xE5\x7B\x90\x87\x12\xE4\x9D\x28\x9E\x89\x1D\xE6\x95\x48\x1E\x49\x7B\x88\x37\xA2\x78\x22\xED\x21\x5E\x88\xE2\x81\xCC\x09\xF1\x3E\x14\xCF\xC3\x0E\xF1\x3A\x14\x8F\xC3\x0E\xF3\x36\x24\x4F\x23\x17\xE2\x65\xAC\x53\xC7\x29\x2F\x21\x2C\x79\x09\x61\xE5\x4B\x08\x57\xBD\x84\x70\xFD\x4B\x08\x5D\x2F\x05\x4E\x88\x4E\x40\x38\xE9\x04\x84\x89\x27\x20\x9C\x71\x02\x42\xE7\x09\x08\x17\x9D\x80\x00\x17\x23\x2C\x58\x80\x00\xA6\x61\x9A\xA6\x65\xA6\xCC\x93\xCC\x36\xB3\xCD\x3C\xD9\x3C\xD9\xB4\xCD\x8B\x4C\xC7\xBC\xC4\xBC\xDC\xAC\x98\xEB\xCD\x9F\x9B\xBF\x34\x3D\x73\x65\x74\x38\xBA\x25\xF6\xA5\xD8\x57\x62\x4F\xC6\x9E\x89\xBD\x18\xFB\x4D\xCC\x8B\xAD\x3C\xFC\xCD\xC3\xCF\x1C\x9E\xFD\xD6\xCA\xB7\x3E\xF0\xD6\x87\xDE\x7A\xF2\xAD\xFD\x6F\xBD\xF5\xD6\xE0\x91\xFA\x91\x0D\x47\xDE\x3C\xA2\x92\x5F\xD3\x82\xB0\xA9\x05\x01\xF6\x21\xE8\xFB\x10\x22\xFB\x10\x32\xFB\x10\xB2\xFB\x10\x72\xFB\x10\x26\x49\x38\x4D\xC2\xE4\x7D\x08\x33\xF6\x21\x9C\xB5\x0F\xE1\x9C\x7D\x08\xE7\x49\x38\x5F\xC2\x5C\x09\x0B\x25\x2C\x96\xB0\x44\xC2\xB2\x7D\x08\x2B\xF6\x21\xBC\x73\x1F\xC2\xA5\xFB\x10\x56\x4B\xB8\x42\xC2\x55\x12\xAE\x91\x70\x9D\x84\x2E\x09\xEE\x3E\x84\xB5\xFB\x10\x7A\x24\xF4\xEE\x43\x28\xEE\x43\x28\xED\x43\xA8\xEE\x43\xA8\xEF\x43\xB8\x61\x1F\xC2\xC6\x7D\x08\x37\xEF\x43\xD8\x24\x61\x64\x1F\xC2\xFB\xF6\x21\xBC\x7F\x1F\xC2\x27\xF7\x21\x7C\x66\x1F\xC2\xFD\xFB\x10\x3E\xBB\x0F\x61\xDB\x3E\x84\xED\xFB\x10\xBE\xB1\x0F\x61\xFF\x3E\x84\x97\xF7\x35\xDA\x67\x7A\xE4\xF2\xC8\x55\x91\x3B\x23\x7F\x1F\xF9\x40\xA4\xF5\xF0\x09\x87\xC1\x42\x38\xDB\x42\x38\xCF\x42\xB8\xC8\x42\x80\x09\x4B\x26\xAC\x9C\x70\xF3\x84\x2D\x0F\xBC\xF1\x00\xF4\x21\x9C\x5B\x6C\xE6\xDD\xEC\x2A\x22\xFC\xB1\x88\xB0\xBF\x88\xF0\x5A\x11\x61\xE7\x84\x3F\x4F\x78\x73\x82\x37\x41\xFD\x3E\x12\xBF\x25\xBE\x39\x7E\x57\xFC\x03\xF1\x7B\xE2\x1F\x8E\x3F\x18\x7F\x28\xFE\x48\xFC\x0B\xF1\x2F\xC4\xBF\x12\x7F\x2C\xFE\x78\xFC\x3B\xF1\x27\xE3\x4F\xC5\x7F\x10\xFF\x61\xFC\xF9\xF8\xF3\xF1\x9F\xC7\x7F\x11\xFF\x55\xFC\x37\xF1\xDF\xC5\x8F\xC4\x1B\x07\x80\x08\xB1\x36\x84\x44\x1B\xC2\xCC\x36\x84\x73\xDB\x10\xE6\xB7\x21\xAC\x6C\x43\x28\xB5\x21\xFC\x97\xF6\x7B\x2D\x32\x7D\xDA\xF4\x2D\xD3\x77\x4C\x27\xFF\x69\x84\x65\x69\x84\x4B\xD3\x08\x95\x34\xD2\xF9\xA7\x00\x68\xF7\xDA\xE1\xA4\x09\x27\x9D\x7C\x52\xE6\xA4\x39\x27\xAC\x3A\xA1\x51\x82\x0C\x0C\xC3\x7D\xF0\x30\x3C\x0C\x60\x46\xA8\x0F\xFE\xC2\xFC\x85\x39\x26\x3A\x26\xFA\xEB\xD8\xAF\x63\x2D\xE3\x5B\xC6\x5B\xF6\x18\xFB\x74\x7B\xAA\xBD\xC6\xEE\xB1\xD7\xD9\x15\x1B\x5E\xF0\x0E\x78\x47\xBC\x23\x07\xBD\xC3\xDE\x13\x69\x04\x01\x4F\x4B\xE8\x3C\x0B\xE1\xDC\xB3\x10\xB6\x3C\x8E\x70\xFF\xE3\x08\xB5\xDD\x08\x37\xED\x46\x40\x68\x21\xB0\xA1\x05\x86\xA1\x85\x24\x0A\xDF\xF2\xC6\x90\x94\xC5\x5B\x82\x18\x1D\x37\xFC\x8E\x47\xC6\xBF\x36\x7E\x8B\x86\xB0\x4B\x6B\xD4\x71\x67\x04\xA1\x3B\x82\x50\x91\xA0\xF2\xAB\x9E\xFF\x4E\xBE\x55\x9E\x83\xED\xF7\xBF\x9F\x7F\xF9\x1C\x87\x70\xD6\x71\x08\x33\x8F\x43\x58\x78\x1C\x02\x3C\x34\xF3\xA1\x39\x0F\x2D\x7B\x68\xE5\x43\x57\x3D\x74\xED\x43\x5D\x0F\x6D\x69\x7D\xAD\xF5\x40\xEB\x5B\xAD\x83\xDB\x77\x6C\x7F\x64\x0C\xC2\x0F\xC6\xA0\x98\xD0\x0E\x34\x22\x08\xD5\x83\xCF\xCB\xD8\x06\xDB\xE1\x31\xF8\x06\xEC\x80\xE7\xE1\x79\x78\x03\xDE\x80\xC7\xB4\x5F\x69\xFF\xAE\xBD\xA4\xFD\x41\xF3\xB4\xBA\x5E\xD7\x37\xE9\x9B\xF4\xCD\xFA\x66\xFD\x76\xFD\x76\x3D\x62\x46\xCC\xA4\x99\x34\x27\x98\x13\xCC\x8C\x99\x31\xE7\x98\x73\xA8\x3E\x44\x5D\xF4\xC4\xF3\xF1\x8B\x5A\x2E\x6A\xD9\x33\xE6\x85\x31\x87\xD3\x87\xD3\xBF\x7F\xC7\x1F\xDF\x71\xF6\x09\x33\x4F\xD0\xC6\xEB\xE3\x23\xE3\x23\xE3\x9F\x69\x7B\xA6\xED\xC5\xB6\x17\xDB\x5E\x6A\xFB\x43\xDB\xBE\xB6\x3F\xB7\xBD\xD6\xF6\x46\xDB\xA1\xB6\x43\x6D\x60\x5B\x76\x0B\xD5\xDD\xA0\x5D\xB7\x37\xD9\x23\xF6\xDD\xF6\x16\x7B\x87\x0D\xD9\x53\xB2\xA7\x65\x27\x67\xA7\x64\xA7\x67\xCF\xCC\x2E\xCB\xAE\xC8\x76\x65\xDF\x97\x7D\x7F\xF6\x9E\xEC\xA9\x93\xE6\x4C\xBA\x64\xD2\x9A\x49\xDE\x69\x30\xB9\x76\xC6\x86\x33\x3E\x72\xC6\xC7\xCE\xF0\xCE\x80\xA9\x97\x4E\xDD\x3F\xED\xF0\x34\xE8\x34\x3A\xCD\x4E\xB3\x33\xD6\x69\x77\x76\x76\x9E\xD3\x79\x41\xE7\x45\x9D\x9F\xEC\xBC\xB7\x73\x4B\xE7\x96\xCE\xC3\x9D\x6F\x76\xDE\x33\xFD\xBE\xE9\x3B\xA7\xFF\x66\xBA\x3D\xA3\x6B\xC6\xF0\x8C\x1F\xCD\xF0\x66\x78\x33\xAE\x3C\xD3\x3B\x73\xE7\x32\x6F\x19\x6C\xCD\x6E\xBD\x63\xEB\x9D\x5B\x3B\x1F\x3C\xEF\xC1\xA9\xDB\xA6\x6E\xFB\xD1\xB6\x1F\x6D\xBB\x66\xFB\x35\xDB\x8B\xDB\xFB\xB7\x5F\xF8\xD6\x45\x6F\xA5\x8F\x9C\x78\xA4\xF3\xC8\xA5\x47\x56\x1F\x59\x77\xA4\xEF\x48\xFF\x11\xD1\xD6\xE8\xD9\xDE\xF9\xDE\x1C\xEF\x1A\x2F\xEF\x0D\x7A\x83\xDE\xA7\xBD\xFB\xBC\xBD\xDE\xEF\xBC\xFF\xF4\xF6\x79\x07\xBD\xB7\xB8\x1F\x20\x82\x86\x08\x51\x44\x38\x13\x11\xCE\x41\x84\x8B\x10\xE1\x1E\x44\xB8\x17\x11\x76\x21\xC2\x9B\x88\xF0\xA2\x86\x70\x58\x43\xF8\x75\x05\xE1\x37\x95\x46\xFF\x80\x5D\x08\xAF\xED\x42\x80\xDD\x08\xA7\xEE\x46\x98\xBC\x1B\x61\xDD\x6E\x84\xA2\xEC\x37\xB7\xEC\x46\xB8\x75\x37\xC2\x5D\xBB\x11\xB6\xED\x46\xF8\xFC\x6E\x84\xFF\xDC\x8D\xB0\xF7\xC7\x08\x7F\xFE\x31\x02\x3C\x83\x70\xF9\x33\x08\x6B\x9E\x41\xB8\xF1\x19\x04\x78\x16\x61\xF5\xB3\x08\x97\x3F\x8B\x70\xEF\xB3\x08\x9F\x7E\x16\xE1\xBE\x67\x11\xFE\x97\x84\xCF\x3E\x8B\xF0\xC0\xB3\x08\xDB\x9E\x45\xF8\xDC\xB3\x08\x9F\x7F\x16\xE1\xB1\x67\x11\xBE\x21\xE1\x89\x67\x11\xBE\xFD\x2C\xC2\x77\x9F\x45\x88\xEC\x41\x88\xEE\x41\x48\xEC\x41\x48\xED\x41\x18\xB7\x07\xE1\x1D\x7B\x10\xDA\xF6\x20\x9C\xBC\x07\xE1\xBC\x3D\x08\xE7\xEF\x41\xB8\x70\x0F\xC2\x9C\x3D\x08\x17\xEF\x41\x58\x20\xE1\x92\x3D\x08\x2B\xF7\x20\x5C\xBA\x07\xE1\xB3\xCF\x21\x3C\xF8\x1C\xC2\x0F\x9E\x47\xF8\xE1\xF3\x08\xDE\xF3\x08\xEB\xFF\x80\xF0\xA5\x3F\x20\xE0\x1F\x11\x66\xFF\x11\x01\xF6\x23\x9C\xB1\x1F\xA1\x73\x3F\xC2\x87\xF7\x23\x6C\xD9\x8F\xF0\xF9\xFD\x08\x8F\xEE\x47\xD8\xB1\x1F\xE1\x9B\xFB\x11\x7E\xB4\x1F\x61\xF7\x7E\x84\xD7\xF6\xF3\x41\xE6\xC3\x2F\x23\xFC\xEE\x65\x04\xEF\x65\x04\x7C\x05\x41\x7B\x05\x21\xFD\x0A\xC2\xF9\xAF\x20\xCC\x79\x05\x61\xD1\x2B\x08\x2B\x5F\x41\x78\xE7\x2B\x08\x6B\x5E\x41\xC8\xBF\xC2\x07\x9D\xCF\xBF\x8E\xB0\xF7\x75\x84\x3F\xBC\x8E\xB0\xFF\x75\x84\x37\x5F\xE7\xC3\xCD\xDA\x1B\x08\x23\x6F\x20\xBC\xF0\x06\xC2\xDE\x37\x10\x7E\xFF\x06\xD2\xC5\x1D\xEB\x00\x42\xFA\x00\xC2\xC2\x03\x08\x2B\x0F\x20\x5C\x75\x00\x61\xCD\x01\x84\xDB\x0F\x20\xDC\x73\x00\xE1\xE1\x03\x08\x8F\x1C\x40\x78\xF4\x00\x02\x1C\x44\xB8\xF1\x20\xC2\x4D\x07\x11\x7E\x70\x10\xE1\xE9\x83\x08\xAB\x0E\x21\xAC\x39\x84\x30\x70\x08\x61\xF0\x10\x42\xFD\x10\xC2\x8D\x87\x10\x6E\x3A\x84\x30\x72\x08\xE1\xB6\x43\x08\xF7\x1C\x42\x78\xF0\x10\xC2\x23\x87\x10\xBE\x72\x08\x61\xE7\x21\x84\x6F\x1D\x42\xD8\x75\x08\xE1\xB9\x43\x08\x70\x18\xE1\x43\x87\x11\xFE\xE1\x30\xC2\xF7\x0F\x23\xEC\x3F\x8C\x70\xF0\xF0\xD1\xF3\xCA\xAF\xF5\x3F\xE9\x23\x53\xFE\x3C\xE5\xE0\x14\x6F\x0A\xBC\xE0\xFD\x0A\x6C\x84\x8F\xFF\x1D\x02\x4C\x42\x28\x4F\x42\x18\x9C\x84\x50\x9F\x84\x30\x32\x09\x61\xDE\x29\x08\x10\x43\x88\xC4\x10\x62\x12\x12\x31\x84\xB3\x63\x08\xE7\xC4\x10\xCE\x8D\x21\x5C\x20\xE1\x22\x09\x60\xDC\x6A\xDC\x6E\x4C\x8B\x8C\xB4\xDD\xD1\x76\xC6\xC4\x33\x26\xDE\x38\xF1\xC6\x89\x7B\x3B\xBC\x8E\x39\x0F\xDC\xF7\xC0\xD4\x23\xD3\x8E\x80\xA4\xAD\xFC\x27\xBE\x32\x7E\x59\xBC\x3B\xBE\x2E\x3E\x1C\xDF\xFB\xA0\xF7\x20\x9C\x88\x60\x9E\x88\x10\x97\x90\x3C\x11\xA1\xF5\x44\x84\xE3\x4F\x44\x78\xC7\x89\x08\xE3\x4F\x44\x38\xFB\x44\x84\x73\x4E\x44\x38\xF7\x44\x84\xF3\x4F\x44\x58\x70\x22\xC2\xCA\x13\x11\xAE\x3A\x11\x01\xC6\x23\x9C\x3F\x3E\x10\xFF\xE3\x08\xC5\xC7\x11\x06\x1F\x47\xB8\xF9\x71\x84\x91\xC7\x11\xEE\x78\x1C\xE1\x9E\xC7\x11\x3E\xFE\x38\xC2\xBD\x8F\x23\xDC\xF7\x78\xC0\x7F\x27\xC2\xD4\xB3\x1A\xF3\x34\x44\x10\x4E\x8B\x20\xEC\x1D\x83\xC4\xA3\x83\xB1\x8B\xC6\x5E\x32\x76\xD9\xD8\x95\x63\x2F\x1F\x7B\xE5\xD8\x2B\xC7\x5E\x3D\xF6\xBA\xB1\x6B\xC6\xDE\x31\xF6\xCE\xB1\x7F\x3F\xF6\x9E\xB1\x8F\x8C\xFD\xC2\xD8\x2F\x8F\xFD\xDA\xD8\x7F\x19\xBB\x73\xEC\xCE\xB1\xDF\x1A\xFB\xDD\xB1\x4F\x8D\x7D\x6E\xEC\x0B\x63\xD3\xAD\x63\x5B\x8F\x6F\x1D\xDF\x7A\x75\xEB\x75\xAD\x37\xB7\x8E\xB4\x7E\xBC\x75\xA4\xE3\xA3\x1D\x5B\x3A\x1E\xE8\x78\xB0\xE3\xF3\x1D\x8F\x74\x7C\xA5\xE3\x6B\x1D\xFF\xD2\xB1\xB3\xE3\x7B\x1D\x4F\x75\xFC\xB0\x63\x57\xC7\x73\x1D\x2F\x74\xFC\xB2\x03\xB7\x9B\xDB\xE3\xDB\x5B\xB6\x1F\xB7\xFD\x1D\xDB\xED\xED\xA7\x6E\x6F\xDF\x3E\x75\xFB\x96\xF4\x77\xD3\x4F\xA6\x9F\x4C\x3F\x9D\x7E\x3A\xBD\x2B\x7D\x28\xFD\x66\xDA\x4B\xDF\xD3\xF6\xF5\xB6\x27\xDA\x76\xB4\x41\xC7\x29\x1D\xA7\x75\x9C\xD6\x21\x1E\x98\x32\x75\x4A\xE7\x94\xEB\xA7\xC0\x5E\x04\x73\x2F\x42\x6C\x2F\xC2\xF8\xBD\x08\x13\xF6\x22\x64\xF7\x22\xFC\xDD\x5E\x84\x49\x7B\x11\x4E\xDD\x8B\x70\xFA\x5E\x84\x4E\x1D\xE1\x12\x1D\x01\x5A\x11\xF4\x56\x84\x48\x2B\x42\xB2\x15\x61\x4C\x2B\x42\xBA\x15\xA1\xB5\x15\xA1\xBD\x15\xE1\xF4\x56\x84\xCE\x56\x84\x19\xAD\x08\x67\xB6\x22\x9C\xDD\x8A\x70\x5E\x2B\xC2\x05\xAD\x08\x17\xB7\x22\x2C\x6C\x45\x58\xD4\x8A\xB0\xA4\x15\x61\x59\x2B\xC2\x4A\x09\x57\x48\xB8\xAE\x15\xA1\xBB\x15\x61\x5D\x2B\x42\xA9\x15\x61\xB0\x15\xA1\xDE\xDA\xA8\xFF\x41\xBD\xA6\x0F\xE9\x37\xE8\x37\xE9\xEF\xD6\x87\xF5\x61\xFD\x56\xFD\x56\xFD\x36\xFD\x36\xFD\x0E\xFD\x4E\xFD\x2E\xFD\x2E\xFD\x6E\xFD\x33\xFA\xFD\xFA\x8B\xFA\x7E\xDD\xD3\x4F\x9D\x78\xFA\xC4\xEB\x26\xBA\x13\xD7\x4D\x2C\x4E\xDC\x31\x71\xC7\x44\xC8\x1C\x9F\x19\x9F\x99\x98\xB1\x33\xF3\x33\x8B\x32\xCB\x32\x2B\x33\x57\x64\xAE\xCA\x5C\x95\xB9\x26\x73\x4D\xE6\xBA\xCC\x75\x99\xAE\xCC\xBB\x33\x23\x99\x2F\x65\xBE\x92\xF9\x4E\xE6\x7B\x99\x67\x32\xCF\x65\x7E\x96\xF9\x45\xE6\x4F\x99\x57\x32\xAF\x66\x5E\xCF\x1C\xC9\x9C\x9A\x3D\x35\x9B\xDF\x9E\xDF\x1E\x5C\xBF\xE7\x20\xC2\xDD\x88\xB0\x45\x02\xFC\x18\x61\xFE\x8F\x1B\xF9\x7D\x4F\xE2\x96\xC4\xE6\xC4\xDF\x27\x3E\x90\xF8\x60\xE2\xC3\x89\x07\x13\x0F\x25\x1E\x49\x7C\x21\xF1\xC5\xC4\x97\x13\x8F\x25\x1E\x4F\x7C\x37\xF1\x64\xE2\x5F\x13\x3F\x48\x3C\x9D\xD8\x95\xD8\x95\xD8\x9B\xF8\x4D\xE2\x77\x89\x97\x13\x07\x13\x5E\xC2\xA7\x9F\x27\x22\xE4\x27\x22\xF4\x4E\x44\xE8\x9B\x88\x50\x9C\x88\x70\xF7\x44\x84\x7B\x26\x22\x7C\x70\x22\xC2\x87\x27\x22\xFC\xE3\x44\x84\x2D\x13\x11\xB6\x4D\x44\xC0\x84\x9E\x88\x24\x12\x89\x31\x89\x74\xA2\x35\xD1\x9E\x38\x3D\xD1\x99\x98\x91\x38\x33\x71\x76\x62\x66\xE2\xDC\xC4\x79\x89\x0B\x12\x17\x24\x2E\x4C\xCC\x4D\x2C\x4C\x2C\x4A\x2C\x49\x2C\x4B\xBC\x33\xF1\xCE\xC4\x55\x89\x6B\x13\xD7\x27\xAE\x4F\xAC\x4B\x6C\x48\x8C\x4C\xFD\xF8\xD4\x4F\x4E\xFD\xF3\x54\x98\xB6\x67\x5A\xA4\x33\xD2\x19\xED\x8C\x76\x66\x3B\x27\x77\x9E\xDB\x79\x7E\x27\x9C\xB5\x63\x19\x2C\x7F\xF3\x3E\x38\x38\x70\x68\xF0\xD0\x4F\x0E\x89\xFC\xED\xAF\x20\xBC\x5C\x41\x92\x22\xFA\xC5\x03\x1A\xC0\x56\x0D\xCE\xFA\xAA\x06\x73\xBE\xAA\xC1\xC4\xAF\x69\x60\x7F\x4D\x83\xCF\xFC\x50\x83\x47\x7E\xA8\xC1\xDE\xDF\x6B\x00\x07\x34\x98\x78\x88\x25\x8E\x2E\x69\xD5\x01\x8E\xF3\x8E\x9B\xDA\x39\xAD\x73\xFA\xF4\xBB\xA7\xC3\x8C\x93\x67\xAC\x99\xB1\x69\xC6\x9A\x6D\x37\x6F\x83\xCF\xDD\xFF\xFC\x23\xCF\x7F\xEF\xF9\x1F\x3C\x7F\xF8\xF9\x2D\xDE\xBF\x78\xDF\xF2\x9E\xF4\xBE\xEF\xFD\xC8\xFB\xB1\xF7\xBC\xF7\x53\xEF\xE7\x0D\x1A\x2A\x85\x70\x5A\x0A\xA1\x33\x85\x70\x5E\x0A\xC1\x3E\xE1\xAC\x13\x82\xED\xB3\x37\x86\xF0\x4A\x0C\xE1\xD5\x18\xC2\x6B\x31\x84\xC3\x31\x04\x2F\xD6\x68\x9F\x0F\x46\x9E\x8C\xEC\x8A\xFC\x57\xE4\x4F\x91\x57\x23\x13\x0F\xCF\x3C\x7C\xEE\xE1\x0B\x0E\x5F\x78\xF8\xC2\xC3\x73\x0E\x3B\x87\xE7\x1D\xBE\xF8\xF0\x82\xC3\x2B\x0E\x3B\x9D\x1F\xED\xFC\x54\xE7\x7D\x24\xAE\x88\x8F\x20\x9C\xFC\x28\xC2\xCA\x47\x11\x2E\x7D\x14\x1B\x97\x3F\x55\x7F\x88\x21\x5C\x16\x43\xB8\x22\x86\xD0\x25\xD2\xD1\x2B\xFA\x66\xE3\x36\x63\x89\x79\x99\x39\x68\x0E\x9A\xEF\x8C\xAF\x8E\x3F\x72\xD2\xCE\x93\x76\xB5\xFD\xB8\xED\xD9\xB6\xBD\x6D\xBF\x6E\xFB\xCF\xB6\x3F\xB6\xFD\xB1\xED\xD5\xB6\x57\xDB\x0E\xB4\x1D\x6C\xDB\x39\xF1\xE0\xC4\xC3\x13\xBD\x89\x49\x3B\x65\xEF\xB2\xF7\xDB\xA7\x77\x76\x74\x7E\xBC\xF3\x13\x9D\x70\x64\xCC\x11\xFB\x48\xC7\x91\xB7\x10\x41\xC0\x5E\x8D\xE1\xFC\x56\x86\xDE\xDD\x08\x85\xDD\x08\xEF\xDA\x8D\xF0\xDE\xDD\x08\x9B\x77\x23\xBC\x7F\x37\xC2\x43\xBB\x11\x1E\xDE\x8D\x00\xD8\x02\x7F\xC2\x16\x58\x63\x21\x54\x2D\x84\x1B\x2D\x84\x61\x0B\x61\x8E\x85\x70\x99\x85\x70\xA5\x85\xD0\x65\x21\x8C\x6C\x7B\x7A\xDB\xAE\x6D\x3F\xD9\xF6\xCB\x6D\xBF\xDA\x26\xCA\x33\x92\x66\x9E\xBE\xA2\x4F\x47\x92\x77\x25\xEF\x4E\xDE\x93\xFC\x50\xF2\xC1\xE4\x43\xC9\x2F\x26\xBF\x9C\x7C\x2C\xF9\x78\xF2\x3B\xC9\xEF\x25\x9F\x4A\x7E\x3F\xF9\x74\x72\x4F\xF2\xB9\xE4\x2F\x93\xBF\x4C\xEE\x4D\xFE\x26\xF9\xBB\xE4\x9F\x92\x2F\x27\x5F\x49\x06\xDB\xE3\x33\x9D\x87\x3A\xDF\xEA\xF4\x3A\xF7\x4F\xF7\xA6\xEF\x9A\x71\x64\x06\x9C\x79\xC5\x99\xEB\xBC\x8A\xB7\xDE\xFB\x14\xB7\xE9\x23\xCC\x48\xED\x79\x14\xA1\x57\xD4\xEF\xB6\x8E\x6D\xD3\xB6\x4D\xDB\xE6\x47\x90\x40\xD0\x13\x08\x91\x04\x82\x99\x40\x48\x26\x10\x5A\x13\x08\xC7\x27\x10\x4E\x48\x20\x9C\x98\x40\x38\x3B\x81\x70\x6E\x02\x61\x56\x02\xE1\xA2\x04\xC2\xA2\x04\xC2\xCA\x04\xC2\x95\x09\x84\xDF\x56\x18\xE0\xFD\x08\x7B\xEE\x42\x18\x39\xE1\x17\x27\xEC\x3D\xE1\xA5\x13\xF6\x9F\x70\xF0\x84\xBD\x27\x7A\x27\xC2\x58\x84\xE3\xC6\x22\xB4\x8E\x45\xB8\x70\x2C\xC2\x23\x63\x11\xFE\x6B\x2C\xC2\xFE\xB1\x08\x07\xC7\xA2\x2F\x1F\xF4\x10\x3C\x04\x5F\x86\x2F\xC3\xD7\xE1\xEB\xB0\x13\x9E\x83\x17\xE0\x75\x38\x00\xDF\xD4\xBE\xA7\xED\xD6\xF6\x6A\xBF\xD5\xFE\xA0\xFD\x41\xDB\xA7\xED\xD3\x06\xF5\xBA\xBE\x21\x30\x5F\x1D\x3D\x57\xBD\xA6\xBF\xA1\xBF\xC7\xB8\xD3\x98\x16\xE1\xBD\xE0\x55\x91\x35\x91\x3B\x64\x9F\x7C\x25\x62\x9B\x97\x98\x65\xDA\x87\x3F\x63\xEE\x31\xF7\x98\xFF\x6E\xFE\xCE\xDC\x67\xFE\xC9\x3C\x64\xBE\x69\x7A\xA6\x67\xA6\xA3\xE9\xE8\xD8\xE8\xB4\xE8\xB2\xE8\x67\xA3\x8F\x46\x1F\x8D\x7E\x3F\xFA\x5F\xD1\x57\xA3\xAF\x45\x0F\x45\x0F\x45\x21\x76\x7C\xEC\xA4\xD8\x49\xB1\x49\xB1\x49\xB1\xF6\x58\x7B\x6C\x4E\xEC\xCA\xD8\x9A\x58\x31\xA6\xF6\xF1\x46\xFC\xBC\xF8\xEC\xF8\xEC\xF8\xCA\xF8\xCA\xF8\x95\x71\x37\xBE\x3E\x3E\x72\xCC\xBD\xE2\x13\xF1\x27\xE2\x3F\x8C\xFF\x50\xEE\x11\x5F\x8C\xEF\x8F\xBF\x1C\x7F\x33\xFE\x66\xFC\xED\xE6\x1A\x35\xAF\x54\x13\xF5\x63\xCE\x81\x4F\x24\x9E\x90\x73\xDF\x8B\x89\x83\x89\x83\x89\x88\x95\xB4\xC6\x58\x69\xAB\xD5\x6A\xB7\x4E\xB7\x3A\xAD\x19\xD6\x99\xD6\xD9\xD6\x79\xD6\x6C\x6B\xB6\x75\xAD\x75\x9D\xD5\x65\xB9\xD6\x7A\x6B\xBD\x75\x8B\x75\x8B\xB5\xD9\xBA\xD3\xBA\xDB\xBA\xC7\xFA\x90\xF5\x11\xEB\xE3\xD6\x27\xAC\x7B\xAD\x7B\xAD\x4F\x5B\xF7\x59\xF7\x5B\xFF\x64\x3D\x68\x3D\x64\x7D\xDE\x7A\xCC\xDA\x65\xED\xB2\x22\xC9\x64\xB2\x25\x99\x4E\x8E\x4D\xB6\x27\x4F\x4F\x9E\x97\x9C\x9D\x9C\x9D\xBC\x32\x79\x75\x72\x4D\xD2\x4D\x8E\x24\x47\x92\x9B\x47\xE9\xD3\x4F\x24\x9F\x90\xFD\xF8\x45\xEA\xC3\x46\x2A\x99\x6A\x49\xA5\x53\x63\x53\xB3\x52\xB3\x53\xB3\x53\xCB\x53\xCB\x53\xAB\x53\x97\xA7\xBA\x52\x6E\xEA\xA6\xD4\x70\x6A\x73\xEA\xA3\xA9\x4F\xA4\x1E\x4D\x7D\x31\xF5\x8D\xD4\x13\xA9\x27\x52\x3B\x53\xDF\x4B\x61\x4B\x67\xCB\x8C\x96\x33\x5B\xE6\xB4\x2C\x68\x79\x4F\xCB\x7B\x5B\x6E\x6D\xB9\xB5\xE5\xB6\x96\x3B\x5B\xEE\x6A\xB9\xFF\xFF\x61\xEF\x4B\xE0\x9B\xA8\xB6\xFF\xCF\x99\x4C\xDA\xA6\x69\x9A\x14\x4A\x97\xA4\xBD\x49\x2C\x10\x02\x43\x08\x8B\x2C\x65\x2B\xBB\xA2\x2C\xAE\x80\x0B\x69\x68\xD3\x36\xD0\x26\x21\x49\x59\x5C\xA0\x6C\x2A\x08\x82\xFB\x86\x52\x15\x67\xDA\x54\x05\x14\x10\x65\x2B\x2E\xB8\x22\xA8\x2C\x0A\x28\x05\x15\x51\x40\x70\x57\x96\xE4\xFF\x99\x99\x9B\x36\x8D\x45\x7D\xEF\xFD\x7C\xEF\xFD\x7F\x3F\x87\xCF\xA1\xB9\xE7\xDE\xB9\xFB\xFD\x9E\x73\xB7\x33\xA9\x4F\xA7\x3E\x9D\x2A\xA4\xAE\x4E\x7D\x3E\xF5\x85\xD4\xCD\xA9\x9B\x53\xEB\x53\x5F\x4B\xDD\x9E\xBA\x3D\xF5\x60\xEA\xA1\x54\xD0\x82\xB6\x40\x3B\x42\x7B\xA9\xB6\x5C\x7B\x87\x76\xB1\x16\x74\x1D\x75\x03\x75\x03\x75\x63\x75\xD7\xE8\xAE\xD3\xDD\xA8\x73\xEA\x9C\x3A\x97\xAE\x44\xE7\xD1\xF9\x74\x95\xBA\x39\xBA\xBB\x74\x77\xE9\x7E\x2B\xF7\xFF\x35\x3D\xA4\xAA\xF5\xD2\xD6\xD1\xB9\x23\xA6\x97\x67\x78\x33\x66\x67\xCC\xC9\x78\x24\xA3\x3A\xE3\xEB\x8C\x6F\x33\x7E\xC9\x80\xCC\x94\xCC\xD4\xCC\xB4\x4C\x53\x66\xD7\xCC\x82\xCC\x2B\x32\x0B\x33\xCB\x33\x3D\x99\xBE\xCC\xAA\xCC\x17\x32\xF7\x65\xEE\xCB\x3C\x98\x79\x30\xD3\x94\x35\x23\xAB\x2A\x6B\x7E\xD6\xED\x59\x35\x59\xA1\xAC\x50\x96\x38\xBF\x86\x6C\x92\x3D\x36\xBB\x22\xDB\x97\x1D\xCC\xAE\xCA\xAE\xCB\x5E\x9D\xFD\x56\x36\xE8\x33\xF4\x26\xFD\xD5\x7A\x41\x2F\xE8\x95\x86\xEE\x86\x61\x86\xCB\x0C\xF3\x0C\xD5\x86\xE7\x0C\xAB\x0C\x2F\x1B\xBE\x30\x40\x4E\x5E\xCE\xA8\x9C\x31\x39\xD7\xE5\xDC\x9A\x23\xEA\x8D\x51\x7D\xA6\x69\xBE\xF8\x93\x3C\x5F\x14\x65\x3C\xF9\xB3\x32\x7E\x93\x71\x8B\x71\x8B\xF1\x15\x49\xD6\xBF\x6B\xDC\x45\xE5\x7D\x83\xF1\x04\x95\xF7\xE7\x8C\x53\x4D\x53\x4D\xB3\x4D\xB3\x4D\xCB\x4C\xCB\x4D\x8C\x99\x31\x27\x9A\x13\xCD\xC9\xE6\xD6\xE6\x36\xE6\x36\xE6\x6C\x73\xAE\xB9\xAD\xB9\xAD\x59\xD4\x09\x3A\x98\x3B\x98\x3B\x9A\x3B\x9B\xBB\x98\xFB\x98\xFB\x99\x07\x9A\x87\x99\x2F\x35\x8F\x36\x8F\x36\xCF\x33\xCF\x37\x47\xF5\xAC\x42\xEE\x73\xEE\x38\xF7\x0D\xF7\x1D\xF7\x3D\x17\xAB\x87\xD9\x3B\x97\x76\xF6\x76\xF6\x76\xFE\x33\xFA\x5E\x17\x5B\x17\x9B\xD2\x9E\x60\xEF\x6A\xBF\xD8\xDE\xD7\xDE\xCF\x1E\x95\x4B\x22\xA6\x9E\xB3\x47\xEC\xCA\xAE\x5D\xBA\x8A\x72\xB4\xBA\xEB\xD6\xAE\x22\xBE\x36\xCA\xEB\xEA\xC5\x2B\x77\xAD\x3C\xBF\x12\x9E\x4E\xE1\x75\xBC\x91\xEF\xC8\x77\xE2\x0B\x78\x0F\x3F\x9B\x7F\x8C\xAF\xE6\xBF\xE0\x33\x05\xA3\x70\x91\x70\x87\x70\xA7\xB0\x55\x78\x45\x78\x5B\xF8\x51\xC0\x1A\x45\x8D\xB2\x26\xB1\x26\xB9\x26\xA5\xE6\xA2\x9A\x82\x9A\x40\xCD\xDC\x9A\x17\x6A\xBE\xAB\xF9\xB9\xE6\x6C\xCD\xD9\x9A\xF3\x35\xE1\x9A\xE4\xDA\x76\xB5\xF6\xDA\xE1\xB5\xA2\xAC\x9E\x5F\xFB\xBC\x34\x1F\x6E\xA8\xFD\xBC\xF6\x68\xED\xA9\xDA\x33\xB5\xE1\x5A\x08\x75\x08\x15\x84\x06\x87\x86\x86\x2E\x0B\x15\x86\xA6\x85\x6E\x0A\xDD\x14\x9A\x15\x5A\x15\x5A\x13\x5A\x13\x5A\x17\x5A\x1F\x7A\x29\xB4\x39\x54\x1F\xAA\x0F\xBD\x12\x7A\x25\x74\x20\xF4\x49\xA8\x21\xF4\x75\xE8\xBB\xD0\x0F\xA1\x96\x74\x5A\x7B\xDD\x75\x75\x37\xD4\xB9\xEB\x7C\x75\x47\xEA\x62\xF5\x82\x46\x3D\xE4\x6C\xC2\xD9\xD6\x67\x33\xCF\xE6\x9E\xCD\x3D\x6B\x3C\xDB\xE1\x6C\xC7\x16\xE5\xF8\x9A\xB8\xB5\xCA\xB3\x8D\x6B\x95\xE6\x48\x7E\x64\x50\xE4\xBA\x48\x49\x8B\x7A\x06\x82\x1A\x50\xB2\x92\xD3\x1E\x10\x3A\x00\x42\x3E\x20\xF4\x03\x84\xFE\x80\x30\x10\x10\x46\x01\xC2\x58\x40\xB8\x11\x10\xAA\x00\xE1\x8C\xF8\x97\x41\x58\xCE\x20\x34\xAE\x27\x29\x10\x8C\x0A\x84\xCE\x0A\x84\x02\x05\xC2\x60\x05\xC2\xA5\x0A\x84\xB1\x0A\x84\x4A\x05\x42\x15\x3D\x2B\x57\xAD\x40\x78\x55\x81\xF0\xA6\x02\x61\xA7\xA8\x6F\xB3\x08\x8F\xB1\x08\xD1\x75\xE6\xE8\x1A\x73\x74\xDE\x11\x5D\x9F\x82\x04\x84\x9E\x89\x08\x05\x89\x08\xD7\x24\x22\x14\x26\x22\x94\x26\xFE\x63\xF3\x33\x51\x6F\x29\x4C\x42\x98\x96\x84\x50\x95\x84\xF0\x78\x12\xFE\x46\x7F\x02\x15\x42\x1B\x15\x82\x49\x85\xD0\x47\x85\x50\xA5\x42\x78\x51\x85\xB0\x45\x85\xB0\x55\x85\x92\x7C\x16\x49\x77\x01\xB9\x5C\x98\x8C\x70\x4B\x32\x42\x55\x32\xC2\xF2\x64\x84\xFA\x64\x84\x37\x92\x11\xDE\x12\x65\xB3\xF8\x2E\x5D\xFF\x8C\xEA\x25\x51\x7D\xA5\x4A\x8D\x70\xB7\x1A\xA5\x83\x78\x97\xA4\x88\x7A\x88\xBC\xBE\x1E\x5D\x5B\x17\xF5\xC0\x3C\x0D\x42\x55\x2A\x42\x6D\x2A\xC2\xEA\x54\x84\x35\xA9\x08\xA0\x45\xC8\xD1\x22\x58\x28\xD9\xB5\x08\xC3\xB4\x08\xAB\xB5\x08\xAF\x69\x9B\xE6\x6D\x0A\x1D\x42\x2F\x1D\xC2\x3C\x1D\xC2\x2A\x1D\xC2\x2E\x1D\xC2\x31\x1D\x82\x22\x0D\xA1\x7D\x1A\xC2\x50\x4A\x23\x28\x8D\x4D\x43\xF0\xA7\x21\x4C\xA3\x34\x2F\x0D\xE1\xF9\x34\x84\x6D\x69\x08\xAF\xA5\x21\xEC\xA7\x74\x90\x52\xAC\x5E\xD1\xA9\x15\x42\x55\x2B\x84\xDB\x5A\x21\xDC\x41\x69\x51\x2B\x84\x25\xAD\x10\x96\xB6\x42\x78\xAC\x15\xC2\x8A\x56\x08\x35\x54\xFF\xF8\xB4\xD5\x3F\x3E\xDF\xEA\x4F\x29\x3A\xBF\x12\xE7\x55\x4E\x71\x1E\x95\x8E\xD0\x23\x1D\x61\x44\x3A\xC2\xC8\x74\x04\x47\x3A\x82\x33\x1D\xA1\x2A\x1D\x61\x55\x3A\xC2\x6B\xE9\x08\xAF\xA7\x23\xBC\x41\xA9\xAA\x0D\xC2\x73\x6D\x10\x3E\x6A\x83\x70\xA0\x0D\x02\x64\x20\x74\xC9\x40\x18\x4A\xA9\x2A\x03\x21\x94\x81\xB0\x81\x52\x74\xBF\x41\x9C\x67\x77\xCA\x42\xA8\xA6\x7B\x1C\x91\xAC\x3F\x9E\xB7\x77\xC9\x46\x18\x48\x69\x10\xA5\xEA\x6C\x04\x21\x1B\x21\x94\x8D\xB0\x2B\x1B\xE1\x30\xA5\xCF\x28\x81\x5E\x26\xB5\x1E\xA1\x9B\x1E\x21\x9F\xD2\x58\x4A\x37\xE8\x11\x16\xEA\x11\x1E\xA3\x54\xAF\x47\xF8\x45\x7C\x27\x66\x5D\xDB\x96\x83\x50\x40\xC9\x9F\x83\xB0\x34\x07\x01\x72\x11\x12\x72\x11\x92\x72\x11\x54\xB9\x08\xEA\x5C\x04\x7B\x2E\xC2\x70\x4A\x2D\xCD\xCB\x16\xD2\xB9\x98\x48\x0D\x04\xE1\x3B\x82\xB0\xDA\x28\x53\x74\x1D\xA5\x71\xDD\x84\xAE\x23\x34\xEE\xC3\xD0\xB5\xFD\x82\xC9\x08\x13\x27\x63\xE3\xBA\x3E\x4C\x41\xE8\x32\x05\xA1\x60\x0A\xC2\x90\x29\x08\x45\x53\x10\xA6\x4F\x41\xB8\x75\x0A\xC2\x52\x91\x4F\xF7\x8A\xC0\x8B\x30\xD2\x8B\x30\x96\xD2\xBD\x5E\x84\x15\x5E\x84\x06\x2F\xC2\x61\xBA\x66\x28\x12\xF8\x10\x7E\xA6\x87\x39\x45\xFD\x17\x96\x20\x24\x2D\x91\x75\xEE\xD8\x79\x4D\x54\xFF\xF6\xAD\x41\x38\xFB\x3C\xFE\xE1\x3A\xC9\x7F\xF3\xBA\x61\xFD\x5E\x84\x57\xF6\x22\xEC\xDF\x8B\x70\x70\x2F\xC2\x99\xBD\x08\xE7\xF6\x22\xA4\xEF\x43\xC8\xD8\x87\xD0\x63\x1F\x42\xCF\x7D\x08\xA3\xF7\x21\x8C\xDD\x87\xE0\xD9\x87\xE0\xDB\x87\x70\xC7\x3E\x84\x45\xFB\x10\x6A\xF6\x21\x84\xF6\x21\xBC\xB2\x0F\xE1\xB5\x7D\xF2\x3A\x24\x1C\x46\xE0\x0E\x23\xF4\x3A\x8C\xD0\xFF\x30\xC2\x68\x4A\xF5\x47\x10\x8E\x1F\x41\x80\x63\x08\xAF\x1D\x43\x69\x7F\x6E\x08\xDD\xA3\x13\xE9\xEF\x7D\x30\x79\x1F\xEC\x9F\x5D\x2F\x68\x59\x8F\x6E\xD2\x9F\xDF\x4C\xDD\x91\xBA\x2B\xF5\x63\x59\x8F\x96\xCE\x7B\x37\xCD\xD7\x9A\xCF\xD5\x36\x48\xF3\x34\xC8\x6D\x97\xCB\xE5\xDE\x90\x3B\x29\xD7\x95\x3B\x25\x77\x7A\xEE\xCC\xDC\x2D\xB9\x40\x22\x24\x56\xF7\xEB\x68\xEE\x64\xEE\x66\xEE\x26\xE9\x74\x85\xE6\x3B\xCC\x85\xDC\x6C\xEE\x22\xE1\xF6\x18\x7D\x29\x22\xC8\x3A\x4A\x71\x5D\x49\x9D\xBB\x4E\xD4\x47\xA2\x3A\x05\xE4\xF4\xCA\xE9\x9B\x73\xA9\xA4\xBF\x4A\xF3\xD7\x6C\x92\x6D\xCA\xEE\x94\x6D\xCF\xEE\x9B\x5D\x90\x5D\x90\x3D\x34\x7B\x4C\x76\xEC\xFC\xF8\x82\x72\xBC\xF1\xF9\xE7\xF4\x91\x5D\x2B\x23\x2B\x1B\xB1\x28\xE6\x11\xEB\x46\x9C\xC7\x1E\x92\xEA\x24\x02\x88\x88\x0A\x54\xA0\x12\x95\x98\x88\x89\xA8\x42\x15\xAA\x51\x8D\x1A\xD4\xA0\x16\xB5\x98\x86\x69\xD8\x1A\x5B\x63\x1B\x6C\x83\x99\x98\x89\xD9\x98\x8D\x06\x34\x60\x2E\xE6\xA2\x11\x8D\x68\x46\x33\xE6\x61\x1E\xB6\xC3\x76\x68\x41\x0B\x5A\xD1\x8A\x9D\xB0\x13\x76\xC6\xCE\xD8\x05\xBB\x60\x57\xEC\x8A\xDD\xB1\x3B\x5E\x8C\x17\x63\x2F\xEC\x8D\xF9\x98\x8F\xFD\xB0\x1F\x0E\xC0\x01\x58\x80\x05\x38\x18\x07\xE3\x50\x1C\x8A\xC3\x71\x38\x5E\x82\x97\xE2\x65\x78\x19\x8E\xC2\x51\x38\x06\xC7\xE0\x15\x78\x05\x5E\x85\x57\xE1\x35\x78\x0D\x8E\xC3\x71\x38\x01\x27\xE0\xF5\x78\x3D\xDE\x88\x37\xA2\x03\x1D\xE8\x44\x27\x16\x61\x11\xBA\xD0\x85\xA5\x58\x8A\x6E\x74\xE3\x14\x9C\x82\x15\x58\x81\x5E\xF4\xE2\x54\x9C\x8A\x01\x0C\x60\x25\x56\xE2\x74\x9C\x8E\x37\xE1\x4D\x78\x0B\xDE\x82\xB3\xB0\x0A\xE7\xE1\x3C\x5C\x80\x0B\xF0\x0E\xBC\x03\x17\xE3\x12\xBC\x07\xEF\xC1\xFB\xF1\x7E\x7C\x18\x1F\xC5\xC7\xF1\x71\x7C\x02\x9F\xC0\xA7\xF0\x29\x7C\x1A\x9F\xC6\x1A\xAC\xC1\x10\xD6\xE1\xB3\xF8\x2C\xAE\xC6\xD5\xB8\x16\xD7\xE2\x7A\x5C\x8F\x2F\xE1\xCB\xB8\x19\xB7\xE2\x76\xDC\x8E\x6F\xE1\x5B\xF8\x2E\xBE\x8B\xEF\xE1\x7B\xB8\x0B\x77\xE1\x07\xF8\x01\xEE\xC6\xDD\xB8\x17\xF7\xE2\x47\xF8\x11\xEE\xC7\xFD\x78\x10\x3F\xC1\x43\x78\x08\x0F\xE3\x61\xFC\x0C\x3F\xC3\x2F\xF0\x0B\xFC\x12\xBF\xC4\xAF\xF0\x2B\x3C\x8E\xC7\xF1\x24\x9E\xC4\x53\x78\x1A\xBF\xC7\xEF\xF1\x47\xFC\x11\x7F\xC5\x5F\xF1\x2C\x9E\xC5\xF3\x78\x1E\x23\x18\x41\x64\x90\x51\x30\x0A\x46\xC9\x28\x99\x44\x26\x91\x51\x31\x2A\x46\xCD\xA8\x19\x0D\xA3\x61\xB4\x8C\x96\x49\x63\xD2\x98\xD6\x4C\x6B\xA6\x0D\xD3\x86\xC9\x64\x32\x99\x6C\x26\x9B\x31\x30\x06\x26\x97\xC9\x65\x8C\x8C\x51\x9C\xD2\x30\x79\x4C\x1E\xD3\x8E\x69\xC7\x58\x18\x0B\x63\x65\xAC\x4C\x27\xA6\x13\xD3\x99\xE9\xCC\x74\x61\xBA\x30\x5D\x99\xAE\x4C\x77\xA6\x0F\xD3\x8F\xE9\xC7\x0C\x64\x0A\x98\xC1\xCC\x60\x66\x04\x33\x82\xB9\x94\xB9\x94\xB9\x8C\xB9\x8C\x19\xC5\x8C\x62\xC6\x30\xF7\x32\xF7\x33\xAB\x98\xA9\x8A\xA9\x8A\x80\x22\xA0\x98\xAE\x98\xAE\xB8\x59\x71\xAB\x62\x99\x62\x99\xE2\x19\xC5\x7B\x8A\x5D\x8A\xF7\x15\x7B\x14\xFB\x14\x1F\x2B\x3E\x56\x1C\x50\x1C\x50\x7C\xA2\xF8\x44\x71\x48\x71\x48\x71\x58\x71\x58\xF1\x99\xE2\x33\xC5\x17\x8A\x2F\x14\x5F\x2A\xBE\x54\x7C\xA5\xF8\x4A\x71\x5C\x71\x5C\x71\x52\x71\x52\x71\x4A\xF1\xBD\xE2\x47\xC5\x8F\x8A\x5F\x14\xBF\x28\xCE\x2A\xCE\x29\xEC\xAC\x83\x75\xB2\x4E\xB6\x88\x2D\x62\x5D\xAC\x8B\x2D\x65\x4B\x59\x37\xEB\x66\xA7\xB0\x53\xD8\x0A\xB6\x82\xF5\xB2\x5E\x76\x2A\x3B\x95\x0D\xB0\x01\xB6\x92\xAD\x64\xA7\xB3\xD3\xD9\x99\xEC\x4C\xF6\x66\xF6\x66\xF6\x56\xF6\x56\x76\x36\x3B\x9B\x9D\xC3\xCE\x61\xEF\x64\xEF\x64\x97\xB0\x4B\xD8\xA5\xEC\x52\xF6\x6E\xF6\x6E\xF6\x5E\xF6\x5E\xF6\x7E\xF6\x7E\xF6\x41\xF6\x41\xF6\x61\xF6\x61\xF6\x51\xF6\x51\xF6\x31\xF6\x31\x76\x05\xBB\x82\x7D\x82\x7D\x82\x7D\x8A\x7D\x8A\x7D\x9A\x7D\x9A\x15\x58\x81\xAD\x65\x6B\xD9\x3A\xB6\x8E\x7D\x96\x7D\x96\x5D\xC5\xAE\x62\xD7\xB0\x6B\xD8\x17\xD8\x17\xD8\x75\xEC\x3A\xF6\x45\xF6\x45\xF6\x25\xF6\x25\x76\x23\xBB\x91\xDD\xCC\x6E\x66\xB7\xB2\x5B\xD9\x57\xD8\x57\xD8\xD7\xD8\xD7\xD8\xED\xEC\x76\xF6\x4D\xF6\x4D\xF6\x6D\xF6\x6D\xF6\x5D\xF6\x5D\xF6\x3D\x76\x27\xFB\x3E\xFB\x3E\xFB\x21\xFB\x21\xBB\x87\xDD\xC3\xEE\x63\xF7\xB1\x1F\xB3\x1F\xB3\x07\xD8\x03\xEC\x27\xEC\x27\xEC\x21\xF6\x10\x7B\x98\x3D\xCC\x7E\xC6\x7E\xC6\x7E\xC1\x7E\xC1\x7E\xC9\x7E\xC9\x7E\xC5\x7E\xC5\x1E\x67\x8F\xB3\x27\xD9\x93\xEC\x29\xF6\x14\xFB\x2D\xFB\x2D\xFB\x3D\xFB\x3D\xFB\x23\xFB\x23\xFB\x33\xFB\x33\xFB\x2B\xFB\x2B\x7B\x96\x3D\xCB\x9E\x67\xCF\xB3\x11\x36\xC2\x8A\x83\x57\xA1\x54\x28\xC5\x27\x51\x99\xA8\x54\x29\x55\x4A\xB5\x52\xAD\xD4\x28\x35\x4A\xAD\x52\xAB\x4C\x53\xA6\x29\x5B\x2B\x5B\x2B\xDB\x28\xDB\x28\x33\x95\x99\xCA\x6C\x65\xB6\xD2\xA0\x34\x28\x73\x95\xB9\x4A\xA3\xD2\xA8\x34\x2B\xCD\xCA\x3C\x65\x9E\xB2\x9D\xB2\x9D\xD2\xA2\xB4\x28\xAD\x4A\xAB\xB2\x93\xB2\x93\xB2\xB3\xB2\xB3\xB2\x8B\xB2\x8B\xBC\x3E\xA5\x3B\xA3\x3B\xAF\x8B\xE8\x7E\x69\x7D\xBE\xB5\x38\x27\x87\xDC\x4E\x14\x3F\x1F\xC9\x45\x82\x44\x41\x14\x44\x49\x94\x24\x91\x24\x12\x15\x51\x11\x35\x51\x13\x0D\xD1\x10\x2D\xD1\x92\x34\x92\x46\x5A\x93\xD6\xA4\x0D\x69\x43\x32\x49\x26\xC9\x26\xD9\xC4\x40\x0C\x24\x97\xE4\x12\x23\x31\x12\x33\x31\x93\x3C\x92\x47\xDA\x91\x76\xC4\x42\x2C\xC4\x4A\xAC\xA4\x13\xE9\x44\x3A\x93\xCE\xA4\x0B\xE9\x42\xBA\x92\xAE\xA4\x3B\xE9\x4E\x2E\x26\x17\x93\x5E\xA4\x17\xE9\x43\xFA\x90\xBE\xA4\x2F\xE9\x4F\xFA\x93\x81\x64\x20\x19\x44\x06\x91\x21\x64\x08\x19\x46\x86\x91\x11\x64\x04\xB9\x94\x5C\x4A\x2E\x23\x97\x91\x51\x64\x14\x19\x43\xC6\x90\x2B\xC8\x15\xE4\x2A\x72\x15\xB9\x86\x5C\x43\xC6\x91\x71\x64\x02\x99\x40\xAE\x27\xD7\x93\x1B\xC9\x8D\xC4\x41\x1C\xC4\x49\x9C\xA4\x88\x14\x11\x17\x71\x91\x52\x52\x4A\xDC\xC4\x4D\xA6\x90\x29\xA4\x82\x54\x10\x2F\xF1\x92\xA9\x64\x2A\x09\x90\x00\xA9\x24\x95\x64\x3A\x99\x4E\x66\x92\x99\xE4\x66\x72\x33\xB9\x95\xDC\x4A\x66\x93\xD9\x64\x0E\x99\x43\xE6\x91\x79\x64\x01\x59\x40\x6E\x27\xB7\x93\x85\x64\x21\xB9\x93\xDC\x49\x96\x90\x25\x64\x29\x59\x4A\xEE\x26\x77\x93\x7B\xC9\xBD\xE4\x7E\xF2\x18\x59\x41\x56\x90\x27\xC8\x13\xE4\x29\xF2\x14\x79\x9A\x3C\x4D\x04\x22\x90\x5A\x52\x4B\xEA\x48\x1D\x79\x96\x3C\x4B\x56\x91\x55\x64\x0D\x59\x43\x5E\x20\x2F\x90\x75\x64\x1D\x79\x91\xBC\x48\x5E\x22\x2F\x91\x8D\x64\x23\xD9\x4C\x36\x93\xAD\x64\x2B\xD9\x46\xB6\x91\x57\xC9\xAB\xE4\x75\xF2\x3A\x79\x83\xBC\x41\xDE\x22\x6F\x91\x77\xC8\x3B\x64\x07\xD9\x41\x76\x92\x9D\xE4\x7D\xF2\x3E\xF9\x90\x7C\x48\xF6\x90\x3D\x64\x1F\xD9\x47\x3E\x26\x1F\x93\x03\xE4\x00\xF9\x84\x7C\x42\x0E\x91\x43\xE4\x30\x39\x4C\x3E\x23\x9F\x91\x2F\xC8\x17\xE4\x4B\xF2\x25\xF9\x8A\x7C\x45\x8E\x93\xE3\xE4\x24\x39\x49\x4E\x91\x53\xE4\x5B\xF2\x2D\xF9\x9E\x7C\x4F\x7E\x24\x3F\x92\x9F\xC9\xCF\xE4\x57\xF2\x2B\x39\x4B\xCE\x92\xF3\xE4\x3C\x89\x90\x44\xA3\xCE\xD8\xC6\x68\x32\x5A\x8C\x76\x63\x2F\x63\x81\x71\x98\xB4\xE6\x52\x68\x2C\x35\xFA\xA4\xB5\x96\xDB\x8D\xCB\x8C\x0F\x1A\xAB\x8D\x82\x71\xB5\xB4\xEE\xF2\x62\xCC\xBA\xCB\x1B\x74\xDD\x65\x9F\xB1\xC1\xF8\x25\x5D\x77\xF9\xD9\x98\x6C\x4E\x36\xA7\x9A\xB5\xE6\xD6\xE6\xD6\xE6\x2E\xE6\x2E\xE6\x1E\xE6\x1E\xE6\x3E\xD2\xDA\x4A\x7F\xF3\x70\xBA\xB6\x32\xDF\x3C\xDF\x6C\xE7\x26\x72\x4E\xCE\xC9\xB9\xB8\x12\xAE\x8C\x2B\xE3\x26\x73\x93\xB9\x72\xAE\x9C\x9B\xCA\x4D\xE5\x02\x5C\x90\x9B\xC6\xDD\xCC\xCD\xE1\xE6\x70\xF3\xB8\x79\xDC\x02\x6E\x01\x77\x3B\x77\x3B\xB7\x90\x5B\xC8\xDD\xC9\xDD\xC9\x2D\xE1\x96\x70\x4B\xB9\xA5\xDC\xDD\xDC\xDD\xDC\xBD\xDC\xBD\xDC\xFD\xDC\xFD\xDC\x83\xDC\x83\xDC\xC3\xDC\xC3\xDC\xA3\xDC\xA3\xDC\x63\xDC\x63\xDC\x0A\x6E\x05\xF7\x04\xF7\x04\xF7\x14\xF7\x14\xF7\x34\xF7\x34\x27\x70\x02\x57\xCB\xD5\x72\x75\x5C\x1D\xF7\x2C\xF7\x2C\xB7\x8A\x5B\xC5\xAD\xE1\xD6\x70\x2F\x70\x2F\x70\xEB\xB8\x75\xDC\x8B\xDC\x8B\xDC\x4B\xDC\x4B\xDC\x46\x6E\x23\xB7\x99\xDB\xCC\x6D\xE5\xB6\x72\xDB\xB8\x6D\xDC\xAB\xDC\xAB\xDC\xEB\xDC\xEB\xDC\x1B\xDC\x1B\xDC\x5B\xDC\x5B\xDC\x3B\xDC\x3B\xDC\x0E\x6E\x07\xB7\x93\xDB\xC9\xBD\xCF\xBD\xCF\x7D\xC8\x7D\xC8\xED\xE1\xF6\x70\xFB\xB8\x7D\xDC\xC7\xDC\xC7\xDC\x01\xEE\x00\xF7\x09\xF7\x09\x77\x88\x3B\xC4\x1D\xE6\x0E\x73\x9F\x71\x9F\x73\x27\xB8\x13\xDC\x37\xDC\x37\xDC\xF7\x71\xEB\x47\x83\xF8\x41\xFC\x10\x7E\x08\x3F\x8C\x1F\xC6\x8F\xE0\x47\xF0\x97\xF2\x97\xF2\x97\xF1\x97\xF1\xA3\xF8\x51\xFC\x18\x7E\x0C\x7F\x05\x7F\x05\x7F\x15\x7F\x15\x7F\x0D\x7F\x0D\x3F\x8E\x1F\xC7\x4F\xE0\x27\xF0\xD7\xF3\xD7\xF3\x37\xF2\x37\xF2\x0E\xDE\xC1\x3B\x79\x27\x5F\xC4\x17\xF1\x2E\xDE\xC5\x97\xF2\xA5\xBC\x9B\x77\xF3\x53\xF8\x29\x7C\x05\x5F\xC1\xCF\xE1\xE7\xF0\xF3\xF8\x79\xFC\x02\x7E\x01\x7F\x3B\x7F\x3B\xBF\x90\x5F\xC8\xDF\xC9\xDF\xC9\x2F\xE1\x97\xF0\x4B\xF9\xA5\xFC\xDD\xFC\xDD\xFC\xBD\xFC\xBD\xFC\xFD\xFC\xFD\xFC\x83\xFC\x83\xFC\xC3\xFC\xC3\xFC\xA3\xFC\xA3\x7C\x9E\x90\x27\xB4\x13\xDA\x09\x16\xC1\x22\x58\x05\xAB\xD0\x49\xE8\x24\x74\x16\x3A\x0B\x5D\x84\xAE\x42\x77\xA1\xBB\x70\xB1\x70\xB1\xD0\x4B\xE8\x25\xF4\x11\xFA\x08\x7D\x85\xBE\x42\x7F\xA1\xBF\x30\x50\x18\x28\x0C\x12\x06\x09\x43\x84\x21\xC2\x30\x61\x98\x30\x42\x18\x21\x5C\x2A\x5C\x2A\x5C\x26\x5C\x26\x8C\x12\x46\x09\x63\x84\x31\xC2\x15\xC2\x15\xC2\x55\xC2\x55\xC2\x35\xC2\x35\xC2\x38\x61\x9C\x30\x41\x98\x20\x5C\x2F\x5C\x2F\xDC\x28\xDC\x28\x38\x04\x87\xE0\x14\x9C\x42\x91\x50\x24\xB8\x04\x97\x50\x2A\x94\x0A\x6E\xC1\x2D\x4C\x11\xA6\x08\x15\x42\x85\xE0\x15\xBC\xC2\x54\x61\x86\x70\x93\x70\x93\x70\x8B\x70\x8B\x30\x5B\x98\x2D\xCC\x11\xE6\x08\xF3\x84\x79\xC2\x02\x61\x81\x70\xBB\x70\xBB\xB0\x58\x58\x2C\xDC\x25\xDC\x25\xDC\x2D\xDC\x2D\xDC\x2B\xDC\x2F\x3C\x28\x3C\x28\x3C\x2C\x3C\x2C\x3C\x2A\x3C\x2A\x3C\x26\x3C\x26\xAC\x10\x56\x08\x4F\x08\x4F\x08\x4F\x09\x4F\x09\x4F\x0B\x4F\x0B\xE2\x53\x2B\xD4\x0A\xAB\x84\x55\xC2\x3A\x61\x9D\xF0\xA2\xF0\xA2\xF0\x92\xF0\x92\xB0\x51\xD8\x28\x6C\x16\x36\x0B\x5B\x85\xAD\xC2\xAB\xC2\xAB\xC2\x9B\xC2\x9B\xC2\xDB\xC2\xDB\xC2\x4F\xC2\x4F\xC2\x19\xE1\x8C\xA4\x2B\x16\xD6\x95\xD5\xF9\xEA\xB6\xD6\x45\xD7\xAE\x24\x5D\x31\xF6\x6C\x21\x5D\xE7\x69\x3C\x3B\x48\xE7\xD1\xD1\xB3\x81\xFA\xDD\x08\xDD\x77\x23\x8C\x8E\x99\x77\x95\xEE\x46\x98\xBB\x1B\xE1\xD1\xDD\x08\xEB\x7F\x67\x7E\xB5\x73\x37\xC2\xD7\xBB\x11\x14\x74\x3E\xD5\x8B\xCE\x97\xA6\xEC\x41\xB8\x6D\x0F\xC2\x8A\x3D\x08\x2F\xEF\x41\xF8\x70\x0F\xC2\x37\xE2\x5C\x6C\x2F\xC2\x45\x7B\x11\xFA\xEE\x45\xB8\x76\x2F\x82\x77\x2F\xC2\xA2\xBD\xF2\xFC\x2A\x76\x5E\x75\x38\x6E\x5E\x65\x88\x9B\x57\x5D\x13\x37\xAF\x5A\x1A\x37\xAF\x7A\x8B\xCE\xAD\x44\x8A\x9E\x75\x5C\xCD\xFC\x76\x4F\x25\xA8\x08\x2A\x6E\x52\xDC\xA4\x98\xA0\x9C\xA0\x2C\x48\x28\x90\xF6\x44\xA2\x7B\x1E\xB1\xFB\x1D\x53\x55\x53\x55\xC3\x53\x87\x4B\xF3\x82\x73\xBA\x73\xBA\x7D\x99\xFB\x32\x87\x64\x0D\xC9\x12\xF4\x82\x7E\x46\xCE\xAD\x39\x5C\xEE\x64\xE9\xFC\xC1\xA3\xB9\x5B\x73\x63\xF5\xFD\x5B\xB8\x5B\x39\x6F\x67\x2F\x5D\x6F\x55\xD2\xF5\xD6\xBE\xF6\xC7\xEC\x8F\xDB\xCF\xD9\xC3\xF6\x36\xD5\x6D\xAA\x7F\x59\x79\x7E\x65\x0A\x9F\xC2\xCF\xE6\x67\xF3\xCB\xF9\xC7\xA4\x35\x54\x9F\xE0\x13\xEE\x10\xEE\x10\x7E\x11\x7E\x15\x76\xD6\xEE\xAC\x3D\x5A\x7B\xB4\xD6\x17\xF2\x85\x3E\x09\x7D\x12\xFA\x3E\xF4\x43\xE8\x86\x3A\x47\x9D\xBB\xCE\x5D\x17\x3D\x0B\x24\xED\x17\xD2\x79\xFD\x85\xE6\xEB\xD1\xF9\x65\x74\xFE\x18\x9D\xC7\x6C\xC4\x8D\x58\x8F\xAF\xE2\x7D\xCC\x7D\x4C\xD3\x5E\x91\x7C\x6E\x33\x76\xBF\xE8\x8F\xF6\x8A\x20\xA9\xCD\x1F\xEE\x0B\xF9\xFF\xDE\x17\xFA\x8B\xF6\x85\x86\xFD\xA9\x7D\xA1\x7F\x76\x4F\x08\xD2\xFE\xF5\x3D\xA0\x7F\x74\xEF\x47\xDE\xF7\x19\x9C\x35\xF4\x1F\xDC\xFB\xF9\xFD\x7D\x9F\xE9\xBF\xD9\xEB\xB9\xD8\xDC\xDB\x6C\xEF\x5C\xFA\xA7\xF6\x4D\x12\xEC\x09\xF6\x7E\x74\xBF\x64\x05\xDD\x2F\x89\xFC\xEE\x7E\x49\x7A\x75\x86\xB4\x67\xF2\xF3\x4A\x78\x5A\xDD\xB8\x67\xE2\xE1\x3D\xD2\x7E\xC9\x52\x61\xA9\xF0\xB3\xF0\xB3\x70\xF6\x9F\xD8\x27\x69\xA8\xFD\xBC\xF6\xCB\x16\xF7\x48\xBC\xA1\xA9\x7F\xB8\x4F\x72\x90\xEE\x93\x7C\xD7\xE2\x3E\x49\xFD\x05\xF6\x47\xFE\xD9\x7D\x91\xE8\xBE\xFC\xEF\x9E\xBF\xF8\x0F\xEF\x8F\x8C\xA5\xFB\x23\xFF\xD7\xF7\x43\xFE\xDE\xFF\xF8\xEF\xDC\xFF\xF8\x7B\xDF\xE3\x5F\xDB\xF7\x88\xDF\xEF\x88\xDF\xE7\xF8\xCB\xF6\x33\xE8\xFA\xFE\x05\xD7\xF5\xFF\x5E\xC7\xFF\x97\xD6\xF1\xA3\xCF\xEB\xF8\x3A\xBE\x89\x6F\xE2\x3B\xF8\x0E\x7E\x87\xDF\xE1\x1D\xC6\xA5\xC6\x87\x8C\x2B\x8C\x35\xC6\x55\xC6\x4D\xC6\x4D\xC6\x77\x8D\xEF\x1A\xCF\x19\xA3\x96\x38\xE5\xF5\x7C\x79\x0D\xFF\x53\x51\xF4\x20\x83\x0C\xB2\xC8\x62\x02\x26\x60\x12\x26\x61\x32\x26\x63\x0A\xA6\x60\x2A\xA6\xA2\x0E\x75\xD8\x0A\x5B\x61\x3A\xA6\x63\x06\x66\x60\x16\x66\xA1\x1E\xF5\x98\x83\x39\x48\x90\xA0\x09\x4D\x78\x11\x5E\x84\x6D\xB1\x2D\xB6\xC7\xF6\xD8\x01\x3B\x60\x47\xEC\x88\x1C\x72\x68\x43\x1B\xDA\xD1\x8E\xDD\xB0\x1B\xF6\xC0\x1E\xD8\x13\x7B\x62\x1F\xEC\x83\x7D\xB1\x2F\xF6\xC7\xFE\x38\x10\x07\xE2\x20\x1C\x84\x43\x70\x08\x0E\xC3\x61\x38\x02\x47\xE0\x48\x1C\x89\x97\xE3\xE5\x38\x1A\x47\xE3\x58\x1C\x8B\x57\xE2\x95\x78\x35\x5E\x8D\xD7\xE2\xB5\x38\x1E\xC7\xE3\x75\x78\x1D\xDE\x80\x37\xE0\x44\x9C\x88\x85\x58\x88\x93\x70\x12\x16\x63\x31\x96\x60\x09\x96\x61\x19\x4E\xC6\xC9\x58\x8E\xE5\xE8\x41\x0F\xFA\xD0\x87\x7E\xF4\x63\x10\x83\x38\x0D\xA7\xE1\x0C\x9C\x89\x37\xE3\xCD\x78\x2B\xDE\x8A\x73\x70\x2E\xCE\xC7\xF9\x78\x1B\xDE\x8E\x0B\xF1\x4E\xBC\x0B\xEF\xC6\x7B\xF1\x3E\x7C\x00\x1F\xC2\xE5\xF8\x18\xAE\xC0\x6A\x7C\x12\x9F\xC4\x95\xB8\x12\x79\x14\xB0\x16\x6B\xF1\x19\x7C\x06\x9F\xC3\x55\xB8\x06\x5F\xC0\x75\xB8\x0E\x5F\xC4\x0D\xB8\x09\x37\xE1\x6B\xF8\x1A\xBE\x81\x6F\xE0\xDB\xF8\x36\xEE\xC0\x1D\xB8\x13\x77\xE2\xFB\xF8\x3E\x7E\x88\x1F\xE2\x1E\xDC\x83\xFB\x70\x1F\x7E\x8C\x1F\xE3\x01\x3C\x80\x9F\xE2\xA7\xD8\x80\x0D\x78\x04\x8F\xE0\xE7\xF8\x39\x1E\xC5\xA3\x78\x0C\x8F\xE1\xD7\xF8\x35\x9E\xC0\x13\xF8\x0D\x7E\x83\xDF\xE2\xB7\xF8\x03\xFE\x80\x3F\xE1\x2F\x78\x06\xCF\xE0\x39\x3C\x87\x61\x0C\x23\x30\xC0\x88\x0F\xCB\xB0\x4C\x02\x93\xC0\x24\x31\x49\x4C\x32\x93\xCC\xA4\x30\x29\x4C\x2A\x93\xCA\xE8\x18\x1D\xD3\x8A\x69\xC5\xA4\x33\xE9\x4C\x06\x93\xC1\x64\x31\x59\x8C\x9E\xD1\x33\x39\x4C\x0E\x43\x18\xC2\x98\x18\x13\x73\x11\x73\x11\xD3\x96\x69\xCB\xB4\x67\xDA\x33\x1D\x98\x0E\x4C\x47\xA6\x23\xC3\x31\x1C\x63\x63\x6C\x8C\x9D\xB1\x33\xDD\x98\x6E\x4C\x3E\xD3\x97\xE9\xCF\x0C\x60\x06\x31\x83\x98\x21\xCC\x70\xE6\x12\xE6\x12\x66\x24\x33\x92\xB9\x9C\xB9\x9C\x19\xCD\x8C\x66\x7C\x0A\x9F\xC2\xAF\xF0\x2B\xA6\x29\xA6\xB5\x70\x1E\x6F\xA9\xE2\x6E\xE9\x4C\x5E\x9D\x62\xA7\x62\xA7\xE2\x03\xC5\x6E\xC5\x47\x8A\x8F\x14\xFB\x15\xFB\x15\x07\x15\x07\x15\x9F\x2A\x3E\x55\x34\x28\x1A\x14\x47\x14\x47\x14\x9F\x2B\x3E\x57\x1C\x55\x1C\x55\x1C\x53\x1C\x53\x7C\xAD\xF8\x5A\x71\x42\x71\x42\xF1\x8D\xE2\x1B\xC5\x0F\x8A\x1F\x14\x3F\x2B\x7E\x56\xFC\xAA\x38\xA3\x38\xAF\xE8\xC2\x16\xB2\x85\xEC\x24\x76\x12\x5B\xCC\x16\xB3\x25\x6C\x09\x5B\xC6\x96\xB1\x93\xD9\xC9\x6C\x39\x5B\xCE\x7A\x58\x0F\xEB\x63\x7D\xAC\x9F\xF5\xB3\x41\x36\xC8\x4E\x63\xA7\xB1\x33\xD8\x19\xEC\x4D\xEC\x4D\xEC\x2D\xEC\x2D\xEC\x2C\x76\x16\x5B\xC5\x56\xB1\x8B\xD8\x45\xEC\x62\x76\x31\x7B\x17\x7B\x17\xBB\x8C\x5D\xC6\xDE\xC3\xDE\xC3\xDE\xC7\xDE\xC7\x3E\xC0\x3E\xC0\x3E\xC4\x3E\xC4\x3E\xC2\x3E\xC2\x2E\x67\x97\xB3\x8F\xB3\x8F\xB3\xD5\x6C\x35\xFB\x24\xFB\x24\xBB\x92\x5D\xC9\xF2\x2C\xCF\xD6\xB0\x35\x6C\x88\x0D\xB1\xCF\xB0\xCF\xB0\xCF\xB1\xCF\xB1\xAB\xD9\xD5\xEC\xF3\xEC\xF3\xEC\x5A\x76\x2D\xBB\x9E\x5D\xCF\x6E\x60\x37\xB0\x2F\xB3\x2F\xB3\x9B\xD8\x4D\xEC\x16\x76\x0B\x5B\xCF\x6E\x63\x5F\x65\x5F\x65\x5F\x67\x5F\x67\xDF\x60\xDF\x60\xDF\x62\xDF\x62\xDF\x61\xDF\x61\x77\xB0\x3B\xD8\x5D\xEC\x2E\xF6\x03\xF6\x03\x76\x37\xBB\x9B\xDD\xCB\xEE\x65\x3F\x62\x3F\x62\xF7\xB3\xFB\xD9\x83\xEC\x41\xF6\x53\xF6\x53\xB6\x81\x6D\x60\x8F\xB0\x47\xD8\xCF\xD9\xCF\xD9\xA3\xEC\x51\xF6\x18\x7B\x8C\xFD\x9A\xFD\x9A\x3D\xC1\x9E\x60\xBF\x61\xBF\x61\x4F\xB3\xA7\xD9\xEF\xD8\xEF\xD8\x1F\xD8\x1F\xD8\x9F\xD8\x9F\xD8\x5F\xD8\x5F\xD8\x33\xEC\x19\xF6\x1C\x7B\x8E\x0D\xB3\x61\x16\x94\xA0\x64\x94\x8C\x92\x55\xB2\xCA\x04\x65\x82\x32\x49\x99\xA4\x4C\x56\x26\x2B\x53\x94\x29\xCA\x54\x65\xAA\x52\xA7\xD4\x29\x5B\x29\x5B\x29\xD3\x95\xE9\xCA\x0C\x65\x86\x32\x4B\x99\xA5\xD4\x2B\xF5\xCA\x1C\x65\x8E\x92\x28\x89\xD2\xA4\x34\x29\x2F\x52\x5E\xA4\x6C\xAB\x6C\xAB\x6C\xAF\x6C\xAF\xEC\xA0\xEC\xA0\xEC\xA8\xEC\xA8\xE4\x94\x9C\xD2\xA6\xB4\x49\xE7\x2B\x63\xCF\xC9\x55\xB7\xFE\xB1\x75\xE3\xD9\x78\x02\x62\x37\x23\x2C\x61\x49\x02\x49\x20\x49\x24\x89\x24\x93\x64\x92\x42\x52\x48\x2A\x49\x25\x3A\xA2\x23\xAD\x48\x2B\x92\x4E\xD2\x49\x06\xC9\x20\x59\x24\x8B\xE8\x89\x9E\xE4\x90\x1C\x22\x3E\x26\x62\x22\x17\x91\x8B\x48\x5B\xD2\x96\xB4\x27\xED\x49\x07\xD2\x81\x74\x24\x1D\x09\x47\x38\x62\x23\x36\x62\x27\x76\xD2\x8D\x74\x23\x3D\x48\x0F\xD2\x93\xF4\x24\xBD\x49\x6F\x92\x4F\xF2\x49\x3F\xD2\x8F\x0C\x20\x03\x48\x01\x29\x20\x83\xC9\x60\x32\x94\x0C\x25\xC3\xC9\x70\x72\x09\xB9\x84\x8C\x24\x23\xC9\xE5\xE4\x72\x32\x9A\x8C\x26\x63\xC9\x58\x72\x25\xB9\x92\x5C\x4D\xAE\x26\xD7\x92\x6B\xC9\x78\x32\x9E\x5C\x47\xAE\x23\x37\x90\x1B\xC8\x44\x32\x91\x14\x92\x42\x32\x89\x4C\x22\xC5\xA4\x98\x94\x90\x12\x52\x46\xCA\xC8\x64\x32\x99\x94\x93\x72\xE2\x21\x1E\xE2\x23\x3E\xE2\x27\x7E\x12\x24\x41\x32\x8D\x4C\x23\x33\xC8\x0C\x72\x13\xB9\x89\xDC\x42\x6E\x21\xB3\xC8\x2C\x52\x45\xAA\xC8\x5C\x32\x97\xCC\x27\xF3\xC9\x6D\xE4\x36\x72\x07\xB9\x83\x2C\x22\x8B\xC8\x62\xB2\x98\xDC\x45\xEE\x22\xCB\xC8\x32\x72\x0F\xB9\x87\xDC\x47\xEE\x23\x8F\x93\xC7\x49\x35\xA9\x26\x4F\x92\x27\xC9\x4A\xB2\x92\xF0\x84\x27\x35\xA4\x86\x84\x48\x88\x3C\x43\x9E\x21\xCF\x91\xE7\xC8\x6A\xB2\x9A\x3C\x4F\x9E\x27\x6B\xC9\x5A\xB2\x9E\xAC\x27\x1B\xC8\x06\xF2\x32\x79\x99\x6C\x22\x9B\xC8\x16\xB2\x85\xD4\x93\x7A\xF2\x0A\x79\x85\xBC\x46\x5E\x23\xDB\xC9\x76\xF2\x26\x79\x93\xBC\x4D\xDE\x26\xEF\x92\x77\xC9\x7B\xE4\x3D\xB2\x8B\xEC\x22\x1F\x90\x0F\xC8\x6E\xB2\x9B\xEC\x25\x7B\xC9\x47\xE4\x23\xB2\x9F\xEC\x27\x07\xC9\x41\xF2\x29\xF9\x94\x34\x90\x06\x72\x84\x1C\x21\x9F\x93\xCF\xC9\x51\x72\x94\x1C\x23\xC7\xC8\xD7\xE4\x6B\x72\x82\x9C\x20\xDF\x90\x6F\xC8\x69\x72\x9A\x7C\x47\xBE\x23\x3F\x90\x1F\xC8\x4F\xE4\x27\xF2\x0B\xF9\x85\x9C\x21\x67\xC8\x39\x72\x8E\x84\x49\x98\x24\x19\xB5\xD2\x39\xC1\x0E\xC6\x2E\xC6\xDE\xC6\x81\xD2\x59\xC1\xE6\xE7\x04\x1D\xC6\x32\xA3\xD7\xB8\xC1\xB8\xD1\xF8\xA6\xF1\x1D\xE3\x47\xC6\x03\xC6\x63\xC6\x13\xC6\x5F\x8C\x67\x8D\xD1\xB3\x80\x6A\xB3\xC6\xAC\x33\xB7\xBA\xE0\x79\x40\xBB\xB9\xBB\x79\x80\x74\x1E\x70\x98\x79\x9E\x79\x9E\x74\x16\xB0\x90\x2B\xE4\x26\x71\xC5\x5C\x29\x57\xCA\xB9\x39\x37\x37\x85\x9B\xC2\x55\x70\x3E\xCE\xCF\xF9\xB9\x4A\xAE\x92\x9B\xC5\x55\x71\x73\xB9\xB9\xDC\x7C\x6E\x3E\x77\x1B\x77\x1B\x77\x07\x77\x07\xB7\x88\x5B\xC4\x2D\xE6\x16\x73\x77\x71\x77\x71\xCB\xB8\x65\xDC\x3D\xDC\x3D\xDC\x7D\xDC\x7D\xDC\x03\xDC\x03\xDC\x43\xDC\x43\xDC\x23\xDC\x23\xDC\x72\x6E\x39\xF7\x38\xF7\x38\x57\xCD\x55\x73\x4F\x72\x4F\x72\x2B\xB9\x95\x1C\xCF\xF1\x5C\x0D\x57\xC3\x85\xB8\x10\xF7\x0C\xF7\x0C\xF7\x1C\xF7\x1C\xB7\x9A\x5B\xCD\x3D\xCF\x3D\xCF\xAD\xE5\xD6\x72\xEB\xB9\xF5\xDC\x06\x6E\x03\xF7\x32\xF7\x32\xB7\x89\xDB\xC4\x6D\xE1\xB6\x70\xF5\x5C\x3D\xF7\x0A\xF7\x0A\xF7\x1A\xF7\x1A\xB7\x9D\xDB\xCE\xBD\xC9\xBD\xC9\xBD\xCD\xBD\xCD\xBD\xCB\xBD\xCB\xBD\xC7\xBD\xC7\xED\xE2\x76\x71\x1F\x70\x1F\x70\xBB\xB9\xDD\xDC\x5E\x6E\x2F\xF7\x11\xF7\x11\xB7\x9F\xDB\xCF\x1D\xE4\x0E\x72\x9F\x72\x9F\x72\x0D\x5C\x03\x77\x84\x3B\xC2\x1D\xE7\x8E\x73\x27\xB9\x93\xDC\x77\xDC\x77\x5C\x01\x5F\xC0\x0F\xE6\x07\xF3\x43\xF9\xA1\xFC\x70\x7E\x38\x7F\x09\x7F\x09\x3F\x92\x1F\xC9\x5F\xCE\x5F\xCE\x8F\xE6\x47\xF3\x63\xF9\xB1\xFC\x95\xFC\x95\xFC\xD5\xFC\xD5\xFC\xB5\xFC\xB5\xFC\x78\x7E\x3C\x7F\x1D\x7F\x1D\x7F\x03\x7F\x03\x3F\x91\x9F\xC8\x17\xF2\x85\xFC\x24\x7E\x12\x5F\xCC\x17\xF3\x25\x7C\x09\x5F\xC6\x97\xF1\x93\xF9\xC9\x7C\x39\x5F\xCE\x57\xF1\x55\xFC\x5C\x7E\x2E\x3F\x9F\x9F\xCF\xDF\xC6\xDF\xC6\xDF\xC1\xDF\xC1\x2F\xE2\x17\xF1\x8B\xF9\xC5\xFC\x5D\xFC\x5D\xFC\x32\x7E\x19\x7F\x0F\x7F\x0F\x7F\x1F\x7F\x1F\xFF\x00\xFF\x00\xFF\x10\xFF\x10\xFF\x08\xFF\x08\x7F\x91\x70\x91\xD0\x56\x68\x2B\xB4\x17\xDA\x0B\x1D\x84\x0E\x42\x47\xA1\xA3\xC0\x09\x9C\x60\x13\x6C\x42\x37\xA1\x9B\xD0\x43\xE8\x21\xF4\x14\x7A\x0A\xBD\x85\xDE\x42\xBE\x90\x2F\xF4\x13\xFA\x09\x03\x84\x01\x42\x81\x50\x20\x0C\x16\x06\x0B\x43\x85\xA1\xC2\x70\x61\xB8\x70\x89\x70\x89\x30\x52\x18\x29\x5C\x2E\x5C\x2E\x8C\x16\x46\x0B\x63\x85\xB1\xC2\x95\xC2\x95\xC2\xD5\xC2\xD5\xC2\xB5\xC2\xB5\xC2\x78\x61\xBC\x70\x9D\x70\x9D\x70\x83\x70\x83\x30\x51\x98\x28\x14\x0A\x85\xC2\x24\x61\x92\x50\x2C\x14\x0B\x25\x42\x89\x50\x26\x94\x09\x93\x85\xC9\x42\xB9\x50\x2E\x78\x04\x8F\x30\x53\x98\x29\xDC\x2C\xDC\x2C\xDC\x2A\xCC\x12\xAA\x84\x2A\x61\xAE\x30\x57\x98\x2F\xCC\x17\x6E\x13\x6E\x13\xEE\x14\xEE\x14\x96\x08\x4B\x84\x65\xC2\x32\xE1\x1E\xE1\x1E\xE1\x01\xE1\x01\xE1\x21\xE1\x21\xE1\x11\xE1\x11\x61\xB9\xB0\x5C\x78\x5C\x78\x5C\xA8\x16\xAA\x85\x27\x85\x27\x85\x95\xC2\x4A\x81\x17\x78\xA1\x46\xA8\x11\x42\xC2\x73\xC2\x6A\x61\xAD\xB0\x5E\x58\x2F\x6C\x10\x36\x08\x2F\x0B\x2F\x0B\x9B\x84\x4D\xC2\x16\x61\x8B\xF0\x8A\xF0\x8A\xF0\x9A\xF0\x86\xF0\x96\xF0\x96\xF0\xA3\xF0\xA3\x60\x8E\xE4\x37\xBB\x9F\x1A\xBD\xF7\x1E\x3D\x7F\x18\x3D\x77\x17\xBD\xC7\xDE\x68\x9B\x77\x37\x42\xF6\x6E\x84\x1E\xBB\x11\x46\xED\x46\x28\xDB\x8D\x30\x87\x9E\x05\xFA\xB3\xE7\x81\xD6\xED\x46\xD8\xB5\x1B\xE1\xAB\xDD\x08\xEC\x9E\xDF\x3F\x07\xD4\xFB\x4F\x9E\x03\x2A\xDF\x83\xB0\x60\x0F\x42\xF5\x1E\x84\x97\xF6\x20\xEC\xDE\x83\x70\x72\x0F\x42\xD2\x5E\x04\xF3\x5E\x84\x7E\x7B\x11\xAE\xD9\x8B\xE0\xDB\x8B\xB0\x30\xE6\xBC\xD0\x11\x7A\x4E\x28\x87\x9E\x0F\xBA\x96\x9E\x0B\x5A\x46\xCF\x03\xBD\x4D\x29\x7A\x3F\x3F\xBA\x9E\x11\x5D\x07\x18\x48\x09\x14\x5E\xC5\x3C\x76\x21\x7B\xB7\x72\xB3\x72\xAB\x72\xAB\x72\x9B\xF2\x15\xE5\x6B\xCA\xD7\x95\x6F\x28\xDF\x50\xEA\x12\xF4\x09\x97\x25\x38\xA4\x7B\x21\x7B\x13\x0E\x26\x1C\x4A\xF8\x3C\xE1\xCB\x84\x63\x09\x5F\x27\x9C\x4C\x48\x4B\x4C\x4B\xB4\x27\x8E\x4C\xE4\x13\x57\x27\x1E\x4F\xFC\x3E\xF1\x7C\xE2\xF9\xC4\x8C\xA4\xEC\x24\x43\x52\x5E\x52\xBB\x24\x4B\x92\x35\xA9\x73\xD2\x84\xA4\xEB\xE9\x5D\x7C\x85\x2A\x5F\xD5\x4F\x35\x40\x35\x46\x75\x85\x6A\x9C\x6A\x92\xAA\x48\x35\x47\x35\x4F\xB5\x49\xB5\x49\xB5\xA5\xF1\x8E\xF7\x0E\xE9\x8E\xF7\x11\xD5\x67\xAA\xB0\x2A\xAC\xC2\x64\x45\x8B\x77\x81\x7C\xC9\x53\x93\x2B\x93\x2B\x93\xE7\x24\xCF\x4B\xDE\x94\xBC\x29\x79\x4B\xCC\x1D\xA5\x23\xC9\x9F\x25\x9F\x49\x8E\x24\xA3\x5A\xA1\xEE\xA7\xEE\xA7\x1E\xA0\x1E\xAA\x1E\xA1\xBE\x44\x7D\x99\x7A\x94\xFA\x1A\xF5\x38\xF5\x24\x75\x91\x7A\xAE\x7A\xAE\x7A\x8B\xFA\x15\xF5\x76\xF5\x9B\xEA\xB7\xD5\x3B\xD4\xFB\xD4\xFB\xD4\x90\xC2\xA6\x0C\x48\x19\x9A\x32\x3C\xE5\x92\x94\x91\x29\xA3\x52\xAE\x49\xB9\x36\x65\x52\x4A\x51\xCA\x9C\x94\x79\x29\x9B\x52\x36\xA5\x6C\x89\xBB\x3F\x72\x24\xE5\xB3\x14\xD0\x28\x34\x7D\x35\xFD\x34\x03\x34\x43\x35\xC3\x35\x97\x68\x46\x6A\x46\x69\xC6\x69\xC6\x69\x26\x69\x8A\x34\x73\x34\xF3\x34\x6F\x6B\xDE\xD6\xEC\xD4\xEC\xD6\xEC\xD5\xEC\xD5\x7C\xA4\x39\xA4\xF9\x4E\xF3\xBD\xA6\x6B\x6A\xD7\xD4\x1E\xA9\xF9\xA9\x23\x52\x47\xA7\xAE\x49\x5D\x93\xBA\x36\x75\x93\x74\x1E\x28\x4B\x9B\xAD\xBD\x58\x7B\xB1\xB6\x97\xB6\x97\xB6\x8F\xB6\x8F\x76\x80\x76\xA0\x76\xAA\x76\xBE\xF6\x36\xED\xED\xDA\x25\xDA\x07\xB5\x0F\x6B\x37\x69\xB7\x6B\xB7\x6B\x3B\xE9\x06\xE8\xAE\xD5\x4D\xD0\x4D\xD4\x15\xEA\x26\xE9\x8A\x75\xA5\xBA\x0A\xDD\x54\x5D\x50\x37\x57\xB7\x44\xB7\x54\xB7\x54\xF7\x88\xEE\x31\xDD\x8D\xAD\x1D\xAD\x5B\x65\xA6\x67\x76\xCB\xEC\x91\x79\x65\xE6\x55\x99\xFE\xCC\x40\xE6\xDA\xCC\x0F\x33\x3F\xC9\xFC\x24\x53\x9D\xA5\xC9\x5A\x90\x75\x5B\x56\x6D\x56\x6D\x56\xF4\x8C\x4F\xA6\xDE\xA0\xBF\x46\x3F\x51\x5F\xA8\xBF\x45\x3F\x5B\x3F\x5B\xBF\x5A\x5F\xAF\x07\x03\x6B\xE8\x61\x18\x6A\x98\x62\x08\x18\xAA\x0C\x73\x0D\x4F\x18\x9E\x35\x1C\x35\x7C\x6F\x68\x9B\xD3\x2B\x47\xBE\x17\xD4\x74\x27\xE8\xE7\x98\x3B\x41\xBB\x4C\xA7\x4D\xA7\xB8\x6F\xB9\xD9\x9D\x67\x77\x6E\xE8\x1C\xE9\xDC\xD1\xDE\x45\xBA\x1B\xE4\xE5\xFD\x7C\x90\xBF\x95\x7F\x9C\x5F\xC1\x9F\xE6\xBF\xE5\x99\x1A\xA6\x26\xA1\x26\xA1\x46\x5D\xA3\xAE\xC9\xAB\xB1\x88\x22\xA0\xA6\xAA\x66\x4E\xCD\xDA\x9A\xD7\x6B\x1A\x6A\xBE\xAD\x89\xD4\x44\x6A\xDA\xD7\x76\xAE\x1D\x51\x7B\x55\x6D\x55\xED\xBC\xDA\x17\x6A\xEB\x6B\xBF\xA8\xFD\xA2\xD6\x1A\xEA\x19\x1A\x12\x1A\x12\xBA\x3C\x34\x2A\x74\x73\xE8\xD6\xD0\xEA\xD0\xEA\xD0\xF3\xA1\xB5\xA1\x17\x43\x1B\x42\x5B\x42\x5B\x43\xDB\x42\xDB\x42\xC7\x43\xA7\x42\x3F\x86\x7E\x0A\x7D\x56\xF7\x75\xDD\x89\xBA\x93\x75\xE4\x2C\x39\x2B\xDF\x4D\xEA\x22\xD9\x7C\x88\xBF\x9F\x34\x4D\x81\x70\x93\x02\x01\x5B\xB8\xB7\xA3\x8D\xBB\xA7\x23\xD2\x17\xC9\x08\x47\x93\x11\xDA\xD2\x7B\x5D\x75\xA9\x08\xCF\xA4\x22\x0C\xD7\x22\x8C\xD5\xA2\x64\x5F\x80\xD1\x21\xF4\xD6\x21\x0C\xA7\xB6\x05\xE6\xEA\x10\x56\xEB\x10\x5E\xD6\xA1\x74\x1F\x9E\x49\x43\xB0\xD0\x7B\xF1\xC3\xD2\x10\x86\xA7\x21\x04\x28\x55\xA5\x21\xCC\x4D\x43\x78\x21\x0D\xA1\x3E\x0D\xE1\xAD\x34\x84\x77\xD3\x10\xDE\x4B\x43\xD8\x99\x86\xC0\xB5\x42\xE8\x45\xEF\xFE\x88\x74\x88\xDE\xFF\x89\xDE\x7B\xEC\x4B\xEF\x32\x0E\xB8\xC0\x7D\xC6\xE8\x3D\xC6\x49\x17\xB8\xC7\x78\x71\x3A\xC2\xF0\x74\x84\x89\x94\x56\xA7\x23\xBC\x9A\x8E\xB0\xAA\x0D\xC2\xBA\x36\x08\x1B\xDA\x20\xD4\xB7\x41\x38\xD8\x06\xE1\x93\x36\x08\xF6\x0C\x84\x82\x0C\x84\xBA\x0C\x84\x17\x33\x9A\xEC\x78\x70\x59\x08\xF9\x59\x08\xF6\x16\xEE\xBB\x0E\xC8\x46\x28\xA0\x34\x38\x1B\x61\x48\x36\xC2\xFB\xD9\x08\xFB\xB2\x11\xF6\x67\x23\x34\x64\x23\x7C\x4E\x09\xF5\x08\xC9\x7A\x84\xEE\x7A\x84\x3E\x7A\x84\xBE\x7A\x84\x01\x7A\x84\x11\x94\xAE\xD0\x23\x5C\xAF\x47\x58\xA4\x47\x78\x58\x8F\xD0\x25\xC6\x0E\xC4\xC0\x1C\x84\x7B\x72\x10\x84\x1C\x84\xDA\x1C\x84\xF5\x39\x08\x5D\x73\x11\x7A\xE6\x22\xE4\x53\xEA\x97\x8B\xD0\x3F\x17\x61\x60\x2E\xC2\xB0\x5C\x84\x11\x94\x16\xB5\x70\x8F\xF1\xC1\x5C\x84\xEF\x09\xC2\x4F\x44\xB6\x67\xF1\xC3\x64\x04\xFB\x14\x84\x9E\x53\x10\xC6\x78\x65\xBA\xC2\x8B\x70\xBB\x17\x61\xA9\x17\xE1\x1E\x6F\xD3\x7D\xAE\xE8\x3D\xC4\xC7\x36\x21\x3C\xBE\x09\xC1\x45\xEF\xC3\x55\xBC\x8F\xE0\xFF\x9D\x7B\x71\x83\x3F\x40\x18\xFA\x01\x02\xEC\x47\xE8\xB9\x1F\xA1\xEF\x7E\x84\xF2\xFD\x08\x95\x94\xE6\x53\x7A\x74\x3F\xC2\x8A\xFD\x08\x4F\xEC\x47\x58\xB5\x1F\xE1\x4F\xDF\x9B\x3D\x8C\xD0\xF3\x30\xC2\x89\x23\x08\xA7\x8E\x20\xEC\x3A\x86\xB0\xF7\x18\xC2\xD0\xAF\x10\x46\x7E\xD5\x74\x2F\x6F\x2C\x5D\x47\x6D\x20\x08\xBF\x90\x98\x33\x7A\x8D\xFB\x4B\x51\x1C\x1C\xA3\xB9\x5A\x53\xA4\x29\xD1\xCC\xD6\x14\x24\x5D\x9F\x34\x31\x69\x62\x52\xAC\xB8\xAE\xA7\xE3\xE7\x78\x32\xC2\x4F\xC9\x08\x3E\xBA\xAE\x17\xDF\x4E\x8D\x4F\x0B\xEB\x7D\x17\x6A\x3F\xB1\xDD\xC6\xE6\x22\x4C\xC8\x6D\x7A\x5F\xA1\x52\xA8\xFA\xAA\xFA\xAA\x06\xA8\x0A\x54\x97\xAA\x2E\x57\x8D\x56\x8D\x51\xCD\x55\xCD\x53\x6D\x51\xD5\x53\xB9\xF3\xAE\x24\x77\x14\xC9\x8A\xE4\x01\xC9\x05\xC9\xF3\x92\xE7\x25\x6F\x49\xAE\x4F\x7E\x4B\x92\x29\xEF\x26\x33\x6A\x85\x7A\x80\x7A\x80\xBA\x40\x5D\x40\x65\xC9\xE5\xEA\x71\xEA\x71\xEA\x2D\xEA\xAD\xEA\x6D\x8D\x72\xE4\x5D\x49\x8E\x60\x8A\x22\x65\x50\xCA\xD0\x94\xB9\x29\xF3\x52\xB6\xA4\x6C\x49\xA9\x4F\x79\x2D\xE5\x0D\x49\x6E\xBC\x23\xC9\x0D\x46\xA3\xD0\x0C\xD0\x14\xD0\x7A\xBA\x5C\x92\x17\x73\x35\xF3\x34\x3B\x35\xEF\x37\xCA\x08\x11\xFB\x67\x6B\x67\x6B\x3B\xE9\x38\x5D\x57\x5D\x57\x5D\x6F\x5D\x6F\x5D\x5F\x5D\x3F\xDD\xB5\xBA\x71\x8D\x98\x3F\x4F\x37\x5F\x77\xBB\x6E\x31\xC5\xFC\xE5\xBA\xF5\x99\xEB\x33\xB7\x64\xBE\x9E\xF9\x46\xE6\x9B\x99\x79\xD9\xED\xB3\xAD\x12\xB6\x77\xCD\xEE\x9E\xDD\x3B\x3B\x5B\xAF\xD7\x5F\xA3\xBF\x46\x3F\x4E\x3F\x4E\xEF\xD4\x3B\xF5\x45\xFA\x62\x7D\x85\xDE\xAF\x67\x0D\xAC\xE1\x62\xC3\xC5\x86\xBE\x86\xBE\x86\xFE\x86\x41\x86\x21\x86\xA1\x86\xB9\x12\xD6\x3F\x61\xE0\x0D\x82\x21\x64\x08\x19\xBE\x34\x7C\x69\xF8\xDA\x70\xC2\xF0\x8D\xE1\x1B\xC3\x77\x12\xFE\x77\xCA\xE9\x91\x73\xB1\x64\x0B\xE3\xE7\x9C\x9F\x73\x6C\xF6\x2E\xF6\xBC\x9A\xB6\x35\x96\x1A\x0B\xC5\xEF\x57\x6B\xAE\xAC\xBD\xAA\x76\x5E\xED\xBC\xDA\xB5\xB5\xEB\x6A\x5F\xAE\xDD\x58\xBB\xA5\xB6\xBE\xB6\x4B\xC8\x1E\xEA\x1E\xEA\x11\x1A\x25\xE1\xB6\x88\xDC\xB7\x86\x8E\x87\x8E\x87\xBE\x91\xB0\xFA\xC7\xD0\x67\x75\x9F\xD7\x1D\xAD\xFB\xB2\xEE\x2B\x09\xB3\x4F\x34\xBB\xBF\x2C\xE2\x69\x14\x53\x19\x8A\xA5\x51\x3C\x7D\x5E\x27\xDB\x76\xDE\xA0\x43\xE0\xD2\x64\x8A\xE2\xE9\xDC\x18\x1C\x5D\x97\x86\xB0\x95\xE2\xE9\x7B\x94\x44\x1C\xB5\xB5\x42\xE8\xD6\x0A\xA1\x7B\x2B\x84\x8B\x29\x35\xB4\x42\x38\xD2\x0A\x81\xA1\x78\x2A\xE2\xE8\xC0\xD6\x08\x83\xFE\x24\x9E\x8A\xF8\xD9\x2B\x1D\xA1\x20\x1D\x61\x50\x3A\xC2\x30\x4A\x22\x8E\x3E\x9F\x8E\xF0\x12\xA5\x8D\xE9\x08\x5B\xD2\x11\xB6\x51\x12\xF1\x75\x0D\xC5\xD7\x8D\x6D\x10\xB6\x50\x12\xF1\xB5\x5B\x06\x42\xDF\x0C\x84\x7E\x19\x08\x03\x28\x3D\x43\xE9\xB9\x0C\x84\x55\x19\x08\xEB\x29\x99\x32\x11\xCC\x99\x08\xED\x29\x89\x38\x6C\xCB\x42\xE8\x4D\x29\x1E\x8F\xFB\x53\x8A\xC5\xE3\xC1\x14\x8F\x3F\xCC\x46\x38\x98\x8D\x70\x28\x06\x8F\xFB\x50\x1C\x1E\xA7\x47\x18\xAF\x47\x78\x90\x92\x88\xBB\x22\x0D\xA0\x54\x4B\x69\x0D\xA5\xB5\x94\xA2\xB8\x2A\xE2\xE9\x7D\xB9\x08\x0F\x50\xFA\x91\xE2\x6A\x14\x3F\xA3\x78\xE9\xA2\xF6\x42\xA2\x38\x79\x07\xBB\x90\xDD\xA2\xDF\xA2\xFF\xC4\xD4\x60\x3A\x62\xFA\xDC\xE4\xE3\xFD\xBC\xA8\xE3\x3F\xE2\x41\x68\x08\xFD\x14\xAA\xAF\x3B\x59\x77\xBA\xEE\xD7\xA6\x3E\x44\xD7\xC3\xDF\xA0\xB8\x56\x4D\xF7\xBF\x36\xA9\x10\x76\xAA\x10\x3E\x50\x21\x44\xE8\xBE\xD8\x0A\x55\xD3\xFA\xFD\x98\x18\x1C\x5F\xE1\x8D\xC1\x23\x49\x5F\xBE\xFD\x2F\xD6\x97\x19\x55\xBE\xA4\x31\xF7\x53\x0D\x52\x5D\xA2\x1A\xA5\x1A\x15\xA3\x37\xCF\x91\xF4\xE6\x6D\xAA\xD7\x54\x3B\x54\x3B\x62\xF4\x65\x46\xD2\x97\x07\x5D\x50\x5F\x9E\x2B\xE9\xCB\xDB\x24\x7D\xF9\xCD\xE4\x1D\xCD\xF4\x65\x94\xF4\xE5\x81\xEA\x81\xEA\x41\xEA\xA1\xEA\x51\x92\xBE\x7C\x6D\xA3\xBE\x5C\xAF\xAE\x57\xEF\x50\xEF\x50\x43\x8A\xA8\x29\x8B\xBA\x72\x41\x0B\xBA\xF2\x1C\x49\x57\xDE\x9A\xB2\x35\x65\x7B\xCA\xF6\x94\x77\x53\x76\x50\x3D\x19\x25\x3D\x79\x90\x66\xA8\x66\x94\x66\x14\xD5\x8F\xE7\x48\xFA\xF1\x07\x54\x3F\xFE\xB3\x7A\xF1\x54\xED\x2C\x6D\x55\x0B\x7A\x71\x67\x9D\x5D\xD7\x4D\xD7\x4B\xD7\x47\x97\xAF\xEB\xAF\x1B\xA0\x1B\x4F\x75\x64\x59\x37\x9E\xAB\x5B\xA0\xBB\x4D\xB7\x44\xB7\x44\xF7\xD8\x05\xF5\xE3\x75\x99\x2F\x66\x6E\xCE\xDC\x9E\xB9\x3D\xF3\xAD\x16\x75\xE5\x8B\xB2\x2D\xD9\x1D\xB2\xBB\x65\x77\xCB\xEE\x23\xE9\xCC\x59\x7A\x83\xDE\xA0\xBF\x56\x7F\xAD\x7E\xBC\xA4\x3B\x17\xEA\x27\xE9\x27\xE9\x5D\xFA\x72\x7D\xA0\x51\x8F\xDE\xAC\xDF\x2A\xE9\xD2\x0A\x43\x0F\x43\x0F\x43\x4F\x43\xBE\xA1\x9F\xA1\x9F\x61\xB0\x61\x30\xD5\xAB\xE7\x18\x9E\x34\x3C\x6D\xA8\x31\xD4\x1A\xEA\x24\xFD\xFA\xA8\xE1\x98\xE1\x2B\xC3\x49\xC3\x49\xC3\x29\xC3\xB7\x06\x2E\xA7\x7B\x4E\xCF\x46\x7D\xFB\xC2\xF7\xF0\x77\x99\x0E\x9A\x0E\x9B\x0E\x9B\xBE\xF8\x8D\xEE\xDD\x99\xEA\xDE\xDE\x0B\xEA\xDE\xED\x6A\xDA\x4B\xBA\xF7\x6B\xBF\xD1\xBB\xAF\xA8\xAD\xAA\x9D\x5B\xFB\x42\xED\x0B\xB5\xEB\x6B\x5F\xAA\xDD\x54\xBB\x99\xEA\xDF\xB6\x50\xD7\x50\xB7\xD0\xC5\x8D\x7A\xF8\xE5\xA1\x5B\x42\xB7\xB4\xA8\x87\x9F\x08\x9D\x0C\xFD\x14\xFA\x29\xF4\x45\xDD\x17\x75\xC7\xEA\x8E\xD5\x9D\x6C\x41\x17\xFF\x77\xE8\xE3\xD2\xF7\x0A\xE2\xF4\xF1\x39\x3A\x84\x17\x74\x08\xEB\x75\x08\x2F\xB5\xA0\x97\x77\x4A\x43\xE8\x4C\xF5\xF3\x58\xBD\x7C\x4E\x1A\xC2\xFA\x34\x84\x2D\x31\x7A\xF9\x4E\x4A\x5D\x5A\x21\x74\x6D\x85\xD0\x83\x52\xCF\x16\xF4\x74\x91\x3E\x8B\xD1\xD7\x31\x46\x5F\x2F\xA0\x14\xAF\x8F\xF7\x4E\x47\x18\x98\x8E\x30\x38\x1D\x61\x28\xD5\xCD\x63\xF5\xF3\x17\xD2\x11\x36\xA4\x23\xBC\x4C\x69\x6B\x3A\x42\x7D\x3A\xC2\x2B\x54\x6F\x7F\x9E\xEA\xED\x9B\xDA\x20\x6C\x6E\x83\xB0\x35\x4E\x7F\xEF\x9E\x81\x90\x9F\x81\xD0\x9F\xD2\xC0\x18\x7D\x5E\xA4\x67\x29\xAD\xCE\x40\x58\x47\x75\xFC\xA8\x9E\x6F\xCC\x44\xB8\x28\x13\xA1\x5D\x26\x82\x85\xEA\xFD\x5D\xB2\x10\x7A\x65\x21\xF4\xA1\xFA\xBF\xA8\xEF\xF7\xA3\x3A\xFF\x00\xAA\xEB\x8B\xB4\x3B\x46\xDF\x3F\x40\x75\xFE\x86\x38\x7D\xBF\xF7\x05\xF4\xFD\x6B\xF5\x08\x13\x62\xF4\xFE\x07\xF4\x08\x0F\x51\xFD\xDF\x1E\xA3\xFF\xF7\xA7\x73\x80\xD8\x79\x40\x28\x07\x61\x75\x0E\xC2\xF3\x39\x08\x2F\xE4\x20\xAC\xFB\x07\xE7\x05\xD1\xF9\xC0\xFD\x94\x1E\x8C\x99\x17\xFC\xF0\x3B\xF3\x82\xF8\xF9\x40\x74\x1E\xF0\x47\x76\x31\xFE\x5B\xF4\xFF\xC6\x87\x9E\x2B\x18\x9B\x81\x30\x21\x43\x3A\x47\x90\xC5\x66\xB1\x59\x09\x59\xA9\x59\xBA\xAC\xEC\x2C\xF9\x4E\x7D\x28\xAB\x30\x03\xA1\x3C\xA3\x69\xDF\xBC\x70\x32\x82\x7B\x32\x82\x67\x32\x82\x37\xC6\x46\xE0\x9F\x39\x67\x51\x2B\x8E\x17\xDD\x0A\x5D\x43\x6D\xB8\xB6\x30\x34\x3B\x14\xAB\x1E\xDA\xA1\x0F\x3C\x0F\x2F\xC0\x4B\xF0\x12\x6C\x82\x2D\x50\x98\xE0\x4E\x38\x9D\xF0\x6B\x42\x7D\xE2\x5B\x89\x25\x2A\xAF\xEA\xA8\xEA\x94\xEA\x07\xD5\xAF\xAA\x92\x64\x6F\xF2\xD1\xE4\x53\xC9\x25\x6A\xAF\xDA\xAF\x9E\xAE\x3E\xAA\xFE\x4E\x5D\x92\xE2\x4D\x99\x91\x32\x4B\xB2\x03\x32\x5E\x33\x51\x53\xA2\x99\xA1\x39\xAA\x39\xA5\x19\x9B\x3A\x41\xBA\xA3\x65\xD2\x76\xD7\x16\xE8\x2E\xD5\x2D\xD3\x3D\xAC\x73\xB7\xBE\xA5\xF5\x37\x19\xA7\x33\xA2\xB6\x38\xC4\xB2\x0E\xCF\x1E\x93\xBD\x2B\x7B\x7F\x76\x95\x7E\xA1\x7E\x99\xFE\x61\xFD\x58\xC3\x04\xC3\x6A\xC3\x4B\x86\x82\x9C\x4B\x73\xC6\xE6\x4C\xC8\xF1\x49\x76\xE1\x66\x9A\xAA\x4C\x0B\x4D\x63\xCD\x73\xCD\x0B\xCC\x0B\xCD\x85\x6D\x1F\x6D\xFB\x75\xDB\x48\xDB\x69\x96\x7B\x2D\xE7\xB9\xF3\x5C\x93\xFD\x9A\x7C\xFB\x3D\x5D\xEF\xEF\x6A\xEA\x66\xED\x76\x49\xB7\x31\xDD\xAE\xE8\xE6\xE8\x56\xD5\x6D\x61\xB7\x35\xDD\xB6\x76\x33\xF1\x56\xFE\x28\x7F\x8A\xB7\xD7\x5C\x5C\xB3\xAB\xE6\xE3\x1A\xA8\x55\x49\x36\x50\x4E\xD7\xFE\x5A\x3B\x36\x34\x21\x24\xEA\x3B\xBA\x48\xB6\xB4\x2E\x1A\xB5\xD3\x56\x80\x08\x33\x10\x61\x11\x22\xDC\x19\x63\x9F\xCD\xA4\x40\xC8\x53\x20\x0C\xA2\x34\x92\xD2\xFB\x0A\x84\x3D\x0A\x84\x6A\x16\xA1\x96\x45\x18\x4F\xED\xC3\xCC\x4C\x42\x98\x9D\x84\x20\x24\x21\xAC\x8A\xB1\x4F\x93\xA1\x42\x30\x50\x3D\x6A\xB3\x0A\xA1\x3E\x4E\x9F\x2A\xA0\x76\x4D\x6E\x4D\x46\x98\x95\x8C\xF0\x58\x32\xC2\x0A\x3A\xEF\x3B\x95\xDC\x64\xCF\x25\x6A\xE7\xA5\x56\x8D\xB0\x4A\x8D\x70\x26\x05\x21\x92\xD2\x64\x2F\x27\x6A\x9F\x33\x57\x8B\xD0\x5E\x8B\x70\x85\x16\xE1\x6A\x2D\xC2\xEB\xD4\x56\x60\xD4\x8E\xE3\x69\x1D\xC2\xAF\x3A\xD9\xCE\xE0\xC0\x34\x84\x5D\x69\x08\x1F\xA7\x21\x1C\x4E\x43\xF8\x21\xAD\xC9\x3E\xCA\xD8\x74\x84\x09\xE9\x08\xBB\xD2\x11\x3E\x4E\x6F\xEA\xC7\xF5\x19\x08\x6F\x65\xC8\xF6\x47\xFB\x66\x22\x34\x64\x21\x7C\x97\xD5\x64\x97\x2B\x6A\x4F\x32\x3A\xBF\x8C\xDA\x33\xAA\x37\x22\xEC\x36\x36\xD9\x1F\x8B\xF6\xF3\xB1\x53\x10\x26\x4C\x41\xB8\x7E\x0A\x82\x73\x0A\x42\x95\x07\xE1\x01\x4F\x93\x5D\xBB\xA8\x3D\xBB\xA8\xBD\xB8\x82\xC3\x08\x97\x1E\x46\x38\x7D\x04\xE1\xD7\x23\xB2\xBE\xBA\xF3\x58\x93\xDD\xD4\xA9\x27\x10\xEA\x4E\x20\x3C\x7B\x02\x61\xD5\x09\x84\x35\x31\xF6\xE5\x3A\x9F\x44\xE8\x12\xB5\x33\xF7\x2D\x42\xCA\xB7\x4D\xF6\xD3\xA2\xE7\xC2\xA2\xED\x17\x7B\x6E\x27\x8A\xBF\xA2\x5E\x1F\x3B\x9E\x5A\x1A\x43\xB1\xE3\xE7\xA8\xFA\x94\x34\x76\xC4\x71\x53\xA2\xF1\xC6\x8D\x19\x6B\xE3\x98\x11\xC7\x4A\xD3\x38\xF9\xF8\x82\xE3\x44\xEC\xDF\x7F\xD4\xAF\xA3\x4F\xB4\x7F\x46\xFB\x47\xC9\x1F\xB4\xFF\x3F\xD2\xEE\x7D\x68\xBB\x7F\x15\xD7\xEE\x13\x5A\x68\xF7\xF8\x76\xBE\x50\x3B\x46\xDB\xAF\xB1\x3D\xE8\xB9\x52\x11\x37\x3E\xCD\x3E\x94\x0D\xE9\x08\xD7\xA7\x23\xDC\x48\xCF\x79\xD5\x27\x9E\x49\x3C\x9F\x18\x89\x7E\x62\x16\x44\xCC\x29\x94\xB0\xE3\x0E\x73\x2C\x56\x88\x78\x00\xD4\xBE\x57\x10\x2F\x3C\x9E\xA3\xFD\x32\xFA\xC4\x62\xA5\x88\x8B\x22\x16\x9E\x56\x7F\xA7\x16\x71\x50\xC4\x40\x9F\x66\x86\xA6\xA3\xB6\xBB\x56\xC4\x3B\x11\xE7\xF6\x67\xEF\xCF\x6E\x8E\x63\x0E\xF3\xC2\x16\x70\xEC\x42\xD8\x25\x62\x56\x6C\xFB\x45\xF1\xA9\xF2\x0F\xF0\xE9\x3F\x8D\x3F\x7F\x06\x77\x5C\xBA\x26\x7C\xC9\xA7\xB8\xF1\x35\xC5\x8D\xEB\x28\x5E\x44\xF1\xE1\x8F\x70\x20\x3A\xDE\xFF\xD1\x71\x7E\x98\x9E\xC7\x6F\x3C\x37\x46\xCF\x85\xD9\xE9\xB9\xFC\x68\x7F\x1C\x7D\x18\x61\xCC\x61\x84\xAA\x8C\xE5\x19\x63\x73\x66\x37\x7E\x36\x2B\xDE\x4E\x45\xB4\xFC\x4D\xF3\x5D\xB9\x3D\x3A\x2B\x10\xBA\x28\xE4\x73\xA0\x62\x3D\x8E\xA5\x7A\x7A\xEC\x79\x57\xA9\xBF\xD1\x73\x9C\x85\xC9\x08\xB3\x93\x63\xCE\x7F\xD2\x07\xD5\x0A\xF5\x85\xCE\xCA\xF7\x8B\xDB\xAF\x91\xCF\xCD\x17\xA9\x4B\xD4\xD3\x1B\xBF\x4C\x1F\x6F\x67\xA4\x8A\x9E\xAF\x8D\xE2\x82\x19\xF2\xA0\x1D\x74\x04\x4E\xFA\xEE\x5D\x5F\x18\x28\xD9\x8B\xBD\x51\xFA\xBE\xDD\xCD\x70\x33\xDC\x0A\xB7\xC2\x13\xF0\x04\x08\x20\x40\x1D\xD4\xC1\x7A\x78\x11\x36\x4A\x56\x64\xB7\xC2\x2C\xC5\x2C\xC9\x56\xEC\x75\x4A\x87\x72\xA1\x72\x91\x72\x8B\x72\x8B\xB2\x5E\x59\xAF\x7C\x55\xF9\xAA\x72\xBB\x72\xBB\xF2\x7B\xE5\x0F\x4A\x55\x42\x72\x42\x4A\x82\x46\xB2\x23\x4B\x12\x8C\x09\x93\x13\x2A\x12\x76\x27\xEC\x4E\x80\x44\x4D\xE2\xCF\x89\xBF\x26\xDA\x93\x06\x48\x6B\x9E\xC5\x2A\x97\xCA\xA7\xF2\xA9\xCE\xAB\xCE\xAB\xA6\x25\x4F\x4B\x3E\x9D\x7C\x3A\x79\x7A\xCA\xF4\x94\xF9\x29\xF3\x53\x7E\xD0\xFC\xA0\x19\x93\x3A\x26\xF5\xBA\xD4\xEB\x53\x59\x6D\x2B\x6D\xBA\x36\x5D\x9B\xAF\xED\xAF\x5D\xA0\x5D\xA0\xDD\xA5\xDD\xAD\xFD\x58\xBB\x5F\x3B\x52\x37\x46\x77\x56\x77\x56\x57\xD8\xBA\xAC\x35\xA4\x43\xBA\x27\xC3\x93\xF1\x68\xC6\xF2\x8C\xA8\x6D\xDA\xDD\x99\x7B\x33\x3F\xCA\xDC\x9F\x09\x59\xC9\x59\x43\xB3\x87\x65\x13\xBD\x51\x5F\xAD\xE7\xF5\x35\xFA\x67\xF5\xD7\x19\x0A\x25\x1B\xCB\x7D\x73\x06\xE6\xCC\xCA\x99\x9D\x53\x9F\xF3\x46\xCE\x87\x39\x1F\xE6\xE8\x4C\x16\x93\xDD\x34\xC4\x34\xCC\x74\x85\xE9\x2A\xD3\x44\xD3\xAD\xA6\x59\xA6\x25\xA6\xBB\x4C\x49\x79\xEA\x3C\x6B\x5E\xC7\xBC\x32\x4B\xA5\xE5\x75\xCB\x76\xCB\x51\xCB\x29\xCB\x3C\xEB\x43\xD6\x8F\xAC\x07\xAC\xE7\xAC\xE7\xAD\xBF\x72\xE7\xB8\x30\x17\xE1\x7C\x9D\x7D\x9D\xC1\x66\xB3\xD9\x6D\x63\x6C\x57\xDA\xAE\xB4\xA1\x5D\x61\x4F\xB2\xA7\xD9\xD3\xED\x46\xBB\xF8\xF4\xB7\xF7\xA7\xB6\x6A\xCF\xDA\xC3\x2B\x23\x2B\x35\xBC\x96\x0F\xF0\x01\x7E\x16\x3F\x8B\xFF\x8E\xFF\x99\x0F\xD6\x4C\xAF\x79\xAF\x66\x67\xCD\x2F\x35\x67\x6A\xCE\xD5\x9C\xAB\xB1\xD5\x76\xA9\x75\xD4\x3A\x6A\xB7\xD5\xEE\x90\xEC\xB9\xDD\x10\x72\x84\x3E\x0D\x1D\x0A\x9D\x0E\x7D\x1B\x3A\x5E\x77\x3C\xCE\x6E\xAD\x33\x5C\x14\x2E\x0A\x97\x85\xCB\xC2\x93\xC3\x53\xC2\x18\x51\x44\x94\x91\xE4\x48\x4A\x44\x1B\xD1\x47\x0C\x11\xA3\x64\xCB\xB6\x7F\x64\x60\x64\x60\xE4\xFA\xC8\xF5\x91\x1B\x23\x37\x46\x1C\x11\x57\x24\x6A\xBF\x76\x85\x42\xA6\x5D\x94\xA2\xF6\x92\xC7\x51\x5B\x78\x46\x95\x4C\xD1\x7D\xD1\xA8\x3D\xB6\xD9\x94\x4E\xD3\xF5\xEA\xA8\xDD\xEB\x87\xD5\x08\xCB\xD5\x08\xCF\xA6\xCA\x74\x8D\x16\x61\x82\x16\x61\x04\xB5\x2F\xBD\x51\x87\xB0\x49\x87\xB0\x85\xDA\xA7\x2B\x48\x43\x18\x92\x86\x10\x4C\x43\xA8\x4C\x43\x78\x3D\x0D\xE1\xCD\x34\x84\x1D\x94\x0E\x50\xFA\x24\x0D\xE1\x50\x1A\x42\xEF\x56\x08\xFD\xA9\xEE\x2B\xD2\x65\xE9\x08\x63\xD2\x11\xAE\x8B\x91\x17\x22\x6D\xA7\xB4\xAD\x0D\xC2\xBE\x36\x08\x83\x32\x10\x86\x64\x20\x44\xF5\xEF\x7E\x99\x08\x03\x32\x65\x9B\xA2\x22\x0D\xA5\xF6\x46\x8F\x50\x1A\xA8\x47\x18\xAE\x47\x78\x44\x8F\xB0\x5C\x8F\xF0\xB8\x1E\xE1\x49\x3D\xC2\x20\x6A\x47\xDB\x97\x83\x30\x35\x07\xE1\x67\xBA\xEE\x1F\x31\xCA\x14\xB5\xB3\x1A\xD5\xE7\x7F\xA4\xD4\x6B\x0A\x42\xDF\x29\x08\x43\x29\x3D\x48\xD7\xE6\x8E\x78\x65\x5A\x41\xED\x95\xDE\xBE\x1F\xE1\xCE\xFD\x2D\xD8\x83\x9F\x82\x30\x6C\xCA\x6F\xF5\xA7\xF8\x73\xB6\x85\x54\x2E\x34\x9E\x1B\xA6\x8F\x38\xCE\x07\x9A\x0A\x4C\x57\x9B\xAE\x36\x75\x0F\xF7\x08\x8F\x0A\x8F\x09\x8B\x7D\x22\xEA\xDF\x19\x3A\xC3\x22\x3A\xB6\xC5\x71\x95\x90\x95\x90\xA5\x33\xB5\x31\x65\xDA\x32\x6D\x7A\x9B\xDE\x96\x6F\xEB\x6B\x2B\xB0\x15\xD8\x72\xEC\x39\x52\x9F\x16\xFB\x73\xD7\x70\xB7\xF0\xF8\xF0\x78\xA9\xFF\x69\x22\x9A\x48\xB4\xCD\x41\xFA\x2E\xA9\x15\x6E\x84\x1B\x25\x5C\xE9\xAB\xED\xAB\xED\xAF\xED\xAF\x5D\x9E\xB1\x3C\x63\xB8\x69\xB8\x69\x96\x69\x96\xE9\x2E\xD3\x5D\x26\x55\x9E\x2A\x4F\x9D\xA7\xCE\xEB\x98\xD7\x31\xCF\x6D\x71\x5B\xA6\x58\xA6\x58\x2A\x2C\x15\x16\xAF\xC5\x6B\x99\x6A\x99\x6A\x09\x58\x02\x96\x4A\x4B\xA5\x65\xBB\x65\xBB\xE5\x4B\xCB\x97\x96\xAF\x2C\x5F\x59\x8E\x5B\x8E\x5B\x4E\x5A\x4E\x5A\x4E\x59\x4E\x59\xE6\x5B\xE7\x5B\x6F\xB3\xDE\x66\xBD\xC3\x7A\x87\x75\x91\x75\x91\x75\xB1\x75\xB1\xF5\x2E\xEB\x5D\xD6\x65\xD6\x65\xD6\x7B\xAC\xF7\x58\xEF\xB3\xDE\x67\x7D\xC0\xFA\x80\xF5\x21\xEB\x43\xD6\x8F\xAD\x1F\x5B\x0F\x58\x0F\x58\xCF\x5B\xCF\x5B\xF3\x6C\x79\xB6\x76\xB6\x76\x36\x8B\xCD\x62\xB3\xDA\xAC\x36\x95\x5D\x65\x57\xDB\xD5\x76\x8D\x5D\x63\xD7\xDA\xB5\xF6\x34\x7B\x9A\xBD\x8D\xBD\x8D\x3D\xD3\x9E\x69\xCF\xB6\x67\xDB\x0D\x76\x83\x9D\xD8\x8D\xF6\x01\xE7\x07\x9C\xCF\x0A\x67\x85\x7B\x86\x7B\x86\x7B\x87\x7B\x87\xF3\xC3\xF9\xE1\x7E\xE1\x7E\xE1\x01\xE1\x01\xE1\x82\x70\x41\x78\x70\x78\x70\x78\x68\x78\x68\xF8\x92\xF0\x25\xE1\xEB\xC2\xD7\x85\x6F\x08\xDF\x10\x9E\x18\x9E\x18\x56\x45\x54\x91\xFE\x91\xFE\xD2\xD8\x2B\x8C\x14\x46\x8A\x22\x45\x11\x11\x5B\xB3\x4D\xD9\xA6\x5C\x53\xAE\x29\xDF\x94\x6F\x52\xD8\x14\x36\xA5\x4D\x69\x4B\xB6\x25\xDB\x34\x36\x8D\x2D\xD7\x96\x6B\x33\xDB\xCC\xB6\x82\x9A\xE9\xB2\x9E\x40\xE5\x7A\x74\x3C\x8A\x18\x9D\x65\xCA\x32\x19\x4C\x39\x26\xA3\xC9\x68\xEA\x63\xEA\x63\x62\x6C\x8C\x8D\xB5\xB1\x36\x95\x4D\x65\x4B\xB1\xA5\xD8\x72\x6C\x39\x36\x93\xCD\x64\x03\xE9\x7B\xA7\x22\xFE\x5B\xA0\xA3\x24\x03\x38\xB0\xC5\xC8\x81\x1B\xE0\x86\x46\xEC\x17\x71\xFF\xB7\x98\xBF\xF0\xDF\x86\xF7\x17\xC2\x7A\x11\xE7\x7F\x8B\xF1\xCA\xAC\xC4\x7F\x00\xE7\x33\x4C\x99\x26\x93\x84\xF5\xBD\x4D\x7D\x4D\x03\x4C\x83\x4C\x43\x4C\x23\x24\xCC\xBF\xCA\x74\x8D\x69\xA2\xA9\x39\x9E\xA3\x2D\xC1\x96\x64\x53\xDB\xD4\xB6\x54\x5B\x86\x2D\xCB\x96\x6D\x33\xD8\x0C\x36\x62\x33\xDA\x3A\x4A\x58\xDF\xC7\xD6\xCF\x36\xD0\x36\xC8\x36\xC8\x36\x24\x06\xF7\x45\xAC\xFF\x9F\xC2\x79\x5D\x38\x23\x9C\x2D\xE1\xBC\x3D\x3C\x2C\x3C\x3C\x7C\x69\xF8\xF2\xF0\x58\x09\xF3\xC7\x85\x1D\x61\x67\x1C\xE6\x27\x46\x92\x25\xDC\x4F\x89\xA4\xC6\x60\x7F\xBF\x48\xBF\x88\x33\xE2\x8C\x14\x8B\x98\x1F\xFB\xFC\xC5\xF8\xFF\x37\xEE\xFF\x6B\xB8\xDF\x01\x3A\xC0\xF5\x70\xBD\xA4\x9B\xE5\x6B\xF3\xB5\xFD\xB4\xFD\xB4\x8F\x66\x3C\x9A\xA1\x37\xE9\x4D\xC4\x44\x4C\xC3\x4C\xC3\x4C\xB7\x9A\x6E\x35\x2D\x31\x2D\x31\x25\xE5\x25\xE5\x25\xE7\x25\xE7\x59\xF3\xAC\x79\x65\x96\x32\xCB\x64\xCB\x64\x4B\xB9\xA5\xDC\xE2\xB1\x78\x2C\x3E\x8B\xCF\xE2\xB7\xF8\x2D\x41\x4B\xD0\xF2\xBA\xE5\x75\xCB\x51\xCB\x51\xCB\x31\xCB\x31\xCB\xD7\x96\xAF\x2D\x27\x2C\x27\x2C\xDF\x58\xBE\xB1\xCC\xB3\xCE\xB3\x2E\xB0\x2E\xB0\xDE\x6E\xBD\xDD\xBA\xD0\xBA\xD0\x7A\xA7\xF5\x4E\xEB\x12\xEB\x12\xEB\x52\xEB\x52\xEB\xDD\xD6\xBB\xAD\xF7\x5A\xEF\xB5\xDE\x6F\xBD\xDF\xFA\xA0\xF5\x41\xEB\x47\xD6\x8F\xAC\xFB\xAD\xFB\xAD\xE7\xAC\xE7\xAC\x17\xD9\x2E\xB2\xB5\xB5\xB5\xB5\xB5\xB7\xB5\xB7\x75\xB0\x75\xB0\x0D\xB6\x0D\xB6\x25\xD9\x93\xEC\xC9\xF6\x64\x7B\x8A\x3D\xC5\x9E\x6A\x4F\xB5\xEB\xEC\x3A\x7B\xBA\x3D\xDD\x9E\x61\xCF\xB0\x67\xD9\xB3\xEC\x7A\xBB\xDE\x9E\x6B\xCF\xB5\x0F\x3C\x3F\xF0\x7C\x66\x38\x33\x7C\x71\xF8\xE2\x70\xAF\x70\xAF\x70\x9F\x70\x9F\x70\xDF\x70\xDF\x70\xFF\x70\xFF\xF0\xC0\xF0\xC0\xF0\xA0\xF0\xA0\xF0\x90\xF0\x90\xF0\x88\xF0\x88\xF0\x84\xF0\x84\xF0\xF5\xE1\xEB\xC3\x37\x86\x6F\x0C\x27\x45\x92\x22\x7D\x23\x7D\x25\x5D\xC6\x11\x71\x44\x26\x45\x26\x49\xFD\x3A\x7A\xAF\x2A\xDA\xFF\xA2\xF3\x16\x7B\xED\x55\xD2\x18\xAB\xCE\xF8\x3A\xE3\x9B\x8C\x5F\x32\xA2\xDD\xBF\x2D\xB4\x85\x4E\xD0\x09\xFA\xC1\x00\x98\x08\x13\xA1\x10\x0A\xE1\x16\xB8\x05\x66\xC1\x2C\x78\x12\x78\xA8\x81\x5A\x78\x06\x9E\x81\xE7\x60\x0D\xAC\x85\xB5\xB0\x01\x36\x34\x7E\x4F\xE1\x15\xE6\x75\xE6\x03\xE9\x9B\x0A\xC7\x99\x93\xCC\x49\xE6\x14\x13\x61\x2A\x15\x95\x8A\xF9\x8A\x05\x8A\x9F\x14\x3F\x29\xE6\xB2\x73\xD9\x25\xCA\xA5\xCA\x84\x84\xA4\x04\x75\x82\x3A\x21\x35\x41\x9B\xF0\x69\xC2\xA7\x09\x5F\x25\x7C\x95\x70\x3E\x21\x9C\xF0\x53\xE2\x4F\x89\xE1\xC4\x48\xE2\x77\xAA\xEF\x55\x67\x54\x67\x55\xDF\x26\x7F\x9B\xEC\x53\xFB\xD4\xDF\xAB\xCF\xA8\x67\xA7\xCC\x4E\x19\xA3\x19\xA3\x99\xA9\x99\xA9\x19\x98\x3A\x30\x15\xB5\x0A\x6D\x6B\x6D\x6B\x6D\x1B\x6D\xA6\x56\xAF\x35\x6A\x7B\x68\x7B\x68\x7B\x6A\x7B\x6A\x7B\x6B\x7B\x6B\xB7\x68\x5F\xD7\xBE\xA1\x7D\x57\xFB\x9E\x76\xA7\x76\x8F\xF6\x23\xED\xE3\xBA\x15\xBA\x65\xAD\x1F\x6E\x5D\x91\x51\x91\x71\x20\xF3\x40\x66\x41\x76\x41\xF6\xA7\xD9\x91\x6C\xA7\x61\xB2\x21\x68\xB8\xC5\xB0\xD9\xB8\xD9\xB8\xD5\xB8\xCD\xB8\xC3\xB8\xD3\xF8\x89\xF1\x90\xF1\xA4\xF1\x94\xF1\xBC\x31\x6C\x1C\x6A\x1A\x6A\xBA\xD2\x74\xA5\xE9\x26\xD3\x2D\xA6\x45\xA6\xC5\xF4\x5B\x0D\x68\x56\x98\x13\xCC\x49\x66\x95\x39\xDD\x9C\x6E\xCE\x30\x67\x99\x89\x39\xCF\xDC\x4E\xFA\x86\x83\xC5\x6C\x35\x5B\x45\xF1\x64\xCE\x37\xF7\x35\x17\x98\x87\x9A\x47\x4A\xDF\x73\x18\x63\x5E\x24\x7D\xCF\x21\x31\x2F\x25\xAF\x43\x5E\xA7\x3C\xF9\xBB\x0E\xCB\xDB\x7E\xD5\x16\xDA\x95\x5A\xEE\xB3\xBC\x66\x79\xC3\xF2\x85\xE5\xB4\x65\xAE\xF5\x61\xEB\x3E\xEB\x41\xEB\x59\x6B\x38\xE6\x7B\x0F\x5F\x70\x5F\x73\x63\x6D\x57\xD8\x62\x6D\x59\x9F\xEE\x72\xB6\x0B\x6B\x67\xED\xAD\xEC\xAD\xED\x26\xBB\xC9\xDE\xD3\xDE\xCB\x3E\x80\x7E\xFF\x61\x59\xD7\xBB\xBB\x3E\x40\xBF\xFD\x00\xDD\x48\xB7\x8E\xDD\x46\x74\x1B\xDB\x6D\x6C\xB7\xC2\x6E\xB3\xBB\x2D\xEA\xB6\xBA\x5B\x7D\x37\xF9\x1B\x10\xCB\x56\x6E\x5F\x09\x42\x86\x60\x12\xCC\xC2\x42\x61\x91\xD0\xA1\xA6\x53\x4D\xCF\x9A\x3E\x35\xD3\x43\x33\x43\xD1\x6F\x41\x58\xCF\x5A\xCF\x3E\x7F\x76\xDB\xD9\x73\xE7\xCF\x9F\x9F\x14\x9E\x14\x2E\x0E\x97\x84\xDD\x61\x77\x98\x8D\xB0\x11\x75\x44\x1D\xC9\x89\x90\xC8\x80\xC8\x80\x48\x41\xA4\x20\x72\x43\xE4\x86\xC8\xC4\xC8\xC4\xC6\x6F\x42\x9C\x8B\x9C\x8F\x44\xBF\xFF\x30\x13\x11\x16\x22\xC2\x62\x6A\x5F\x3C\xFA\x3D\x88\xA8\x9D\xF1\xE8\x77\x21\xA6\x27\x21\xCC\x48\x42\x78\x33\x59\xA6\x81\xF4\xFB\x3E\x7B\x8C\x08\xDF\x1A\x11\xFA\x4D\x41\x18\x48\x75\x3E\x91\x96\x53\x9B\xF6\x2D\x7D\x37\xA2\x98\x7E\x37\xA2\xFC\x77\xBE\x1B\x01\x1F\x20\x0C\xA2\x36\xCE\x87\xC5\x7C\x3F\x62\xDB\x5E\x99\x0E\x50\x3A\x4B\xA9\xCD\x3E\x99\x2E\xA6\x34\x86\x92\x97\xD2\x42\x4A\xB5\x94\x5E\xA5\x04\x1F\x21\x44\x3E\x46\xE8\xB5\x1F\x21\x7F\x3F\x42\xC5\x7E\x84\xE0\x7E\x84\x69\xFB\x11\xE6\xED\x47\x58\xB0\x1F\xE1\xB6\xFD\xF2\x9C\x5B\xA4\xC8\x11\x99\x9E\x39\x21\xD3\x6A\x4A\xB6\x93\x32\x35\x7E\x77\xE7\x1F\xFC\x4E\x84\x26\xE6\x3B\x11\xFF\x97\xBE\x0F\x01\x49\x9D\x93\x44\xDD\xAB\xAA\xE6\xF5\x9A\xF7\x6A\x3E\xAE\x89\xC5\xBB\x27\xE1\x69\x58\xAA\x5C\xAA\x14\x31\x29\x8A\x3F\x67\x29\xFE\xFC\xAA\xFE\x55\x2D\xE2\x8D\x88\x1B\xE2\xF8\xEF\x5D\xD3\xBB\xE6\xDC\xF9\x73\xE7\xA3\x63\xA0\x21\x72\x38\xF2\x45\xE4\x68\xA3\x0E\xF1\x89\x11\xA1\xC1\xD8\xD4\x7E\xD1\x76\xAB\xA2\xF2\x7A\x2C\x8B\x30\x3B\xE6\x7B\x10\x55\x31\xE7\xE4\x3E\x69\x23\xF7\x93\x3B\x5B\xDA\x9F\xA1\xCF\x1C\xCD\xBC\x16\xEF\xE0\xC7\x9F\x1F\x3D\xAA\x39\xA5\xF9\x4E\xF3\x83\x26\xBA\xDE\x14\x7D\xA2\x98\x5E\x03\x35\xB0\x0A\x56\x35\x62\xF8\xEF\x61\xF7\x85\xF0\x51\xC4\x9A\x58\xFC\x88\x62\x86\x88\x17\x51\x4C\xF8\x2C\xF2\x59\xE4\xEC\xF7\x08\x91\xEF\xE5\x3C\x44\x65\x4B\x54\xA6\x88\xB2\x64\x0D\xAC\x69\x94\x21\xA2\x9C\x10\xE5\x43\x3C\xF6\x66\x99\xB3\x24\x3C\xBD\xCC\x7C\x99\x79\x99\xF9\x3E\xF3\x23\xE6\x47\xCD\xD5\xE6\x6A\xF3\x53\xE6\xA7\xCC\xBC\x99\x37\x3F\x67\x7E\xCE\xFC\x9E\x79\xA7\xF9\x03\xF3\x07\xE6\xDD\xE6\xDD\xE6\x1F\xCC\x91\x8B\x4C\x79\xE6\xBC\x5B\xF2\x6E\xC9\x7B\x34\xEF\x85\xBC\x83\x79\x87\xF3\x5E\x6C\xF7\x62\xBB\x6D\xED\xB6\xB5\xFB\xA5\x5D\xA4\x9D\xB7\xBD\xB7\x7D\x7D\x23\xEE\x46\x2C\x60\x8D\xC5\xDE\x48\x47\x7B\xA7\xA1\x9D\x46\x74\xBA\xBC\x93\x88\x7F\x51\xDC\x8B\xE2\x5D\x14\xE7\x8E\x44\x8E\x44\xBE\x8A\x9C\x68\xA6\x3F\xFE\x4F\xE3\x46\xFC\xF7\xB5\x78\xE0\xA1\x16\x6A\xE1\x39\x78\x0E\x56\xC3\x6A\x90\x65\xEA\x5D\xCA\x96\x64\xE9\x19\xD5\x19\x95\x2C\x3F\x7F\x51\x9F\x89\x93\xA1\xFF\xAC\xFC\x8C\x97\x9D\xBF\x95\x87\x99\x17\x94\x87\x23\xCD\x23\xCD\x97\xC7\xC8\xC3\xFB\xCD\x0F\x9B\x97\x9B\x57\x98\x9F\x30\x3F\x69\x5E\x69\x7E\xDA\x2C\x98\x9F\x35\xAF\x32\xEF\x30\xEF\x32\xBF\x6F\xFE\xD0\xFC\xA1\x79\x8F\xF9\x7B\x33\xE4\x89\x32\xD3\x98\x77\x91\x24\x37\x6F\xCE\xBB\x35\xEF\x91\xBC\xB5\x79\x07\xF2\x8E\x34\x93\xA1\xEB\xDB\x6D\x68\x57\xDF\xEE\x95\x76\x3F\xB7\x83\xF6\x9E\xF6\xBE\xF6\xA2\x4C\xDD\x6A\x91\xBE\xD1\xD5\xA9\x4B\xA7\x61\x9D\x86\x77\x1A\xD5\xE9\x9F\x93\xA7\xFF\x88\x2C\x95\xE5\x67\xAF\x9A\x3E\x54\x86\x9E\x3F\x7F\xFE\xFC\xE7\x91\xCF\x23\xC7\x22\xC7\x22\x27\xFF\x02\x79\x78\xD0\x88\x70\xF8\xDF\x2C\x17\xFF\x51\x39\x16\x95\x5F\x51\xB9\xF5\xEF\x90\x57\x67\x44\xAC\xF9\xE1\x7F\x87\xDC\xB2\xD3\xF9\x62\xF4\x89\xDA\x31\x88\xEE\x5F\x8D\xA5\xF3\xB0\x2A\xC3\x56\x83\x38\x9F\x87\x1A\xAE\xD9\xFE\x09\x24\x6A\x12\xB5\x89\x23\x13\x47\x25\x8E\x49\x2C\x4C\x9A\x9C\x24\xDB\x3A\x49\x6F\xB4\x71\xE2\xCF\x0C\x64\x46\x6D\x98\x88\xC1\x4D\x8D\x67\xFB\xE5\xBD\xB7\x6A\xFD\xB3\xFA\x2A\xC9\x56\xC7\x21\x69\x1F\xBD\x8A\xDA\x05\x88\xEE\x83\xCD\xFD\x93\xF6\x69\x62\xEF\x50\xEC\x52\xEF\x92\xCE\xBF\x1E\x55\x9F\x51\xD7\xD3\xFE\x1B\x9D\x2F\x36\x50\x9B\x41\xB1\xF7\xE0\x23\x8B\x62\xEE\xC1\xFF\xC6\xC6\x4D\xF3\x7B\x18\xB2\xBD\x9B\xA2\x94\x92\x14\x6F\xCA\xF4\x94\xD9\x29\x55\x89\x6B\x12\x31\x35\x3F\xB5\x20\xF5\x7A\xF9\x50\x02\xB5\x39\x33\x95\x9E\x07\xDB\x22\xE1\x9C\xBC\xAE\x22\xDB\xAE\xF7\x75\x9E\xDD\x79\x76\xE7\x46\xF9\x4C\xED\x26\x44\xF7\x01\xA3\x76\x3B\xA2\xEB\x03\xF0\x74\x27\x3E\xB6\xBA\xEB\xE9\x3E\x5E\x54\xFE\x57\xD3\xFD\x9D\x08\xA5\xA8\x3D\x8B\x3A\xBA\x26\xD8\x68\xD7\xA2\x7A\xF1\x4A\x11\x43\x4C\x60\x82\x6A\xA8\x86\xAA\x8C\xAA\x0C\x30\x25\x9B\x3A\x98\xAC\xA6\x2E\xA6\x2E\x26\x87\xC9\x61\x02\x3B\xD8\x81\x9E\x6B\xE9\x20\xF9\x58\x4D\xF1\xE1\x9B\x85\x05\x80\x49\x49\x08\xB1\x16\x52\xAB\x46\x22\x28\x00\x60\x64\x1C\xBF\x9A\xF2\xC7\x33\xCD\xF9\x1B\x46\xCA\xEE\x05\xA9\xF2\xDF\xFA\x91\x28\x7D\x7B\xDD\x74\x19\x4A\xDF\x3C\xEF\x92\x28\xF3\x0B\x2E\x47\x60\x63\xDE\xFB\x38\xA1\x79\x3C\x63\x2F\x47\x60\x00\xA4\xB9\xA7\xC8\x2F\xA4\xEE\xE8\x93\x4F\xF3\x53\x76\xB9\xFC\xB7\x9C\xFE\xBD\x29\x39\x2E\xFF\xF4\x3D\x93\x4A\xE6\x2F\x8B\x8B\x67\x03\xE5\x57\x5F\x8E\xB1\x5F\x15\x94\xF6\xFB\x62\xE3\x69\xA0\xF9\xFD\x49\x49\xF9\xA3\xE4\xF2\x47\x9F\x35\x71\xE1\x75\xD4\xBF\x53\x8A\xCC\xEF\x30\x2A\xAE\x9E\x68\xB9\xB8\x51\xCD\xF3\xA3\xD7\xC8\xFC\x1E\x71\xE1\x17\xA5\xD1\x7A\x1B\x85\xD2\xF7\x85\xAB\x46\xA1\xF4\x4D\xE3\xD3\xB4\x1E\xDE\x8C\x8B\xA7\x43\x62\xF3\xF7\x77\xD1\xF8\x7A\xC7\xF1\x1B\xE8\x7B\x9B\x69\x7B\x9D\x8E\x8B\x67\x34\xCD\x27\x8C\x46\xD0\x88\xE9\x8F\x46\xE9\xDB\xCB\xF9\xDA\xB8\x7A\x1E\x2D\xBF\xB7\x90\xE6\x67\xD9\xE8\xE6\xED\xBB\x81\xD6\x43\xF5\xE8\xE6\xF5\x76\x59\x5C\x7E\x9E\x19\x2D\xBB\x73\x68\x7F\x58\x4B\xDD\xF5\x34\xFE\x02\xCA\xDF\x35\x1A\xE1\x7A\xD1\x3D\x16\xE1\x4A\x00\xE8\x48\xF3\xBF\xEB\xCA\xE6\xF1\x2F\xA6\xFD\xF3\xE0\x95\xCD\xD3\x29\x88\xEB\xCF\x0D\x57\xCA\xF9\xDD\x86\xCD\xF9\x70\x15\x42\x02\x00\x38\x68\xFC\xF6\xAB\x10\x62\x2F\x1C\xE8\xE2\xFA\xFF\xD8\xAB\x64\xF7\xDD\xB4\x3F\x14\x5E\xD5\xBC\x1E\xEE\x83\xE6\xE1\x7D\x34\x3E\x43\x5C\x7D\xAE\xA6\xF1\xEC\x8A\x1B\x17\xF5\x57\x21\x28\x01\x60\x48\x5C\xBA\xA7\xAF\x92\xEB\xE7\xFB\xB8\xF0\x70\xB5\xEC\x5E\x9C\xDA\x9C\x9F\x44\xF9\xC3\x28\x5F\x77\xB5\xFC\xED\xDD\xE8\x33\x9D\xB6\xCB\xB2\xAB\x11\x92\x63\xF8\xF9\x34\xFC\x86\xAB\xE5\xF4\xEA\xAF\x96\xF3\x33\x2B\x2E\xFE\x63\x34\xFE\x4B\xE3\xCA\x75\xFA\x6A\x94\xBE\xB7\xAF\xA7\xF5\xE3\xBB\x06\xC1\x2C\x7A\x5C\x2B\xB7\x9B\x9B\xF2\x4D\xD7\x36\x1F\x8F\x49\x71\xE3\xBA\xF0\x5A\xB9\x5D\x4E\x44\xC7\xF5\xB5\xF2\xB8\x88\x3E\x47\x69\xFE\x77\x5D\x8B\x20\x6D\xEE\x8E\x93\xEB\xF9\x79\xDA\xEE\x05\xE3\x9A\xC7\xFF\x75\x5C\xFE\xAB\xC6\xC9\xE5\xFB\x8C\x8E\xF7\x65\xE3\x9A\xFB\x9F\x8D\xAB\xE7\xEA\x71\x72\xFE\x17\xD0\xF6\xAD\x1F\x27\x8F\xCF\xE8\xD3\x91\xD6\xC3\xE7\x34\xDE\xD3\x34\xFC\x7D\x71\xE9\xEA\xC6\xCB\xFE\x65\x14\x07\x4C\xE3\x11\xD2\xC5\x7A\x1A\x2F\xE7\xBF\x4F\x1C\xCE\xAC\xA6\xE1\x0B\xE3\xC6\x51\x3D\xE5\x6F\xA3\xE3\x77\xD7\xF8\xE6\xE3\x62\x7F\x3C\xBE\x45\xE3\xA1\xF9\x3C\x3D\x5E\x1E\xEF\xD1\xC7\x4C\xF9\xF6\x09\xF2\xF8\x1F\x3B\x41\x0E\xDF\x93\xB6\x4B\xE1\x84\xE6\xB8\x71\x38\x2E\x7E\xDF\x04\x39\xFF\xCB\xE3\xC6\xD7\x6A\xFA\xDE\x0E\x96\xD6\xDB\x84\xE6\xE3\xE5\xED\xB8\x78\x76\xD1\xF0\xB9\x71\xFD\xA1\x81\xF2\x17\x50\x9C\x3C\x3D\x01\xE1\x16\x04\x68\x70\x20\x7C\x07\x00\x29\xB4\xBD\x7C\xA5\xB4\x3F\xD0\xE7\x21\xCA\xAF\x2E\x6D\xDE\xFF\xAF\xA0\xE5\x35\x95\x35\xAF\xB7\x67\x69\xFB\xDA\xCB\xE4\x7E\x1F\x7D\xEE\x4F\x8B\xC3\x01\xFA\xDE\xE2\x38\xBE\xAF\x4C\xAE\xBF\xA5\x34\x9E\x9A\x32\xF9\xEF\x33\xF4\x6F\x32\x2D\x6F\x7D\x19\x42\x11\x03\x30\xD6\x2F\x7F\xFB\xFA\x6E\xDA\x6F\x0B\x03\x08\xBD\xC4\xFE\x1B\x94\xFB\x7F\x76\xB4\x5C\x41\x84\x8B\xC4\x1F\x33\x11\xDE\x10\xF1\x9F\xE2\x43\xC1\xEC\xE6\xE3\x77\x2B\x4D\xD7\x37\x1B\x61\x02\x03\xD0\x70\x07\x42\x3B\x00\xB8\x37\xAE\xFF\x24\x2D\x92\xDD\x55\x71\x7C\xDD\x22\xB9\x9E\x6D\xB4\x5C\xA6\x45\x08\x1B\x45\xFE\x12\x84\x52\xB1\x3C\x71\xED\x75\xC9\x32\xD9\xBD\x5A\x19\x57\x3F\x94\x5F\x16\x2D\xD7\x32\xB9\x5E\xA2\xCF\xD3\x74\x5C\xDC\xB2\xAC\x79\xBF\x3A\x10\x97\x9F\x65\xD4\xFF\xCD\xF8\xF1\xB8\x4C\xAE\xFF\x33\x71\xF9\xA9\xA7\xE9\xBC\x1C\x1D\x77\x77\x23\x54\x88\xFC\x7B\x10\x06\x03\xC0\x0F\x54\x4E\xED\xBA\xAF\x79\xFB\x3E\x1E\x17\xFF\xE7\xF7\xC9\x6E\x0D\x8D\xFF\xC4\x7D\xCD\xFB\xC9\x98\xB8\x74\xE1\x7E\x39\x9F\x6E\x75\x5C\x7D\xDE\x2F\xBF\xE7\x89\xC3\xF3\x0E\xF7\xCB\xEE\xC4\xB8\x74\xED\x34\xFC\xD7\xB4\x1E\xC6\xDE\x8F\xF0\x81\x58\xDE\x87\x10\x2A\x01\xE0\xC7\xB8\x74\xC7\x2E\x97\xD3\xBD\x96\x86\x2F\x5C\x2E\xF7\x9B\x19\xCB\x65\xF7\xA1\xB8\x76\xA9\x5A\x2E\xC7\x5F\x1D\x17\x4F\xF5\x72\xB9\x3E\x36\xD0\xF0\x6F\x2E\x6F\x5E\x5E\x8E\xF6\x87\x86\xE5\x08\x43\xC4\xF1\xF7\x98\x5C\x9F\xEF\xC7\xB5\x17\x54\xD3\x71\x1A\xAF\x2F\x55\xCB\xF1\x7D\x43\x71\xCF\x5E\x2D\xF7\xCB\x5D\xD5\x08\x59\x00\xB0\x84\xD6\xDB\xB2\x27\x9A\xE3\x83\x22\x4E\xBE\x54\x3F\x21\xC7\x7F\x9C\xE6\x73\xF5\x13\xCD\xFB\x4F\x62\x74\x7C\x3D\x21\xF7\x83\x86\x27\xE4\x72\x65\xD0\x7A\xD6\x3D\x89\x90\x2B\xA6\xF3\x24\x42\x47\x31\xBE\x38\x79\xDD\xF0\x94\x1C\xDF\xF3\x71\xF8\x73\xFA\x29\x39\x5F\x7E\xDA\x9F\x75\x2B\x9B\xFB\xBF\x4F\xD3\x4D\x8F\xE3\xEB\xE3\xEA\xC7\xB4\x52\xAE\x87\xB2\x38\xB9\x50\xB0\x52\x4E\xF7\xC5\xB8\xF0\x63\x69\x7C\x57\xC7\x85\x1F\x4F\xF9\xA9\x71\xE1\x0B\x69\x3C\xB7\xC7\xF1\x7D\x34\xFC\xD3\x71\xFC\x19\x94\x1F\x88\x97\x8F\x94\xFF\x39\xE5\x2F\x5B\xD9\x5C\x9E\x3E\x9F\x12\x87\xDB\x34\xDD\x83\x71\xF1\x34\x50\xFE\xF7\x3A\x8A\xDB\x2B\x11\xE6\x8B\x1E\x02\xC2\xC5\x62\x7E\xE2\xEA\x79\x75\x8D\xDC\x5E\xE6\xB8\xF8\x3F\xAA\x91\xDD\x2F\xC4\xB7\x17\xE5\xBF\x18\xD5\x6F\x6B\xE4\xFE\x1F\x7D\x8E\xD3\xFE\xA6\xAB\x45\x48\x13\xEB\xB1\x56\x76\x7F\x4B\xE3\x2F\xAC\x45\xB8\x24\x26\xFC\x65\xB4\x1F\x56\x85\x9A\xE3\x6A\x19\xED\x87\x35\xA1\xE6\xE9\x3F\x12\x57\x9F\xAB\x43\x72\x79\xBB\xD3\x72\xD5\x87\x9A\xCB\xA3\xD6\xB4\x1F\x9E\x0E\x21\xCC\x11\x19\xCF\x22\xF4\x10\xF5\x6A\x8A\x0F\xD5\xCF\x21\x5C\x16\x13\xDE\x1A\x2F\x1F\x57\xC9\xFD\x47\x49\xF3\x79\x6C\x55\x9C\x3E\x1F\xAD\x87\x38\xFE\xB7\xB4\xFE\x61\x35\xC2\x7D\x62\xBF\x7A\x1E\xE1\x0A\x51\x8E\xD3\xFC\x7C\xB4\xB6\x79\x78\x1D\x2D\x6F\x43\x1C\x7F\x6D\x34\xFE\xB5\x08\xAD\xC4\x7E\xBB\x8E\xCE\x13\xE3\xC6\xE9\xF8\x75\xB2\x3B\x21\xBE\x7F\x52\xFE\x8C\x38\xFD\xDC\xB7\x4E\x4E\xE7\xAD\xF8\xF9\x1D\x8D\x7F\x10\x8D\xBF\x7A\x1D\x42\x8A\x58\x0F\x34\x9E\xF6\xB4\x7E\x1A\xD6\xC9\xE3\xFA\xEA\xF5\xB2\xFB\xAE\xA8\xDC\x59\x2F\xF7\xB3\x82\x17\xE5\x76\xD0\x45\xF5\x81\x17\x11\x7C\xA2\xFF\x4B\x08\xC3\x01\xE0\xCE\x78\xB9\xB3\x51\x76\xFF\x12\x97\x9F\xEA\x8D\x72\x3E\x23\x71\xB8\xBA\x9A\x86\x7F\x9B\xB6\xCB\x86\x8D\xCD\xEB\x6D\x46\x14\x97\x36\x36\x7F\xEF\xD9\xE8\xBC\x66\x63\x73\xDC\x7B\x8C\xA6\xDB\x10\x17\xCF\x19\x1A\xFE\xD8\xC6\xE6\xF8\xFC\x33\xED\xCF\x3F\xC4\x85\xAF\x8A\xD6\xF3\xA6\xE6\xF2\x2E\x8B\xF6\xB7\xF4\x4D\xCD\xC3\xFB\xE3\xFA\x9B\x89\xFA\x7F\x42\xE3\xB1\x6F\x92\xF5\x70\x1F\xE5\x8F\x8F\xD3\xF7\xAA\x36\xC9\xE5\xE0\xE9\x38\xAD\xDE\x24\xB7\x57\xF4\xE9\x4E\xDB\x71\xD7\xA6\xE6\xEF\x59\x68\x79\x77\xC7\xE5\xE7\x38\x6D\xAF\x86\x4D\xCD\xF5\x87\x81\x34\xFC\xB9\x4D\xCD\xEB\x21\x29\xAE\x1F\xEA\x36\xCB\xF1\x75\x8D\xC3\x25\x13\xE5\x73\x34\x1E\xFB\xE6\xE6\xFE\xA7\xE2\xDA\xB7\x80\x86\xBF\x31\x5E\xFE\x52\x7E\x46\xFC\xFC\x65\xB3\x9C\x2F\x3B\xE5\xCF\xD8\xDC\xBC\x5C\xEF\xD1\xF1\x58\xB5\xB9\xF9\x7C\x73\x77\x9C\xFE\x50\x4D\xF3\xC5\xD2\x74\x6B\xE2\xF2\x79\x90\xE6\xF3\x99\x38\xFE\x6D\xB4\x7D\x57\x6F\x6E\x5E\x6F\xF5\xE2\x7F\x83\x1A\x56\x24\xAA\xEB\x15\x52\xBC\x70\xE1\x47\xF4\x4F\xFA\x03\x7F\xDD\x1F\xF8\x17\xFC\x8E\xBF\xF8\x54\xFD\x8E\x5F\xBE\x1A\xA1\xF7\x07\x52\x39\x66\x21\xC0\x0C\x04\x28\x17\xFB\xB0\x51\xD4\xCB\x00\xE0\x64\x24\x32\xBD\xA7\x9C\xCE\x4F\x00\xBA\xBE\x28\x59\x6B\x94\xCB\x94\x44\xF3\x66\xA2\x79\xA8\x8A\x66\xE8\x2F\x0C\x20\x7A\x8A\x32\xA3\x58\x1C\x9F\x0A\x00\x9D\x05\xA0\xFA\x36\x84\x82\xC1\x5A\xA8\x7A\xE0\x21\x80\xC3\x3F\x2A\xE1\xED\x47\xFA\x8A\x11\x68\xE9\xCB\xA2\x4C\x6A\x45\x23\x46\xA9\x5C\x72\xBB\x2D\x7E\x9F\xCA\xDD\xF7\xA3\xED\x1A\x91\xBE\xD5\x18\x89\xE8\x40\x2C\xB3\x38\x47\x80\x07\xDF\x63\x58\x8F\xB3\xC2\x85\x5F\xEE\x60\xB6\x28\x21\xC5\xEB\x73\x3A\x26\x55\xBA\xCB\x83\x6E\x8F\x1D\x63\x5D\x5D\x99\x58\x57\x37\x45\xAC\xAB\x3B\x1B\xEB\xEA\xA1\x4C\x16\x5D\xCE\x49\x5E\x7F\xD0\x91\xA0\x91\x7E\x97\x96\x3A\x8A\xBC\x95\x9E\x60\xA2\x3A\xEA\x0C\x54\x56\x24\x69\xA3\x0E\x9F\xDF\x5B\x5C\x59\x14\x54\x35\xFA\x56\x38\x67\x24\x37\x39\xDC\x1E\x75\x4A\xE3\x7B\x5E\x7F\x30\xA5\xD1\xCB\x59\x5E\xAE\x69\x72\x78\x66\xA6\xA6\xD2\x4C\x38\x2A\x5C\x15\x93\x5C\x7E\xAD\xB6\xB9\xBB\xBB\x4E\xCE\x8F\xDF\x1D\x2C\x73\x38\x27\x05\xD2\xB4\x4D\x4E\xBF\xB7\xD2\x53\xDC\x2A\xB5\x89\x51\xE4\x72\x97\xB7\x8E\x09\x50\x52\xEE\xF5\xFA\xD3\x63\x02\xF8\xCA\x2B\x03\x6D\x62\x02\x54\xB8\x3D\x95\x81\x8C\x56\x31\x8C\xCA\xF2\xA0\xDB\x57\x3E\x33\x53\xD7\xC4\x2B\x76\x4F\x73\x17\xBB\xB2\x62\xF2\xE1\x77\x55\x64\xD3\x00\x7E\xE7\x4C\x47\x91\xD7\x53\xE4\x0C\xEA\xB5\x4D\x9C\x40\xB9\xBB\xC8\x65\x48\x6B\x62\xF8\x5D\xD3\x5C\xFE\x80\x2B\x47\x2A\xFA\x24\x77\x30\xE0\xF0\xFA\x73\x53\x1A\x1D\x4E\x4F\x31\xD1\x36\xBA\x3C\xAE\x52\x67\xD0\x65\x6C\xF2\x9E\xE1\xF5\x9B\x5A\x35\xBA\x02\x65\xEE\x92\x60\xB9\xAB\x24\x68\x6E\xDD\x9C\xE7\x77\x97\x96\x05\x2F\x92\x52\x2D\x72\x17\xFB\xC5\x7C\x05\x9D\x6E\x4F\x20\x2F\xD9\xE7\xF4\x07\x5C\x12\xB3\x6D\x92\xFC\xDB\xED\x6B\xD7\xBA\x31\xA0\xDB\x13\x14\x73\x57\x14\x0C\xB4\x97\x7A\x42\x51\x85\xCF\xE1\x9A\x6A\x51\x47\x7F\x7B\x5C\x53\x3B\x34\x7A\x94\x06\xAD\xEA\xA6\xDF\xAE\x8E\x8D\x1E\xE5\xC1\x4E\xEA\xA6\xDF\x2E\x4E\xAA\x21\xD7\x34\x67\xB9\xA3\x28\x38\xC3\xE1\x71\x4D\xEF\x9C\xD1\x8C\x13\x70\x05\x1D\x6E\x8F\xAF\x32\x68\x4B\x8A\xF2\xBB\xB0\xE2\xFF\xF6\x36\xBF\x09\x57\xEC\x0C\x3A\xBB\x1A\x7E\xC3\x76\x79\x82\xFE\x99\x3E\xAF\xDB\x13\xEC\x96\xD9\xCC\xB3\xD4\x15\x74\xF8\x5D\x81\xCA\xF2\x60\x77\x83\xC3\x51\xE2\xF5\x17\xB9\x1C\xEE\x0A\x9F\xD8\xC1\x63\x3A\x7E\xA0\x87\xD4\xA8\x41\xAF\xC3\x53\x29\x76\xB6\x8B\xE5\x0A\x75\x06\x5C\x3D\x7B\x38\xDC\x01\xC7\x34\x67\xB9\xBB\xB8\x67\x1A\x65\x94\xBA\x3C\x8E\x62\x57\x91\xB7\xD8\xD5\x2B\x2D\x26\x9C\xCC\xEA\x1D\xCB\x72\x79\x44\x56\x9F\xD8\x17\x65\x56\x7E\x9B\x98\x50\x95\xFE\x72\xFA\x72\xDF\x78\xB6\x1C\xBA\x9F\xD4\xE6\x93\x03\x5E\x8F\xA3\xD2\x53\xE1\xF4\x07\xCA\x9C\xE5\xFD\x75\x8D\x3C\xCA\x19\x20\xA5\xEC\xAF\xF4\x04\xDD\x15\x2E\x87\xCB\xEF\xF7\xFA\x07\x66\x46\x07\x51\xA9\xDF\xE9\x13\xFB\xAB\xB3\xA8\xCC\x39\xA9\xDC\x55\x90\xD5\xF8\x76\xB9\x6B\x86\xC8\x2F\xA6\x45\x1F\xD4\x82\x4F\x20\xE8\x77\x7B\x4A\x07\xA7\xFD\xC6\x67\x48\x9B\x46\x96\xDC\x9B\xE4\x90\x43\xD3\xE3\xD8\x41\xEF\x14\x97\x67\x58\xAB\xF8\xC0\xAE\xE0\xF0\xF8\x08\xBC\x93\x26\xBB\x8A\x82\x23\x52\x9B\xB3\x2F\x91\xC6\xC4\x34\x67\x79\xA5\x4B\x66\x5C\x9A\xD3\x18\x60\xBA\xDF\x1D\x74\xF9\x1D\xAE\x0A\x77\xD0\x31\xC9\xEB\x2D\x77\x39\x3D\x23\x5B\xF6\x15\x7B\x77\xA9\xCB\x7F\x99\xA1\x45\x5F\xB9\xF8\x97\xB7\xEC\x29\x97\x6B\xD4\x45\x2D\x7A\xCA\x23\xDB\x55\xEE\xAA\x70\x79\x82\xA3\xF5\x2D\x86\x91\xF2\x3E\xC6\xD8\xA2\x5F\x91\xB7\xBC\xDC\x55\x14\x74\x7B\x3D\x63\x4D\x2D\xA7\x2E\xF6\x70\x39\xFA\x2B\x2E\x1C\xA2\x5C\x74\x3A\xCB\xAF\xCC\x6B\x31\x84\x5C\xB1\xD1\x68\xAE\x6A\x13\x1F\x48\xFA\x73\xB5\xA6\x91\x5D\x5C\x59\xE1\xBB\x26\xB5\xA9\xDA\x45\xF7\xB5\x52\x33\x54\x38\xCB\xCB\xBD\x45\x0E\xB7\xC7\x1D\x1C\x27\x75\xC2\x32\x97\xD3\xE7\xF0\x05\xFD\xE2\x60\x1B\xDF\x9C\x13\x70\x05\x27\x24\x37\xBD\x74\x9D\x34\xC0\x4B\xFC\x2E\xD7\xF5\x12\x42\xF8\x5D\x12\xFB\x86\x36\x31\x63\xD1\x51\xE4\x2C\x2A\x73\x89\x71\xDD\xD8\x02\x3B\xE0\x0A\x4E\x94\x92\xA8\x70\x55\x78\xDD\x37\xB9\xA4\x6C\x38\x9A\x71\x7C\x95\x81\xB2\x42\x6D\x33\x8E\xD7\xE7\x6C\xD5\xFC\xA5\x80\xCB\x1F\x9C\xD4\x2C\x50\xA9\x2B\x58\x24\x21\x6D\x85\xAF\x58\x8A\xB5\x58\x2B\xFE\x2A\x76\x95\x38\x2B\xCB\x83\x22\x98\xB8\xD4\x22\xA3\xC2\x39\x43\x74\x94\xA8\xA3\x61\x8B\x5D\xE5\xA5\x52\x06\xE4\x1E\x24\xA2\xC8\xA4\x92\x32\x89\x33\xA9\xA4\x09\x53\xDC\x59\xF1\x1C\x87\xC7\x2B\xD5\xC5\xE4\xB4\x26\x9F\x49\x25\x12\x8E\x4F\x89\xB2\x24\x50\x97\xF8\xE5\xEC\x54\xE7\xA4\x40\x45\xD2\x54\x67\x71\xB1\xC3\xEB\x71\x79\x58\xF1\x97\x37\x69\x6A\xA0\x72\x92\xE8\xF6\xB1\x53\x2B\x2A\xCB\xA7\xB2\x53\x9D\x9E\x62\x7F\x92\xF8\xBF\xC3\xE3\x0D\x06\x14\x53\xBD\xFE\x20\x3B\x75\x86\xD7\x5F\xC9\x4E\xF5\xB8\x4A\xA7\xA5\x35\x65\x35\xE0\xF0\x3B\x3D\xA5\xAE\xE9\x69\x0E\x87\xCF\x19\x2C\x93\x12\x92\xFB\xC9\x8C\x14\x07\x1D\x90\xA2\xC7\x4C\x9D\x43\x02\x52\x3A\x20\x83\x65\x81\x9B\x34\x0E\xD9\xE5\x77\x55\x78\xA7\xB9\x6E\x8E\x3A\x4B\xDC\xE5\x41\x97\xFF\x96\x36\xD1\x46\xA3\xBD\x4E\x66\xDF\xDA\x2A\x8E\x5D\xEA\x0A\xCE\x8A\x0F\x2A\x47\x38\x3B\x3D\x8E\x5D\xE9\x71\x7B\x3D\x55\x98\xE8\x70\x54\xB8\xFC\xA5\xAE\x39\xD8\x3A\x1A\x20\x26\x1B\x73\xE3\xB8\x72\xB2\xF3\x64\x6D\x48\x92\x1E\xEE\x92\x92\xF9\x98\x1E\x75\x36\x8A\x3B\xB7\xD7\xB3\x00\xDB\x50\x76\xA0\x19\xFF\x36\xD4\x44\x83\x4B\x79\xB8\x1D\x53\x1B\xC3\x49\x8C\x3B\x50\xEA\xE0\x81\xA0\xBF\xDC\xE5\x59\x88\x6A\xEA\xF0\x14\x55\xF8\x16\xC9\x2E\x77\xA0\xD8\x5D\xEA\x0E\xDE\x19\x75\x05\x7C\xCE\x22\xD7\x62\x54\xC9\xAE\x32\xD7\x8C\x25\x28\x8D\x0B\x77\xD0\xEB\xBC\x4B\x8E\xCD\x19\xF4\xBA\x7B\xF6\x58\xDA\xE8\x28\xE9\xD9\x63\x19\xB6\xA2\x51\xBB\x3D\xA5\x01\xAA\x6B\xDC\x4D\x0B\xD3\xC4\x94\x04\xFD\x3D\xCD\xD9\x2E\x4F\x71\x60\xBA\x3B\x58\x76\x2F\x66\xC4\xB2\x4B\xBC\xFE\x0A\xA7\x54\x0B\xF7\x61\xEB\x58\x0F\xB7\xA7\xD8\x35\xC3\x5B\x72\x7F\x73\xAE\xDF\xE5\x2B\x77\x16\xB9\x1E\xA0\x15\xD5\x9C\xEB\xF0\x3C\x18\x1F\x5A\x52\x74\x1E\xC2\xB4\x58\x6E\xC0\x57\xEE\x0E\x3E\xDC\x3C\x1B\x81\xA0\xD3\x1F\x94\xF2\xF7\x48\xF3\xA8\x03\x95\x93\xE4\x9F\x8F\xA2\x2E\x96\x1F\xF4\xBB\x2B\x96\x37\x0F\x2A\xB2\x1C\xA2\x2A\xF4\x58\xF3\xB8\x25\xBE\xA4\x0E\x3D\x8E\x99\xBF\xF1\xF0\xF9\x5D\x25\xEE\x19\x2B\x5A\xF0\x09\x54\x96\x94\xB8\x67\x54\xB7\x10\x99\xD4\x78\x4F\x60\x72\x93\xE3\xC9\xE6\x45\x2C\xF7\x4E\x77\xF9\x9F\x6A\xCE\xAB\xF4\xF9\x5C\xFE\x95\x72\x05\x05\x67\xFA\x5C\x01\x51\xA9\x90\xC7\xE0\xD3\x71\x5C\xF9\x15\x5E\x6E\xC0\x46\x2E\x15\x6C\x82\xDC\x07\x1A\xD9\x92\xE4\xA9\x91\x6B\xA7\x29\x06\x57\xB0\x36\x2E\x52\x79\x10\x85\xE4\x4C\xC5\x64\xA0\xBC\xBC\x4E\xEE\xCE\x32\x4F\x9C\x4E\x3C\x23\x57\x6C\xA5\xC7\x2D\x6A\x1E\x8E\x40\xA5\xDF\xEF\x15\xB5\xD0\x67\xE5\x5A\x8A\xF2\x65\x7D\x45\x74\x06\x9F\xC3\x9C\x16\x7C\x1A\x5F\x5C\xD5\xF2\x8B\xC1\x92\xDE\xAB\x9B\xFB\xC8\xBA\x8E\xE4\xB3\xA6\x79\x26\xCA\x9D\x81\xA0\xC4\x7F\x5E\xAE\x96\x28\x5F\x2C\xAC\xD8\x02\x2F\x34\x67\x07\xBD\x72\x23\xAC\xFD\x0D\x5B\x6A\x87\x75\x18\x23\xD6\xC4\x82\xAF\x8F\x65\x94\x39\x03\x65\x2F\xCA\x15\x25\x33\x8A\xBC\x15\x3E\xA7\xDF\xB5\x41\x0E\xD4\x84\x5C\x2F\xD1\xA1\xEE\x92\x5C\x2F\xCB\x2D\x13\x15\x01\xFE\x99\xE2\xB0\xDA\x28\x63\x87\x1C\x4F\xA9\x2B\xB8\x09\xB3\x7E\x13\x2F\x7D\x65\x73\x4B\x5E\x72\x5F\xD8\x22\x57\x53\x73\x2F\xA9\xE5\xB7\xB6\xF4\x92\x9C\xC3\x7A\xB9\x06\xE3\xE2\x73\x05\xB7\xA1\x36\x26\x9F\x62\x61\x5F\x89\x2D\xBD\xA8\x0A\xBC\x2A\xF7\x27\x99\x51\xEE\xF2\x94\x06\xCB\x5E\x93\xDF\xA2\x85\x9F\xE2\x9A\x19\x78\x3D\xF6\x2D\x51\x88\x6D\xC7\x18\x1D\x4D\xC2\xE8\x37\xE4\x5A\xA4\x2F\xC9\x22\xF7\x4D\x4C\x73\x38\x62\xEB\xD1\xEF\x9D\xFE\x96\x5C\x91\xB4\x8B\xBF\x2D\x47\x4C\x33\xE8\x77\x95\xBC\xD3\x8C\xE1\xF6\x04\xDF\x6D\x04\x5C\xB7\xA7\x74\x07\xE6\x36\x25\x2B\x6A\xC1\xE5\xDE\xE9\x8E\x22\xAF\x6F\x26\x4D\xE1\x3D\xD4\x5F\xC0\x3F\xE0\x0A\xEE\x6C\x6A\x42\x67\x71\xF1\x2E\x4C\x95\xF3\x26\x35\xA9\xDF\x3B\xFD\x7D\x79\xF8\xFF\xF6\xDD\x0F\xE4\x1E\x4F\x7B\x90\xDF\xE9\x09\xB8\x83\xEE\x69\x2E\x47\x51\xB9\x37\x50\xE9\x77\x7D\x88\x44\x8E\xE8\x42\x01\x76\x63\xCC\xA4\xD1\xE9\xF3\xB9\x3C\xC5\x7B\x64\x11\x20\x0E\xC9\xBD\xCD\x9A\x28\xE0\xBE\xC9\xB5\x4F\x6E\xE7\xA6\x3A\x71\x48\x6A\x93\x33\xE8\x2A\xFE\x48\x1E\xE9\x8D\xB5\x23\xAB\xB9\x1F\xC7\xA2\xA3\x23\xE8\xF2\x57\xB8\x3D\x62\xE8\xFD\xB1\x72\xA1\x29\x92\x03\xB2\x2C\x92\xF2\x73\x10\x5B\x35\xE5\x6D\xBA\x34\x99\x76\xFA\x3E\x91\xAB\x5C\xAE\xD2\x4F\x31\x91\xD6\xD2\x21\x8A\xCB\xAE\x60\x63\xC8\x86\xD8\x2E\x19\x2D\x82\x2C\x6A\x0F\x63\x6A\x53\xCC\x01\xAF\x3F\x78\x44\x4E\x4A\x0E\x2B\x6A\x35\xA2\x6A\xF1\x59\x6C\xC7\x95\xC5\xBA\xC4\xFF\x5C\xCE\x42\xB9\xD7\x3B\xA5\xD2\xF7\x85\x9C\x70\x85\xD3\xE7\x13\x0B\x22\x16\xFB\xA8\x1C\x59\x94\x25\x87\xFB\x12\x4B\x3D\xE2\x6C\x2B\xDF\xEF\xEA\xD6\x21\x10\x2C\xCE\xCF\x77\x38\xBA\xE6\xE7\x4F\x72\x06\xDC\x45\xB4\x16\xFA\x15\x95\x39\xFD\x9C\xA9\xC9\x53\x74\x8B\x6D\xE6\x0E\x06\x24\xBF\x01\xB1\x9E\xB4\xCA\xBC\x7E\xD9\xCB\x34\xC0\x54\xE4\xF5\x04\x82\xED\xAD\xC7\xB0\xD8\x15\x28\x72\xFA\x5C\x7F\x6D\x32\x5F\x61\x03\x53\x5A\xEE\x9D\x24\x77\xAA\x72\x67\xD0\xD5\x41\x0C\x21\x7B\x77\xE4\x4C\x95\x9E\x80\xBB\xD4\xE3\x2A\x36\x95\x7B\x3D\xA5\xB1\x31\x4E\x73\x15\x89\xD1\xFD\x25\x99\x6B\xD9\xF3\xAF\xA9\x87\xA6\xBA\x88\x0D\xF7\x3F\x98\x42\x47\xEB\xD7\xD8\xA1\xDC\x35\xC3\xE5\xCF\xCF\x97\xFE\xFC\x4E\x05\x5B\x8F\xA3\x96\x06\x9D\x25\x87\xB5\x9E\xC0\xD6\x94\xE3\x71\xCD\x08\x76\x90\x26\xBE\x1D\xAD\x27\x1B\xB9\x25\xAE\x60\x51\x99\x08\xB3\x15\x1D\xAC\xDF\x60\x7A\x33\xAE\xA4\x8C\x77\xB0\x9E\xC2\xDC\x66\xEC\xA0\x18\x91\xDB\x13\xA4\x79\xB0\x9E\x46\x8D\xDC\xA9\x67\x89\x7F\x3A\x58\xBF\xC5\xD6\xB2\x5B\x86\xD8\x0E\xA2\xA3\xA3\xF5\x3B\xD4\x4B\x1D\x45\xD2\xE2\xE5\x02\x75\xE4\x4C\x92\x5F\x47\xEB\xF7\x68\x90\xF8\x7E\x47\x85\xD3\xED\xE9\x10\x08\x3A\x83\xAE\x8E\x9C\x49\x0E\x65\xFD\x01\x73\xA8\xAF\x9C\xA3\x38\xEF\x1F\xE5\x71\x2C\xC5\x5E\xE1\x0C\x16\x95\xFD\x84\x47\x12\x9B\x6A\xD4\x21\x49\x15\x49\x92\xC4\xF5\x03\xEA\x23\xE6\xE1\xB7\xDC\x26\x71\xDC\x4F\x9E\xE9\x4D\x71\xCD\xFC\x8B\x9A\x58\xEC\xB1\xD3\xBC\xEE\xE2\x8E\x03\x3A\x0E\x30\xFD\x26\x23\x41\xE7\xA4\xF2\xFF\x7C\xFE\x62\xD3\xAF\xF4\x78\xFD\xC5\x2E\xBF\x4B\x9C\x7B\xFA\xA4\xDC\xB8\xFC\x2D\x67\xE2\x3F\x9D\x57\x31\xF5\xA6\x24\x9B\x79\xB9\xA6\x56\x3A\xCB\x1D\x41\x6F\x33\xEF\xA0\xBF\xD2\xF5\xBB\x85\x95\xDE\xFA\xEF\x2C\x6B\x8B\x05\xFA\x9D\xAA\xF8\x4D\x59\x5B\x82\xCA\xFF\x44\xA1\xC4\x7F\xF9\xF9\x25\x6E\x4F\x71\x4C\x7E\x3B\x34\xFE\x8C\x8A\x9E\x9F\x31\x51\xE2\x75\xB0\xFE\x82\xDB\x99\xA6\x18\x7D\x4E\xF7\x05\x7A\xE3\xFF\x68\x2E\xE3\x13\xFA\xAB\xF0\xBF\x3D\x67\x2A\x71\x96\x07\x5C\x31\x35\xF0\x97\x25\x65\xFD\x15\x7F\x52\xC5\xD5\xE4\xFF\xAF\x40\xCA\x99\x44\x05\xFE\x6F\x3C\xFD\x1B\x4F\xFF\xC6\x53\x11\xAD\x1C\x0E\x57\x85\xBC\x32\x56\xE9\x71\x4F\xAD\x94\x92\x74\x38\xFD\xA5\x81\x96\x73\xF0\x6F\x42\x51\xD3\x6F\x81\xFD\x3F\x90\x8B\xF6\xED\xAD\x67\xF0\xA6\xFF\x80\x08\x99\x25\xA6\xD4\xC1\x7A\x16\x97\x32\xFF\x87\x00\x21\x3F\xDF\xEB\x93\x65\x48\x07\xEB\x6F\xE5\xBA\xFC\xF7\x1C\x5E\xF5\xBB\x51\xFD\x6E\x1C\x9C\xE9\x02\xB1\x9E\xC7\x15\x09\x7F\x0B\x84\xFF\x74\xFB\xFF\x2D\x10\xFE\x3B\x04\x82\xDF\x25\xA6\xDD\xA1\xF9\x12\x42\x58\x5E\xB7\xF2\xBB\x4A\x5D\x33\x1A\x8F\x1A\x44\xE4\x25\x40\x99\x29\xCD\xB1\xAB\x98\xF4\x22\x6F\x85\xCF\x5D\xDE\x6C\xC1\xC7\x3A\x87\xD1\xF9\x5D\x95\x01\x57\x07\xBF\xAB\x5B\x7E\xFE\x95\xC3\xBA\x75\xB4\xCE\x65\x5E\x49\xF9\x9F\x54\x22\xFF\xA2\x25\xA3\xC6\x0C\xFF\x8B\x93\xF1\x7F\x43\xF6\xFE\x10\x2A\xFE\xA2\x3C\xFC\xD7\x14\x5B\x1A\x6A\xFF\xCE\x11\xFF\x97\xA5\xF5\x27\xF1\xF0\x7F\x7B\x83\xFE\xF5\x15\xFD\xEF\xEA\x3C\xFF\x24\xE8\xFF\xE5\xB5\x1D\xB3\xA8\xF2\x17\x95\xFD\xAF\xDD\x68\x98\xC7\x7C\x98\x1A\xA7\xA0\xFF\x7F\x2A\x54\xFE\x85\x85\x89\xBF\x65\xCB\xDF\xB2\xE5\x6F\xD9\xF2\xB7\x6C\xF9\xAF\x94\x2D\x17\x5E\x60\xFA\xCB\x7B\x58\x9C\x38\xF8\xCB\x8A\xF9\x97\x8A\xB8\x7F\x7F\x81\xDA\xB7\xB7\xCE\x67\x2E\x6A\x9A\xD8\x89\xDA\x81\xC3\x59\x5E\x1E\x3D\x8F\x11\xA8\x9C\x24\xCD\xF5\x16\x30\x5F\xAA\xFE\x96\x56\xFF\x95\xE0\xF6\xB7\xB4\xFA\x5F\xD6\xA0\x7F\x4B\xAB\x7F\x97\xB4\x6A\x71\xF5\xEB\x36\x26\xD1\x1D\x70\x96\xFB\xCA\x9C\xB7\x8B\xBF\xA4\xF3\x99\x77\x88\xBF\xA4\x73\x9D\x0B\x19\x65\xB9\xB7\xB4\xAB\x7D\x11\xA3\x0A\x78\x7C\x7E\xB7\x27\x58\xE2\xB8\x93\x49\x76\x4C\x8B\xBA\x16\x33\x6A\x87\xB7\x32\xE8\x98\x54\x59\x52\xE2\xF2\x2F\x61\x54\x92\xCB\x53\x59\x5E\x7E\x17\x93\xE2\xF0\x04\xBD\x4E\x7A\xC8\x79\x29\xA3\x74\x94\x04\xBD\xCE\x65\x8C\xD2\xE1\x0A\x7A\x9D\x77\x33\x49\x52\x50\xBF\x6B\xDA\x3D\x4C\x62\x89\x1C\xDB\xBD\x4C\x42\x89\x74\x0D\xE2\x3E\x46\x59\xE2\xAB\x0C\x16\xDD\xCF\x28\xA5\xBB\x70\x0F\x30\xF2\xF9\x38\xF1\xF7\x83\x4C\x82\x7C\x97\xE1\x21\x86\x2D\xF1\xBB\x5C\x0F\x33\x09\x45\x92\xFB\x11\x26\x91\x5E\x67\x78\x94\x49\x08\x04\xFD\x41\x6F\xF9\x72\x26\xA1\xC2\x55\x51\x54\xE6\x7F\x4C\xFE\x51\xE1\x7B\x5C\xFE\xE1\x9B\xB9\x82\x49\xAC\x70\x55\x54\x78\xA7\xB9\xAA\x25\x56\xC0\x15\x7C\x42\x7A\xAD\xDC\xE5\x79\x92\x49\x91\x2E\x16\x4C\x72\x06\x5C\xCE\xE2\xE2\xA7\x98\xD4\x58\x67\xD0\xBB\x92\xD1\x48\x8C\x40\x99\xD7\x1F\x74\x16\x17\x3F\x4D\xDD\x62\x00\xB7\xA7\xC8\xCF\xC7\xBC\x1F\xA8\x9C\x24\x30\xBA\x58\x67\x89\xDF\x5B\x51\x13\x1B\x43\x45\x65\x79\x6D\xCC\x1B\x15\x95\xE5\xA1\x58\xEF\x62\xF7\xB4\x3A\x46\xDB\xE8\x5D\xEC\x9E\x56\xE1\x2D\x7E\x26\x86\x23\x5F\x43\x7B\x36\x9E\xE3\x7F\x8E\x72\xA2\x89\x38\x8A\x56\x31\xAD\x62\xAE\x54\x14\x79\x3D\x41\xD7\x8C\xE0\x6A\x26\x95\x5E\xAB\xA0\x8C\x35\x8C\xA6\xC4\x13\x74\x14\x79\x3D\xD3\xBC\xE5\x95\x41\xD7\xF3\x4C\x9A\xE8\x76\x56\x06\xBD\x8D\xBC\x17\x18\xB6\xC8\x1F\xEC\xBE\x96\x49\x12\xBD\x8A\xDD\x25\xDD\xD6\x31\x89\x81\x60\xB1\xA3\xC4\x13\x5C\xCF\xA8\xC5\x5F\x6E\xCF\x34\xD1\xF5\x22\xA3\x29\xF1\x56\xFA\x1D\x81\xA0\xCB\x27\xBA\x37\x30\x69\x92\x4F\x2C\xEF\x25\x46\xBA\xEA\x31\x35\xE0\x8A\xDE\xF4\x79\x99\x49\x16\x39\x41\xAF\x23\x50\xE4\xDE\x48\x1B\x40\x74\x49\xBE\x9B\x98\xD4\xE9\x5E\x7F\x0C\x63\x33\xA3\x12\x03\x48\xAD\xBF\x45\x7E\x55\xEE\x14\x5B\x19\xE9\xCE\x08\xED\x18\xF5\x4C\x8A\x5C\x21\x72\xC8\x6D\x4C\x92\x94\xAC\xC7\x35\xFD\x15\x26\x4D\xF2\x99\xEE\x96\x4E\x7C\x79\x1D\xC5\x33\x3D\xAF\xCA\xB9\xA2\xEF\x8A\x9C\xD7\x64\x8E\x94\xB6\x74\xA5\x20\xF0\xBA\x9C\x58\x45\x40\xE4\x6D\x97\x6B\xD2\x1D\x70\x7B\x4A\xDC\x1E\x77\xD0\xF5\x86\x9C\x2D\x77\xC0\xE3\xF4\xBC\x19\xF5\x94\x2E\x15\xBA\xA7\xB9\xDE\x92\x5F\x75\x07\xA6\x7A\x9C\x9E\xB7\x19\x8D\xEC\x08\xF8\x5C\x45\x6E\x67\xF9\x3B\x51\xCF\x9B\x5C\x7E\xEF\xBB\x72\x3E\xC5\x71\xBB\x43\x0E\x57\xE9\xF6\x04\x1D\xA2\xD7\x7B\x4C\x22\xBD\x02\xB3\x53\x2E\xE9\x54\xBF\x2B\xE0\xBE\xC9\xB5\x4B\x0E\x17\x70\x05\xE5\x7C\xBE\x4F\x4B\xEE\x0A\x3A\xC4\xE4\x3E\x60\x74\x51\x67\x34\x3B\x1F\x36\xB1\x7C\x5E\xF9\xB8\xEC\x6E\x26\x3D\xCA\x2A\x9E\xE9\x71\x56\xB8\x8B\xA4\xEB\x7F\x7B\x1A\xE3\x76\x94\x94\x3B\x4B\x03\x7B\x65\xB7\x98\x9D\x22\xAF\xAB\xA4\x64\x9F\xEC\x9E\x2A\xF5\x28\x57\x49\xC9\x47\x72\xAD\x89\xE0\x55\x2E\x85\x0A\x7C\x1C\x2D\x2D\xBD\x11\xB6\x9F\x49\x76\x34\x32\x0E\xC8\x15\x15\x70\x05\x69\x5D\x1C\x6C\xCC\xBB\x74\xA7\xEE\x93\x68\x67\x91\x7A\x8B\x58\xD8\x4F\x69\x82\x25\x6E\x8F\xB3\xDC\x7D\x93\xEB\x10\x1D\x4D\x25\xEE\x19\x0E\x8F\xD3\xD3\x40\xBB\x4F\xD1\xFF\x63\xEF\x3B\xA0\xE3\x28\xD2\x84\x91\xAA\xFA\x27\xFC\x1C\x0B\x3E\x6E\xD9\x5B\x96\x5B\x73\x0B\x43\xD7\x00\x5E\x4B\xC2\x0B\x68\x85\x6E\x65\xC9\x02\x3D\x6C\x59\x48\xB2\x7D\xEF\x71\xDC\x6C\x6B\xA6\x24\x35\x6A\x75\xB7\x3A\x58\x92\xDF\x9D\x0E\x30\xC9\x44\x63\x72\x32\x39\xE7\x9C\x61\xD9\xDD\x63\x03\x9B\x97\xCD\x91\xBC\xE4\x9C\xD3\xBD\xAE\xAA\x4E\xD5\xD5\x3D\x3D\xB6\x47\x7E\xC7\xF1\xFC\x9E\x3C\x53\xF5\xD5\x97\xFA\xAB\xAF\xBE\xFE\xEA\xAB\x9A\x51\x5C\x1E\x2B\xE1\x29\xF3\xF1\xC6\xBF\xA7\x18\xC8\x34\x29\xA9\x3A\x09\xE3\x9F\x68\xDC\xC6\xA3\xE1\x58\xAA\xA9\xE1\x27\x29\x4D\x62\x91\x6A\x4B\xF3\x53\x0C\xC5\xC4\x08\x76\x88\xF2\x9F\xF6\xFB\x29\x8E\x67\xD8\xD7\x11\x0A\xFE\x2C\x7D\xEE\x13\x65\xC3\x9C\xFE\x1B\x7B\x2E\x74\x96\x3E\xD7\xB8\x13\x41\xA4\x98\xA6\x36\x4D\xCF\xCD\x96\xF0\x54\x19\xDB\xF6\xF3\xCC\x18\xCB\xE3\xE6\x0B\x8D\x5B\x51\x86\xC7\xCD\x17\x1B\x77\xF4\x3F\x96\x6C\x65\x1C\x97\x94\xCA\xE1\x78\xCA\x7C\x89\x49\x48\x4E\x57\xEA\x95\x97\xA3\x34\xAC\x57\x18\x26\x45\xAF\xBC\xEA\x63\x52\xCC\xD7\xA8\xA5\x4C\x18\xD6\xEB\x6C\xF0\x84\x52\xA9\xD8\xEE\xD0\x1B\xD1\xC1\xFA\x9B\x6C\xF0\x94\x61\xBD\xE5\xE3\x19\xB2\xDF\xA6\xE6\x38\x41\x8E\xE9\xBE\xC3\xA4\x33\x35\xD7\x7E\xD7\x87\xA9\x54\xDE\x63\x1F\x6D\x77\xE8\x7D\xF6\xB1\xA2\xAE\xFC\xA0\x71\xEB\x92\xFF\xF9\xC3\xC6\x1D\x02\x17\x55\xD2\xA9\x23\xFB\xC8\x67\x86\x7E\xFD\xD8\x07\x1F\x77\xB5\x4F\x98\xD2\x3D\xD7\xD5\x5C\xD2\x70\xF3\x11\x80\x76\x8E\x8D\xBB\xDA\x91\xE0\xFF\xD3\x67\xAC\x7B\xAE\xED\x28\xF0\x77\x41\x8F\xD7\xB4\x1A\x6C\xE5\x63\x39\x1A\x50\x8D\x4F\x50\x65\x3B\x06\x3D\x86\x68\x29\xDA\x31\x0C\xC8\xC2\xE3\xC7\x82\xCF\x05\x64\x4B\x78\x4A\x29\x3B\xC7\x81\x1D\x12\xA3\x8E\x07\x73\x12\x6D\x53\x6B\x00\x55\x8D\x63\xB9\x7A\xF9\x04\xF6\x85\x9C\x77\x3E\x11\x30\x2B\xC0\xAA\x76\x12\xA0\x46\xAD\xAE\xC2\xAA\xEE\xC9\x7F\x32\xA0\x56\x8C\xA7\xC8\x59\x58\xB7\xE9\x6B\xA7\xB0\x16\x76\x3A\xD6\x6D\xFA\xDA\xA9\x20\x5C\x15\xCA\xE3\xE6\x5A\xB0\x1D\x15\x70\xD2\xB0\xC6\x3C\x4C\xA7\x79\x0D\x8A\xA5\x38\xB6\x3B\xE4\xBD\xD3\x94\xD7\x81\x39\xF1\x06\x4F\x17\xA7\x83\xED\x7C\x5B\x19\xC3\x96\x8E\xB5\x33\xC0\x8E\xC1\x19\x3B\x0F\xA0\x64\x2A\x96\x32\x6E\x9F\x09\x76\x08\x9B\x27\x5B\xC8\xFB\xCE\x59\x60\x5B\x5B\x9D\x0A\xFC\xF4\xD9\x60\x7B\xCF\x77\x47\x9B\xCE\x01\xDB\x91\x52\x68\xD3\xB0\x71\xC9\x34\x26\x9B\xCF\x05\xDB\xDB\x13\xAE\x62\x61\xD2\x4C\x9A\xCE\x03\x73\xEC\x49\xF2\x8E\xA2\x0D\x5B\xC6\x24\x6D\x3C\x1F\xEC\x1E\x0D\x8B\x74\x3C\xE5\x94\x4C\x4B\x1D\xC7\x5C\x84\x72\x01\xD8\xD9\xDF\xF9\x9E\xAB\xE3\x49\xAE\xF7\x42\xF0\xF9\xA0\xB7\x82\x35\xEC\x60\x99\xE4\x1D\xD1\x7A\xB0\x4B\x74\xD8\xA1\x87\x71\x03\x2F\x02\x5F\xE0\x06\x1E\x7A\x18\x1B\x7A\x31\x98\x53\x2A\x95\xA7\x94\x92\xE9\x5A\xB8\xB4\x52\xB5\x1C\x57\xD1\x2E\x01\x0F\x37\xD4\x25\x54\x8B\x23\xAB\x6F\x52\xF9\x52\xF0\x60\xBD\x84\xF0\x95\x79\x40\x7D\x25\xB8\x0C\x58\x75\x12\x80\xAE\x9B\x32\x57\xC0\xEF\x75\xA3\xCB\xEB\x46\x94\x9E\x3D\x11\x12\xBD\x02\x4C\xD7\x97\x68\x39\xBD\xAC\xFE\x4A\x70\xF8\xAC\xD1\x46\x57\x81\xE1\x3A\x11\x33\x5D\x7B\xB4\x34\xA4\x94\xC7\x08\x3D\x74\x35\x38\xBA\x5E\xD6\x5F\x2A\x29\xB6\xA7\xC0\x12\x9E\x72\xB0\xA5\x2B\x5A\x96\x76\xAF\xA9\xDF\x93\x25\x4C\x64\xD1\xBE\xB6\x7E\x4F\x36\x41\x1B\x5D\x07\xD6\xD7\x4B\xE1\xEC\x44\x1D\x3F\x73\x04\x13\x29\x45\x11\xB4\xF9\x7A\xA0\x78\x3E\x3F\x96\x0E\xB1\x0D\xCB\x89\xBD\xB1\x6B\xD8\xB6\xDB\x54\xDD\xD9\x6B\xAE\xAA\x3B\xED\x05\xF2\x5F\xB1\x5D\xF6\xFE\xD2\xCF\xF1\x74\x0A\x07\x8E\x6E\x00\xFF\x11\x90\x56\x75\x27\x41\x6B\x41\xAD\xC4\x92\x7F\xB3\xC8\xDF\x08\x56\x0E\x19\x86\x16\x83\xA1\x07\x55\x54\x43\x27\x0C\x94\x54\xDD\x53\xA7\xB7\x06\x6E\x4A\xB9\x6F\x02\xDD\xD1\x6E\xEF\x7D\xAC\x64\xE8\x65\xEE\x99\xCD\x5D\x69\x68\x8A\xA3\x6A\xB8\xC0\x76\x0C\xE9\x7F\x73\xE5\x22\x62\xCB\x31\xBA\x19\xEC\x4E\xB2\x19\x0B\x55\x67\xC0\x51\x1C\xDC\xDA\xDA\xE7\xDA\xA3\x32\x21\x15\x35\xB7\x5B\x40\x91\x83\x1B\xB4\xA6\x07\xB0\x62\x95\x05\xC0\xB7\x82\x31\x0E\x98\x41\x92\xD6\x01\x62\xA5\x7D\x2A\x2E\xE3\x20\x5B\x9D\xDA\xE1\xA9\xD7\xFF\xCB\x03\x51\x4D\xA1\xDB\xC0\x49\x0D\xA4\xAB\xCF\x32\x46\x7C\x52\x3E\xE9\x0D\x20\x19\xC1\xD5\xA1\x97\x47\x0D\x2B\xD6\xB4\x44\x71\xCA\xA3\x07\xAB\x7A\x25\x95\x9F\xDB\xC1\x2E\xA4\xA7\x93\x96\x34\x59\xAD\xAD\x1D\xDE\x1C\xEB\xD1\x6D\x72\xD2\x09\xDD\x01\xBE\xC8\xF5\xCF\xF8\x1F\x65\x74\x27\x40\x34\xBB\x84\x47\xF0\x94\xD9\xDA\xBA\x42\xD1\xC6\xB0\xD5\x46\xDA\xBA\x2D\x65\xA4\xBD\xB5\x75\x86\xB6\xC9\xE8\xAE\x2C\x44\x77\x03\x99\xEB\xEC\x54\x1C\x39\xC0\xC3\xB8\xF7\x3E\xA2\x7B\xC0\x6E\x1C\xE8\x80\xA3\x58\x51\x58\x4F\xFF\xE8\x5E\x66\x2A\x21\xD8\x21\x2E\xB6\x9D\x04\xDC\x7D\x60\x47\x0E\xAE\xD7\x30\x65\x74\x7F\x82\xA1\x85\xD3\x0E\xEE\x27\x87\xB0\x7C\xCB\x66\x18\x1E\x00\xFF\xC8\x81\x12\xAD\x13\xED\x3D\x08\xF6\xE0\xFA\x16\x8D\x9B\xCE\xF4\x0A\xB5\xE2\x30\xFB\x22\xDF\x97\x9A\xE8\x21\x50\x48\x68\xC0\x74\x5C\x0B\x47\x39\xF6\x50\x7E\x0B\xEC\xCC\x73\x86\x47\x54\x9D\xB2\x86\x1E\x06\xBD\x5C\xEF\x32\x9D\x94\xD5\x55\xFA\x5D\x1D\x7B\x32\x0C\x90\xF3\xF4\xE1\xD4\xA3\xFE\x97\xFB\x4A\x4D\xD8\x23\xF7\x6D\x70\xB7\x94\x48\x5A\x6E\x44\xB9\x41\x7C\xCA\x53\xE7\xEC\x79\x89\x8D\x2B\x3D\x4B\xC7\x5A\x75\x9F\x44\x34\x74\x63\x49\x91\xF4\x72\x12\x50\x9C\xEE\x16\xC2\xE5\xDC\x13\xA8\x07\xF3\xD9\x8C\xE5\x11\x72\x03\xF3\xDF\x69\xCC\x45\x0A\x78\x04\xE4\xE4\x64\x9B\xFF\x8A\xF0\x1D\x70\xE7\x96\xA9\x3B\x8B\xB3\x61\xCA\x1B\x51\xF0\xF2\x99\x45\x7F\xEA\x2D\x3A\xBD\x6C\x20\x5B\x09\xA6\xB7\x7C\x4F\xAA\x36\x2E\x11\x3B\xB7\xDC\xB2\x53\x72\x04\x5B\xE9\x8E\x6B\x6A\x22\x1E\x18\x68\x7B\x12\xB6\x7D\x6E\xC6\x74\xAA\x07\x0B\x05\x01\x74\x7B\xA1\x80\xBE\x0B\xF6\xE4\xA3\x92\x4A\x85\xAE\x5C\xFD\xB8\xEC\x5A\xB6\xBA\x32\x5C\x87\xD1\x7F\x27\xD6\xCE\x6E\x55\xAF\x24\x17\x6C\xF4\x48\x62\x39\x0E\x80\x16\x79\x36\x12\x42\x7E\x4F\xC4\x80\xB7\x84\x0A\x43\x80\xEF\x83\xAF\x66\x40\x2F\x1B\xEC\xDE\x8F\x1F\xF1\x03\xD0\x94\x31\x62\xB1\xE2\xA8\x7A\x13\x3F\xE6\x87\xE0\x9F\x92\x63\x4A\xFB\xCD\x2F\x35\xCD\x1F\x1E\x1E\x1E\x96\xD1\xA3\x09\x80\x4E\xC3\x9C\x0E\x03\x08\xF4\x23\xD0\xCC\xC7\x4E\xA3\x86\xE5\x2C\x57\x6D\x95\x45\x46\x34\x92\x2B\x46\x03\xAE\x1F\x83\x7D\xB9\x41\x7D\x16\xCE\x1C\x42\xF9\x2D\xA2\x9F\x80\x5D\xB9\x91\x8B\xE9\xC5\x68\x72\x28\xD4\x4F\xC1\x12\x1E\xBB\x61\x67\x73\x24\xFE\xC8\xE2\xD9\x9F\x81\x05\x09\x1D\xD0\x62\xFD\x38\x36\x1A\xE1\x04\x06\x89\x7E\xCE\x6C\xA8\xC7\xA6\x61\xB4\x17\x54\xC6\x39\x60\x04\x7E\xC1\xE2\x4F\x1F\x70\x91\x5E\x11\x81\xFD\x12\x1C\x5C\x2D\x34\xF6\x9A\x7A\x74\x96\x25\xC8\x54\x25\x7A\x0C\x7C\x81\x93\xAA\xCB\x70\x48\xDC\x8B\x7E\x05\xBE\x92\x34\x7E\xD5\x1E\x8D\x61\x44\xBF\x66\xE1\xB0\xCF\x0E\x83\x1E\x34\xBC\xF7\x04\x39\x54\xC3\x6F\xC0\x5E\x62\xB8\x7E\x7A\x89\x14\x07\xFE\x5B\xE0\x44\xFD\x5F\x05\x7B\x6E\x6C\x48\xB1\x31\x15\xD4\x93\x90\xBC\xD8\x44\xE4\x16\x7B\xD1\x54\x70\x76\x5A\x2C\x8A\x5B\x46\xBF\xAB\xE1\xBD\xE3\xF7\xE0\xA0\x6A\xB0\x79\xED\xF9\x0F\xE0\xDF\xAB\xA2\xDA\x28\xE3\xFD\x23\xF3\x23\x19\x04\xB8\x19\xFD\x27\x70\x03\xFC\x2C\xBA\xF8\x2C\xBA\x10\x45\x17\xC2\x32\x8F\x3F\x03\x3B\xC4\x49\xE6\xD4\x26\x9E\xAA\x25\x72\x7F\xD0\x90\x52\x1E\x2B\x95\x15\x53\x29\xAB\xCE\xB4\x8C\xFE\x02\x4E\x6D\x88\x65\xBB\x4C\x4D\xF5\x6B\x45\xD2\x11\x16\x6B\x64\xA0\xC8\xE5\x7A\xD3\x01\x0B\x05\xF4\x57\xB0\x76\x76\x59\x1A\xB6\x0C\xDD\xC9\xE6\xE9\xF1\x59\x52\x53\x21\xBF\x9A\x9E\x00\x67\xCE\x2E\x4B\xD5\xD4\xE4\xBF\xDC\x3D\x09\x5A\x49\x6B\x57\x77\x07\xF9\x23\x07\x69\xAF\x62\x5A\x0A\x2C\x5C\xB7\x9E\x02\xDB\x87\x83\x67\xBC\xD1\xE8\x69\x96\x39\x21\x4D\x2B\x0C\x6B\x6C\x62\xD0\xE8\x24\x99\x93\x48\x6A\x2E\xEC\xF4\x89\xC4\x5A\xA2\xD9\x5D\xF4\x0C\x98\x17\x42\x44\x31\x05\x69\x53\x6E\xC0\xB3\xE0\x9D\x86\x5A\xDF\x4F\x43\x0A\x03\xEC\x6A\xA0\x6A\x69\x94\xE4\x08\xAE\xE5\x20\xC5\x1E\x4D\x34\x92\xF0\x38\xE3\xA1\x46\x10\x86\x6F\xEB\x89\x1E\x99\x6F\x09\xEE\xEC\x02\x8F\xF1\xF7\x98\x6C\x3A\x4D\x54\x79\x0B\x9F\x1D\x85\xA4\xBF\xEC\x55\xA5\xEF\x6B\x29\x55\x7D\xA9\x23\xD0\x73\x60\x3E\xE9\x1A\x20\xB7\x48\x0D\x60\x67\xB0\xCD\xD3\x4A\x7B\x6B\x6B\x0F\x49\xFC\x07\xE1\x67\x98\xF0\x7B\x1E\xEC\x1F\x62\xEB\xA8\x54\x06\x8D\x43\x5C\xEC\x8A\xEC\x3F\x69\xC1\x2F\x80\xE5\x21\x58\xBF\xAB\x13\xC8\xA5\xBA\xF7\x92\x95\x6B\xFE\x24\x30\xFA\x91\xD7\x8B\x2C\x30\xF2\x11\x13\x29\x13\x88\x7D\xFD\x79\xAC\xBC\xC4\x42\x44\x3A\x02\xDB\xD8\x21\x53\x30\x02\xDD\xBF\x62\xB1\x51\x1E\xC3\x56\x11\xBD\xCC\x82\x5E\x8A\x84\x64\xE4\xBB\xBB\xBB\x65\xBE\xAD\x8F\x14\x47\x14\xD1\x2B\xA0\x97\xEC\xA3\x84\xFD\x3D\xBA\xA6\xEA\xB8\x42\xC1\x16\x1B\x86\xD9\x46\x2E\xAE\x61\xF7\xD7\x04\xD7\xD8\xA4\x21\x7C\x55\x44\x7F\x30\x15\xFC\x35\xB0\xA4\x46\xFA\x24\xC0\x48\xC5\xF7\xBA\x80\xFC\x60\xBA\xF8\x6F\xE4\x27\xEF\xD1\xAD\x2A\xFD\x9B\x22\xF2\xE9\xD2\xBF\x05\x16\xD7\x46\x3E\x5B\xF8\xB7\x93\xD4\x07\x33\x9E\xFD\x3B\x39\x84\x8F\x48\x5D\x55\xF8\x77\x45\xE4\xD3\x85\x7F\x2F\x87\xF0\x31\xF2\xD9\xC2\xBF\x2F\xA0\x9E\xF1\xE4\x3F\xC8\x4D\x3D\xD7\x83\xFF\x50\x44\x3D\x5D\xF6\x8F\xC0\xC1\x35\x51\xCF\x16\xFD\x63\xE6\x19\xA9\x9B\xD3\x15\x6D\x7A\x15\x8E\xEE\xFB\x09\x86\x7C\x02\x8E\x6C\xE0\xFB\x36\x72\x93\x30\xFC\x5B\x8C\x6D\x4A\xFA\xDE\x91\xF7\xDB\x45\x74\x04\xDC\x23\x12\xDC\x1C\x88\x9D\x58\xE8\x13\x09\x77\xD0\x91\x70\x32\xB1\xA3\x1D\x6C\xBD\x96\x4C\xCB\x98\x9A\x6E\xE3\x73\x6F\x39\x51\xB7\xB6\xEE\x56\x9A\x5F\x88\xEE\x3E\x16\x0B\x85\xF6\xB9\xED\x6C\x9F\xF6\xA8\xFA\x52\x6E\x4A\xA7\xBC\xBA\xBE\x94\x9B\xD3\x29\x1F\x0D\xBF\x1C\x19\xD6\x45\xEA\xC8\x02\x5C\x5D\xDD\x1D\x45\x74\x0C\xBC\x24\xB9\xF3\x1B\x80\xCC\xDA\xA6\x2F\x33\xB6\x14\xF3\x3A\x16\xCA\x29\x01\x43\xD9\xC2\x8A\x83\xE9\x85\xE4\x64\x6F\xF3\x38\x78\x46\x32\x5C\x9D\xDD\x90\x4A\xF8\x86\x7B\x3C\xA4\xFB\xB1\xBD\x1E\x78\x6F\xEC\xC5\x00\xAD\x81\xD1\x78\xA8\xC3\xB2\x94\xE9\xB6\x10\x78\x70\xD4\xC2\x4A\xA5\x18\x96\x80\x79\x62\x9E\x00\xB7\x0F\x21\x66\x3C\x7C\xE8\x44\xA8\x64\x65\xBD\x22\xB8\x32\xA4\x89\x40\x89\x52\x5C\x27\xC1\xFF\x0A\xC1\x48\x18\x46\x61\x27\xE4\x1C\x02\xF8\xA1\x14\xF9\x93\x6A\x45\xB1\x9A\x98\x04\x12\x74\x32\xFC\xA6\x30\x53\xB0\x61\xE2\x09\xD3\x02\xA7\xC0\x63\x1A\x42\xE8\x01\x07\x9B\x39\xA5\xAB\x41\x07\x79\xC4\x47\xA7\xC2\xE1\x08\x1F\x75\x2C\xFF\x58\x0B\x9B\x23\x33\xB5\x8F\xDC\x7A\xDF\x51\x2E\x63\x8D\xB8\x91\x94\x0A\xA5\xD3\xE0\x71\x49\xCF\xD1\x3B\xEB\x9E\xC3\xE3\x7F\x1D\xDC\x3F\xA1\x7B\x55\x77\xDA\x3D\xAE\x04\x6F\x13\xE4\x8F\xFF\x32\x72\x3A\x4C\x3E\x5D\x3A\x36\xE1\x5B\xCE\x80\x5F\x4F\x27\xB3\x68\x4A\xB5\x1D\x55\x1F\x09\xC8\xF1\x84\xCE\x84\x4E\x76\xCA\x22\x6A\x28\xF9\x6C\x58\x98\x5A\x8A\xF6\x17\x0A\xE8\x2C\xE8\xD6\x91\x6C\x24\x2F\xC2\xD1\x3D\xBB\x1E\xE2\x0A\x52\x44\x1C\xD9\x73\xE0\xAA\x3A\x92\x4D\x11\xD7\x7F\xC4\xE7\xC2\x13\x93\x53\x62\xA9\x8E\xFB\x14\xDB\xDE\x0C\xD3\xE2\x3C\xF8\xF9\x08\x7C\x8F\xED\x73\x82\xCE\x87\xD1\xC2\x47\x76\xF5\x37\x57\x6C\x23\xD2\x4C\x0C\x84\x18\x01\xBB\x52\x3A\xAA\xF2\x49\x4B\x31\x49\x5A\x24\x0E\xCE\xBC\x48\x7B\xA2\x08\x52\x00\x55\x40\x17\xC0\x75\xB1\x65\xDC\xB1\x30\x73\xF4\xC1\x76\xA4\x7F\x86\x30\xD8\x9E\xC4\xB6\x9D\xF1\x40\x03\x38\xC2\x77\x05\xDB\x8E\x65\x4C\xCB\x3C\x8D\x48\xD6\x26\x42\x88\xE5\x6C\xD0\x85\x90\xEE\x8F\xFA\x3B\x21\x7D\x24\x1C\x09\x6B\xFE\x68\x7B\x7C\x8F\x6B\x3D\x3C\x28\x75\xCC\x12\x65\x7A\x08\x77\x92\x1F\x6A\xA1\x4F\x4F\x0E\x17\x87\x18\x7C\xB7\xA6\x8C\xD8\xE8\x22\xB8\x13\xE9\xEA\x9C\x2E\x6B\xB8\xDB\xD0\xC8\xD6\x2C\xF1\x4C\x17\x43\x94\xC9\x58\x64\x87\x13\x5D\x02\xF7\xC9\x84\x1D\x50\xC7\x4D\x0D\x2F\x8D\x89\xB1\xD4\x44\x97\xC2\x7F\xCE\x1C\xD6\x65\x38\x32\xBA\x0C\x0E\x54\x51\x90\x89\x15\x87\xC7\x5D\x65\xF1\x42\x97\x33\xF7\xEE\x8D\x76\x54\x47\x35\x74\xBA\x05\x95\xBA\x47\x16\xC9\xD1\x5C\x01\x3B\xD3\xC6\xA6\x6C\x8A\x85\x31\x4A\x90\x12\x45\x57\x06\x3A\xE3\xB1\xA4\x6E\x55\x7B\xC3\xAE\x82\x7D\x55\xD5\x41\xD1\xC9\x39\x22\x23\xA2\x8B\xAB\xE1\xBF\xC4\x50\xB2\xCD\x38\xBA\x0C\x65\xEC\xDD\x86\x7B\xDB\xD7\xC0\xAF\xA4\x63\x08\x36\x28\xAF\x85\xA3\xF9\xF6\x50\xF9\x8D\xA7\xD4\xBC\x7A\xB8\x09\xC4\x05\x95\xD7\xC1\x03\x52\xB5\xD4\x65\x2C\xC6\xC3\x4E\x9F\x62\x61\x3D\xCD\x81\xA2\xEB\xA1\x9C\x31\x7E\x39\xB6\x1C\xB5\xAC\x68\x0B\x15\x4B\x46\x37\xB0\x40\x47\x0C\xDA\x69\x68\x9A\x62\xDA\x98\xB3\xFC\x1B\xD9\xCB\xAD\x78\x50\xBF\x3A\x32\xCA\x18\x44\x37\x71\xAA\x8D\x43\xB2\x1D\x70\x74\x33\x6C\x89\x41\xF5\x93\x1F\xB1\x58\x8C\x95\x8A\xAA\x8F\x30\x3F\x90\xB4\xA5\x5B\x60\x7C\xEB\xBE\x5B\xF1\xBC\x76\x87\x46\x9E\x36\x31\xA1\xE8\x98\x58\x9C\x29\x70\x25\xB7\xC2\x35\x0D\xDC\x3B\x29\x5B\x06\xC8\x88\x6E\x4B\x19\xC7\x19\x4F\x95\xF4\x73\x19\x64\x12\x45\xDB\x9A\x31\x49\x7E\x8E\xA3\x2D\xC6\x0D\x2D\xCE\xF6\x73\xC5\x5C\x2B\xBA\x0D\xAE\xA3\xCB\x66\x42\xA6\x9E\x71\x53\x6B\x6D\xED\x37\x5C\xBD\xD2\x5C\x8B\x7C\x7B\xA5\xC8\x35\x60\x6A\x6A\x39\x4B\x30\x0A\x40\x7E\xE6\xE1\xF6\x3C\x5C\xB5\xCC\x36\x57\x77\xE4\xE1\xAA\x69\xB6\xB9\xBA\x13\xD2\x6D\xAF\x8E\x4A\xC5\x5B\x97\x70\x85\x16\x24\xD1\xD5\x6A\x54\xB1\x3A\x35\xC5\xB6\x17\xBA\xAA\x56\x21\xBF\x32\x11\x73\xB1\xE8\x2E\x78\xA0\x10\x90\x56\x1F\x79\x88\x08\x9F\xBC\x8B\x14\x58\xF5\xDD\x50\xA7\x01\x8F\xD7\xB4\x8C\xFE\xAC\xD3\x81\x96\xE1\x9A\x09\xDF\x51\xCC\x50\x47\x1A\xD7\x91\x01\xDE\x9C\x76\xED\x22\xBA\x87\xB9\xAE\x1E\x7B\xB9\xA2\xA9\x15\x52\x53\x95\x1D\xCE\xC5\xC7\xDF\x0B\xFF\xD5\xD7\xDB\xB2\x08\xA3\x69\xC4\x29\x4C\xF0\x3E\x96\xA9\x8B\xFB\xD8\xC2\x17\xE1\x64\xD0\xF0\x03\x86\xA2\x30\x66\x14\x30\x78\x3F\x34\xD2\x57\x30\xEF\x73\x67\xA7\xC7\xAC\x52\x76\xB0\x25\x52\xB2\x98\x56\x96\x4A\x1E\x80\x0B\xC3\x67\xB8\x88\xFE\x64\x4E\x36\xE2\xD8\x78\x6A\x52\x0F\xC2\xC3\xAA\xB0\xED\x6B\xB8\x8A\x65\x14\xC5\x6A\x79\x88\xD9\x7B\x2A\xFE\x3E\x6C\x69\xD4\x6A\x13\xF8\xD1\xB7\x58\xBA\x87\x19\x0D\xAB\xA8\xEF\x55\xC6\x53\x4F\x5A\xA0\x87\x99\x9D\xC4\xE8\xE5\xB2\x34\x81\x69\xC7\x45\xF9\x36\x2C\x86\xFA\xEE\xA1\xE7\xDD\xD3\x14\x8E\xBE\xC3\x80\x93\x71\xD0\x0C\xDF\x24\xA3\xEF\x66\x05\x1A\xA4\x62\x88\x94\x49\xE6\x0A\x47\x1E\x81\x6D\xE9\x50\xD5\xA3\xC0\xEF\xC1\xEE\x8C\xE1\x35\x04\x82\xDF\x87\x23\x55\x0A\x55\x36\x20\x1E\x12\x66\xA1\x7E\x00\xFF\x33\x6F\x85\x83\x87\x29\x5F\x6D\x03\x81\xCC\xAE\x47\x21\x20\x85\x02\xFA\x21\x9C\xA9\x3B\x7D\x61\x55\x85\xCF\xC0\xA3\xF5\x53\x40\x66\xA5\x89\x4F\xFF\x47\x70\x75\xEE\x1A\x93\x0D\x66\x20\x5D\x03\xFE\xD4\xFF\x31\x7B\xF5\x60\x2F\xF1\xBA\xED\x78\x7F\x55\xA7\x43\x73\xE4\xF8\xA6\x74\x6C\xD3\xFB\x27\x2C\x57\xC5\x0F\x13\x1D\x18\x4A\x8E\xFE\x29\xFC\xAA\x70\xB4\x7F\xFA\x27\x39\xE2\x67\x29\xF4\x52\x4E\x16\x71\xA3\x7F\x1E\xDB\xA7\x08\x47\x87\x67\x96\x7E\x01\x0B\x42\x88\x5E\xC3\x8C\xA9\x01\xFD\x12\xEE\x2C\x04\xEC\x56\x54\x4D\x46\x8F\xC1\x1D\x22\xBD\xA4\x6E\x14\xFD\x0A\xCE\x89\xB4\xCD\xD0\xC6\x5F\xC3\x7F\x88\x34\x2E\x35\x1D\x75\x5C\x5D\x85\x65\xF4\x9B\x98\x9C\x44\x1A\xB1\x9F\x17\x66\x74\x7F\xCB\x9C\x9D\xF7\x14\x96\x28\x66\x10\xE8\x2C\x51\xAC\xB1\xB0\xCC\xFB\x77\xF0\x4B\x62\x28\x6C\x8D\x60\x19\xFD\x1E\xEE\x2A\xEC\xEE\xC7\x65\x43\x33\x2C\xA2\xAE\x3F\xB0\x35\x86\x87\x21\x1F\xE2\xC7\xAC\x7C\xDF\xFE\xC7\x98\xE2\x3A\x8D\x71\xD3\x75\x30\x1B\x2F\xA3\x3F\xC1\x1D\x23\xBD\xDD\x9A\xE2\x38\xDE\x1B\xCF\x9F\xE1\xA5\x8D\x91\x76\x4F\x8E\x01\xB7\x5C\xC6\xB6\x6D\x58\xB6\x38\xB1\x2A\x4A\x94\x27\xE6\x4F\xE2\xE7\xE8\x58\x03\xD1\x91\x68\x9A\x51\x6F\x5A\xAD\x5C\x31\x1F\x16\xEF\x5F\xEA\x66\x94\x80\xCB\x2A\xE8\x8A\xE8\x2F\x2C\x8A\x4A\x26\x8E\x35\x6C\xA7\xA8\xC9\xB3\xDA\x0A\x9E\x5A\xAE\x68\x2E\x9F\x2A\xAC\x0E\x89\xFE\x0A\x1F\x6A\x14\x9F\x1C\x26\xBB\xD5\x72\xB1\x80\xEA\x41\x36\x07\xA8\x5F\x6F\x95\x09\x93\x07\x0F\x5D\xDB\xEB\x26\x0A\x7A\x1C\x5E\xC9\x9B\x76\x97\x41\x7E\x27\x93\xCE\xB1\x74\x33\xFE\xBF\x66\xDE\x4F\xC0\x47\x1A\x62\x4E\x51\x75\x16\xAB\xF4\xA4\x6E\x0D\xCA\xE0\xBC\x76\xC6\x6A\x1A\x02\x6D\x6A\x49\x9E\x84\x5A\xD2\x07\x1E\xA4\xEA\x8E\x2D\x6F\x42\x7E\x03\x47\xFF\x14\x3C\xB9\x61\x36\xE9\x91\x3A\x85\xD8\x2F\xB5\x04\x5D\xD4\xEC\x9F\x66\xD9\xC5\xC4\x46\x60\xA9\xDB\x0B\x54\x3A\xF4\xCA\x42\x2F\x62\xCA\xD8\x15\x7C\x06\xAE\x05\x99\xD7\x09\xEC\xF3\x69\xF2\x40\xFF\x3B\x3C\xD9\xB3\xF0\xD1\xE4\x62\xC0\xDD\xB1\xD0\xF2\x69\x7A\x2C\xF5\x55\xE7\xDF\xE0\x63\x8D\x35\x5C\x59\xF1\x99\x5E\x73\xEA\xF5\x39\xD8\x41\x5F\xC0\x17\x35\xD3\x90\xBD\x5A\x82\xC3\x83\xF3\x22\x73\x43\xB7\x7D\x24\xCF\xC3\xD5\x7C\x5E\x3B\x67\xAD\xD5\xC6\x50\x65\x95\x67\x61\xDD\xD5\x0B\xF0\x1B\x21\x70\xFF\xA2\xE6\x0D\xC0\xF9\x22\xDC\x25\x82\x22\x72\xCE\x8D\xB9\xEA\x97\xA0\xB5\x11\x82\x0A\x10\xC6\x8B\xD8\xFA\x17\x35\x33\x0F\x1F\x95\xEB\x65\x56\x76\x44\x70\xCC\x78\x82\xA1\x57\xE0\xEA\xFF\x97\xD8\xA9\x8D\x36\xD4\xFF\x2E\xD9\xE4\x19\xB1\x71\xC5\x8C\xFF\xB4\xFD\x6C\xDC\x53\x3C\xFB\x82\x92\x2B\x6D\xEA\x42\xAB\x86\x0B\x83\x67\x47\xEC\xB9\x39\xB6\xED\x37\xC7\xC3\xF0\x4B\x03\x5E\x85\x4F\xC0\xBC\xD3\x80\x0B\x43\xEB\xF8\xC3\x75\xC9\x79\xC0\xD1\xDE\x2C\x7C\xF9\x37\x31\x6D\x90\x8D\xD5\x93\xC9\x8D\x30\xB2\xFA\xEA\xCE\xB7\xB2\xD7\xA0\x13\xFA\xDF\x2E\x83\xA6\xC7\x72\xAC\x2D\x7E\xF9\x50\x2C\x56\x2F\xC6\x20\x2C\x76\xFD\x41\x31\xBA\xBD\xC4\x96\x86\xD7\xE1\xCA\x10\xB2\x0A\xD1\xEC\x1B\xCF\x92\x2C\x89\xB7\x1A\x18\xE1\x37\x60\x39\x1C\xD1\xA7\x58\x8E\xAA\x68\x84\x7E\x6F\x75\xA9\x45\x6A\x10\x0B\xF9\x26\xA4\xB7\xAC\x11\xC0\x51\xC3\x18\xB3\x3D\xE5\xF6\x1A\xCE\xA8\xF7\xF4\x62\xAD\xDD\x1D\x24\x45\x4B\x4E\xED\x90\xF3\x3B\x2C\xB8\x4A\xED\xF7\x97\xF3\xB7\x58\x11\x01\x03\xA4\x35\xDD\x1C\xEC\x41\x86\x31\x26\xA3\xB7\xE1\x61\x35\x70\x43\x4F\xA6\x28\xAA\xE6\x5A\x38\xC9\x4B\xB4\x37\xF8\x9D\x6E\xB8\xBB\x80\x93\x28\x24\xE5\xE3\x5D\x96\xE9\x0C\xCB\x14\xF8\xE2\xA6\xA0\x72\x47\xB0\xFD\xF8\x1E\xAB\x3D\xF3\xBB\x66\xD8\x70\xF4\x3E\x3C\x2B\x19\xA1\x91\xE9\x35\xA4\x68\x8A\x17\xBB\x28\xC3\x0E\xB6\x58\x40\xDD\x26\x9C\x84\xB4\x10\xC5\x3F\x49\x27\x9E\xA8\x31\x98\xBD\x92\xD4\x12\x30\xE8\x03\x96\xF2\xF5\x59\xEE\xD1\xCB\x16\x1E\x96\xD1\x87\x70\xC5\x06\x46\x5A\x1C\xA2\x44\x98\xF8\x11\x47\xB1\x0B\x53\xC0\x8F\xE1\xF1\x29\x3A\xB2\x48\xAD\xC8\x6C\x6A\xE5\x13\xEE\x41\x76\x31\xEF\x88\x8E\x90\x76\x8D\x75\xB0\xAB\x42\x06\x8D\xB0\x94\x0D\x1D\x29\xF5\xC7\x60\x06\x1C\xC5\xEA\xD3\x5C\x7B\xA9\x15\xB9\xF7\x4C\x68\x4C\xE9\x1B\xF4\xE8\x28\x69\xBF\x78\x97\xE6\xDA\xA2\x8B\x0D\x04\x43\x57\x73\x43\xC3\x4B\xDA\xAA\x0E\x3D\x5A\xDA\x3F\xD6\x95\x10\x20\x63\xEC\x31\xD2\x37\x63\x5D\xB4\xE4\x6F\x69\x50\xBD\x81\xB3\x14\x51\xB5\x7C\x83\x54\x75\x1D\x2B\x7D\x43\x40\xA2\xA6\x9A\xA0\xE3\xA4\x78\x85\x51\xC0\x5D\xAF\x41\x8B\x4D\x6A\xC2\x76\x3C\xCF\x50\xF4\xE2\xB8\x6A\x1A\xA3\x7E\x79\x8D\xD4\xC5\x79\x20\x13\x2B\x39\x75\x1E\xC9\x7A\x9D\x20\xB5\xC7\x80\x58\x31\x64\x68\xA6\xD5\x44\x39\x51\xEA\x8C\x75\xF5\xE2\x49\xAE\x7E\x20\xF8\x9A\x61\x04\x27\x49\x5F\x8E\xDB\xDE\xA4\xC2\x95\x8A\x9E\x2C\xCD\x8B\x41\xD0\xDB\x7F\x52\xC5\x2D\xA2\x53\x24\x5A\xF6\x39\x68\x98\x55\x61\x4F\x95\x0E\x15\xA7\x14\x03\xF0\xF4\xD2\x59\x0A\xE1\xEF\x55\x0B\xEE\x04\x46\x6B\x63\xC2\xD1\xDA\x82\xD6\xD6\x41\x3C\xE5\xF8\xAF\x97\xA7\x49\x5F\x8C\x2B\xD1\x1D\x67\x26\x61\xCB\x68\x9D\xC4\x0E\xB1\x75\x1A\xFA\x4A\x6C\x39\x9E\x23\xB1\x07\x8D\x85\xD3\x0E\xB6\xC3\xBA\x7D\xFF\x39\xD5\x25\xCC\x2A\xA2\xD3\xA5\x35\x0D\x9C\xBD\x4D\xB8\xAA\x85\x2B\x34\x0F\x5A\x9F\x8B\xA3\xB9\xB3\x4F\xFE\xE4\x42\x67\x48\x47\x65\x31\xD3\x6D\x58\xF4\x80\x46\x5D\x99\x42\x67\x32\x03\x4B\x96\x66\xF1\x2D\x32\x3A\x2B\x15\xD6\x2F\xE3\x0A\xB3\xCD\x67\x4B\x5F\x4F\x87\x4D\x9B\x5B\x7E\x31\x14\x3A\x47\xDA\x2D\x8D\x2B\x43\x77\x14\x55\x27\xB5\x62\xE8\x5C\x69\x8F\x14\x30\x5A\x71\xD9\x31\x64\xD0\xFB\xBA\xD0\x79\xCC\x7A\x93\x90\xBD\x78\xC4\x73\xCF\xE8\x7C\x69\xF7\x14\x88\x03\xB1\x13\x32\x8C\x2E\x90\xE8\xE9\xD1\x88\x75\x07\x15\x32\x89\x36\x19\x5D\x28\xB5\xA6\x81\x57\x2F\x6A\x59\x2F\x2D\x48\x1B\x9C\x59\x95\x7C\x91\xB4\x53\xDC\x11\xA9\xE3\xA6\xA6\x0E\x4F\xCB\xE8\x62\xE9\x90\x58\x4F\xF4\xA6\x1F\xDF\x09\x54\xBF\x21\x2A\xF8\x46\x16\xA6\x4B\xA4\xBD\x72\xE0\x0C\x0A\x7D\x2E\x95\xBE\xC4\xAD\x63\x64\x4F\x84\x16\xA5\xCB\xE8\x32\xFF\x99\x1A\x8A\x86\xED\x32\xF6\x05\x0E\x2F\x20\xF2\xFD\xDD\xE5\xBE\x89\x71\x90\xD9\x77\x8B\xF9\xA3\xAF\x90\xFE\x4D\x38\x3A\xF3\x26\xA5\x1C\xDF\xFC\x7A\x72\xA9\x4D\x2C\x86\xA2\xFB\x4D\x59\xDC\x5D\x25\x1D\x20\x1C\xDE\x65\x08\x47\x27\x9C\xCB\xD5\x4C\x8B\xFE\x93\x4F\xD7\xE2\x35\x4C\x8B\x3C\x64\x3E\x2D\x5E\xCB\x62\x00\x7E\x74\xD6\xD5\x56\x31\xEB\x29\xA2\xEB\xD8\x83\x48\xA0\xD8\x34\x0F\xE2\x7A\x69\x1F\xB1\x7C\xEC\x3B\xE7\x8B\x7C\xC1\x6E\x90\x06\x33\x87\x09\x23\x95\xEA\xF5\xAB\x37\x4A\x05\xE1\x73\x9D\x89\x37\xC8\xE8\x26\xA9\x37\xC7\xA4\xAA\x41\xCD\x37\x33\xCA\xBC\x40\x33\xF1\x06\x19\xDD\x22\xAD\xCA\x77\x2E\x20\xC6\x4B\xFE\x8B\x77\xC2\x58\x43\x70\x52\xE0\xD6\xDA\x7C\xC9\x6D\xD2\xE1\x79\xB4\xB4\x69\x2C\xE9\x76\xE6\x8D\xB3\x89\x25\x67\xD8\x1D\xD2\x74\x95\x8A\xC2\x4D\xA2\x49\x61\x8D\xE1\x9D\xD2\x39\xB5\x5D\xA3\xE4\xE3\xAC\xE1\x26\xA5\x60\x48\x8E\x6B\xB0\x02\xD8\x42\x01\xDD\x25\x9D\xBB\x39\x78\x4B\xBF\xE8\x29\xCA\xDC\xDD\xB3\xAA\xB8\xEA\x17\x63\x45\x79\xBB\x47\x5A\xBF\x39\x78\xCB\xA1\x38\x3F\x03\x75\xAF\x44\x8F\xD1\x45\xB2\x77\xF4\x6E\x26\x99\xBB\xF6\x3B\xF2\x4B\x08\xF7\x49\xDB\x78\x9D\x8E\x61\xB9\x3A\xBE\x5F\xDA\xC6\xFB\xCF\x31\xBC\xA6\x07\xA4\xAD\x86\x5D\x4D\xF3\x1A\xB6\x9C\xD3\xB0\xC5\xE7\x4A\x25\xDB\xF1\xCC\xDC\x34\x54\x2F\x48\xD9\x7A\x87\xC6\x2D\xB6\x9C\x67\x19\x15\xC5\x51\x1A\xA4\x79\xDE\x7F\x5B\x54\xB6\x36\x2D\xA3\xE2\x96\xB1\x65\x37\x6C\x6B\x5A\x46\x19\xDB\x36\xAE\xEC\x3D\x34\xDD\xB0\xED\xB2\x21\x57\x77\xDC\xB9\x65\x4D\xD1\x47\x0E\x68\x6A\x99\x37\x7F\x5E\xD3\xDE\x7B\xEE\xD9\x3C\xBF\xB9\xA9\x69\x7E\x73\x4B\x53\xF3\x82\x96\xA6\x7D\xF6\xDC\xB7\xA5\xA2\xE0\xA1\x96\xCA\x82\xF9\xFB\x0E\xEF\xDD\x34\x83\xA7\xCC\xA6\x99\x08\xC8\x7E\x0B\xF6\x9B\xD7\xD4\xF2\x3F\x01\x00\x00\xFF\xFF\x89\x10\x7A\xF3\x3E\x70\x06\x00") -var gzippedCallGraphCSV = []byte("\x1F\x8B\x08\x00\x00\x00\x00\x00\x00\xFF\xEC\xBD\x5B\x6F\xDC\x38\xB2\x38\xFE\xEE\xCF\xF1\x43\x1E\x82\xE0\x8F\x8C\x67\x77\x91\x31\x72\x02\x38\x17\xEF\x19\x9C\xD9\x24\x27\xC9\xEE\xBC\x04\x10\x68\x89\xDD\xAD\x13\xB5\x24\x53\x94\xD3\x9E\x87\x7C\xF6\x3F\x44\xEA\xC2\x4B\x91\xA2\xEE\x6A\x47\x0F\x33\x71\xB3\x8A\x75\x63\x55\xB1\x44\x52\x54\x92\x22\x0F\xED\xF7\x9E\x9F\xE4\x31\x7D\x56\xFC\xBA\x47\x51\x8E\x3D\xFA\x90\xE2\x0B\x19\xE8\x1F\x10\xA1\x09\xC9\x63\x15\x50\xFC\x8A\xF3\xE3\x2D\x26\x5E\x18\xD3\x1A\x98\xE5\x47\x13\xC1\x02\x74\x4C\x03\xEF\x2E\xC6\xDF\xB5\xC6\x23\x3A\x79\x3E\x3D\xE9\xC8\x19\xA6\x5E\xF8\xEB\xA5\xC6\x00\xDD\x26\x84\x6A\xE8\x01\x8E\x34\xCC\x52\x4A\x9A\x78\xB7\x3B\x09\x78\x87\x82\x40\xC3\xBE\xDD\x15\x98\xBC\x4F\x0D\x4C\x49\x12\xE4\xBE\xD1\x54\x15\x58\xD3\x4E\x04\xA8\x1A\x4A\x9D\x54\x2D\x45\x86\xB2\xA6\x62\x37\x51\x5B\xB1\x07\xA8\x71\x85\x70\x77\xCC\xE1\x5E\xA0\xE6\x47\x74\x32\x69\x2D\x83\xFC\xE4\x98\x22\x22\x40\xC3\xD8\xD8\x51\x02\xA9\x1D\xB3\x84\x18\x0D\x5D\xC3\x10\x21\xE8\xC1\xFB\x1E\xD2\x83\xE7\xA3\xD4\x04\x47\x69\x8A\xE3\xC0\x44\x39\x3B\xA0\x28\x4A\xBE\x7B\x7E\x92\x3E\x98\x28\x64\xA2\xE9\x51\x14\x99\x04\xAB\x40\xB7\x49\x12\x61\x14\x37\xED\xF1\x83\xB1\x4B\x09\x32\x75\x29\x5C\x62\x8F\xE9\xC5\x6D\x1E\x46\x34\x8C\xBD\x23\x2E\x06\x46\xA0\x16\x52\x4C\xCC\x50\x6B\xD7\xCA\xE6\x00\x42\x25\x8E\x0C\xFA\x55\xD5\xC2\x0C\xD6\x19\xFF\xDA\xCA\xF9\x57\xDD\x12\xA4\x18\x5B\x74\x9B\x19\x1C\xBA\x06\xCB\x31\x27\x35\x4B\x11\x27\x77\x40\xB7\x19\xD0\x5C\xC7\x93\xC4\x5C\x88\x3F\xA9\x5D\x8F\x17\x06\x26\x49\x1E\x07\x36\xA9\x39\x02\x20\x77\x03\xD0\x25\x17\x3A\xB1\x3F\x0B\xAA\x75\xDE\x55\x30\x64\x35\x1A\x79\x54\x45\x1A\x88\x41\x15\x1F\x87\x91\x4D\x13\x06\x07\x14\xA9\xDB\x75\x3D\x9A\x2E\xC5\x5F\x10\x40\x96\xBE\x16\xC1\x20\xE2\x2E\x4A\x12\x62\x93\x91\x23\x00\x42\x36\x00\x5D\x4A\xA1\x13\xFB\x13\x04\xC9\x82\x36\x82\x18\x24\x4D\xA3\xDC\xEA\xCC\x35\x5C\xA7\xCD\x40\x80\x0A\x75\xBB\xAE\x41\xD3\xA5\x9E\xE4\xE4\x76\x80\x03\xE4\x23\x35\xC0\xA0\xD5\x31\x8C\xED\x6A\x71\x04\x40\xF8\x06\xA0\x4B\x2F\x74\xCA\xF2\x5B\x10\x20\xCB\xDF\x88\xA1\x2A\xD0\x40\x5A\x35\xB0\x91\x2D\xB2\x71\x10\xEE\x24\xCD\xF2\x88\x86\x69\xF4\x60\xD5\x5E\xC4\x01\xE8\x57\x60\xC8\x3E\x22\x0C\x30\x91\xD4\xB5\x9E\xD3\x75\x98\x81\x23\x68\x2B\x11\x68\x30\x57\x10\xDE\x87\x01\xB6\xE9\x2C\x60\xE8\xFC\x4B\x20\xA0\xAF\x00\x09\xF0\x0E\xE5\x11\x55\x34\x16\xBB\x06\xE1\xBD\xA9\x2B\xC0\x0D\xD2\x55\x00\x99\x12\x39\x36\xD5\x8F\x35\xB8\xE0\x18\x66\x61\x4C\xF1\x1E\xEC\x0A\x64\x63\xAC\x16\xC1\x52\x33\x90\xF8\xAB\x0E\x04\x1F\x81\x66\x9D\xBA\x41\xA3\xA2\x9A\xF1\x93\xD8\x47\x70\x71\xA5\xC2\xA1\x02\x0B\xC6\x11\x8B\x2C\x5E\x33\x45\xA1\x8F\xCD\x4C\x1A\x70\x65\x58\xF2\x20\x4C\x65\x32\x8A\x51\x0C\x15\x45\x93\x82\xE0\x7B\x4C\x32\x8B\x1C\x22\x82\x91\x8D\x8E\x24\x30\xBA\x0D\x69\xE6\x99\x66\x9F\x0A\xA8\xBB\x88\xD8\x4D\x1C\x42\xB1\x47\x16\xEE\x63\xA9\xB1\xAE\x58\xEA\x86\x2C\xBF\xF5\x92\xF2\xB1\xAC\xC1\x52\x44\x2B\xB2\xBF\x8E\x15\xE3\xBD\x26\x89\xE6\x32\x35\x76\x42\x34\x1E\x5E\x9C\xD0\xA6\x11\x99\xEA\x9D\x1A\x6A\x30\x42\xD5\x51\xB3\x42\xD5\x47\x36\x43\xD1\x2A\xDB\x81\xB5\x68\x86\x60\xAD\x09\x51\x3B\xAA\x96\x60\xAD\x92\x29\x90\xA9\x1C\x12\xA8\x88\x06\xAE\x5A\x64\x73\xC4\x78\x8F\xA8\x21\x43\x8A\x08\x06\xA3\x08\x50\x59\xFD\x12\x20\x5B\x40\x60\xC6\xA5\xBE\xDD\x31\x80\x09\x83\x35\xC1\xF2\xE8\x86\xAC\x39\xAA\xA6\xAB\x00\x92\xF5\x34\x51\x54\x03\x9E\xAC\xB1\x72\x32\x07\xCB\xC9\x14\x2D\x27\x30\x5C\x4E\x5A\xBC\x9C\xC0\x80\x61\xAD\xA7\x04\x60\x64\x96\x5F\xB7\xC5\x49\x8B\xA8\xEC\x10\xEE\x68\x84\x77\x86\x87\x71\x00\xC7\x30\x74\x32\x92\xA6\xBD\x0C\x56\x92\x9C\x45\x12\x21\xDF\x2A\x58\xD2\xCC\x04\xC0\xC4\xE9\x09\xEA\xCA\x7E\xC6\x26\xB0\x45\xF8\x66\x76\x86\xAD\x03\xB9\x2D\x64\x42\x75\xC8\x18\x0E\x09\xF7\x87\xD6\xB1\xE0\x48\x06\x07\x54\xC8\x18\x4D\xDD\x10\xD1\x75\xD5\xA5\x00\xC7\xA1\x21\x21\x3B\xB5\x00\x94\x7D\x5B\x04\x68\x2E\xAE\x70\xB6\xF9\x59\xC3\x98\x0F\x23\xF1\xC2\x38\x8D\x90\x6F\xA5\x66\x1C\x97\x4A\x52\x35\x5A\x44\xA0\x1E\x34\xAA\xA4\xC2\x70\xFA\x61\x40\x8A\xE2\x83\xA2\x30\xCE\xA0\x51\x90\x11\x52\x44\x32\xCC\xDA\x8C\xC0\x30\x35\x10\xAE\xD6\x21\x1A\x1A\x4D\x0F\xA1\x8D\xF9\x2D\x4D\xC2\x7F\xFC\xED\xA2\x02\x3F\x3B\xE2\xA3\x7F\x10\x58\x16\x9E\x44\x32\xEC\x53\xB3\xC8\x02\x0A\x24\xB4\x42\x41\x5C\x23\xF1\x8F\xA9\x87\xEF\x0C\x4B\x69\x02\x50\xED\x13\x5B\x3B\xC5\x86\x5E\x7B\x6A\xE9\x54\x02\xF5\x3E\xD8\xDA\x09\x83\xBD\x22\x1B\xA7\x08\xE6\x14\x59\x39\x45\x00\x27\x7C\x8F\xA2\x22\x99\x79\x31\xFE\xCE\xF3\x2B\x8A\xA2\xC4\xAF\x61\x4A\x5E\xAA\x9B\x0E\x18\xA5\x5E\x4A\x49\xF1\x48\x28\x43\x38\x6F\x36\x88\x0D\xA0\xF8\x1F\x84\x16\xE4\xC7\x54\x6E\xFF\xBF\x2C\x89\x79\xB3\xE7\xED\x12\xE2\x63\x2F\x3C\xA6\x09\xA1\x1E\x93\x9C\xAF\x97\x95\x4E\xC0\x7F\x3C\x77\x45\xFC\xC5\x15\xF1\xD2\x15\xF1\x57\x57\xC4\xBF\x31\x1D\xEB\x50\x86\xA2\x40\x06\x2A\x7B\x0A\x32\x10\xD1\x64\xF7\x0F\x88\x64\xD9\x8B\xE0\x32\x0F\xA1\x0C\xFF\xE3\x6F\x5E\x98\x15\xBC\xC2\x00\x4C\xDA\x0A\x4A\xF9\x7B\x8F\x63\x2F\xC0\x7E\x12\xC0\x58\x3B\x82\x61\x80\xE8\x5D\x1A\xA9\x22\x2D\x14\xDE\x02\x00\xB8\xCF\xE9\x00\x95\x51\xD9\x6C\x56\xA4\x44\xB0\xAA\x21\x10\xC9\x28\x09\x63\xB6\x6E\x9D\xF8\x88\xE2\x40\x44\xC3\x71\x0B\xAF\x12\x41\xE0\xC5\x5B\x0C\x44\x34\x5E\x5A\xC7\x67\x42\xEC\x95\xC0\x9C\x44\xED\x4A\x0B\x48\x56\xC5\x15\x62\x36\xE5\x0B\xD4\x56\x03\x08\x48\x56\x23\x28\xC4\x40\xBE\x2C\xEC\xF3\xF8\x88\x48\x76\x40\xE0\x06\x03\x80\xC1\x9A\x9A\x4C\xC3\x7E\x6A\xF0\x3A\xC5\x48\xE0\x8C\x92\x08\xC7\x70\x37\x50\x40\x92\xC7\x34\x3C\x62\x0F\x13\x52\x96\xC7\x21\x4D\x90\x01\x24\x50\xD7\x81\xC2\x28\xCB\xC0\x2C\x4E\x49\x18\xD3\x9D\x67\xE8\xC8\xD3\x70\xB5\x67\xB0\x27\x28\x3D\x78\x04\x23\xFF\x80\x6E\x23\x6D\x9C\x6C\x68\x19\xB6\x93\x61\xCF\xF8\xED\x18\xD5\x2A\x40\xBB\x44\xFB\x16\x86\xEA\x2E\x8F\x09\xA7\x5A\xC5\x65\x43\x16\xE1\x53\x81\x12\x88\x19\x30\xCC\x82\x70\x1F\x52\x00\x89\x0F\x6A\x89\x74\xC0\x27\x1D\xA5\x1A\xB9\xD8\x17\x1D\x46\x82\x86\x59\x96\x56\xB5\xA0\x0E\x85\x18\x1A\x09\x19\xE4\x04\x08\x09\xE5\x5F\xE3\xF0\xA2\x3E\x82\x43\xC1\x08\xE6\x88\xD3\x50\xF3\x38\x2C\x02\xB5\xCC\x13\xC5\x4F\xEA\x80\x9D\xE5\x84\x24\xC5\x73\xAF\x05\xB7\x29\x22\xDC\x38\xD3\xDD\x0B\x07\x6C\x9E\x57\x5C\xB1\x4B\xDA\x46\x71\x69\xF2\x0D\xC7\xE5\x54\x1A\x45\x66\xA8\x58\x41\x81\x08\xF0\x90\x88\xE4\xAB\x99\xDA\x3C\x2A\x0D\xB6\xA6\x94\x19\x95\x87\xAE\x9D\x52\xE5\x59\x4E\x0C\x59\x63\x0B\x3F\x71\x39\x10\xC4\x4A\x6E\xFF\x0F\xFB\xDA\xE0\xC3\x0A\x62\x37\xBC\x36\x92\x00\x9D\x0C\x53\x07\x90\x9C\x64\x64\xB0\xD5\x7C\x12\x92\xC5\x78\x5C\xF2\x76\x7A\x2A\x9E\x1B\x49\xD8\x2E\x1A\x82\x17\xC6\x19\xD6\xE2\xB1\x55\x28\xB3\x34\x42\xD9\x6F\xA0\x02\x62\x80\x4A\x7D\x27\x21\xC5\xC4\xC3\xC7\x90\x56\xA1\x06\x46\x95\x88\x57\xAE\x56\xC8\x33\xB3\x15\x4B\x2D\x01\x4C\x78\x36\xAE\xC2\xD4\x63\xA2\xD1\xDA\xD1\x91\xBE\x92\x3F\x45\x2C\xCB\x5C\x00\xA0\xC9\x65\x86\x86\xC5\x23\x1A\x47\xF8\x88\xA5\x63\x51\x7B\x6C\xE0\xAE\x77\xD0\x50\x18\x05\xB8\x37\x03\x19\xEB\x3D\x18\xD3\xA6\x64\x83\x65\x72\xA5\xCE\xFD\xD4\x9C\xEB\xD8\x4D\x9D\xB7\x1D\xBB\xF9\x49\x14\x61\x9F\x86\x49\x0F\x41\x31\xF5\xA2\xE2\x27\x8A\xE0\xBE\x0D\xED\x56\x3B\x2A\xA8\xC2\x61\x1B\x23\xEB\xBE\x2E\x20\x88\x2D\xF0\x8A\x70\xBC\xA7\x07\xB7\x1E\x56\xAF\x57\x70\x3B\x1B\xBC\x4C\x98\x7A\x44\x98\x5D\x15\xE8\xD2\xC5\x20\x2D\xDD\xD9\x3F\xCE\x3D\xCB\xE2\x8F\x62\x72\x0C\x63\xB9\xCE\x70\xD2\xB2\x7E\x06\x77\xE9\xD2\x09\xD9\x36\x6A\x46\x71\xA0\x34\xC4\xFE\xB1\x91\x6C\x10\x1C\x87\xA1\xE9\x20\x6B\x54\x3C\x4C\x5A\xC6\xA3\x59\xD4\xB2\x20\x71\x19\x95\x9C\x4E\x70\xD3\x2A\x3E\x20\x96\xCD\x47\x7C\xF4\xCB\xF3\x79\x22\x66\xB3\x3E\x52\x3E\x38\xF9\xC8\x3F\x30\x1B\x29\xE4\x65\x78\xA6\xC1\x8F\xF8\x98\x84\x7F\x61\x2F\x8C\x43\x6D\x6C\x34\x98\x50\x62\x54\xB0\x34\xCF\x0E\xA6\x7E\x35\x0C\xE8\xC7\xAB\x10\x68\xC9\x0B\xC0\xD0\x2B\x97\x0A\xA9\x52\xB8\xA6\xC1\xA7\x5F\x10\xA7\xA4\x52\xF9\x11\xDB\xEB\x28\x34\x13\x8E\x5C\xF8\x49\x4C\xF1\x09\x80\x1F\xD1\xC9\x08\xAB\x77\x8C\xE4\x56\xF1\x4C\x6B\x0D\x51\x8C\xCF\x77\x4A\xA4\x1D\x13\x71\x8F\x06\x4A\x39\x12\x5C\x62\x2F\x43\x9A\xE1\x90\xDB\x1B\x7F\xD2\x29\x15\x22\x0B\x73\x9E\x26\x4A\x23\xB9\x04\xAA\x9D\x11\xA6\x58\x19\x41\x82\xCA\xB5\x88\xB8\xDD\xC1\x3B\xEE\x85\x8E\x1A\xB4\xDE\x5D\xDA\xD9\x57\x4F\xB5\x8E\x34\xF1\x32\x3F\x84\xFB\x0A\x55\xA1\x89\x6E\xBD\xBE\x2A\xC0\xBD\x38\x61\xD1\x68\x17\xBB\xC6\x6A\x91\x52\xA2\x66\x90\x56\xA2\x65\x90\x1A\xE2\x27\x4B\x5F\xEE\x5E\x29\x7B\x9E\x3A\x48\x3C\x6E\x79\x32\x20\xC9\xE3\x21\x75\xAF\x96\x54\x64\x90\x92\xA3\xB4\x6E\xCD\xDE\x9F\x4E\xB1\xDE\xFD\x03\x7A\xFD\x85\xC3\xF8\x16\x95\xAB\x82\x0A\x82\xB0\xC4\xAA\x13\xC5\x27\xB6\x8C\x9F\xFF\xF2\x0F\x13\x46\xB9\xD0\x0F\x63\x34\xE9\x58\xDC\x13\xD4\x54\x51\x40\xA2\xD9\x55\x90\xA0\xA5\x00\xD2\xEC\xA6\x74\x13\x06\x42\x13\x43\xB5\x8D\x88\x20\xD9\x46\x15\x45\xB7\x8D\x8A\xA1\xDB\x46\xC0\x60\xB6\xB9\x93\xCE\x2A\xDF\x49\x07\x91\xEF\xE4\x93\xC6\x77\xE2\x71\xE4\x6A\x07\x55\xEC\x2B\xB6\x14\x8F\xEC\x75\x8B\x48\x43\x40\xAA\xE8\xC8\x34\xE4\xFE\x4A\xDF\xA6\x5F\xB9\xA5\x2C\xF4\x95\x5A\xB2\xFC\xB6\x69\x11\x68\x88\x48\x8C\xCE\x31\x17\x8E\x0A\x37\xBF\x8E\x79\x09\x13\xFA\x56\x40\x2E\xB7\x74\x26\xA7\xDA\xBA\xBE\x43\xC2\x59\xE7\xFA\x47\xA9\x5B\xF5\x0B\xC5\x41\xD3\xBF\xD2\xAD\x21\x27\xEC\x5D\x57\x87\x78\x0C\x9C\x18\x48\xE4\x56\x37\x08\x5E\xD5\xB4\x73\x67\x92\xF0\x44\x1F\x92\x98\x35\x62\x35\xB8\x95\x16\x4D\x8B\xE0\x5F\x35\xA0\x74\x2B\x11\xED\x94\x10\xA5\xA5\x32\x81\xA8\x9C\xA8\xB7\x74\xDE\xA5\x56\x59\x38\xE0\x7C\x27\x1E\x9E\xAE\x7F\x14\x7C\xA4\xA5\x79\x81\x90\x48\xFE\x04\xD3\x3F\x89\x0C\x4E\x12\x87\xFA\x17\xD3\xE5\x24\x33\x39\xC1\x5C\x62\xBC\x87\xB8\x14\xCD\x0D\xE1\xFA\x17\x3B\xD3\xDB\xFC\x66\x32\x54\x14\x4A\x3E\x02\x41\xE5\x80\x03\x9F\x49\x32\x8F\xA0\x78\x6F\x38\xE2\x25\xA3\xE8\x07\x4A\x74\x78\x3D\x47\xE8\xD4\xD5\x82\x43\x82\xD5\x0B\x9E\x3A\x0C\x9A\x0A\x6D\x64\xC4\x75\x4C\x19\x45\x3A\x53\xA2\x80\xC4\xD3\x1D\x3A\xE1\xCA\xBA\x9E\x97\x22\x7A\x60\x56\xD4\x57\xEB\x60\x60\xF3\xC0\x63\xEB\x5C\xD5\xC3\x36\x1A\xAC\x6C\x84\x11\xD8\x42\xB7\x57\xAE\xC3\x15\x08\x82\x55\xB5\x66\x85\x9A\xDE\xA5\xB2\xA0\x06\xD5\x9F\x42\x0D\x28\x99\x47\x49\x78\xF4\x22\xBC\xA3\x66\x94\x2C\x8D\x42\x0B\x98\x60\x7E\x64\xC7\x2B\x8C\x57\xAD\x33\xD2\x43\x26\xA9\x06\x80\x84\x45\x0E\x0B\xBC\x54\xDF\x08\xE7\x03\xA6\x80\x45\x59\x2D\x72\x35\xF6\x63\x50\x82\x8F\xC9\xBD\xB6\x3A\xA6\x03\x6B\x37\x82\x7B\x95\x1A\xC1\x40\x2E\xAE\x08\x93\x7E\x19\xB9\x35\x7E\xA7\xC2\x33\x8D\x62\x22\x2C\xAD\xEB\x90\x6A\x44\xD4\x76\xA1\x40\x06\xC8\x95\x63\xBD\x4B\xC8\x11\x51\x03\x12\x64\xD3\x5D\x18\xD1\xBA\x9E\x67\xBE\xAF\x36\xCB\xA7\x67\x4C\x70\x71\x28\x04\xA0\x3C\x14\x5A\x2F\x71\x28\x34\xA0\x30\x14\x25\x4C\xFA\x65\xE4\xA6\x0C\x85\x00\xCF\x34\x8A\xC0\x50\x08\x10\x69\x28\x24\x4B\xA9\x43\x21\x92\x33\x0E\x85\x4A\x59\xDD\x31\x2E\xC5\x87\x6D\x6B\x46\x2A\x6D\xDC\x46\x45\x7A\x35\xD0\x84\x24\x6E\x3B\x9B\x38\xB9\x21\x95\x83\xA0\xE0\x55\xCB\x0E\x66\xC5\x80\x85\x09\x05\x03\x4E\x03\x66\xA4\xCC\x81\x04\x64\x1C\x30\x37\x58\xD9\x30\x37\x32\x63\xC0\x03\xA5\x90\xB0\x33\x71\xB2\x8A\x75\x04\xF2\x58\x5E\xC4\x86\x8C\xC7\x71\x3C\xEF\x88\xC9\xBE\x88\x6A\xF6\xAF\x1C\xCF\x4D\x93\x14\xC9\x2A\x66\x19\xC4\x2A\x76\x99\x2D\x78\xB3\x99\x4F\x13\xC9\x2A\x01\xD1\x00\x96\xB9\x01\x42\x51\xA7\x1B\x03\x92\x52\x1D\x18\xB0\xC4\xA9\x41\xC2\xB0\x07\xB1\x9C\xD4\x6C\xF2\xD4\x48\x56\x79\xC0\xFC\x98\x08\xAF\xA8\x41\x8B\x66\x12\xAC\xDA\x05\x56\x1B\xEB\x95\x41\x0D\x50\x6D\x0C\xB3\xD5\xAC\xEA\x40\x2A\xE0\x5B\x46\x9C\xA2\x51\x7A\xBD\xC6\x88\x25\xCA\x00\x22\x08\xB2\x64\x2E\xC2\x00\x48\x82\x01\x60\x28\x8F\x8A\x16\x1C\xB1\xB1\x55\xA2\xBA\x00\xAF\xA9\x9B\x6C\xD7\x00\xC5\x71\x92\x5A\x65\x1B\xD8\xA8\x65\x30\xB9\xAC\x0B\xBD\x46\xF4\x72\xB6\x33\xBF\x46\x06\x60\x08\x4B\x2B\x0A\x54\x58\x89\x05\xFA\x81\x27\x77\x04\x3C\xE3\x39\x70\x10\x47\x3C\xDC\x04\x22\x88\x3B\xA4\x15\x02\x8E\x83\xAC\x70\x5A\x1B\x17\x09\x07\x22\xD2\x94\x06\x36\x32\x0A\x16\xF0\x86\x40\x0B\x66\xFD\xF8\x09\xE0\x49\x4B\x6D\x06\xB8\xF8\x66\x87\x89\x04\x25\x79\xEC\xDB\x84\x69\x9E\x56\x4D\x24\xC4\xE5\x61\x03\x11\xC0\x61\x04\x0C\x79\xD9\xDC\x40\x42\x58\x13\x36\x63\x98\xBD\x2B\x8C\x03\x7C\x4A\xE0\x3C\x0A\xA0\x40\xBE\x25\xC2\x95\xD5\x6E\x08\xC5\x74\x02\x0C\xC2\xD5\x8D\x5C\x3E\xF4\xD9\xE4\x15\x51\x00\x03\x8B\x60\x48\x1D\x11\x5E\xEE\xC2\x81\x70\x20\xA4\x15\xD2\x66\xAB\x97\x88\x1E\x9C\xCA\x40\x24\xB3\x26\x5E\x6C\x91\xA5\xEC\xDD\x41\x1A\xF5\xF1\xBA\x4D\x6E\x2D\x65\x5A\xDE\x46\x85\x50\x40\xC5\x1A\x70\x9B\xBB\x48\x6F\xAD\x02\xEE\xC2\xA1\xA0\x81\x9A\x8E\x56\xF3\xB0\x65\x08\x9B\x36\x0D\x42\xB3\x4C\xA5\xC3\x20\x5F\x6B\xA0\x80\x15\x38\x10\x90\x5C\xA2\xE9\x22\xB7\xB6\xEC\xA5\xA3\xB4\x99\x59\x20\xA6\x19\x39\xA3\x88\xD0\xD6\xD9\x43\xC1\x02\xCD\x21\xA3\x40\x53\x4C\x96\xDF\x0A\xE7\xA0\x4C\xAC\x24\x24\xCB\x04\x23\x23\xC2\x67\x28\x60\xDC\x56\x7B\x49\xD8\x80\xCD\x24\x38\x34\xFA\x35\x02\xE4\x01\x90\xDC\xB0\x17\x50\x12\x82\xB7\x54\x69\x70\x7D\x61\xAE\x1D\x8D\xBD\xC3\x66\x63\xA7\x25\x87\x9A\x7A\x9B\x4C\x0D\x52\x9B\xA9\x65\x6C\xDD\xD4\x32\x1C\xF2\x3B\x19\x03\x18\x8C\x06\x01\x18\x0C\x8D\xBE\x7D\x30\x3C\xE3\xBB\x95\x06\xAC\x4A\xFF\x08\x65\xD4\xA0\x7D\x83\x6C\x50\x5F\xA7\x66\xB5\x66\x83\x6E\x34\x57\x83\x62\xB2\x57\xF9\x92\xA5\xC1\x60\x12\x8B\x16\x8B\xA5\x04\xEF\x42\xF0\x6E\x2E\x13\x9A\x51\x6C\x01\xC7\x24\x77\x89\x62\x12\x5C\xE6\xD2\x22\x79\x96\xEF\x5C\x24\x17\xD0\x8C\x92\x0B\x38\x26\xC9\x4B\x14\x93\xE4\x32\x97\x36\xC9\xD3\x96\x32\x4B\xC0\x6A\xFE\xB4\xD1\x31\x4A\x9D\x1A\xCA\x29\xA5\xBF\x26\xB3\x02\x87\x3C\x5B\x41\xE1\xD1\x61\xE8\x17\x66\xA5\x0E\x06\x78\x13\x7F\xB0\x62\xBA\x32\x6D\x0A\x88\xCA\x46\xC9\x77\xF8\x95\x41\x1D\x01\x30\x64\x03\x74\x60\x62\xED\xDF\x96\x1F\x1A\x4C\x3D\xD5\xE8\x54\x68\xC2\x1B\x0D\x68\x50\x7D\xAD\x13\x51\xDF\xFE\xA8\x30\xF3\x34\xB5\x9B\xAC\x41\x00\x54\x6E\x80\x56\x93\x71\x34\x6B\xFF\x36\x93\x35\x98\xBA\xC9\x74\x2A\x34\xE1\x8D\x06\x34\xC8\x64\x3A\x11\xD5\x64\x85\x69\xB2\xC2\xC3\x2D\xAF\xA6\x02\x28\x62\x05\x56\xC3\xCD\xF5\x17\x84\x02\x92\x10\x8F\xFB\x9B\x68\x88\x38\x20\x11\x56\xD5\x5A\x49\x34\x18\xB0\x22\xFA\xA2\x3D\x08\x07\x3B\xC3\xBB\xAF\x26\x14\x90\x44\x9C\xC3\x97\x2C\xEA\x08\x7A\xF7\x18\x1D\xC1\xEC\xAC\x40\xF9\x40\x5C\x34\x88\x07\x94\x89\x55\x7A\xF1\x13\x82\x96\x35\xB3\x02\x2E\x37\xCB\x9E\x69\x2D\xE2\x29\xF8\x36\x54\x4D\x24\x33\x6A\xF3\x38\xD5\x86\x29\x1C\x3A\x6D\xE5\x8F\x4D\x78\x4D\x74\x82\xBB\x3A\xB5\x31\x40\xA8\xF8\x1A\x7E\xB9\xB6\x0B\xF5\xD5\x41\x62\x47\xF9\x51\x85\x0B\xC5\x6F\x5D\x30\x41\x6B\x91\xEB\xFD\x03\x7B\x7B\x43\x4D\x06\x80\xEE\x60\x16\x14\x1A\x7D\x98\x01\x84\xA3\x08\x07\xA0\x68\x4B\x7D\x20\xA6\x74\xA0\xC4\x88\x51\x9D\x93\x83\xDC\x50\x2B\xB9\x00\x07\x74\x32\x81\x7E\x60\xE3\x1B\x7E\xC8\xEC\x88\x1D\x68\x82\xC3\xD3\x46\xD2\x18\x95\x62\x0F\xB3\xF2\x46\xB2\xA6\x60\x33\xF7\x00\x63\x4E\x44\x37\xB9\x84\x80\x52\x3F\x4A\x6A\xA4\xD5\x19\x1E\x46\x68\xE9\xDF\x36\x18\x20\x9E\xD9\xC2\xF5\xAB\x86\x46\xF3\xC2\x04\x8D\xB6\x85\xD1\x61\xC3\xEA\x27\xFF\x85\x7C\x5E\x85\xE9\x4E\xCE\x2A\x0D\x48\x19\x8A\x90\x1A\xC2\xB6\x01\xA8\x31\x1F\xCA\x5B\x84\x8A\xDB\x86\xF0\x09\x0C\x25\x74\xD4\x41\x55\x41\xF5\x70\xAA\x80\x76\xAA\xC6\x21\x54\x11\xC5\xC1\xB3\x13\x11\x86\x4D\x45\xD4\x66\x15\x03\x09\x79\x28\xEB\xF3\xDC\x8A\xDB\x36\x1B\xC6\x9A\xC3\x43\x7B\xC9\x10\x54\x7F\xC1\x02\xEE\xCF\x77\xAF\x21\xB8\xEA\x09\x02\x44\x75\x05\x95\xA8\xE2\x0B\x2A\xD8\x18\x57\x0D\xA2\x1E\x51\x06\x22\x0E\x6C\xB4\x68\x33\x21\x56\x83\x23\xD9\xCE\x52\x1B\x68\x08\x80\x3F\x96\x38\x9E\x27\xD6\x14\x24\xF9\x6E\xA0\x53\x8E\xB7\x86\xDE\x0E\x14\x44\xB4\x21\x28\xB6\x87\x51\x2B\xEB\x3B\x10\x72\x62\x56\x1F\xBC\x68\x47\xE5\xA7\x9C\x20\x3C\xEE\x8F\x30\xAC\x39\xAD\x0F\xBF\x62\xDC\xBC\x2F\x61\x80\x18\xF7\xEA\xE0\x28\x14\x6F\x5A\x17\xA7\x2F\x67\xC4\x26\x49\x3A\x23\xCB\x91\x6E\xC2\x35\xC5\xBD\x84\x6F\x9C\x4E\x41\xAC\x76\x59\xA1\x57\xEE\xCB\xBF\x4D\xE5\xB1\x0C\x52\xEB\xEA\x02\xCA\x47\x9A\x95\xD2\x55\xBC\x88\x1D\xA5\x60\xA8\xB0\x5A\x20\x5A\x8C\x00\x50\x30\x40\x24\x3C\x39\x3A\xCC\x24\xDA\x79\x48\x71\x61\xC1\x6B\x82\x42\x42\x12\x23\x42\x02\x28\xD3\x89\x38\x56\xED\xE3\xFE\xCC\xEA\x64\x1D\x3B\x65\x46\xA7\x05\xE7\x18\x4A\x50\x9C\x85\x34\xBC\xC7\x9E\x1F\x25\x59\x4E\x0C\xB3\x20\x80\xC7\xCD\x60\x02\x5F\xD8\xC1\xBA\xDF\x58\x31\x99\x7D\x1D\xF0\x94\x33\xAB\xAD\xF8\x42\xA4\x39\x63\xEF\x6B\xDF\xE8\x6B\x1A\x55\x52\xD5\xE4\x1A\x4C\x38\x7B\x1F\x69\x2F\x6D\x97\x09\x35\x0B\xFF\xD2\xC6\x0E\xBA\x29\x05\xCE\xBA\xC2\xF6\x9C\x1B\x82\x76\x40\xC2\xCC\xA1\x79\xDC\xD3\x74\xAC\x0E\x74\x39\x40\x95\xCA\x54\x93\x52\xCF\xB0\xE2\x89\x31\xD8\xAB\x51\x10\xA8\xE7\xF3\xE5\xA4\x2F\x21\xD8\xBB\x9B\xE6\x01\x80\x87\x92\x2D\xF8\x81\x40\xBB\x20\x30\x8E\x52\xF6\xC1\x48\x7A\xDD\x06\xE3\xB5\x48\x55\x83\xA3\x24\xF9\x96\xA7\x80\xA4\x1A\xA0\x9E\xC1\x04\x88\x50\xCA\x8A\xAD\xCD\xFB\xBC\x28\x4D\x0B\x6F\xAA\x5F\xF5\x6D\x6E\x3A\x91\xE0\x42\x67\xFE\xE7\x45\x9C\x04\xF8\xEA\x8A\xE0\xCB\xAF\x97\x2F\x32\x1A\x5C\x5D\x79\xDE\x2F\x57\x57\xB7\x28\x0B\xFD\xD2\x45\x5F\xFA\x07\x44\xBE\x5E\xFA\x5F\x2F\x9F\x37\x08\x45\x5B\x11\xA4\x21\xCD\x18\xFC\x95\x8A\x50\xFA\x75\x42\x2A\xF0\xF3\xE2\x3F\x3F\x89\x33\xFA\xE4\xEB\xE5\x6F\xCF\x26\xE6\x75\x75\x85\xB2\x2C\xDC\xC7\x5F\x2F\x5F\x70\x92\x9C\xF5\xD3\xAF\x97\xBF\x2D\xA6\xF2\x42\x7C\x1F\xF1\x10\xB3\x5C\xAF\x0F\x71\xD1\x35\x8F\x8B\xE1\xC7\xC1\xD7\xCB\xE7\x51\x12\xEF\x97\xB4\x43\x92\x62\x52\x80\xBF\x5E\x3E\x0F\x70\x84\x29\xFE\x7A\xF9\xE2\x3E\x09\x83\x45\x9D\x71\xB1\xC1\x59\x4E\x65\x9C\xF9\x28\xC5\x3F\x49\xF0\x09\x4E\x17\xE3\xEF\x5F\x2F\x5F\xAC\x28\x20\xCA\x1D\xE9\xA5\xD8\xF3\x69\x73\x09\x6F\x98\x3E\xEA\xD2\x3C\x3B\x78\xB7\xC8\xFF\x56\x07\xDE\x6F\x17\xFB\x28\xB9\xE5\x05\x75\x84\x58\xF2\x71\x49\x97\x32\xAF\x7B\xEC\x17\x8C\x26\x37\x95\x09\x65\xFA\x31\x52\xC7\x4A\xEE\x33\x09\xD7\xA7\xAE\x71\xBA\x0D\xE0\x8A\x07\x30\xC2\x27\x4C\xAE\xAE\xD8\x3F\xCE\xA5\xC8\x36\xA2\x2B\x1E\x51\x36\x38\xEC\xD9\xE9\xEB\xE5\x0B\x36\xAE\xE5\x88\x14\x13\xD6\xD3\xA7\xDB\xF8\xAD\x7C\xFC\x26\x9F\x64\xCB\xB5\x01\x28\x59\x1B\x07\xDE\x29\x2F\x54\x08\x9B\x7F\xAD\xD9\xBF\xCA\x8C\xFF\xA3\x4A\xF9\x5B\x42\x58\xF9\x80\x59\x1F\xC0\xB7\x91\x5B\xF1\xC8\x2D\xB8\x4A\xB8\xF9\xC5\x8A\xFD\x22\x31\x9C\x74\xDE\x06\x6D\xC5\x83\xB6\xC8\xFA\xDB\xE6\x11\x2B\xF6\x88\xE9\xD3\x7B\xB7\x1D\x82\xCD\x59\x36\x67\xD9\x6A\x81\x33\xF3\x8B\xA5\xB6\x55\x37\xA7\x58\xBD\x53\xFC\x28\xFE\x29\x9F\xD0\x81\xA7\x76\xFB\x83\x61\xD9\x21\xC6\x27\xFA\xF5\xF2\x05\xFB\x5E\x4C\x99\x87\xD8\x19\x1E\x13\xD8\x65\x41\xDF\xD4\xB7\xDC\x18\x73\x20\xDD\x45\xDC\x89\xD3\x66\x25\xD5\x7F\xCD\x1C\x7F\x26\x85\xF9\xF6\x9E\x09\x5A\xB6\xEF\x30\xF5\x0F\x5E\x48\xF1\x51\x76\x0F\xB5\xDD\xF8\xA4\x61\xC6\x77\x1E\x7E\xAD\xAF\x3C\xFC\x36\xD2\xC6\xE1\xD7\x3A\x49\xED\xEC\x8E\xD8\x16\x85\xA5\x76\xCA\xCD\x17\xC6\x54\x9A\x0F\x4D\x7D\x25\xD3\x6B\x2C\xDD\x6C\x29\x77\xE8\x68\x4C\xB1\x33\x64\x4D\x03\xF1\x16\x73\x8A\xBD\x3A\x9B\x47\xEC\x0C\xD8\x07\x26\xE1\x66\x2A\x63\xDF\x8E\x56\x33\xD0\x81\x0C\xD8\xCE\xB2\xC5\x96\x16\x7D\xCB\x93\x82\xAE\xE8\xEC\x68\x9D\x0B\x32\xB7\xBA\x36\x23\x00\x73\x84\x8E\xE3\x70\x76\x87\x9F\x5D\xFC\x7A\xF9\x82\x6D\x0F\x75\x3B\x7B\xA1\xF7\x95\xCE\x46\x58\x49\xBB\xCB\xC3\x0D\xD0\xBE\xAD\xE5\xBE\x19\x6D\x27\xF3\x58\x36\x60\xEC\x5A\xEA\xEE\xD3\xCD\xC2\x86\xA5\xE8\x85\x4D\x2B\x92\x9D\x79\x42\x67\x9A\x13\xEF\x88\xC2\x98\x71\x46\x14\x97\xFA\x57\xB6\xA8\xD3\xAF\x3A\xAD\x3B\xF5\x7D\xB4\x95\x90\x8B\xF2\x2E\xA1\xBD\x0A\x23\xAE\xDC\x01\x0D\x39\x76\x1D\xB6\x33\xEF\x66\x74\x74\x11\x3D\x37\x95\x04\xAA\x6A\xA6\x57\x74\xDA\x3B\x2F\x6F\x9D\x85\xE5\x5B\x38\x7D\xD8\xB5\x37\x15\xA4\x4E\x9D\x8B\xA2\xE6\x98\xAE\xC4\xCC\xD3\x17\x0F\x8E\x06\x75\x4E\xC8\x0B\x1B\x6C\xF9\x8C\x6C\x37\x40\x4B\x4A\x76\x1E\x06\x3D\xE9\x15\x4E\xCF\x8A\xB2\x23\xA2\x3E\x78\xA1\x24\x08\xAD\xDF\xE0\x01\xA1\xD5\x8B\x3F\x12\xB0\xDD\x17\x94\x2E\xC2\x15\x65\x30\x21\x27\x6D\xB4\x8F\x38\xB6\x23\x65\x3A\xD2\x63\xF6\x40\xA3\xAA\x1E\x7B\x8F\x8C\x8D\xB5\xB2\x96\x5A\x42\x0A\x3F\xD4\x5B\x1B\xFF\x79\xC9\xED\xF9\x0D\x3F\xCC\xB2\x2C\xDA\xAC\x04\x33\x87\x78\xF5\xF4\x95\xD4\xA3\x14\x8F\xA2\xDB\x68\x9D\x52\x8B\x32\xE5\x71\x42\x02\x4C\x70\xE0\x1D\x51\xCA\x24\xC4\xC4\x2C\xD8\x1A\x75\x28\x24\x6A\xC4\xD0\xC0\xF8\x2E\x47\x91\x47\x13\x0D\x85\x92\x1C\xB7\x1A\x84\xF5\x3E\x2F\x7B\x18\x15\x6E\x31\x19\x68\x0F\x68\x83\x63\x1D\x4A\x3F\x2F\x13\xDA\x2E\x8C\x03\x49\x97\x17\x82\x34\x2E\xD9\xE7\x91\x95\xA0\x8A\x9E\xDB\xD6\xDA\x7A\xB7\xD6\xD4\xB1\x62\x8E\x5B\x2E\x80\x19\xDD\x35\x45\xA1\x25\x41\x4F\x66\x21\x95\xF1\x3C\x56\x2A\xB9\xEC\x50\x94\x61\x29\xB4\x67\x63\xDF\x36\x12\x8F\xAB\x8A\x29\x7E\xDD\x26\x49\xB4\x15\x34\x5B\x41\xB3\x1A\x7B\xFC\x64\x05\x8D\xE7\xE1\x23\xFF\x1C\x47\x1E\x87\x77\x39\x13\xC3\x43\x64\x9F\x99\xA5\x5A\x68\x4E\xE0\xFF\x81\xF5\xD6\x2A\xE4\x7B\xE2\x90\xBF\x97\x98\x49\x7F\x14\x9C\xE1\x79\x3E\x91\x2F\xF7\x55\x20\x04\xEF\xF1\x89\xFF\x56\xC1\x7C\x63\xF2\x71\xCD\x46\xDB\x14\xB4\x4D\x41\xEB\xB0\xC7\x4F\x36\x05\x39\x3E\x53\x3F\xDB\x9C\xBE\x1E\xE4\xE6\x81\x9F\x27\x76\x93\xCD\xEA\x9D\x8D\x2D\x55\x2F\x3F\xE2\x3F\xBD\xD7\x6E\xA9\xFA\xE7\x4B\xD5\x80\x81\x1C\x93\x57\x21\x9A\x3D\xA9\x09\x2B\x58\x86\x4D\x31\x19\xA3\x7D\xAF\xCE\x4E\x31\xC3\x62\x22\xDD\x96\xC6\xE6\x5D\x1A\x7B\xD4\xBB\x95\x9B\x5B\x2D\xE6\x56\x2E\x79\x61\x1B\x9E\xC5\x86\x47\x5D\x6A\xD8\xD6\xC1\xD7\xA2\xC0\x56\xD9\x6E\x95\xED\xA3\xA9\x6C\xB7\x75\xF0\xB1\xE4\x7B\xB2\x2D\xD6\x8C\xB6\x58\xB3\x4D\x76\x6B\x51\x60\xF3\xE7\x6D\xB2\xDB\x26\xBB\x6D\xB2\x6B\x99\xEC\xA6\x5C\xEE\xDA\xA6\x85\xF5\x29\xB0\x4D\x0B\xDB\xB4\xB0\x4D\x0B\xDB\xB4\x00\x4E\x0B\x3D\x56\x16\xB7\xA4\xBE\x02\x05\xB6\xA4\xBE\x25\xF5\x2D\xA9\x6F\x49\xBD\x75\x61\x2B\xC6\x27\xEA\xA5\x24\x3C\x82\x2F\x25\x6F\xC9\x7D\x8D\x0A\x6C\xC9\x7D\x4B\xEE\x5B\x72\xDF\x92\x7B\x6B\x72\xDF\xF2\xD8\x96\xC7\xB6\x3C\x36\x57\x1E\x23\xB8\x90\x67\xB5\xA7\x6F\x84\x97\x68\xEC\x77\x75\x58\x87\x70\xAC\xA5\xE0\xEA\x5E\x9E\x2D\x5D\x6D\xE9\x6A\x15\xF6\xD8\xD2\x55\x93\xAE\x3A\x2E\xFB\x6D\x91\xBB\x45\xEE\x16\xB9\xAB\x8B\x5C\xDB\xCC\xBE\x85\xEC\x16\xB2\x5B\xC8\xAE\x23\x64\xB7\x4A\x78\x0B\xCE\x35\xD9\x63\x0B\x4E\x21\x38\xF9\xBB\x1B\xCD\x5D\x12\x61\x56\xC8\x1D\x06\xD0\x85\x9C\x00\xC6\x6D\x92\x44\x18\xC5\x30\xD8\xED\xE2\x4D\xA5\x9B\x70\xF9\xA6\x02\x21\xF8\xF2\xEA\xEA\xD3\x3B\xFE\xBF\xAF\x97\x2F\xD8\xEF\xCF\xCC\x74\x1F\x43\xEC\x63\xE5\xF1\xBC\x41\xFF\x90\xD2\x30\x89\x33\xED\x56\x34\x23\xFD\x1F\x9C\x81\x01\xAF\xF5\x22\x50\x05\x5F\x35\xB1\xF1\xC2\x53\x19\xDC\xC5\x7A\xDA\xBD\xA5\x62\xB3\x9F\x1C\xD3\x30\x02\xAE\x7A\xD3\x29\x34\x26\xF8\x88\x08\x0D\x51\xF4\xAF\xA2\xF9\xBD\xBB\xB1\x8D\x63\x70\x4D\xF6\x02\x67\xE9\xB6\x39\xF6\x4D\x09\x48\x96\x9C\x5D\xD6\x5F\xD1\x00\x05\x86\x7D\xD0\xF1\xC2\x56\x11\x99\x0F\x92\xD9\x54\xA6\xB7\x57\x6D\x1D\x1C\x5E\x66\xED\xC2\x2F\xB3\xF3\x2B\xBF\x9D\x64\xC1\x28\xFD\xC3\x4A\x43\xC9\x61\x03\xB6\x3F\x27\x4F\x72\xA2\x83\x3D\xD5\xF6\x3E\xF5\x49\xA8\xBD\xDE\x58\x44\xE4\xD6\x62\x63\x16\xA9\x56\x6C\x16\x36\x19\x2F\x59\x27\xCC\xC6\xBB\x43\xC5\xF5\xB3\x3B\xC5\xFC\x83\xB3\x94\x43\x0E\x28\x3B\x67\x1F\xA3\xAA\xE6\x64\x97\x50\xCC\x62\xA1\x59\xAF\x00\x70\x9B\xED\xF5\x62\xC3\xD2\x6F\xD4\xA2\xD6\x89\x8F\x50\xDC\x5A\xF0\x79\x3D\x04\xD5\x60\x86\x4A\xC8\x80\xDA\x5E\x03\x39\xF2\xC8\x4C\x3C\x26\x76\x80\x45\xEF\x9C\x68\x51\x78\xF4\x23\x6B\x0B\xD7\x6C\xFE\xB0\xF3\x6A\x5B\xF9\xB6\x46\xB3\x6C\xE5\xDB\x56\xBE\xAD\x60\x70\xB6\xF2\xCD\xBD\x7C\x33\x9F\x59\x9C\xD9\x6F\x95\x09\x6E\x46\x33\xCC\x38\xCD\x2F\xAF\xF2\x13\x73\xB1\xD1\x6D\xBF\xF5\x7C\x4B\x8F\xAD\xDE\x78\x2C\x53\xCB\x56\x6F\x6C\xF5\xC6\x0A\x06\x67\xAB\x37\x7E\xF6\xE5\x22\xFD\xE0\xC3\xF6\xBC\xBE\xBC\xF4\xDB\xFC\xB9\xCD\x9F\xDB\xFC\xB9\xCD\x9F\x8F\x69\xFE\xDC\x9E\xD7\x7F\xC2\xE7\xF5\xAD\xBC\xD8\xCA\x8B\x33\x9C\x49\xB6\xF2\x62\x2B\x2F\x56\x30\x38\x5B\x79\xB1\x95\x17\x5B\x79\x61\x2B\x2F\x16\xBE\x1A\x6D\xAB\x3D\xB6\xDA\x63\xAB\x3D\xB6\xDA\x63\xAB\x3D\xB6\xDA\x63\xAB\x3D\x7E\xB2\xDA\x43\x74\xC5\x05\x6F\xF0\xDB\x6A\x90\xAD\x06\xD9\x6A\x90\xAD\x06\xD9\x6A\x90\xAD\x06\xD9\x6A\x90\x9F\xB8\x06\xD9\x66\xDC\x33\x4E\xAE\xDB\x8C\xBB\xCD\xB8\x2B\x18\x9C\x6D\xC6\xED\x32\xE3\xDA\x2E\x1B\x6D\xEE\x6E\xD8\x85\x71\xE0\xA1\x28\x2A\x25\xF3\xB2\xFC\xB6\xE5\x76\x0D\x6B\x8F\x38\x3F\xDE\xE2\x42\x8D\x07\x2F\x8C\xA9\x5B\xAF\x2E\xD7\x74\x18\xC9\x68\x37\x77\x18\x31\x9D\x2E\xF3\xB0\x2A\x89\x08\x41\x4E\xAC\x8E\xF8\x98\x61\x27\x2B\x94\x57\x4F\x38\x60\x36\xAF\xA1\xB2\x9B\x45\xDA\x5F\x78\xD5\xEC\x69\x6C\x14\xEE\x1A\x89\xFD\x43\x42\x84\x56\x81\xBC\x74\xE7\x48\xFD\xDD\x2E\x07\xC9\x77\x79\x14\x91\x3C\x76\xF2\xA5\x62\x68\x68\xE2\x8A\x5D\x0F\x8A\xF7\x3D\xA4\x07\xCF\x47\xA9\x6B\xAF\x23\x0B\x2A\x57\xEC\xB2\xAD\x0C\x44\x1C\x74\x93\x0D\xA5\x29\x8E\x9D\xFA\xB4\x5C\xDC\xE2\x12\x4B\xB6\xFB\x5A\x8C\xFD\xD5\xCF\x50\x6F\x65\xDB\x59\xCF\xD0\x5B\xD9\xB6\x95\x6D\x2B\x18\x9C\xAD\x6C\x1B\xA9\x6C\x9B\xEA\xD2\xE5\x2D\x7B\xAF\xD1\x2C\x5B\xF6\xDE\xB2\xF7\x0A\x06\x67\xCB\xDE\x13\x64\xEF\x51\x2E\xDE\xDE\xD2\xF6\x1A\xCD\xB2\xA5\xED\x2D\x6D\xAF\x60\x70\xB6\xB4\x3D\x56\xDA\xEE\x7F\xF9\xFA\x96\xA0\xD7\x68\x96\x2D\x41\x6F\x09\x7A\x05\x83\xB3\x25\xE8\xB1\x12\x74\xB9\x6A\x1D\xA7\x24\x8C\xE9\xCE\x7B\xE6\xDD\x57\x7F\x5F\x34\x7F\x3E\xF3\x62\x9A\x20\x6F\x97\x90\x23\xA2\x52\xFB\x8E\x26\x48\x6A\xC0\xAC\x41\x40\xAF\xF6\x71\x18\xEA\x33\x2F\xC9\xA9\x47\xF0\x7D\xF5\x93\xA3\xB3\xBF\x2B\x3C\xCC\x00\x9C\x30\x16\x00\xBB\x92\x05\xDB\x13\x60\x52\xEF\xBE\x93\x90\x62\xB1\x21\xCD\xA9\x2F\xFC\x66\xFF\x6F\x7E\x7B\x17\xF5\x9F\x62\x23\xDF\xC6\x10\x77\x34\x76\x04\x73\xB2\xC5\x1F\x17\xBE\x06\x2F\x5B\x4A\xC1\x08\x6E\x10\xCA\xBF\x2F\x32\x4A\x68\x12\x3D\x0B\xB3\x2C\x45\x3E\x6E\x7E\xA2\x28\x3D\xA0\xE6\x67\x9E\xA6\x98\x5C\x1C\xF1\xF1\x98\xDC\x63\x91\x83\xD8\xC4\x84\xF0\x8E\x69\xE0\xDD\xA2\x0C\xA3\x20\x10\x34\x64\xCD\xD9\x21\x21\x14\x68\x2F\xD0\xC3\xD8\x27\x50\x7B\x96\xDF\x82\x64\x8E\x79\x04\xA1\x5B\x9A\xAB\x71\xAB\x49\x04\xE1\x3D\x84\x1B\x84\xF7\xC7\x04\x94\xB1\x84\x14\x3F\xB9\xF6\x2A\x64\x57\x79\xA4\xDA\xCE\x1C\x00\x40\xCF\xA9\x4E\x44\x35\xC0\x21\xDC\x51\x50\xA9\x12\x52\xFC\xCC\xC3\x98\x7A\x7F\x61\x92\x28\x50\xCD\xA4\x95\xED\x3C\xD1\xFD\x7C\x42\x7F\x15\xBD\x33\xA6\x5E\x10\xEE\x2E\x85\xA6\x8C\x06\xDE\x2E\xA6\x40\x0B\xA3\x1A\xC6\x21\x2D\x7E\x79\x29\x22\xE8\x98\xD5\xC0\x8A\x12\x6B\x08\xE3\x7B\x80\x46\xD5\x6A\xA4\x53\x21\xD4\xB4\x76\x49\x4E\xBC\x8C\xE2\x54\xA1\x26\xB7\x37\xF4\xBE\xFF\xCA\x8A\x38\x08\xBE\xC7\xF4\x1B\x26\x31\x8E\x14\x60\x16\x9E\xEA\x1F\x17\x4C\x00\x03\x4F\x1D\x56\xB4\xB4\x74\x57\x58\x1B\x10\x24\xD9\x8B\x86\xBB\x0C\xD3\x32\x1B\xB3\x31\x2F\x7E\xEE\x22\xB4\xCF\x8C\xD0\x18\xEF\x11\x0D\xEF\x8D\xDD\xB3\x14\xFB\x21\x8A\x34\x30\x0F\x7B\xB0\xD7\x1D\xC1\x59\xF8\x97\x91\x22\x26\x24\x21\x26\x60\x10\xEE\x43\x0A\x4B\x7B\xB7\x0B\x63\x14\x55\x84\x69\xE2\x65\x7E\xC8\xCD\x50\xFC\xCD\xF0\x2E\xE4\x9F\xAC\x5B\x91\xB6\xB8\x0A\x20\x30\x46\x31\x04\x10\x62\x57\xEB\x51\x59\x0C\x84\xDE\x19\x08\x1E\xB3\xEF\x09\x09\x20\x48\xD1\xEE\x95\x7A\x2B\x60\x06\x6A\xD1\x2E\x8C\x77\x85\x1B\x68\xE2\x08\x01\xDD\x34\x2A\x91\xAE\xD2\x6A\xB2\x83\x00\xE1\x29\x91\x8D\xCF\xF7\x90\xFA\x87\x02\x16\x3C\xC4\x02\x2D\x1D\x56\x79\x17\xB3\x86\x19\x9C\x26\x59\x58\x3B\x9F\x82\xC2\x8F\x71\x98\xFB\x06\x0F\x31\x3A\x86\xBE\x17\x20\x8A\x18\x5A\x39\x5D\xC1\xFC\x21\xA0\xC4\x9D\x8F\x8F\xA2\x14\xB7\x88\xD2\x58\xE7\x51\xD1\x34\xA5\xCF\x2B\xA8\x55\xAB\xA6\x85\x06\x15\xE4\xBB\x90\x42\x41\xA1\x58\xF0\xF5\x13\xBC\xDB\x19\xDB\x05\x6A\x9C\xCB\x11\x9D\xA0\x1E\x4D\xBB\x41\x3A\x09\xAE\xCA\x57\xF8\x1D\xC5\x7B\xAC\x4D\x1E\x0C\xA0\xFA\x46\x9D\x46\xCC\x00\x55\xEC\x2A\x51\xE8\x3D\x78\xB3\xA6\x66\xC6\xD2\x45\x6D\x50\x39\x5B\xD4\xBF\x78\xC2\xD8\x85\x27\xAF\xF2\x0C\x05\xE4\x1F\xB0\xFF\xCD\xC3\xA7\x54\x01\x72\x1E\x6C\xBE\xF4\xC2\x98\x1D\x61\x86\xBA\x57\x65\x8A\x02\x53\x8C\x54\xF2\x37\x35\x8B\xBA\x69\x00\x79\xA4\x4C\xFD\x1A\x60\xAD\x90\xCA\xAD\x01\x68\xFC\x64\x50\xED\x09\x10\x50\x98\x22\x14\xA8\x91\x15\x37\xA3\x86\x0F\x59\x58\x41\x61\x3F\x51\x9A\x46\x0F\x1E\x49\xF2\x38\xF0\xF0\xC9\xC7\x59\x06\x51\xAA\x03\xE2\x02\xA0\xAB\x86\x82\x02\xAD\x26\x5F\x8F\xC4\x01\x88\xA0\x3B\x1F\x40\xA0\x29\xB0\x8C\x34\x80\xA8\x33\x73\xE2\x43\x9A\x61\x4A\x49\x98\x46\x92\x4F\x35\x8D\xBA\x64\x98\x7A\xE1\xAF\x97\x9A\xC6\x65\xB3\x11\x5F\x89\x29\x6E\xE2\xBB\xC2\x28\xB9\x1C\xE0\x0A\xA0\x49\x02\x2A\xA4\xA4\x28\x5A\x44\xAB\x3F\xA5\x66\x61\x0A\xA8\xDB\x80\xA1\xA9\xB0\xF5\x84\x20\x74\x51\x07\x63\x5F\xAA\x29\xCB\xC8\x61\x7E\x92\x3E\xA8\x32\xB1\x36\xC3\x90\xD5\x30\x35\x49\x96\x10\x51\x0D\xB5\x50\x17\x5B\xEB\x2E\x5A\xAB\xC9\x4F\x22\x98\x6D\x09\x53\x1E\x02\x2E\x0C\xD1\xA3\x24\x2E\x13\x96\x32\xE2\x00\x86\x21\x3B\x19\x30\x75\x63\x1D\xCB\x00\xF7\x8F\xE9\x45\xF5\x87\x96\xB4\x04\x24\x2F\x43\x47\xEC\xA1\xE0\xFF\x8A\x6C\x0D\x35\x36\x9A\x15\x24\x85\xF0\x82\xE3\x1F\x1C\x00\xD0\xE5\x4C\x1E\x67\x71\x38\xA1\x8F\x31\xEA\x61\xBB\xA0\x58\x2D\x4B\x58\x93\xC6\xBD\x6A\x04\xC8\x2B\xF8\x3A\x40\xA9\xBC\x8B\x66\x6E\x50\x94\x6A\x88\xBC\x7E\xAF\xC0\xDA\x00\xA1\xD4\xE4\x08\x15\x48\x9F\xA2\x50\xFA\xAC\xC9\x61\x9C\xA1\x36\xF3\xDF\x41\x93\x7E\x62\xB4\x68\x62\xB0\x66\xD5\x43\x56\x38\x21\x8A\xBE\x0D\x9A\xA0\xEE\x1D\x0A\x02\x60\xAD\xA1\x6A\xD6\xE6\x36\x11\x20\xCD\xAF\x22\x00\x30\x14\xD0\x4F\x07\x8A\x8B\x28\x06\x48\x96\xDF\x0A\xEE\x15\x83\xFE\x1D\x03\x15\x91\x88\x5E\xEA\xA3\x75\xA8\x2D\xA5\x21\x2B\xF5\x51\x09\x87\x22\x2E\xD6\x07\xF4\xA4\x8F\xFB\x09\x1A\xF8\x93\x79\xE4\x4F\xA6\xA1\x3F\xC1\x63\x7F\xD2\x06\xFF\xA4\x8E\x3E\xB7\xEE\x6D\xA6\x5A\xB0\x68\xD2\xCD\x57\x23\x1E\xC3\x38\xCF\x94\xB6\x34\xAA\x9A\x18\x54\xA5\xC8\x1B\x75\x9A\xBC\x5D\xAD\xE0\x1B\x0A\x72\xB5\x5B\x30\x51\x29\xB3\x36\x9D\x30\x6B\x56\xE8\x36\xDD\x65\xB2\x28\x08\x34\x0B\x04\x01\x64\x81\x40\x4F\x59\x01\x94\xB2\xAA\xDE\xA5\xEB\x2A\x6C\x64\xEE\x75\x88\x09\x1E\x5C\x79\xBB\xEC\xBE\x52\x88\x4A\x7D\xE5\x74\x5D\xF5\x96\xB8\xD7\x6C\x64\xEE\x41\x78\x5F\x22\x07\xE1\xFD\x45\xFD\x97\x20\x91\x27\xE3\x55\x22\x35\xCD\x6A\xDA\xA8\xBA\xCB\xA9\x81\x11\x68\x92\xA1\xC2\x49\x4A\x2F\x55\xAB\x92\xA5\xEA\x66\x20\xB3\x28\x3C\x15\x88\xB4\xEE\xA9\xB6\x37\xCB\x8F\x10\xC4\x8B\x35\xD8\x11\x78\x3C\x10\x01\xE2\x9A\x47\xDD\x4E\x09\x0A\x23\x56\xBD\xAB\x10\xF0\xD9\xA0\x81\xD6\xC3\xA5\x89\xC4\xE1\x31\xFE\x6E\x00\xB1\xC7\x9B\x98\xE2\x13\x05\x10\x80\x35\x55\xA9\x33\x60\x64\x0D\x47\x33\xB6\x84\x01\x18\x5D\x87\xDF\x1D\xF3\xC8\xC3\x27\xE4\x9B\xE4\x28\xAD\x63\xEB\x6F\xEA\xD9\x66\xB9\x22\x34\xAC\x8A\x19\x99\xF2\xA2\x8F\x26\x1E\x5B\x2C\x20\x08\x12\x41\x5B\x0F\x6F\x20\xCA\x8A\xB8\xD0\x45\x5A\x13\x97\x99\xD6\xD5\xA3\xAE\x48\x3D\x53\xEA\x20\x39\x82\x81\x81\xBF\xB3\x63\x43\x11\x2B\x0F\x4D\xA4\x03\x0C\x01\x6A\x74\x9B\x3B\xB3\xC7\x48\x20\x28\x52\x25\xA0\x1E\xAC\xB5\x48\x7A\x58\x82\x83\x7D\xC7\x76\x4B\x14\x5B\x14\x6D\x40\xEE\x03\xF6\x5B\xEA\xEE\x3A\x51\xF6\x57\xE1\xED\x97\x5E\x84\x2F\x45\x74\x61\x6F\xA6\xC1\xAC\x36\x2B\xD4\xF6\x72\x47\x47\x61\x07\x59\x5C\x11\x05\x80\xF8\xA0\x8C\xDF\x00\xAE\xBB\x18\x92\xA5\xC0\x64\xEB\xFC\x8A\x30\x52\xDA\xFE\x06\x98\xE9\x1B\x24\x01\x6B\xE4\x34\xBF\x27\xE4\x5B\xF3\x74\x5E\x02\x10\x41\x34\xCB\x6F\x91\x47\xB0\xD8\x43\x8A\x33\xDE\x22\xC4\x57\x89\x52\xC7\x15\xFB\x2D\x2E\x4D\xC5\xD0\x6E\x5A\xD9\x2A\x91\xAE\xDA\x04\xE2\x35\x5A\x4D\xBE\x6C\xD1\x49\xA9\xCA\x56\xCD\xFC\x19\x5A\xA2\x16\x53\x0F\xE5\x34\xF1\x93\xF8\x3E\x89\x72\x95\x57\x4C\x3D\x18\x22\xBA\x51\xD9\xE4\x13\xFA\xEB\x85\x34\x52\xD0\x40\xB0\x76\x68\x30\xF8\x7E\x8C\x61\x40\xF8\xBE\x94\x6A\x7B\xDE\xAA\xD8\xBF\x44\x95\xC6\x80\xB5\x41\x82\xC8\xE3\xCC\xBC\x4B\x76\x3A\x96\xF1\xE5\x50\x95\x13\xBD\x9E\x9D\xB5\x80\xD6\x31\x80\xF0\xD6\x91\xD4\x60\x07\xC8\xA8\xE9\xD2\x20\x8B\x38\xA9\x99\xA4\x01\xD6\x03\x39\x3A\x3E\xAA\xF5\x62\xD1\xA4\xD7\x8B\x45\xAB\xB6\x6E\x7F\xD4\xEB\xC5\xA2\x51\xC1\x11\x93\xA4\xC2\x55\x31\x76\x3D\x7F\xAB\xF3\xB1\x00\xB1\xF7\x03\x52\xA8\xD0\x4F\x96\x54\xB0\x94\x71\x2A\xD6\x9A\x4F\x76\x54\x4A\xF2\xD8\xB7\xA3\xEC\xA2\xA4\x7A\x98\x32\xA1\xF8\x38\x8C\xAC\x18\x45\xF8\x84\x71\x91\xBE\xD5\xA5\xFF\x06\x10\x25\xFB\x5F\x9E\x73\x7A\xF8\x94\x26\x84\x7A\xF9\x2F\xFF\x50\xC7\x50\x80\xC8\xDD\x41\xB0\x60\x3D\x05\x22\x3D\x02\x83\x30\x02\xC2\xA4\x11\x11\x60\xF2\xEC\xCD\xA0\xE1\xD1\xA4\x84\x00\x11\x94\x16\x5A\x35\x4E\x0A\xCC\x0C\x01\x1E\x9D\x61\xBA\x20\x5C\x9A\x7C\x3D\x9D\x43\x03\x2F\xAA\x2E\x88\x3A\x50\x77\x2A\x0B\x7E\x52\x4A\xD5\x32\xAC\x0C\xAD\x53\xAC\xD2\x5C\xE6\x58\x15\x99\x25\x59\xB9\xB1\x62\x2A\xE6\x55\x49\x1A\x09\x20\x17\x19\x32\xAC\x9A\x62\x94\x1E\xCA\xCC\x6C\xA0\x87\x82\x80\x26\x00\x45\x36\xFF\x19\xFA\x64\xF9\xED\x8E\x24\xC7\x0B\x7D\x52\x30\x2A\xD0\x1C\x22\x80\x95\x68\xE0\x55\x4D\xA3\x83\x41\xA9\x2A\x88\x6E\x01\xD3\xBC\x65\x13\x0C\xB2\x86\x8C\x51\xEB\x7E\xD4\x0F\x87\x68\x9B\x93\x0A\x98\xFB\x68\x19\x23\xE2\x89\x0C\xF1\x00\x8A\xD8\x4C\x09\x8A\xB3\x34\xC9\xB0\x97\x26\xDF\x2F\x65\x18\xC4\x41\xC6\x68\x8E\xBA\x68\xFD\xE4\xA3\x1E\x06\x51\x34\x10\xC8\x52\xC3\xAA\xD9\xC2\xFD\x2D\xAC\x15\x75\xE5\x9F\x82\x60\x0A\x20\xBB\xCB\x11\xC1\xAC\x11\xEC\x98\x7D\x67\x87\x90\xA3\x1D\x49\xBE\xC3\x18\x20\xD9\x2A\xC4\xD5\x66\x1E\xE2\x1A\x72\x11\xE2\xAA\x24\x95\xCB\xEA\x12\x88\x23\xAE\x03\x85\xB2\x0F\x82\x1A\x89\x96\x47\xDF\x20\x50\xA5\x0D\x04\xE2\x1A\x81\x9D\x98\x56\x3A\x40\x7F\x61\xBF\xE5\xAE\xC7\xB2\x87\xC3\x8B\xA4\xF2\xF5\x08\xE6\x37\x97\x9A\xF3\x7D\x0A\xD1\xBF\xDF\x7E\xFD\x7B\xD0\x83\x74\xD5\x4F\x67\xE0\x79\xFE\x09\x79\x69\x4E\xB0\x77\x1F\x12\x9A\x4B\x35\xE7\xC4\x87\x53\x65\xB2\x33\x7F\x06\x4D\xB0\xD7\x63\x56\xB3\x0A\xA8\x47\xAC\xE2\x4C\xBE\x5A\x45\xD5\x7F\x3D\xD2\x11\x5C\x4A\xBF\x39\x03\x71\x49\x1D\x59\xBA\x7D\xB4\x1A\xCE\x14\x84\xFC\x9C\x1F\x34\x03\x16\x9D\x39\x1F\xF1\x92\xC4\x89\xC4\xE0\xD7\x0E\xB5\x88\x31\xBD\x35\x9C\xC4\x98\x35\xC0\xDC\x24\x9A\x2B\xA5\xB9\xDB\x67\x9E\xE0\x74\xB6\x4E\xF1\xC4\xB7\x12\x69\x66\x0A\xEC\x5A\x18\x5F\xBE\x40\xCE\x87\x2E\x56\x5B\xC0\x83\xDC\xC4\x5A\x24\xD4\xDC\x45\x9B\xD9\xCB\xDD\x04\x5B\xD8\xC1\x7E\xAB\xEE\x09\x5C\x54\x82\x55\x0D\xC9\xF4\xA6\x48\xF3\xEC\xE0\xDD\x22\xFF\x5B\x2D\xD2\xBC\xB1\x03\xF1\x9F\x2B\xA5\x98\x74\x9F\x27\x38\x21\xEE\x33\x45\xA0\xE7\xA1\xAC\x70\x35\x0F\x9F\x28\x26\x31\x8A\xBA\x64\x7B\xBE\xF6\xB3\x5E\x11\xE7\xF4\xDE\x41\x86\x9C\xC5\xC7\x87\xD9\x71\x9E\x48\xE8\x2F\xE3\x5C\x33\x16\x13\xCF\x59\xAA\xF5\x1A\x6C\x39\x53\xCD\x38\xB9\x5B\x24\xF8\x79\xC7\xC5\x4F\x8E\x29\x22\xC6\xE7\x75\xCB\x83\x47\x8B\xE8\x15\x82\x7E\x93\xCC\xD9\x6A\xC2\xB3\xCA\x7D\x12\x06\xCA\xB5\x1E\x59\x42\xA8\x74\x87\x44\x84\xB3\xEC\x25\xBB\x2B\xDA\xE7\x97\x46\xBF\x7A\x52\xFF\xF9\xF4\xD5\xD7\xCB\x17\xC5\xBF\x4D\x8B\xAA\x9F\x81\xC0\x6F\xCF\x04\xE1\x18\x50\x93\xE2\xEF\xC3\xC5\xB0\xFF\xD5\x2E\xE4\xA2\x06\xBA\x4D\x92\x48\xC1\x0C\xE3\x0C\x13\x1A\x26\x31\x93\xC2\x0B\xE3\xC2\x51\x22\x4C\xF1\x1C\x02\x2D\x6A\x8C\x05\x79\x5F\xAC\x6D\x20\xCE\x23\x74\xD8\x35\x30\xAF\x43\xFA\x99\x22\x8A\xAF\xAE\x3E\xE6\xEC\xF6\x97\x1A\x4F\x9B\xB9\x5C\xAE\xC6\xED\x4A\xB3\x2A\xE5\x07\xC8\xA2\x5F\xF4\xD8\x95\x18\x4F\xB5\x4A\xAF\x2F\xE4\xE1\x33\x46\xC4\xB7\x77\xED\xC8\xAA\x1F\x93\x66\x07\xB6\xBF\x8C\x1F\x49\xB2\xBF\xBA\x7A\x77\x4C\xE9\xC3\x4D\x84\xF6\x59\xFB\x47\x0D\xDA\x64\xAF\x39\xB7\xD0\x69\x01\xF3\xD8\x95\xFF\xB2\x7E\x0F\xA1\x8F\x1F\xAE\x41\xD6\x56\x3F\x5D\x4C\xC8\xFA\x86\xA2\x75\x88\xD3\xC7\xC9\x57\x23\x7C\x51\x80\x1E\xC8\x6A\xC4\x11\x62\xFF\x23\xC1\xBB\xF0\x74\xED\xFB\x38\xF2\x6E\x48\x12\xD3\xEB\x38\x78\xCD\x57\x7E\xCA\xB9\xBB\xE5\xF1\x60\x2D\x3A\x09\xA9\x90\x6B\xC6\x25\xA9\x24\x1B\x2C\x91\x40\x59\xFB\x48\x0B\x6F\x66\x9F\x85\xF9\x9F\x30\x0E\x7A\xB8\xF3\x52\x66\x3B\x1F\x83\xB5\xE7\xCC\x37\xFC\xFB\x42\xE4\xEA\xEA\xBA\x78\x38\xFB\x3D\xCE\x68\x95\x1C\x3A\x4C\x0F\x56\x2A\x62\x4E\x6C\x43\x6C\x2A\x18\x57\xC1\xFA\xE8\x25\xB8\x7D\x83\xF7\xA3\xFA\xF3\xEB\xE5\x0B\x25\xDE\x7F\x14\xFF\xF0\xE6\xB6\x5E\x5D\x24\x03\x39\x7E\xC2\x7B\x7C\x4A\xAF\xAE\xFE\x44\xD1\x37\x4C\x5E\xB2\xB6\x1B\x82\xF6\xAF\xAE\xAE\x7E\xF0\x36\x51\x10\x57\x6C\x40\x2E\xF5\xE4\xDA\x60\x82\xBD\x49\x89\x05\x76\x80\xEF\x72\xCC\xCE\xCC\xF2\x0E\x05\x22\x0B\x2D\xA1\xBF\x79\x59\xC1\xD2\xA5\xBA\x9D\xF1\x87\xC8\xC3\x69\x4C\xFF\xBF\x5F\x9E\xD9\x11\xDA\xFB\x77\xF1\x8A\x37\x88\x56\x69\xA4\x10\x5D\x08\x73\xFE\x53\x2B\x62\x9B\xAE\x9F\x69\x31\x95\xAB\x7D\x79\x56\xFB\x4D\x55\x42\x8F\x8B\x1E\xE4\x78\x7C\x14\x54\x8A\xFF\x87\xF4\x3A\xA2\x72\x9A\xA8\xCB\x0C\xB5\xAD\x23\x37\xA3\xCE\xFF\x9B\xE3\x0C\x30\x58\x6F\xA5\x5D\xE8\xA9\x5A\xBF\x4F\x52\x40\xEB\x51\x48\x0F\x30\x28\x97\xAA\x97\x11\x94\xAE\x3D\xF5\x7D\xFD\x40\xF1\x27\x14\xEF\xB1\x58\x6E\xD6\x7F\xF4\x1E\xA1\x0E\x64\x55\xC1\xED\x5D\xDD\x4D\x5B\x7D\xCC\x4E\xAC\x4B\xBA\xA8\x00\xF5\x57\x65\x95\x71\x54\x0A\xEC\xF1\xF3\xCF\x30\xA0\x75\xFD\xC3\x5A\x3E\xA4\xBD\xE4\x69\xA7\xA6\x4A\x67\xE9\xE1\x64\xC1\x37\x28\xA5\x39\xC1\x7A\x34\xF4\x35\xA9\x13\x41\x55\x8B\xA6\x93\xFB\xD8\xB7\x31\x32\x66\xAA\xD7\x78\x1F\xC6\x95\xF7\x75\xAD\x15\xFE\x1D\xFB\xC8\x3F\xE0\xE0\x53\x1E\xE3\xC2\x8B\x3F\xE7\xBB\x5D\x78\x92\xA3\xB0\x59\x93\x07\x1B\x9B\xEA\xB6\xAF\x8D\x27\x13\x62\xAA\x38\x9D\x42\x60\xE8\x10\xBF\xF2\xD5\xEA\xF2\x32\xEA\x90\xF2\xF1\xD5\x21\x71\x12\xD8\x2F\xE3\xD7\x0A\x6D\x61\xF3\x23\x8C\x69\x59\x02\x31\x67\x79\x55\xDD\x0F\x5E\x48\xFA\x0A\xBA\xB8\xBC\xFD\xEE\x7F\x37\x76\x22\x05\xF0\xB2\x7E\x1B\x99\x31\xD9\xB3\x3B\xC8\xE1\x4E\xE6\xBB\xD2\xAD\xF8\x1D\x2E\x9F\x9F\x4B\x49\x37\xC1\xBB\x18\x65\xC0\x65\xEA\xED\xC2\xB7\x7F\x3C\xDF\xCD\x70\x69\xF1\x14\xFB\x3D\xCC\xB0\xC7\x9E\x8B\x49\xEE\x53\x8F\x1A\x3F\x17\x4F\xF3\x34\x32\x4B\x57\x76\xD2\x74\xE6\xBD\x4A\xA1\xF5\x87\x5A\xB9\xF7\x1C\x02\x3E\x31\xF4\x2B\x3F\x4D\xDF\xED\x4B\x7A\x5B\x32\xDA\x92\xD1\x96\x8C\xB6\x64\x34\x51\x32\x12\x47\xA6\xE5\xFD\xCD\x2D\x29\x6D\x49\x69\x4B\x4A\x5B\x52\x9A\x37\x29\x6D\x11\xBD\x45\xF4\xDC\x11\x6D\xF9\xE2\x93\xB6\x1F\x15\x04\x7C\x19\xE2\x13\xF6\x73\x92\x85\xF7\xFA\x22\x87\x65\x11\xC9\xA9\xB7\xD2\xE7\x26\x8C\x03\xF3\x8A\x4A\x3F\x01\x3B\xAF\x1D\xB9\x53\x1D\x71\x09\xDC\x89\xE9\x92\x55\xC9\x56\x8A\x6C\x89\x6B\xC1\xC4\xB5\x0B\xE3\xC0\x2C\x46\xFB\xBC\x3F\x62\x94\x8F\xB5\xE4\xDB\x2B\x7B\x39\xF4\x51\xF9\xB4\x24\x55\x73\x06\x6F\xEB\x68\xDA\xED\x7A\x57\xB8\x58\xBB\x5C\x6D\xF8\xD6\xA9\xE5\x53\x1E\xF7\xD8\xB1\x13\xFA\xFD\xFB\xCB\xCD\x0B\x5B\xDF\x29\x78\xFE\x81\x68\x18\xFF\xD2\x97\x6B\x9B\xC4\x00\x67\xEF\xC5\x73\xEF\x97\xE7\xBB\xDD\x6E\x07\xEF\xFF\x77\x23\x9F\xC7\x98\x26\xFE\x01\x91\x91\xA5\x1C\x65\x4C\x5A\xF9\x3A\x79\x53\x2B\x95\x6D\x0A\xDE\xA6\xE0\x6D\x0A\x1E\x6D\x0A\xEE\x9D\x36\x26\xD8\x2F\x1D\x29\x2F\x6C\xEB\x87\x5B\xC6\xD8\x32\xC6\xB6\x7E\x38\xCD\xFA\xE1\xC8\xA5\x57\xE7\x87\x89\x41\xFC\x5C\x96\x5E\x3A\x52\x1F\x77\x09\x66\x20\xF3\x11\x9F\x0C\x3B\x3C\x33\x0C\x33\xB3\x2B\xFD\x95\xEA\x3A\xD0\x85\xFB\x6A\x3F\xCC\xCD\xD4\xC7\xB2\x5E\x23\x08\x13\x99\x70\x98\xDA\xA4\x9E\xA7\x24\x6B\xB5\x5D\x3F\x8F\x70\xB3\xE6\x80\x61\xFF\x83\x55\x80\xD2\x32\x47\xEF\xF8\x6D\xA3\x35\x95\x13\xD8\xF8\x1A\x17\x07\xDC\x15\x6F\x7D\x9F\x40\x25\xFD\x31\xC9\xE8\x7F\xC2\x2C\xAC\xBB\xF1\x17\x39\x9E\x6A\x3D\x6D\x3F\x81\x17\xB7\x80\xE3\xE5\xF3\xB3\xB6\x9F\xAA\x9E\x9E\xBF\xED\xA4\xF5\xFC\xD2\xAC\xD0\x37\xDA\xE3\x74\x2A\x19\x46\x4C\x4B\xD3\x9B\xC9\xFA\xB6\xCC\xFC\xE2\xD8\x5F\x6D\x99\x5F\x1E\x73\x7A\x9C\x41\x16\xE3\x9A\xE8\xF4\x7A\xAB\xEF\x18\x2C\x10\xBF\x8E\x2B\xFA\x0B\xD8\xC6\xE5\xBD\xA1\x65\x0D\xD6\x75\xA9\x7E\x5E\xE9\xBA\x6F\xB4\xCC\x31\x83\x59\xDF\xC6\xD1\x5E\xDE\xE1\x7F\x81\xE2\x34\x65\xB2\xB4\x82\xD0\xEB\x35\xE8\x5E\x7C\x84\x59\xC8\xFC\xC2\xF1\x50\xCA\xCD\xFC\x76\x83\xC2\x68\x6C\x1E\xD5\x3B\xBF\x9F\xC3\x63\x1A\x85\xBB\x87\xB1\xE9\xFF\x9E\xF1\xD7\xE2\x8B\xE9\x4F\x75\x2A\xE0\x1D\xFD\x31\x59\xBE\x63\xD7\xBC\xCE\xC0\xD0\xF2\xDE\x74\xD1\xF4\x7B\x5C\xDF\x45\xD7\x1A\x52\x86\xB0\x1C\x2E\xDB\x5B\xEC\x13\x6C\xD8\xFE\xEC\x45\x1D\xBE\x24\xA0\x2F\x29\xE8\x1A\x81\xA1\xA1\xDF\xFE\xE2\xF6\x68\x91\x69\x7F\x48\xE9\xCF\x67\x40\xE1\x3F\x06\xD3\xB7\x09\x2D\xCB\xD6\x89\x18\xDC\x84\x71\x98\x1D\x34\x12\x93\xF0\x6A\xB9\x03\xA0\x17\x0B\xA1\x68\x74\x4E\x74\x72\x5C\xFE\x1E\xAB\x71\xD9\x91\x50\xC7\x04\xEB\x4E\xDD\x79\x12\xED\xA9\xF9\x9B\x24\xF6\x91\x0D\xBF\xF6\xF0\xAA\xC7\x47\x44\x32\x5C\xDE\xE0\x35\x8C\xB7\x9E\x0D\x7B\xD9\xC5\x90\x61\xBA\xD0\x92\x1E\x3C\xFA\x5A\x52\x2E\xA9\xF4\xF9\x05\x32\x60\x7F\xC7\x90\x48\x96\xCF\x6D\x9F\xAB\xAF\x1D\x49\xF7\xF1\x75\x1B\xC3\x96\xF9\xDA\x35\x6E\x9C\xC8\x74\xAA\x11\x5C\x29\x77\x8E\x98\x4E\x1A\x8F\x1A\x2F\x9D\x38\x9B\xA3\xA5\x83\x45\x5A\x62\xA5\x9D\x12\x18\x29\xDD\x2C\x38\x66\x9C\x74\xE2\x3C\x42\x94\x8C\x5F\x64\x8E\x71\xC5\xD0\x78\xAC\x67\x60\xDA\x6C\x25\xB3\x0B\x85\x26\xBA\xAF\xC8\xF3\x50\x10\xB0\xEF\x0A\x78\x3E\x4A\x91\x1F\xD2\x07\xC7\x0B\xA0\x46\x30\xA6\xFD\xCB\x88\x6A\xCD\x23\xD5\x77\xDD\xD7\x75\x81\xEE\x53\x6D\xB9\x38\xB0\x1A\xB0\xFA\x2B\x53\xB7\x1D\x04\x06\xAB\xD5\x4E\x0F\x1D\x46\x22\x82\x52\x1F\x52\x1A\x1E\xF9\xC7\xBD\x7A\x52\xB8\x89\x10\xA5\x38\xEE\x4F\xA0\xC0\xCE\x29\xDB\xC0\xFC\x17\x32\x6C\xFE\xD8\xE9\x54\x8E\xFE\x09\xDF\xE5\x21\xC1\x01\xBF\x83\xF1\x26\x21\xEC\x1A\xC6\xE9\xBF\xC9\x26\xD4\xEE\x7A\x56\x29\x95\xF8\x92\x94\xEB\x36\x2D\x0F\x5F\x3D\x9E\x0D\x0C\xFC\x3E\xE1\x7B\x4C\x32\x3C\x1D\x5B\xF1\xC8\xD2\x12\x77\xB3\xD9\x83\x61\x3B\x88\xB7\x1D\xC4\x5B\xF2\x20\x9E\xE5\xB5\xBF\x8E\x97\x76\x6C\xFE\xBB\xF9\xEF\x6A\xFD\x77\x4B\xBC\x9B\xE3\x9E\x8F\xE3\xAA\x97\xB3\x2D\xF2\x74\x06\x7C\x59\x6D\x19\x39\xBA\x4D\x43\x8B\xCB\x68\x4B\x35\xCB\x08\x27\xBA\x6C\x96\x46\x21\xF5\x6E\xF3\xDD\x0E\x5B\xA8\x6A\xDF\x0D\x71\x90\xE4\x29\xF0\x15\x41\x33\x32\x3F\xB4\xFE\x13\x5A\x66\x47\x92\x98\x6E\xA6\xA9\x4D\xF3\x64\x73\x19\xBB\x5D\xDA\x1D\x46\x7A\x97\x6E\x69\xFB\xA8\x73\xD7\xCA\x52\x0E\x30\xAB\xAD\x4D\xC2\xAE\x8F\x5D\xAB\x95\xDE\x5E\x74\xAF\x4B\xEC\xC5\xDC\xD6\x71\x3E\x58\xD2\x6F\x5D\x45\x5C\xA9\xE3\xF6\x10\x7F\x15\x9E\xEB\x2A\xF7\xBC\xAE\xDB\x71\xBE\x9E\xDD\x6F\xBB\xCA\xB7\x26\xA7\x1D\x20\xFB\x72\x1E\xDB\x55\xE8\x85\xDC\xB5\x5B\x19\xB5\x9C\xDB\x76\x94\x73\x95\xEE\xDB\x5F\x87\x15\xB8\x71\x47\xE1\x85\x8D\xD1\xB7\x37\xD7\xEC\x7F\x55\xE7\x8F\x24\xD9\x8B\x1B\xD5\xE0\xE7\xAF\xFA\x9E\xD4\x1E\x83\x99\x2A\xFB\x0F\x4E\x0F\xD8\xF3\xD5\xC1\x6D\x1B\xBC\xAC\xC7\x9F\x09\xF9\x76\xF7\x25\x79\xC3\xDE\xFF\x94\xBE\x2F\xD6\x80\x45\xA1\xB5\x56\x6D\xA7\xBA\xA3\x81\x26\x11\x40\xB7\xCC\x04\x6C\x96\xFC\x84\xEE\xA4\x8A\xCD\x60\xBB\x06\x4F\xA6\x0A\x9C\x32\x32\x1C\x85\x98\xC1\x7B\x6D\x71\xD3\x5D\xEC\xCE\x17\x50\x35\xCC\x18\x9B\x92\x41\xFB\xA5\x52\x70\x3F\xA5\xF5\xBF\xD9\xD2\x36\x00\x28\xAF\x74\xB3\xE6\x67\x81\xB8\x78\xC3\x10\x00\x7B\xA1\x0B\xA3\xDE\x2A\xD4\xD7\x9C\xDD\xD2\x4C\x0F\x06\xE2\xEB\x02\x03\x68\x34\xEF\x09\x0C\x77\x9C\x5E\x37\x14\xB5\xF9\x91\xDF\x7E\xEB\xD0\x12\x2E\x65\xBE\x82\xC6\x49\x9A\xA7\xE2\xC5\x31\x36\x37\xB4\xF6\x05\xF7\x21\x7F\xF6\xB8\x1D\xEF\xF3\x44\x9B\x6B\x0E\x72\xCD\x6D\x20\xD6\x33\x10\x93\x7F\x3D\x67\x1B\xA2\x61\x43\x24\xDA\x64\xE2\x6F\x8A\x6C\x43\x35\xDE\x50\x2D\x6E\x93\xD6\x6B\xF7\x3F\xA7\x88\x64\xF8\x33\xA6\x5F\x5E\x16\x43\xFC\x8A\x9D\xF0\xC6\x84\x0A\xE7\xE2\x2D\x1F\xF8\xEB\x4D\xC2\xD0\xD1\x27\x18\x51\xEC\x85\x71\x80\x4F\xCA\x31\x78\xA6\xDD\x75\x10\x7C\x49\xFE\x37\xC7\xB9\xF9\x51\xC9\x5A\x74\xEB\x8F\xD6\xC3\x28\xF6\xD5\x5F\x10\xE0\x53\x1E\x33\x66\x1F\xE2\xD7\x0F\x9D\x9F\x00\x8D\xA2\xF9\xC2\x49\xEC\x73\x92\x72\xF8\x98\xCC\x2B\x34\xE4\x52\x9F\xF2\x98\x45\x21\xC0\x56\x0C\xFD\x51\x5D\xA8\x2B\xC3\xB1\xED\xDC\x8B\xFF\xF4\xE3\x33\x82\x90\xE3\xAF\xD2\x88\x42\xE1\x0C\x53\x46\x5A\x22\xFA\xE9\xCF\x3F\x12\xFF\x1B\x26\x4D\x5C\x1C\x92\xE4\x5B\x76\x75\xF5\x4F\x4C\xDF\xDE\x5C\x33\x29\x58\x2F\xD6\xFF\xBF\x93\xE4\x9B\xF8\xC2\x88\x33\x65\xC7\xD5\xA2\xCF\x18\x11\xFF\x70\x73\x73\x23\x1B\x8D\xB5\x7E\x44\x04\x1D\x33\x46\x8D\x3B\xA8\x60\x84\xDF\xE3\x28\x8C\x71\xC0\x11\xFF\x48\x92\xF4\xE5\x0E\x45\x19\xE6\x56\x51\xFF\x7C\x65\x25\x7E\x31\x25\xF1\xFE\x21\x38\xBD\x54\xDD\x5D\x77\x5A\xA1\xDC\x5F\x7D\x9B\x54\x8C\x6A\x65\x6A\xAE\x01\x68\x0F\xA6\xB9\x44\xE9\xB1\xE2\x3C\x97\x43\x58\xDF\x69\x95\x32\xC9\x97\x89\x32\x49\x79\xB6\x7A\x92\x44\xD2\x4E\x7B\x81\x3C\xE2\x2A\xD4\x9C\x69\xC4\x41\xA6\x19\xB2\x88\x83\x14\x83\x93\x48\x37\xEB\x4F\x99\x43\x3A\x49\x32\x67\x0A\xE9\xE6\x0C\xCE\x19\xE4\xCB\xA8\xB5\x48\x21\xE4\x54\xA5\x48\x27\xDA\x33\x05\x6B\x37\x99\xA6\x0A\xD6\x6E\x52\xF4\x0B\xD6\xDE\xD6\x1F\x3D\x58\xFB\x4A\x32\x79\xB0\xF6\x76\x06\x53\xB0\x4E\x6E\x97\x61\x0F\xCB\x55\x06\x19\xB5\x06\x69\xB4\x18\xBD\x04\xE9\x42\x7A\xF6\xFC\xB1\xE0\x5C\xDF\x49\x88\xA1\xD9\x63\xE1\x99\xBE\xA7\x20\x33\xE6\x8E\x71\xE6\xF9\xA9\xAD\x32\x46\xE6\xF8\x32\xC2\x3A\x88\x92\xF4\x46\xAD\x3D\xFA\xD2\x9E\xF4\xE9\xA5\xB7\x50\xCD\xFA\xEA\x94\x5C\x8E\xF8\xE8\x1F\xC8\xF4\xD6\xAD\xAE\xAE\xC5\xBB\xF0\xC4\x6E\x65\xF1\x6E\x48\x12\xD3\xEB\x38\x78\xCD\x0F\x37\x97\xA7\xF2\xD8\x66\x92\xBE\x7C\xD8\x21\x51\x0E\x93\x73\x9A\x19\x64\x80\x13\x8C\x38\x85\x0C\x71\x12\xD7\x39\x64\x0C\xEB\x8F\x33\x89\x8C\x20\xC9\x34\xB3\xC8\x18\xCE\xE0\xFA\xB8\xF8\x65\x84\x05\x27\x50\xDE\x71\x8A\xBD\x9E\xA4\xE7\x4F\xD8\x4E\x93\xFC\xD0\x7C\xED\x56\xCD\x0D\x49\xD7\xAE\xA6\x5D\x38\x5B\x2F\x5B\xEE\xF7\x77\x80\xA9\x73\xF5\xB8\xE5\xFE\x08\xA6\x9F\x30\x53\xAF\xA0\xDC\x1F\xC1\x11\x9C\xF3\xF4\x08\xCB\x7A\xF2\x43\xC1\x44\x95\x75\xD7\x35\x93\x5E\x39\xB1\xF3\xFA\x58\xE7\x9C\xD8\x63\xED\x65\x89\x9C\xB8\x9A\x25\xD4\xFE\x0E\x30\x49\x4E\x9C\x6C\x01\x75\x04\xD3\x8F\x9D\x13\xD7\xB5\x7C\x3A\x82\x23\x0C\x58\x02\x19\x64\x95\x51\x96\x40\x46\x58\x3C\x85\x94\x18\xBB\x9C\xEE\xB8\x3C\x35\x30\x4B\x4F\x54\xB8\x76\x5F\xE5\x5A\x36\x47\xAF\xA5\x6C\x5D\x6E\x91\xBA\xBF\x73\x0C\x49\xD0\xCB\xD6\xAC\xAB\x5A\xA1\x1E\xEE\x04\x23\x67\xE7\x79\xD6\xA7\xAF\x63\x14\x3D\xFC\x85\x39\x69\xD7\x11\x18\xEB\x0C\x68\x2F\xE6\x93\x9E\xAD\xEC\x25\x91\x4B\x60\xA8\x84\x2A\x4C\xFE\xB5\x81\x8F\x21\xF6\x31\xF4\xBE\x80\x19\xDC\x8C\xA9\xE9\xAF\x52\x5D\x7E\x55\x77\x99\xB8\x45\xC3\xA8\x8E\xA3\x0E\xB2\xB3\x29\x2E\x84\x19\x84\x1F\x36\xAD\xD0\xB5\x17\xFE\xE5\x37\x1D\x7C\x14\x45\x5E\x12\xFB\xE0\x3B\x29\xCF\xEF\x93\x08\xD1\x30\xC2\xA5\xC2\x65\x3C\x55\x7F\x7E\xBD\x7C\xFE\xF5\xF2\x45\xC1\x5E\x88\xB5\x42\x18\xE0\x25\xF5\x9A\x8F\x97\x92\xE4\xF4\x20\xBC\xC3\xC2\x3F\x57\xDF\x41\xFE\xAB\xAB\xFF\xE7\x3D\x7F\x22\x0E\x11\xBB\xE4\xE0\xC9\x93\xEA\xAB\xFC\x95\x28\x4E\x2F\x48\xAD\x45\xD6\x66\x6C\x87\xDD\xDE\x30\x83\x42\xBF\x9C\x91\xF1\xDD\x64\x3D\x23\xE3\x5F\x9E\x91\xF1\xDD\x64\x1D\xCD\xF8\x02\xE2\x5B\x56\x08\x08\xF4\xDE\xDE\x5C\x2B\xCC\x9A\x6B\x4B\x9C\x3A\xB6\x1F\xBA\xE7\x04\x78\x66\x16\x08\xF4\x9E\x5E\x04\xA2\xFC\x53\x3A\x6D\xE6\x50\xE9\x3D\xD5\x26\xA2\x96\x49\xC7\x6D\x64\x1F\x8B\xB6\xEB\x2F\x05\x1E\x8B\xA5\xE5\xB7\x60\xF8\x41\x2E\x14\x46\x39\xC1\xEA\x3B\x30\x8E\xAF\x16\x82\xEF\xA0\xAF\xFB\xB5\xCD\x21\x9F\x4D\x58\xB5\x32\xCE\x77\xE8\xAF\x51\x8B\xB3\xF3\x22\xE1\xA5\xC5\xF7\x45\xD7\xF7\xDA\x94\x29\xDF\x21\x63\x44\x12\xC2\xED\x9A\x10\xF4\xF0\xB2\x41\xFF\x72\x20\x18\x05\x4F\x5F\x5D\x5D\x11\x9C\xF1\xCF\x2B\x89\x4F\x69\x46\x92\xCE\x97\xE3\xB8\x50\x30\x4C\xB2\xC6\xAE\xDA\x8B\xCD\xCE\x5A\xC1\xEF\x45\x77\xE9\xEE\xA8\x76\x57\xC2\xD2\x3D\x3E\x03\xA4\x32\x98\xB2\x2B\x45\xD5\xF5\x7E\xBC\xB7\xDC\xFB\x06\x6F\x1A\xAA\xFD\xC4\xB0\x53\x3F\xC2\x24\x48\xD3\x12\x43\x02\xA6\xF9\x7B\x4B\xAE\x82\xB7\x89\x0C\xE7\x8C\x89\x85\x77\x95\x93\xAD\x0A\x71\x62\x77\x75\x9D\x60\x1D\x66\x75\xAD\x48\xFE\xCE\xA2\xB9\x88\x90\xEA\x18\xA1\x87\x48\x1C\x88\xAD\x95\x4A\x09\xDE\x8E\x3E\x64\x0C\x5B\x3E\x6E\xB8\x52\x33\x74\xFC\x40\xE2\x4A\xB5\x90\xD2\xE6\x4A\x65\x14\x8A\xED\x77\xC7\x94\x3E\x94\xDF\x30\x6D\xAB\xEC\x15\xF2\x86\x6B\xFD\x27\x71\x5C\xFB\x77\x5F\xA6\x61\xD9\xFB\x13\x2F\x93\x8B\xE3\xF8\x35\x97\x69\xE4\xB0\xDF\x7D\x2B\xBB\x9A\x2B\x53\xE3\xE5\xF8\x22\x8E\xE5\x53\x1B\x67\xAA\xA9\x74\xAB\xEF\x23\x54\x15\xBC\x85\xFB\x31\xEB\x69\x1C\xD0\xF6\xAF\xA1\x4C\xA3\xAF\x5A\xFC\x7C\xA6\x38\xED\x34\x07\xF5\x98\xAD\x3A\xCE\x22\xC0\x8C\xB9\x46\x29\xA1\x52\x72\x8D\x72\xAE\xB8\xEE\x10\x2D\x37\xDD\xC2\xA7\xDA\x45\xBE\xA2\xA7\x4F\x91\xB9\x98\xB0\xDA\x52\xCA\x62\x92\xAC\xDA\xE7\xD7\x60\x20\x30\x3D\x2C\x3A\x5A\xDA\x81\x2E\xE7\x33\x5C\x2B\xD0\x60\x91\x47\xE2\x25\xE3\x5C\x9D\x01\x97\x0F\xF4\x95\x4E\x1F\xBD\xDD\x5A\x0D\xD1\xFE\x84\xCA\x03\x99\x03\xA9\xC0\xDD\xFB\x9D\xBA\x14\x68\x71\xCF\x79\xBF\xA2\x9D\x3A\xCD\xB3\x80\x05\xF4\x33\xD3\x60\xA9\xF0\x3C\x33\x33\x35\x0B\xD7\xFA\x9A\x7F\x18\xD3\x57\x85\x12\x96\x63\x8A\xF5\x1F\xE0\xF5\xDF\x63\x51\x34\xD0\xB1\x5C\x2A\x3B\x1D\xD3\xCF\x98\xBE\x3B\x85\x19\x0D\xE3\xBD\x40\xCF\x44\xC5\x24\x4D\xDB\xA6\xF5\x28\xCC\x0D\x5B\x11\xF3\xAF\x08\x39\x7E\x15\x6B\x72\x31\x86\x7F\xF4\x6A\x4E\x11\xBB\x7E\xD3\x6A\x72\xD9\x66\x72\x25\xFB\x92\xDB\x7C\xBE\xD4\x22\xC7\x2A\x9C\xC9\x5D\xC6\x05\xBC\xA9\x45\xB8\x29\xDD\xC9\x6D\x59\x73\x62\x5F\x72\x14\x62\x39\x47\xEA\x2E\xE0\x5C\x5E\xE4\x28\xD9\x2C\x2E\xE4\xB4\x62\x3C\x97\x2B\xB9\x09\xB3\x02\x97\xEA\x2C\xE8\xEC\xAE\xE5\x26\xA1\xF6\x38\xCA\x8B\xFA\x0F\x31\xFE\x88\x32\x87\x1D\xD9\x79\x0B\x7B\x71\x39\xF2\x0C\xC4\x15\xFA\x0D\xDA\xE5\x3E\x17\x7D\x8F\xF8\xE8\xA7\x0F\xA2\xB8\xBF\x67\x8D\xA8\x9D\xCE\x8A\xC1\xDD\xF5\xF1\xD7\x19\x00\x4B\x2C\x2A\x92\xF0\xF8\xD1\xE3\xA5\x35\x98\x66\x13\x9D\xF7\xD8\x2F\x42\x53\xD0\x8C\x0F\xA8\x29\x86\x15\xC4\xB2\xC4\x08\x99\x30\x5F\x2F\x5F\x88\x69\xE2\x3B\x41\x29\xFB\x0E\x8E\x4E\xBD\x74\x97\x57\x86\x15\x1A\xA5\x51\xEC\xF3\xA4\x45\x33\xD7\x93\xFF\xC0\x60\x35\x6B\x8A\x30\x8A\x3A\xC1\x9D\xA3\xE9\x80\xA9\xF1\x2C\xD5\xE8\x36\xA9\x9E\xA5\x8A\x65\x7E\x3A\x73\x2D\x1C\x8B\x8A\xB3\xD4\x4D\xAF\x78\x29\xC1\xE5\xE6\xCB\xA7\x3C\xC6\x9F\x50\xBC\xC7\xC2\x54\x54\xB7\xFD\x81\x8B\xA4\x62\x2D\x93\x6A\xDC\x52\xB5\x00\x67\x94\x24\x0F\x8A\x6E\x05\x3F\xE1\x83\x62\x0A\xD3\xF2\x73\x62\xEA\x41\x8E\xB5\x0A\x79\x1E\x96\x6C\x9F\x60\x3E\xE1\x3D\x3E\xA5\x57\x57\x1F\xD9\x94\x4D\x11\xC5\x57\x57\x1F\xF3\xEC\xC0\xDB\xAB\x32\x88\xFF\x6A\xCE\x20\x40\xBD\xFE\x85\x1E\x6E\xF1\x9B\x24\xF6\x11\xE5\x65\x8C\xB8\xC0\xA8\xF7\x2A\x2B\xB6\x41\x62\xBC\x39\x20\xF2\x26\x42\x59\xF6\x3A\x0F\xA3\x00\x93\x02\xA3\x98\x30\xAE\x6F\x93\x7B\xF5\xDD\x82\x61\x8A\xBE\xC5\x3E\xC1\x3B\x71\xBD\xBB\x33\x3D\xE7\x02\x6D\x98\xA4\x10\xD6\x87\x74\xDE\x51\x78\x93\xC4\x14\x85\x71\x36\xEA\x10\xBC\x49\x8E\x69\x4E\xF1\xE7\xF0\x98\x46\xB8\x6D\x24\x7A\x3A\xA3\xCC\xF1\x3A\x08\x8A\xD0\xFA\x92\xC8\x24\x26\x60\xEB\xFA\x96\xC3\x64\xAA\xBA\xFB\xF7\x1F\xEC\x63\x91\x51\xB3\xF7\xC0\x7D\xE0\xC1\x8F\xF0\x4D\x12\x31\x83\x39\x0E\xBB\x46\x69\x70\x84\xC0\xB2\x4D\x19\x19\x06\x6B\xE8\x11\xA1\xB4\x0C\xB0\xB4\x46\xBB\x70\x53\x9E\xFD\x95\x0D\x9D\x41\xD6\x9A\x3C\xB5\xBB\xB3\xB7\xE5\x08\x3B\x0F\x9E\x2C\x3E\x00\x83\x3E\x82\xB7\xD9\x88\x4F\xEE\x78\xCE\xCC\xC7\x31\xE4\xDB\x84\x76\x5D\x67\x68\x27\x35\xB9\x91\x0C\xAC\xC6\x36\xC9\x88\x31\xDF\x4A\xB4\x6F\xB0\x7F\xC2\x29\x46\x14\xF2\x17\xE7\xB3\x0C\xA3\x94\x30\x23\x48\x31\x43\xB9\x33\x96\x94\xFA\xF8\xFD\x13\xD3\xBA\xD1\xAD\x9C\x1C\x41\x96\x73\x78\x56\x99\xD7\x7D\x3B\x3F\x0D\x8D\xED\xB7\x9D\xAA\xD9\x4A\x04\x1A\xD2\x30\x89\xB5\xD0\x9F\x3B\x82\x87\x8B\x31\x4B\x08\x8F\x24\xE6\x38\x31\x3C\x5C\x98\xC7\x11\xC4\xA3\xBA\x70\x8F\x28\x1E\xD9\x77\x17\x0A\x63\x89\xC5\x9F\x28\xFA\x86\x49\x79\xD0\xAB\xF8\x21\xEC\xB3\xC8\x15\x8D\xAF\x7E\x11\x79\x71\x71\x7F\xF0\x5F\x90\xE5\x86\xA9\xD5\xF9\x91\x7A\x74\x76\x13\x31\x02\x4F\xCF\x17\x64\xD8\xA8\x31\xB2\x2D\x21\xAF\x22\x3F\x77\x3B\x46\x3F\xBE\xA1\x5C\x5E\xD2\x69\xF7\x9A\x61\x23\xDD\x46\xAA\x07\x11\xDB\xED\x0D\xA3\x0C\x14\x74\x07\xC5\x12\x4C\x7B\xA6\xE2\xB7\xC9\x1F\x78\x47\x3F\x22\x82\x63\xFB\x36\xFB\xC0\x72\xA5\x03\x9F\x49\xEB\x91\x0E\x72\x88\xDB\xFC\x63\xAA\x34\xFC\x89\xB7\x03\x67\xE1\x90\x00\x4C\xE9\x3F\x98\xD0\xD0\x47\xD1\x6B\x44\xDA\x1E\xCF\x47\x5F\x79\x82\xB8\x0F\xF4\xB2\x16\x75\xC6\x77\xA7\x0E\xF6\xEB\x3B\xD8\x1D\x58\xBC\x4D\xDE\x24\x51\x84\xD2\x0C\x03\x6B\x51\x03\x78\xB8\x2D\x49\x5B\xB9\x0F\x1D\x59\x1B\x69\xA9\xDB\xEF\xF1\x70\x59\x47\x57\x7E\x84\xA7\x17\x3B\x83\xF3\x7E\x22\x71\xF6\x9C\x4E\x53\x9B\xF3\x08\xF3\xA4\xF6\x81\x5C\x47\xAC\x7A\xA2\x40\x0F\x25\x43\x3C\x05\x5F\xA5\x85\xB2\x87\x53\x79\xDF\x49\xD6\x0E\x4F\x39\x76\xBA\xAD\x93\xC3\xA7\x70\x7F\xA8\xE7\x99\x96\xC4\xA3\x65\x90\xFE\x84\x5D\x03\xAE\x8B\x78\x3D\xF3\x22\xCC\x62\x68\x20\x6B\x54\xCF\x3D\x7A\x35\x85\xFA\x86\xAC\x7D\x48\x3B\xFA\xBE\xBB\x7F\xF4\x9D\x9A\x6F\xC2\x38\x64\x77\x38\x8E\x19\x1C\x66\xA2\xAE\x81\xE1\x2A\x56\xCF\xA0\xD0\xC9\x0F\x0D\x08\x89\xE2\xB9\x07\x83\xA4\x8C\x7B\x20\xF0\xE3\x32\x7F\x60\x14\x84\xF1\xBE\x2E\xB0\x0D\x4F\xF1\xDD\x9F\xB4\x3B\x92\x97\x8E\xB5\xF6\xA4\xE1\xE8\xBB\x43\xA8\x7E\xFE\x8E\x5A\x43\xF6\x06\xF9\x34\xA9\xE7\x76\xBE\x7C\xD6\x6B\x2A\xFF\xED\x99\x76\x99\x7A\x79\xFC\x90\x75\xB9\x21\xE8\x88\x5B\xBC\x90\xE1\xD4\x6B\x3C\xF8\x98\x46\xC8\xC7\x7C\x9D\x27\x8B\x92\xEF\x5E\x8A\xE8\xE1\xA5\x24\xDC\x93\x5A\xBA\x27\xAF\x54\xC1\x05\xD8\x94\x5A\x4B\xA0\xBA\x46\x7A\x9F\x70\x16\xBD\xE9\xCE\x23\x2F\xAF\xED\x56\x2A\xA4\x46\xEF\xF7\x63\x1A\x5D\x5D\x7D\x4A\xF2\x38\xB8\xEC\x4B\xDB\x37\x7A\xE8\xE7\x34\x0A\xFD\x36\x17\xE5\x48\xCC\x47\x27\x8D\xA5\x56\x03\xFC\xFA\xC8\x0D\xE0\x3E\x35\xCC\x69\xF5\x5F\x56\x62\xF5\x55\xA7\xDA\x7E\x5F\xDE\x58\xAB\x06\xBA\xF7\xAD\x5B\x74\xE1\xB9\xF5\x0C\x12\xA8\x3C\x1D\x95\xB7\xC5\x03\xE5\x0D\x58\xC3\x9C\x9D\x7E\xFA\xEA\xDB\xD9\xA9\xA0\x57\xAA\xE7\xA0\x82\x54\xB0\x9F\x83\xC0\xCE\xEB\xC1\xE7\xA0\x8C\xF2\x40\xA3\xAF\x24\xB8\xEE\x32\x9C\x83\xB2\xE2\xCE\xD8\x39\xC8\xDB\x5E\x68\x9D\x83\x16\x8E\xB3\xDE\x4A\xAA\xE6\x3E\x07\x71\xCF\x58\xAD\xB6\xA3\xC0\x67\xA3\xDA\x75\x10\xDC\x24\x51\x80\x6B\x75\x40\x85\x8D\x37\x10\x9E\xA5\xCE\xFD\x26\xFE\x55\xA9\xE0\xBE\x00\x7B\x36\x2A\x55\x7C\xDE\xE3\xEF\xA2\x22\xB2\xBA\x7A\xED\xDC\x79\x56\x5D\x8B\xD2\xA3\xD4\x43\x6B\x51\xC6\xB5\x44\x58\x8B\xBC\xC3\x4A\x84\xB5\x68\x71\x3E\x3B\x27\xE7\x60\x4D\xC7\x82\x6B\x25\x0B\x66\x72\xD2\x2C\x5F\xE8\x13\xCF\x69\x75\x5D\xF3\x3E\x3B\x9D\xBB\xED\x23\x9D\x8F\x92\xA3\x4C\x0C\x6B\x51\xC6\x75\x62\x58\x8B\xBC\xC3\x26\x86\xB5\x68\x21\xA4\xB2\x51\x0A\xFC\xBE\x8F\x41\x23\x32\x1F\xF1\x41\xC5\xAC\x47\x75\x95\x99\xE1\x6D\x0B\xCB\x16\xCB\xCA\xC5\xEB\x3E\x74\x8C\xC8\xBF\xE3\xD0\x4F\x02\xFC\x4F\x92\xE4\x29\x70\xF4\xB8\xAD\x1A\xAF\x81\x06\x3B\x08\x1D\x3F\x53\x44\xF3\x8C\x39\xEF\x2E\x8F\x22\x92\xC7\x78\x69\x39\xFC\x03\x22\x34\x59\x83\x24\x2A\x97\xAB\xAB\x1D\xBB\xBB\xEE\x45\x73\xF7\x91\x3E\x47\x54\xE7\xC2\x57\x21\xFC\xEF\xD9\x7F\x50\x14\x06\xFF\xFE\x72\xF3\xC2\xF1\x7A\x3F\x95\xCE\xD2\x6A\x14\x13\xD9\x31\x5D\x5A\x8A\x8C\x92\x08\xC7\x4B\x4B\x51\x65\xBC\x7F\x4B\x6C\x6D\x24\x2A\x4C\xE9\x83\x0B\xCE\xA5\xE9\x70\xEF\x51\x92\xCA\x08\x04\xD5\xEC\x30\x8B\x39\xDA\x92\x79\x9F\x09\x62\x05\xE2\xDB\x16\x89\x57\x61\x57\x4B\x7D\xB3\x9C\x70\xEF\xF1\x9E\x1F\x5D\x5F\x87\x38\xD7\x41\x60\x5E\xB7\x6B\xF8\xCF\x2A\xEA\xF9\x2C\x90\x08\xE9\xE7\x4B\xD2\x5C\xE4\x24\x7D\xEB\xCB\x96\xD6\x8D\xA9\x6E\x44\xC2\x6A\xCA\x93\x6C\x5E\x9D\xAC\x2E\xFE\x7E\xF3\xA6\x18\x4D\xE4\x53\x4C\xCC\x73\x92\x59\x08\xA7\xFC\xDB\x4C\x81\xEF\x32\x1F\xA5\xD8\x99\x91\x42\x49\x0D\xE9\x45\x95\x92\x87\x6E\x51\x51\xC0\xEA\x77\x2C\x53\x43\x35\xFE\x68\xB4\xA7\x95\x3C\xCC\x50\x94\x1E\xD0\x14\xA4\x47\x8E\xD7\x36\x27\x52\x93\xB5\x95\xEC\x53\x6B\x66\x18\x76\xB5\xC9\xD8\xD2\x48\x9C\x86\x1C\xCE\x98\x4F\xDA\x11\x6E\xB0\x9A\x49\xAE\x5E\x97\x60\x8D\x2D\x9B\x6B\x9E\x1C\x91\xA5\xFB\xE4\x37\xF6\x18\x2C\xF6\x9C\x37\xA3\xA2\xE2\x73\xED\x2C\x0C\xC5\xC7\xF9\xF9\x93\xD2\x12\x33\xFA\x7A\x52\x47\xFF\xA7\xD4\x25\xA5\xD7\x1F\xB7\xE6\x93\x66\xC6\xB5\x96\xF9\x83\xC1\xF1\xFE\xDB\xE2\xEF\x8F\x98\x44\xA6\xCF\x8D\x8C\xB5\x52\x3A\x90\xF9\x58\x2B\x9D\xE3\x88\xF1\x06\xA5\x34\x27\xF8\x3D\x3B\xC1\x6F\xBD\x3A\x64\x14\xAE\x50\x77\xF7\xDB\x4B\x86\x8A\xE0\x58\x15\xB4\xD2\x71\x9E\xEA\x5B\x29\x0D\xAF\x8B\xBB\xD9\x7D\x92\x62\xB7\xFF\xD0\x77\x7C\xCD\xBA\x53\xBC\x18\xEA\x9B\x91\x1E\x4D\x26\x11\xA5\xF0\x28\x9A\x14\xDE\x35\x17\xC7\x66\x3D\xEA\x16\x65\xA1\xEF\x65\x8C\xF8\x4B\xE8\x93\x19\x45\x9B\x47\x09\x0A\x69\xF6\xB2\xFC\x52\x86\x69\x11\x4C\xF8\x90\x06\x4A\x53\x2C\xE4\x57\xA7\xEF\x29\xCF\x3A\xE6\x2E\xB7\x11\x4C\x22\x83\x25\x19\x4D\xC2\xCF\x9A\xB4\xA6\x71\x67\x5B\xD8\xAB\x37\x8C\x2F\x2E\x94\xFC\x9E\xFE\xE2\xE2\x38\x65\xD6\x45\xA4\x1C\x6D\x0E\x99\xDF\xE9\x96\x9E\x6B\xA6\xA9\x81\xE6\x16\xD5\xE5\xF2\x92\xB9\x65\x52\xAF\x76\x59\x5C\x24\xE9\xC2\xF8\xE5\xA5\x19\xF9\x01\x6D\x79\x85\x46\xBA\xF8\x7B\x56\x45\x18\xEA\xEF\x31\xC5\xFB\xF6\x55\x9C\x75\x98\xD8\xF2\x25\x8B\x59\xE5\x1B\xB0\xFC\x3D\xF1\x70\xAE\x64\xDD\x75\x7A\xD7\x68\x59\x85\x99\xE8\xF1\xC4\xB0\xEA\x3B\xF1\xA0\x4E\xB8\x63\x3C\x9D\xE4\x4B\xAD\x01\x2E\x93\xF8\x07\x5F\x7C\x5E\x69\x52\x91\xE2\xD7\x44\x5F\x5D\xFD\x50\x9B\xA0\x6B\xD3\xDA\xEF\x49\x77\x25\xEB\xF8\x59\xCB\xD9\xEE\x13\x07\xBE\xEB\x3A\x1F\xEF\x6E\x1F\x63\x5D\x44\xAE\x75\x0C\x92\x78\x8E\x48\xFF\x88\xBA\x4C\xEF\x69\x27\xEE\x4F\x4B\xDE\xDA\x77\xFA\x15\x34\xF6\xA9\xFE\xC7\xAC\xBB\xF4\x21\xF9\x9F\x42\xF9\x27\x3F\xED\xB0\x3F\x71\x19\x74\xE9\x29\x7C\x09\x0B\xE8\x5F\x09\x5E\x30\xF4\x81\x79\x62\x59\x79\xBA\xCD\x1D\xAB\x91\xD5\x36\x9F\x2C\x2B\xE4\xAC\xEE\xD6\x9A\x6D\xE7\xF6\xB7\x76\x81\x56\xE4\x70\x9D\x84\x5D\xCC\xE3\xDA\xA5\x9C\xDE\xE5\x9C\xE7\xB8\x59\xFC\xCD\x5D\x9A\xA5\x9D\xAD\x97\xA4\xF3\x7A\x9A\xBB\x88\x33\xBA\x99\x6B\x41\x31\xAF\xBB\x39\x4B\xB5\x1A\xB7\xEB\x23\xF1\x42\xEE\xE7\x2C\xAA\xF0\x12\xF6\x47\x92\xEC\xAF\xAE\x7E\x8F\x33\x5A\xFC\x3F\xA4\xD7\x11\x95\x8D\x5D\x2F\x6E\xA8\x6D\x85\xC6\xC8\xB3\x90\x7A\xFD\x40\x31\x7C\x20\xD7\x1F\x44\xB7\x3C\x39\x24\x52\xED\x4C\xE3\xDD\x31\xA5\x0F\x7F\x86\x01\x3D\x54\xF6\x62\x2D\xD5\x12\x7E\x67\x7A\xFF\x42\xD4\x3F\x88\xDF\x2C\xB7\x62\xBF\x4F\x52\xC0\xC8\xAD\xDD\x6E\x50\x18\xD5\xEB\x02\x00\xE6\x8F\xE2\x1F\x71\xD1\x88\x37\xBF\x65\xDE\xF8\xF6\xE6\xBA\x52\xF5\xED\xCD\xB5\xF0\xDE\xAA\xD6\xB5\xFD\x8C\x02\xEF\xF3\x21\xA5\xE1\x31\xFC\x0B\xF8\xEC\x97\xFD\x38\x0B\xD8\x9B\x2F\x9D\xA5\x6C\xBD\x0D\xD3\x2F\x2F\x0B\x86\xAF\x98\xF2\x98\x50\xE1\x5B\x72\x7C\x0D\x4D\x5D\xE8\x34\x08\x04\xD8\x48\xC1\x69\x83\xB7\xDB\xA2\xF0\xF2\x7F\xA1\xB4\xDE\x1E\xF8\x17\x22\xDF\xF4\x13\xE8\xAA\x30\xCE\xBD\x1C\x6D\xEA\x48\x4F\xBC\xF3\xA2\xBB\x08\xC3\x2C\x60\xD3\x1E\x93\xBD\x61\xD4\xEC\xA8\x7D\xAD\x23\x10\xB1\x9A\x04\x66\xE6\x6A\x07\xA1\xB7\x45\xA3\x4F\xD8\x4F\x22\x76\x79\xFD\xD0\x41\xD7\x28\xD9\x74\xB3\xB1\x75\x54\x50\x23\x61\xD1\x92\xFD\x21\xEB\xE2\x1F\x10\x91\x77\x1E\xDB\x06\xDF\x8D\x88\x93\xAC\x43\x18\x1C\xF1\x31\xC3\x52\xD6\x28\x3F\x07\x53\x92\x13\x53\x9A\x5B\x70\xB4\xD2\x02\x73\x99\x01\xB3\x53\x1A\x06\x69\x88\x7E\xE3\xC6\xCA\x3A\x51\x74\xB0\x4E\x13\x32\xAD\x04\xDC\xCC\xD1\x7F\x98\x45\xD2\x37\x11\xA2\x14\xFA\xA4\x90\x8B\x89\xE5\xCE\x02\xA0\xF0\x85\xCF\xB9\xEF\xE3\x2C\x4B\x48\x73\x36\x82\x4D\x82\xD7\x84\xA0\x07\xB1\x1C\xB4\xC1\xB4\xFB\x79\xB4\x86\xDA\xDF\xA0\xAA\xB2\x59\x06\x34\x61\xF4\xA5\xF7\xBC\xBA\xF8\x47\xD1\xA0\x99\xE2\x4D\x1A\x38\x31\xB0\x0F\x93\x74\x0F\x35\x84\xA0\xDD\x2D\xEF\x79\x59\x42\xE8\x4B\x5E\x63\x3C\xFF\x7A\xF9\xE2\xE9\x13\x76\x08\x1F\x1E\x98\xA2\x3A\x09\xF0\xE9\x3F\x28\xCA\xC1\x7D\x47\x47\xFC\xDF\x1C\xBB\x3C\x7D\xE5\x22\x88\xCD\x5B\x00\xBC\xD9\x55\x75\x8F\x8C\xB7\xC9\x31\x8C\x79\x98\x29\x1F\xF7\xDC\x22\x64\xBC\x08\x51\x26\x32\xE3\x88\xBC\x3B\x86\xF4\x8F\x30\xA3\xAE\x83\x01\x5F\x19\xD6\x3C\xD1\xB4\x3C\xE3\x36\x88\x4B\xDA\x47\x9F\x5C\xFE\x3B\x8C\x69\x26\x5D\xA7\x30\xA2\x7A\x96\xA2\x40\x9D\x80\x80\x6A\xC0\x38\x47\x59\xE7\x66\xB9\x97\x46\x74\x9B\xA2\x46\x70\xB0\x67\x86\x1C\xF9\x19\x5B\x9E\x6C\xEB\x3F\xD4\x13\xB8\xDB\xC0\x8C\x37\x30\x60\x20\x6D\xA6\x1D\xC7\xB4\x9D\x8A\xE4\xCD\xE8\x23\x1B\xDD\x9A\xF5\x37\x6B\x8F\x9C\xD6\x7B\xAF\x59\x6E\x23\x31\xC6\x48\xE8\x0B\x16\x9B\x5D\xC7\xB0\x2B\x9F\x1B\xB7\xE7\xE4\x05\x9E\x93\x9F\xA9\xDB\x43\x9A\xFD\xFF\xF6\x33\x0D\xC0\xF9\x0F\xE8\x16\x46\x8B\x84\x11\x60\xF5\x90\x4D\xD0\x61\x12\x33\xFB\x7B\xBF\x6E\x23\xB0\xF9\xFD\x63\xB3\xFA\xB3\x92\xA5\xC5\xEF\xC3\xD8\x4F\x8E\x69\xF1\x94\xB0\x0D\xC5\x16\x00\x8F\xCD\xEA\xDB\xEE\xCE\xD2\xBB\x3B\xDB\x26\xCE\xC2\xEB\x69\x9B\xF1\x97\x58\x57\xDB\xAC\x3E\xF9\xEA\xFC\x66\xE2\xB5\x2D\x61\x6E\x23\x32\xED\x52\xE6\x66\xDF\x89\xF7\x62\x7D\x82\x11\xC5\x5E\x58\x14\x36\xCA\x21\xC1\x6D\x14\xE6\xDC\x11\x7F\x77\x0A\x33\x1A\xC6\x7B\x21\xF9\xB8\xEE\x87\x6F\xC3\x33\xEE\xBA\xBF\x60\xDE\x47\x7C\xFC\xA7\x6B\x65\xFD\x93\x98\xC2\x5A\xE7\x3E\x6E\x1B\x00\x55\xE7\x63\x56\x78\x8C\x1A\xF0\x31\xDB\x47\xAF\xC8\x1E\xB3\xB6\xA6\x63\xF5\x73\x9E\x7C\xB4\xBE\xEE\x30\xAB\x20\xCB\xB2\xBF\xBA\xFA\x7F\x45\xD7\x26\x2B\xBF\xE0\xAB\x78\x0A\x9A\xF2\x3D\xCD\x33\x91\x16\x1E\xE4\x8F\x04\xEF\xC2\xD3\xB5\xEF\xE3\xC8\xBB\x21\x49\x4C\xAF\xE3\xE0\x35\x7F\xAD\xBA\x5C\xD0\xB5\x5F\xE4\x3C\x11\xD5\x22\x07\x1C\xC8\xC5\xB6\xA9\xB2\x96\x85\xE6\xED\x78\xC4\x63\x3B\x1E\xC1\x3A\x7C\x7A\x77\xC9\xDF\x75\x76\xBD\xDB\xAF\xC0\xFF\x90\x16\xF1\x97\x49\xAF\xB8\x8B\xDE\xE0\xA3\x28\xF2\x92\xB8\xA0\xA0\xD7\xF6\xCF\xEF\x93\x08\xD1\x30\xC2\x25\xD9\xB2\xDC\xAD\xFE\x2C\x8D\xC0\x6D\x50\x95\xC2\x53\x8A\x3B\xD5\x75\xF8\x59\xA1\x77\xC7\xEB\xF0\xC7\xD2\x8E\x43\xA7\xBE\xBA\x71\x64\xA1\xDD\x3F\xCF\x30\x81\x99\xB8\x56\x57\x57\x5F\xF0\x89\x56\x13\xA9\xF4\x29\x98\xB1\x78\x8A\xC5\xED\x14\xA6\x33\x7D\xE5\x60\x12\xD7\xFA\x84\xEF\xF2\x90\xE0\x80\x4F\xF7\x52\xD9\x33\x49\x58\x09\xC9\xF2\xA9\xE6\x98\x4F\xA7\xD6\xF6\xF7\x58\xBB\x17\x77\x12\x3E\x45\x29\x19\x46\xF8\x4B\x52\xDE\x11\x51\xE6\xCD\x19\x52\xC5\xFB\xFC\x58\x5E\xF6\x91\x4D\xA7\x66\x59\xDB\x66\x1F\x62\xFC\x91\x5F\x51\x3E\x3E\x23\xE3\x39\xDA\x7A\x6A\xF2\x52\x92\x9C\x1E\x84\x45\x41\x9A\xA7\x51\xF5\x81\xDE\x91\xC4\x28\x0A\xF4\xE7\x4F\x9E\x94\x6B\x88\x75\x44\x76\xCC\x74\x9F\xDE\x5D\x8E\x33\xC8\x23\x69\x25\x8A\x86\xEF\x31\xC9\x70\x73\x97\x49\xFD\xAC\x31\x79\x39\x30\x68\x6C\x8D\x72\xB3\x47\xAA\x27\x8A\x15\xEA\xD9\x1B\x18\x4A\x38\x70\x65\xFA\xC6\xF8\xFD\xC1\x87\xD6\xE5\x02\x6E\x08\x5D\xBB\x49\x06\xC6\x75\x9C\x1C\x84\x1E\xA2\x51\xD9\xB7\xB2\xC5\x86\xFB\xA2\x96\xF4\xE8\x43\x2A\x36\x4F\x92\xED\xEB\x87\x58\x0D\xD9\xF3\x8E\x28\x2D\x25\x29\x1E\xFF\x10\x99\x49\x98\xB5\x19\x22\xC2\x59\x36\x3D\xF7\x4A\x02\x4A\x72\xEC\xB4\xC1\xB2\x94\x71\xAA\x9B\x55\xDB\x3E\xC3\xBE\xBC\xA4\x3E\xF0\xE1\xF7\xBE\x61\x08\xAC\x35\x4E\xEC\x08\xF6\x48\x04\xE4\x59\x8D\xBC\x2C\x60\x9A\x21\xE8\xE7\xD1\xF3\x2A\x30\xD0\xA9\xE7\xB7\xB6\xE0\xD7\xDB\x4C\xB2\xCD\x24\xDB\x4C\x32\xE3\x4C\xB2\xD5\x6E\x5B\xC4\x6D\x11\x37\x6B\xED\xB6\x45\xDC\x16\x71\x5B\xC4\xCD\x39\xC7\x39\xDF\xFC\xBC\x3D\x35\x6D\x4F\x4D\xE7\xF7\xD4\xB4\x3D\xF8\x6F\x2E\x7C\xE6\x2E\xBC\x65\xE1\xCD\x85\xCF\xDC\x85\x5D\xF7\x4D\xDE\x26\xD5\xB7\x08\x9C\x37\xD5\xAE\x63\xFF\x90\x94\x82\x2B\x1B\x62\x4F\x35\x5C\xB2\x17\x4E\x8F\xE8\x1F\xF1\x6C\xB6\xDC\xC4\xCB\x45\xD7\x25\x99\xB2\xDF\xF9\xF7\xDB\xAF\x7F\x0F\xDA\x76\x3D\x57\x23\x7C\xD3\xC7\x51\x20\x4D\x2D\x63\x23\x2C\xBA\x4A\x1E\x14\x6B\x8D\x96\xD2\x03\xA6\x1A\x69\x20\x6C\xD6\x64\x4C\x76\x28\xE7\x98\xAE\x55\x3A\x87\x7D\xEB\x55\x8B\xCE\xF7\xA9\x3F\x63\x44\xFC\x83\xF0\x55\x94\x16\x47\x6D\x01\x73\xA2\x9A\x44\xD5\xCB\x69\xD4\x3F\xFC\x4F\x18\x07\x56\x59\xD5\x63\x44\xC0\xD9\xFE\xD5\x7A\xAC\x66\xD8\xE6\x34\xCD\x3A\x8C\x2B\xBE\xB2\x72\x06\xE6\x7B\x1D\x52\xF6\xF1\xA6\xCD\x7E\xFD\xEC\xF7\x7E\x45\x71\xBD\x6E\xD3\x69\x65\xDA\x47\x44\x68\x88\x22\x26\xE4\x7B\xF7\x59\xDC\x3C\xB9\xB7\x4D\xD8\xCF\xD6\x57\x3B\xC8\x1F\x9B\xFF\xC1\xFF\x70\x3F\xBF\xE4\xD8\xCF\x50\x8E\x80\xBD\xB5\x07\x58\x8E\x9B\xC7\xD5\x27\xD6\x1A\x4D\xAB\xB6\x3F\x70\x96\x99\x1F\x36\x64\x5C\xC7\xE7\x22\x88\xA9\x76\x6C\x04\x38\x9E\x3A\xD7\x71\x7C\x88\xB5\xFB\xE9\x46\xA8\x37\x70\xAE\x8F\x99\xE3\x16\x45\x28\xF6\xB1\x87\x76\x14\x93\xF2\x25\xA0\x97\xA0\xD1\xBC\x5B\x94\xE1\x97\xA5\x95\x5E\x99\x4C\x2B\x61\xE9\x8F\xFC\x26\xBC\x01\x07\x0F\x01\x6D\x87\x1F\x10\x05\x4E\x0C\x0E\xEC\xBD\xFA\x01\xE8\x2E\x3A\xC1\xC7\xE4\x1E\xAF\x4C\x58\xF7\xB4\x26\x75\x53\x20\x55\xFA\x00\x3A\x09\xA0\x67\xE6\x44\x67\xEB\xD5\x45\x44\x23\x37\xE0\x30\xAB\x23\x33\x7B\xBA\xBE\x0E\x82\x22\x31\x7E\x49\xF8\xA4\x65\xFC\x16\xA4\x03\x7E\xB7\xD5\x89\x0E\xF4\x9C\x14\xF9\x4C\x11\xF9\x18\xE5\xD9\x07\xF2\xBF\x39\xCE\xEA\xD3\xD9\x1C\x5C\x7D\x20\x53\x68\xD2\xDF\x7C\x90\xDF\xD7\xE9\x91\x00\xC6\x17\xC1\x98\xF2\x66\x65\x6C\xF6\xBD\x82\xA9\xCC\xAD\x23\xE9\xD1\x25\xD7\x0D\xB3\x66\xF9\x00\xA2\xEB\x12\xF0\x4D\x12\xFB\x88\x7E\x20\xD7\x11\xBB\x4F\xA1\x79\xB2\x33\x33\x50\xDF\xB2\x36\xB2\xF2\xEB\x17\x36\x7B\xC4\xDA\x5C\x82\x49\x2F\x9A\xCD\xCC\x5B\x42\xBA\x41\x3E\x4D\x6A\x96\x61\x12\x2B\x6E\xE3\xCC\x61\x1D\x23\xEC\x3A\x3F\x2C\x62\xEB\x99\x98\x2E\x35\x10\xE0\xC4\xBE\x80\x1F\xB4\x54\x43\x73\x49\x64\x34\x46\xEF\x08\x5B\x81\x37\xD5\xCC\xDE\x27\x3C\x73\x9C\xB3\x32\xCD\xA7\xD3\x3B\xCC\x64\x7E\x8F\xEF\x51\x37\x6F\xE4\xA6\x58\x1B\x7F\x57\x7E\x43\x18\xFF\x11\x52\x4C\x50\x24\x16\xC3\x1D\x46\x6A\x66\x76\x4E\x85\xBC\xCC\xF0\x3D\xFE\xFE\xE6\x80\xC8\x9B\x48\x58\x64\xAF\x1B\x26\x28\xC9\xDF\xDD\xE5\xEC\x12\x28\xEB\x38\x36\xAF\x20\x7E\x49\x52\xE7\x1E\x7D\x19\x4D\xA8\x03\x54\xAC\xF4\x12\xAC\x25\x33\x77\xA1\x09\x4E\x36\xEE\x04\x9A\xF5\x01\xF1\x06\x20\xA9\x8B\x79\x99\x90\x63\x95\x8B\x84\x9E\x87\xD2\x14\xC7\x2E\x45\x87\xBB\x78\x82\x6E\x5D\x7C\xA7\xDA\x2A\x9D\x49\xBB\x6A\xB5\x7C\x2E\x76\x2E\x2E\x3E\xA3\xEA\x45\x4C\x2C\xA0\xBA\xED\x50\xED\xB4\x42\x68\x11\x67\xBE\x97\xE3\x59\x46\x49\x84\x63\x47\xE4\x8E\xB9\xCB\x42\x49\x4F\x55\x56\x19\x27\x3D\xB7\x74\x75\x55\x1B\xD3\xEF\x70\xCD\x4C\x07\x63\x59\xF3\xE9\x1A\x15\x77\x15\x10\xC8\xED\xDA\xED\x17\x72\x9D\xF0\x27\x8A\xBE\x61\x52\x5E\xAB\x54\xFC\x10\xAE\x6A\xD4\xC3\xA1\x2E\x43\xC0\xBA\xB0\x0B\xAB\x1F\xFC\x97\xB8\x8A\xF5\x26\x89\xEF\x31\xA1\x45\xD9\x92\x7D\x49\x5E\x3F\x50\x46\x46\xBA\x2F\x52\x15\x63\xE2\x01\x79\x3A\xCB\xA5\x4A\x04\x67\xE1\x5F\xE0\x36\x56\xD1\x99\xF3\x59\xB7\x89\x48\x1E\x63\x9A\x14\x6D\xAB\x16\xD3\x39\x5D\xAE\x54\x7E\x31\x49\xAF\x54\x44\xF7\xA2\x75\x7D\xB7\x2C\xB9\x2E\xEF\xAF\x5D\xF2\x45\xD6\x04\x56\x68\x94\x1E\xCB\x0C\xAB\xD3\x61\xC5\xB1\x6E\xB3\xDD\x4D\x42\xD8\x35\xA9\x73\xDB\xF0\x0C\x0C\x56\x2F\xAF\xBC\xCE\xC3\x28\xC0\x84\x2F\xD5\xF0\x83\x30\xEA\x25\xB7\xAB\xDD\xFB\xEF\xA4\x44\xFB\x9C\xD0\x97\x5C\xCB\x2C\x3E\x86\xA5\xD7\x75\x40\x04\x54\xC9\xBC\x82\x57\x62\x09\x81\xD1\xC1\x24\x26\x8E\x9F\x98\xEF\x5D\xDF\x26\xF7\xB8\x39\x1E\x70\x3E\x8E\x0A\x8A\xDF\xC7\x45\xDB\x08\x75\x75\xCE\x4E\x76\x5D\xBB\x5B\xBE\xC7\x7B\x5E\x78\x0C\x34\x8A\x44\x47\xAA\x7F\x9D\x59\x3A\x0F\xA8\x44\xE0\x7C\x4E\x4A\xB6\x28\xF2\x28\x3C\x48\x58\xD7\xD0\x31\xFF\x89\xA9\x98\x02\x7B\xEC\x6E\x3B\xD2\x94\xD6\xCE\x85\xC5\x0E\xBE\x94\x71\x75\xF5\x43\x6B\xEB\xB3\x0C\xD2\x85\xB0\xFB\xE3\xD6\xE7\xF0\x98\x46\xE1\x0E\x3C\xCC\x56\x4A\x24\xAD\x6B\x3A\x2C\x06\xE9\x2D\xE0\x93\x81\x8D\xB3\xF9\x28\x53\x77\x79\x01\x53\x4E\xA5\x60\xE7\xD3\x70\x33\x08\x30\x1B\xEB\x26\xAA\x03\x7C\x97\x97\x39\xAA\xA0\xC7\x5E\x35\x51\x57\xC7\xAD\xE9\xD0\xD8\xEB\x79\xB3\xAE\x1E\x04\xDE\x2D\xF2\xBF\x79\x3E\x4A\x91\x1F\x52\xF0\x58\xE8\x94\xE6\xED\x72\x3A\xC6\xC1\x31\x47\x72\x9D\x36\x9A\x43\xA8\x89\x69\x9B\x0D\x31\x4F\xD9\x53\x8E\xF3\x0F\x91\x93\xBC\x26\x8C\x22\x9C\xF9\xB8\xCA\x84\x6F\x92\xF4\x41\x1B\xCB\xD6\x95\x1B\x95\xCA\xE7\x43\x42\xE8\x7F\xC2\x2C\x6C\x3D\x53\xD0\x87\xFA\xC7\x24\x73\x26\xDE\xDE\x02\xBD\xE4\xB2\x1E\x51\xB4\xF1\x41\x71\xD5\xE4\xBE\x69\xBF\x98\x01\xF5\xE9\x67\x71\x03\xBE\x4D\x0C\xF6\xB3\x5E\x0D\x3F\xA7\xD8\x1D\x56\xD1\x17\x1A\x56\xFE\x6F\xEB\xAA\x27\x78\x0E\x73\x5E\x4B\x4A\x15\xE5\x82\x83\xD8\x5A\x20\x0F\x0B\x73\x79\x78\x3A\x1E\xE9\x19\x1A\x21\x5D\x73\xE6\x30\xFA\x23\x1D\x55\x1B\x49\xB4\x16\x0F\x1B\xA6\xEA\xA0\xD3\x6A\xA3\x29\xD8\x33\x19\x0D\x1D\xE5\xF1\x32\xCC\x30\x49\xF4\x29\xAC\x7A\x72\x1A\x56\x32\xA9\x54\xC6\x2D\x99\x54\xEA\x1F\x09\xEE\x92\xE9\xE4\xD5\xFE\x0E\x6C\x96\x2B\x2C\x16\x13\x65\x05\x56\xD1\x7C\xA9\xFC\xAD\x2D\x5C\xAB\xA9\x7F\x35\x92\x5A\xB2\xBA\x9C\xB8\x5D\x12\xC0\x9C\x5A\x39\xE7\xC7\xC5\xFC\x73\x48\x2A\x9D\xD7\x92\xD2\x54\xBA\xE0\x20\xB6\x15\x6B\x3D\xC2\xAD\x4D\x3B\x37\x12\x3D\x5D\xCD\x85\xF8\x64\x2E\xD3\x99\xB9\xC3\x54\x39\x6E\xD2\x58\x15\xFB\x1E\x6F\x70\x2E\x20\x65\x8F\xF7\x60\xE7\x91\x72\x70\x8C\x4C\x62\xAE\x81\xAF\x57\xCD\x6E\x3A\xC3\x1A\xE7\xAC\x36\x9B\x22\x07\x4D\x22\x68\x9F\x77\x9A\x17\x09\x86\xBE\xD3\xDA\x58\xE2\x58\x9E\x93\x7F\xC8\x0D\x83\xF6\xA4\xDC\x68\x77\xF7\xF6\x1F\x72\xC3\x20\x19\xDD\x68\x3B\x5F\x58\xBE\xDC\x26\x82\xAB\x8C\x0B\xEE\x64\x3D\x3B\xE2\xE3\x31\xB9\xC7\x2B\x91\xA6\xDB\xBB\x42\x2B\x91\x74\xAD\x43\x2B\x06\x41\x96\x46\x21\xF5\x6E\xF3\xDD\x0E\xDB\x09\x6B\x87\x27\xDC\xE4\xA9\x5E\x45\x4A\xF3\xEC\xC0\x84\xA9\x32\xA2\x09\xFF\xC9\x93\xCD\x50\xDE\x8E\x24\x31\xDD\x2C\x65\xB2\xD4\x93\xCD\xA1\x3A\x99\xC9\xC9\x9D\xA4\xCF\xCE\xAE\xC3\x5C\xBC\xF6\x59\x7F\xBA\x02\x26\xCB\xD5\x8A\xDA\x6D\x26\x3D\x07\x35\xEC\x55\xDE\x4A\xE5\x5F\xD8\xB5\xDD\x27\x98\xE5\x7D\xBB\x83\xAC\xAB\x76\xEE\x7E\x7A\xAC\xC8\xBB\x3B\x28\xB0\x84\x7B\x77\xAF\x0A\x16\xF2\xED\x1E\x82\xAE\xCF\xB1\x87\x29\xB1\xB4\x57\xF7\x90\x7E\x51\x97\xEE\x5C\xC1\x2D\xED\xDA\xDD\x05\x5E\xB1\x8B\x0F\x52\x66\x35\xAE\xDE\x5D\x0B\xEE\xF2\xFF\x7F\x00\x00\x00\xFF\xFF\x9B\x75\x3E\x10\x44\xCD\x03\x00") diff --git a/vendor/github.com/open-policy-agent/opa/internal/compiler/wasm/opa/opa.wasm b/vendor/github.com/open-policy-agent/opa/internal/compiler/wasm/opa/opa.wasm index 6840e639bdcb4f71f9b12fc52dc8f321b9c40114..302e089cc46bf3c910123bd43da7a4a4c6ca16e1 100644 GIT binary patch delta 79521 zcmdSC2YeL8`#*j&dzW^XWCKYEse4BYy@w82q>2iNfMNqH1{;W72`WfYimosq3ZaPw zML`XyAgCxRQdAT{nxZJ6fTE(P{NK;a?%rMqSiYa%=llK3%iQhk%rj4)XPzmyYIpUc z-&KG99HzY%c*W|=+E^3V`PMFWxt}wiwyZnrYn6tQ=%nT+D%vT_6wWLr|=nCIbXqNYRma6 zd?tU4&*G2ra&3jSQQNG2pl#5WYumNYwH?|{ZJu_Ye~f>u|FHIhc2xUWJEr}n{jU9~ zJ>y^Nf5N}W|F-{0|2zJ5{&)R*{0IGq{6G7T`Tz9K4$KLZ1?Cq9o)5ebSQJmB2fU^HMg-nwyZ5#Ea>l;nSB}UE(^PxpM62(LRHh zM7ZHAf`+DVu)>Wqtj7{+bX`u(%sWd` zG#_U?sK0qHfzb?R`igjhp_xAY>(SIJGZRYmujnlhVud`RYqHOL_r5WsefkWmXMCL$ z1xf3c>_Z2etWojR$2rmNm4uZiuFX&E%=)k z?CzG#4N^%~sApzW%1rZ-h7|IUzRS8b)RKK_JsYZ9BW?NO01Wp=^4xukT zJz9#c?&a-Rib1-SQlh^mHPP2yUr)+e+Mcme6AHtNHRzkc^|kaq!n-CjYJq}PRG_~? zf0KPM18!1ddJMK!jWu43dJgd>>jm9JOT*`F+cgY^Xm-PKjPcSM;DEJ1HGQB|t_Yw5 z5nohuIS__*gd`-#Iw00x$T|RgQ`AbN25*=+28hm}QedfnYpqXh_D^9?xUX{9Qy-`R zyT9JFmltoZ#SuRp+%+KT!F`e!KG`?K1+IK+ZT@%g9y~R?4C}JA>OqNP81m4+vHISc z*%RpOBh5pB{ETHetX;7jXrqQgoSR_!Z(gS3)h6q!v`CW8Y$717N~5sq3|>w7i?E*; z^I}7Tdold@qrYhlu9BYXfqGm*g&&ijen6dCB~s6Z>eH2{(V!CGkwV*Ve-e6Q?y-ia zr(1`s{O8CV8_Q21bW_y|WM)_|Rz0c1>Xq=7@CM1>r`JgP{Uq+SORw*6&kyc3&B(CE zrdI@8wHdL_r*Eg`^p&Zv$zaIIa&-I4s_$X(xQY}>RN&M}^HkS0YAV|ebeCOtA20xs zQ^AR4Mtl6-nNi%qThF$BjUYI#fF7V;e)s~7Mzxj7g}|IOeWi6_=18{7 z+LYNE&=RY)O7lRY4$)uVY7OviRJHIBj}pt_jY+*y9Qg+vN+qssAo|HZ?QdkD8y?Q8 zUizJM$cJs?@I2{1$T#21zdy^rQZWe}z#OeuV)%1(PjiqDM(5~1Rt_b25dOW?@Lgux z41rYi{+Fp#bexJx#RG?KMG}!;fmoYLSuc3A8XiCgcr0m(luDsRHChQ$i9$(Iaqa*y zD}u^2iX4=jlai6+1ByIohT-iUS3x7I!_Z3kc{WgyhV-9Oo2z3a?UlR!Q$@w7P;zP} zs+Jxa;URWrM@l6Rb(=S8xWrg59itlmnb!_61jm?NT`i$==|FPB(gf%=hF3CA*escx z<2C#x`cKkJM8{aX_jj$0F8@%WD7^-+ZP6${O+z;%vB_GPQ>0KO0s2tn_%A4eiy;ND zHhhUYy{io2{>s_O5dz+XoE)m!b`0?H!6O94o+~4oD(;xrSH3Z`rlA@)?kOpmtuVjO)7EVOa)50tv~oXp|RQdZt}B z%UiefCbQs{%79c}%6iy!3*yN|==wIRQJrq-_P8I=KO=E~>3;oPYkZxKDW6F3*!s1_ z`nb+Z_J~zllH&Id>G!CAP)*p^t9dOg)iaIZY7Eg-XoB#CiQ?!o^YSN0l9*r5zHJGh(J*QWLd6g3r3X zZcR3CnN_zjV>7Kc>;23gwO*-z0V4Qm(1QgRH5HW{aYP=$1p7PoGcZ=sRR= zZ+e~gsUbR?*iOJq7A9k1m|NV}xUk1$0+em=#fBd&F_Uz=z^6~3Hz-~0D^cG|WEJPT zW`8jQB@qVY__1+O>4VC8xeD-6{jksGa&UHl#oLn@I0l7l9BdIHblgomxf|yJaSkH0 zorFV}<}OUS5l?M*o^H>;F%KyXbJHL8W0HjM7FuNZOK4vFsZB^i3E~`XGRhgytetWO z9Fryb9ti+GTo>1hsSj$*O+J#^M<^_oF#R5As1fmFt|eQC12pAgr35(uu)(0dM`DgJ z3cwZUI0+giL%`!qjez?j@oN4Rap*f8g?NWKAE6mgbW(vrR#I9TNl;Tq@KG}o6yZUL zK~f$d$^#N#qFf6GqkvAsunNswh7naBgorrn^~qU<+>mUNQ&@>dlxsjG$~NnqW(C&j z$?3hwhn$kJ2|>d~fW$>JTO(foka!ye4?$3z@+B}hy#bt0g#uo&K+`8JJEy3Kv5D5$R?YEuh5Y+ntER!d&y5-F zgUtQ&kRUe&sgyWviIY)GmMrC&qlmr72H=?_||Z3eMz zR!N&T;3gWjZN;9qu54S#mROV9)<^j(ZF{j9R$RMw^_LQyNP<~Pk@{2f(XBImh9ZRw zUkN6T@kRpJvdp@+-P@(ld+To)@2T%I<4VE-8O zx=Jtt`U})H4AI#A(ql!#Dh*LdUI~j%ECS>3WNdgavyhjMz4L5a4@8+#c zP6M9F?$k_%q5;`h9AuX0WKdoldKLF~@mW+n(ET1T&tuz`4KmMUY}@=nCbJfvF(O5c z50Xz@pJ3JMQV(_dbO~|xzIAc84tSW=Z5D&_w(2n!4=a1zg4W9>^x@1Z>eUioukAI0 zN_L&qiax5Hg9H@-^p+nJ`p}1UZLjTCYOgxhTjzF!O=BH!E^Ee`B zpE<^^`SHM3z+~m1tyafDb*$h8VYb$4ctLBoY!v!jImlpX8>vBXipBz^5EIkiw!Xix z5nkq9)R%JlH(cD%o;PE#CqDq0ZnU1gxM`D*sbM4l-Mu^Prx@4=+xncu9f{K=RE$?~ z`Z_AXo9$N0B~1ah_>vq}XmfxdgkfNsX*l7G-$BJ3oq%EazDrxMjn?u@8zf0%&9`~xp-Re;yslFOMQy*ExA5j$2{OX_$+gDn|C2eSq;`L5(5toFh z(#6qIiM1+yFj^`xRHX-PTmVd>WBLe><*H};`h4{3llWSSeyO;Oii1Hs6)H&tk8+}E zEA{gKP!6T0B&mr1cjVxznIi zkhVcQX|sqU(xc{&Sm$3A&Q_B!ViigVVZ78wmO!U1>#FuJ z3zV+L00T@FL1SE|K(%L%UPd#7Fh(g|PDYRks^bG~K9j;bQ;Z%1FYTn35r3Oy)}(84 zN@Gfh-V%cqabK_@Rwzt(Iq68H;@D#+NrAlBYRMdo^vI!LB?Ps>F< zMX{`{8p=2RV5@21!58PSHwT20`s6OUTPQni};uYR4i*8lQl5n)7uEhoJ#Ku`7Q(n zW2!X@JZo*>fn1FFQRRu+Q(H_1OxD&XQNc*$PLG2LHeU2aViK$ZQdEYHM$~D;3NZ$+ z8dkql;()9zfyc5b{vkluXA*QZP1lt8Bfb*T*VYH6ML!ZG5N%(gze+U$8ndtV2yD!l zwIZJ`=kb{I+o>^xYK9N#vWUMk#6LRWE^Fwusil&I3Y=V^MSTN?)Cx?;RtBd9-Nc~z()|*Yb>i9{-M%4; zD5xZwD3@<1(=gq3w1yCbwDi|h)o2Zb333LGRLG?3dK8@m>Qo@m34l0FgV(p~uH>RY z2Z<)A71o(OoQGmeywBx4DF)JSGL=WTp(Yxk&OvL^_0?QUbQkt$a_p%u<3cQ|6E)x~ ztt@Ds^}-11x*`WqxlCxYheZC)gHbs+P)$Y=4gu%&Z&}x;^|5`SEg~BS$cPA&NsE6<*k?DkE^tiO zRu8XXl9a3_!!w+fFtYjH!&?_t(gcPt-H#W8->fCEZ8wf++r!pK8q#cS1f9wgL>Drl z7PJCJYO#$h)*Tn@l}K9O?0ZvH>+%t`{w^}dt>;I?5b}4Bk;{!9WEMPhN@N-YWmY;2 zap(uIlxqx=c0`F56wz`$kR~BV-B|2uu9pc%sG{o+?1mmZCOzuG0||4i?q*npBdgmu z(B$-hb=F9urHca`A>n`}TpI^h0FEApX`F^}=##CvBh#~8h2*4=0VpJC7&KyvwQ*!7 z=6GL^eEbqA%m^Bi1_m&UIHo0QI;Q1_GR8WoiSp~N9%HG)%Y>h;JvV1uV(Yw(xWh!d z4a~^(X0l7gIA_7yHaQRDp0BVxVx4zOqyMDfe*6o;eK4*P!8N?q5!`=7d4B#296r9k z5*+rV<_(NlxQeQ&J91QT4adCwfX6h>k;OhZ-2-yWPmHrKYa>42^01`avh%&Q;E^S*=3xv{9fGOaO2K~mWx48eXt1yeNao-4bfOT0|K zAwWHFM?p1}V~6LbKt=rnKFh7}oei>F=1Ds`MLZ2hUS#QZKOWawr!-NHH z9n<@EhOEE9^LF%Tx27)N>eNeiK4s$NG$?pG^OxFUB|Y#VJX&itxhvhXLyvJE&6wnJ zpaHI^EIPI?Ng{uBB~Pm+jfOuhiD{Y^ivx}!)434in^b9MlQ8g_HFJ<(kiz(|R})jIF)CYdUe z69dApe`q~=cOlzpy?J-NdUDCEbCL=e_34C;y=SXJqff1Bqng)IqE$2Zqm0J2p!>3-C8 z;X+Vg>K-dp+Sw*oHB+%8`1PT6MQL4-JGr#+X~?b6=u>N7X^}0$3M)x*F)K;lhj_WH zB$L1@8g>K(3P`?^J8nydX!(eSP9uO74u=V(A+G#s z?IMW8s7A>IjapZ}3Kk%AX$+98MiLk*m7`V-q|(q^`G`7&yp&c{oT`KZa_XV5Pa{`| z`;w~TP@g5>Yd7-JC+>os*=q$z=pnpKrRF7e~eM*1T8gj4E z&4x}=3^;B#;D|a}N~&V_2B+h38xrhV>z#W$WJnE=v|yi4xQj*cwYN>IYlyZQ=kANx}tK<%uth4rvZPik$ad7KDp$qGuqo#$4qD08k zCgAS-`>_wL&G)woxfp>FR{FU1K(z0;s`fT+P)DhLw~ilm?jF|)O#%iJ z^YQK3x@9+xPhtFU*|O0SUeZe6=E|YO4BgTk@4~hV=F_mnnf#4Md&wtg#Uw4Ln(I64 zzOhC@nH#1*L_p-Q2liC>;X7?YIAJMx@T%u zg$-NQqN$_tqVS>m_6xbsmf8(@D1)uEZhfeccf%b+b{nhOF`1BbSe~;J9g6`a`U7@e z3iIT2I6CtIZyto1V_q0TO9wOomeo~`5Zs}Ol5FsUH%Gt{;U$q7&%m-^!-pGTW%Z(m zbE>L6J2U{>J9@A}Kk;G5?YhKoc(@seXI5Rd!K!byY$3NLsBHpkW&#aoN)y+a-4%|L zGc^O-Gk}|#;lv?+l{>+ov{G_hLzj#t%{OI7*p+N;w$dS^uW?3dqSgA5Yx6#*5es@1 zq%2^-h(}_KbPf54B)!a9^GIVz@yH_$A>)E+*_{;Rn6X`v09QcSXtgpSJpp+KDw0bQ z(pP};+orXz>taVkwC8~a>42KD03VK#V#l=RaLZ}aYp1)_q_0t3S-qyWu5vP;tZCD0 z0Ou9c31>Qc^!xMzdl+!I?Q%rlX|)iYJ4u%XEsQq;!y+C!Bqsvcmt`l`<6v?swXPZX zu6BuGdi9d%TFqXQA>N?{T7tS^;9%N}aJ;kt)N3=N@hJsM37jYgWzUcWl>*3PGjd!Y zt7f#UXc}y0{bbwi(yR?TJ=)A{ZU-?jG@p8JdRENdbWDP=J$>=ag82DMNclduKAhR7 zGG4cM^s-tW%2||Ga6dM$@C}lG_0dREj})CTOrl3>Dkr`X`c_8bq0ti>kTT0T?d8y6 zoiVGpGEAn;Y6OjbYgXUd9(?@Bs!3TmYAEGGyJVEH^nR?MA}{ZHtXBGqgh9mKrDL0) z_0nUtE9>{W$Ee@j$9vWA_G>E)CabQ(kh(|Dx+d8@n*AJZ=CjD#$J>oVth}vC|a6nu6|C6Ey61COH6;sYEjmpRI!eqTmxQ{W({yym?Y^6P%MX4&L*J0 zMBJBy6-Lcr5$>S|>?Dk+W(28Hn;LvmrWp2HyUH4ZrsTQj!^t(6-_Ww=w#~2yAeHzD z00CMyv!0wEZl=)CD#cdQUzTbR`_?)>zh0X1nedm8FR;MQup8$&-e%dn0uKUL|Le}n z|3&AeRpB0Om#THM_ZNH0{FRjPO3DJ2l*Lz47FVGx1i6^ZF0}4kmUWU^M^JD}hlAU-6dGFmPaIvJgfDqlK|)KR~x z{_c9MPLkY3K||_xOJ8ZAu<1&Ab^c2Cm4;Un@M@d&^h(;j_T|d+P@(mztMR>T)g}0T zVpSO5!>>;p=jFiCw_irxYu+iV zBhL#^Ks+<5wosk=T7t|hnJbk}E^D&Rpq+@0Qr;~UuvRjXU`SPxg5ey66@G7uEN=?n z;r9$nd%pxD&Gqj$1jdiOPxNegpXjN#p6GdaJ<)SyJ<(J317g|ZA6!T*`*j1+6WK`g zY~0u@Lup2`+}DR732`?ap=i8`01n+m09S4zfVDOgz%iQ%;1in(V3Q9egU(*BvQ&%ZlR-mhOTYBlLs|@OjD!ggc_~cSsGXv75FitDY40!oD z{}U3~p-%{-!cPgKyFVq2zWS6f>avwEx@s$l?1!xqqtG_OsO2^@u)#naI^6UpgvO{4+Tku_5xpK!?064H?Fobs2&RM|Xo1HcBH+k2< zEL+nVrEL+!+MDm8+;{EjU*V;*Ghe>Usz&Q8uw^>-fF&iGPM`qqvUX3bB7=#DX2#Ke z6wNdsq8E2RTvz2@;}CLV1SLh0Y1a41fYm|@J%fgKtsC}KXYW}P_RImwb-y6z*Yk@` z(5DGsd;p~w_$BdW^p`V1R`y<-FUf%rM2YD!_;T~!dT3a-w~lO*Z+*RY$=`3b>+2Ti zJAGd}H14yHob;%DwG!VaEX@!oS-h_s%6{Ben|lA|vZ^t}a-2ZWw=J9UO(sSQ>xpka zLBDtJCv>0K{}dYZIpBzql!oF6p!>{$20(Y~fqr4hM+_%gUG8rMUck*r&IWTaY+$Fx zHtVc|xlo{M4%RPF{W1d^8)r= z#E&F5X(w&jp-w2st$pC%dv-9zDxL--qrf~ydJi_zzpaZB4q~EQX1v!okcZGed7?2 zyI>ZtBnDFrNky~P{7iV}{8FpHX@)s8yp@urLV=#wT9^KU{oK~0zmOOD^%tTv=NLAQ zTYZmZ;P2JP@)MOpaqLJRelXX1`dIbGwu+g4QZ0M?6L|{S+6SF8u#Xb?z+g>B<$gF; z6R@iNN`Kq`+5~^E`}MqfHnFNYyhZRfn^~mV5k!KdSw86{4*Ytx?K}`LdW#2JZ~fLZ zRu{2ziA*iwct6foSr`40W4H7xfPM*Z!XH<_n7{v{MY9ob^Bim^-Bev!vnAzLIG#<46mM_kHT!}z!7jv4LKpA=z@W$8~+f-9TmDU9-6 z2gD|g)wdf&=Fm&1thqupk#$3HRN@7~o)m30)?8mC-{a*2PpBU5)>yN!TN(od=e1(G z%(AN?(Txeha5^LerX}ceL{&c<#Ab`@{j7UL&g+0JMB&VuO^Z6iF%|_tyv|u2;D5r; zGC^==fHkVB-jEOVnfsnos0|FT8eq?z0oE|eo;L!lel^7r!j3XEaViYf_X~fVmy_M% zSlwn{C}4^$avR~pigH;JggF8Ra<%C9Tf5)6aV(t)@p2p+&7Kl#5?F0|axT0ne(6mW1%p6>+#QR-3*#fuJM0Acyy6-gfg4Sl*)Z>&ug@+_wXij8P-A*lE^nKfhcMQ#eaGsy-U zK=IrZ)(L&?NdfBfMYUAc92E3QWnEdBcqo;%!n=1fZCMB+KF{l*iaNTtqQPZ;+{-akWFXLBsL+O zJ%iWDrR_7|h(24)t9lZEe(5IxpjcD^Kmg_dz`0idXqe#zFh7GWjOn}vNL!f63>3bd z>8%M|+a|&D9U&2f;>4Z|Rv_k7V?AP;=ZJP$EGffgrBgc)lwA+XssM0)OxuD0<)P<@ zP1V_8aI9%Iy9n#Jk<!`7As1gXAtYMvluz=Obv>D6H)y{!qY&a2O zY5|)AmNhZh1n}w=Lz0a}2~x5qVfHu*H-sgWxQGJ<6DC*+VhfA{*anaOIz=k9D-a(? z*!^hyXHD6-s20mlK?{t^2zw}qnIm4W!SYdKXN}6*W{C%CvJo)=8biUZss$gvRA6DM zA>OU5C7XUzOQPGnHb|K*KB_Hy4Ax;aZ1a*Rwyq=VUS5Zt4bonyBN;TXP}ZGN2-h-4 zyjsZGRR5iLt4nSYx(FobP9$Pga$Po;EiGSCmo;JNXLmgSEEB={pkj`wSD$?j#Nr#U zkMM4L16HrPGZrS$mJ;Ha+&CTr<@pU+9kxw$Ysl8%BcT!N4M68NVzuxwsu44Tc2kM? zdyORGKdD+>8%x9wG?s|hXu@hVuq7MkOx&DqNVAbCfHu}dJVJ3Vh#1j?b%?i}p?J9o z>jK=5H(@z#6ck{IrO0&K)%na^8eMHCNbhJ`DKA38v_|er1~t90k_eg)H0#q9fuX#?)2$770GGbslcgN{OD#9#@U^c!;KnBJmeZCwjXq{d8 zXUalnDeT1X<}43%Oli(~vP0st=B$7E0$DwjD4iIBL9;#7v~vqq6JGi17Qp1F7}o-= zW{ZU_SaHEovYu>=17lU_=BN25sKusA+0x&#(n=I*MJrv7N`SkdC3=}H-fhXwNK$Ym z2oT;rKr!S|F|bJXd~*@&hWf7+VLVtY5?Zlb5K*fYTLB_=wPI~hSiLo?fw7`PYd~Hq zMzv-gs@h|wJWmf>hAAx&8(O3GQjySxb*N&$r!6yP6ZLvv8+Iu`m$zY!o!68@f~%6P zkF{avB)TFIz1xC4i^UCX5#r7kTs5>%zFuQgJ8L0u6`qq$^C zvLyS*w$SzYBBvdTfYy%fU_5ig4eg}BCbeT(rLN-fhQ4B3@sL>TVQU@w+vIqx;P0s{ zC@)+a?K4NSVG&JfHjLEIi%CDDi;+G&IxAE$J9cP!9mG2|<)Lv}J%yAssEnR9G_-)h2(O_8uWGQZy2ZaFhdh;uX$4wDsZiI?b8o7?WTmA;ajp z7b8)H+OiW^(zTHg8PtuDL>z20@Eu2hT45Lp0e?^lA}D1mM<&e(A-{k-4~qD;)SGNL zG?FyT3|)uCGptT0n4dhogY#hLs~=!*dC4WN5^>m&6aK8WX^#maLg90W_|-#LBiL6C09p&G(j3$ zsq9R*5#z8xeFyo#2;IP#uEZ4yDei?#y!joDUzf@x3rz;9c2Wyz_O|L{>G!WyAMDeN zfqz5w9n1R*0|0$+6^hFF;-XG0uM~^}CnE{NV3DEg(i?`QW9{02=mPR>yFfY>m*> zeJ7b0i&Sg{0{U_4ljyoESJATJNec&g?9%Pir zqZq0Rr@Z!qQ3=p11LNgy9~WMRA~ z>!VNrh9ALgEHq@$-9Sd1dO@H{Fo*GzmB&fUBh1Dfx0}E}_-PnI9+f_*Cl>;BR2rc@ zuLR2OUFJ~mWNm3p{+EOUY1q6Q7)2(9rOpru;>s)eV_wJFa$QN zZ~-{ll8BibCIL>4!S2oOM#3k>Rs6Go1ZQyEN@dXDXoORLh?36iY4)ecI0G|>KgFOk zSZ?q=xh4SA^!Z{fmCqJQT`(7zFFJQ&%|dqUs3rlUyCA6iL(J&Ho~Zl{H+COBdUVGm>x8(vJNvFYz9)M|Lkzj>EcO*XW}Yn* z2~#hpX;?WshjryB?cRs2#Yb{qb{&#=!>^Wev`720M$Wb!cynZ^){9U2vKBx%crNRX z$ydK~+1P5|jUg9@5VVPXpShd3wD+tao4HR%C6YS{w z^H^pz8w(2XZ6vVo7_+;a{w$oVhHtK%zBn>~)e5Qc8M4LYXyT6km{ZIav-?W~_Vj02 zIb)!dg_KTLJcHRcA>-5Z!$cA8 zOjB9}$zr}isgW3L-yBdWwq5LKf&df3{F3-IV)(YG%0~{@llatP1cF@fVC@*fK;BLmt&?jTSP7Ca#mjDN;Z*! zUyomfyoPH?Q?Ev1sI0uvP=@XQlgi^u*fc+zDjvU{ZN$fQH?Sl4xb#N0L49P3F2mU( zd_OoGNs_7J-4USeA(4C&D`5|p-*yxG2(8P{x{cLi;LokML*M3=zi~SosWGd()!mG1 z=%npv_6}nY3w|$h!}G+I_p)cvsOo*JFGuYw$FiR|d#L=v@vJ94#lne5t;{RG`T@2_ zQz!CGUr@(7Np!cbD%c!Tkm@tlZb_L2KwM57x4$RRcPWaX{Hj*M(#h}1Qj@;TBY_7) zsEP-<$1He{cxMV`yA#X5nt};_+IXsgOo!Rie5w?dGNHW6L+nq+ww0$@NO-aB#&r zti{_c#Ep-$C*ti_B5yWCGfy0x&9dAI!8zRiw>$_PAruMsyYlDr7BYZLEYeVKJA zd_=uVzBiI=;&&JxlaG;<&c^sfB}U33#MQEVhA_(6SMi_Pq7N-+kp`OIYvtzyG#EdPHDoSg!ApOvhrbS{x)PxhG!)Es1Scp{1{cA87V zK`~g4I8WV1iK`z=5u9lzHtrG$f(8)CBOGtr3hwhc6V4WUVDWaE99|M=B7y-JAR>t+ z<$+aLzG2hK+r17?1uZXK&H6X=AZ2s$G>8fKp9XBEXtajql4_iVPXN4T4T~@)${&1# zZ2ePl;!V{0STuNx&0ssk+P7FK+fm+iEei2D;%ydYkr}j#jJZ65;zS^b8(fq0$BFK6 z94{t;%n&cVgRI(&@~_`veVM+Q-pR3KH0?RrS>F9!)}H5nC?yCkp=>vOmlT`yxKqq| zpS5Ny#n$&(VbAsS5`&S6;{+UvroU?!P(XnF?KrF;S6T+dp&t?knH#Y5{^tZuJyzCzz2dMfhCgbXfVL?mwag!n>#hk!#%AyMG^ zHa#J7TUCB+J-ddnb>gxOtU-<)(^ALOl;mq3TL7r=oKJ1QLi-0IVI#X5F73vRSl@k5 zjNQmCuP?nd7A1}NVJ1%IP>f=CN1b7Q;pwjGCUz66b}D2Z-^^yST=f=f(#RJisI~;W z1P8ZPl@I)gP2`C#{$l?YtPRc(`JccEzefy!JL~WXoDo|u{@eY)hKOR581fkw z2-b@iK4ba%x_QusFoWc?NcQW+uFsJAT`#I{!_dD~wA{wd1<>)^SOkU7Y-0t<9}&k4 zpT31Y#Y5X!uJ~~qOG{JvLj%I3$+J)YL}YAdjnG|}?W}#G+e%D8X=xdCi{QcVolSFj z(|*QW1<&x~+$r%JzFs(sYr^`RqL3SN>%7LH#mSD$By$9Ax+c5kZ( zVowlAqAST{oGT`ZBmT2M;e(G!0V!6Z6w3lsPaN98+N8ck#qwM@ZTVR%TJ2={)ot9I z7TDbg;KT!Q{Z3|7bs5NA28y{mS;pn|i;%&4<;bNcFW+S47`~lc%P?duhjo&3PAyY0 zulg?bxXXiwjza-$9*F0UBLrS2j_+dO%r%6&47nTnQl&$YBp>+!%a1%`#-$7+IcN%TTpqWa;k}qAsu}R0%U}AG_Q}toL6c zhE_3_sk$W9tPC-+XCJFt*9KZSE`$n#PK8R|H!!%};*4)t^VB`AQ7{Pc3o-T^R+oJ# zmVCoL&RIn79%MsfV4Tt=p(R-nuhfdT0=oYoYu!a!QML+d?DINn ziRqXQB9_2n6E8+c;TUJ}kY*oDMuJW*o*G=QGjt2%O@ zsIs_r>JDPBxw=W1_I6`VbJYa41g3DQ~31m!Z8$+ z5KKzV!JA9rk_~rbg!vkyrFE? zT9NxRGu`ew9rFw*mmFjHZB-XEm?8c}9GeQB17f-nXJQC2Q%iJb@C4V8fAP?@{bzP1 z^x6D{*;D&=C5o3 z`&?Z2EApTF#JpeG`DxORkdH?i>w0Vy#?G1FSbK1y$8YSa)}x<}DjQga!-o}B5Tz8_ zS5&{Rf5WU7F7!A%J69@THY)l^|7;kJ{#6#{Y&>4UhQgxG??_ND5uJW#1KHO?{LUVB zc|>u=DLwPwQMdd4V67^PZ`S`I_atTi$#VZaBd-4w9_3px2OloRpZtj}fiZzu98NaV zv`*=N>IqgYOT8g55q+_4cL z$9&c8NE)O)cAeHB^B8wHM7yrgGbz_vaTVijPYwzwo&qUlVg=*I|CNF23eIn2--#`p z-{P9Xxf*ZnK0KuHw(i4+8lSryfJg~cU_yhYv;IRp`0MlPTN+$D+p)CP>PveF3( zl{$vOlX&yS_IQEK>e%&6+DUy?%e( zbK66Q)`+HAoMt*QR+-#}XNyrGUe9ef_!^#7DNehxc#KDo`Pg-$T|*bAIAxb~cfwlH zr#iR9&Sahf7yMl^?}Cq3*}T2DK7~`zR+`N_hK573zHn7EgQdP3Ey3`B`av76Cwa)C7}(enY`M`UYa%w6SqaD%>O%H zdVeNw={n)s)p%33SDassk8s6SS-AS&Hm$jk9jvO2sGlv(38qDh)UYbD7?=`$z^LdsC=ksjm`+501M}2SZ6ussaWr@T2Jl83&Rp5NjD8HaNAHrP0{%_jj zU17e`)w<#cf6{$86yay)?I#1KasF=yu#eQ>HL5!yI@-lG^lij~7d(KP-_+pOHM2)m zf^E<>Q`mcnnn825Pj26@Y-GBjW~HNmsOgRZa%-m#JFjqurPy7QkHjYB{+&K@ z~u%u@`IRMk7wn4OR9m5 z#i~r{2^O>YF#SuhrXH{6GQ2XMm{-5DaHG1ITAyEk+J^saXL_SJ+K@MZe+W0?o$xWF z5$}wT1&w6X@~!%4)L1^<)L2Fsuc!~!L^{&5oBZvtqe`XDv&!Fcp0Twmcb=kNQ|Zr( zno75Mep6lwQnogA+-9Z-H*@@7rZ}$|&%MgH%#S$dBGO6b@Upz)SY~RyM&%cV)BHrBFnC%hR*lV~}{!kI8 z$@9uqJU3M?j@qs9i{RqBBAy#B3xoT`)Rw&A-vm^<0w6+4re6sq8(tjqi@Eh7m&0{= zyvS|MOBC?hK=8TNJeq>4%=9X^pxURkx?kJyaF%Dlz`4eh=y-wILi@;$r^dK>KW-02 ze$}B(cKop@iP`OW=hLG5{|6GY#gdLZsREsE zcf`{5A7Woez6#_%(TS5pKj_38x#TuI!?RWpE{4gi6UD``(!p?UKss`p%sSE6e_?2_@Nl1z&eqInqDcN1gA~BEoJeL$r~Z|U2P(pFCrm21K7T{vN)K#m+N$wu{q=GmuKyD_VEq19TNMF)#_+9Ris2%!-tt$Gw(Z0^ddvYq0quDmhMSY>qM4Kki33yJ8I1M!eCpc`+4EwWR) z@j7{O^#HaAIpX-8S%95CbU9Bhb+uJ|+>O`3(0{BO&vQAH+Z{KEY!&Bsmr=o}?tGN1 zFeHFV=l+d)xFMOOgeyu0^oWU)u$m+Zd8`NW5g&>*J@}bv+av{alp=U$4mLY{DRO&4 zt+t9@J!SvHdxGfQVr)?i#cJG6r`>MeW<;Twx0<&& zjcTe!bPA8-H@w8LvoOjyHobGeX>tZQ2}X&k%G9q2h@>=NuJ+?ZlbG0_H?8gUuJBwqNKF$@UctMH zJ^hK`=>uYCvItDdXeMhfNwgmT`F|%$2LQ|C{|d|9lGKV=hKm?*iaU9|NNVJN0P9v_ z*7>}SVb8Gw)EQWT#VN;$rzQ8)n4w%|ITA?O5-KPgNfqVNNy?!d z8`#L1CRKE%R2sIE-wiHthjM$HDBZHLR@`(6?-0|k3?Y)J-=x3lH%WyMDHZ#b18<6Q z;z^gfxe!W#gI7ti-+q_!TM#?EcPVd^FJlL~`;-D39JsY-`|#4J`lkpF=A&IX{+)b@ zRuLh|2)XYUyTS&?L;|!pM|?1cRF?K7K7i9@V)&5%?G#DwWqg*)g~3;xW^9EikvjM) zazTp4wvDvQdCSOKax9JNtuwaDb20c;N&Q1H6!qL*D_ed3;__G_H<#}6}$&LLX9h7sar+&EBRB=iC5NDyuKSNo_Q7T=ps7hD$jIc z#ZVcZ?;0wncI+BCwWDD~w7G_-iY{04G`ArwDMc~7Mqzl3x8bMP@OC0)C=a=lkMK}l z+ePD?p}b>5@0cOYnD*ovaW~PfVneH7RpPYx8UGBmZF%=lTmg}yCwbnVE)qf>EDiyh zK2K~Y;Wb-7O=2gh(>uDsF5O6?KUk2}pRt{bO5SR*OPYUYqiI4#t&_HACr0lc<~jq4 zo($ygFea=OZx7>#*lO|ewY))|!VWF~N0VLD;t0c#{){+rErfZmsC6B0diH&C)4NRK zQv8B*@1O{4IAc^fAkZBNSV!`slbw=iIe3#uoCyf9v&XLa*YPzjvuU(BWmYy$x!cov znN`>GmZeeqf&hq6X>TMNbQ_8wD)8*^rgySD9A`TnC+;WN(HptB5F1ZUvyQHssEL3k z&OV1H{67kL%Xi!W|DE^2{4w4%3nKB}m<&BVNtTKmhx0*attP_g7Ei37+J}ufu7r7* zgcQNG0oWQPlLK;-gyqp*)32+xvXTlgGDq-&kgTL)s;$Bt!JDOe&*vbh6l@x=YMeff z#P+p2ZsU1XVw%Dp{&wrn(YkAFYn*Dg1KKz40SEr}*`V8RLI&wm#me$GZo-tbQi>Y= z6-C5pdH+{D@lF)o%xhLrEDR#E2(?q?*WSzv^2(@kI0P%h2&m5E;rKpOo1VmzH}l*f zFH#kXZxYdv>Fy&^j@TL~ACnoyG7_v5Gy36}E0Vj@ptTX_q#xDs>B|&ACg0#vVX=x$zcWEvI6fDv!V?stW8@goIl~^;>zzztHiKxAKeNx;Ed+tJigP z8mb-_!m$!;q;qu`;YXNGSxN+|-o|THD&zl#)(-uTXf3f;Dx0)lXbw%oA0hiCn^t5F zxgAOSt)ldH-XK{`c$LTBEtcQTJ76UL`F390-DJ%Gf@TQy@8H?_GKS8UsVtls#=S~0 z&Qe@{2X9h+FX14opeghM75&2i;;>kJ2QS2Jdt2_{yQA^Kjys{bPIjWQSZ40sv9SoI zZ{ApDQw`owJaZQh*HOqJPIZoO;&h*4h`ACAEmB}eAkSbmN>>qAUq-*5wu zrSx>zSYEGSqo;ewg5zwUJlXiL>Rh}ymbVAfj*NvdEf%Tw^J%cDh4;gNwu&$Bmw|Eo zINpB1cd>B%4xDu{L>P$9&6BEn*+e=aLlDDHjF>r&UmcSMks)oR>+j;$iBim6$H%7b z<6{c@j<0tZ_on+W0r;*^M+(jFV ziDQYr(`+Jeh?qX{BqBUJpg}~JjxwN=*Fe#gR=gPfVF01Cg`5sCR}ic#kOj72`#!+? zv**R42Rw5u+SyTQVmsx*Sc>IHwN4a0$UE36dUvxaZe_GI2LCf^~onYKdUHmFnYCXsoTkYdqE z`3|XpcE026DZFKUZ{Uw8zzO^fBpC6uTN)VUk|-8=b_$ZHiW2_Y#0BbvP|!sv39+=_2ojx9 zgyPIm!!dm&aDIA$qxp7M_|-$ydrDG>2+XB0-~v<>;&;K~kuR4$?u6smXv6a}b)QsF%nXNO_3wX7Fl-s;MJ7G@`7g z-H4(G@LSbxFjH>3>oJoTWZ60ZfKkEFugrCyvdPfJURbR-6LIP55rccy*l&3XSbYVL; zR1zA6;XVean_qu}%G5+2N2Afxfmzg3dHiGi5qW_Gdd`S9wd=$1 zCgm$0=W#6TMZjE3pxw(Gbn0!2*gc!4W0CZy+59p6Q3?fV4@`o*)L^=pJBK$p*S@s@ z$^=`Z%O_>%0e*rbvIQE*B)2zGH8C&n-l))$(`47aETOyxR%KYaw?3q!E{fOd$!$4!V-pH{wm9xjAQaqzco zKgHwr(d7#~0x(`V5GzQy5sGV`wCbYs>m(Q_dWH$CYM0N9n`k}-9caFXGA zaBMX|Cj@6LpZ?PLXP>Bxj1FshLg&vKig1P`Yy z5q~}YwAG;Jcwh9o_&F?Glqtha!VqeXbbxJRb~O zUjICAJ=h!4z?=UIU8E+y-~*{FPK7f*^CLNV!ad{W^IwVgU%<(cvKM$=e)wy8=#X*v zYgivPcHA8HX~l4-*eQWmc}saVKlrt{a4FA?BkMW*bveF|jB{=QF}S*+$$MFo*tMKz z*f(US_{=XKl6EapSMZx~1^0a`c-@fX0(QpA%dnQOkXz9^yuzCVB^LFHFJpRd8HDdp@D9FA|@x;?EfpGNL;AIThKSuFgSt zD{tM!J>%l(*Rhr$#HQEzt!%F7v6|njm=W!f&i*O9;%FSit*kj{_x)<#ptjw0it3u{ zPm?1IR>0(GQM?AbA?Ju^)?la<w2%7?rlSGGjQmA38qA-MdRxZy)6`BE|ULr}9+JoO=O75~GtQhm$! zeaP2w+}Za079cs2=a?tH+rsPgkaQw#3Q7yZNd|_jBIMbyG);~*isU2HTG1T8Daz91daGB`( zDR9N{_D}gz_M)i1m0wU`-&F_-o8U4^r1>ha`k}2rdX8AK6$Uq3wD=4=Jm!fDKjS&I zrC=abjh5m31|LR0lO)4C+P{ul3|gS;IpVR;cr7QJ_!z^_5rLfLV)JJ_Gyc;tK$I>` z{EUauhrSKK=ZIF@cwUZ;7H$kD;Q?UqI|j#uwp+yw+xS4}wXxaXTAbL1G!bqrU+=Fb zjO|*LjBQSR9SW?M(-V3avYkHxt!T4@=Zo5(^ZW4q>CX|<$fEv;X{PMp_1Q-8>JHu; zg~6Tt{rH!uZG?+PyLe6_{X7=v)ISz%7qEO;EM{7DaX42CiL-ZM``kt`WEaoC$E~~g zYo}?NC+^&hbcVP6v%7f`?)BNhv^=q8H`noUcCMBQj_Myx--C^x3K2~r64YeG4n4pc zcAq`GYYeDY_wZo;X5uQXl(#@u4)cIG*+RqRxoEUm?A;?tZ1Dvj4aU9n1wRfDFMJ82 zHi@lY@~d!DRj0lD9u3l7vyZ=q`xi%j!^awr6F2D6SllOze~Gw8JjqDtO$(9O`JnGA zukkJBOI6i{1h8C8Hpz8y{T(rQKW0QPi#zwjRcsVuKc-1Da5D!#FMgG6wCxV??pbe9 z59Dj*y~7l0k!#Xl7LOj_-Q24C4q)BgX;=3kl0eSGB?qyKe3N+iAfN7*HU19U(%%v{ zd;6BVa%v+_sM^t-u!@pm? z`!K8l*F*6i_)VQwjm7IoDoxQS6D=RHoy6Hmf_l58zA1Je<(XNS#v{fw4}Sx8;H!c8AKj3-Rt))(-(*NB&{P-W8|5GEt`aei zfIM{tqxGp^5Ha*qgN5jUZmv=X9%)8nx9)rttE*mo^3p0rBtz8r2|2{K#O*)9zsX)J ztNHd%yZ~>%{)u-@e_LuOZRdsVq}saE>1QuX=!0Bai{xNKD-XtD6#<$wNGSw@K z_L{5t^^M}OU*+uPtzY4tH%GU^6)8BdZX%}mGl{dZQ1kcY^Xv9U)2=Zc1Ar_fK z@p1Ta0+uOCk7H-on?yGR1Gq&tOnFgn)PM*knDFTD(hku+KX{m#M={rjM<7V~;`iVA zQ^__Y8{X1Ccu}TgD)@j&r!UN09RB=6Ld^SN6);Zg`s<>DU#pJw(Sd&L z-1t}My}?D_fR-)Z^J}Bguvb8P7SMhT0QN><#A#o%S4Dihb{RfO;-_z=(r!=)^vxD`g|zG07I7d1#?2MiVqre*BLXh3iF^dJCmMGjPu4C?+(N|& z-cN`?iZ&Zf7o=!)5Rj}-(Q@lJ0SW!!1{`LBHz1KeA+erDsyQMxRqMb$5@)AsH-f9H zQnj1dC!(3I4aCz#T`ST*m7Rk!&76T>GB3w(eQBAAc3`|uv-&=kD}rfSe;gqhlBT6s zk;szA^fzX5KN5bIB~9ys*Xz@?hLQ2)!zp_6>nEgt!Otr>yraccP?J+dGt&Q{m-I^` zbYPnVQKEk&dRNg3yDw1VB*k?k1;kwBWOaR_4Vi`mFooLFwgspKaiB5d`h#|jAayQQ zRnew7NPkQy#3Fr1RqbqrbgtN2RZIV$VE*xc9P=!(C0)B33eq7%yAYwvpV?ZDcrimO zVY|xx2_b+ zkXnAibJ7c%#Bf&YbVgE&V`OjOHr!<57 zfOqZY5a(>Mq`8(IujW*VVpDUiZay|HBgIQ&c*v}2yBv5;GQ$K+o2UWWz1Ljpk)zr% zGOQvj#mPokJ&=iphAlMwV6Yg}LaQB*g>dPjjg}fsi8|uL`Gr6Rx3V5}PE$427BQ+y zRS_%C)@W8Zp{3RwA1hmG2k`M!5rE*Q@`_{^-xon8XP38WrOjjUE7Yt~e9>B4i0SkA zHfZ-;`D<+eC1U#~;Eo?*F{&bMtdAU_e?uKn5P_fA5Wlz8F2RowOOA8Uas@t|o@P z+5ca2-yI)S(fz%5XZEJUhJ=;^NoWZ*6cGYQLX%Et3aBhgvMG>c!|sM=2`C_)AZ4OL z2pvQrsJNg4QBlDLKB6Lmf?@>`6&r%Q-zj&qS$v-7eSd%aK5zWU-8pmS%*>fHXU?2C zcP{%SgY3c9VRIkCKM>HWFB-YXJD@KK;x7(4cacUcrysHUSo&Z;k`#1N@wd@V(lqGA zR>^GSGzsIOC%aVcul>ltrt9Qt$rU12IdeKxrnsZ#-A-%yTmI4N&a~x?z1p+pM|l9SU!l1 zHTDqR?*~DzG*FEbRx5W^VfhZWUSdOrkgM7ymOhkZ;_Zx;L!pVjVLuKfv$Wq?{xIT# zpq&{86A52p7*4KeKe4}ulPzdNf)jh&mq!o>?mjw^LmEJZcF!a;t(Cl4*lu8BV8UmG zHnLYT$$;A>G+cZ!Iw4B>Yhqw6@{hZ5UNo})sJau1CDH^_x=w=C`YpR-Bx$OB%O;M* zOuo#fk0dSYZ{&!a!yL0NU!3Zqmc1j%<7m~8QRFcoel`lleq?`-BAxoms?lTKAd9Rz z(Ne2gBrKQ`L{5ez@rkYs{-HyD&Jt$5%-o|P`x{xsXtG0A(wI#e1Eul{n>$8;T8||Q zQ19Ncf)PIykWOrBCJAG$3rP>fk~+deBv_Jpmmf<+yZl%p4KII|gb0?@0aG|j-n~n( z35SC<=Av$?Z%O*XwqS)V8!Q7KUQ#871nxT3&Ek|KLSYb zQ$w?21wYQj7tOd63eZrnqWyTmiqc}T-;jbf6HvvkY{&$%KJKy@{s=xSA&IyPU0ic& zPR~G~!8u0n@d=2P$28*r5kt519v?R%xXEHlh^`S3X*%A@4J)`232scYlPDHH}eF?oRdh2Pksc)&?0tql8Vp8rBZ}?MLj4=ri7c`Ul{}D5mig5csr|{iI%?^emMkWRv;D@%%u!_MunM%{e@lJc zVAG}%mruODpXTF?cwr3F4BGD-*4#w~sM#tbrfcc9hh4teifC>=C3MWAAh&O};ynls zjr{NWEg}Jt6qqvYW|5}1s5V5aLEO1E7E|fCn?%(ti(J16{mXUh?`~2UB()1+m+plt z;YT*#pL=RBgLr&QT{$C=sb;~{w`Y)nwV6tU4J8+iHZy%p4X^)LT`c zTtdSUd15dXDsb2=p#mMV{8b>wR3CcW$SP;S4!X=@W($p%HrtPJqR#)oI0RoQ#+m&X zhi{95aeHP9#+{yR=)PFC@qV~(F0$?qaPE0`KLBS(3&X;LK?G|NMk5rQSYE{|X3A`0 z84rA9JboEB3-%(xP5X7R)1L0sg~W?0+g)%)=xm@hjdy z9L4ans)Dda+&~p4tD5y|o@3qM9s4j`K6s01_9LWeeZ?!nfWRiH%#=MQ}ag%y$utMSO!~5!{o;4srJ)(zUVNR`N@p@h18lGmc8|6Gvnf z>;?#=i1n0z-2w+a94yn^&cH5|b4?r@^H0zBTa1b}u!6;;2P)dJnBY83azkuA#6I=x zV(jNOvOgAMzXj803F)4;jCTRUg=57;G2C5+6Il(J7Ra z4DQcr~EjKst&uh}~Cokj!%&Db>t)+mbI7NklrLHg4&QqL!hSIgirE@uN*k`^)kL-Jqf zCa~(qU~_%A5Sp9IseYf z<*i;vdcz&Jwt}?x$!lc=X~>3FkYWCi2|>rym}(o~aVuxp8?at(WMv!3Qas`-p;0%o z!Ih+&f2oG7va&XvG}eA2X{;~jjS6EK>q*19<^;TlhLzfwFQ`ApyKE!oTRmy5BS;`y zoV6iplYO*_I2)Jo;`}4mVw23bPdlETC)`*QHiw~pI_ZPE(1U<+9p<2U|%jG!8_ zaTTE$_>NmiI^G3wZ6(e9o85@*Lq;P}3%}c&+Z_ItL~Mhospfhg*hbRe$5NC1;C9lq zu837Ijz3|9UEGcpWHGbufV*%d8?%G-)RwZ9JK%0DXVp9WRP2vC{+-7NIki1TxjW%8 z@|*Vm=rOvb)bBgVuoT3OgZT+uD2R2fNc^Q~gnzkAUAkrHj=-gCuup9SE?*PS6?nj! z#|LPP0D@J!{M6rvyT}ONkr4#*oX`;Ei-SIBTyd=T5x zi|oOJB*(YNe18xJ4i{O^Ls+BNvzdo5OAL*;`4H*lOMG*PEb=90942YL#Qwt=PNU3) z!{kTbl6JbX; z_^X2Q)mO=4U$+1o9-(uE3qIr3SA3`9+-vYSR^Xdx!U!62gd{@l3Xi~wE@#V*`1w%1 zM@W2{;q=B!-tf5z2YfW-z_4w&-N@wv4N49)Lh8KkN37fHBz|ZOv9<=Ww*O2lJxSCc z?tAEmW~-9wQ`2x*;MC%E`16=||Lde5PK>-a-o$|&`qk$c{O~wPdJEQ$QDIPxx!Bgk zTg#1E!9|l5-s3qc0l^aq!^0o;r?M?eGx$5f>)wVt?@9LB+oW{^aa5PlNUm@hu(qK# zaWM~hhqP?|1V6Han#U>9KcOLx2bja&0XMg>qIYmI@Gy&f7i-hQY~;J7Q)B5vMN9|I zI`}2t35clY=Ig_3-Mi$r)`BEXHp6zBwP5Hm{_euJtLMY7ivvfz;FVGN3zmmOb4I22 zuXjlct;0syC+>kZ+8Ygv_3xXeY1V5zFOe7-ILT*h#BtIqtep3OzrzfZfgLB!OXY;- z@j$8!Zy8t!KqV)F8&VAF1~)huf${wNXB-uS$RROsS%xKn;|4xxNbd1mg#(nwHH{)% zjGIy*jYDVF^8|?t618LWZ6`?mA+jmHntAhiyM=5;5ut{go+X}Yi3?#jhy?urx zG?ns@q1DXJ*UcaS(U21^^ekzXAO%mHTw;d#$p{89!-7xO3_DAT<;iD8!&;KW-akvm zMawCOIv|2rWiXC7NAL9+KbK<*KO+kxMFojL@M>Vr!BnbXiJ!x>aDzShIpRoevSXi< z)EX%uTnc!MKcDb&Ecq6%(K*s@@}pc-@b1hOA%y%JhvEujMBnc+#Zoy~ZK{&1aqv_$Qxg3tp>gepDb7aGRZOjAbN!yX45vZ3-jZ#Djh)d&a zfVhjQFw2f;n)ZKN_n+qx^s=61e&Lg19+_G61$hTj)8k86dK=lyFG)Hc2fl=p@+J$p zK>CI(=iCq>A7d{-A8lm!Um$iwub#a?KEPw|S7f_&5y!#41)Mcw>Miuy~$F3ASp6d zR?a{~WhDBdHGJX@%N`Sb<`3jU$kp)6WGICE$;*VtwjRC=>zA?1mvP9z*p46Js4r(x zSIE6F`$hjSCZRZ+2vYhe48HoCY|$0cGDPSe@M9U;mJPiFXdyi_~J1} zAryuMmlRYkv?O0Jk>KVAq*JapxEGkeP)vBtqD3w{m_gyVA5tgj3HS_TMkTxdS8`d~ z&!+rF##(*71u3qO+40{7g)ugBq3n~-?R3Q5TB%~-tZ_UD7r8-LVQFJ zU8!cD{Ye_s^~=E}mDP+~BOTJK|5`8yL3kwHjQkpvCp<*xwCftI$(PygYh+xcuk|80h<}5;;a@n5Kfuoa1zmK2wY?7e1)l)F z4zJ1qw&Oa+=m7imI!R47)?$lwotR2oNFSGziYp_2SCUXhY}gHq`T_Rn4frV!uBSDz?Fl733g0Vy0Diln{D{cF_Agp^@fR#>$Ae3>5UYK7x%ogcnm8 zyJVp}xVu=ViQ!cj=Yjuqxx}4d>vj5A9beNhVXX(fy@KEcK+xC3VA>F`t4|N6&9u$z ziD1g(G7bb&{yCG-5E=_$%MdygF?Q|{IvS5RL+E|rO^Z<4R$IqLh0@zXUsk#%ql&Ez zrTG}Pzd~tC%>Se?daUuwoZsBQ<$g^bn5pr8gj-)?G@K5B$YqApI0Ww%hSN6MGwktj zI-#bqTxT8hY7umSCO1C5#gW&UgwN=3MbC$IgLhjUU}2KBvr@z(^IGxNh|5a5`VxDr z^iCse1qyg=B+Y85_+rMqBXELZh2j?DK{h**79&dJVkB+aO5h;$hlKICI0Ovu<+UR| zm|t0$AU6HzQWfi1kG@=A5yVAz{^ZP2f}A^w@@pTeqG;m`xjZ+(rcJEPm|UVWyaFs~ za-HF`h4eAqP6mpKB~2@0D7ZDqQln|35QJumHtlBV^=W)aluA~ykO| zuuh8wVPyyiZ03!oDRLRlh{Ciq@_ve@4I9e#A)43#`_~G)RMfhK)9Dml&GY(4G*GQf zHV?%5wxB+JSKI0x(*UxCaORl}>BtyK)S3|qMfQbhK&;VwpdW5Td?`EUb=fPDX@vGpDoM7zKn2`j5io#$pscXiPgcm(xniz^>VUPWg=Mq#(?x ze-FFm!4UPXx3tmYjkZ+C9GVdp_sj>QAIx?}^#U57D99(5hEsXy0g| zHTp+H`mt#fYaB}_VzSJPrA?9~d!i9>?zhPZ@=bG!Chd=$9{d8qPYz(Q0icUeQ3r*8M-c`QI_c-zYZzAx5 z)rz0NGTYMzK@TgJ;^KC+hcEF?J37{vC~Qv?*`W3`-XC&@FY#=9XefiqkL_t|Um~Fc zE$^^EbRHH(vN_MH63OOqiO`~an>!AI=6|fg;!oplg$+zgr94otX(~;FN}P~NdB|T$ zDwdUt?DoW$v=!Eil8)3FFTL+PW`}=kTsjc2U&@ev53LjBA&YG~p#b8zI?-Wx zY(z@EM8da*GzkZS+&mGb_*Xr@>_l%vz+B_bQ17UxYZuz0g`eM#uaMFBq$`S`AKWt{ z-YkSBXLGvn#(OKe(2bh?I3&{VItdl+_GxV>yy9lQgFhp0VG+)DP&6y>_qD+=$M}sX zbCs{UvMb#O?%Z`dxKqj2+)i7nSs{*>Uca4=5BftAU@f}Q+kJ_wZWLdBRXLA!qx+hE z!&@aj@0%guSnY^t0@Sz#7Ftes+A&@tg=Q#ZZXpRPNk^0ZV9$4_Ex@hQ-DzU9q=B0R z3>u-JQ8$eq4iaVYS@sXp=o5U+EAA=UzNaVF-3{y`o?_nmy+8_?J?;>hv+tnE;cEI5 z(Y4)o&=v+McKQx*`wV-aH%-L4SJ9hZXm(o29xgnak{AMR@?FF&A$eQV>2SQKdNrMX ziKn&LfwWzga-Gg?X^$6v-;C#{>QXqnEOteU8{oT;BKdZba0pv zb&B4xM+eg84bJjT^2rE8AUH`})5%`}d}Sc5k1`(*q!Z+RJdta-Z%zKLTU^aj03S~U z8##z}LUr>7VcaU%Q-jcj3iin$%(@Du4Hi>4ZZLHEIhH+`b^~zzV9c$J?8U)YVZUIX z4Tk&SOO`f-_Ezo~eqjhM3SkR|(A$B2Xb7DK<34pL2I(tr;ZQn53%a26*TrG9X%JJ1 z$l-Jh@NaauMU@$?(n!QrR{8>HdfMqE1rDq)ki^$1F}NIBBzym(=lzoptclh#l0 zh0)?-cs}3!CO_V#1C}R~rgt|Mqy%y&kH^dKOu-t8PylXgsntrdH91MmeIso3hfG=% zw%Twc^>?|p1recGxR;b$XnaNUbGhOhSG8kci$?kpD+jR=hWsX*`x(Ug{WGy~N&iqo zUSJJ$lwS?)M+yBrbQFzkU{sVbN1M`|ADGMd>jy_c8dkHaQ8e1;?tXg|t%(}gIhsyn zBS+H)eO7bfixI|XvN<>@s8&R9Re z2?0N1wc;K}{vt?XP*{ctf=&z;%81NZr)8pSSeZ_K7b#J z8Yk($-9z61!Gre-g@65COxO)9;Xc~gcdC&w5k{i(K3JUP?7jOiYHBy^77i1%LCCw&yy4*%v`B!->LD9M}W|>6W z`uN#x67AK%7a#yx5xZ-+4>A~AGKn@uU$#!7?eO?u5`7#DD6~W0RkE#i+AL1Zt_W`1 z@OOAK=yY662KB;UG`V1>b2Y?F9C=YHVzN*pK3qn)6soo z+&W|%CwW*6@guB8$c-t>CrDB#goaNM9#>zB&ZAT4gXptmDt!`lZ=5Rn{K-@?=r^X) zVZJ7eEbyC=$B|&G3uqT#HXup9F-4+2izuYG`M5u*kdE--qBk{4;7B1I?aOIWP?p)(Mv8&A!B~_a?@)lJ8`Bc3tuFkpQ}A|EZFqD zM-1bF5|MYPMC5%@Lb>y@c7boFL%fVBnmPjmZP1@LgMM-g`inE6%bEAPnNUi;Q`UB~ zX+ke`TFDf5^IHY@WOak++ROP~+WK0v!9D6?3Z$GF}@{s&Bc2_r1zjt^37 zV9;oG$AdJ!x!7mJs=|6ckn2Lj@HV3>93TwuZ4c68K1+A{L-dcJGP!|ZC8e}m%pqy_ zNJbD&F%EKMnQE=@zFA7IP^gCe^Jo%u!l(1-dNXe9ob@Pe(&FhYaEijh$Cl@LzHD3f zs?D{~Py8Q>6Q`FSrG2%k7Z%c$T8q~>Lz8f>m#liFp-bV$J6a$G=h9>Bg+(+M!`Ngo zUVFp$02jjm#LH5PY4^r!xLb=`Gw@Oc%X==+*kpL#KV3`{o1EhHC*iDS4CWGSD1$W4 z0aIaL#w?*rB3E$|*f=J@BuX}qv94n;ETOGRm6|lHBs3+}&lusHF@+W5#KRz&q`@r( zseqZ))m$YQA@+VC?WMW-*M(>T|Yj6*F z*?xGU_574Wlf(KGA4q;2Qk;yuf0Y(@=H zM2{Ls7;vBSCTt*IF#ONrWqZ&fE^|MzuoX1MaP9CT+72sd!`oyV6pl!6A{o(BYHHDN zJ|G$*1(_GUe+882CboYCy*)xjT4Ah0u&cPf0ygYb7E?xdYa6}C%II|deq;J$P&Y3# z=VNpZVvMiw)T6BFDlE^BvSF(r%9mM*Ol@98X8{<$nwB6qVAX1x8nchr8zG|h`H_Hz z$Yya@9&UO(yBcb86YIQ&Hfx1*dLBH1MH~W{Y+kq-%&ti;70Di^1XgDktk2bJXh9Dp z-fF=38;;m&MDxD#LBpV0MeqR6kb{q7MB&fCDSnT&xRcQr-{8$&OFd@Xi1A4|ZPM(h zl6=I7?5|M)38oGjO~i2s8h3=nJWk6nop(J>$7$CYeS)s3N&WH!9cl#oRdZT=ezA5- zX##UT35|1vJ@zEFX%#GVohY5P4$H$4_T)NxG-eK;$;Kv$x10Ybt9)?1QK$oJUV*LJ zbvC>L`^>rXy$dR6xFzT>&Owas6*dvAl~2AgRc z9n|y@-W9w;oDq}&`Lz0yAu`Ec*#=%zunXJhrg-VN<>v_MTB;=2Za9bJaca-8&D+K1 z`Pgxj3jC65+#^*Bgu_#Dn9rO~VRux)yiZ}FtY9I#aNJSB#_obv zuVCf7V52|GKHi1;D%kJ4=vA$ZU3!{s#-r>Rx*Wi!yM-xm_ikYd+`pSH1u%Gz0H*H| z!0bIVJ!rmK%69F+LCs3`#~$##jAd5g^oFs5DsX25TV4fU{04UKUV03Vme0Z>+QgZH8Ls8z24949ytfZ$QCrzx`{)aJJo^Gl9by+=z>sfcO}un6nmX4@ zJJ+kZ!%QrpaL;((@WQKtfQ9q>;U~Sp;ttThw@60O0s07b#y=mRXSCJc<1ZrAAw*6+ z?&)ZJklyf1Ha$eI`X%EI!xLY@&K`!7>^f^(jiE31I;$}g6%zLm?S55kS_Zy`Q@88v z@z-dfR^bggLg#Vk-O|_TZ3)KRX%IXd7~>V^V4M_#nMN#`_x$Vd!vOp1H)zMk1~%WX z)IwvaZ(>B?Eqs$EHdfLNg)=3ID?=;xHeB_?@$@Er1y}_~@p}Aq_VQ6oA;!KriV3xW zwLgXpP$e62jD8Zd0_RZt-i{7$(f0o52Dxw1>FtkTfq^XmI|oZl@)T5nmCU+{&vW4| z;qZ2*e$q2@lZCxadyc#y%kbIchmM#eoU!nMS3Q`4neKbu1w*oUD&S$=P-Y@?##7BG1yX}2CNNXWf8^7I8 zv&x@ehRcWm4HsjGZ=xWBMSp;c0(P+bKA?-Bb-(!lGwvq)^kW1SH9twm*V(~a&shRT zC$4{8a*}=}PrNv4`agELW5MxlzHhl;|uVg1b#4z1t%|D`pV@1uluSUg_)he~< zBlrp{z1u#b^)%og_}Hj>%zrF4_!QiImEQbQv_M0w^hck-@wkH3Jxv$NekKq!3S(=+ zOCsA(Q&-UWZ5Vh+MZ%}F#Ftq2DK=e3&bOapvt%S1p223xNc22|Llh(N&>85qZ?;Qn zub+YQ%(y7#M_$HAM4zPtd}ZusX&Yb8sUnsC9z0BS7zGinW! z;;|tJF^J6mkyf9r{)`qjlhnAdWS9p<;L1hrbw_ATVla`OcdVGrKc{y~uETuMth>cz zV#hy+18d{w@QUr>0KOIi_pQ(A0M$f9H}WVdE`JHpXrET9dLJgA zZZ(3Gf$F5}y?738Vs$4AJ9mz@{g1-a>O2=7HsCz{I%umJ$aD&g+=;R z7G4XC$$x{*`+_FRn!xz^yqnyBGbGSDkG=T?ZKB@q!ia!vJdnmB%3k||w#*PrN7;G2 zEFV8y^n;|hpC%EKAWiE8i!lG7W?k4kDKS7?6~U~te+hHt2D|@DnxZPADCl3&fiG#( z_L3t1ETUn>WAITKj|vJB;^o)w1x&v{6Pt?0CE_YF6%>n8BPn(03BOu=zy+Ai3s}(w z+N`TQA?9rtaw(**#z6VEB;LG0Tc=3ni27yID7X+7WkNb2<2+l|6oFUjMJB(9MS!k; zMfXBa{==d0EjP35=e+*6hr$PV?tj0T<)03QUz{&rT6u|+0CQnw|Nn6{%e1R>vpH<- zW`h0goH;b?h_~tQ^f{~dRFGwyS!?g@5Ng>*z31y#j>OP=-`7lRWr`(QOJ;{sEDg1L zy&t7m9x<1iZqrPfspfBPq-MH_zd-mSW{HPWRuic~H+^rF_<``Hpvvi5flvMLzft%K z&Q7(ptSo&#fM}RP3WRS5oD;1EzS)oHI)xtyAFJ?>0^h%rgB8Alm-N2phc>JK={+D0|~%af?P+8f2FD*f&JXU(m8Q2$`5O(neN8_FX6YH zXr_^2nrS|s>+!$WJ0R8a(!BA-gC(88!L{34rr?;0wfeCGaNe)^b>y=O=>|w!hiIl| zcy zSMV~xB|k(-_QyU;WW5sra^C)Ck%g}eWw9?a zLgF0M092q#7{?X+J?LHSJfuFGGgEY8(>KpmoUSy^GW3 zbBjt>;mIpVSMX}2??E~>Q!|aivvnPbUp=Ze{iQ;(P~$u7EiXjn1oC;qUeP2zeu}>< zfa~~WQ*;tWYo=h8#~;=J2~9^nZ|6!S5}8Q*b=b5Tx&EU(NzyAd#RhP$QV}|#I%d>a z1-J!?z5x}V1DrGBmjL`Liq2Bi0fWvT3f^D9xfUCvnG%d#JZGp8m;-pIA3RMF+5tGP zunJG!poan~I0ZQG>7am)oddis;4z9rlg4VM#ApDn;0G6he3fBkvkzFBW@O$aN8pkI z&q12YxAFJ2YG*azVE|silaJ7|O5{@6X^@F5zFRi;OEnGWu+SGV?qvY;O6RL_sY061 z8so3EYW+!7{G7sH9nf3TI7vP)LiXBSAzgpqP2WZL@3SO#|(K)0afE1jtX{Mh|rczV563f;NVh2mSOt)sb9qCk$6wk&+$B<3}p(`^r)0@aoR`WEq6w}DBe$!!9@g!u7 zM+GMqYNi$_uqr?x{#Ya{oD%>yEtVskta@xk`X!V%Ez?Z7cTww5U5fWy{Ia< zf^;D=s~*!#eek@h)`n_!?I33G$&9X6361V^BA(k<}IjD9T2F z^X>)(G&4crf3Lcgj`Y8PS5~f>O7Q$OfS>8>M7^b^yHtTp0QgdohbOP#^8h{<1J3zS z89=BEa4y3875;vO2axz{sRsB2;C%r(w_dX6eNnHUNL|?=IWa$=0#l`|__ESX2}pBE ztH!etp34;H=4`Az>4}wlaZVY3CpOCpo(<3{=K$y0$fYFIw56t;sQ-OcG4;#fL}LO()QWsQ1bG44 zI~8!=Gj9NXx}rBG0Dqu@*9p>cYV{yf5qdO$&?Hr0YXCe?!QBBB6f5`z1)sxOzXDmW zR^-nFpr26iAFPrw=a8O<{#@CnnQp^#YXCzlcgTTBR5J`H;_oQnd_bNH92f~NHF;IV z=K$ccZv0(TqGj5tnIZw~hbJfWld2#S>463^o$B**|&?~#&XQ4KwX zbS+sQZg7HU7^U#MsD|JQ;H!~Ne->LGJkP5J9eutwzO}CG>4^aN^9Fsu>!Ez=KB#Xl zyW>;={1kr!k-=G|2Z&atg3k=D&B{rD^9CPNoRd^wssqmHXDIq*NQWa`t?0B2!CseF zbOHcAi~GGG2`^U_WFpOlE!8WB!m4^)hIBFDnFln}06dojG<42OlFlCi4JreiH}s}Y zkE-#nR0ZBrW3xjc0$2RaRAX7C@C^7filxK$HBv%HU#>m&t^lF4s{FHxP6E>0)SaZ@rQ@{PV_A$0-ji(@7(O&* zNb@l>{yM9%tOlIt8-J}-kB=zA2K=HLvJ-&&n*hBf{$SH3WC&KM62Sq&XgVl+*io_0 zAZ%6ewyNR;7WxMCS*ilx7r=;g1#h6x4KkU4^XgL-JWt^n`A5~{D+ZjKxyIi~B`HR_ z4Ef+krirSG%O$|bH=R*3w=Mv_K#gQ2;QkD%<|Q*uus`0g?1Yza)0>vxJ3a8eWcnQS zge2cIq1u}_t){L(=hmZX;rnJB12gwnJrZfB7#-))<2FUZgC9qBB# z+gv3@p8P_G&Eax6UFi)cm*f|C@{4S_E_-pF&E>FX<=H0}I5ODvcPx!d`!wQ31eF5E zOrD)>b9-F*MY(+iv06F(>idw3?Jl=NVfK#>0DGKM9Yq7`1wh>n&z*q;lAV*MII=tg z>(s^_RF|jGrxJ%v)HImA{H`Tl8^TWGVICUG+FqlvO@_t#ip_B4dmJvCqcGp2S~Z+? zde74AjuA2ch$CocHvPxEEN4Lh8kO%X8p&3@XGzpXv3>k8ntlA9rFrR?nog)%#ba!Q zk63m|Vews}OR@_*eYfaRp}nBMnPtl_%J+=3a{N4pz1UXlaoKVmp7FII;Masujw1`* z6U=|-xav9__lEJG7%NVURUWrQQxu=bHXgS$D76VHg^ogJ{%nV+X;Ljnafv(6E*qn8 zik-!i#bDJGaywj}tXfE@AUj;NrZ`(rb<{1yuWUz-y`;cn%ks?33FCn7h4z^|l`GmP zQ3~=z3yMk#Cu58}PTS<1d;yu9!&4GIg*i_^Ag2nklNtDTa=yo1V1Rf+7B8gJ?33L^ z!PD&7**0g9!-;fuc5(1DcgbX)o<^q?mK3-EEy{KSt0>!6sMp0mO^hzmT=m~FN3(UOx!VBQ~LlPMQCI^FSr zk<7u&bGRO?Z%jc+%_hMQv1#vHqDo5xb0v*AT693Ek|GG?TrJ3ED|EPW9S>_!27;KQ zve+X5P+8SHO^h4_%9fpDx0Ln;tj@nI4ND)@ge3E9QRhNUh%(Y{ z30R~Fv4ol^a1<@p#6TlalvP-~M5glH+4;Hoo~1_0U2M;CEYm``{N%gy95a_|LYnhE zPWuW;&F*pLckNQ9g_d$8q;k4;S*gYGgw#iFo4sh3t=Q$r$)AZS;mOK-?AB~|Nscdj zl_rGRD4pdj%CdV_tG){r=e7apvF8`L*VKS}7UjBa4v5W+d{5q5t$}|Fa-1&I4Dt6c zEy_P5zbM-=)0tDw5RD@Bp%TZi_{@aS4U?Y${{z*-UossP>F39(+t6h=X zV|RH(3)a`B1^JfcDl{>J3?k@HA@1D&z|xQ|2XH(#1$6L(4u z`=FDZ&H{(M=qWi!KCs>8vd`Kj`{n~e^qvkNDyPyj?5C5KsMfosgi3tALhg}FgZO4S zv$0H-U@6SCdmL4){fCyuh(I0qp`{5ffIzAN_2hJW4$&8u#&DEq!(&~p~(bmyzD&4mhP90k}Cjj+5vw+p52}IqU52($Z{4!#W-G)NJ>*c&AhDU z7iJ?;2PL}jYulsk0FTQWcQIq7M8FD>&Z^}}tH&Y$6+(-RO<+vP< zV-iJ{5}NHT$re=zlM`#vn71{kW!^@EGcMQF3rJqWvR&C~10UysZ{!`ut z&KBkuL3p#z)+ni>T$@<@IruZFS4G{QOZkxP;0nx^WiLJ_`z<+lUJDY+z~T8qyF;oF zbj?8h(q9l%vv}9hstb~xlC<0D@_fbSeu8uEuUAd7G|c!$)=}VePAw_^R&t{di~y%m zIN$S~Y?uL(c;9QeMVOyyX)Z@6w((EPedZLrz$=ud4E zVW0QZ8?awaTaGkJOM{&N^U5~ao;B6xhBgoeLr)fY$#OS4^t0uzpq4FDu4!Kcu@1jj zhO!f%Te7{^&ssVW`j_^GnU#KHakD+=Eg#V9+HWB&;TOw20*3)7*gNb?O9!gm@Lv1I zQcoj)!yH|C-4ex;zO$s$o7x7QecnLt%<+rZ@~f7H-X`B$a?NAsn8hmSa^%9iH#Q1$ z&0;o7pjb;EHb?V$pI_ie#{R&SfQ8ZRX`S+j+3IqXxE;wDp|rG7{X4ZznP+|#cM<$z z@v`t}y^HtwHOqR+CjV@CQJe3L{{>w{OpE85r8#}n{7r~x6UTX;mM%00<-6?##d-Ec z{FhiV7V}?1nJzI~3Y@teQj_xUC45giS8bE6o;L4w8@I>A-W9IhdoC!eE}h+8mT4iLWv1Il^vVYqsPRmw2+6 z*0^eHiz<7VGVa@&$SvWVQet zHivL@Aq&;As1%G>%wiX$gyU6nI7fpp{P((_N=1U zWwTUa4tYaDMCHPycE2fWD|B<%QOV1EcYaY0*RqadvR(P^B74zWMkY2oc29okbjRDW zY`%LMvfnXB@CNYd=g7*p7rbj^&UUz*@5zR`^K*-iOS}@yNE^>O0bxeDY)8TSvai!z z4!)%QOX7j0vY{WSzJtZQx|1!Hbt=aMpOB`&?(BOJsimI4fC$wyMs5 z=4-6*nv{NS2#Amn-o10GM(`P~mf4Q;Y7X)32DC&zkPma}-) z_i8BQ7nQjGRQ)S1C~;pikg~J?Qa?fSy7~zvbpz{x;5UB%&3>}zQKdJH21p$? zhbRnaytzawS^TthvK2TwJ*+aP0`dq66QrO#keWxN`sF!XpUfw63M09INH$NC9L5|L zvgWcEJPM@teDtv^Ur3}j>hBIdzZ3SlX|vV0N(#Kigl2QsSa0>qhKwO!XvC#AaGBKz0l1_q~M|e z&*<#oKF4wr?#6U6RP*D+%2l%;sE6kXQU{Z`$PIk}Z;NwArzeTkJ*@;gBLNnmb)=qq z2Ha+O_5v(#Gu#5Wp8Xx9x9V0w;1hJ96HZqGv}|%o5!Ydk?1TbmQEtixg6+J@g#vSU z9Ldw2`Pr>gD%pL(dQ(KeJ{YXWG~R@c+p=cbVa&N4w(0pUPl>%?GkHbhhTK?Y3DFDa z7IH`>-wM(5$yVZ$nM*?Tc(P5U)`lUmoy-zhNQAJ5!t_|OgG{Lf38N`HNscT(FQF<6#RTsH?+2?ijf!OyZTlH4>dU~c+Z;KlzAGGR)+CFyCs;|HoB=3&| zBfRX%NIec06{a@P+u&k2k$RD7_4L-b+pTRqy(L}-99>Uu)#xCJ7PetuX+6S3AMPwp zPC3LL}!FqZuzAF1kJv{|iq>(5+8Mnf6WpGiGP>f$sg~ryjZ;pK8?9PR`0HzU?0co&9wK~U$OcK?O$w2oPIk# zqxwi32%lt6$LTG}hhzv#EzldY%W--qZ1xi4^%(7A)-zsjp`BurW!5J6=VH9xPy1A`kend{eT0J8LrwH#a+XZ=108Fk-$6bj9sEG8o9cDx z=j0jcg=OE!%x+KB>*JP~sfl`tcAlM2)E^^XkTRwX&{J4Ka}4qw&Ge4!(dHclFA=LDy#6Lm{yYia4XO!bBP7$nRvRgx}R#Z_fT~4g3|Yf&Zt(Ph{`5*4wffZ9qGs z4QOv`qqk*>mmC-Ff`?EN#97ZEp|I0`^*aJrVE0e$`&zLKf0w z$(@ZI^eDQBuGaX*s3AMvL4Ob*2N;$LrLY87n51^no3Z7odPMY6D#u~C7zg~DoU)AV zN!7FP$%3ek`dqxNv!*zE`oPo%O|JH7+!^k3coA zb=H&2YbcvELT|)wc7}pk%Obnzz3>9qh%WkcQcg?Q$S(Rltl4c4@5ecC414T0J&8O) zXY-s5qP*2j4=0;x7n%C4o8Ay_0@mpc9k-Q^mcUWn^{!+aIuMbD z4vf#oKZpD3VPre)D?y)ihnVkRVQIkINi$_Bdz#*vJVkr+Nfga?rD0%qvGl%rT=+wG z+Y7K>C8s=1^L+DkDCSl#Ua~QJqz7u>jbY-cdi|Vt!ReF_lfI7~W%kfc66Z6=Y1_0&V5W5@N>Z-b6q-cuh29ebvyz7#t4W-rL=vR?W$ zC|dkp14S!92kro9KWo!l?@12OJ6T7E-jprs4W7Tqp6#s{L+Lh4*Ly?hW~J*BaF^(z zbiF@0M4i0K7?zNskA<$Co}u?6)$~3e(8A7SK&`w&(|jPxdh~&2f0d2v1C90?Tha&g zjT6P=AQjseXD3@(z87&F!y`W8?Z`<#?Bs_SZ|HjBobW7m(wSgTn(bE7uP|!M1lo zL{AQY>V2PfknnnULMZ>m+TE%5A|KGE0vI-o9{?o7-by*imfeYld`M$OPU-+`0g_Wb zqLD(6Wy0~Bmd5{0PWhPD^MgGkfv4DwJ0X9cu=)crAx^W=1JUPC*{p$j6Wp6tJ`fAU zS@!Zk@ai-6<3K&W(dSeyc#`sf0=vhX>ifd`@?gD>rhVzX zIRr`r*L0-}(~oIiv0sMiMYv{T@^HNjpD_JpIId&+mX;CrT^&8%J9dQb*0k@u$1?Td zTJZPuq}jKcd$F2#(QuVQ@o2p-y@YpI+57GEuIz`=5VIdxk1={5a+$`n_s78U_>sC~ z?nh(vo(-=kl>#pgR6G8KXN*3gv(#}=u*x)qt#XVHTSa}>-aNZ`T%(1Zh zC+dr1=2{vA*?Au7BX4y59K;yGSv18Hf^$=8dhdW5mk?Y zJADp^uVn8{*6)LwZJUMEDmFX|Q)xAunx)Uv*03vCnB!~NgW39gXx0u6sFgT}UJoyI z;&&VJxTOc5*D-8{Lr)}6Seo-x!#aJ{5tI<-Ct0-vEL~@5Dl)@{^Q}9al zF4nrumbNlAHCGQw+G2^TRf^klaK?zjXuwuWD^{J0S-LA1J=(?&&?_x9`Qt>qFlaCR223IQS6UlB%7ZyJSD+if^ ziMWTwPSNioRhE{r_@XI#B-v{jE}L3C1uFenc4-P`{BxGsK209V(hKznHh3!9@w|V( zpmSfDsuw`rwlC0cCtk}euArK;Lj`&;NAJeoDbVY~1TB{~63EnHSBms5Y(yb=binfX zKfpuS)k0{f7Xu6RVo%-%yYEte{KE<9@bLNh4mL5Yud>YZZ`T0*(26+4xZ UBF7<1D1X4^1<+x(-Kn$x18n|Cj{pDw delta 76718 zcmd3P2Y6J)+W*d+-E5LgIY0ub6eTvKiHOov^wOkQ5K#mam7@6ny));W-Aw}G{om*NK77pXnKSdw+vlBk%AP!$ z{_cVF<$aj;mhTO#KWk+@!UkDiv!Ong;L&_5dhorB9k6P78?vj!>g~R2)<|zovn!Kt zA2sy4aYIMkdDC^HhK{-8CQtXIaLJ8#jvVC~m`DYMciev4b@KUgYmIkELbq5dy?M+{ zH+inM>iLYSgDQvH4ZZP3KpQ*E^QEZ2-&d{3^A(Gm!I;NnfgYu-cfZSiW+&OB{4qXL zd!3hZp{?X^aKWGEv-nf|b*)s}qHWVY(l%==wO!h7ZI8BBo3GvPeZV`x`=fST`$;>g z{i>bP&S<}DFL>8`7kU?Y-}64_-Qa!SyV1Mfd&GOxd(!)>_jm6c-&|jbFSx+>vTvDh zk?&dGLf>-V65ory#l9DOuliQ_Uh~EOaj?o8R(ec5_PW(IrhU)1A1p5R7`!mV4No5T z_DJ+3cmn~$)11fZAKj04L7t$==LGkemw9FZL(@O9)-*`9N@8m?e4Uz^cNZpU9?p0` zU;6-o(F|sKTJTsyGd=n@#ndY^V+-}KtmCn%Y^K%AUpq;))Vd~m0C1~S>`(W9OEvW$ zt@-|7$HVj;_^>iC6QW=y04DXwI~pvHCv-{ln42a{1dy2)i*1o^_t7oUYiI_~;~w4V zqF-!&m2ek?0f8~Ck7JtuHBIbq({%hVX$r&q4{17Xfu^KrnygOo`H=mH_%^K6S{y$P ze{%xeR7q^96dmhsEsMV$FESE(R1Hg$X&%x9@KoPx{VSn4+hHwB$gljd)Dq|bGw}_4 zK4=|Ds1AT}iI?K<6^TRf_l?9hY?E~|u|B?sk{abGsJt5_O=<@Pge2;F7)guio2=1E zxz#pFz$9*nlDfq^58l0$lpg=4MAXv_kEC{ACsn#(9TkOmmqcbjwG6JmO$E9nDbWL+ z<|fsr!az&LicIJv0CU};&}6-U4q6(X$hKXApjwj+t%J!qEv34IJcLZp6V^qp^gBT| zck~-_`-PMU5Vu;(l2hYn(i2e84_jN4n+*D!uqI8gVU^MECmxad9fmzfZ+vMa>`0pd z+Agjd5R1U|kx2N`dZZ;DF1FS-NU;v)%>G-chv&kUFFs zg!+tTG^~~!35?#LCJKFSg4L%SR?ANYmVa9*O?7(&qc(v7KaG*Zw4oCs+sP zrCJ}}@!unSLNtqk&z)7u5Z<6Hm(wfYGNG)mwT`ExWqmCLAlZ7R99!F^)OFbE1zXo< zrCO6xqRDwNCB5D*YDi3Fy8N05hLjg=xYN23vV5>CNfHV8Yj^>0Ty@$#H7QqFua7(l zvrSn&e!h&3L?n!n;wGI4G* z@qmoi{zlfj*&33OUi5=>yvJ;#iFnda$Tvsj-(TckshtE5aDFaZVt6z4h~}Ul+|Ja0 zsu)V(2gbA_!*jiDM+8z4{69^mBIJBzDkeCDE1HP@GSu2+%6fr2)er$fz-38Ps7Q(} zEHNcYB@87?#kwQJtq>~HNbIMKW)Opc<1lClbvRlfLwf@isVM(7#knO) z){&CePnK1UG9~BdqG}mIB&_v7MyN*(s;;;jO1|h|uG7!jxHn&4=ukAjq+iKRd^0SvAR@;?5g+0FF z#_BzJrNzr87JH01Gi|JycBdK8=UA(aAX~iRppnbkt)Z7{1mzt~`DFN2)lj@yNuNV_ zhvGob0aa6d8uFdhDfEyHo<)G)70L$q-$J+YGWe~yt!8xw zDt>jtyy()NcHOkdx2^+QOI!N18Yj{ z_DP>f(b&4R!}`4T>-K0=)l6rB}^(m#@7{8q8_@4Y4Z2pS*`3)Hp*7iETvZt(# zb+1woc=2+peZ4HU8*y|~Rm6%dyZCVD0v{gTdq?09SUF4aXAoq^aX^pzT5hwVP5ygi0%x7zhrn#z0i!;?E;TJnckpu zji*q3FO*fB@0$I^^c98>a#)=jwdgZNnGM&ah*15wr!W`Y!vRe4F@F&!15^We42)Xe zM_nWb^gZ+$3N#z4SF?r6^f9(hL@%;`VJel*g`}Nn#l@7nq@Rj zuc2uj2?kZvOg@(ChA76A?Y*+4b-YPRkq>4nTNysg!2NRXj1c{j8`JQ5>;d89ux zVoZ&|c|(3RTM1G8N{FI~=1ywtNw}d&AOud%u{=SaqJj$WvdAGM6sya*l?--b6-AvQWeRxN0!@E(#cM5EF!r$Z zeL++F&1p%02efP)IQ;U&VhG;!>JM8}TV}IItfeihZtHjhXek zsZnPQsd;jMxTim2-P!7Dw#(YyY8@=wYF)sVS|7E}V=q}JTh~Ro(WV!YFeBQuX|{~u zgksGiih-9si*A+eNaiLOo{7Du~QvfTV$%s*V{49_wGIzUT!i zzh^b*P_0N+@SFcEO!9d&&BG}3VvsIwPzcDn@2l^&uRyH81H}vuSmBe$Q($1~Wr!xb zQjMWQab0DO#70B65k_4&B~>pDBydI26CE@B5*uLlV+TWFX177I#YUgtWA?O>NKp+5 zHA3IZ8S_wR<&YiSpN)i>2%RlgM<#QkA#23uNSt2jIMBr}(H!J{A8uaGcCA=r_GE0= zvujLd1v-yOQX`3E57#GKk94ksIx9OTaQ30KzH2)?ux_&%3}J5f2k}s&$2hcJajYL_ z)|_6=VJ3N(?Xi|#Dp1&^cLBY-w-0hjfYDce9P1}P@-FMh;0kUXP&ZzxWylXNU>)qW z*)sa%S<^0uA+9)jc>(jQ6xCL%?iE1{uf4BmjTe)zXiFi^$AhYJ1Oi_Ss>ZfizYH>J zZi^87$KmhhS%kG7b0U@K+pN}C768r7SN&;?zADJpS=Lo80dwC@3hYTqkE;T zR5>tJV?G-Hn7-b+d~gF)n=rUPi;^u8nT9=r8=k;q4dbGLiNO44Cqdsl9X&z( zH4o3nQwx5IKJa)_Jsww&oje7atbjKks5d+9iaXTf_Had60TmY)saJ2>XG`E6$r_B3Q77PB-P__^(c{4Uk#6`k8M%J7b+5I+Vi^&9A71E zXpXOY!=;iCRr*f2R3fcP=Y&fohN|>;8yA8b#^VF(7$Y>{v_>I8U*8CST~1#E0br4* zBoR2tiKfHU%YR1>lK%iH+1!HvkIMn{N9fuAybwZC3KtZDrQcMw-`6BJ!_yb$h%t%; zGLeL!O1LUn7%tIJa)2I#oa(-(-t-^~XQpqgd0@r3o9eMFX@V zXuW%LFhk9NNQ+QH+JKk(*uoZgefs9MSKC;bW6_>MV67Z}vLp&WSrW$28^%wIz|Wd` zi(vDukB6mO9fyVR?S^3^<#QYSO}w>-{chB)p*E5Tj8m}~bpwGw6O5E_e$qo5U=WZa z)G%UGXx}4KFQK_Z5W^CD)|Ok-i<|^6f>Tp!Jk<~6d%y{gNuj1GiYLMgm#=5A7N}XD z@gg%;br_NsYmc5J6;iJ4l5ucxNNiw@0=2dA-cnVGWJw^$;4LT5NFj{AE!B#>t!ayL;LZmI`~jeF z^EO5HiZ1ko2!ZJ3VvNfz2yZf_?-ey9JTTYEl_A1}@{xd#NF=@CTIR+_!<=$($7`A| zM?m039Z{&Myel|Gd3r5kDK3P{k+{eexbyYmtnxCNIxvVFlQP0Rq?v52F4cn;tx5j@ z6*_^hULgON_44i2`nl>LsS67ivk6K9nemZ$JRZ%HXlfz$Sp6w0byUwGGR;r=DpPtk zP;f00)&*&@rEanyf-zdbh_`8i(SRmv^{BLAj}bDIwj+g;_7%l^n=N+dAetz`GOfuY zvA7tDI*e}FUg?XgE0LB3R+gpZU>Pm2ULV~O=6Y;QYGdgyA~BFlei1}WeQIKH*a5@f zo8QqIZAaaalAvr^EfiW)$D~${79JepR%_*$^Y(O_v^Aoqv*KcUnqWO~M}<9=-cc_g z6SYAwS3iPvPa{hz5#=gSM9bK~h&okX1|+nJAmNy_YsbFrnm`YutCZ$rPe-7Kq_JH4 ze0*$rk?NGDdFmTffc~>rKQZn*-)$8jd@$}4K3M3q@u87lTD_xMsDdnt^8^b7_J*fN zW(bLx2s7fQRqM{AG^uc^r>xL{fB)$~kmKX)QHTp4nAg zED+fGBAcJIrRX%D6)8nY-(|%lkVt!m1-Zs2fhgYF{E6^W4$$RgRvE0f#x?lc!p@2i z_KC8>wyyi9yO+QbuJzCKzuifsMoHE<)Hp1NkkO8ZCWcMR>2%Ot*@L7WU3f!8q{;po zNetjS{^!w|5%4)Q_M^L6sFivfwnmyLM>uCmhD<~)&5^wjEPJ7u1_h1M>0Ie4>7x|g z5v%(6YO%`l^*z?;f2B43PEKEuJv7fmIv!f`S4mxuE?4iQQip-`vfYQ*Eb6I}j9N!v z;L$Gvo?N^s1FyFWk9C*ZRqUlP=?_3PQH1|#=56mKVD=|>!yiKn9AzxFNdLlmVEo`b znO&fzBzW#v6)%P~74~7k^a3alMyl9~xx2GpCdu@TR{y(m7#MSRD&nL`ch|=<_OvLqB7+La^^~c@0?WQht>zI*mXUO_XBHk7kyES$BR;Qlwb5-J< zgb5+)?xr3OahLuB`LA0q-jh*BQir`cl7>^R5NSn$T$L?51U~DmWA`*IdR=0GWtW&3 zhJP|8h495m{^Nz;hkqJF=xxvnfK1C#4Hpe0*fk|&a7}iPp6qY>8HE9bD#(Nb0htS< zWQayoBv449HrMInIOP?gEZ^UkgH%zVsCo~H6wyGL8iz#tn|@Z3 z7^6t^A*ExJ=tW&6C?DnEIheQ48dKECCQ&t0K^*w9&HAt?A2gmUYIp$}%QV_y^)7BP zNXalp5(HbfA89StdGJL(_$g84$`OI8jU_FVxavf!I=hQAiX^cJ0-;W@7yS)lRLG}j z1%a3}IpmVe6}b>7?JEm3Q;S|2wRNxpk_N}FN5V}3Q_IN!BsY`=&M@qqfJJlJ%GS@n zNt;})lH1tA)Tz}aTjCC1Xg4PFcn}YIW=v?8Dw!cs#|jjnOYE(0 zCp52EGUXKw^qiIkA~`@}|BW7%g)6F%gbhnP%n8uv$f*LbJoq=1fDk;EF z{;*ll18g#9R&6V3atuXYj8xhavDPOaw^?-`Y+WPAFR&yTaQ0@StTiLC11?S zDcyYH!Iofporik1aeIXo0KU&zd7#BPkN%L9q>n~#pDLNCWDCJ;u$DbkP+zKYVCO|r zQe{$sPcCOzjUEhIqb4;#l6>Bzf&>=}BKONlZR@)Qnj|=gZ9vL-9BRz2Ame~&$xhreml+90)k`LxEU^U$ufjAaXYhL+%SJUv>gOg;yyQAG_OcQVR8l?TIepVOoJQADKN7c3{Uh%$)uT9t(pM zR&x{8RU!iHqmW-V_$hMo!s4Jn$dILCX>P}i1`XAUK4}uh<`!JX{5I8K0Hqh|>K!>W zdoKVoca3K{8twAn&Y4ZXwZ${@*+Ka_uaMtCcPcyrhkqeMOS(QJnT9 z8&f_wx+EW95?)#%rZvRJM1S zI~P93&T0w`e`8jy z6t@~2vD20H)2x;iai;ClH9*Lar|a80M4o**$KF(5jz5U7_dMOH#>2!gcq6~z!wM@Q zZRcF&3=exa^b)J%>{P#G48obgv%6G_u0cqlG3vcBJLuLsFuUOuTWL zk{A%eh-=eEWAAS3`jSxN3UF15i{VEG^k~KOxrCC8j5k=XmpF$q9NAh&N;+1AN!xi1 zpzy=z^^e5ITL^DYvUAi_%2BS%ADq{$%-$}m#{8NouTbYm8pzEDjx7$JU#p@nr_ZM@ z*U#@2ieLeBNzxCAhGh%-VaZvOm`thL5mz)fEQlA-9$QeoLj*LAq@QD9y0Tjtp1>zl za=v0nM;0_3;HW%uULb$C)R;2r$Zbp!ExD57k)1qGF%UXo$~9Ocp6SXKSSy~n4u7jW z+c}ca5WE8pp3nw|Zy@}hXFFv}BMGr6t|~KO`m5HSXX_OyzVq@msBbu0IBVLCsQ^Wb zM*;HbuPRkWK#Ul2c!)=k3t71M(OD0bV8=I`T(9-eLOK~T{<$kL7A7vbCDrD&n)hPI z5djlc-{*r(6bf2}m}>e8sXx(gt-XutRN6za80C1SsTbZEoXvYO1EL}SD`sS3sg{vY|`1(a^6aG#^%D6iqIcznZIx?RHBIR zFDp?r|Cg0mWF;@rXT7+X_B3uYnys`FS0I&z&naC7!=c2MSOoZ z;`{1|?`za|^UrW*jC>Bwk9tF80A;0BX=QyJ)H?Bc4tiAIJFL7k`Ek4DoCm=I9DI9v z>vw^Xr;OViMPR%y_E4p_TN6H)eU={=}@;~NdKWaLKK zG!=MY?S?qTY5aQJdijlJ_3Y5j-X~A#Yv&k%iVzcsWPGJXA1mq2Y89p43*KxA|2yi< zb?Cb3Tm6#XBNE8L@{R!ao;B^Q4rwyGfxX07&Z2-IR&IZc2N%A3YY%(J+VFNBK>Yaj z5~ApxUO~yrU?svl(TLY2JMoB(V>U-E->OSN_hqZL`K4@vvDQa#ha`+_YuLMw0nV@Q zI&fm6;lxJ5xqdaBrYl+912NIz)wS?9c}?y3-Q;s1EqLLMYYY?)T|=)XhhJ%U^(J2J zvUab*J{_yd+RIU4*xFn0ef!#L@x9@?Ainos_b@A6v1eU61DU_9Co-$Q_Y%7N=)EZP zxDjWQ8HHZKhF$P#5Di zIbFoaj4Hwj#mSgR4t%+3CF;)laDMHnBsdEDXNA=ks#8~sm6-=~mD0%-Lq0NSr{`-zZ<+DH__FpoxOC` zRVsBw71me-c3o#{rcc@wA`Hb(UnD<2-9;kH*-aQ-xtlOrx|=Xc-9s3S-9s47+(RM@ z?Ufj{{em#M>5E8=5O4#dfA7PaY5NJGxAr>-r4d5(evOrSz(GjvKxdc~D}-q4tZOPW z4m=0|nFohJX7?PN4J6XO%*Ee!Uk*yM6`dh!Bp0E!;xg#=OJ5Ew^U~g@t6rv6L$p9K zp&Y$*8w_o?=>FA|$}$cMX=V(q#%rbl34Q(5w0xPU3B_POfuWrAM0#DBic}eGdIk;e zTl2n7XB(|Izn%w_ulk03-@m@;z*bssezOHSG3gKqYWbmAAgkxMHeV8b36LaG6yVFE zZ|k7p_HS#;Ce^K~hnHP^v*aVq(0At}ZP2**2>IxjkJOC&fNaA|03~~mbVXT>?`u)- z-(O!Pidc@V-TJN-tG};`vBUcOhtJXP(xZg#=SN>agW?|@F_O;E+y&_F`LQ0*jXgFX zDESDNtL1ZV0eAr~BRx?divkUf-tRe<1^te{$*B?!s@pgef|NG+8oq;!{txjQ^O*{l?gy~se%>#W8nvRn{;l4Te~ld>S}ETn8m zyrO-yGR1nDvLFDr(z@p-;^vZ{vg5W(Zel}?wf(0KDARu?ZWjD}gUe0Xoy*Nc7Vizl z8kA1}oWI4^Uw<7?r2@8EgMaG?M$Y}M3I1;XEf0TB|3;>nc`65g+nl-ze`lPk-Czr$ zDi85(Ro~40$_>is8KXMJ2~XfsnpKb|I&zBS<3DXw`jo0eAm4Fp)|BsBo~CyLPhZ~M zwilQI9fU&C#$NP5Ud3Jh(9|~$0l6tu>_+uKVldSNsc6=T(}ZWQGc|LZRTCr#;H;D^ z6$bzcnCT_e zvNw{Dr=aW&bk4wrGE9$EfJ-VD`aKu0y2-yIes6^DbAP|Qj!mqpj`1XLm(46vZS1C! zUcw{2MCKp2*vfn*8B3rg*K>wC~ILRnZKRSQ`)A%;pKVX)^! zVGL`kFOu)^a_V!cgQYR7Nzg5g0z!vjB6>`-YiY!9HkwXUfMK!vTycq?UCri*56JTJC1uu@; z{eB$BQrHY}D2^4g=f#Ntt0f+bXYtPWcO$+Zi}-E?+&Y1_q>UbPuSZt^0`1EbB<%5& zE%~BMNn^XiK*KUM#dr*`4mK}EeBT%GUB{Z61E5YB0D-z?3bHzc3y3dZ6V5fsbA}*b zO(N@>H;2k`>IR0dA0cNL#L@+=>es`EQHXM%S+7w4Ni;}eb*syK864e<)Czk7;3)L( zsZ|Cdg1O?KNvu!SD^g#VNI3v;w8vJw$@56|OT@t>c0F4rIw!N{P^|IEY&<(Geo2;U zAJnCpwANXJrjhze0Rxs#2q8>~zNPLUUg;f3TV;eo+BzEz`yHp(x@|m>$Pqpeod;f$ zw8K~5qf2GSN^Blab6TL-2p8u8YI-Hs1Ps5l61zLz2J1ud$CX$ID0!vIAY_5)R+%*g zJ$F}TU08`&TbUK$-Ra7#7kfr@slw_xEj6^fy9yg0m_Y**$wNOn6G{n|;A1&JbW3G* z@Da>lJz2@jEY?PxN@c@PGPEjymWWr+mW~=;M@-UPnL;I4d9MGM|@JXZ2Jk- z%C)D!RJBJm=CsFYK(xOmEuy{nG>t8a>bnN$LWpjl@YnQ+n!vPmJgE7R*q;z1DgjNg zErWHBYMm*DXR`QIn~hFwUqJReB{Le{vr$cRe3WmWJ2RUN0kei?v%z@%ZZ>O(50_bG zngnDM5`2PJoT$rc#%Z+kSN1kCC%U&bqPkI*Peqi`SA4=8m-c8u-(eJ_(BRsEub|V| zDvUBuzP6K|3bZ`+v|T+JcuJF%Cx7i$rq=mdF%=YOcI#8BKE;T5cU(O&kLZ}gYKjR# zmXUBwauyp^NJ=jRrCj=kq)toZp-ayP~#f;_cdOOca2I(5;zy@bt^X>O71Ma|B{?GKRE#iS7;gAZ3ow z>c}43)L}Jjvyvzdt0U`Lby#nZcCe0Q(4@Mu?&`X5FmuKCby=(QM~UycDZW-POnf&wgY4^WEj z;yh*+jj^`#qjxmDloz#NdLwrrf|{acY(8ic>Ozd`-JwMT$<)GT~ z=Iq%%QR4)*HX@!#pxqvvj_4uIk_cwtiv(m~RMHp70mDELqWm(Y#M2aZVnGW;baTb( z7OV&RLBtiXfho_(>Ireue<3C`+rliy6tG-)V^IK1PKeS1w3;J6D_|XSPLTCv!2dr; zoC4A`4+XWzS<9BqT2@$z!mV&63zYzOM@#fFN1SfSE{RugB?u7S)<;3)2{Eaa?0Hcu z))n=CXoYcLiD=oHWr2u+t=XF(BDoD~g~A?fSPhI3BijJ-GVyX7)~<>@aw16MWXmw6 zXT&+Gy-c)h%i2}8-_uSDvypl|sV%z>pucU)8al7xusB?mY@OGR^@(#uB8u99Jxj#A zb_j9jh|+efNo$+=0SxsLsw0g6?h#<4TYFj}!xzvMu{5SjrX)(TYqW>1FA%-jvk+*# ztv!rqu9(+e3T#z-mR95{9?rbI76^Qk%_UHaw&$&4!VB zc}LRE@QN!Or~M%Bqj44l)x&TTC{_KJtwqM@`0X z*xO#W!bT_Rl~PE41d=$gJE|`NV|dHb_<%ZXG>V$M4tps zXMWXNLMjwz#2Nk`ndWCy2YBM+GumYwE13~TFk>A6G5DG=!X_DJ7-t0Vj2>L{{1QDJ z$%()(PPT%(#3dFuf}9|+h&Q~nuPi* z%T_pS!T8$>S4Q6u#5fA)C_Tu$klMjGV3E>996z9$U(>lA;x!o+A58iet5^NN?5}K_h8L$boBhwf{AE*U=X~AzqD;NMY0Me&(@6s6srKUh` zQc<7~O2TM^GHFF()g`Q&6ZSyW32FUT^7Fx1apV$~SxMreNhcZ$UkE>1wKL1T`8%pZ z1T>@3rV(&whKs<6!tIAi@36^CL$(`i{DRzE|ZxHtxoZ082Z zEh*0L!qS^du>MTytuys8&hTfd6*SDZ{-mbPMq!1lYPzWl%MSZZv>hCZ1796~Fa!wD z(RL#)7^`gZq7*!?tb!9@#mqR83Fc5LqyR1b%px_nLY*t!@;p8)3Xu{2Jw%4Nh>TE( z04;G$8=2-M*<5S)5LK=+cX0o%1kMZKK7s~TPC1o3M@n% z=)oFeRIS_-!O{ZJvnOkk>(mS5zuB+(i@f*odlBw~1`ZnL_hiq3hHkxBmlR0@ zX-}V7PoGmdzZa{^*le-+QZ@k}{dyxT{X^W|n;k8Bg;0VOA+k%ebr9Te~|SeJz{FpDkta1rK7}H?QzEHXh%s+5ap}7h6XIFNZ_<^3P|MCIJDoG z?oIRC`c!&#AGTCOSP**|`vxCBUMA;-A5PIIvi@$0_HJp)-Ee+3(f zRL%A)WbA$H3f92c%_7HTeUqp=h&2PBhYVuf5L7Q7#2)PS{Y0`&gcgnLYm0&&pz|luqUWMsNvFLl1 zESz|iEZle%>wo3X)EoNs>i3hXB4F7hjT{WC?;&SE?hz>JPWLEwWQJnHEkq6zHofL>CX7LtP~^#?@X zf3S2g`PP51p6M!N!7B>op&aT8Gy=ByA1p0H@f1#=5h1QQ+$0H~BPcUO+TaQiOSTP0 zp6nxV;|H^>^a;cibOLEO%z!Y!jM8O;*=-z1Rlg1?p}D2Au49kWV;-_a_}Fg?N_~|BTskt{x>D_WOtDjIz z=+SKw9(|9Pe>3aI=8JB#lxT|}8F4h$-$*6&9ism*mdSRAk;4#=-!C2;#?q_VN{PIJ~tMFm|6Z6~0MfJNt$5e6sU96BjQM%9}YYO{?v#F)a zA7ef6Db7sAtayIuJ5R8KS_8SdYkC5@$9yxqD^(SI9958v-Fo7q=`6c}Y3vOm>d3U8Ofa zg>(elUAk))^J3w=bl=lhQ};i;Xd*;4Q|Kk^gnzpI5Ic`$`yWvcrS;~qX`1_L!!ztT z=am@qEI2%0misdWKZbrB-a_+S7oUwUbzuT7`V zHx{uH2T18P&#`(MpiFsz?Q{yotxMQ5NTw#dh^g0MQRhY0w9~`|Sfnr~F9QvjVNaue zGR&C|;L*%CF(bm|1yn<~)SLKl$#hwQe8RjJ!M&NK>tAF(#+HjsFR>Q>l4m7TN@JG7 zOS@0wU&bZe>078_U`K!@(uhQ^>c~P_Tjaa~r@378euZVZZAQJq)^PW!;5Ak=4DW{5 z*<;M5?zfffH$Hhm>G4vy0Q&OLn^^k67cui~)|fsw;j`*|qUS!txAXxc=Dtwk^SDpT z#5?^R)=poj-X%T|N;Gi+_e9Rjp^2J^NcvKmZjXLw(`j0A`DK`X$(@g5vnUg zp^~pr2Z($iox{(7no1Z_ifJ@0SxTG&M_f%<50jc9^4CM-tkQ1lF{i4 z1aSfMAaJMoHuk5`HE4eHiL1rX&w}^Sk5&5a`z*xRjM9S}$x=TPmv3VA*eBxtP3$SQ zM`V1+irAjgr$0m?J~w{Ef-JO{R_P2M&3xnF_pu*2UY||;4aV?}#NovvehV@di%Z*V zVf}H16TQPghcFOx8+~u-oUN=a&)P0ULL;cA@1yS&bz^Chs)&=@Sxc-z*8iC0RsN7( zV)!rv>g>0^QQYz|3szMoP6;rxK6OnBpno7rK4vZTjdmNmbdxyrF{_z;dI9*ac4m0; zJimyHPgqv94OA7uR}AJ)n*JU=A+7OFY41H$~zfrOWKvMD?+47I}^*3@t{H-@&hYv!N$ zCwED1C+_GoOQ-FD|G>CAU>}6|wV1dML;3-+avxHudqvECR$bpoCPaID%`}qtUeRnn zQfYg|ZTm6y9u`yfv&*1{U+iZgws+s zKqs>fu(lDE_o1|CIdz73#_;r}K-;vRalIs-;c&TAf}o*SCiY-zrq?OL41$Wnt!P*so~kyUi`on=Fg!@2I|V)j9n<9CF=_8_|}@*!JX^CcYad@|wASUNbw!TNgf=vSV zEqIE72sc?e63KVzg;0V>yi91{Qi zoh0$s?^rO+ArTG+iGm0s8y;ap{@Vb9qQUoUD%k(-_ehDj_T(+63SOuJWox5NN|o{` zRgi6?RH1w=!;rP2RKeRzT>S$U-CdDKwXhpMPZe z>E&#b4(^gYiRQ1F+B6f;1$0N;DYA`b}jFPizd;%*q%f-DX z*sv%VRU!@5woqZV=G7|65lWhAkSiD+eqt>99F;m6U(jQsyuU9ElfX8y}^Egd!4{>y0?MEqsE^!LspLBZTVV;sujgs zru4#nJ zjBqZ0hIso|L=A_<5BP8itMMDV(tUW~H`cvk9BnF-Pq83dEgGLf-#Z1od{EqSiiMz# zQ%*tKz7x-%g3pf_fV0J)r!bTHT=-72?rmp94Xl_`&|MeiwngqxDzBx0u8^JLXkVnl z2#6p}EI7^bvyT#8@|e1(o(Fq66&Pm-k1_%q@0!3gJXRqG72(D~MLt_99y-GcI+hzA z5ckH=N{JUp(N=@VQG&3Ln|ljoyd^DaIgd6HC_Nck&9kg!N~Dos(4ySbdNKMeGhLQv zCq)diZIh&8{dP85Exb|4H~O93gWV$k73WpQ zvBQeJSCsQ={|D=PL300*#DjUxkV~7M!jdAa5tNckfs4n4;0uv?j`eWWZTLBM1=}sw zokN!L8}ZvYc4Z~$AZ&izhFzV@#gIQ)Te!Trf3lmiCoTyq9^{MTY7y!5LCT5dQOH*Mf$W}ydA?^mk$Ogo+3jMNq%np?~YJ!`uQmKy{Hq*$2rO=o{!}%-G@W5 zytVsKJB~kvy%%fZ_?1XRrpH4Qr;0Z5e0_w=AWxx{@f7qNJ$=(E^c1QzdMK7{qlV)0 zwTz%4t3{2)PBk?amzD(h7S{5DCTg35ocAN|6*XEso5)*2arY+jTy|8jB%bOjQAJ*s z+)j9ah;ln>m|Ky=n{{6*2ObXVXJq|2xJnKq(snToLKX@7f^esT(ZaFahRN1Z66Ar+A`?i%N{LON_p-%lVc0 z0(k9iIlLA2lgjz&qdB~7!f3GH6ReD?u+D8&cn?&nZt(W@ixl-@Shyrrl`Ku+6!@Mq zct_DHRl@5T3QX{A1;egZ`J}G+C%T32sX|z#P9G}wu81gF$Ud8b# zVt=Y;2dG6=!RqCu3##(ry!@CEPbw0BA-y|K41a5D-n@BactYE{+yGv8(t(`ps7X!IB540gkT;UXg z_HRlDJw5Y(+d+58;>}(Edv_LZ%)S&$v-p@O=9Ha0{LfgBC0ge2BnS&;)EOVJplBMqsFvJO}f84}EfK42>y?e|sHdzcMb4=ko1KK_QW&L)dIR{IQ(fr%6RdhOC~t3D!u`aNG$#D=6YXmz*|sa zpaHLmkID`CjTAby_ zS-2Jk4aC{Q?Vlc#Mz3a3e>H)xVW;0AZOS6G_9O!+lnysVnV7_y$%XoA5e`3Ei|r5c zMDbH2KGYTP|6bfOO7v*T>%oPLYsx#|V`Wp`2_L7MN*~<5nf#cbK0atBdfuQ;Yj(@Ex#uYeDv#NNZ zfM?@1{BBk8Ma1_?EyLwqS|Zh9*BS4A|3BqqKWz=^Nux*?_1f^nx^}*qcf>lY+U7(i zLet1!N%QUHZF>Gq+}MUUjcm;Ji`8v#Xv9gB9c;@R_4qbyFxc3JHB8!wPS!*)=d8Z2^T`r9Kze;FE75q*i8;yLSS1&_yz0N$DofU06^LIh{ z+0MKcL>%hE8@QCO>K2it+}lkiDbIG3;+oo3ifd~ZeyLkM*cD^DvnJ84D@K@|Vr17c z)q`SVS3tF;D-Lz#4G=e{c0=mNv5^kl5RfkP-=;t1oF&`FbJ;9W{qM#@5<97IH&kKZ5 z$&|v-LzT*gGcq?&NrfEyd-CSBWXKUrrf`NC$rN*t!Uo|OC{60((YhCJh!Zb`y?DLU z7pX=_r&L7*rZ4v5t+4&}n_j#&Ru2*{MZkMllR&qB&^P+Fx0 ze-Bwp>~wGKSwmltSssFqJA$8sw^Kbozq?KNiH4C3q- z5rf^e*GL-gy+(praE+w#$$vmsz85?G0US>MM>rh+ng}V2Lr~;h!*!RQi$bJ89KQxS z@U2L@7MLCu1=m7qhsBlGVw;p5@A;rUq-j{X%E2a2X~e|-!(zIs{Q9+gM7ZfC*I^`4 z?`lLT6Trw7Fn?4IT+rmbR>bZB8K2@8G^tlpYBl=Ol~$j;uAEk{!4_Zx(?s$;jfU_x zQQK=|ZruhIom&@A4v~^sKBQuZQo(kLABOPocA%1>ygfMe#ZXA`OYzH4-l?7(&v7Zg z!3-P_qJ1X#Ml*YxcXApG&xk#UUy2*9m${XfujgrC(8lZe5LZ?WZn&V^B@SFTU2^{o zyt%!GCl5T~^qw3UKe&Onb9+g+@dCZXtB4}0taFzUMUrChbvL>R5_#|sC-P)3i*MxP z5IyAFgb?9Han((j%l$6K-NcLCt|?t|-jScH+>BPXYa%;{3@*vG-_5+a86HjJ}1>j>-p0b)}_449(H%N+Sx&6)V~gb2Cg* zIJWlRJd8IFg*l}3&>49VDbaabrf(OkP%la@u1B`F=@r}>Z4)sVJFrdNv9ddE0~HG)@i`9sT5E`PR< zkUryBRGWIY^EOytlJ1Jr>XI`h88?#Gaxr{%ByaBmJv=gEf^$AZ!ZYNH!eMvzpQI!* zabw6|$e_DmD9477N}n%kjN-X17u!`Zbm)b24Oxm2Pk#_x=`Ywirku1E?W8Yelj{Xj ztDlT=JqHC>1}4n3U?eg}^JDB?5g5bkWh?C9GI3PeMJ-IiIgT0%f|6k zZ1>(Uj$aMA82{u|n?E6sV8p-!V7o1*jO2SY{Yfgd)y?$k?-J$mQbf5#cUTPnC(lWc zm7s#)#$hq@pS(#Y71pT(H*Q$>!~N*|q7U3`yO(EoiGl-db|S>T2deRXj}& z`&(FTy$iXXFBBI`tNjZT(tc4mI#B->g2Ysrw(=8;#UtZ+Ze_*k05+#epF`zq$Mc-* z5~>_bfJHz+>Qo($h(nd>N&GRMXAOOos!+IsOAK)S-hD)3(d6nekk)V#kU>PC$6cD-%;<5#~jC5GEy@#i#M}?uv zIoc|D+dXj0WmJFDJ$!JcVlo{s!`PJ|w`XuV_?)IcAv)g6Q&Jp#M|gs(1Z|xkb}z5_ ze^A4Vq7)G`I7b)%mWC7Or7lV@0L3CNFpazbd4{rL09)jSG#ZNhjtAIrAFr2qPKrr%7+YNRBv(;`P)HJMjU`7olPp}k?SH}r?agvorG z|1GuMQ)E5Nb6u%*dzin?9u?XYXu>K{dkU}PzPfS>?~`HYcxefg7zW=4cc!#)4Jzk+ z7SSDW4b7A+5{wH6UwD%RcU0NI?a%6qjrIouVtuO zQJX*$=&(xy0#Q3*n>n4=cNL-U^b17!2}--lWjrKX?2N}V(|Pl{kx@S8luncnk43*h zxR1OmF8Ej8l+QrEbE&dH(RK#E7b^pA&EVl9;lItmT7!F_d$!=o92Wz;IE%NAiX8ZV zG)9ZsPnQX3`F|W6Km0V$YV*#biIJ|>YgWRUW4hA>x%Dz{bV?2?Cr+y>FNgzA^Oj9z zEQq7Ubbkhh+p(B-;_@yq&g;$~q=)q4B52b)CKQXgYOmQmqpqAfP%NujVY-Md%GNec4Qc1^R35pn`0gii*RT%hlV#ba}M_beqWiols#iUIlYN?6mxiMdGY z%@;`}JmV@E_F*DH@{wyc#3h&Oi2X1r2cdDcjN;1>3Ye6hw4u<@kK5g3b8&wO78Xmy z+>!`E;Ubi^*fzxoW`J3^?nHk|eDVVC)#P0CSvYRrak1g~XmwE+RzU&v zS)%U}-XQq2k`5FQ`QKUefNwFl&CtAOG+hriOO!0(!N%p@0DxT?{s8F$2`3Q_3};Jy zbUN_(5}qA3Xkk7mor?LlCYGY_i)98aEn?96d!Sw^59&Xh?v_@03A_B^igK3n`kB&M zl7r5K-*O+a=t11F6f0gOrD7?+pS6+iU?x?Kg3|NMZ1SMkX zGCmj|KP>x;ew!_)eg*m+DmEA3Yvat{amfW6O-@wKcC}$Kd6!^UKpz`%LJ$W>lfkjs~$T zeV#y-A($`)SMruon(O_~G7iYt;%{M6C?|8pRc?eD{(b;w3mo!t9he z`x<_;Aq_bt2QwZsQh(py0^ss z{X>}qm%m;ei>BSz;ZWdw@vn7QAeyjH>qY{S}12Xm`o`yk0H4>m=1R*B6tk43xoS zv6%5bFJyCPZp45*qcm$H&){rnX~8CbH^b+O4|#1~^<`=)af>%GRQe>rFN@=wd6SHn zqywgMc_JIbCH9XHdM}l>-NGOA0?4|L`C3*}UQ$Q%nMR;kqibj95ieTsMh??(leAd; zDW3<5yYJu+z~DaG!SC@uqH-ys&u9Dy^8T$p=Tq?Nz0Y|lW4Y`cdv@?6CZkOaY%}fJ zGyl!=+?Eah&FgovUr?X4j!elv$izn&;bK^|kScz6%b4yX*aSsa5avtLo=+xNhyy&@9!@&T3S z5S_>i13?G_%JhvGy%&kbSH#nMVZt~!zn8ztUKO`}!LI_y55C~pIrhDEAU!XLDGF^L z%L9(mZb&xXeK65E;^BSR6*6Be-^Vj+NfAOJG+M{Q`S0O~Y*kPSyQWJLnxX5t;>12) zvx%*ba~OV<2DEH4DH3+)uQ6M>6e0mh8 z;bx1geuVOUQF_;pn6{}a{}5hN_K+MKekz%`#XCY@FyhIdH=`(we3Jf}*nSM&WsCUX z81D|oH9yWT_ph_9`ibMbTiQBmOg=A2_ftnSJa(LSb*r{L!P~eG_npA@T?fdD6WAQS zReXDb&veTs{=_>}w*dny7*|_NatuyRV#$lpJxmP_{=~bo_e9>${2}1A>}Rm}eX;3h zJ{%v-Px4;)xc?;Y?!w%DlIJ@QGk?LtjDtXzUyuQI9*TcKg2j1w>la?neK`6HFR1hZ z5gW0Uep6}lUwM$R55=J0_#GYIp^GU)$*?APmO@=VcEg&N+^B;g#7G{4x0)`&7|C!t zxH$wrzi~hXFe2_0PT;Q-Lr+1ZYekRKJiAiycOVzwurW^Fo4H=BJcVI+ov3pfGCgml zf1c(!cq>k0_0; z_jHFUKwiz@|0u12uQ$RoC|lyRG1y}y0AIO)!CYJtzeb8IQp+ES}rV2m0lZD#-HGW=L z^@($Gf|vd${P0%sil!|QfBwlUv9&WQX)RJJf+)5#+@HH;#&|6wMp?CJ{3lNj9AM@# z{B%G=CUaAkuy*O3he?PD~py&}3)(t8moe&W^gk|hEmuFKCL6SsS{ zYQpeo>F1>`OM@+)0&-MVi&-g$7r*a*K7pj;cpv9Js~UOHFx43buWG{ z3brVQ__YkwxZAIhzkJ584Zzc{eyu*9a$~hS*&E{NSnZqoZxg!`Qb<$b9FfGz9oStg@GHGWpSK+fU-17)SCLgUkrPPdn-)TTDcEP615<(+?=QhbaG>o)~fOv zg%kAb&p+3bvTYR0lC%cw9kDM-D`0Cx)nx7M%F@xOKmy5R{Dj0N@l~=mJ#Do_gMPdS zWT1Q6M%5@2G$EYL&DHs@j3WQ_MWY?OW-+xYBP~CD1nqzv-mi#y%AtD{I9F0@ha6j7lF7Z6~maQ-MuvWMG3G{Vn)r z>?}kKabt?s4#V;@DcUG-wMwdX2QCAil&THF)BaSgMeWaI=b%h8pTZAx%h&^RMsq$d z%*ARwFpzhmi?&s@f7L!q;ey&H$mPZ8@b0=qpHf%ehD>ginj`jB)jH!qOR$<&KlBid z7Zfad^*>~^fuHT6J`>On5-OEu0J5KtmDw*OoL))e5 zEop6sj>=oY=9|HRz*~}`*)CX0%1dz}((2d;$UBlq5GsWO$lf9BL)#**4rx6RoGc1y zAEK*KHMG2HpUc)57IXD4!e94_Wi?>D)5X3T(89UmvRrMvzMVK9pgABJDE3zF&DEOW zWu=rBU0VyW%1*RRa{({(K%Y_CKTqq*K>P7rFMb%T z^kBZ$i7u~RP+zMd=GF4HY$)eC^m{4N@(cC-QX66N%sif>3^wBJ0r7HO?dA&M^#w{N z>T3;CZHVw86mP%yc(RE%A1~Xk)CT`YElY1G(5lfdPu<-T ziRd|EPD>0J_<^35T07fUK>^5G@q-%9%bZqP_e?cIM4&;=1=ohsU8T4yROUt=Xr|Ro`l@(J#U6YppfKN98u!_xL#1 zMyplVkvg_G%FqW0!dx+Qrr`iwD`BdPB`Uu);6KWkDaxr zB5L>|YCPQ)cS)0|V?H2P_Vd%6SDNu{&9Yks;1`Pm+bHXPOmdur9i@gADj zf6Nxf$d|Po|L2R8WbiuVUuM6-LltpNPpx^*d^ISOm&ITT%Itf#&fn~4Ppx&#LuB9q z@#j*FHa%X_ORM4HF|wC-seil8=rz5xei`2pOEX{u0B`S)b->rTRIBb{)%Q~EO6)mW ze5tnDf5ZlGTW_tvztoN#Ug`~Id?yC?(S~OoA)o*lu+uDZ2IWj|zUre5bf!0@9s0tX zRJD{6S-*ib^?UAg-m~9HVF~O+RXr7f+K8oIgx5bp4PpM`}6jkQh1= z*8GF`=nAb?>O87QoTZw{G@2!&mzY7?EMPf*kT&iAHTUN6Q58$yc+PasNkSGTEC~q^ zk^q_jLf98UGHe3MrYIolFi9pPn9L+I6BhN7prE26dkTd>07Xz)T!)>wvbcfb0*C|< z1x3N-x`4?0?e23XljyzA=lA>LdEeYCXS%w(y1Kfmy1KegpHxL)q&U!{{wH=GjXh`C z6x}+Wb-7N_1l9;jt%{~pYaM!!uwoaAffA<&BY&RujfcRmqMygJ^{TlZ&AO8{#$)N7Y&+;4eHW`lXMVT~-2I+n?`B;F zn%oc(H@Fef@k011baapqyRR0TY)$SH<3m`OInA~thA+~DNkJ~)vrVGG7!2XbNkJ~esL!Q0sTCCElm49xHgF*hT)5jQ1!bmF3QF7z zsQU&dYg{WWf@wA}AT~60$jJ)O|Le15|0iY#`wuTbkP~V2KeB%RaU2%rPxT+)w!o@X zOBEuFw->?dgW0nGm$R{&IY*8uEQwysVLS2Loy+c1kJc39Vi8cbco(ZM1Vw#?(0mtB z>e5>-R%+-{tYQ&pqG5SK7C|Znr6Gc%{Pa*3Mb{COck+UQf}$c=Z3qhXTtWO_d7;WB zs>TX-#iH5XlQDX-L#^r3WX#v+$(1izHzz-Yb&?>cyo`B9*JqtL^AK*ZPJaIrt%swK z_*~n0k|P#VfYDn;Z421*Xzow}o2OcBul+`kO)>OE*HS@?8P=ztGh) z^w~XZ#D6tYUC3R?hWy{MRj(A8wyFc&>t>VxZy2nIKcpFbDVmN?V5`Aq*sa0jD`D3L zxx2a~$Xyx0Zh!@e1>8MW!bbg93n#f7MP*)geNcp@SrOC0|DLVe?lWy&7kbvu3jY@k z-er1cD)R<8*?(G)lk%NA=q_yY8JaN-K0NMFoW|l|6+fF6RC%0+7s{=JUw2Lqsyw3B z0vh)}Tg8|ZoMeGjJTg6~@-U!c2LVclEA^o#Vs3!ah0LeILdFP{7smMJ8LTK=O{9$M z*(?$B#L*c+Wyd$<1St&+Wp7Yl*@dh`8p;k!BkiOxrg(81?T)%v+R1zGt+kVx_eq^N za3AwS82jJP+T)8{_uS9?K_-5Ee|;ueBokBq#fH`uNAae&IpLQ7E6CL7TBcfOhA>r{ ztq`XE-HxgWHyWpc>M^y}jtZuVK|N2~XUYlp7W=W~1 z1%YDQ3VU;fQO&|ES>}r8E0KfmTRrzMqEL9l>%&J_tEN{4P;S5C?L&362)E*}x$L#z zN}zf!OAY$m-)?3%G*-Ke^8Ab4SP;#MHS-ONQ zZUS2$2czmP(xc>x@xyu0>?-xYY_Od;(rlB|Y{8 zyEe_(pSxnwtnx#2D2rgcybbiqN(=!4iie*FSq8EBtjAEbY{ibSc=e>*GAag92824) zS{E@lG>7m8SP4;5p(P8pS`9;j`e2rbMO;cmVLL^0KU{v!IaUmO0F<^kXEJWGlD8UhP!3x0eq(!xBRk7d+ ztx$&q6waheOIcF5QA4yH-%M<;r^OnrV#C00O82k|1k5X_vVyfu3gP;HZ3ErBIef9P zE*(p(UqiXhS9|!;QTF9jMXW{b;{QJq3q%=|yo|L8VjH%Mc|gv#Wvp#*8GL3L%MTtL z82co9K$}OOJ;@g0G)Tcyu%)YL?o(L7R?yL>Sf;u1rMTtLbaPGIdpYZb0tL%ic7m~> zmclDm(Ad?(lJ7G*wVWLWuKy53>)y3I)t#`bV5bAgx0} zZN1P~8qwn!ynWMU(oXzyC@s)*4eKwWez&b<9fPv@F8MkNVv!6iIGqLSpQT7g{RUVem}FK;sF4 z6YE$Xt&v(n;>1L*IG~6{WEHhv&pe4wh#sXjfCP)Rt#CqKctIJmm)EmhIMZIT0n4#@ zbZmnh-=gPOGtlzTb1Vyu?R~CZV-)#3zGg0QCIO}JLqbtzKM%n%hzlHg9&~j#mJ4F= zg<)x?;=IhRzSk*0E#TP7qZ;OZFr2fjjhJN~A=gGGzT&rGBkQX@N?&h8Xt#ozZVEBL zgEk@1^p7E^#UjVl=$)I`&{iQD{g2V7IyG+I%tmA^7R-YoL~xbamdM0qO?{)(xZBXQ zA@!%Z;X!Lqe`>n}U3d#K+<&JTBf;X3bNyP(#)3MK&s^$>lGC zS-|}wG}*vS7hh!J)YSiG^p}dZvq^F4XbyJxi9bQEodn_yb%)gjN2PBGfjJg-2spAqc>PyPzw9MiLH`zwD3(#?91u> zH=$gnz5M-6)-Rao`xcuUOuY9N>k~}Gyp6#%>-2t`T?kH#o!$v)eC#{2@y+jGVl^eS zK6N8$_FiTKIZOAl>q1M#(}}$-B1rTPdnK-6``DwQu#uF$AM5^e)O|k!o-6S6I_X7K z?PsZ}G6<0b{>dOLWAU=K{6T>(|OT5K?QNs`b2j>_5Po zXBbg=`t#Ii@ zzIc!&WZ;`%!hDJO7UCom=683a{xHsRmD$Cc)ju9&*Jw{tvqP+1yxfYkH&fCn{B2lE z+(R_}5W8mZGO=rjl?C>r{;8N<^V)tF?Fc^%;wL_j0c(ITa+tlTh^C5)Yv|2G*k>rG zypJ$3mD9S9SeHZnIa zh3K7lM;uP=`NOPDnNnp90M#oQF%rGNp-<63jAkMU@=1VkY)gEuPFP@Os}ztV1pT7Y zFcxNc;81)Oxkecjc~XOGEOSzU8WRjH{DigWBwG)b!h~b7;}d70y$KQ`d-KI7EU}g9 zjhyW4AVmcE3=n#NeVZnqvc)m7hk{bkkhnjzeb>y1B zkk8mbEnd|Rw>Q9<|LQQGSQaz~q7Fycgjz)SIz3KB!lnvIQgSFZj-Eau5YdN6uz&P9 zbvz1pVm%E$%F=8bB+sN6h=WucXxdRW6B9!IA#NNDOrT?j_-M<%*@5dm;$y|euI)Ip zXaEoZskH7e7psYOpK!6ui<5*2wD=R=j~pl1aLe!K#Bsyhk&lyxy=dd7T&NSv&i8jd&zv<&OG6*ToL#2GHr#jkJ%<_aaB!4A%H8g+&> zjZ|X<*`=a0Y*6GJArZLka1Cug!;%w~p{ec7HVb@LD5<2) zKY*>3bo>W4S6dO7d_iqeZ@!3y$Wr?7BJ6D?{d$p|f)jh>N6aLb>9-$oYT*jq`V%DR z3eEh9WvD~C@Sp?_5d) ze_`A3xbh2P@D-H*Z+2(WYl6Fi?+vgC5pGz8C~PZTp)dcCfH?50 zM9hCC0eHo&3_{9aMR68c@FM8KSJ3@`vJFjw zUl`~17E_e6?V#RQ*pbBTQWbQ28BQ$NwN!8cW(`(;8@2xn3*>E7_80Tz{UTnoLT03W zUODI=5`z55-M+MW{q1+%O+lE1`FNH`R2m~vWU;k{Am3k&z-F^2Kcc&uUky4tqE zXkqquPz~ezuHB{BZOKC9(~6^I_-|NqTP>kR;lPKSx6(q#0nLIj0*^)Kf8bKgKXv{a z&XxTV&c*Sv#0b6xj{_0>25m;)`N~wBcnB+m67ZXso{zw~B7WTEZJZiEKh(k%>1}?bY;d3?`G+ zbRmX|=v-_pII@KtvAj8e_r-$6J854mzZH)b4f$Q**1ZjR2kj|(wjuA{V7pXI76m+C zH{=D{b{cKNiI*!h&4z*7PWCvS02!MQ$8SpBCdvv|4Cj;Z$tkQ^W=@eWdSG|jR!wij z@%Feb?}s>kM{O5GAb3aM_D1|6t(BTe%;`%Y!|f+!V&)K|T^YFC7^R@X4>aXBgv+DM zc=u&*Q{E$(;PL!M^8kymwYSCdoCJei7U(Lo@5A{^gsg2CwGZQYF~XWz3A|M!Sq3M5 zSd=(h*KCSyGnFLpCOEbDPy*lC)IfoIX+sf=YbFuwYsUNb-XqA3X%?Lf?-HWK5mk}* zubEac9?K^=heEAnmqaU=MJlOZB1YnQawqZRNSyJMt-nC!iM(yZCdJSsawPI*^mZbT zOH@2+WRHa%v6VCj24iLxHCj@163^(Z%HUL}nJ3WTm|zM#*=EAXiZ)Dnv*Wn5Sxyjf z4g4R8>ZK&!w8?HoLA9(}bY|5J;8-klu=SZX*eR z@7)Bv5Amsk?rmB)^y*)&K=M{v-HJDkQ4@M}3@kEcnb%vvY;2*+t)yBcr0{`Fq+u8w z6Bp9=Djdw?cSHK$PvOaFs?SYeL_-=3HyNA8(O^s}Pm7g0q_AKmMg3C2h5a;C~JQc2dqCj#gpN&9ijxs`%e~0{Q<#N9kxs7v3`b zAtPdUq6@!1m`Lr)Zwn^=)fKkIEQUlXo$eY^F}xeBjhQ{D8*dj(ly>9GVrR=tSN0B}FvIuo1DkYq*^aSTC=s*uHBF}w#LUzy5?LFn8!}6X`A)LzU$w%Pv zt4LK)axb2URIgsVJC3ordx1PaAM4F)16_gMJoxe!rpag=jFHiyXb>y6T`aJ~KlkRV zHRs1_S0NbrQIUxgb)kqaTf}m>M%^Y;TUddAs0)S%B<_frV}yGDeLa5_oqxVBI=_;> z?aQw*6hKB4lKSxp;eRRwbZD_pc4 zH}K9aRiOrUqp+~#(wgfa=}(I6&)b4qo%-|CCJG0xQ!sHf0FK=Ld{4Nni&yp9W$~xP zWc9XP(#{7!=_{%I0Hl@%rVIcoWG)yeGd~>2)6HNBlU-wjFeC;l>NE)4K1zoM@l@E2 zp9k@;T7NDDO&2dpMq*%e;Z5MB#X6_yE)C`*v5Flwgr7!J*WJkbf`?OXsoJ^;b{u*xZW`1&a=>g+jKKeq7R1g`$fwmcsI0M zK3w`AtA_KA+L6Eq!}-Sl->7HS!P<=O9f7NeE9v18e0Y?xqlXpS5&B{TJh5Y>jpR)d zRG)>$rCXkkNRwBFinj?njO2F!-NPe6-3nSal6S%5$ViOO3c5NHw636zqxiMz9BC@T z686ksSQODD6&5WS1=IZ{Z5zdVB}+3aM@6cOe4SR@2gB6XWW&*zf@-MiXv7Up()`i9 zzY*~ev1##o=CRSdCmL-$hL_;lb%DR=e3SVv{@@_@6VFm5D zRSMXtTXB|t6*U@*R^r z7}(GlJ9hLdY5#3J3p{UdJ8w}h)>vsaeNxwO;}rybHg~8$qJeF}1&dPtR7{}KhWXLUV2#z|cyj|Aa?7B&3z1eo-n z?|=kW(Df5}W-tWZ|L&0HXd=HC1sdJSGg^cUrhK0W;d$|L2W}y^U!ZY!V!kuGIrA0`2?ft6I6&ASGp{mxw8lt#ID6n&U2I!di` zFkI#q6LK3{FaVw$ES=2#wm270rjEJ1LvR}mklO9Izh?tWEkEby%4SXrnAuF7`X)s^ zgT0vN;{D=-o8piOx&BR!5PRb_F1QaBbjii9!y`S9KZ*Xnkq3LVk}l=R{w7a`9xe^^ zp3G-yIP$zBA9L2%bUYsu$v3pV0LJX=z?lL*mBpVG>Lr(l5pW*y8b9WK+gU0p;$4zz zL>>%MgmDjm_$BjeXjc)xh*mebv6!r+qi$}muP?aj5x2(+Ybx|5qIpVWsNg30cq_{D z@NO-Y8wFLx<`iZ<{D=yX-GC=VdA{)Qf1$^B6!Yb1``2RG<`S2juS9y?)#pM80q;PmHe9&1F7b2#T$K9y~=tEBrh; z1$!RqG>7^|6qbYNIL^ril~$ChbBF_g8&gBJd=b9F46s-Z5q@ErkE^Jc5HH2xB6LxT z5Y7sEs8o*6OQn2pkjlSGLzvTgDj!oj1duW{1URG?qy)XHHJ7hVc;WWwn{nI4x zzn=y_FT~o!H`JS_gom=pK0SoXi>C8Sz;W_EiQ~I}NgU^9@Y}$t{`bnUJ$A3mYkZ%~ z>v2i*;LXw_d^6s^K;~We&nC(Ao3ZwVJ7dV?LVur=q5iT4o z(sf~&PvnaQ_5Chq?Lo|^orARrAp9bKhj2UF>njlXDSb+?#QljQ3Ce zOpt(YJw+Y1pha{AgpM{3#a*db%$u~hA}SkB&R8?+69I%LmWdK&Mnd*Rq3B?g6LCyQ zvC?;SK(rv3bb+_ zk53M1?YgjTW<1{BS8fkhQpytEEZrQVsG(u#kzBWf2&kS-ksuS?c7Y}>;Ypp0(k5L| zH`TAC$S^8hqzy}WomcM;E`c?zqL`(;XN++o3F8-u^}(2>aG`%9&r-fgTNOyI;8Vq` zhD!q?&7`g233ap+}ht#+q`b<#M_ zLWtH6lV>GgpslD_#m8%xXu>MKG?*A##cx*8xgM5-0@FWg!eM%?3a0BYeN%g>R=wLIDy z{)eawF8r__Ve>0Aegj-R6E=SX@72sWU;tiTuZdUNZD#HX2lX2ly8XP-u;1)YgFVgQX z$n|paMm_{&xi)gK;k;lYLaHn2z((wwAb7Kp4{JC_nLn}9C&?JF3DKzvTChnL*|Lcr z&N387uiebW1=?0Y5oVPzo5UdQ)W&JxIv~7oE4RSut<6{ot)S>FnEzJL-CMXgRP)>x zSo0P1{TBG<<<#Ltw6}t8ev$v8Jwf)D_*y*9yu_aX(EqY@0A6@mIsgY>=8pq-*DDgZ z^c4x*_6pAmpR4AuidV6LGM`4g%Ez^QLiBEMjCjfGx-gvPj)rR3HM|fdUUjC|Uj@f1 z>GZ1zo>$U~uVMS7lI{z@TdJb!054VBAL1)6xB$Tq=3+*YSEs_PV|1!$>{cLHNmIA- zwxLn5XSO1Qa1pvGkF{e@d^P>J6?CqkrrWsK@xN^w?-Nphqri%7vOu*gP_do&8Sspd zaIx66fONwEVx9~dWQ#a|_RMCqVp}JRVl=SLD5hEQPXu;i?%&Qk!vWm1oe%2#%1;-bOs_63u!WqEr!h>uo5zQAos* z*gm-$s@#X|wM+EFK3=4)2;8}!&lYjMlLruRT}cV=if~Mz$GeEb0DAp<=;TT&dJkJ@ zl{Du)-d|U0Mho|2Z{W-Kc!Y7X0gQ{Dk_Nc~&(p+NQK<0M_xS5T^7{LH6dH{C06K?P z%|GBn(v%8Ct+C)LD|;9uDa_tP%w&kM;L>4yJ7gt2{Q*A`zSwwKclL+8W9UZ0&JX$2 zj)&C8uBMs$akDG0e{jb$l9fvTj4D3FRg&t09v_HXnMa~ms#=ugN_Blcg zvrh1PY_jPTY|C4qd|`N7z8CVs5JYR$~)cM>ZcGf{XFOEELC z@FaFm%*4khVGO@}R$)sz#ankUE{u^5J){Kp5n@3;QZBHiagiZr^H!OHQ+!Ae*K4PE z``{HbXHW4i!9?n5o)}EnPxGcp^0p8a&^Pv2({P^&I`}&kpXLc|q@<--QgOuHIH6*U zxG+BE;6HyFJiJT?PQ%Jyp>bbH&d>OYkBc$}3?lLZ9s7#+j@l?#E2cb%a;r0Zh;cs( z4*TK6kd&m-GrV=IRABVkkcf3>_<(B_DYYd91+}tZSMxLIZum9Ay2kw})ctGT;lGam z{a=gmr{!PsufsPO{D0&do~EwMutNos7<~iQFhin?m43>a6)w(AC5k-q{ zdAnbuok}e{~MR1yz>WNYl&L7mSN3XUe2Qq2mF8Xmu-CF-FJ2j zw2iUeVF}D?Xnlq^2n+rN+BC7glhnEJkQTPU0BU%e|MmJv3tRXP@WMdzPSywU?q7FJ zmNoG`9H)ihZr-c);SX9=xh5#j%))qHw5js-;g18Wpdkh2c}V&1jq(PZT6VE6#0`U+ zyI2zv->QfES(Ftx4$=vceGu7}5LCAr*`HIxuE_oz*`l=hmt|ysLH$Mc7xiklH?qGX zkI4S2UiKRe6}I!VN@Sm}hkS}r`yzcPvM<)lK5x|ig|3S1U+QI77}>v5r*77Sq~Ggh ze`aKd^`J@JtnrCqJ?iqX%xK(3^HDv~hU`$jXhwE3+9R@?Av=^Wi;e6Qx*)Ps>LGu@ zXgrfzb+;yHnbfnpwQFhsmBu7!VH5G+>-g-Cdb)EDAL~4UEvh`6Yj!0(#Zo{)2&qzEk8l2ci7tlHm5;ojvO_)%F zT!D3$!HA1S8X);M(&%y6NG&V|kSsjI@$6%YhAI|T4!FQG*Jxz{l>t4LU$oQOYQhW1 z6*RTb)$}zFE&xm7L4cz~t#PX7FBxN%MXmOr zRyn|emeYnjE}&5YTm`T&v~$2j!NzJT+XLDE$|!x2Rw0>a8?P$o$EZ>%NH;<0V)_u- z^w%D1MCv3%sQ{LL)kdAy4H<(-R-?fh13bwXn2XfE8ZBf_P?9$qPeEpn2CAh=xp!hUx`7Y*$<@C1g05U*j_x+4bjdjt1Fy)jE0K= zp8!_ZJfelQMft_`#xLSg(H;$Y!`dj(hTmdTIvSlxL3#($HuAgy0Uo+Qk+#LOb;uHb zqYZezNjhytaqWKE`vz2W0dfV>eMSdl75r00R|V2)WD7=ZH`6!q6`W43-h%NiM^Y$JmC-hS%D*bWg^X-9__EI^XJm(UFxXiG_&@+J zQW?UwkbLye90WO;P!O+4?q{YY`#IqTm^9)T}K%c#3Z8fZFwL|uH1*2>56x7BW<6&E)g|$R_0rh_y<59C#p?j&Go?Qf7OmtpjkZn|k;<$~D-o_xO zJgsQXH%AdA#a|YEh*F88ku79)Y`yI9bQReRCLvph$iv|@`W%b0G|@ztwS zZot1X;0ws}j+_eTzXR{5#z6J6O|^It>DhvkXQ>*f6Su5ay-zj7N@R(@&*=iHPpCn* zAT(egzlgMu0Q2uVV@zxtw6Hh;$Kol<|7f6_g!E9eFrGIVV^vO$z2Me*nzq;4%D&|} z)!(iLkE@ZEHr=S%*ch6NfX@=JjapbYJmwlwT}7V(&4h!S71OOoBWIBoY&8Eynw_Wz z|Iy%0*cL5p2?~#X3GNr3Cyi!5r&0Sb-POEO7u_o+I=~yDc-E_``SFH{H=7+#t57|0 z60!wrrW*odW@9ozc3HVm+iujZLOL2{4jR?2F~qh8X(66tUsLFwFwjjxx;x-m0mVC; zL3|a`#eh%RriBf`^RasNZov+Thg|!uaT!%e2##H?XVJE(LSY9C&Tgeu`_ZLp1B@E- zZ==DHgY==ut}(Eg*_Vv$vvd{NNx#&~E;kvolQIuj$=wnqVJoz^5DK%=`}w;VY-Tz?M;k zY|#yF4EK5?Z5IE+7?~HS;d|DEge}MxM$G)H7C-DY)bBm(2CWx;^Pcs$E{`8lC>Iz{ z4DVI3gvBy6tOtsVSv8D4uyD7t$fbSrzGZj7_JOqlw|h(7{(>Tx!{zmQy!{ihOA8A9 z1#ZV=ud_Jc;dME4@}1d*uB>LBVy9z@&*OF!x~27aW?kh?BeOl8LYLD$+(;I<{jSL_?@i4`Zgb1c^4T;(ku@Ss)}gA8tugq1 zsZ4}RoTFN)oSKiV4J@NmWFiLSIb4OVBA44gCInPi;CFeQg}1bprOFh+X2Dx)`!mfe z3OeLtY)r5}xur$Lx5@6Q-iY+==$_M2yKfW#m zv`uIr%5+n;?uZbHJg>_&F-rWx=m}(ZN^;Br&%sE$1Ts2#*xIBFIPR8Mi(EyXf_q)E zWk(%Iaj7qVk|NG1Qw;2K9BV82TwZ^6okD0LCt8qERGcfZ<~9_+a$R}O(n7x@$3NW_ zRa6E@k#o98$vkUyb zdrXi>D2#=?#95Z@D~c#_=H@y)ZkHQ?++0sYiLW$Uq>Fh;QE6cbpzd5RO1Ys69>0&3 zc)WgIGTq}XJJ-W-hj*-)Eq!V))e?(jp~7cbY}gc3Sn83dPPg!1)dw-VjSFT;H?;fBs@EUbJJD&hvPSoPN>WB^|WJp*aQaT-S6@-qN~IudBEaq*u@f zpIZ~!Q~k`Tf@#Y%DN<$=zG9f0C&_li+HBBM28*SM^*JD^s?>5#>W5hc*+nbr78M{E z^*nzi6#`T9m0Ch5wKBWd>&h#bj{9_uz|>dO?Wimu*}ht9P@7m&m)1hhG?#a6U07*x zvCI3k>ZyN5u?xcCGj#ZwdQe6GI>Grerz2R{F!0Z+(F=l|Uaxb;de!qF5K^_F9#*9? z&nc>c1>L2Eh0iNVR4_4RzMz-}sifDAT4QmYa{`?_YHe%<#J`D79JMCnI%5SJS?bU0 zwMDTJ**TtEOs{VF5IHXfQAtpt(+BYf{3Rtxd8KYHtes@vtBO!H zT?let3kBpmefa@=yUE(PY^#DBRs!Z=o1sonKBTrQ7MhmTJ3~nB4yB(8ISTF6q-j;@ z98VGKmdhBhUH_D&Ms2s?1P%Pc+O%xXKb2A>RV%L4#f>q0U6ZqiL73>!8=*ZE)99OO zq*YZIipl=`w^S{oxA(Ya_})gXGU%r&B__jnR2c=3mTs?-1fv#I9h28S4K`aSgvmLt zXWD+nIz##nD0&PEp(*dy0f3M18D>m}dblUQPj?=NOZfraf85$4?Zet(z$9}}i&d=& zn>ZOw9n#|I?&ENjKcYj&t;zW8#Kq&*l+K4$m!+UPoX{h8?kAd@=7scrT8C#oOy8f8 z{Yz_7+2=|D3{r(NA%^)#s6lXmp+yC5r{9%(G}O>Jq|YG0$CP!IZGWM}M$sXxpd-gw zd|Y)`G3-k%T$&S?|Af{^*&=k!ELub7LD#h>6~`1BAEwb$3a=U)F%65U@HAa_!kW5vi5qJRBqBjNpuT@vp(I)zBiMJ>h+a{$!hk)H7*IZj!d_Wemqnq26~R*pcrtlzFf ziYU^qUB=H^bLk*>;Sm+M(|vvn0&-QUj9+NWDQj}me=813ESPt4@_*G%h0}tctvA!C zi`HDqa_ZrMtTWd3Eb2EcTsnK{8NUY#&sv9RcuoD+NV2umCx(@|?B$ zB4!pl3%uzWmjhdVur|`zpW3Zd@{P46ow{J{!LMkmbaMC569TKgwf=0OYksyiz2z@8 z3|_e0(}iKjSQg}7Rg$TIlFo8V6JbdT3SH?~YIsvH_4@qnGGC@+y z!Cl&AJP?@ovvsh}AGCZNtEgRd#hOF?f3ZdcR$Q{q)3H0-PSe|Ha{{9@{aar4kR`mp z=PWGFcRnnBNqhN-_$BS?T#L2PGr4o-qn1dYyBLb;&vQIxY2cV@rXIIMIXtC)M|Npm zp36JW5-C$czn-u}JKXRm9BPuEZ?QV^kX(SL3(tj?2vM{Qj`q|=mhe1A>T zc22&xQvE0@Ua5W*&sb#%FLIR?34gIl!eCBT%X%=6Ye2nlr?Z_tEaP(5T4FAic zJU3Pv9`{sFVX5Esk|j>0ouz(H5caZ#=Xm{HU$I1pEcmirUbTe#{J9P!U$aDsUj^=| zA{78OPpKDaDotD^X z9D{!y4~(ON2;cIWp}HOPtJt>4mXE zOSv=LZz{5k$^!D1g2=|DXYG4iwN~U4uy+(M3w#CcJYj2Hdlg*;KDX1o&&K6D)kbS@sBM1=c=gKK?7QSm{-s|#u-cyA73MRYXSLI5fkq(jd0fZU#a$SWV zs=k(ZU1ClT=VaffDlO>A@woExK2zxuDOE^+uK4EnItvOV)xINTs)*o~j5{2K zaZwBa;F#hSh&Qs8a&p&LQ}H#>|&G=l$A9ifk1BMvZkz zj;DCWSz{=aa-Op!$gW`)E}Y@;dP=b*c1_Q5`Fv&H8WM;l+jp>wf&ut>-qOVU>F~k5 zb~tmVfadRwK8Xgs?%W?j8}VK+Fk!uY(Zu8|{!vK>*4{rEmBnHmhUjNmxy(@_Np$~W z)W-7s-zK7L->*srOW^U8`hGL|S6o=?``s*)oBN0H6F4s!KVhUUV?Gd_DZ%eQEphU< zu&55VvS3#fH4|k4ShT;)^00qb4Zh>2qYKtvUCNn4b`K!4Sd>HtQ-ah3Oc`I1Bdp7V zOer*yv#CXlo>(@=fM8kv5L0$kgeSm%XO912CY8QK4GEMUgZv1SRQHdY% zz??^oL>_!6?_*3_o{}6_LE+;}no=Q-1#Z#Xd8|wsHVhtS3rA_^9#1d@DJU`_^I7eL zgRx)0VjZ$OFevv3R0{>b>2>EYZ<}C(dveAqhUHbAiYTJ0R)Hgqoas(SaVeHu zQwzNQQfJ|6_PR#SIDH)LYN$`)YuIih`G8HIj04h+IDG;QjMJOrL)G~0XV0)1qWs;o zrI8**{Tt~v9Bv)mNH1j1vOI+;zOmkruV=FjOy!OB>(~ZWZ9GLZ(Nl5r9BSMle#coe zp2t`1RK*{g=r`gkOud`xIod|5Y^u*>o7hx3G+u8{!{YUu@FLODczq~tQ~5GpzZM?} z))Vv&xVWrqf?k9h%2p@n^R-u~b2Ggaj`EIarnkVoS?*@2JQ^U{+Dy;Dg+iyA>DS5W^?0_ErBh0(-i%zSx~=6dmM9&@fy$+X4?jXU!|4$F416^(5m@%nwnRO> zRW)l~ujByNo z;WC0T?et~ZY5KOE-nYqD3~oCd_CZ7JSmWT|^o%pqr#*Q2H9gi|&*a~*-!!o-$)pSI z^}YNoJ8TiYe1DNlqQ=+hm+{(q!;X3vti}d()UU^PoMv{^$Kyq%&pLv{?*-o5-x~ey1b0K}wTYnU?VU(Xivbh6*BC(x^X^f-7q6T3)t^XuipNXQ5q;{@ zXP9r4+cnzh^QC7jrv?4=)v&HzZqS#qO5T}L2Iz@&`UbrjTgiv2R6>8fQ}b0kL21a0 zoI=+SPhsvYrEZth{VHvA|15Ce z89tEgS+Fh#++c$zOSjVY;d(=u&DXQ^9_(2@TGeT6*L$+{yuVPYCgilEtqr`jOf`b- zxdF;B&RK|gD?Q^mo-fAa8mhTj=XAmj{d=C02k8A_J?|c%w__W57lE`nts9^xXq#yF z06l|k1~vhV9D*s%m!F=2s|yF}-C;)u4b(?Oyo47N#7sdYgFxq&fw~(uwbdYfDQqep zZRp4#y@ZknL1sOJ^%B_DlY{lXEWmH1F@>n{*FAa!^&bM$vz79P=*6(LM}~l;?UZ<< zeg|x=|3-Z<+sQquyPw>s->&VV4ny@p+HNWtsykp-4-M7(;%cbe98~S4S;O_Ac$;Q2CHpWRNZwr- zuH(yl{9i(#(gi4!p14VG%MS3FA)wfsVM*T&Db<^n-K@7@@9_tPNy`uwC&DQA9ijJt zQTC0{`w62wLhnr85g7SPBQWP2qSTR4&5vl{NKo@J6^_(f4mix?LL^qmEi5r_6at&6 zfS>ZFA;3|^*l>m0lb-Pzy*pCx$36!OI*-!hW1Zf6gtT@@5!2NX>NHAEW=DCd$ZXp1 z7FUj^(BqZje2iyGAk7`6H}CNUS5ukN95MA_Q9EFU-!;NnoSt!Kva2qZc z`kF6bw5qAzG7xpU?$fky0%ha$k(zck@WFU}oW{;^JFUC}1;541Rit&-d(hfD^bxS+ zKi;7aVCQk~+rty}mh5}(Rk@E(#LV%7VZ9LQfwJIV#LNO;Ow@ITx4ln6Q6}o?fmny$ z#{vsK%!!HZC#rVp)A0ee-r2fK`-N6z!}a(#u^gz&ulz1roejHwM-HU%H=37&dGU98 zHAnB4^oL<+ZV{2GV$qoVk)CmhqI2~p*=4*V^JA{wl>N!aDFT1Z)l-wM7=vh58|2DC ztT#R5FHW6Z`lFVsoOZdOu;uiL3)AB)`opDwr0zDIkdwc=x%Y5rZ zS~*ycrxgYI;H(AKR&taTb?^%OrM^CW#)@!dhKi{zv?hg=z$8*yG|*Y>hd&7$yNFV! z=(DxOv~G(2I9pYW`PQok|v{(Mif^FeOX4M3ZZ3B zTCY{+WPTxB%crQSP)}vct!)+XSfL&ZEB$5c415>u17I;u(18w)9$aD0&2kd^Ht}jN(7wD;CX#Yk!R1CwmiPB4;<(p|t z2^^O#v<4~mqV-;3w%gOBQaz%=tKZ08`WrRuW$Tm>0#}Gqub^*cH(J28ZQ;}JgZVw= z*Nbou2Raq7J}K-_M}at!iu@4bts;9l+h$!^H>VrjAj%|;fY;&h>#>n)WUtBCP7SB( fU7PQ)HV`pXEIcsp?X)%!55%+p+C_Iw)#?8LpXhYd diff --git a/vendor/github.com/open-policy-agent/opa/internal/compiler/wasm/optimizations.go b/vendor/github.com/open-policy-agent/opa/internal/compiler/wasm/optimizations.go index fed367ca..ecdd8d82 100644 --- a/vendor/github.com/open-policy-agent/opa/internal/compiler/wasm/optimizations.go +++ b/vendor/github.com/open-policy-agent/opa/internal/compiler/wasm/optimizations.go @@ -113,10 +113,7 @@ func unquote(s string) (string, error) { } func (c *Compiler) removeUnusedCode() error { - cgCSV, err := opa.CallGraphCSV() - if err != nil { - return fmt.Errorf("csv unpack: %w", err) - } + cgCSV := opa.CallGraphCSV() r := csv.NewReader(bytes.NewReader(cgCSV)) r.LazyQuotes = true cg, err := r.ReadAll() diff --git a/vendor/github.com/open-policy-agent/opa/internal/compiler/wasm/wasm.go b/vendor/github.com/open-policy-agent/opa/internal/compiler/wasm/wasm.go index 77b0150a..5a7b3813 100644 --- a/vendor/github.com/open-policy-agent/opa/internal/compiler/wasm/wasm.go +++ b/vendor/github.com/open-policy-agent/opa/internal/compiler/wasm/wasm.go @@ -8,20 +8,19 @@ package wasm import ( "bytes" "encoding/binary" + "errors" "fmt" "io" - "github.com/pkg/errors" - "github.com/open-policy-agent/opa/ast" "github.com/open-policy-agent/opa/internal/compiler/wasm/opa" "github.com/open-policy-agent/opa/internal/debug" - "github.com/open-policy-agent/opa/internal/ir" "github.com/open-policy-agent/opa/internal/wasm/encoding" "github.com/open-policy-agent/opa/internal/wasm/instruction" "github.com/open-policy-agent/opa/internal/wasm/module" "github.com/open-policy-agent/opa/internal/wasm/types" "github.com/open-policy-agent/opa/internal/wasm/util" + "github.com/open-policy-agent/opa/ir" opatypes "github.com/open-policy-agent/opa/types" ) @@ -137,6 +136,7 @@ var builtinsFunctions = map[string]string{ ast.GlobMatch.Name: "opa_glob_match", ast.JSONMarshal.Name: "opa_json_marshal", ast.JSONUnmarshal.Name: "opa_json_unmarshal", + ast.JSONIsValid.Name: "opa_json_is_valid", ast.ObjectFilter.Name: "builtin_object_filter", ast.ObjectGet.Name: "builtin_object_get", ast.ObjectRemove.Name: "builtin_object_remove", @@ -184,6 +184,11 @@ var builtinsUsingRE2 = [...]string{ builtinsFunctions[ast.GlobMatch.Name], } +func IsWasmEnabled(bi string) bool { + _, ok := builtinsFunctions[bi] + return ok +} + type externalFunc struct { ID int32 Decl *opatypes.Function @@ -317,11 +322,8 @@ func (c *Compiler) Compile() (*module.Module, error) { // are about to be compiled. func (c *Compiler) initModule() error { - bs, err := opa.Bytes() - if err != nil { - return err - } - + bs := opa.Bytes() + var err error c.module, err = encoding.ReadModule(bytes.NewReader(bs)) if err != nil { return err @@ -851,7 +853,7 @@ func (c *Compiler) compileFunc(fn *ir.Func) error { for i := range fn.Blocks { instrs, err := c.compileBlock(fn.Blocks[i]) if err != nil { - return errors.Wrapf(err, "block %d", i) + return fmt.Errorf("block %d: %w", i, err) } if i < len(fn.Blocks)-1 { // not the last block: wrap in `block` instr if withControlInstr(instrs) { // unless we don't need to @@ -895,7 +897,7 @@ func mapFunc(mapping ast.Object, fn *ir.Func, index int) (ast.Object, bool) { } func (c *Compiler) emitMappingAndStartFunc() error { - var indices []uint32 + indices := make([]uint32, 0, len(c.policy.Funcs.Funcs)) var ok bool mapping := ast.NewObject() @@ -994,7 +996,7 @@ func (c *Compiler) compileBlock(block *ir.Block) ([]instruction.Instruction, err for _, stmt := range block.Stmts { switch stmt := stmt.(type) { - case *ir.ResultSetAdd: + case *ir.ResultSetAddStmt: instrs = append(instrs, instruction.GetLocal{Index: c.lrs}) instrs = append(instrs, instruction.GetLocal{Index: c.local(stmt.Value)}) instrs = append(instrs, instruction.Call{Index: c.function(opaSetAdd)}) @@ -1065,7 +1067,7 @@ func (c *Compiler) compileBlock(block *ir.Block) ([]instruction.Instruction, err return nil, err } case *ir.DotStmt: - if loc, ok := stmt.Source.(ir.Local); ok { + if loc, ok := stmt.Source.Value.(ir.Local); ok { instrs = append(instrs, instruction.GetLocal{Index: c.local(loc)}) instrs = append(instrs, c.instrRead(stmt.Key)) instrs = append(instrs, instruction.Call{Index: c.function(opaValueGet)}) @@ -1073,8 +1075,8 @@ func (c *Compiler) compileBlock(block *ir.Block) ([]instruction.Instruction, err instrs = append(instrs, instruction.I32Eqz{}) instrs = append(instrs, instruction.BrIf{Index: 0}) } else { - // Booleans and strings would lead to the BrIf (since opa_value_get - // on them returns 0), so let's skip that. + // Booleans and string sources would lead to the BrIf (since opa_value_get + // on them returns 0), so let's skip trying that. instrs = append(instrs, instruction.Br{Index: 0}) break } @@ -1095,13 +1097,13 @@ func (c *Compiler) compileBlock(block *ir.Block) ([]instruction.Instruction, err instrs = append(instrs, instruction.Br{Index: 0}) continue } - _, okA := stmt.A.(ir.Bool) - if _, okB := stmt.B.(ir.Bool); okA && okB { + _, okA := stmt.A.Value.(ir.Bool) + if _, okB := stmt.B.Value.(ir.Bool); okA && okB { // not equal (checked above), but both booleans => not equal continue } - _, okA = stmt.A.(ir.StringIndex) - if _, okB := stmt.B.(ir.StringIndex); okA && okB { + _, okA = stmt.A.Value.(ir.StringIndex) + if _, okB := stmt.B.Value.(ir.StringIndex); okA && okB { // not equal (checked above), but both strings => not equal continue } @@ -1133,7 +1135,7 @@ func (c *Compiler) compileBlock(block *ir.Block) ([]instruction.Instruction, err instrs = append(instrs, instruction.Call{Index: c.function(opaSet)}) instrs = append(instrs, instruction.SetLocal{Index: c.local(stmt.Target)}) case *ir.IsArrayStmt: - if loc, ok := stmt.Source.(ir.Local); ok { + if loc, ok := stmt.Source.Value.(ir.Local); ok { instrs = append(instrs, instruction.GetLocal{Index: c.local(loc)}) instrs = append(instrs, instruction.Call{Index: c.function(opaValueType)}) instrs = append(instrs, instruction.I32Const{Value: opaTypeArray}) @@ -1144,7 +1146,7 @@ func (c *Compiler) compileBlock(block *ir.Block) ([]instruction.Instruction, err break } case *ir.IsObjectStmt: - if loc, ok := stmt.Source.(ir.Local); ok { + if loc, ok := stmt.Source.Value.(ir.Local); ok { instrs = append(instrs, instruction.GetLocal{Index: c.local(loc)}) instrs = append(instrs, instruction.Call{Index: c.function(opaValueType)}) instrs = append(instrs, instruction.I32Const{Value: opaTypeObject}) @@ -1336,7 +1338,7 @@ func (c *Compiler) compileWithStmt(with *ir.WithStmt, result *[]instruction.Inst return nil } -func (c *Compiler) compileUpsert(local ir.Local, path []int, value ir.LocalOrConst, loc ir.Location, instrs []instruction.Instruction) []instruction.Instruction { +func (c *Compiler) compileUpsert(local ir.Local, path []int, value ir.Operand, loc ir.Location, instrs []instruction.Instruction) []instruction.Instruction { lcopy := c.genLocal() // holds copy of local instrs = append(instrs, instruction.GetLocal{Index: c.local(local)}) @@ -1483,7 +1485,7 @@ func (c *Compiler) compileCallDynamicStmt(stmt *ir.CallDynamicStmt, result *[]in instruction.CallIndirect{Index: typeIndex}, // [arg0 arg1 tbl_idx] -> [res] instruction.TeeLocal{Index: c.local(stmt.Result)}, instruction.I32Eqz{}, - instruction.BrIf{Index: 2}, // mapping found, "undefined" result counts + instruction.BrIf{Index: 3}, // mapping found, "undefined" result counts ) *result = append(*result, instrs...) @@ -1749,7 +1751,7 @@ func getLowestFreeElementSegmentOffset(m *module.Module) (int32, error) { // It returns the instructions that make up the function call with // arguments, followed by Unreachable. func (c *Compiler) runtimeErrorAbort(loc ir.Location, errType int) []instruction.Instruction { - index, row, col := loc.Index, loc.Row, loc.Col + index, row, col := loc.File, loc.Row, loc.Col return []instruction.Instruction{ instruction.I32Const{Value: c.fileAddr(index)}, instruction.I32Const{Value: int32(row)}, @@ -1770,8 +1772,8 @@ func (c *Compiler) storeFunc(name string, code *module.CodeEntry) error { return nil } -func (c *Compiler) instrRead(lv ir.LocalOrConst) instruction.Instruction { - switch x := lv.(type) { +func (c *Compiler) instrRead(lv ir.Operand) instruction.Instruction { + switch x := lv.Value.(type) { case ir.Bool: return instruction.I32Const{Value: c.opaBoolAddr(x)} case ir.StringIndex: diff --git a/vendor/github.com/open-policy-agent/opa/internal/deepcopy/deepcopy.go b/vendor/github.com/open-policy-agent/opa/internal/deepcopy/deepcopy.go index cf8cc1cf..00e8df6f 100644 --- a/vendor/github.com/open-policy-agent/opa/internal/deepcopy/deepcopy.go +++ b/vendor/github.com/open-policy-agent/opa/internal/deepcopy/deepcopy.go @@ -16,12 +16,16 @@ func DeepCopy(val interface{}) interface{} { } return cpy case map[string]interface{}: - cpy := make(map[string]interface{}, len(val)) - for k := range val { - cpy[k] = DeepCopy(val[k]) - } - return cpy + return Map(val) default: return val } } + +func Map(val map[string]interface{}) map[string]interface{} { + cpy := make(map[string]interface{}, len(val)) + for k := range val { + cpy[k] = DeepCopy(val[k]) + } + return cpy +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/future/filter_imports.go b/vendor/github.com/open-policy-agent/opa/internal/future/filter_imports.go index b0fdc8c6..2863aad4 100644 --- a/vendor/github.com/open-policy-agent/opa/internal/future/filter_imports.go +++ b/vendor/github.com/open-policy-agent/opa/internal/future/filter_imports.go @@ -18,3 +18,20 @@ func FilterFutureImports(imps []*ast.Import) []*ast.Import { } return ret } + +// IsAllFutureKeywords returns true if the passed *ast.Import is `future.keywords` +func IsAllFutureKeywords(imp *ast.Import) bool { + path := imp.Path.Value.(ast.Ref) + return len(path) == 2 && + ast.FutureRootDocument.Equal(path[0]) && + path[1].Equal(ast.StringTerm("keywords")) +} + +// IsFutureKeyword returns true if the passed *ast.Import is `future.keywords.{kw}` +func IsFutureKeyword(imp *ast.Import, kw string) bool { + path := imp.Path.Value.(ast.Ref) + return len(path) == 3 && + ast.FutureRootDocument.Equal(path[0]) && + path[1].Equal(ast.StringTerm("keywords")) && + path[2].Equal(ast.StringTerm(kw)) +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/gojsonschema/jsonLoader.go b/vendor/github.com/open-policy-agent/opa/internal/gojsonschema/jsonLoader.go index 65ada35c..0475dbc5 100644 --- a/vendor/github.com/open-policy-agent/opa/internal/gojsonschema/jsonLoader.go +++ b/vendor/github.com/open-policy-agent/opa/internal/gojsonschema/jsonLoader.go @@ -39,6 +39,7 @@ import ( "path/filepath" "runtime" "strings" + "sync" "github.com/xeipuuv/gojsonreference" ) @@ -48,8 +49,11 @@ import ( // add extra parameters to all calls and interfaces involved, so we're // using a global variable instead: var allowNet map[string]struct{} +var netMut sync.RWMutex func SetAllowNet(hosts []string) { + netMut.Lock() + defer netMut.Unlock() if hosts == nil { allowNet = nil // resetting the global return @@ -61,6 +65,8 @@ func SetAllowNet(hosts []string) { } func isAllowed(ref *url.URL) bool { + netMut.RLock() + defer netMut.RUnlock() if allowNet == nil { return true } diff --git a/vendor/github.com/open-policy-agent/opa/internal/gojsonschema/schema.go b/vendor/github.com/open-policy-agent/opa/internal/gojsonschema/schema.go index 3e6f8c08..6bdf5ebc 100644 --- a/vendor/github.com/open-policy-agent/opa/internal/gojsonschema/schema.go +++ b/vendor/github.com/open-policy-agent/opa/internal/gojsonschema/schema.go @@ -517,17 +517,12 @@ func (d *Schema) parseSchema(documentNode interface{}, currentSchema *SubSchema) } } - if pattern, err := getString(m, KeyPattern); err != nil { + // NOTE: Regex compilation step removed as we don't use "pattern" attribute for + // type checking, and this would cause schemas to fail if they included patterns + // that were valid ECMA regex dialect but not known to Go (i.e. the regexp.Compile + // function), such as patterns with negative lookahead + if _, err := getString(m, KeyPattern); err != nil { return err - } else if pattern != nil { - regexpObject, err := regexp.Compile(*pattern) - if err != nil { - return errors.New(formatErrorDescription( - Locale.MustBeValidRegex(), - ErrorDetails{"key": KeyPattern}, - )) - } - currentSchema.pattern = regexpObject } if format, err := getString(m, KeyFormat); err != nil { diff --git a/vendor/github.com/open-policy-agent/opa/internal/gqlparser/LICENSE b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/LICENSE new file mode 100644 index 00000000..1221b9d3 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/LICENSE @@ -0,0 +1,19 @@ +Copyright (c) 2018 Adam Scarr + +Permission is hereby granted, free of charge, to any person obtaining a copy +of this software and associated documentation files (the "Software"), to deal +in the Software without restriction, including without limitation the rights +to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the Software is +furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in all +copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE +SOFTWARE. \ No newline at end of file diff --git a/vendor/github.com/open-policy-agent/opa/internal/gqlparser/ast/argmap.go b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/ast/argmap.go new file mode 100644 index 00000000..43f6a3d6 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/ast/argmap.go @@ -0,0 +1,37 @@ +package ast + +func arg2map(defs ArgumentDefinitionList, args ArgumentList, vars map[string]interface{}) map[string]interface{} { + result := map[string]interface{}{} + var err error + + for _, argDef := range defs { + var val interface{} + var hasValue bool + + if argValue := args.ForName(argDef.Name); argValue != nil { + if argValue.Value.Kind == Variable { + val, hasValue = vars[argValue.Value.Raw] + } else { + val, err = argValue.Value.Value(vars) + if err != nil { + panic(err) + } + hasValue = true + } + } + + if !hasValue && argDef.DefaultValue != nil { + val, err = argDef.DefaultValue.Value(vars) + if err != nil { + panic(err) + } + hasValue = true + } + + if hasValue { + result[argDef.Name] = val + } + } + + return result +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/gqlparser/ast/collections.go b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/ast/collections.go new file mode 100644 index 00000000..94b800ee --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/ast/collections.go @@ -0,0 +1,148 @@ +package ast + +type FieldList []*FieldDefinition + +func (l FieldList) ForName(name string) *FieldDefinition { + for _, it := range l { + if it.Name == name { + return it + } + } + return nil +} + +type EnumValueList []*EnumValueDefinition + +func (l EnumValueList) ForName(name string) *EnumValueDefinition { + for _, it := range l { + if it.Name == name { + return it + } + } + return nil +} + +type DirectiveList []*Directive + +func (l DirectiveList) ForName(name string) *Directive { + for _, it := range l { + if it.Name == name { + return it + } + } + return nil +} + +func (l DirectiveList) ForNames(name string) []*Directive { + resp := []*Directive{} + for _, it := range l { + if it.Name == name { + resp = append(resp, it) + } + } + return resp +} + +type OperationList []*OperationDefinition + +func (l OperationList) ForName(name string) *OperationDefinition { + if name == "" && len(l) == 1 { + return l[0] + } + for _, it := range l { + if it.Name == name { + return it + } + } + return nil +} + +type FragmentDefinitionList []*FragmentDefinition + +func (l FragmentDefinitionList) ForName(name string) *FragmentDefinition { + for _, it := range l { + if it.Name == name { + return it + } + } + return nil +} + +type VariableDefinitionList []*VariableDefinition + +func (l VariableDefinitionList) ForName(name string) *VariableDefinition { + for _, it := range l { + if it.Variable == name { + return it + } + } + return nil +} + +type ArgumentList []*Argument + +func (l ArgumentList) ForName(name string) *Argument { + for _, it := range l { + if it.Name == name { + return it + } + } + return nil +} + +type ArgumentDefinitionList []*ArgumentDefinition + +func (l ArgumentDefinitionList) ForName(name string) *ArgumentDefinition { + for _, it := range l { + if it.Name == name { + return it + } + } + return nil +} + +type SchemaDefinitionList []*SchemaDefinition + +type DirectiveDefinitionList []*DirectiveDefinition + +func (l DirectiveDefinitionList) ForName(name string) *DirectiveDefinition { + for _, it := range l { + if it.Name == name { + return it + } + } + return nil +} + +type DefinitionList []*Definition + +func (l DefinitionList) ForName(name string) *Definition { + for _, it := range l { + if it.Name == name { + return it + } + } + return nil +} + +type OperationTypeDefinitionList []*OperationTypeDefinition + +func (l OperationTypeDefinitionList) ForType(name string) *OperationTypeDefinition { + for _, it := range l { + if it.Type == name { + return it + } + } + return nil +} + +type ChildValueList []*ChildValue + +func (v ChildValueList) ForName(name string) *Value { + for _, f := range v { + if f.Name == name { + return f.Value + } + } + return nil +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/gqlparser/ast/decode.go b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/ast/decode.go new file mode 100644 index 00000000..d0092055 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/ast/decode.go @@ -0,0 +1,216 @@ +package ast + +import ( + "encoding/json" +) + +func UnmarshalSelectionSet(b []byte) (SelectionSet, error) { + var tmp []json.RawMessage + + if err := json.Unmarshal(b, &tmp); err != nil { + return nil, err + } + + var result = make([]Selection, 0) + for _, item := range tmp { + var field Field + if err := json.Unmarshal(item, &field); err == nil { + result = append(result, &field) + continue + } + var fragmentSpread FragmentSpread + if err := json.Unmarshal(item, &fragmentSpread); err == nil { + result = append(result, &fragmentSpread) + continue + } + var inlineFragment InlineFragment + if err := json.Unmarshal(item, &inlineFragment); err == nil { + result = append(result, &inlineFragment) + continue + } + } + + return result, nil +} + +func (f *FragmentDefinition) UnmarshalJSON(b []byte) error { + var tmp map[string]json.RawMessage + if err := json.Unmarshal(b, &tmp); err != nil { + return err + } + for k := range tmp { + switch k { + case "Name": + err := json.Unmarshal(tmp[k], &f.Name) + if err != nil { + return err + } + case "VariableDefinition": + err := json.Unmarshal(tmp[k], &f.VariableDefinition) + if err != nil { + return err + } + case "TypeCondition": + err := json.Unmarshal(tmp[k], &f.TypeCondition) + if err != nil { + return err + } + case "Directives": + err := json.Unmarshal(tmp[k], &f.Directives) + if err != nil { + return err + } + case "SelectionSet": + ss, err := UnmarshalSelectionSet(tmp[k]) + if err != nil { + return err + } + f.SelectionSet = ss + case "Definition": + err := json.Unmarshal(tmp[k], &f.Definition) + if err != nil { + return err + } + case "Position": + err := json.Unmarshal(tmp[k], &f.Position) + if err != nil { + return err + } + } + } + return nil +} + +func (f *InlineFragment) UnmarshalJSON(b []byte) error { + var tmp map[string]json.RawMessage + if err := json.Unmarshal(b, &tmp); err != nil { + return err + } + for k := range tmp { + switch k { + case "TypeCondition": + err := json.Unmarshal(tmp[k], &f.TypeCondition) + if err != nil { + return err + } + case "Directives": + err := json.Unmarshal(tmp[k], &f.Directives) + if err != nil { + return err + } + case "SelectionSet": + ss, err := UnmarshalSelectionSet(tmp[k]) + if err != nil { + return err + } + f.SelectionSet = ss + case "ObjectDefinition": + err := json.Unmarshal(tmp[k], &f.ObjectDefinition) + if err != nil { + return err + } + case "Position": + err := json.Unmarshal(tmp[k], &f.Position) + if err != nil { + return err + } + } + } + return nil +} + +func (f *OperationDefinition) UnmarshalJSON(b []byte) error { + var tmp map[string]json.RawMessage + if err := json.Unmarshal(b, &tmp); err != nil { + return err + } + for k := range tmp { + switch k { + case "Operation": + err := json.Unmarshal(tmp[k], &f.Operation) + if err != nil { + return err + } + case "Name": + err := json.Unmarshal(tmp[k], &f.Name) + if err != nil { + return err + } + case "VariableDefinitions": + err := json.Unmarshal(tmp[k], &f.VariableDefinitions) + if err != nil { + return err + } + case "Directives": + err := json.Unmarshal(tmp[k], &f.Directives) + if err != nil { + return err + } + case "SelectionSet": + ss, err := UnmarshalSelectionSet(tmp[k]) + if err != nil { + return err + } + f.SelectionSet = ss + case "Position": + err := json.Unmarshal(tmp[k], &f.Position) + if err != nil { + return err + } + } + } + return nil +} + +func (f *Field) UnmarshalJSON(b []byte) error { + var tmp map[string]json.RawMessage + if err := json.Unmarshal(b, &tmp); err != nil { + return err + } + for k := range tmp { + switch k { + case "Alias": + err := json.Unmarshal(tmp[k], &f.Alias) + if err != nil { + return err + } + case "Name": + err := json.Unmarshal(tmp[k], &f.Name) + if err != nil { + return err + } + case "Arguments": + err := json.Unmarshal(tmp[k], &f.Arguments) + if err != nil { + return err + } + case "Directives": + err := json.Unmarshal(tmp[k], &f.Directives) + if err != nil { + return err + } + case "SelectionSet": + ss, err := UnmarshalSelectionSet(tmp[k]) + if err != nil { + return err + } + f.SelectionSet = ss + case "Position": + err := json.Unmarshal(tmp[k], &f.Position) + if err != nil { + return err + } + case "Definition": + err := json.Unmarshal(tmp[k], &f.Definition) + if err != nil { + return err + } + case "ObjectDefinition": + err := json.Unmarshal(tmp[k], &f.ObjectDefinition) + if err != nil { + return err + } + } + } + return nil +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/gqlparser/ast/definition.go b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/ast/definition.go new file mode 100644 index 00000000..d2039081 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/ast/definition.go @@ -0,0 +1,94 @@ +package ast + +type DefinitionKind string + +const ( + Scalar DefinitionKind = "SCALAR" + Object DefinitionKind = "OBJECT" + Interface DefinitionKind = "INTERFACE" + Union DefinitionKind = "UNION" + Enum DefinitionKind = "ENUM" + InputObject DefinitionKind = "INPUT_OBJECT" +) + +// Definition is the core type definition object, it includes all of the definable types +// but does *not* cover schema or directives. +// +// @vektah: Javascript implementation has different types for all of these, but they are +// more similar than different and don't define any behaviour. I think this style of +// "some hot" struct works better, at least for go. +// +// Type extensions are also represented by this same struct. +type Definition struct { + Kind DefinitionKind + Description string + Name string + Directives DirectiveList + Interfaces []string // object and input object + Fields FieldList // object and input object + Types []string // union + EnumValues EnumValueList // enum + + Position *Position `dump:"-"` + BuiltIn bool `dump:"-"` +} + +func (d *Definition) IsLeafType() bool { + return d.Kind == Enum || d.Kind == Scalar +} + +func (d *Definition) IsAbstractType() bool { + return d.Kind == Interface || d.Kind == Union +} + +func (d *Definition) IsCompositeType() bool { + return d.Kind == Object || d.Kind == Interface || d.Kind == Union +} + +func (d *Definition) IsInputType() bool { + return d.Kind == Scalar || d.Kind == Enum || d.Kind == InputObject +} + +func (d *Definition) OneOf(types ...string) bool { + for _, t := range types { + if d.Name == t { + return true + } + } + return false +} + +type FieldDefinition struct { + Description string + Name string + Arguments ArgumentDefinitionList // only for objects + DefaultValue *Value // only for input objects + Type *Type + Directives DirectiveList + Position *Position `dump:"-"` +} + +type ArgumentDefinition struct { + Description string + Name string + DefaultValue *Value + Type *Type + Directives DirectiveList + Position *Position `dump:"-"` +} + +type EnumValueDefinition struct { + Description string + Name string + Directives DirectiveList + Position *Position `dump:"-"` +} + +type DirectiveDefinition struct { + Description string + Name string + Arguments ArgumentDefinitionList + Locations []DirectiveLocation + IsRepeatable bool + Position *Position `dump:"-"` +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/gqlparser/ast/directive.go b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/ast/directive.go new file mode 100644 index 00000000..5f6e8531 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/ast/directive.go @@ -0,0 +1,43 @@ +package ast + +type DirectiveLocation string + +const ( + // Executable + LocationQuery DirectiveLocation = `QUERY` + LocationMutation DirectiveLocation = `MUTATION` + LocationSubscription DirectiveLocation = `SUBSCRIPTION` + LocationField DirectiveLocation = `FIELD` + LocationFragmentDefinition DirectiveLocation = `FRAGMENT_DEFINITION` + LocationFragmentSpread DirectiveLocation = `FRAGMENT_SPREAD` + LocationInlineFragment DirectiveLocation = `INLINE_FRAGMENT` + + // Type System + LocationSchema DirectiveLocation = `SCHEMA` + LocationScalar DirectiveLocation = `SCALAR` + LocationObject DirectiveLocation = `OBJECT` + LocationFieldDefinition DirectiveLocation = `FIELD_DEFINITION` + LocationArgumentDefinition DirectiveLocation = `ARGUMENT_DEFINITION` + LocationInterface DirectiveLocation = `INTERFACE` + LocationUnion DirectiveLocation = `UNION` + LocationEnum DirectiveLocation = `ENUM` + LocationEnumValue DirectiveLocation = `ENUM_VALUE` + LocationInputObject DirectiveLocation = `INPUT_OBJECT` + LocationInputFieldDefinition DirectiveLocation = `INPUT_FIELD_DEFINITION` + LocationVariableDefinition DirectiveLocation = `VARIABLE_DEFINITION` +) + +type Directive struct { + Name string + Arguments ArgumentList + Position *Position `dump:"-"` + + // Requires validation + ParentDefinition *Definition + Definition *DirectiveDefinition + Location DirectiveLocation +} + +func (d *Directive) ArgumentMap(vars map[string]interface{}) map[string]interface{} { + return arg2map(d.Definition.Arguments, d.Arguments, vars) +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/gqlparser/ast/document.go b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/ast/document.go new file mode 100644 index 00000000..43bfb54f --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/ast/document.go @@ -0,0 +1,79 @@ +package ast + +type QueryDocument struct { + Operations OperationList + Fragments FragmentDefinitionList + Position *Position `dump:"-"` +} + +type SchemaDocument struct { + Schema SchemaDefinitionList + SchemaExtension SchemaDefinitionList + Directives DirectiveDefinitionList + Definitions DefinitionList + Extensions DefinitionList + Position *Position `dump:"-"` +} + +func (d *SchemaDocument) Merge(other *SchemaDocument) { + d.Schema = append(d.Schema, other.Schema...) + d.SchemaExtension = append(d.SchemaExtension, other.SchemaExtension...) + d.Directives = append(d.Directives, other.Directives...) + d.Definitions = append(d.Definitions, other.Definitions...) + d.Extensions = append(d.Extensions, other.Extensions...) +} + +type Schema struct { + Query *Definition + Mutation *Definition + Subscription *Definition + + Types map[string]*Definition + Directives map[string]*DirectiveDefinition + + PossibleTypes map[string][]*Definition + Implements map[string][]*Definition + + Description string +} + +// AddTypes is the helper to add types definition to the schema +func (s *Schema) AddTypes(defs ...*Definition) { + if s.Types == nil { + s.Types = make(map[string]*Definition) + } + for _, def := range defs { + s.Types[def.Name] = def + } +} + +func (s *Schema) AddPossibleType(name string, def *Definition) { + s.PossibleTypes[name] = append(s.PossibleTypes[name], def) +} + +// GetPossibleTypes will enumerate all the definitions for a given interface or union +func (s *Schema) GetPossibleTypes(def *Definition) []*Definition { + return s.PossibleTypes[def.Name] +} + +func (s *Schema) AddImplements(name string, iface *Definition) { + s.Implements[name] = append(s.Implements[name], iface) +} + +// GetImplements returns all the interface and union definitions that the given definition satisfies +func (s *Schema) GetImplements(def *Definition) []*Definition { + return s.Implements[def.Name] +} + +type SchemaDefinition struct { + Description string + Directives DirectiveList + OperationTypes OperationTypeDefinitionList + Position *Position `dump:"-"` +} + +type OperationTypeDefinition struct { + Operation Operation + Type string + Position *Position `dump:"-"` +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/gqlparser/ast/dumper.go b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/ast/dumper.go new file mode 100644 index 00000000..dbb7a7ef --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/ast/dumper.go @@ -0,0 +1,159 @@ +package ast + +import ( + "bytes" + "fmt" + "reflect" + "strconv" + "strings" +) + +// Dump turns ast into a stable string format for assertions in tests +func Dump(i interface{}) string { + v := reflect.ValueOf(i) + + d := dumper{Buffer: &bytes.Buffer{}} + d.dump(v) + + return d.String() +} + +type dumper struct { + *bytes.Buffer + indent int +} + +type Dumpable interface { + Dump() string +} + +func (d *dumper) dump(v reflect.Value) { + if dumpable, isDumpable := v.Interface().(Dumpable); isDumpable { + d.WriteString(dumpable.Dump()) + return + } + switch v.Kind() { + case reflect.Bool: + if v.Bool() { + d.WriteString("true") + } else { + d.WriteString("false") + } + case reflect.Int, reflect.Int8, reflect.Int16, reflect.Int32, reflect.Int64: + d.WriteString(fmt.Sprintf("%d", v.Int())) + + case reflect.Uint, reflect.Uint8, reflect.Uint16, reflect.Uint32, reflect.Uint64: + d.WriteString(fmt.Sprintf("%d", v.Uint())) + + case reflect.Float32, reflect.Float64: + d.WriteString(fmt.Sprintf("%.2f", v.Float())) + + case reflect.String: + if v.Type().Name() != "string" { + d.WriteString(v.Type().Name() + "(" + strconv.Quote(v.String()) + ")") + } else { + d.WriteString(strconv.Quote(v.String())) + } + + case reflect.Array, reflect.Slice: + d.dumpArray(v) + + case reflect.Interface, reflect.Ptr: + d.dumpPtr(v) + + case reflect.Struct: + d.dumpStruct(v) + + default: + panic(fmt.Errorf("unsupported kind: %s\n buf: %s", v.Kind().String(), d.String())) + } +} + +func (d *dumper) writeIndent() { + d.Buffer.WriteString(strings.Repeat(" ", d.indent)) +} + +func (d *dumper) nl() { + d.Buffer.WriteByte('\n') + d.writeIndent() +} + +func typeName(t reflect.Type) string { + if t.Kind() == reflect.Ptr { + return typeName(t.Elem()) + } + return t.Name() +} + +func (d *dumper) dumpArray(v reflect.Value) { + d.WriteString("[" + typeName(v.Type().Elem()) + "]") + + for i := 0; i < v.Len(); i++ { + d.nl() + d.WriteString("- ") + d.indent++ + d.dump(v.Index(i)) + d.indent-- + } +} + +func (d *dumper) dumpStruct(v reflect.Value) { + d.WriteString("<" + v.Type().Name() + ">") + d.indent++ + + typ := v.Type() + for i := 0; i < v.NumField(); i++ { + f := v.Field(i) + if typ.Field(i).Tag.Get("dump") == "-" { + continue + } + + if isZero(f) { + continue + } + d.nl() + d.WriteString(typ.Field(i).Name) + d.WriteString(": ") + d.dump(v.Field(i)) + } + + d.indent-- +} + +func isZero(v reflect.Value) bool { + switch v.Kind() { + case reflect.Ptr, reflect.Interface: + return v.IsNil() + case reflect.Func, reflect.Map: + return v.IsNil() + + case reflect.Array, reflect.Slice: + if v.IsNil() { + return true + } + z := true + for i := 0; i < v.Len(); i++ { + z = z && isZero(v.Index(i)) + } + return z + case reflect.Struct: + z := true + for i := 0; i < v.NumField(); i++ { + z = z && isZero(v.Field(i)) + } + return z + case reflect.String: + return v.String() == "" + } + + // Compare other types directly: + return reflect.DeepEqual(v.Interface(), reflect.Zero(v.Type())) +} + +func (d *dumper) dumpPtr(v reflect.Value) { + if v.IsNil() { + d.WriteString("nil") + return + } + d.dump(v.Elem()) +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/gqlparser/ast/fragment.go b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/ast/fragment.go new file mode 100644 index 00000000..57ab56c7 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/ast/fragment.go @@ -0,0 +1,38 @@ +package ast + +type FragmentSpread struct { + Name string + Directives DirectiveList + + // Require validation + ObjectDefinition *Definition + Definition *FragmentDefinition + + Position *Position `dump:"-"` +} + +type InlineFragment struct { + TypeCondition string + Directives DirectiveList + SelectionSet SelectionSet + + // Require validation + ObjectDefinition *Definition + + Position *Position `dump:"-"` +} + +type FragmentDefinition struct { + Name string + // Note: fragment variable definitions are experimental and may be changed + // or removed in the future. + VariableDefinition VariableDefinitionList + TypeCondition string + Directives DirectiveList + SelectionSet SelectionSet + + // Require validation + Definition *Definition + + Position *Position `dump:"-"` +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/gqlparser/ast/operation.go b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/ast/operation.go new file mode 100644 index 00000000..3b37f81b --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/ast/operation.go @@ -0,0 +1,30 @@ +package ast + +type Operation string + +const ( + Query Operation = "query" + Mutation Operation = "mutation" + Subscription Operation = "subscription" +) + +type OperationDefinition struct { + Operation Operation + Name string + VariableDefinitions VariableDefinitionList + Directives DirectiveList + SelectionSet SelectionSet + Position *Position `dump:"-"` +} + +type VariableDefinition struct { + Variable string + Type *Type + DefaultValue *Value + Directives DirectiveList + Position *Position `dump:"-"` + + // Requires validation + Definition *Definition + Used bool `dump:"-"` +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/gqlparser/ast/path.go b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/ast/path.go new file mode 100644 index 00000000..be1a9e4e --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/ast/path.go @@ -0,0 +1,67 @@ +package ast + +import ( + "bytes" + "encoding/json" + "fmt" +) + +var _ json.Unmarshaler = (*Path)(nil) + +type Path []PathElement + +type PathElement interface { + isPathElement() +} + +var _ PathElement = PathIndex(0) +var _ PathElement = PathName("") + +func (path Path) String() string { + var str bytes.Buffer + for i, v := range path { + switch v := v.(type) { + case PathIndex: + str.WriteString(fmt.Sprintf("[%d]", v)) + case PathName: + if i != 0 { + str.WriteByte('.') + } + str.WriteString(string(v)) + default: + panic(fmt.Sprintf("unknown type: %T", v)) + } + } + return str.String() +} + +func (path *Path) UnmarshalJSON(b []byte) error { + var vs []interface{} + err := json.Unmarshal(b, &vs) + if err != nil { + return err + } + + *path = make([]PathElement, 0, len(vs)) + for _, v := range vs { + switch v := v.(type) { + case string: + *path = append(*path, PathName(v)) + case int: + *path = append(*path, PathIndex(v)) + case float64: + *path = append(*path, PathIndex(int(v))) + default: + return fmt.Errorf("unknown path element type: %T", v) + } + } + return nil +} + +type PathIndex int + +func (PathIndex) isPathElement() {} + +type PathName string + +func (PathName) isPathElement() {} diff --git a/vendor/github.com/open-policy-agent/opa/internal/gqlparser/ast/selection.go b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/ast/selection.go new file mode 100644 index 00000000..5ef26c6a --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/ast/selection.go @@ -0,0 +1,39 @@ +package ast + +type SelectionSet []Selection + +type Selection interface { + isSelection() + GetPosition() *Position +} + +func (*Field) isSelection() {} +func (*FragmentSpread) isSelection() {} +func (*InlineFragment) isSelection() {} + +func (s *Field) GetPosition() *Position { return s.Position } +func (s *FragmentSpread) GetPosition() *Position { return s.Position } +func (s *InlineFragment) GetPosition() *Position { return s.Position } + +type Field struct { + Alias string + Name string + Arguments ArgumentList + Directives DirectiveList + SelectionSet SelectionSet + Position *Position `dump:"-"` + + // Require validation + Definition *FieldDefinition + ObjectDefinition *Definition +} + +type Argument struct { + Name string + Value *Value + Position *Position `dump:"-"` +} + +func (s *Field) ArgumentMap(vars map[string]interface{}) map[string]interface{} { + return arg2map(s.Definition.Arguments, s.Arguments, vars) +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/gqlparser/ast/source.go b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/ast/source.go new file mode 100644 index 00000000..2949f83f --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/ast/source.go @@ -0,0 +1,19 @@ +package ast + +// Source covers a single *.graphql file +type Source struct { + // Name is the filename of the source + Name string + // Input is the actual contents of the source file + Input string + // BuiltIn indicate whether the source is a part of the specification + BuiltIn bool +} + +type Position struct { + Start int // The starting position, in runes, of this token in the input. + End int // The end position, in runes, of this token in the input. + Line int // The line number at the start of this item. + Column int // The column number at the start of this item. + Src *Source // The source document this token belongs to +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/gqlparser/ast/type.go b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/ast/type.go new file mode 100644 index 00000000..5f77bc7c --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/ast/type.go @@ -0,0 +1,68 @@ +package ast + +func NonNullNamedType(named string, pos *Position) *Type { + return &Type{NamedType: named, NonNull: true, Position: pos} +} + +func NamedType(named string, pos *Position) *Type { + return &Type{NamedType: named, NonNull: false, Position: pos} +} + +func NonNullListType(elem *Type, pos *Position) *Type { + return &Type{Elem: elem, NonNull: true, Position: pos} +} + +func ListType(elem *Type, pos *Position) *Type { + return &Type{Elem: elem, NonNull: false, Position: pos} +} + +type Type struct { + NamedType string + Elem *Type + NonNull bool + Position *Position `dump:"-"` +} + +func (t *Type) Name() string { + if t.NamedType != "" { + return t.NamedType + } + + return t.Elem.Name() +} + +func (t *Type) String() string { + nn := "" + if t.NonNull { + nn = "!" + } + if t.NamedType != "" { + return t.NamedType + nn + } + + return "[" + t.Elem.String() + "]" + nn +} + +func (t *Type) IsCompatible(other *Type) bool { + if t.NamedType != other.NamedType { + return false + } + + if t.Elem != nil && other.Elem == nil { + return false + } + + if t.Elem != nil && !t.Elem.IsCompatible(other.Elem) { + return false + } + + if other.NonNull { + return t.NonNull + } + + return true +} + +func (t *Type) Dump() string { + return t.String() +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/gqlparser/ast/value.go b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/ast/value.go new file mode 100644 index 00000000..c25ef150 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/ast/value.go @@ -0,0 +1,120 @@ +package ast + +import ( + "fmt" + "strconv" + "strings" +) + +type ValueKind int + +const ( + Variable ValueKind = iota + IntValue + FloatValue + StringValue + BlockValue + BooleanValue + NullValue + EnumValue + ListValue + ObjectValue +) + +type Value struct { + Raw string + Children ChildValueList + Kind ValueKind + Position *Position `dump:"-"` + + // Require validation + Definition *Definition + VariableDefinition *VariableDefinition + ExpectedType *Type +} + +type ChildValue struct { + Name string + Value *Value + Position *Position `dump:"-"` +} + +func (v *Value) Value(vars map[string]interface{}) (interface{}, error) { + if v == nil { + return nil, nil + } + switch v.Kind { + case Variable: + if value, ok := vars[v.Raw]; ok { + return value, nil + } + if v.VariableDefinition != nil && v.VariableDefinition.DefaultValue != nil { + return v.VariableDefinition.DefaultValue.Value(vars) + } + return nil, nil + case IntValue: + return strconv.ParseInt(v.Raw, 10, 64) + case FloatValue: + return strconv.ParseFloat(v.Raw, 64) + case StringValue, BlockValue, EnumValue: + return v.Raw, nil + case BooleanValue: + return strconv.ParseBool(v.Raw) + case NullValue: + return nil, nil + case ListValue: + var val []interface{} + for _, elem := range v.Children { + elemVal, err := elem.Value.Value(vars) + if err != nil { + return val, err + } + val = append(val, elemVal) + } + return val, nil + case ObjectValue: + val := map[string]interface{}{} + for _, elem := range v.Children { + elemVal, err := elem.Value.Value(vars) + if err != nil { + return val, err + } + val[elem.Name] = elemVal + } + return val, nil + default: + panic(fmt.Errorf("unknown value kind %d", v.Kind)) + } +} + +func (v *Value) String() string { + if v == nil { + return "" + } + switch v.Kind { + case Variable: + return "$" + v.Raw + case IntValue, FloatValue, EnumValue, BooleanValue, NullValue: + return v.Raw + case StringValue, BlockValue: + return strconv.Quote(v.Raw) + case ListValue: + var val []string + for _, elem := range v.Children { + val = append(val, elem.Value.String()) + } + return "[" + strings.Join(val, ",") + "]" + case ObjectValue: + var val []string + for _, elem := range v.Children { + val = append(val, elem.Name+":"+elem.Value.String()) + } + return "{" + strings.Join(val, ",") + "}" + default: + panic(fmt.Errorf("unknown value kind %d", v.Kind)) + } +} + +func (v *Value) Dump() string { + return v.String() +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/gqlparser/gqlerror/error.go b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/gqlerror/error.go new file mode 100644 index 00000000..58d1c1bd --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/gqlerror/error.go @@ -0,0 +1,147 @@ +package gqlerror + +import ( + "bytes" + "errors" + "fmt" + "strconv" + + "github.com/open-policy-agent/opa/internal/gqlparser/ast" +) + +// Error is the standard graphql error type described in https://facebook.github.io/graphql/draft/#sec-Errors +type Error struct { + err error `json:"-"` + Message string `json:"message"` + Path ast.Path `json:"path,omitempty"` + Locations []Location `json:"locations,omitempty"` + Extensions map[string]interface{} `json:"extensions,omitempty"` + Rule string `json:"-"` +} + +func (err *Error) SetFile(file string) { + if file == "" { + return + } + if err.Extensions == nil { + err.Extensions = map[string]interface{}{} + } + + err.Extensions["file"] = file +} + +type Location struct { + Line int `json:"line,omitempty"` + Column int `json:"column,omitempty"` +} + +type List []*Error + +func (err *Error) Error() string { + var res bytes.Buffer + if err == nil { + return "" + } + filename, _ := err.Extensions["file"].(string) + if filename == "" { + filename = "input" + } + res.WriteString(filename) + + if len(err.Locations) > 0 { + res.WriteByte(':') + res.WriteString(strconv.Itoa(err.Locations[0].Line)) + res.WriteByte(':') + res.WriteString(strconv.Itoa(err.Locations[0].Column)) + } + + res.WriteString(": ") + if ps := err.pathString(); ps != "" { + res.WriteString(ps) + res.WriteByte(' ') + } + + res.WriteString(err.Message) + + return res.String() +} + +func (err Error) pathString() string { + return err.Path.String() +} + +func (err Error) Unwrap() error { + return err.err +} + +func (errs List) Error() string { + var buf bytes.Buffer + for _, err := range errs { + buf.WriteString(err.Error()) + buf.WriteByte('\n') + } + return buf.String() +} + +func (errs List) Is(target error) bool { + for _, err := range errs { + if errors.Is(err, target) { + return true + } + } + return false +} + +func (errs List) As(target interface{}) bool { + for _, err := range errs { + if errors.As(err, target) { + return true + } + } + return false +} + +func WrapPath(path ast.Path, err error) *Error { + return &Error{ + err: err, + Message: err.Error(), + Path: path, + } +} + +func Errorf(message string, args ...interface{}) *Error { + return &Error{ + Message: fmt.Sprintf(message, args...), + } +} + +func ErrorPathf(path ast.Path, message string, args ...interface{}) *Error { + return &Error{ + Message: fmt.Sprintf(message, args...), + Path: path, + } +} + +func ErrorPosf(pos *ast.Position, message string, args ...interface{}) *Error { + return ErrorLocf( + pos.Src.Name, + pos.Line, + pos.Column, + message, + args..., + ) +} + +func ErrorLocf(file string, line int, col int, message string, args ...interface{}) *Error { + var extensions map[string]interface{} + if file != "" { + extensions = map[string]interface{}{"file": file} + } + return &Error{ + Message: fmt.Sprintf(message, args...), + Extensions: extensions, + Locations: []Location{ + {Line: line, Column: col}, + }, + } +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/gqlparser/lexer/blockstring.go b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/lexer/blockstring.go new file mode 100644 index 00000000..4065a610 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/lexer/blockstring.go @@ -0,0 +1,58 @@ +package lexer + +import ( + "math" + "strings" +) + +// blockStringValue produces the value of a block string from its parsed raw value, similar to +// Coffeescript's block string, Python's docstring trim or Ruby's strip_heredoc. +// +// This implements the GraphQL spec's BlockStringValue() static algorithm. +func blockStringValue(raw string) string { + lines := strings.Split(raw, "\n") + + commonIndent := math.MaxInt32 + for _, line := range lines { + indent := leadingWhitespace(line) + if indent < len(line) && indent < commonIndent { + commonIndent = indent + if commonIndent == 0 { + break + } + } + } + + if commonIndent != math.MaxInt32 && len(lines) > 0 { + for i := 1; i < len(lines); i++ { + if len(lines[i]) < commonIndent { + lines[i] = "" + } else { + lines[i] = lines[i][commonIndent:] + } + } + } + + start := 0 + end := len(lines) + + for start < end && leadingWhitespace(lines[start]) == math.MaxInt32 { + start++ + } + + for start < end && leadingWhitespace(lines[end-1]) == math.MaxInt32 { + end-- + } + + return strings.Join(lines[start:end], "\n") +} + +func leadingWhitespace(str string) int { + for i, r := range str { + if r != ' ' && r != '\t' { + return i + } + } + // this line is made up entirely of whitespace, its leading whitespace doesnt count. + return math.MaxInt32 +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/gqlparser/lexer/lexer.go b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/lexer/lexer.go new file mode 100644 index 00000000..f25555e6 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/lexer/lexer.go @@ -0,0 +1,517 @@ +package lexer + +import ( + "bytes" + "unicode/utf8" + + "github.com/open-policy-agent/opa/internal/gqlparser/ast" + "github.com/open-policy-agent/opa/internal/gqlparser/gqlerror" +) + +// Lexer turns graphql request and schema strings into tokens +type Lexer struct { + *ast.Source + // An offset into the string in bytes + start int + // An offset into the string in runes + startRunes int + // An offset into the string in bytes + end int + // An offset into the string in runes + endRunes int + // the current line number + line int + // An offset into the string in rune + lineStartRunes int +} + +func New(src *ast.Source) Lexer { + return Lexer{ + Source: src, + line: 1, + } +} + +// take one rune from input and advance end +func (s *Lexer) peek() (rune, int) { + return utf8.DecodeRuneInString(s.Input[s.end:]) +} + +func (s *Lexer) makeToken(kind Type) (Token, error) { + return s.makeValueToken(kind, s.Input[s.start:s.end]) +} + +func (s *Lexer) makeValueToken(kind Type, value string) (Token, error) { + return Token{ + Kind: kind, + Value: value, + Pos: ast.Position{ + Start: s.startRunes, + End: s.endRunes, + Line: s.line, + Column: s.startRunes - s.lineStartRunes + 1, + Src: s.Source, + }, + }, nil +} + +func (s *Lexer) makeError(format string, args ...interface{}) (Token, error) { + column := s.endRunes - s.lineStartRunes + 1 + return Token{ + Kind: Invalid, + Pos: ast.Position{ + Start: s.startRunes, + End: s.endRunes, + Line: s.line, + Column: column, + Src: s.Source, + }, + }, gqlerror.ErrorLocf(s.Source.Name, s.line, column, format, args...) +} + +// ReadToken gets the next token from the source starting at the given position. +// +// This skips over whitespace and comments until it finds the next lexable +// token, then lexes punctuators immediately or calls the appropriate helper +// function for more complicated tokens. +func (s *Lexer) ReadToken() (token Token, err error) { + + s.ws() + s.start = s.end + s.startRunes = s.endRunes + + if s.end >= len(s.Input) { + return s.makeToken(EOF) + } + r := s.Input[s.start] + s.end++ + s.endRunes++ + switch r { + case '!': + return s.makeValueToken(Bang, "") + + case '$': + return s.makeValueToken(Dollar, "") + case '&': + return s.makeValueToken(Amp, "") + case '(': + return s.makeValueToken(ParenL, "") + case ')': + return s.makeValueToken(ParenR, "") + case '.': + if len(s.Input) > s.start+2 && s.Input[s.start:s.start+3] == "..." { + s.end += 2 + s.endRunes += 2 + return s.makeValueToken(Spread, "") + } + case ':': + return s.makeValueToken(Colon, "") + case '=': + return s.makeValueToken(Equals, "") + case '@': + return s.makeValueToken(At, "") + case '[': + return s.makeValueToken(BracketL, "") + case ']': + return s.makeValueToken(BracketR, "") + case '{': + return s.makeValueToken(BraceL, "") + case '}': + return s.makeValueToken(BraceR, "") + case '|': + return s.makeValueToken(Pipe, "") + case '#': + if comment, err := s.readComment(); err != nil { + return comment, err + } + return s.ReadToken() + + case '_', 'a', 'b', 'c', 'd', 'e', 'f', 'g', 'h', 'i', 'j', 'k', 'l', 'm', 'n', 'o', 'p', 'q', 'r', 's', 't', 'u', 'v', 'w', 'x', 'y', 'z', 'A', 'B', 'C', 'D', 'E', 'F', 'G', 'H', 'I', 'J', 'K', 'L', 'M', 'N', 'O', 'P', 'Q', 'R', 'S', 'T', 'U', 'V', 'W', 'X', 'Y', 'Z': + return s.readName() + + case '-', '0', '1', '2', '3', '4', '5', '6', '7', '8', '9': + return s.readNumber() + + case '"': + if len(s.Input) > s.start+2 && s.Input[s.start:s.start+3] == `"""` { + return s.readBlockString() + } + + return s.readString() + } + + s.end-- + s.endRunes-- + + if r < 0x0020 && r != 0x0009 && r != 0x000a && r != 0x000d { + return s.makeError(`Cannot contain the invalid character "\u%04d"`, r) + } + + if r == '\'' { + return s.makeError(`Unexpected single quote character ('), did you mean to use a double quote (")?`) + } + + return s.makeError(`Cannot parse the unexpected character "%s".`, string(r)) +} + +// ws reads from body starting at startPosition until it finds a non-whitespace +// or commented character, and updates the token end to include all whitespace +func (s *Lexer) ws() { + for s.end < len(s.Input) { + switch s.Input[s.end] { + case '\t', ' ', ',': + s.end++ + s.endRunes++ + case '\n': + s.end++ + s.endRunes++ + s.line++ + s.lineStartRunes = s.endRunes + case '\r': + s.end++ + s.endRunes++ + s.line++ + s.lineStartRunes = s.endRunes + // skip the following newline if its there + if s.end < len(s.Input) && s.Input[s.end] == '\n' { + s.end++ + s.endRunes++ + } + // byte order mark, given ws is hot path we aren't relying on the unicode package here. + case 0xef: + if s.end+2 < len(s.Input) && s.Input[s.end+1] == 0xBB && s.Input[s.end+2] == 0xBF { + s.end += 3 + s.endRunes++ + } else { + return + } + default: + return + } + } +} + +// readComment from the input +// +// #[\u0009\u0020-\uFFFF]* +func (s *Lexer) readComment() (Token, error) { + for s.end < len(s.Input) { + r, w := s.peek() + + // SourceCharacter but not LineTerminator + if r > 0x001f || r == '\t' { + s.end += w + s.endRunes++ + } else { + break + } + } + + return s.makeToken(Comment) +} + +// readNumber from the input, either a float +// or an int depending on whether a decimal point appears. +// +// Int: -?(0|[1-9][0-9]*) +// Float: -?(0|[1-9][0-9]*)(\.[0-9]+)?((E|e)(+|-)?[0-9]+)? +func (s *Lexer) readNumber() (Token, error) { + float := false + + // backup to the first digit + s.end-- + s.endRunes-- + + s.acceptByte('-') + + if s.acceptByte('0') { + if consumed := s.acceptDigits(); consumed != 0 { + s.end -= consumed + s.endRunes -= consumed + return s.makeError("Invalid number, unexpected digit after 0: %s.", s.describeNext()) + } + } else { + if consumed := s.acceptDigits(); consumed == 0 { + return s.makeError("Invalid number, expected digit but got: %s.", s.describeNext()) + } + } + + if s.acceptByte('.') { + float = true + + if consumed := s.acceptDigits(); consumed == 0 { + return s.makeError("Invalid number, expected digit but got: %s.", s.describeNext()) + } + } + + if s.acceptByte('e', 'E') { + float = true + + s.acceptByte('-', '+') + + if consumed := s.acceptDigits(); consumed == 0 { + return s.makeError("Invalid number, expected digit but got: %s.", s.describeNext()) + } + } + + if float { + return s.makeToken(Float) + } + return s.makeToken(Int) + +} + +// acceptByte if it matches any of given bytes, returning true if it found anything +func (s *Lexer) acceptByte(bytes ...uint8) bool { + if s.end >= len(s.Input) { + return false + } + + for _, accepted := range bytes { + if s.Input[s.end] == accepted { + s.end++ + s.endRunes++ + return true + } + } + return false +} + +// acceptDigits from the input, returning the number of digits it found +func (s *Lexer) acceptDigits() int { + consumed := 0 + for s.end < len(s.Input) && s.Input[s.end] >= '0' && s.Input[s.end] <= '9' { + s.end++ + s.endRunes++ + consumed++ + } + + return consumed +} + +// describeNext peeks at the input and returns a human readable string. This should will alloc +// and should only be used in errors +func (s *Lexer) describeNext() string { + if s.end < len(s.Input) { + return `"` + string(s.Input[s.end]) + `"` + } + return "" +} + +// readString from the input +// +// "([^"\\\u000A\u000D]|(\\(u[0-9a-fA-F]{4}|["\\/bfnrt])))*" +func (s *Lexer) readString() (Token, error) { + inputLen := len(s.Input) + + // this buffer is lazily created only if there are escape characters. + var buf *bytes.Buffer + + // skip the opening quote + s.start++ + s.startRunes++ + + for s.end < inputLen { + r := s.Input[s.end] + if r == '\n' || r == '\r' { + break + } + if r < 0x0020 && r != '\t' { + return s.makeError(`Invalid character within String: "\u%04d".`, r) + } + switch r { + default: + var char = rune(r) + var w = 1 + + // skip unicode overhead if we are in the ascii range + if r >= 127 { + char, w = utf8.DecodeRuneInString(s.Input[s.end:]) + } + s.end += w + s.endRunes++ + + if buf != nil { + buf.WriteRune(char) + } + + case '"': + t, err := s.makeToken(String) + // the token should not include the quotes in its value, but should cover them in its position + t.Pos.Start-- + t.Pos.End++ + + if buf != nil { + t.Value = buf.String() + } + + // skip the close quote + s.end++ + s.endRunes++ + + return t, err + + case '\\': + if s.end+1 >= inputLen { + s.end++ + s.endRunes++ + return s.makeError(`Invalid character escape sequence.`) + } + + if buf == nil { + buf = bytes.NewBufferString(s.Input[s.start:s.end]) + } + + escape := s.Input[s.end+1] + + if escape == 'u' { + if s.end+6 >= inputLen { + s.end++ + s.endRunes++ + return s.makeError("Invalid character escape sequence: \\%s.", s.Input[s.end:]) + } + + r, ok := unhex(s.Input[s.end+2 : s.end+6]) + if !ok { + s.end++ + s.endRunes++ + return s.makeError("Invalid character escape sequence: \\%s.", s.Input[s.end:s.end+5]) + } + buf.WriteRune(r) + s.end += 6 + s.endRunes += 6 + } else { + switch escape { + case '"', '/', '\\': + buf.WriteByte(escape) + case 'b': + buf.WriteByte('\b') + case 'f': + buf.WriteByte('\f') + case 'n': + buf.WriteByte('\n') + case 'r': + buf.WriteByte('\r') + case 't': + buf.WriteByte('\t') + default: + s.end++ + s.endRunes++ + return s.makeError("Invalid character escape sequence: \\%s.", string(escape)) + } + s.end += 2 + s.endRunes += 2 + } + } + } + + return s.makeError("Unterminated string.") +} + +// readBlockString from the input +// +// """("?"?(\\"""|\\(?!=""")|[^"\\]))*""" +func (s *Lexer) readBlockString() (Token, error) { + inputLen := len(s.Input) + + var buf bytes.Buffer + + // skip the opening quote + s.start += 3 + s.startRunes += 3 + s.end += 2 + s.endRunes += 2 + + for s.end < inputLen { + r := s.Input[s.end] + + // Closing triple quote (""") + if r == '"' && s.end+3 <= inputLen && s.Input[s.end:s.end+3] == `"""` { + t, err := s.makeValueToken(BlockString, blockStringValue(buf.String())) + + // the token should not include the quotes in its value, but should cover them in its position + t.Pos.Start -= 3 + t.Pos.End += 3 + + // skip the close quote + s.end += 3 + s.endRunes += 3 + return t, err + } + + // SourceCharacter + if r < 0x0020 && r != '\t' && r != '\n' && r != '\r' { + return s.makeError(`Invalid character within String: "\u%04d".`, r) + } + + if r == '\\' && s.end+4 <= inputLen && s.Input[s.end:s.end+4] == `\"""` { + buf.WriteString(`"""`) + s.end += 4 + s.endRunes += 4 + } else if r == '\r' { + if s.end+1 < inputLen && s.Input[s.end+1] == '\n' { + s.end++ + s.endRunes++ + } + + buf.WriteByte('\n') + s.end++ + s.endRunes++ + s.line++ + s.lineStartRunes = s.endRunes + } else { + var char = rune(r) + var w = 1 + + // skip unicode overhead if we are in the ascii range + if r >= 127 { + char, w = utf8.DecodeRuneInString(s.Input[s.end:]) + } + s.end += w + s.endRunes++ + buf.WriteRune(char) + if r == '\n' { + s.line++ + s.lineStartRunes = s.endRunes + } + } + } + + return s.makeError("Unterminated string.") +} + +func unhex(b string) (v rune, ok bool) { + for _, c := range b { + v <<= 4 + switch { + case '0' <= c && c <= '9': + v |= c - '0' + case 'a' <= c && c <= 'f': + v |= c - 'a' + 10 + case 'A' <= c && c <= 'F': + v |= c - 'A' + 10 + default: + return 0, false + } + } + + return v, true +} + +// readName from the input +// +// [_A-Za-z][_0-9A-Za-z]* +func (s *Lexer) readName() (Token, error) { + for s.end < len(s.Input) { + r, w := s.peek() + + if (r >= '0' && r <= '9') || (r >= 'A' && r <= 'Z') || (r >= 'a' && r <= 'z') || r == '_' { + s.end += w + s.endRunes++ + } else { + break + } + } + + return s.makeToken(Name) +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/gqlparser/lexer/lexer_test.yml b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/lexer/lexer_test.yml new file mode 100644 index 00000000..5c4d5f0f --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/lexer/lexer_test.yml @@ -0,0 +1,692 @@ +encoding: + - name: disallows uncommon control characters + input: "\u0007" + error: + message: 'Cannot contain the invalid character "\u0007"' + locations: [{line: 1, column: 1}] + + - name: accepts BOM header + input: "\uFEFF foo" + tokens: + - + kind: NAME + start: 2 + end: 5 + value: 'foo' + +simple tokens: + - name: records line and column + input: "\n \r\n \r foo\n" + tokens: + - + kind: NAME + start: 8 + end: 11 + line: 4 + column: 3 + value: 'foo' + + - name: skips whitespace + input: "\n\n foo\n\n\n" + tokens: + - + kind: NAME + start: 6 + end: 9 + value: 'foo' + + - name: skips comments + input: "\n #comment\n foo#comment\n" + tokens: + - + kind: NAME + start: 18 + end: 21 + value: 'foo' + + - name: skips commas + input: ",,,foo,,," + tokens: + - + kind: NAME + start: 3 + end: 6 + value: 'foo' + + - name: errors respect whitespace + input: "\n\n ?\n\n\n" + error: + message: 'Cannot parse the unexpected character "?".' + locations: [{line: 3, column: 5}] + string: | + Syntax Error: Cannot parse the unexpected character "?". + GraphQL request (3:5) + 2: + 3: ? + ^ + 4: + + - name: lex reports useful information for dashes in names + input: "a-b" + error: + message: 'Invalid number, expected digit but got: "b".' + locations: [{ line: 1, column: 3 }] + tokens: + - + kind: Name + start: 0 + end: 1 + value: a + +lexes strings: + - name: basic + input: '"simple"' + tokens: + - + kind: STRING + start: 0 + end: 8 + value: 'simple' + + - name: whitespace + input: '" white space "' + tokens: + - + kind: STRING + start: 0 + end: 15 + value: ' white space ' + + - name: quote + input: '"quote \""' + tokens: + - + kind: STRING + start: 0 + end: 10 + value: 'quote "' + + - name: escaped + input: '"escaped \n\r\b\t\f"' + tokens: + - + kind: STRING + start: 0 + end: 20 + value: "escaped \n\r\b\t\f" + + - name: slashes + input: '"slashes \\ \/"' + tokens: + - + kind: STRING + start: 0 + end: 15 + value: 'slashes \ /' + + - name: unicode + input: '"unicode \u1234\u5678\u90AB\uCDEF"' + tokens: + - + kind: STRING + start: 0 + end: 34 + value: "unicode \u1234\u5678\u90AB\uCDEF" + +lex reports useful string errors: + - name: unterminated + input: '"' + error: + message: "Unterminated string." + locations: [{ line: 1, column: 2 }] + + - name: no end quote + input: '"no end quote' + error: + message: 'Unterminated string.' + locations: [{ line: 1, column: 14 }] + + - name: single quotes + input: "'single quotes'" + error: + message: "Unexpected single quote character ('), did you mean to use a double quote (\")?" + locations: [{ line: 1, column: 1 }] + + - name: control characters + input: "\"contains unescaped \u0007 control char\"" + error: + message: 'Invalid character within String: "\u0007".' + locations: [{ line: 1, column: 21 }] + + - name: null byte + input: "\"null-byte is not \u0000 end of file\"" + error: + message: 'Invalid character within String: "\u0000".' + locations: [{ line: 1, column: 19 }] + + - name: unterminated newline + input: "\"multi\nline\"" + error: + message: 'Unterminated string.' + locations: [{line: 1, column: 7 }] + + - name: unterminated carriage return + input: "\"multi\rline\"" + error: + message: 'Unterminated string.' + locations: [{ line: 1, column: 7 }] + + - name: bad escape character + input: '"bad \z esc"' + error: + message: 'Invalid character escape sequence: \z.' + locations: [{ line: 1, column: 7 }] + + - name: hex escape sequence + input: '"bad \x esc"' + error: + message: 'Invalid character escape sequence: \x.' + locations: [{ line: 1, column: 7 }] + + - name: short escape sequence + input: '"bad \u1 esc"' + error: + message: 'Invalid character escape sequence: \u1 es.' + locations: [{ line: 1, column: 7 }] + + - name: invalid escape sequence 1 + input: '"bad \u0XX1 esc"' + error: + message: 'Invalid character escape sequence: \u0XX1.' + locations: [{ line: 1, column: 7 }] + + - name: invalid escape sequence 2 + input: '"bad \uXXXX esc"' + error: + message: 'Invalid character escape sequence: \uXXXX.' + locations: [{ line: 1, column: 7 }] + + - name: invalid escape sequence 3 + input: '"bad \uFXXX esc"' + error: + message: 'Invalid character escape sequence: \uFXXX.' + locations: [{ line: 1, column: 7 }] + + - name: invalid character escape sequence + input: '"bad \uXXXF esc"' + error: + message: 'Invalid character escape sequence: \uXXXF.' + locations: [{ line: 1, column: 7 }] + +lexes block strings: + - name: simple + input: '"""simple"""' + tokens: + - + kind: BLOCK_STRING + start: 0 + end: 12 + value: 'simple' + + - name: white space + input: '""" white space """' + tokens: + - + kind: BLOCK_STRING + start: 0 + end: 19 + value: ' white space ' + + - name: contains quote + input: '"""contains " quote"""' + tokens: + - + kind: BLOCK_STRING + start: 0 + end: 22 + value: 'contains " quote' + + - name: contains triplequote + input: "\"\"\"contains \\\"\"\" triplequote\"\"\"" + tokens: + - + kind: BLOCK_STRING + start: 0 + end: 31 + value: 'contains """ triplequote' + + - name: multi line + input: "\"\"\"multi\nline\"\"\"" + tokens: + - + kind: BLOCK_STRING + start: 0 + end: 16 + value: "multi\nline" + + - name: multi line normalized + input: "\"\"\"multi\rline\r\nnormalized\"\"\"" + tokens: + - + kind: BLOCK_STRING + start: 0 + end: 28 + value: "multi\nline\nnormalized" + + - name: unescaped + input: '"""unescaped \n\r\b\t\f\u1234"""' + tokens: + - + kind: BLOCK_STRING + start: 0 + end: 32 + value: 'unescaped \n\r\b\t\f\u1234' + + - name: slashes + input: '"""slashes \\ \/"""' + tokens: + - + kind: BLOCK_STRING + start: 0 + end: 19 + value: 'slashes \\ \/' + + - name: multiple lines + input: | + """ + + spans + multiple + lines + + """ + tokens: + - + kind: BLOCK_STRING + start: 0 + end: 36 + value: "spans\n multiple\n lines" + + - name: records correct line and column after block string + input: | + """ + + some + description + + """ foo + tokens: + - + kind: BLOCK_STRING + value: "some\ndescription" + - + kind: NAME + start: 27 + end: 30 + line: 6 + column: 5 + value: 'foo' + +lex reports useful block string errors: + - name: unterminated string + input: '"""' + error: + message: "Unterminated string." + locations: [{ line: 1, column: 4 }] + + - name: unescaped control characters + input: "\"\"\"contains unescaped \u0007 control char\"\"\"" + error: + message: 'Invalid character within String: "\u0007".' + locations: [{ line: 1, column: 23 }] + + - name: null byte + input: "\"\"\"null-byte is not \u0000 end of file\"\"\"" + error: + message: 'Invalid character within String: "\u0000".' + locations: [{ line: 1, column: 21 }] + +lexes numbers: + - name: integer + input: "4" + tokens: + - + kind: INT + start: 0 + end: 1 + value: '4' + + - name: float + input: "4.123" + tokens: + - + kind: FLOAT + start: 0 + end: 5 + value: '4.123' + + - name: negative + input: "-4" + tokens: + - + kind: INT + start: 0 + end: 2 + value: '-4' + + - name: nine + input: "9" + tokens: + - + kind: INT + start: 0 + end: 1 + value: '9' + + - name: zero + input: "0" + tokens: + - + kind: INT + start: 0 + end: 1 + value: '0' + + - name: negative float + input: "-4.123" + tokens: + - + kind: FLOAT + start: 0 + end: 6 + value: '-4.123' + + - name: float leading zero + input: "0.123" + tokens: + - + kind: FLOAT + start: 0 + end: 5 + value: '0.123' + + - name: exponent whole + input: "123e4" + tokens: + - + kind: FLOAT + start: 0 + end: 5 + value: '123e4' + + - name: exponent uppercase + input: "123E4" + tokens: + - + kind: FLOAT + start: 0 + end: 5 + value: '123E4' + + - name: exponent negative power + input: "123e-4" + tokens: + - + kind: FLOAT + start: 0 + end: 6 + value: '123e-4' + + - name: exponent positive power + input: "123e+4" + tokens: + - + kind: FLOAT + start: 0 + end: 6 + value: '123e+4' + + - name: exponent negative base + input: "-1.123e4" + tokens: + - + kind: FLOAT + start: 0 + end: 8 + value: '-1.123e4' + + - name: exponent negative base upper + input: "-1.123E4" + tokens: + - + kind: FLOAT + start: 0 + end: 8 + value: '-1.123E4' + + - name: exponent negative base negative power + input: "-1.123e-4" + tokens: + - + kind: FLOAT + start: 0 + end: 9 + value: '-1.123e-4' + + - name: exponent negative base positive power + input: "-1.123e+4" + tokens: + - + kind: FLOAT + start: 0 + end: 9 + value: '-1.123e+4' + + - name: exponent negative base large power + input: "-1.123e4567" + tokens: + - + kind: FLOAT + start: 0 + end: 11 + value: '-1.123e4567' + +lex reports useful number errors: + - name: zero + input: "00" + error: + message: 'Invalid number, unexpected digit after 0: "0".' + locations: [{ line: 1, column: 2 }] + + - name: positive + input: "+1" + error: + message: 'Cannot parse the unexpected character "+".' + locations: [{ line: 1, column: 1 }] + + - name: trailing dot + input: "1." + error: + message: 'Invalid number, expected digit but got: .' + locations: [{ line: 1, column: 3 }] + + - name: traililng dot exponent + input: "1.e1" + error: + message: 'Invalid number, expected digit but got: "e".' + locations: [{ line: 1, column: 3 }] + + - name: missing leading zero + input: ".123" + error: + message: 'Cannot parse the unexpected character ".".' + locations: [{ line: 1, column: 1 }] + + - name: characters + input: "1.A" + error: + message: 'Invalid number, expected digit but got: "A".' + locations: [{ line: 1, column: 3 }] + + - name: negative characters + input: "-A" + error: + message: 'Invalid number, expected digit but got: "A".' + locations: [{ line: 1, column: 2 }] + + - name: missing exponent + input: '1.0e' + error: + message: 'Invalid number, expected digit but got: .' + locations: [{ line: 1, column: 5 }] + + - name: character exponent + input: "1.0eA" + error: + message: 'Invalid number, expected digit but got: "A".' + locations: [{ line: 1, column: 5 }] + +lexes punctuation: + - name: bang + input: "!" + tokens: + - + kind: BANG + start: 0 + end: 1 + value: undefined + + - name: dollar + input: "$" + tokens: + - + kind: DOLLAR + start: 0 + end: 1 + value: undefined + + - name: open paren + input: "(" + tokens: + - + kind: PAREN_L + start: 0 + end: 1 + value: undefined + + - name: close paren + input: ")" + tokens: + - + kind: PAREN_R + start: 0 + end: 1 + value: undefined + + - name: spread + input: "..." + tokens: + - + kind: SPREAD + start: 0 + end: 3 + value: undefined + + - name: colon + input: ":" + tokens: + - + kind: COLON + start: 0 + end: 1 + value: undefined + + - name: equals + input: "=" + tokens: + - + kind: EQUALS + start: 0 + end: 1 + value: undefined + + - name: at + input: "@" + tokens: + - + kind: AT + start: 0 + end: 1 + value: undefined + + - name: open bracket + input: "[" + tokens: + - + kind: BRACKET_L + start: 0 + end: 1 + value: undefined + + - name: close bracket + input: "]" + tokens: + - + kind: BRACKET_R + start: 0 + end: 1 + value: undefined + + - name: open brace + input: "{" + tokens: + - + kind: BRACE_L + start: 0 + end: 1 + value: undefined + + - name: close brace + input: "}" + tokens: + - + kind: BRACE_R + start: 0 + end: 1 + value: undefined + + - name: pipe + input: "|" + tokens: + - + kind: PIPE + start: 0 + end: 1 + value: undefined + +lex reports useful unknown character error: + - name: not a spread + input: ".." + error: + message: 'Cannot parse the unexpected character ".".' + locations: [{ line: 1, column: 1 }] + + - name: question mark + input: "?" + error: + message: 'Cannot parse the unexpected character "?".' + message: 'Cannot parse the unexpected character "?".' + locations: [{ line: 1, column: 1 }] + + - name: unicode 203 + input: "\u203B" + error: + message: 'Cannot parse the unexpected character "â".' + locations: [{ line: 1, column: 1 }] + + - name: unicode 200 + input: "\u200b" + error: + message: 'Cannot parse the unexpected character "â".' + locations: [{ line: 1, column: 1 }] + diff --git a/vendor/github.com/open-policy-agent/opa/internal/gqlparser/lexer/token.go b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/lexer/token.go new file mode 100644 index 00000000..79eefd0f --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/lexer/token.go @@ -0,0 +1,148 @@ +package lexer + +import ( + "strconv" + + "github.com/open-policy-agent/opa/internal/gqlparser/ast" +) + +const ( + Invalid Type = iota + EOF + Bang + Dollar + Amp + ParenL + ParenR + Spread + Colon + Equals + At + BracketL + BracketR + BraceL + BraceR + Pipe + Name + Int + Float + String + BlockString + Comment +) + +func (t Type) Name() string { + switch t { + case Invalid: + return "Invalid" + case EOF: + return "EOF" + case Bang: + return "Bang" + case Dollar: + return "Dollar" + case Amp: + return "Amp" + case ParenL: + return "ParenL" + case ParenR: + return "ParenR" + case Spread: + return "Spread" + case Colon: + return "Colon" + case Equals: + return "Equals" + case At: + return "At" + case BracketL: + return "BracketL" + case BracketR: + return "BracketR" + case BraceL: + return "BraceL" + case BraceR: + return "BraceR" + case Pipe: + return "Pipe" + case Name: + return "Name" + case Int: + return "Int" + case Float: + return "Float" + case String: + return "String" + case BlockString: + return "BlockString" + case Comment: + return "Comment" + } + return "Unknown " + strconv.Itoa(int(t)) +} + +func (t Type) String() string { + switch t { + case Invalid: + return "" + case EOF: + return "" + case Bang: + return "!" + case Dollar: + return "$" + case Amp: + return "&" + case ParenL: + return "(" + case ParenR: + return ")" + case Spread: + return "..." + case Colon: + return ":" + case Equals: + return "=" + case At: + return "@" + case BracketL: + return "[" + case BracketR: + return "]" + case BraceL: + return "{" + case BraceR: + return "}" + case Pipe: + return "|" + case Name: + return "Name" + case Int: + return "Int" + case Float: + return "Float" + case String: + return "String" + case BlockString: + return "BlockString" + case Comment: + return "Comment" + } + return "Unknown " + strconv.Itoa(int(t)) +} + +// Kind represents a type of token. The types are predefined as constants. +type Type int + +type Token struct { + Kind Type // The token type. + Value string // The literal value consumed. + Pos ast.Position // The file and line this token was read from +} + +func (t Token) String() string { + if t.Value != "" { + return t.Kind.String() + " " + strconv.Quote(t.Value) + } + return t.Kind.String() +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/gqlparser/parser/parser.go b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/parser/parser.go new file mode 100644 index 00000000..c0d2b4a3 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/parser/parser.go @@ -0,0 +1,136 @@ +package parser + +import ( + "strconv" + + "github.com/open-policy-agent/opa/internal/gqlparser/ast" + "github.com/open-policy-agent/opa/internal/gqlparser/gqlerror" + "github.com/open-policy-agent/opa/internal/gqlparser/lexer" +) + +type parser struct { + lexer lexer.Lexer + err error + + peeked bool + peekToken lexer.Token + peekError error + + prev lexer.Token +} + +func (p *parser) peekPos() *ast.Position { + if p.err != nil { + return nil + } + + peek := p.peek() + return &peek.Pos +} + +func (p *parser) peek() lexer.Token { + if p.err != nil { + return p.prev + } + + if !p.peeked { + p.peekToken, p.peekError = p.lexer.ReadToken() + p.peeked = true + } + + return p.peekToken +} + +func (p *parser) error(tok lexer.Token, format string, args ...interface{}) { + if p.err != nil { + return + } + p.err = gqlerror.ErrorLocf(tok.Pos.Src.Name, tok.Pos.Line, tok.Pos.Column, format, args...) +} + +func (p *parser) next() lexer.Token { + if p.err != nil { + return p.prev + } + if p.peeked { + p.peeked = false + p.prev, p.err = p.peekToken, p.peekError + } else { + p.prev, p.err = p.lexer.ReadToken() + } + return p.prev +} + +func (p *parser) expectKeyword(value string) lexer.Token { + tok := p.peek() + if tok.Kind == lexer.Name && tok.Value == value { + return p.next() + } + + p.error(tok, "Expected %s, found %s", strconv.Quote(value), tok.String()) + return tok +} + +func (p *parser) expect(kind lexer.Type) lexer.Token { + tok := p.peek() + if tok.Kind == kind { + return p.next() + } + + p.error(tok, "Expected %s, found %s", kind, tok.Kind.String()) + return tok +} + +func (p *parser) skip(kind lexer.Type) bool { + if p.err != nil { + return false + } + + tok := p.peek() + + if tok.Kind != kind { + return false + } + p.next() + return true +} + +func (p *parser) unexpectedError() { + p.unexpectedToken(p.peek()) +} + +func (p *parser) unexpectedToken(tok lexer.Token) { + p.error(tok, "Unexpected %s", tok.String()) +} + +func (p *parser) many(start lexer.Type, end lexer.Type, cb func()) { + hasDef := p.skip(start) + if !hasDef { + return + } + + for p.peek().Kind != end && p.err == nil { + cb() + } + p.next() +} + +func (p *parser) some(start lexer.Type, end lexer.Type, cb func()) { + hasDef := p.skip(start) + if !hasDef { + return + } + + called := false + for p.peek().Kind != end && p.err == nil { + called = true + cb() + } + + if !called { + p.error(p.peek(), "expected at least one definition, found %s", p.peek().Kind.String()) + return + } + + p.next() +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/gqlparser/parser/query.go b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/parser/query.go new file mode 100644 index 00000000..319425f5 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/parser/query.go @@ -0,0 +1,349 @@ +package parser + +import ( + "github.com/open-policy-agent/opa/internal/gqlparser/lexer" + + //nolint:revive + . "github.com/open-policy-agent/opa/internal/gqlparser/ast" +) + +func ParseQuery(source *Source) (*QueryDocument, error) { + p := parser{ + lexer: lexer.New(source), + } + return p.parseQueryDocument(), p.err +} + +func (p *parser) parseQueryDocument() *QueryDocument { + var doc QueryDocument + for p.peek().Kind != lexer.EOF { + if p.err != nil { + return &doc + } + doc.Position = p.peekPos() + switch p.peek().Kind { + case lexer.Name: + switch p.peek().Value { + case "query", "mutation", "subscription": + doc.Operations = append(doc.Operations, p.parseOperationDefinition()) + case "fragment": + doc.Fragments = append(doc.Fragments, p.parseFragmentDefinition()) + default: + p.unexpectedError() + } + case lexer.BraceL: + doc.Operations = append(doc.Operations, p.parseOperationDefinition()) + default: + p.unexpectedError() + } + } + + return &doc +} + +func (p *parser) parseOperationDefinition() *OperationDefinition { + if p.peek().Kind == lexer.BraceL { + return &OperationDefinition{ + Position: p.peekPos(), + Operation: Query, + SelectionSet: p.parseRequiredSelectionSet(), + } + } + + var od OperationDefinition + od.Position = p.peekPos() + od.Operation = p.parseOperationType() + + if p.peek().Kind == lexer.Name { + od.Name = p.next().Value + } + + od.VariableDefinitions = p.parseVariableDefinitions() + od.Directives = p.parseDirectives(false) + od.SelectionSet = p.parseRequiredSelectionSet() + + return &od +} + +func (p *parser) parseOperationType() Operation { + tok := p.next() + switch tok.Value { + case "query": + return Query + case "mutation": + return Mutation + case "subscription": + return Subscription + } + p.unexpectedToken(tok) + return "" +} + +func (p *parser) parseVariableDefinitions() VariableDefinitionList { + var defs []*VariableDefinition + p.many(lexer.ParenL, lexer.ParenR, func() { + defs = append(defs, p.parseVariableDefinition()) + }) + + return defs +} + +func (p *parser) parseVariableDefinition() *VariableDefinition { + var def VariableDefinition + def.Position = p.peekPos() + def.Variable = p.parseVariable() + + p.expect(lexer.Colon) + + def.Type = p.parseTypeReference() + + if p.skip(lexer.Equals) { + def.DefaultValue = p.parseValueLiteral(true) + } + + def.Directives = p.parseDirectives(false) + + return &def +} + +func (p *parser) parseVariable() string { + p.expect(lexer.Dollar) + return p.parseName() +} + +func (p *parser) parseOptionalSelectionSet() SelectionSet { + var selections []Selection + p.some(lexer.BraceL, lexer.BraceR, func() { + selections = append(selections, p.parseSelection()) + }) + + return SelectionSet(selections) +} + +func (p *parser) parseRequiredSelectionSet() SelectionSet { + if p.peek().Kind != lexer.BraceL { + p.error(p.peek(), "Expected %s, found %s", lexer.BraceL, p.peek().Kind.String()) + return nil + } + + var selections []Selection + p.some(lexer.BraceL, lexer.BraceR, func() { + selections = append(selections, p.parseSelection()) + }) + + return SelectionSet(selections) +} + +func (p *parser) parseSelection() Selection { + if p.peek().Kind == lexer.Spread { + return p.parseFragment() + } + return p.parseField() +} + +func (p *parser) parseField() *Field { + var field Field + field.Position = p.peekPos() + field.Alias = p.parseName() + + if p.skip(lexer.Colon) { + field.Name = p.parseName() + } else { + field.Name = field.Alias + } + + field.Arguments = p.parseArguments(false) + field.Directives = p.parseDirectives(false) + if p.peek().Kind == lexer.BraceL { + field.SelectionSet = p.parseOptionalSelectionSet() + } + + return &field +} + +func (p *parser) parseArguments(isConst bool) ArgumentList { + var arguments ArgumentList + p.many(lexer.ParenL, lexer.ParenR, func() { + arguments = append(arguments, p.parseArgument(isConst)) + }) + + return arguments +} + +func (p *parser) parseArgument(isConst bool) *Argument { + arg := Argument{} + arg.Position = p.peekPos() + arg.Name = p.parseName() + p.expect(lexer.Colon) + + arg.Value = p.parseValueLiteral(isConst) + return &arg +} + +func (p *parser) parseFragment() Selection { + p.expect(lexer.Spread) + + if peek := p.peek(); peek.Kind == lexer.Name && peek.Value != "on" { + return &FragmentSpread{ + Position: p.peekPos(), + Name: p.parseFragmentName(), + Directives: p.parseDirectives(false), + } + } + + var def InlineFragment + def.Position = p.peekPos() + if p.peek().Value == "on" { + p.next() // "on" + + def.TypeCondition = p.parseName() + } + + def.Directives = p.parseDirectives(false) + def.SelectionSet = p.parseRequiredSelectionSet() + return &def +} + +func (p *parser) parseFragmentDefinition() *FragmentDefinition { + var def FragmentDefinition + def.Position = p.peekPos() + p.expectKeyword("fragment") + + def.Name = p.parseFragmentName() + def.VariableDefinition = p.parseVariableDefinitions() + + p.expectKeyword("on") + + def.TypeCondition = p.parseName() + def.Directives = p.parseDirectives(false) + def.SelectionSet = p.parseRequiredSelectionSet() + return &def +} + +func (p *parser) parseFragmentName() string { + if p.peek().Value == "on" { + p.unexpectedError() + return "" + } + + return p.parseName() +} + +func (p *parser) parseValueLiteral(isConst bool) *Value { + token := p.peek() + + var kind ValueKind + switch token.Kind { + case lexer.BracketL: + return p.parseList(isConst) + case lexer.BraceL: + return p.parseObject(isConst) + case lexer.Dollar: + if isConst { + p.unexpectedError() + return nil + } + return &Value{Position: &token.Pos, Raw: p.parseVariable(), Kind: Variable} + case lexer.Int: + kind = IntValue + case lexer.Float: + kind = FloatValue + case lexer.String: + kind = StringValue + case lexer.BlockString: + kind = BlockValue + case lexer.Name: + switch token.Value { + case "true", "false": + kind = BooleanValue + case "null": + kind = NullValue + default: + kind = EnumValue + } + default: + p.unexpectedError() + return nil + } + + p.next() + + return &Value{Position: &token.Pos, Raw: token.Value, Kind: kind} +} + +func (p *parser) parseList(isConst bool) *Value { + var values ChildValueList + pos := p.peekPos() + p.many(lexer.BracketL, lexer.BracketR, func() { + values = append(values, &ChildValue{Value: p.parseValueLiteral(isConst)}) + }) + + return &Value{Children: values, Kind: ListValue, Position: pos} +} + +func (p *parser) parseObject(isConst bool) *Value { + var fields ChildValueList + pos := p.peekPos() + p.many(lexer.BraceL, lexer.BraceR, func() { + fields = append(fields, p.parseObjectField(isConst)) + }) + + return &Value{Children: fields, Kind: ObjectValue, Position: pos} +} + +func (p *parser) parseObjectField(isConst bool) *ChildValue { + field := ChildValue{} + field.Position = p.peekPos() + field.Name = p.parseName() + + p.expect(lexer.Colon) + + field.Value = p.parseValueLiteral(isConst) + return &field +} + +func (p *parser) parseDirectives(isConst bool) []*Directive { + var directives []*Directive + + for p.peek().Kind == lexer.At { + if p.err != nil { + break + } + directives = append(directives, p.parseDirective(isConst)) + } + return directives +} + +func (p *parser) parseDirective(isConst bool) *Directive { + p.expect(lexer.At) + + return &Directive{ + Position: p.peekPos(), + Name: p.parseName(), + Arguments: p.parseArguments(isConst), + } +} + +func (p *parser) parseTypeReference() *Type { + var typ Type + + if p.skip(lexer.BracketL) { + typ.Position = p.peekPos() + typ.Elem = p.parseTypeReference() + p.expect(lexer.BracketR) + } else { + typ.Position = p.peekPos() + typ.NamedType = p.parseName() + } + + if p.skip(lexer.Bang) { + typ.NonNull = true + } + return &typ +} + +func (p *parser) parseName() string { + token := p.expect(lexer.Name) + + return token.Value +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/gqlparser/parser/query_test.yml b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/parser/query_test.yml new file mode 100644 index 00000000..a46a01e7 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/parser/query_test.yml @@ -0,0 +1,544 @@ +parser provides useful errors: + - name: unclosed paren + input: '{' + error: + message: "Expected Name, found " + locations: [{line: 1, column: 2}] + + - name: missing on in fragment + input: | + { ...MissingOn } + fragment MissingOn Type + error: + message: 'Expected "on", found Name "Type"' + locations: [{ line: 2, column: 20 }] + + - name: missing name after alias + input: '{ field: {} }' + error: + message: "Expected Name, found {" + locations: [{ line: 1, column: 10 }] + + - name: not an operation + input: 'notanoperation Foo { field }' + error: + message: 'Unexpected Name "notanoperation"' + locations: [{ line: 1, column: 1 }] + + - name: a wild splat appears + input: '...' + error: + message: 'Unexpected ...' + locations: [{ line: 1, column: 1}] + +variables: + - name: are allowed in args + input: '{ field(complex: { a: { b: [ $var ] } }) }' + + - name: are not allowed in default args + input: 'query Foo($x: Complex = { a: { b: [ $var ] } }) { field }' + error: + message: 'Unexpected $' + locations: [{ line: 1, column: 37 }] + + - name: can have directives + input: 'query ($withDirective: String @first @second, $withoutDirective: String) { f }' + ast: | + + Operations: [OperationDefinition] + - + Operation: Operation("query") + VariableDefinitions: [VariableDefinition] + - + Variable: "withDirective" + Type: String + Directives: [Directive] + - + Name: "first" + - + Name: "second" + - + Variable: "withoutDirective" + Type: String + SelectionSet: [Selection] + - + Alias: "f" + Name: "f" + +fragments: + - name: can not be named 'on' + input: 'fragment on on on { on }' + error: + message: 'Unexpected Name "on"' + locations: [{ line: 1, column: 10 }] + + - name: can not spread fragments called 'on' + input: '{ ...on }' + error: + message: 'Expected Name, found }' + locations: [{ line: 1, column: 9 }] + +encoding: + - name: multibyte characters are supported + input: | + # This comment has a ਊ multi-byte character. + { field(arg: "Has a ਊ multi-byte character.") } + ast: | + + Operations: [OperationDefinition] + - + Operation: Operation("query") + SelectionSet: [Selection] + - + Alias: "field" + Name: "field" + Arguments: [Argument] + - + Name: "arg" + Value: "Has a ਊ multi-byte character." + +keywords are allowed anywhere a name is: + - name: on + input: | + query on { + ... a + ... on on { field } + } + fragment a on Type { + on(on: $on) + @on(on: on) + } + + - name: subscription + input: | + query subscription { + ... subscription + ... on subscription { field } + } + fragment subscription on Type { + subscription(subscription: $subscription) + @subscription(subscription: subscription) + } + + - name: true + input: | + query true { + ... true + ... on true { field } + } + fragment true on Type { + true(true: $true) + @true(true: true) + } + +operations: + - name: anonymous mutation + input: 'mutation { mutationField }' + + - name: named mutation + input: 'mutation Foo { mutationField }' + + - name: anonymous subscription + input: 'subscription { subscriptionField }' + + - name: named subscription + input: 'subscription Foo { subscriptionField }' + + +ast: + - name: simple query + input: | + { + node(id: 4) { + id, + name + } + } + ast: | + + Operations: [OperationDefinition] + - + Operation: Operation("query") + SelectionSet: [Selection] + - + Alias: "node" + Name: "node" + Arguments: [Argument] + - + Name: "id" + Value: 4 + SelectionSet: [Selection] + - + Alias: "id" + Name: "id" + - + Alias: "name" + Name: "name" + + - name: nameless query with no variables + input: | + query { + node { + id + } + } + ast: | + + Operations: [OperationDefinition] + - + Operation: Operation("query") + SelectionSet: [Selection] + - + Alias: "node" + Name: "node" + SelectionSet: [Selection] + - + Alias: "id" + Name: "id" + + - name: fragment defined variables + input: 'fragment a($v: Boolean = false) on t { f(v: $v) }' + ast: | + + Fragments: [FragmentDefinition] + - + Name: "a" + VariableDefinition: [VariableDefinition] + - + Variable: "v" + Type: Boolean + DefaultValue: false + TypeCondition: "t" + SelectionSet: [Selection] + - + Alias: "f" + Name: "f" + Arguments: [Argument] + - + Name: "v" + Value: $v + + +values: + - name: null + input: '{ f(id: null) }' + ast: | + + Operations: [OperationDefinition] + - + Operation: Operation("query") + SelectionSet: [Selection] + - + Alias: "f" + Name: "f" + Arguments: [Argument] + - + Name: "id" + Value: null + + - name: strings + input: '{ f(long: """long""", short: "short") } ' + ast: | + + Operations: [OperationDefinition] + - + Operation: Operation("query") + SelectionSet: [Selection] + - + Alias: "f" + Name: "f" + Arguments: [Argument] + - + Name: "long" + Value: "long" + - + Name: "short" + Value: "short" + + - name: list + input: '{ f(id: [1,2]) }' + ast: | + + Operations: [OperationDefinition] + - + Operation: Operation("query") + SelectionSet: [Selection] + - + Alias: "f" + Name: "f" + Arguments: [Argument] + - + Name: "id" + Value: [1,2] + +types: + - name: common types + input: 'query ($string: String, $int: Int, $arr: [Arr], $notnull: [Arr!]!) { f }' + ast: | + + Operations: [OperationDefinition] + - + Operation: Operation("query") + VariableDefinitions: [VariableDefinition] + - + Variable: "string" + Type: String + - + Variable: "int" + Type: Int + - + Variable: "arr" + Type: [Arr] + - + Variable: "notnull" + Type: [Arr!]! + SelectionSet: [Selection] + - + Alias: "f" + Name: "f" + +large queries: + - name: kitchen sink + input: | + # Copyright (c) 2015-present, Facebook, Inc. + # + # This source code is licensed under the MIT license found in the + # LICENSE file in the root directory of this source tree. + + query queryName($foo: ComplexType, $site: Site = MOBILE) { + whoever123is: node(id: [123, 456]) { + id , + ... on User @defer { + field2 { + id , + alias: field1(first:10, after:$foo,) @include(if: $foo) { + id, + ...frag + } + } + } + ... @skip(unless: $foo) { + id + } + ... { + id + } + } + } + + mutation likeStory { + like(story: 123) @defer { + story { + id + } + } + } + + subscription StoryLikeSubscription($input: StoryLikeSubscribeInput) { + storyLikeSubscribe(input: $input) { + story { + likers { + count + } + likeSentence { + text + } + } + } + } + + fragment frag on Friend { + foo(size: $size, bar: $b, obj: {key: "value", block: """ + block string uses \""" + """}) + } + + { + unnamed(truthy: true, falsey: false, nullish: null), + query + } + ast: | + + Operations: [OperationDefinition] + - + Operation: Operation("query") + Name: "queryName" + VariableDefinitions: [VariableDefinition] + - + Variable: "foo" + Type: ComplexType + - + Variable: "site" + Type: Site + DefaultValue: MOBILE + SelectionSet: [Selection] + - + Alias: "whoever123is" + Name: "node" + Arguments: [Argument] + - + Name: "id" + Value: [123,456] + SelectionSet: [Selection] + - + Alias: "id" + Name: "id" + - + TypeCondition: "User" + Directives: [Directive] + - + Name: "defer" + SelectionSet: [Selection] + - + Alias: "field2" + Name: "field2" + SelectionSet: [Selection] + - + Alias: "id" + Name: "id" + - + Alias: "alias" + Name: "field1" + Arguments: [Argument] + - + Name: "first" + Value: 10 + - + Name: "after" + Value: $foo + Directives: [Directive] + - + Name: "include" + Arguments: [Argument] + - + Name: "if" + Value: $foo + SelectionSet: [Selection] + - + Alias: "id" + Name: "id" + - + Name: "frag" + - + Directives: [Directive] + - + Name: "skip" + Arguments: [Argument] + - + Name: "unless" + Value: $foo + SelectionSet: [Selection] + - + Alias: "id" + Name: "id" + - + SelectionSet: [Selection] + - + Alias: "id" + Name: "id" + - + Operation: Operation("mutation") + Name: "likeStory" + SelectionSet: [Selection] + - + Alias: "like" + Name: "like" + Arguments: [Argument] + - + Name: "story" + Value: 123 + Directives: [Directive] + - + Name: "defer" + SelectionSet: [Selection] + - + Alias: "story" + Name: "story" + SelectionSet: [Selection] + - + Alias: "id" + Name: "id" + - + Operation: Operation("subscription") + Name: "StoryLikeSubscription" + VariableDefinitions: [VariableDefinition] + - + Variable: "input" + Type: StoryLikeSubscribeInput + SelectionSet: [Selection] + - + Alias: "storyLikeSubscribe" + Name: "storyLikeSubscribe" + Arguments: [Argument] + - + Name: "input" + Value: $input + SelectionSet: [Selection] + - + Alias: "story" + Name: "story" + SelectionSet: [Selection] + - + Alias: "likers" + Name: "likers" + SelectionSet: [Selection] + - + Alias: "count" + Name: "count" + - + Alias: "likeSentence" + Name: "likeSentence" + SelectionSet: [Selection] + - + Alias: "text" + Name: "text" + - + Operation: Operation("query") + SelectionSet: [Selection] + - + Alias: "unnamed" + Name: "unnamed" + Arguments: [Argument] + - + Name: "truthy" + Value: true + - + Name: "falsey" + Value: false + - + Name: "nullish" + Value: null + - + Alias: "query" + Name: "query" + Fragments: [FragmentDefinition] + - + Name: "frag" + TypeCondition: "Friend" + SelectionSet: [Selection] + - + Alias: "foo" + Name: "foo" + Arguments: [Argument] + - + Name: "size" + Value: $size + - + Name: "bar" + Value: $b + - + Name: "obj" + Value: {key:"value",block:"block string uses \"\"\""} + +fuzzer: +- name: 01 + input: '{__typename{...}}' + error: + message: 'Expected {, found }' + locations: [{ line: 1, column: 16 }] + +- name: 02 + input: '{...{__typename{...{}}}}' + error: + message: 'expected at least one definition, found }' + locations: [{ line: 1, column: 21 }] diff --git a/vendor/github.com/open-policy-agent/opa/internal/gqlparser/parser/schema.go b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/parser/schema.go new file mode 100644 index 00000000..32c29339 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/parser/schema.go @@ -0,0 +1,535 @@ +package parser + +import ( + //nolint:revive + . "github.com/open-policy-agent/opa/internal/gqlparser/ast" + "github.com/open-policy-agent/opa/internal/gqlparser/lexer" +) + +func ParseSchema(source *Source) (*SchemaDocument, error) { + p := parser{ + lexer: lexer.New(source), + } + ast, err := p.parseSchemaDocument(), p.err + if err != nil { + return nil, err + } + + for _, def := range ast.Definitions { + def.BuiltIn = source.BuiltIn + } + for _, def := range ast.Extensions { + def.BuiltIn = source.BuiltIn + } + + return ast, nil +} + +func ParseSchemas(inputs ...*Source) (*SchemaDocument, error) { + ast := &SchemaDocument{} + for _, input := range inputs { + inputAst, err := ParseSchema(input) + if err != nil { + return nil, err + } + ast.Merge(inputAst) + } + return ast, nil +} + +func (p *parser) parseSchemaDocument() *SchemaDocument { + var doc SchemaDocument + doc.Position = p.peekPos() + for p.peek().Kind != lexer.EOF { + if p.err != nil { + return nil + } + + var description string + if p.peek().Kind == lexer.BlockString || p.peek().Kind == lexer.String { + description = p.parseDescription() + } + + if p.peek().Kind != lexer.Name { + p.unexpectedError() + break + } + + switch p.peek().Value { + case "scalar", "type", "interface", "union", "enum", "input": + doc.Definitions = append(doc.Definitions, p.parseTypeSystemDefinition(description)) + case "schema": + doc.Schema = append(doc.Schema, p.parseSchemaDefinition(description)) + case "directive": + doc.Directives = append(doc.Directives, p.parseDirectiveDefinition(description)) + case "extend": + if description != "" { + p.unexpectedToken(p.prev) + } + p.parseTypeSystemExtension(&doc) + default: + p.unexpectedError() + return nil + } + } + + return &doc +} + +func (p *parser) parseDescription() string { + token := p.peek() + + if token.Kind != lexer.BlockString && token.Kind != lexer.String { + return "" + } + + return p.next().Value +} + +func (p *parser) parseTypeSystemDefinition(description string) *Definition { + tok := p.peek() + if tok.Kind != lexer.Name { + p.unexpectedError() + return nil + } + + switch tok.Value { + case "scalar": + return p.parseScalarTypeDefinition(description) + case "type": + return p.parseObjectTypeDefinition(description) + case "interface": + return p.parseInterfaceTypeDefinition(description) + case "union": + return p.parseUnionTypeDefinition(description) + case "enum": + return p.parseEnumTypeDefinition(description) + case "input": + return p.parseInputObjectTypeDefinition(description) + default: + p.unexpectedError() + return nil + } +} + +func (p *parser) parseSchemaDefinition(description string) *SchemaDefinition { + p.expectKeyword("schema") + + def := SchemaDefinition{Description: description} + def.Position = p.peekPos() + def.Description = description + def.Directives = p.parseDirectives(true) + + p.some(lexer.BraceL, lexer.BraceR, func() { + def.OperationTypes = append(def.OperationTypes, p.parseOperationTypeDefinition()) + }) + return &def +} + +func (p *parser) parseOperationTypeDefinition() *OperationTypeDefinition { + var op OperationTypeDefinition + op.Position = p.peekPos() + op.Operation = p.parseOperationType() + p.expect(lexer.Colon) + op.Type = p.parseName() + return &op +} + +func (p *parser) parseScalarTypeDefinition(description string) *Definition { + p.expectKeyword("scalar") + + var def Definition + def.Position = p.peekPos() + def.Kind = Scalar + def.Description = description + def.Name = p.parseName() + def.Directives = p.parseDirectives(true) + return &def +} + +func (p *parser) parseObjectTypeDefinition(description string) *Definition { + p.expectKeyword("type") + + var def Definition + def.Position = p.peekPos() + def.Kind = Object + def.Description = description + def.Name = p.parseName() + def.Interfaces = p.parseImplementsInterfaces() + def.Directives = p.parseDirectives(true) + def.Fields = p.parseFieldsDefinition() + return &def +} + +func (p *parser) parseImplementsInterfaces() []string { + var types []string + if p.peek().Value == "implements" { + p.next() + // optional leading ampersand + p.skip(lexer.Amp) + + types = append(types, p.parseName()) + for p.skip(lexer.Amp) && p.err == nil { + types = append(types, p.parseName()) + } + } + return types +} + +func (p *parser) parseFieldsDefinition() FieldList { + var defs FieldList + p.some(lexer.BraceL, lexer.BraceR, func() { + defs = append(defs, p.parseFieldDefinition()) + }) + return defs +} + +func (p *parser) parseFieldDefinition() *FieldDefinition { + var def FieldDefinition + def.Position = p.peekPos() + def.Description = p.parseDescription() + def.Name = p.parseName() + def.Arguments = p.parseArgumentDefs() + p.expect(lexer.Colon) + def.Type = p.parseTypeReference() + def.Directives = p.parseDirectives(true) + + return &def +} + +func (p *parser) parseArgumentDefs() ArgumentDefinitionList { + var args ArgumentDefinitionList + p.some(lexer.ParenL, lexer.ParenR, func() { + args = append(args, p.parseArgumentDef()) + }) + return args +} + +func (p *parser) parseArgumentDef() *ArgumentDefinition { + var def ArgumentDefinition + def.Position = p.peekPos() + def.Description = p.parseDescription() + def.Name = p.parseName() + p.expect(lexer.Colon) + def.Type = p.parseTypeReference() + if p.skip(lexer.Equals) { + def.DefaultValue = p.parseValueLiteral(true) + } + def.Directives = p.parseDirectives(true) + return &def +} + +func (p *parser) parseInputValueDef() *FieldDefinition { + var def FieldDefinition + def.Position = p.peekPos() + def.Description = p.parseDescription() + def.Name = p.parseName() + p.expect(lexer.Colon) + def.Type = p.parseTypeReference() + if p.skip(lexer.Equals) { + def.DefaultValue = p.parseValueLiteral(true) + } + def.Directives = p.parseDirectives(true) + return &def +} + +func (p *parser) parseInterfaceTypeDefinition(description string) *Definition { + p.expectKeyword("interface") + + var def Definition + def.Position = p.peekPos() + def.Kind = Interface + def.Description = description + def.Name = p.parseName() + def.Interfaces = p.parseImplementsInterfaces() + def.Directives = p.parseDirectives(true) + def.Fields = p.parseFieldsDefinition() + return &def +} + +func (p *parser) parseUnionTypeDefinition(description string) *Definition { + p.expectKeyword("union") + + var def Definition + def.Position = p.peekPos() + def.Kind = Union + def.Description = description + def.Name = p.parseName() + def.Directives = p.parseDirectives(true) + def.Types = p.parseUnionMemberTypes() + return &def +} + +func (p *parser) parseUnionMemberTypes() []string { + var types []string + if p.skip(lexer.Equals) { + // optional leading pipe + p.skip(lexer.Pipe) + + types = append(types, p.parseName()) + for p.skip(lexer.Pipe) && p.err == nil { + types = append(types, p.parseName()) + } + } + return types +} + +func (p *parser) parseEnumTypeDefinition(description string) *Definition { + p.expectKeyword("enum") + + var def Definition + def.Position = p.peekPos() + def.Kind = Enum + def.Description = description + def.Name = p.parseName() + def.Directives = p.parseDirectives(true) + def.EnumValues = p.parseEnumValuesDefinition() + return &def +} + +func (p *parser) parseEnumValuesDefinition() EnumValueList { + var values EnumValueList + p.some(lexer.BraceL, lexer.BraceR, func() { + values = append(values, p.parseEnumValueDefinition()) + }) + return values +} + +func (p *parser) parseEnumValueDefinition() *EnumValueDefinition { + return &EnumValueDefinition{ + Position: p.peekPos(), + Description: p.parseDescription(), + Name: p.parseName(), + Directives: p.parseDirectives(true), + } +} + +func (p *parser) parseInputObjectTypeDefinition(description string) *Definition { + p.expectKeyword("input") + + var def Definition + def.Position = p.peekPos() + def.Kind = InputObject + def.Description = description + def.Name = p.parseName() + def.Directives = p.parseDirectives(true) + def.Fields = p.parseInputFieldsDefinition() + return &def +} + +func (p *parser) parseInputFieldsDefinition() FieldList { + var values FieldList + p.some(lexer.BraceL, lexer.BraceR, func() { + values = append(values, p.parseInputValueDef()) + }) + return values +} + +func (p *parser) parseTypeSystemExtension(doc *SchemaDocument) { + p.expectKeyword("extend") + + switch p.peek().Value { + case "schema": + doc.SchemaExtension = append(doc.SchemaExtension, p.parseSchemaExtension()) + case "scalar": + doc.Extensions = append(doc.Extensions, p.parseScalarTypeExtension()) + case "type": + doc.Extensions = append(doc.Extensions, p.parseObjectTypeExtension()) + case "interface": + doc.Extensions = append(doc.Extensions, p.parseInterfaceTypeExtension()) + case "union": + doc.Extensions = append(doc.Extensions, p.parseUnionTypeExtension()) + case "enum": + doc.Extensions = append(doc.Extensions, p.parseEnumTypeExtension()) + case "input": + doc.Extensions = append(doc.Extensions, p.parseInputObjectTypeExtension()) + default: + p.unexpectedError() + } +} + +func (p *parser) parseSchemaExtension() *SchemaDefinition { + p.expectKeyword("schema") + + var def SchemaDefinition + def.Position = p.peekPos() + def.Directives = p.parseDirectives(true) + p.some(lexer.BraceL, lexer.BraceR, func() { + def.OperationTypes = append(def.OperationTypes, p.parseOperationTypeDefinition()) + }) + if len(def.Directives) == 0 && len(def.OperationTypes) == 0 { + p.unexpectedError() + } + return &def +} + +func (p *parser) parseScalarTypeExtension() *Definition { + p.expectKeyword("scalar") + + var def Definition + def.Position = p.peekPos() + def.Kind = Scalar + def.Name = p.parseName() + def.Directives = p.parseDirectives(true) + if len(def.Directives) == 0 { + p.unexpectedError() + } + return &def +} + +func (p *parser) parseObjectTypeExtension() *Definition { + p.expectKeyword("type") + + var def Definition + def.Position = p.peekPos() + def.Kind = Object + def.Name = p.parseName() + def.Interfaces = p.parseImplementsInterfaces() + def.Directives = p.parseDirectives(true) + def.Fields = p.parseFieldsDefinition() + if len(def.Interfaces) == 0 && len(def.Directives) == 0 && len(def.Fields) == 0 { + p.unexpectedError() + } + return &def +} + +func (p *parser) parseInterfaceTypeExtension() *Definition { + p.expectKeyword("interface") + + var def Definition + def.Position = p.peekPos() + def.Kind = Interface + def.Name = p.parseName() + def.Directives = p.parseDirectives(true) + def.Fields = p.parseFieldsDefinition() + if len(def.Directives) == 0 && len(def.Fields) == 0 { + p.unexpectedError() + } + return &def +} + +func (p *parser) parseUnionTypeExtension() *Definition { + p.expectKeyword("union") + + var def Definition + def.Position = p.peekPos() + def.Kind = Union + def.Name = p.parseName() + def.Directives = p.parseDirectives(true) + def.Types = p.parseUnionMemberTypes() + + if len(def.Directives) == 0 && len(def.Types) == 0 { + p.unexpectedError() + } + return &def +} + +func (p *parser) parseEnumTypeExtension() *Definition { + p.expectKeyword("enum") + + var def Definition + def.Position = p.peekPos() + def.Kind = Enum + def.Name = p.parseName() + def.Directives = p.parseDirectives(true) + def.EnumValues = p.parseEnumValuesDefinition() + if len(def.Directives) == 0 && len(def.EnumValues) == 0 { + p.unexpectedError() + } + return &def +} + +func (p *parser) parseInputObjectTypeExtension() *Definition { + p.expectKeyword("input") + + var def Definition + def.Position = p.peekPos() + def.Kind = InputObject + def.Name = p.parseName() + def.Directives = p.parseDirectives(false) + def.Fields = p.parseInputFieldsDefinition() + if len(def.Directives) == 0 && len(def.Fields) == 0 { + p.unexpectedError() + } + return &def +} + +func (p *parser) parseDirectiveDefinition(description string) *DirectiveDefinition { + p.expectKeyword("directive") + p.expect(lexer.At) + + var def DirectiveDefinition + def.Position = p.peekPos() + def.Description = description + def.Name = p.parseName() + def.Arguments = p.parseArgumentDefs() + + if peek := p.peek(); peek.Kind == lexer.Name && peek.Value == "repeatable" { + def.IsRepeatable = true + p.skip(lexer.Name) + } + + p.expectKeyword("on") + def.Locations = p.parseDirectiveLocations() + return &def +} + +func (p *parser) parseDirectiveLocations() []DirectiveLocation { + p.skip(lexer.Pipe) + + locations := []DirectiveLocation{p.parseDirectiveLocation()} + + for p.skip(lexer.Pipe) && p.err == nil { + locations = append(locations, p.parseDirectiveLocation()) + } + + return locations +} + +func (p *parser) parseDirectiveLocation() DirectiveLocation { + name := p.expect(lexer.Name) + + switch name.Value { + case `QUERY`: + return LocationQuery + case `MUTATION`: + return LocationMutation + case `SUBSCRIPTION`: + return LocationSubscription + case `FIELD`: + return LocationField + case `FRAGMENT_DEFINITION`: + return LocationFragmentDefinition + case `FRAGMENT_SPREAD`: + return LocationFragmentSpread + case `INLINE_FRAGMENT`: + return LocationInlineFragment + case `VARIABLE_DEFINITION`: + return LocationVariableDefinition + case `SCHEMA`: + return LocationSchema + case `SCALAR`: + return LocationScalar + case `OBJECT`: + return LocationObject + case `FIELD_DEFINITION`: + return LocationFieldDefinition + case `ARGUMENT_DEFINITION`: + return LocationArgumentDefinition + case `INTERFACE`: + return LocationInterface + case `UNION`: + return LocationUnion + case `ENUM`: + return LocationEnum + case `ENUM_VALUE`: + return LocationEnumValue + case `INPUT_OBJECT`: + return LocationInputObject + case `INPUT_FIELD_DEFINITION`: + return LocationInputFieldDefinition + } + + p.unexpectedToken(name) + return "" +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/gqlparser/parser/schema_test.yml b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/parser/schema_test.yml new file mode 100644 index 00000000..8b6a5d0c --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/parser/schema_test.yml @@ -0,0 +1,646 @@ +object types: + - name: simple + input: | + type Hello { + world: String + } + ast: | + + Definitions: [Definition] + - + Kind: DefinitionKind("OBJECT") + Name: "Hello" + Fields: [FieldDefinition] + - + Name: "world" + Type: String + + - name: with description + input: | + "Description" + type Hello { + world: String + } + ast: | + + Definitions: [Definition] + - + Kind: DefinitionKind("OBJECT") + Description: "Description" + Name: "Hello" + Fields: [FieldDefinition] + - + Name: "world" + Type: String + + - name: with block description + input: | + """ + Description + """ + # Even with comments between them + type Hello { + world: String + } + ast: | + + Definitions: [Definition] + - + Kind: DefinitionKind("OBJECT") + Description: "Description" + Name: "Hello" + Fields: [FieldDefinition] + - + Name: "world" + Type: String + - name: with field arg + input: | + type Hello { + world(flag: Boolean): String + } + ast: | + + Definitions: [Definition] + - + Kind: DefinitionKind("OBJECT") + Name: "Hello" + Fields: [FieldDefinition] + - + Name: "world" + Arguments: [ArgumentDefinition] + - + Name: "flag" + Type: Boolean + Type: String + + - name: with field arg and default value + input: | + type Hello { + world(flag: Boolean = true): String + } + ast: | + + Definitions: [Definition] + - + Kind: DefinitionKind("OBJECT") + Name: "Hello" + Fields: [FieldDefinition] + - + Name: "world" + Arguments: [ArgumentDefinition] + - + Name: "flag" + DefaultValue: true + Type: Boolean + Type: String + + - name: with field list arg + input: | + type Hello { + world(things: [String]): String + } + ast: | + + Definitions: [Definition] + - + Kind: DefinitionKind("OBJECT") + Name: "Hello" + Fields: [FieldDefinition] + - + Name: "world" + Arguments: [ArgumentDefinition] + - + Name: "things" + Type: [String] + Type: String + + - name: with two args + input: | + type Hello { + world(argOne: Boolean, argTwo: Int): String + } + ast: | + + Definitions: [Definition] + - + Kind: DefinitionKind("OBJECT") + Name: "Hello" + Fields: [FieldDefinition] + - + Name: "world" + Arguments: [ArgumentDefinition] + - + Name: "argOne" + Type: Boolean + - + Name: "argTwo" + Type: Int + Type: String + - name: must define one or more fields + input: | + type Hello {} + error: + message: "expected at least one definition, found }" + locations: [{ line: 1, column: 13 }] + +type extensions: + - name: Object extension + input: | + extend type Hello { + world: String + } + ast: | + + Extensions: [Definition] + - + Kind: DefinitionKind("OBJECT") + Name: "Hello" + Fields: [FieldDefinition] + - + Name: "world" + Type: String + + - name: without any fields + input: "extend type Hello implements Greeting" + ast: | + + Extensions: [Definition] + - + Kind: DefinitionKind("OBJECT") + Name: "Hello" + Interfaces: [string] + - "Greeting" + + - name: without fields twice + input: | + extend type Hello implements Greeting + extend type Hello implements SecondGreeting + ast: | + + Extensions: [Definition] + - + Kind: DefinitionKind("OBJECT") + Name: "Hello" + Interfaces: [string] + - "Greeting" + - + Kind: DefinitionKind("OBJECT") + Name: "Hello" + Interfaces: [string] + - "SecondGreeting" + + - name: without anything errors + input: "extend type Hello" + error: + message: "Unexpected " + locations: [{ line: 1, column: 18 }] + + - name: can have descriptions # hmm, this might not be spec compliant... + input: | + "Description" + extend type Hello { + world: String + } + error: + message: 'Unexpected String "Description"' + locations: [{ line: 1, column: 2 }] + + - name: can not have descriptions on types + input: | + extend "Description" type Hello { + world: String + } + error: + message: Unexpected String "Description" + locations: [{ line: 1, column: 9 }] + + - name: all can have directives + input: | + extend scalar Foo @deprecated + extend type Foo @deprecated + extend interface Foo @deprecated + extend union Foo @deprecated + extend enum Foo @deprecated + extend input Foo @deprecated + ast: | + + Extensions: [Definition] + - + Kind: DefinitionKind("SCALAR") + Name: "Foo" + Directives: [Directive] + - + Name: "deprecated" + - + Kind: DefinitionKind("OBJECT") + Name: "Foo" + Directives: [Directive] + - + Name: "deprecated" + - + Kind: DefinitionKind("INTERFACE") + Name: "Foo" + Directives: [Directive] + - + Name: "deprecated" + - + Kind: DefinitionKind("UNION") + Name: "Foo" + Directives: [Directive] + - + Name: "deprecated" + - + Kind: DefinitionKind("ENUM") + Name: "Foo" + Directives: [Directive] + - + Name: "deprecated" + - + Kind: DefinitionKind("INPUT_OBJECT") + Name: "Foo" + Directives: [Directive] + - + Name: "deprecated" + +schema definition: + - name: simple + input: | + schema { + query: Query + } + ast: | + + Schema: [SchemaDefinition] + - + OperationTypes: [OperationTypeDefinition] + - + Operation: Operation("query") + Type: "Query" + +schema extensions: + - name: simple + input: | + extend schema { + mutation: Mutation + } + ast: | + + SchemaExtension: [SchemaDefinition] + - + OperationTypes: [OperationTypeDefinition] + - + Operation: Operation("mutation") + Type: "Mutation" + + - name: directive only + input: "extend schema @directive" + ast: | + + SchemaExtension: [SchemaDefinition] + - + Directives: [Directive] + - + Name: "directive" + + - name: without anything errors + input: "extend schema" + error: + message: "Unexpected " + locations: [{ line: 1, column: 14}] + +inheritance: + - name: single + input: "type Hello implements World { field: String }" + ast: | + + Definitions: [Definition] + - + Kind: DefinitionKind("OBJECT") + Name: "Hello" + Interfaces: [string] + - "World" + Fields: [FieldDefinition] + - + Name: "field" + Type: String + + - name: multi + input: "type Hello implements Wo & rld { field: String }" + ast: | + + Definitions: [Definition] + - + Kind: DefinitionKind("OBJECT") + Name: "Hello" + Interfaces: [string] + - "Wo" + - "rld" + Fields: [FieldDefinition] + - + Name: "field" + Type: String + + - name: multi with leading amp + input: "type Hello implements & Wo & rld { field: String }" + ast: | + + Definitions: [Definition] + - + Kind: DefinitionKind("OBJECT") + Name: "Hello" + Interfaces: [string] + - "Wo" + - "rld" + Fields: [FieldDefinition] + - + Name: "field" + Type: String + +enums: + - name: single value + input: "enum Hello { WORLD }" + ast: | + + Definitions: [Definition] + - + Kind: DefinitionKind("ENUM") + Name: "Hello" + EnumValues: [EnumValueDefinition] + - + Name: "WORLD" + + - name: double value + input: "enum Hello { WO, RLD }" + ast: | + + Definitions: [Definition] + - + Kind: DefinitionKind("ENUM") + Name: "Hello" + EnumValues: [EnumValueDefinition] + - + Name: "WO" + - + Name: "RLD" + - name: must define one or more unique enum values + input: | + enum Hello {} + error: + message: "expected at least one definition, found }" + locations: [{ line: 1, column: 13 }] + +interface: + - name: simple + input: | + interface Hello { + world: String + } + ast: | + + Definitions: [Definition] + - + Kind: DefinitionKind("INTERFACE") + Name: "Hello" + Fields: [FieldDefinition] + - + Name: "world" + Type: String + - name: must define one or more fields + input: | + interface Hello {} + error: + message: "expected at least one definition, found }" + locations: [{ line: 1, column: 18 }] + + - name: may define intermediate interfaces + input: | + interface IA { + id: ID! + } + + interface IIA implements IA { + id: ID! + } + + type A implements IIA { + id: ID! + } + ast: | + + Definitions: [Definition] + - + Kind: DefinitionKind("INTERFACE") + Name: "IA" + Fields: [FieldDefinition] + - + Name: "id" + Type: ID! + - + Kind: DefinitionKind("INTERFACE") + Name: "IIA" + Interfaces: [string] + - "IA" + Fields: [FieldDefinition] + - + Name: "id" + Type: ID! + - + Kind: DefinitionKind("OBJECT") + Name: "A" + Interfaces: [string] + - "IIA" + Fields: [FieldDefinition] + - + Name: "id" + Type: ID! + +unions: + - name: simple + input: "union Hello = World" + ast: | + + Definitions: [Definition] + - + Kind: DefinitionKind("UNION") + Name: "Hello" + Types: [string] + - "World" + + - name: with two types + input: "union Hello = Wo | Rld" + ast: | + + Definitions: [Definition] + - + Kind: DefinitionKind("UNION") + Name: "Hello" + Types: [string] + - "Wo" + - "Rld" + + - name: with leading pipe + input: "union Hello = | Wo | Rld" + ast: | + + Definitions: [Definition] + - + Kind: DefinitionKind("UNION") + Name: "Hello" + Types: [string] + - "Wo" + - "Rld" + + - name: cant be empty + input: "union Hello = || Wo | Rld" + error: + message: "Expected Name, found |" + locations: [{ line: 1, column: 16 }] + + - name: cant double pipe + input: "union Hello = Wo || Rld" + error: + message: "Expected Name, found |" + locations: [{ line: 1, column: 19 }] + + - name: cant have trailing pipe + input: "union Hello = | Wo | Rld |" + error: + message: "Expected Name, found " + locations: [{ line: 1, column: 27 }] + +scalar: + - name: simple + input: "scalar Hello" + ast: | + + Definitions: [Definition] + - + Kind: DefinitionKind("SCALAR") + Name: "Hello" + +input object: + - name: simple + input: | + input Hello { + world: String + } + ast: | + + Definitions: [Definition] + - + Kind: DefinitionKind("INPUT_OBJECT") + Name: "Hello" + Fields: [FieldDefinition] + - + Name: "world" + Type: String + + - name: can not have args + input: | + input Hello { + world(foo: Int): String + } + error: + message: "Expected :, found (" + locations: [{ line: 2, column: 8 }] + - name: must define one or more input fields + input: | + input Hello {} + error: + message: "expected at least one definition, found }" + locations: [{ line: 1, column: 14 }] + +directives: + - name: simple + input: directive @foo on FIELD + ast: | + + Directives: [DirectiveDefinition] + - + Name: "foo" + Locations: [DirectiveLocation] + - DirectiveLocation("FIELD") + IsRepeatable: false + + - name: executable + input: | + directive @onQuery on QUERY + directive @onMutation on MUTATION + directive @onSubscription on SUBSCRIPTION + directive @onField on FIELD + directive @onFragmentDefinition on FRAGMENT_DEFINITION + directive @onFragmentSpread on FRAGMENT_SPREAD + directive @onInlineFragment on INLINE_FRAGMENT + directive @onVariableDefinition on VARIABLE_DEFINITION + ast: | + + Directives: [DirectiveDefinition] + - + Name: "onQuery" + Locations: [DirectiveLocation] + - DirectiveLocation("QUERY") + IsRepeatable: false + - + Name: "onMutation" + Locations: [DirectiveLocation] + - DirectiveLocation("MUTATION") + IsRepeatable: false + - + Name: "onSubscription" + Locations: [DirectiveLocation] + - DirectiveLocation("SUBSCRIPTION") + IsRepeatable: false + - + Name: "onField" + Locations: [DirectiveLocation] + - DirectiveLocation("FIELD") + IsRepeatable: false + - + Name: "onFragmentDefinition" + Locations: [DirectiveLocation] + - DirectiveLocation("FRAGMENT_DEFINITION") + IsRepeatable: false + - + Name: "onFragmentSpread" + Locations: [DirectiveLocation] + - DirectiveLocation("FRAGMENT_SPREAD") + IsRepeatable: false + - + Name: "onInlineFragment" + Locations: [DirectiveLocation] + - DirectiveLocation("INLINE_FRAGMENT") + IsRepeatable: false + - + Name: "onVariableDefinition" + Locations: [DirectiveLocation] + - DirectiveLocation("VARIABLE_DEFINITION") + IsRepeatable: false + + - name: repeatable + input: directive @foo repeatable on FIELD + ast: | + + Directives: [DirectiveDefinition] + - + Name: "foo" + Locations: [DirectiveLocation] + - DirectiveLocation("FIELD") + IsRepeatable: true + + - name: invalid location + input: "directive @foo on FIELD | INCORRECT_LOCATION" + error: + message: 'Unexpected Name "INCORRECT_LOCATION"' + locations: [{ line: 1, column: 27 }] + +fuzzer: + - name: 1 + input: "type o{d(g:[" + error: + message: 'Expected Name, found ' + locations: [{ line: 1, column: 13 }] + - name: 2 + input: "\"\"\"\r" + error: + message: 'Unexpected ' + locations: [{ line: 2, column: 1 }] diff --git a/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/error.go b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/error.go new file mode 100644 index 00000000..f31f180a --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/error.go @@ -0,0 +1,55 @@ +package validator + +import ( + "fmt" + + "github.com/open-policy-agent/opa/internal/gqlparser/ast" + "github.com/open-policy-agent/opa/internal/gqlparser/gqlerror" +) + +type ErrorOption func(err *gqlerror.Error) + +func Message(msg string, args ...interface{}) ErrorOption { + return func(err *gqlerror.Error) { + err.Message += fmt.Sprintf(msg, args...) + } +} + +func At(position *ast.Position) ErrorOption { + return func(err *gqlerror.Error) { + if position == nil { + return + } + err.Locations = append(err.Locations, gqlerror.Location{ + Line: position.Line, + Column: position.Column, + }) + if position.Src.Name != "" { + err.SetFile(position.Src.Name) + } + } +} + +func SuggestListQuoted(prefix string, typed string, suggestions []string) ErrorOption { + suggested := SuggestionList(typed, suggestions) + return func(err *gqlerror.Error) { + if len(suggested) > 0 { + err.Message += " " + prefix + " " + QuotedOrList(suggested...) + "?" + } + } +} + +func SuggestListUnquoted(prefix string, typed string, suggestions []string) ErrorOption { + suggested := SuggestionList(typed, suggestions) + return func(err *gqlerror.Error) { + if len(suggested) > 0 { + err.Message += " " + prefix + " " + OrList(suggested...) + "?" + } + } +} + +func Suggestf(suggestion string, args ...interface{}) ErrorOption { + return func(err *gqlerror.Error) { + err.Message += " Did you mean " + fmt.Sprintf(suggestion, args...) + "?" + } +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/messaging.go b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/messaging.go new file mode 100644 index 00000000..f1ab5873 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/messaging.go @@ -0,0 +1,39 @@ +package validator + +import "bytes" + +// Given [ A, B, C ] return '"A", "B", or "C"'. +func QuotedOrList(items ...string) string { + itemsQuoted := make([]string, len(items)) + for i, item := range items { + itemsQuoted[i] = `"` + item + `"` + } + return OrList(itemsQuoted...) +} + +// Given [ A, B, C ] return 'A, B, or C'. +func OrList(items ...string) string { + var buf bytes.Buffer + + if len(items) > 5 { + items = items[:5] + } + if len(items) == 2 { + buf.WriteString(items[0]) + buf.WriteString(" or ") + buf.WriteString(items[1]) + return buf.String() + } + + for i, item := range items { + if i != 0 { + if i == len(items)-1 { + buf.WriteString(", or ") + } else { + buf.WriteString(", ") + } + } + buf.WriteString(item) + } + return buf.String() +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/prelude.go b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/prelude.go new file mode 100644 index 00000000..86796fab --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/prelude.go @@ -0,0 +1,16 @@ +package validator + +import ( + _ "embed" + + "github.com/open-policy-agent/opa/internal/gqlparser/ast" +) + +//go:embed prelude.graphql +var preludeGraphql string + +var Prelude = &ast.Source{ + Name: "prelude.graphql", + Input: preludeGraphql, + BuiltIn: true, +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/prelude.graphql b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/prelude.graphql new file mode 100644 index 00000000..bdca0096 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/prelude.graphql @@ -0,0 +1,121 @@ +# This file defines all the implicitly declared types that are required by the graphql spec. It is implicitly included by calls to LoadSchema + +"The `Int` scalar type represents non-fractional signed whole numeric values. Int can represent values between -(2^31) and 2^31 - 1." +scalar Int + +"The `Float` scalar type represents signed double-precision fractional values as specified by [IEEE 754](http://en.wikipedia.org/wiki/IEEE_floating_point)." +scalar Float + +"The `String`scalar type represents textual data, represented as UTF-8 character sequences. The String type is most often used by GraphQL to represent free-form human-readable text." +scalar String + +"The `Boolean` scalar type represents `true` or `false`." +scalar Boolean + +"""The `ID` scalar type represents a unique identifier, often used to refetch an object or as key for a cache. The ID type appears in a JSON response as a String; however, it is not intended to be human-readable. When expected as an input type, any string (such as "4") or integer (such as 4) input value will be accepted as an ID.""" +scalar ID + +"The @include directive may be provided for fields, fragment spreads, and inline fragments, and allows for conditional inclusion during execution as described by the if argument." +directive @include(if: Boolean!) on FIELD | FRAGMENT_SPREAD | INLINE_FRAGMENT + +"The @skip directive may be provided for fields, fragment spreads, and inline fragments, and allows for conditional exclusion during execution as described by the if argument." +directive @skip(if: Boolean!) on FIELD | FRAGMENT_SPREAD | INLINE_FRAGMENT + +"The @deprecated built-in directive is used within the type system definition language to indicate deprecated portions of a GraphQL service's schema, such as deprecated fields on a type, arguments on a field, input fields on an input type, or values of an enum type." +directive @deprecated(reason: String = "No longer supported") on FIELD_DEFINITION | ARGUMENT_DEFINITION | INPUT_FIELD_DEFINITION | ENUM_VALUE + +"The @specifiedBy built-in directive is used within the type system definition language to provide a scalar specification URL for specifying the behavior of custom scalar types." +directive @specifiedBy(url: String!) on SCALAR + +type __Schema { + description: String + types: [__Type!]! + queryType: __Type! + mutationType: __Type + subscriptionType: __Type + directives: [__Directive!]! +} + +type __Type { + kind: __TypeKind! + name: String + description: String + # must be non-null for OBJECT and INTERFACE, otherwise null. + fields(includeDeprecated: Boolean = false): [__Field!] + # must be non-null for OBJECT and INTERFACE, otherwise null. + interfaces: [__Type!] + # must be non-null for INTERFACE and UNION, otherwise null. + possibleTypes: [__Type!] + # must be non-null for ENUM, otherwise null. + enumValues(includeDeprecated: Boolean = false): [__EnumValue!] + # must be non-null for INPUT_OBJECT, otherwise null. + inputFields: [__InputValue!] + # must be non-null for NON_NULL and LIST, otherwise null. + ofType: __Type + # may be non-null for custom SCALAR, otherwise null. + specifiedByURL: String +} + +type __Field { + name: String! + description: String + args: [__InputValue!]! + type: __Type! + isDeprecated: Boolean! + deprecationReason: String +} + +type __InputValue { + name: String! + description: String + type: __Type! + defaultValue: String +} + +type __EnumValue { + name: String! + description: String + isDeprecated: Boolean! + deprecationReason: String +} + +enum __TypeKind { + SCALAR + OBJECT + INTERFACE + UNION + ENUM + INPUT_OBJECT + LIST + NON_NULL +} + +type __Directive { + name: String! + description: String + locations: [__DirectiveLocation!]! + args: [__InputValue!]! + isRepeatable: Boolean! +} + +enum __DirectiveLocation { + QUERY + MUTATION + SUBSCRIPTION + FIELD + FRAGMENT_DEFINITION + FRAGMENT_SPREAD + INLINE_FRAGMENT + VARIABLE_DEFINITION + SCHEMA + SCALAR + OBJECT + FIELD_DEFINITION + ARGUMENT_DEFINITION + INTERFACE + UNION + ENUM + ENUM_VALUE + INPUT_OBJECT + INPUT_FIELD_DEFINITION +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/fields_on_correct_type.go b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/fields_on_correct_type.go new file mode 100644 index 00000000..d536e5e5 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/fields_on_correct_type.go @@ -0,0 +1,97 @@ +package validator + +import ( + "fmt" + "sort" + "strings" + + "github.com/open-policy-agent/opa/internal/gqlparser/ast" + + //nolint:revive // Validator rules each use dot imports for convenience. + . "github.com/open-policy-agent/opa/internal/gqlparser/validator" +) + +func init() { + AddRule("FieldsOnCorrectType", func(observers *Events, addError AddErrFunc) { + observers.OnField(func(walker *Walker, field *ast.Field) { + if field.ObjectDefinition == nil || field.Definition != nil { + return + } + + message := fmt.Sprintf(`Cannot query field "%s" on type "%s".`, field.Name, field.ObjectDefinition.Name) + + if suggestedTypeNames := getSuggestedTypeNames(walker, field.ObjectDefinition, field.Name); suggestedTypeNames != nil { + message += " Did you mean to use an inline fragment on " + QuotedOrList(suggestedTypeNames...) + "?" + } else if suggestedFieldNames := getSuggestedFieldNames(field.ObjectDefinition, field.Name); suggestedFieldNames != nil { + message += " Did you mean " + QuotedOrList(suggestedFieldNames...) + "?" + } + + addError( + Message(message), + At(field.Position), + ) + }) + }) +} + +// Go through all of the implementations of type, as well as the interfaces +// that they implement. If any of those types include the provided field, +// suggest them, sorted by how often the type is referenced, starting +// with Interfaces. +func getSuggestedTypeNames(walker *Walker, parent *ast.Definition, name string) []string { + if !parent.IsAbstractType() { + return nil + } + + possibleTypes := walker.Schema.GetPossibleTypes(parent) + var suggestedObjectTypes = make([]string, 0, len(possibleTypes)) + var suggestedInterfaceTypes []string + interfaceUsageCount := map[string]int{} + + for _, possibleType := range possibleTypes { + field := possibleType.Fields.ForName(name) + if field == nil { + continue + } + + suggestedObjectTypes = append(suggestedObjectTypes, possibleType.Name) + + for _, possibleInterface := range possibleType.Interfaces { + interfaceField := walker.Schema.Types[possibleInterface] + if interfaceField != nil && interfaceField.Fields.ForName(name) != nil { + if interfaceUsageCount[possibleInterface] == 0 { + suggestedInterfaceTypes = append(suggestedInterfaceTypes, possibleInterface) + } + interfaceUsageCount[possibleInterface]++ + } + } + } + + suggestedTypes := append(suggestedInterfaceTypes, suggestedObjectTypes...) + + sort.SliceStable(suggestedTypes, func(i, j int) bool { + typeA, typeB := suggestedTypes[i], suggestedTypes[j] + diff := interfaceUsageCount[typeB] - interfaceUsageCount[typeA] + if diff != 0 { + return diff < 0 + } + return strings.Compare(typeA, typeB) < 0 + }) + + return suggestedTypes +} + +// For the field name provided, determine if there are any similar field names +// that may be the result of a typo. +func getSuggestedFieldNames(parent *ast.Definition, name string) []string { + if parent.Kind != ast.Object && parent.Kind != ast.Interface { + return nil + } + + var possibleFieldNames = make([]string, 0, len(parent.Fields)) + for _, field := range parent.Fields { + possibleFieldNames = append(possibleFieldNames, field.Name) + } + + return SuggestionList(name, possibleFieldNames) +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/fragments_on_composite_types.go b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/fragments_on_composite_types.go new file mode 100644 index 00000000..1af9c4f9 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/fragments_on_composite_types.go @@ -0,0 +1,41 @@ +package validator + +import ( + "fmt" + + "github.com/open-policy-agent/opa/internal/gqlparser/ast" + + //nolint:revive // Validator rules each use dot imports for convenience. + . "github.com/open-policy-agent/opa/internal/gqlparser/validator" +) + +func init() { + AddRule("FragmentsOnCompositeTypes", func(observers *Events, addError AddErrFunc) { + observers.OnInlineFragment(func(walker *Walker, inlineFragment *ast.InlineFragment) { + fragmentType := walker.Schema.Types[inlineFragment.TypeCondition] + if fragmentType == nil || fragmentType.IsCompositeType() { + return + } + + message := fmt.Sprintf(`Fragment cannot condition on non composite type "%s".`, inlineFragment.TypeCondition) + + addError( + Message(message), + At(inlineFragment.Position), + ) + }) + + observers.OnFragment(func(walker *Walker, fragment *ast.FragmentDefinition) { + if fragment.Definition == nil || fragment.TypeCondition == "" || fragment.Definition.IsCompositeType() { + return + } + + message := fmt.Sprintf(`Fragment "%s" cannot condition on non composite type "%s".`, fragment.Name, fragment.TypeCondition) + + addError( + Message(message), + At(fragment.Position), + ) + }) + }) +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/known_argument_names.go b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/known_argument_names.go new file mode 100644 index 00000000..d9aa8873 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/known_argument_names.go @@ -0,0 +1,59 @@ +package validator + +import ( + "github.com/open-policy-agent/opa/internal/gqlparser/ast" + + //nolint:revive // Validator rules each use dot imports for convenience. + . "github.com/open-policy-agent/opa/internal/gqlparser/validator" +) + +func init() { + AddRule("KnownArgumentNames", func(observers *Events, addError AddErrFunc) { + // A GraphQL field is only valid if all supplied arguments are defined by that field. + observers.OnField(func(walker *Walker, field *ast.Field) { + if field.Definition == nil || field.ObjectDefinition == nil { + return + } + for _, arg := range field.Arguments { + def := field.Definition.Arguments.ForName(arg.Name) + if def != nil { + continue + } + + var suggestions []string + for _, argDef := range field.Definition.Arguments { + suggestions = append(suggestions, argDef.Name) + } + + addError( + Message(`Unknown argument "%s" on field "%s.%s".`, arg.Name, field.ObjectDefinition.Name, field.Name), + SuggestListQuoted("Did you mean", arg.Name, suggestions), + At(field.Position), + ) + } + }) + + observers.OnDirective(func(walker *Walker, directive *ast.Directive) { + if directive.Definition == nil { + return + } + for _, arg := range directive.Arguments { + def := directive.Definition.Arguments.ForName(arg.Name) + if def != nil { + continue + } + + var suggestions []string + for _, argDef := range directive.Definition.Arguments { + suggestions = append(suggestions, argDef.Name) + } + + addError( + Message(`Unknown argument "%s" on directive "@%s".`, arg.Name, directive.Name), + SuggestListQuoted("Did you mean", arg.Name, suggestions), + At(directive.Position), + ) + } + }) + }) +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/known_directives.go b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/known_directives.go new file mode 100644 index 00000000..016541e8 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/known_directives.go @@ -0,0 +1,49 @@ +package validator + +import ( + "github.com/open-policy-agent/opa/internal/gqlparser/ast" + + //nolint:revive // Validator rules each use dot imports for convenience. + . "github.com/open-policy-agent/opa/internal/gqlparser/validator" +) + +func init() { + AddRule("KnownDirectives", func(observers *Events, addError AddErrFunc) { + type mayNotBeUsedDirective struct { + Name string + Line int + Column int + } + var seen = map[mayNotBeUsedDirective]bool{} + observers.OnDirective(func(walker *Walker, directive *ast.Directive) { + if directive.Definition == nil { + addError( + Message(`Unknown directive "@%s".`, directive.Name), + At(directive.Position), + ) + return + } + + for _, loc := range directive.Definition.Locations { + if loc == directive.Location { + return + } + } + + // position must be exists if directive.Definition != nil + tmp := mayNotBeUsedDirective{ + Name: directive.Name, + Line: directive.Position.Line, + Column: directive.Position.Column, + } + + if !seen[tmp] { + addError( + Message(`Directive "@%s" may not be used on %s.`, directive.Name, directive.Location), + At(directive.Position), + ) + seen[tmp] = true + } + }) + }) +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/known_fragment_names.go b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/known_fragment_names.go new file mode 100644 index 00000000..94583dac --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/known_fragment_names.go @@ -0,0 +1,21 @@ +package validator + +import ( + "github.com/open-policy-agent/opa/internal/gqlparser/ast" + + //nolint:revive // Validator rules each use dot imports for convenience. + . "github.com/open-policy-agent/opa/internal/gqlparser/validator" +) + +func init() { + AddRule("KnownFragmentNames", func(observers *Events, addError AddErrFunc) { + observers.OnFragmentSpread(func(walker *Walker, fragmentSpread *ast.FragmentSpread) { + if fragmentSpread.Definition == nil { + addError( + Message(`Unknown fragment "%s".`, fragmentSpread.Name), + At(fragmentSpread.Position), + ) + } + }) + }) +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/known_root_type.go b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/known_root_type.go new file mode 100644 index 00000000..ab97cd90 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/known_root_type.go @@ -0,0 +1,37 @@ +package validator + +import ( + "fmt" + + "github.com/open-policy-agent/opa/internal/gqlparser/ast" + + //nolint:revive // Validator rules each use dot imports for convenience. + . "github.com/open-policy-agent/opa/internal/gqlparser/validator" +) + +func init() { + AddRule("KnownRootType", func(observers *Events, addError AddErrFunc) { + // A query's root must be a valid type. Surprisingly, this isn't + // checked anywhere else! + observers.OnOperation(func(walker *Walker, operation *ast.OperationDefinition) { + var def *ast.Definition + switch operation.Operation { + case ast.Query, "": + def = walker.Schema.Query + case ast.Mutation: + def = walker.Schema.Mutation + case ast.Subscription: + def = walker.Schema.Subscription + default: + // This shouldn't even parse; if it did we probably need to + // update this switch block to add the new operation type. + panic(fmt.Sprintf(`got unknown operation type "%s"`, operation.Operation)) + } + if def == nil { + addError( + Message(`Schema does not support operation type "%s"`, operation.Operation), + At(operation.Position)) + } + }) + }) +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/known_type_names.go b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/known_type_names.go new file mode 100644 index 00000000..aa9809be --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/known_type_names.go @@ -0,0 +1,61 @@ +package validator + +import ( + "github.com/open-policy-agent/opa/internal/gqlparser/ast" + + //nolint:revive // Validator rules each use dot imports for convenience. + . "github.com/open-policy-agent/opa/internal/gqlparser/validator" +) + +func init() { + AddRule("KnownTypeNames", func(observers *Events, addError AddErrFunc) { + observers.OnVariable(func(walker *Walker, variable *ast.VariableDefinition) { + typeName := variable.Type.Name() + typdef := walker.Schema.Types[typeName] + if typdef != nil { + return + } + + addError( + Message(`Unknown type "%s".`, typeName), + At(variable.Position), + ) + }) + + observers.OnInlineFragment(func(walker *Walker, inlineFragment *ast.InlineFragment) { + typedName := inlineFragment.TypeCondition + if typedName == "" { + return + } + + def := walker.Schema.Types[typedName] + if def != nil { + return + } + + addError( + Message(`Unknown type "%s".`, typedName), + At(inlineFragment.Position), + ) + }) + + observers.OnFragment(func(walker *Walker, fragment *ast.FragmentDefinition) { + typeName := fragment.TypeCondition + def := walker.Schema.Types[typeName] + if def != nil { + return + } + + var possibleTypes []string + for _, t := range walker.Schema.Types { + possibleTypes = append(possibleTypes, t.Name) + } + + addError( + Message(`Unknown type "%s".`, typeName), + SuggestListQuoted("Did you mean", typeName, possibleTypes), + At(fragment.Position), + ) + }) + }) +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/lone_anonymous_operation.go b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/lone_anonymous_operation.go new file mode 100644 index 00000000..2af7b5a0 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/lone_anonymous_operation.go @@ -0,0 +1,21 @@ +package validator + +import ( + "github.com/open-policy-agent/opa/internal/gqlparser/ast" + + //nolint:revive // Validator rules each use dot imports for convenience. + . "github.com/open-policy-agent/opa/internal/gqlparser/validator" +) + +func init() { + AddRule("LoneAnonymousOperation", func(observers *Events, addError AddErrFunc) { + observers.OnOperation(func(walker *Walker, operation *ast.OperationDefinition) { + if operation.Name == "" && len(walker.Document.Operations) > 1 { + addError( + Message(`This anonymous operation must be the only defined operation.`), + At(operation.Position), + ) + } + }) + }) +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/no_fragment_cycles.go b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/no_fragment_cycles.go new file mode 100644 index 00000000..a7de611f --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/no_fragment_cycles.go @@ -0,0 +1,95 @@ +package validator + +import ( + "fmt" + "strings" + + "github.com/open-policy-agent/opa/internal/gqlparser/ast" + + //nolint:revive // Validator rules each use dot imports for convenience. + . "github.com/open-policy-agent/opa/internal/gqlparser/validator" +) + +func init() { + AddRule("NoFragmentCycles", func(observers *Events, addError AddErrFunc) { + visitedFrags := make(map[string]bool) + + observers.OnFragment(func(walker *Walker, fragment *ast.FragmentDefinition) { + var spreadPath []*ast.FragmentSpread + spreadPathIndexByName := make(map[string]int) + + var recursive func(fragment *ast.FragmentDefinition) + recursive = func(fragment *ast.FragmentDefinition) { + if visitedFrags[fragment.Name] { + return + } + + visitedFrags[fragment.Name] = true + + spreadNodes := getFragmentSpreads(fragment.SelectionSet) + if len(spreadNodes) == 0 { + return + } + spreadPathIndexByName[fragment.Name] = len(spreadPath) + + for _, spreadNode := range spreadNodes { + spreadName := spreadNode.Name + + cycleIndex, ok := spreadPathIndexByName[spreadName] + + spreadPath = append(spreadPath, spreadNode) + if !ok { + spreadFragment := walker.Document.Fragments.ForName(spreadName) + if spreadFragment != nil { + recursive(spreadFragment) + } + } else { + cyclePath := spreadPath[cycleIndex : len(spreadPath)-1] + var fragmentNames []string + for _, fs := range cyclePath { + fragmentNames = append(fragmentNames, fmt.Sprintf(`"%s"`, fs.Name)) + } + var via string + if len(fragmentNames) != 0 { + via = fmt.Sprintf(" via %s", strings.Join(fragmentNames, ", ")) + } + addError( + Message(`Cannot spread fragment "%s" within itself%s.`, spreadName, via), + At(spreadNode.Position), + ) + } + + spreadPath = spreadPath[:len(spreadPath)-1] + } + + delete(spreadPathIndexByName, fragment.Name) + } + + recursive(fragment) + }) + }) +} + +func getFragmentSpreads(node ast.SelectionSet) []*ast.FragmentSpread { + var spreads []*ast.FragmentSpread + + setsToVisit := []ast.SelectionSet{node} + + for len(setsToVisit) != 0 { + set := setsToVisit[len(setsToVisit)-1] + setsToVisit = setsToVisit[:len(setsToVisit)-1] + + for _, selection := range set { + switch selection := selection.(type) { + case *ast.FragmentSpread: + spreads = append(spreads, selection) + case *ast.Field: + setsToVisit = append(setsToVisit, selection.SelectionSet) + case *ast.InlineFragment: + setsToVisit = append(setsToVisit, selection.SelectionSet) + } + } + } + + return spreads +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/no_undefined_variables.go b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/no_undefined_variables.go new file mode 100644 index 00000000..e45a5e3d --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/no_undefined_variables.go @@ -0,0 +1,30 @@ +package validator + +import ( + "github.com/open-policy-agent/opa/internal/gqlparser/ast" + + //nolint:revive // Validator rules each use dot imports for convenience. + . "github.com/open-policy-agent/opa/internal/gqlparser/validator" +) + +func init() { + AddRule("NoUndefinedVariables", func(observers *Events, addError AddErrFunc) { + observers.OnValue(func(walker *Walker, value *ast.Value) { + if walker.CurrentOperation == nil || value.Kind != ast.Variable || value.VariableDefinition != nil { + return + } + + if walker.CurrentOperation.Name != "" { + addError( + Message(`Variable "%s" is not defined by operation "%s".`, value, walker.CurrentOperation.Name), + At(value.Position), + ) + } else { + addError( + Message(`Variable "%s" is not defined.`, value), + At(value.Position), + ) + } + }) + }) +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/no_unused_fragments.go b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/no_unused_fragments.go new file mode 100644 index 00000000..f82cfe9f --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/no_unused_fragments.go @@ -0,0 +1,32 @@ +package validator + +import ( + "github.com/open-policy-agent/opa/internal/gqlparser/ast" + + //nolint:revive // Validator rules each use dot imports for convenience. + . "github.com/open-policy-agent/opa/internal/gqlparser/validator" +) + +func init() { + AddRule("NoUnusedFragments", func(observers *Events, addError AddErrFunc) { + + inFragmentDefinition := false + fragmentNameUsed := make(map[string]bool) + + observers.OnFragmentSpread(func(walker *Walker, fragmentSpread *ast.FragmentSpread) { + if !inFragmentDefinition { + fragmentNameUsed[fragmentSpread.Name] = true + } + }) + + observers.OnFragment(func(walker *Walker, fragment *ast.FragmentDefinition) { + inFragmentDefinition = true + if !fragmentNameUsed[fragment.Name] { + addError( + Message(`Fragment "%s" is never used.`, fragment.Name), + At(fragment.Position), + ) + } + }) + }) +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/no_unused_variables.go b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/no_unused_variables.go new file mode 100644 index 00000000..c019ae42 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/no_unused_variables.go @@ -0,0 +1,32 @@ +package validator + +import ( + "github.com/open-policy-agent/opa/internal/gqlparser/ast" + + //nolint:revive // Validator rules each use dot imports for convenience. + . "github.com/open-policy-agent/opa/internal/gqlparser/validator" +) + +func init() { + AddRule("NoUnusedVariables", func(observers *Events, addError AddErrFunc) { + observers.OnOperation(func(walker *Walker, operation *ast.OperationDefinition) { + for _, varDef := range operation.VariableDefinitions { + if varDef.Used { + continue + } + + if operation.Name != "" { + addError( + Message(`Variable "$%s" is never used in operation "%s".`, varDef.Variable, operation.Name), + At(varDef.Position), + ) + } else { + addError( + Message(`Variable "$%s" is never used.`, varDef.Variable), + At(varDef.Position), + ) + } + } + }) + }) +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/overlapping_fields_can_be_merged.go b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/overlapping_fields_can_be_merged.go new file mode 100644 index 00000000..bda9a90b --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/overlapping_fields_can_be_merged.go @@ -0,0 +1,562 @@ +package validator + +import ( + "bytes" + "fmt" + "reflect" + + "github.com/open-policy-agent/opa/internal/gqlparser/ast" + + //nolint:revive // Validator rules each use dot imports for convenience. + . "github.com/open-policy-agent/opa/internal/gqlparser/validator" +) + +func init() { + + AddRule("OverlappingFieldsCanBeMerged", func(observers *Events, addError AddErrFunc) { + /** + * Algorithm: + * + * Conflicts occur when two fields exist in a query which will produce the same + * response name, but represent differing values, thus creating a conflict. + * The algorithm below finds all conflicts via making a series of comparisons + * between fields. In order to compare as few fields as possible, this makes + * a series of comparisons "within" sets of fields and "between" sets of fields. + * + * Given any selection set, a collection produces both a set of fields by + * also including all inline fragments, as well as a list of fragments + * referenced by fragment spreads. + * + * A) Each selection set represented in the document first compares "within" its + * collected set of fields, finding any conflicts between every pair of + * overlapping fields. + * Note: This is the *only time* that a the fields "within" a set are compared + * to each other. After this only fields "between" sets are compared. + * + * B) Also, if any fragment is referenced in a selection set, then a + * comparison is made "between" the original set of fields and the + * referenced fragment. + * + * C) Also, if multiple fragments are referenced, then comparisons + * are made "between" each referenced fragment. + * + * D) When comparing "between" a set of fields and a referenced fragment, first + * a comparison is made between each field in the original set of fields and + * each field in the the referenced set of fields. + * + * E) Also, if any fragment is referenced in the referenced selection set, + * then a comparison is made "between" the original set of fields and the + * referenced fragment (recursively referring to step D). + * + * F) When comparing "between" two fragments, first a comparison is made between + * each field in the first referenced set of fields and each field in the the + * second referenced set of fields. + * + * G) Also, any fragments referenced by the first must be compared to the + * second, and any fragments referenced by the second must be compared to the + * first (recursively referring to step F). + * + * H) When comparing two fields, if both have selection sets, then a comparison + * is made "between" both selection sets, first comparing the set of fields in + * the first selection set with the set of fields in the second. + * + * I) Also, if any fragment is referenced in either selection set, then a + * comparison is made "between" the other set of fields and the + * referenced fragment. + * + * J) Also, if two fragments are referenced in both selection sets, then a + * comparison is made "between" the two fragments. + * + */ + + m := &overlappingFieldsCanBeMergedManager{ + comparedFragmentPairs: pairSet{data: make(map[string]map[string]bool)}, + } + + observers.OnOperation(func(walker *Walker, operation *ast.OperationDefinition) { + m.walker = walker + conflicts := m.findConflictsWithinSelectionSet(operation.SelectionSet) + for _, conflict := range conflicts { + conflict.addFieldsConflictMessage(addError) + } + }) + observers.OnField(func(walker *Walker, field *ast.Field) { + if walker.CurrentOperation == nil { + // When checking both Operation and Fragment, errors are duplicated when processing FragmentDefinition referenced from Operation + return + } + m.walker = walker + conflicts := m.findConflictsWithinSelectionSet(field.SelectionSet) + for _, conflict := range conflicts { + conflict.addFieldsConflictMessage(addError) + } + }) + observers.OnInlineFragment(func(walker *Walker, inlineFragment *ast.InlineFragment) { + m.walker = walker + conflicts := m.findConflictsWithinSelectionSet(inlineFragment.SelectionSet) + for _, conflict := range conflicts { + conflict.addFieldsConflictMessage(addError) + } + }) + observers.OnFragment(func(walker *Walker, fragment *ast.FragmentDefinition) { + m.walker = walker + conflicts := m.findConflictsWithinSelectionSet(fragment.SelectionSet) + for _, conflict := range conflicts { + conflict.addFieldsConflictMessage(addError) + } + }) + }) +} + +type pairSet struct { + data map[string]map[string]bool +} + +func (pairSet *pairSet) Add(a *ast.FragmentSpread, b *ast.FragmentSpread, areMutuallyExclusive bool) { + add := func(a *ast.FragmentSpread, b *ast.FragmentSpread) { + m := pairSet.data[a.Name] + if m == nil { + m = make(map[string]bool) + pairSet.data[a.Name] = m + } + m[b.Name] = areMutuallyExclusive + } + add(a, b) + add(b, a) +} + +func (pairSet *pairSet) Has(a *ast.FragmentSpread, b *ast.FragmentSpread, areMutuallyExclusive bool) bool { + am, ok := pairSet.data[a.Name] + if !ok { + return false + } + result, ok := am[b.Name] + if !ok { + return false + } + + // areMutuallyExclusive being false is a superset of being true, + // hence if we want to know if this PairSet "has" these two with no + // exclusivity, we have to ensure it was added as such. + if !areMutuallyExclusive { + return !result + } + + return true +} + +type sequentialFieldsMap struct { + // We can't use map[string][]*ast.Field. because map is not stable... + seq []string + data map[string][]*ast.Field +} + +type fieldIterateEntry struct { + ResponseName string + Fields []*ast.Field +} + +func (m *sequentialFieldsMap) Push(responseName string, field *ast.Field) { + fields, ok := m.data[responseName] + if !ok { + m.seq = append(m.seq, responseName) + } + fields = append(fields, field) + m.data[responseName] = fields +} + +func (m *sequentialFieldsMap) Get(responseName string) ([]*ast.Field, bool) { + fields, ok := m.data[responseName] + return fields, ok +} + +func (m *sequentialFieldsMap) Iterator() [][]*ast.Field { + fieldsList := make([][]*ast.Field, 0, len(m.seq)) + for _, responseName := range m.seq { + fields := m.data[responseName] + fieldsList = append(fieldsList, fields) + } + return fieldsList +} + +func (m *sequentialFieldsMap) KeyValueIterator() []*fieldIterateEntry { + fieldEntriesList := make([]*fieldIterateEntry, 0, len(m.seq)) + for _, responseName := range m.seq { + fields := m.data[responseName] + fieldEntriesList = append(fieldEntriesList, &fieldIterateEntry{ + ResponseName: responseName, + Fields: fields, + }) + } + return fieldEntriesList +} + +type conflictMessageContainer struct { + Conflicts []*ConflictMessage +} + +type ConflictMessage struct { + Message string + ResponseName string + Names []string + SubMessage []*ConflictMessage + Position *ast.Position +} + +func (m *ConflictMessage) String(buf *bytes.Buffer) { + if len(m.SubMessage) == 0 { + buf.WriteString(m.Message) + return + } + + for idx, subMessage := range m.SubMessage { + buf.WriteString(`subfields "`) + buf.WriteString(subMessage.ResponseName) + buf.WriteString(`" conflict because `) + subMessage.String(buf) + if idx != len(m.SubMessage)-1 { + buf.WriteString(" and ") + } + } +} + +func (m *ConflictMessage) addFieldsConflictMessage(addError AddErrFunc) { + var buf bytes.Buffer + m.String(&buf) + addError( + Message(`Fields "%s" conflict because %s. Use different aliases on the fields to fetch both if this was intentional.`, m.ResponseName, buf.String()), + At(m.Position), + ) +} + +type overlappingFieldsCanBeMergedManager struct { + walker *Walker + + // per walker + comparedFragmentPairs pairSet + // cachedFieldsAndFragmentNames interface{} + + // per selectionSet + comparedFragments map[string]bool +} + +func (m *overlappingFieldsCanBeMergedManager) findConflictsWithinSelectionSet(selectionSet ast.SelectionSet) []*ConflictMessage { + if len(selectionSet) == 0 { + return nil + } + + fieldsMap, fragmentSpreads := getFieldsAndFragmentNames(selectionSet) + + var conflicts conflictMessageContainer + + // (A) Find find all conflicts "within" the fieldMap of this selection set. + // Note: this is the *only place* `collectConflictsWithin` is called. + m.collectConflictsWithin(&conflicts, fieldsMap) + + m.comparedFragments = make(map[string]bool) + for idx, fragmentSpreadA := range fragmentSpreads { + // (B) Then collect conflicts between these fieldMap and those represented by + // each spread fragment name found. + m.collectConflictsBetweenFieldsAndFragment(&conflicts, false, fieldsMap, fragmentSpreadA) + + for _, fragmentSpreadB := range fragmentSpreads[idx+1:] { + // (C) Then compare this fragment with all other fragments found in this + // selection set to collect conflicts between fragments spread together. + // This compares each item in the list of fragment names to every other + // item in that same list (except for itself). + m.collectConflictsBetweenFragments(&conflicts, false, fragmentSpreadA, fragmentSpreadB) + } + } + + return conflicts.Conflicts +} + +func (m *overlappingFieldsCanBeMergedManager) collectConflictsBetweenFieldsAndFragment(conflicts *conflictMessageContainer, areMutuallyExclusive bool, fieldsMap *sequentialFieldsMap, fragmentSpread *ast.FragmentSpread) { + if m.comparedFragments[fragmentSpread.Name] { + return + } + m.comparedFragments[fragmentSpread.Name] = true + + if fragmentSpread.Definition == nil { + return + } + + fieldsMapB, fragmentSpreads := getFieldsAndFragmentNames(fragmentSpread.Definition.SelectionSet) + + // Do not compare a fragment's fieldMap to itself. + if reflect.DeepEqual(fieldsMap, fieldsMapB) { + return + } + + // (D) First collect any conflicts between the provided collection of fields + // and the collection of fields represented by the given fragment. + m.collectConflictsBetween(conflicts, areMutuallyExclusive, fieldsMap, fieldsMapB) + + // (E) Then collect any conflicts between the provided collection of fields + // and any fragment names found in the given fragment. + baseFragmentSpread := fragmentSpread + for _, fragmentSpread := range fragmentSpreads { + if fragmentSpread.Name == baseFragmentSpread.Name { + continue + } + m.collectConflictsBetweenFieldsAndFragment(conflicts, areMutuallyExclusive, fieldsMap, fragmentSpread) + } +} + +func (m *overlappingFieldsCanBeMergedManager) collectConflictsBetweenFragments(conflicts *conflictMessageContainer, areMutuallyExclusive bool, fragmentSpreadA *ast.FragmentSpread, fragmentSpreadB *ast.FragmentSpread) { + + var check func(fragmentSpreadA *ast.FragmentSpread, fragmentSpreadB *ast.FragmentSpread) + check = func(fragmentSpreadA *ast.FragmentSpread, fragmentSpreadB *ast.FragmentSpread) { + + if fragmentSpreadA.Name == fragmentSpreadB.Name { + return + } + + if m.comparedFragmentPairs.Has(fragmentSpreadA, fragmentSpreadB, areMutuallyExclusive) { + return + } + m.comparedFragmentPairs.Add(fragmentSpreadA, fragmentSpreadB, areMutuallyExclusive) + + if fragmentSpreadA.Definition == nil { + return + } + if fragmentSpreadB.Definition == nil { + return + } + + fieldsMapA, fragmentSpreadsA := getFieldsAndFragmentNames(fragmentSpreadA.Definition.SelectionSet) + fieldsMapB, fragmentSpreadsB := getFieldsAndFragmentNames(fragmentSpreadB.Definition.SelectionSet) + + // (F) First, collect all conflicts between these two collections of fields + // (not including any nested fragments). + m.collectConflictsBetween(conflicts, areMutuallyExclusive, fieldsMapA, fieldsMapB) + + // (G) Then collect conflicts between the first fragment and any nested + // fragments spread in the second fragment. + for _, fragmentSpread := range fragmentSpreadsB { + check(fragmentSpreadA, fragmentSpread) + } + // (G) Then collect conflicts between the second fragment and any nested + // fragments spread in the first fragment. + for _, fragmentSpread := range fragmentSpreadsA { + check(fragmentSpread, fragmentSpreadB) + } + } + + check(fragmentSpreadA, fragmentSpreadB) +} + +func (m *overlappingFieldsCanBeMergedManager) findConflictsBetweenSubSelectionSets(areMutuallyExclusive bool, selectionSetA ast.SelectionSet, selectionSetB ast.SelectionSet) *conflictMessageContainer { + var conflicts conflictMessageContainer + + fieldsMapA, fragmentSpreadsA := getFieldsAndFragmentNames(selectionSetA) + fieldsMapB, fragmentSpreadsB := getFieldsAndFragmentNames(selectionSetB) + + // (H) First, collect all conflicts between these two collections of field. + m.collectConflictsBetween(&conflicts, areMutuallyExclusive, fieldsMapA, fieldsMapB) + + // (I) Then collect conflicts between the first collection of fields and + // those referenced by each fragment name associated with the second. + for _, fragmentSpread := range fragmentSpreadsB { + m.comparedFragments = make(map[string]bool) + m.collectConflictsBetweenFieldsAndFragment(&conflicts, areMutuallyExclusive, fieldsMapA, fragmentSpread) + } + + // (I) Then collect conflicts between the second collection of fields and + // those referenced by each fragment name associated with the first. + for _, fragmentSpread := range fragmentSpreadsA { + m.comparedFragments = make(map[string]bool) + m.collectConflictsBetweenFieldsAndFragment(&conflicts, areMutuallyExclusive, fieldsMapB, fragmentSpread) + } + + // (J) Also collect conflicts between any fragment names by the first and + // fragment names by the second. This compares each item in the first set of + // names to each item in the second set of names. + for _, fragmentSpreadA := range fragmentSpreadsA { + for _, fragmentSpreadB := range fragmentSpreadsB { + m.collectConflictsBetweenFragments(&conflicts, areMutuallyExclusive, fragmentSpreadA, fragmentSpreadB) + } + } + + if len(conflicts.Conflicts) == 0 { + return nil + } + + return &conflicts +} + +func (m *overlappingFieldsCanBeMergedManager) collectConflictsWithin(conflicts *conflictMessageContainer, fieldsMap *sequentialFieldsMap) { + for _, fields := range fieldsMap.Iterator() { + for idx, fieldA := range fields { + for _, fieldB := range fields[idx+1:] { + conflict := m.findConflict(false, fieldA, fieldB) + if conflict != nil { + conflicts.Conflicts = append(conflicts.Conflicts, conflict) + } + } + } + } +} + +func (m *overlappingFieldsCanBeMergedManager) collectConflictsBetween(conflicts *conflictMessageContainer, parentFieldsAreMutuallyExclusive bool, fieldsMapA *sequentialFieldsMap, fieldsMapB *sequentialFieldsMap) { + for _, fieldsEntryA := range fieldsMapA.KeyValueIterator() { + fieldsB, ok := fieldsMapB.Get(fieldsEntryA.ResponseName) + if !ok { + continue + } + for _, fieldA := range fieldsEntryA.Fields { + for _, fieldB := range fieldsB { + conflict := m.findConflict(parentFieldsAreMutuallyExclusive, fieldA, fieldB) + if conflict != nil { + conflicts.Conflicts = append(conflicts.Conflicts, conflict) + } + } + } + } +} + +func (m *overlappingFieldsCanBeMergedManager) findConflict(parentFieldsAreMutuallyExclusive bool, fieldA *ast.Field, fieldB *ast.Field) *ConflictMessage { + if fieldA.ObjectDefinition == nil || fieldB.ObjectDefinition == nil { + return nil + } + + areMutuallyExclusive := parentFieldsAreMutuallyExclusive + if !areMutuallyExclusive { + tmp := fieldA.ObjectDefinition.Name != fieldB.ObjectDefinition.Name + tmp = tmp && fieldA.ObjectDefinition.Kind == ast.Object + tmp = tmp && fieldB.ObjectDefinition.Kind == ast.Object + tmp = tmp && fieldA.Definition != nil && fieldB.Definition != nil + areMutuallyExclusive = tmp + } + + fieldNameA := fieldA.Name + if fieldA.Alias != "" { + fieldNameA = fieldA.Alias + } + + if !areMutuallyExclusive { + // Two aliases must refer to the same field. + if fieldA.Name != fieldB.Name { + return &ConflictMessage{ + ResponseName: fieldNameA, + Message: fmt.Sprintf(`"%s" and "%s" are different fields`, fieldA.Name, fieldB.Name), + Position: fieldB.Position, + } + } + + // Two field calls must have the same arguments. + if !sameArguments(fieldA.Arguments, fieldB.Arguments) { + return &ConflictMessage{ + ResponseName: fieldNameA, + Message: "they have differing arguments", + Position: fieldB.Position, + } + } + } + + if fieldA.Definition != nil && fieldB.Definition != nil && doTypesConflict(m.walker, fieldA.Definition.Type, fieldB.Definition.Type) { + return &ConflictMessage{ + ResponseName: fieldNameA, + Message: fmt.Sprintf(`they return conflicting types "%s" and "%s"`, fieldA.Definition.Type.String(), fieldB.Definition.Type.String()), + Position: fieldB.Position, + } + } + + // Collect and compare sub-fields. Use the same "visited fragment names" list + // for both collections so fields in a fragment reference are never + // compared to themselves. + conflicts := m.findConflictsBetweenSubSelectionSets(areMutuallyExclusive, fieldA.SelectionSet, fieldB.SelectionSet) + if conflicts == nil { + return nil + } + return &ConflictMessage{ + ResponseName: fieldNameA, + SubMessage: conflicts.Conflicts, + Position: fieldB.Position, + } +} + +func sameArguments(args1 []*ast.Argument, args2 []*ast.Argument) bool { + if len(args1) != len(args2) { + return false + } + for _, arg1 := range args1 { + var matched bool + for _, arg2 := range args2 { + if arg1.Name == arg2.Name && sameValue(arg1.Value, arg2.Value) { + matched = true + break + } + } + if !matched { + return false + } + } + return true +} + +func sameValue(value1 *ast.Value, value2 *ast.Value) bool { + if value1.Kind != value2.Kind { + return false + } + if value1.Raw != value2.Raw { + return false + } + return true +} + +func doTypesConflict(walker *Walker, type1 *ast.Type, type2 *ast.Type) bool { + if type1.Elem != nil { + if type2.Elem != nil { + return doTypesConflict(walker, type1.Elem, type2.Elem) + } + return true + } + if type2.Elem != nil { + return true + } + if type1.NonNull && !type2.NonNull { + return true + } + if !type1.NonNull && type2.NonNull { + return true + } + + t1 := walker.Schema.Types[type1.NamedType] + t2 := walker.Schema.Types[type2.NamedType] + if (t1.Kind == ast.Scalar || t1.Kind == ast.Enum) && (t2.Kind == ast.Scalar || t2.Kind == ast.Enum) { + return t1.Name != t2.Name + } + + return false +} + +func getFieldsAndFragmentNames(selectionSet ast.SelectionSet) (*sequentialFieldsMap, []*ast.FragmentSpread) { + fieldsMap := sequentialFieldsMap{ + data: make(map[string][]*ast.Field), + } + var fragmentSpreads []*ast.FragmentSpread + + var walk func(selectionSet ast.SelectionSet) + walk = func(selectionSet ast.SelectionSet) { + for _, selection := range selectionSet { + switch selection := selection.(type) { + case *ast.Field: + responseName := selection.Name + if selection.Alias != "" { + responseName = selection.Alias + } + fieldsMap.Push(responseName, selection) + + case *ast.InlineFragment: + walk(selection.SelectionSet) + + case *ast.FragmentSpread: + fragmentSpreads = append(fragmentSpreads, selection) + } + } + } + walk(selectionSet) + + return &fieldsMap, fragmentSpreads +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/possible_fragment_spreads.go b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/possible_fragment_spreads.go new file mode 100644 index 00000000..79cb20c4 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/possible_fragment_spreads.go @@ -0,0 +1,70 @@ +package validator + +import ( + "github.com/open-policy-agent/opa/internal/gqlparser/ast" + + //nolint:revive // Validator rules each use dot imports for convenience. + . "github.com/open-policy-agent/opa/internal/gqlparser/validator" +) + +func init() { + AddRule("PossibleFragmentSpreads", func(observers *Events, addError AddErrFunc) { + + validate := func(walker *Walker, parentDef *ast.Definition, fragmentName string, emitError func()) { + if parentDef == nil { + return + } + + var parentDefs []*ast.Definition + switch parentDef.Kind { + case ast.Object: + parentDefs = []*ast.Definition{parentDef} + case ast.Interface, ast.Union: + parentDefs = walker.Schema.GetPossibleTypes(parentDef) + default: + return + } + + fragmentDefType := walker.Schema.Types[fragmentName] + if fragmentDefType == nil { + return + } + if !fragmentDefType.IsCompositeType() { + // checked by FragmentsOnCompositeTypes + return + } + fragmentDefs := walker.Schema.GetPossibleTypes(fragmentDefType) + + for _, fragmentDef := range fragmentDefs { + for _, parentDef := range parentDefs { + if parentDef.Name == fragmentDef.Name { + return + } + } + } + + emitError() + } + + observers.OnInlineFragment(func(walker *Walker, inlineFragment *ast.InlineFragment) { + validate(walker, inlineFragment.ObjectDefinition, inlineFragment.TypeCondition, func() { + addError( + Message(`Fragment cannot be spread here as objects of type "%s" can never be of type "%s".`, inlineFragment.ObjectDefinition.Name, inlineFragment.TypeCondition), + At(inlineFragment.Position), + ) + }) + }) + + observers.OnFragmentSpread(func(walker *Walker, fragmentSpread *ast.FragmentSpread) { + if fragmentSpread.Definition == nil { + return + } + validate(walker, fragmentSpread.ObjectDefinition, fragmentSpread.Definition.TypeCondition, func() { + addError( + Message(`Fragment "%s" cannot be spread here as objects of type "%s" can never be of type "%s".`, fragmentSpread.Name, fragmentSpread.ObjectDefinition.Name, fragmentSpread.Definition.TypeCondition), + At(fragmentSpread.Position), + ) + }) + }) + }) +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/provided_required_arguments.go b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/provided_required_arguments.go new file mode 100644 index 00000000..20364f8b --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/provided_required_arguments.go @@ -0,0 +1,64 @@ +package validator + +import ( + "github.com/open-policy-agent/opa/internal/gqlparser/ast" + + //nolint:revive // Validator rules each use dot imports for convenience. + . "github.com/open-policy-agent/opa/internal/gqlparser/validator" +) + +func init() { + AddRule("ProvidedRequiredArguments", func(observers *Events, addError AddErrFunc) { + observers.OnField(func(walker *Walker, field *ast.Field) { + if field.Definition == nil { + return + } + + argDef: + for _, argDef := range field.Definition.Arguments { + if !argDef.Type.NonNull { + continue + } + if argDef.DefaultValue != nil { + continue + } + for _, arg := range field.Arguments { + if arg.Name == argDef.Name { + continue argDef + } + } + + addError( + Message(`Field "%s" argument "%s" of type "%s" is required, but it was not provided.`, field.Name, argDef.Name, argDef.Type.String()), + At(field.Position), + ) + } + }) + + observers.OnDirective(func(walker *Walker, directive *ast.Directive) { + if directive.Definition == nil { + return + } + + argDef: + for _, argDef := range directive.Definition.Arguments { + if !argDef.Type.NonNull { + continue + } + if argDef.DefaultValue != nil { + continue + } + for _, arg := range directive.Arguments { + if arg.Name == argDef.Name { + continue argDef + } + } + + addError( + Message(`Directive "@%s" argument "%s" of type "%s" is required, but it was not provided.`, directive.Definition.Name, argDef.Name, argDef.Type.String()), + At(directive.Position), + ) + } + }) + }) +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/scalar_leafs.go b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/scalar_leafs.go new file mode 100644 index 00000000..cd17b47c --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/scalar_leafs.go @@ -0,0 +1,38 @@ +package validator + +import ( + "github.com/open-policy-agent/opa/internal/gqlparser/ast" + + //nolint:revive // Validator rules each use dot imports for convenience. + . "github.com/open-policy-agent/opa/internal/gqlparser/validator" +) + +func init() { + AddRule("ScalarLeafs", func(observers *Events, addError AddErrFunc) { + observers.OnField(func(walker *Walker, field *ast.Field) { + if field.Definition == nil { + return + } + + fieldType := walker.Schema.Types[field.Definition.Type.Name()] + if fieldType == nil { + return + } + + if fieldType.IsLeafType() && len(field.SelectionSet) > 0 { + addError( + Message(`Field "%s" must not have a selection since type "%s" has no subfields.`, field.Name, fieldType.Name), + At(field.Position), + ) + } + + if !fieldType.IsLeafType() && len(field.SelectionSet) == 0 { + addError( + Message(`Field "%s" of type "%s" must have a selection of subfields.`, field.Name, field.Definition.Type.String()), + Suggestf(`"%s { ... }"`, field.Name), + At(field.Position), + ) + } + }) + }) +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/single_field_subscriptions.go b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/single_field_subscriptions.go new file mode 100644 index 00000000..98cb984b --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/single_field_subscriptions.go @@ -0,0 +1,88 @@ +package validator + +import ( + "strconv" + "strings" + + "github.com/open-policy-agent/opa/internal/gqlparser/ast" + + //nolint:revive // Validator rules each use dot imports for convenience. + . "github.com/open-policy-agent/opa/internal/gqlparser/validator" +) + +func init() { + AddRule("SingleFieldSubscriptions", func(observers *Events, addError AddErrFunc) { + observers.OnOperation(func(walker *Walker, operation *ast.OperationDefinition) { + if walker.Schema.Subscription == nil || operation.Operation != ast.Subscription { + return + } + + fields := retrieveTopFieldNames(operation.SelectionSet) + + name := "Anonymous Subscription" + if operation.Name != "" { + name = `Subscription ` + strconv.Quote(operation.Name) + } + + if len(fields) > 1 { + addError( + Message(`%s must select only one top level field.`, name), + At(fields[1].position), + ) + } + + for _, field := range fields { + if strings.HasPrefix(field.name, "__") { + addError( + Message(`%s must not select an introspection top level field.`, name), + At(field.position), + ) + } + } + }) + }) +} + +type topField struct { + name string + position *ast.Position +} + +func retrieveTopFieldNames(selectionSet ast.SelectionSet) []*topField { + fields := []*topField{} + inFragmentRecursive := map[string]bool{} + var walk func(selectionSet ast.SelectionSet) + walk = func(selectionSet ast.SelectionSet) { + for _, selection := range selectionSet { + switch selection := selection.(type) { + case *ast.Field: + fields = append(fields, &topField{ + name: selection.Name, + position: selection.GetPosition(), + }) + case *ast.InlineFragment: + walk(selection.SelectionSet) + case *ast.FragmentSpread: + if selection.Definition == nil { + return + } + fragment := selection.Definition.Name + if !inFragmentRecursive[fragment] { + inFragmentRecursive[fragment] = true + walk(selection.Definition.SelectionSet) + } + } + } + } + walk(selectionSet) + + seen := make(map[string]bool, len(fields)) + uniquedFields := make([]*topField, 0, len(fields)) + for _, field := range fields { + if !seen[field.name] { + uniquedFields = append(uniquedFields, field) + } + seen[field.name] = true + } + return uniquedFields +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/unique_argument_names.go b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/unique_argument_names.go new file mode 100644 index 00000000..4466d567 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/unique_argument_names.go @@ -0,0 +1,35 @@ +package validator + +import ( + "github.com/open-policy-agent/opa/internal/gqlparser/ast" + + //nolint:revive // Validator rules each use dot imports for convenience. + . "github.com/open-policy-agent/opa/internal/gqlparser/validator" +) + +func init() { + AddRule("UniqueArgumentNames", func(observers *Events, addError AddErrFunc) { + observers.OnField(func(walker *Walker, field *ast.Field) { + checkUniqueArgs(field.Arguments, addError) + }) + + observers.OnDirective(func(walker *Walker, directive *ast.Directive) { + checkUniqueArgs(directive.Arguments, addError) + }) + }) +} + +func checkUniqueArgs(args ast.ArgumentList, addError AddErrFunc) { + knownArgNames := map[string]int{} + + for _, arg := range args { + if knownArgNames[arg.Name] == 1 { + addError( + Message(`There can be only one argument named "%s".`, arg.Name), + At(arg.Position), + ) + } + + knownArgNames[arg.Name]++ + } +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/unique_directives_per_location.go b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/unique_directives_per_location.go new file mode 100644 index 00000000..09968a74 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/unique_directives_per_location.go @@ -0,0 +1,26 @@ +package validator + +import ( + "github.com/open-policy-agent/opa/internal/gqlparser/ast" + + //nolint:revive // Validator rules each use dot imports for convenience. + . "github.com/open-policy-agent/opa/internal/gqlparser/validator" +) + +func init() { + AddRule("UniqueDirectivesPerLocation", func(observers *Events, addError AddErrFunc) { + observers.OnDirectiveList(func(walker *Walker, directives []*ast.Directive) { + seen := map[string]bool{} + + for _, dir := range directives { + if dir.Name != "repeatable" && seen[dir.Name] { + addError( + Message(`The directive "@%s" can only be used once at this location.`, dir.Name), + At(dir.Position), + ) + } + seen[dir.Name] = true + } + }) + }) +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/unique_fragment_names.go b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/unique_fragment_names.go new file mode 100644 index 00000000..3da46467 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/unique_fragment_names.go @@ -0,0 +1,24 @@ +package validator + +import ( + "github.com/open-policy-agent/opa/internal/gqlparser/ast" + + //nolint:revive // Validator rules each use dot imports for convenience. + . "github.com/open-policy-agent/opa/internal/gqlparser/validator" +) + +func init() { + AddRule("UniqueFragmentNames", func(observers *Events, addError AddErrFunc) { + seenFragments := map[string]bool{} + + observers.OnFragment(func(walker *Walker, fragment *ast.FragmentDefinition) { + if seenFragments[fragment.Name] { + addError( + Message(`There can be only one fragment named "%s".`, fragment.Name), + At(fragment.Position), + ) + } + seenFragments[fragment.Name] = true + }) + }) +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/unique_input_field_names.go b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/unique_input_field_names.go new file mode 100644 index 00000000..f9f69051 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/unique_input_field_names.go @@ -0,0 +1,29 @@ +package validator + +import ( + "github.com/open-policy-agent/opa/internal/gqlparser/ast" + + //nolint:revive // Validator rules each use dot imports for convenience. + . "github.com/open-policy-agent/opa/internal/gqlparser/validator" +) + +func init() { + AddRule("UniqueInputFieldNames", func(observers *Events, addError AddErrFunc) { + observers.OnValue(func(walker *Walker, value *ast.Value) { + if value.Kind != ast.ObjectValue { + return + } + + seen := map[string]bool{} + for _, field := range value.Children { + if seen[field.Name] { + addError( + Message(`There can be only one input field named "%s".`, field.Name), + At(field.Position), + ) + } + seen[field.Name] = true + } + }) + }) +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/unique_operation_names.go b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/unique_operation_names.go new file mode 100644 index 00000000..dc937795 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/unique_operation_names.go @@ -0,0 +1,24 @@ +package validator + +import ( + "github.com/open-policy-agent/opa/internal/gqlparser/ast" + + //nolint:revive // Validator rules each use dot imports for convenience. + . "github.com/open-policy-agent/opa/internal/gqlparser/validator" +) + +func init() { + AddRule("UniqueOperationNames", func(observers *Events, addError AddErrFunc) { + seen := map[string]bool{} + + observers.OnOperation(func(walker *Walker, operation *ast.OperationDefinition) { + if seen[operation.Name] { + addError( + Message(`There can be only one operation named "%s".`, operation.Name), + At(operation.Position), + ) + } + seen[operation.Name] = true + }) + }) +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/unique_variable_names.go b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/unique_variable_names.go new file mode 100644 index 00000000..94dab3cf --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/unique_variable_names.go @@ -0,0 +1,26 @@ +package validator + +import ( + "github.com/open-policy-agent/opa/internal/gqlparser/ast" + + //nolint:revive // Validator rules each use dot imports for convenience. + . "github.com/open-policy-agent/opa/internal/gqlparser/validator" +) + +func init() { + AddRule("UniqueVariableNames", func(observers *Events, addError AddErrFunc) { + observers.OnOperation(func(walker *Walker, operation *ast.OperationDefinition) { + seen := map[string]int{} + for _, def := range operation.VariableDefinitions { + // add the same error only once per a variable. + if seen[def.Variable] == 1 { + addError( + Message(`There can be only one variable named "$%s".`, def.Variable), + At(def.Position), + ) + } + seen[def.Variable]++ + } + }) + }) +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/values_of_correct_type.go b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/values_of_correct_type.go new file mode 100644 index 00000000..88d8f53c --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/values_of_correct_type.go @@ -0,0 +1,170 @@ +package validator + +import ( + "errors" + "fmt" + "strconv" + + "github.com/open-policy-agent/opa/internal/gqlparser/ast" + + //nolint:revive // Validator rules each use dot imports for convenience. + . "github.com/open-policy-agent/opa/internal/gqlparser/validator" +) + +func init() { + AddRule("ValuesOfCorrectType", func(observers *Events, addError AddErrFunc) { + observers.OnValue(func(walker *Walker, value *ast.Value) { + if value.Definition == nil || value.ExpectedType == nil { + return + } + + if value.Kind == ast.NullValue && value.ExpectedType.NonNull { + addError( + Message(`Expected value of type "%s", found %s.`, value.ExpectedType.String(), value.String()), + At(value.Position), + ) + } + + if value.Definition.Kind == ast.Scalar { + // Skip custom validating scalars + if !value.Definition.OneOf("Int", "Float", "String", "Boolean", "ID") { + return + } + } + + var possibleEnums []string + if value.Definition.Kind == ast.Enum { + for _, val := range value.Definition.EnumValues { + possibleEnums = append(possibleEnums, val.Name) + } + } + + rawVal, err := value.Value(nil) + if err != nil { + unexpectedTypeMessage(addError, value) + } + + switch value.Kind { + case ast.NullValue: + return + case ast.ListValue: + if value.ExpectedType.Elem == nil { + unexpectedTypeMessage(addError, value) + return + } + + case ast.IntValue: + if !value.Definition.OneOf("Int", "Float", "ID") { + unexpectedTypeMessage(addError, value) + } + + case ast.FloatValue: + if !value.Definition.OneOf("Float") { + unexpectedTypeMessage(addError, value) + } + + case ast.StringValue, ast.BlockValue: + if value.Definition.Kind == ast.Enum { + rawValStr := fmt.Sprint(rawVal) + addError( + Message(`Enum "%s" cannot represent non-enum value: %s.`, value.ExpectedType.String(), value.String()), + SuggestListQuoted("Did you mean the enum value", rawValStr, possibleEnums), + At(value.Position), + ) + } else if !value.Definition.OneOf("String", "ID") { + unexpectedTypeMessage(addError, value) + } + + case ast.EnumValue: + if value.Definition.Kind != ast.Enum { + rawValStr := fmt.Sprint(rawVal) + addError( + unexpectedTypeMessageOnly(value), + SuggestListUnquoted("Did you mean the enum value", rawValStr, possibleEnums), + At(value.Position), + ) + } else if value.Definition.EnumValues.ForName(value.Raw) == nil { + rawValStr := fmt.Sprint(rawVal) + addError( + Message(`Value "%s" does not exist in "%s" enum.`, value.String(), value.ExpectedType.String()), + SuggestListQuoted("Did you mean the enum value", rawValStr, possibleEnums), + At(value.Position), + ) + } + + case ast.BooleanValue: + if !value.Definition.OneOf("Boolean") { + unexpectedTypeMessage(addError, value) + } + + case ast.ObjectValue: + + for _, field := range value.Definition.Fields { + if field.Type.NonNull { + fieldValue := value.Children.ForName(field.Name) + if fieldValue == nil && field.DefaultValue == nil { + addError( + Message(`Field "%s.%s" of required type "%s" was not provided.`, value.Definition.Name, field.Name, field.Type.String()), + At(value.Position), + ) + continue + } + } + } + + for _, fieldValue := range value.Children { + if value.Definition.Fields.ForName(fieldValue.Name) == nil { + var suggestions []string + for _, fieldValue := range value.Definition.Fields { + suggestions = append(suggestions, fieldValue.Name) + } + + addError( + Message(`Field "%s" is not defined by type "%s".`, fieldValue.Name, value.Definition.Name), + SuggestListQuoted("Did you mean", fieldValue.Name, suggestions), + At(fieldValue.Position), + ) + } + } + + case ast.Variable: + return + + default: + panic(fmt.Errorf("unhandled %T", value)) + } + }) + }) +} + +func unexpectedTypeMessage(addError AddErrFunc, v *ast.Value) { + addError( + unexpectedTypeMessageOnly(v), + At(v.Position), + ) +} + +func unexpectedTypeMessageOnly(v *ast.Value) ErrorOption { + switch v.ExpectedType.String() { + case "Int", "Int!": + if _, err := strconv.ParseInt(v.Raw, 10, 32); err != nil && errors.Is(err, strconv.ErrRange) { + return Message(`Int cannot represent non 32-bit signed integer value: %s`, v.String()) + } + return Message(`Int cannot represent non-integer value: %s`, v.String()) + case "String", "String!", "[String]": + return Message(`String cannot represent a non string value: %s`, v.String()) + case "Boolean", "Boolean!": + return Message(`Boolean cannot represent a non boolean value: %s`, v.String()) + case "Float", "Float!": + return Message(`Float cannot represent non numeric value: %s`, v.String()) + case "ID", "ID!": + return Message(`ID cannot represent a non-string and non-integer value: %s`, v.String()) + //case "Enum": + // return Message(`Enum "%s" cannot represent non-enum value: %s`, v.ExpectedType.String(), v.String()) + default: + if v.Definition.Kind == ast.Enum { + return Message(`Enum "%s" cannot represent non-enum value: %s.`, v.ExpectedType.String(), v.String()) + } + return Message(`Expected value of type "%s", found %s.`, v.ExpectedType.String(), v.String()) + } +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/variables_are_input_types.go b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/variables_are_input_types.go new file mode 100644 index 00000000..51cb77ca --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/variables_are_input_types.go @@ -0,0 +1,30 @@ +package validator + +import ( + "github.com/open-policy-agent/opa/internal/gqlparser/ast" + + //nolint:revive // Validator rules each use dot imports for convenience. + . "github.com/open-policy-agent/opa/internal/gqlparser/validator" +) + +func init() { + AddRule("VariablesAreInputTypes", func(observers *Events, addError AddErrFunc) { + observers.OnOperation(func(walker *Walker, operation *ast.OperationDefinition) { + for _, def := range operation.VariableDefinitions { + if def.Definition == nil { + continue + } + if !def.Definition.IsInputType() { + addError( + Message( + `Variable "$%s" cannot be non-input type "%s".`, + def.Variable, + def.Type.String(), + ), + At(def.Position), + ) + } + } + }) + }) +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/variables_in_allowed_position.go b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/variables_in_allowed_position.go new file mode 100644 index 00000000..08a8e18c --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/rules/variables_in_allowed_position.go @@ -0,0 +1,40 @@ +package validator + +import ( + "github.com/open-policy-agent/opa/internal/gqlparser/ast" + + //nolint:revive // Validator rules each use dot imports for convenience. + . "github.com/open-policy-agent/opa/internal/gqlparser/validator" +) + +func init() { + AddRule("VariablesInAllowedPosition", func(observers *Events, addError AddErrFunc) { + observers.OnValue(func(walker *Walker, value *ast.Value) { + if value.Kind != ast.Variable || value.ExpectedType == nil || value.VariableDefinition == nil || walker.CurrentOperation == nil { + return + } + + tmp := *value.ExpectedType + + // todo: move me into walk + // If there is a default non nullable types can be null + if value.VariableDefinition.DefaultValue != nil && value.VariableDefinition.DefaultValue.Kind != ast.NullValue { + if value.ExpectedType.NonNull { + tmp.NonNull = false + } + } + + if !value.VariableDefinition.Type.IsCompatible(&tmp) { + addError( + Message( + `Variable "%s" of type "%s" used in position expecting type "%s".`, + value, + value.VariableDefinition.Type.String(), + value.ExpectedType.String(), + ), + At(value.Position), + ) + } + }) + }) +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/schema.go b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/schema.go new file mode 100644 index 00000000..c9c54219 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/schema.go @@ -0,0 +1,513 @@ +package validator + +import ( + "sort" + "strconv" + "strings" + + //nolint:revive + . "github.com/open-policy-agent/opa/internal/gqlparser/ast" + "github.com/open-policy-agent/opa/internal/gqlparser/gqlerror" + "github.com/open-policy-agent/opa/internal/gqlparser/parser" +) + +func LoadSchema(inputs ...*Source) (*Schema, error) { + ast, err := parser.ParseSchemas(inputs...) + if err != nil { + return nil, err + } + return ValidateSchemaDocument(ast) +} + +func ValidateSchemaDocument(ast *SchemaDocument) (*Schema, error) { + schema := Schema{ + Types: map[string]*Definition{}, + Directives: map[string]*DirectiveDefinition{}, + PossibleTypes: map[string][]*Definition{}, + Implements: map[string][]*Definition{}, + } + + for i, def := range ast.Definitions { + if schema.Types[def.Name] != nil { + return nil, gqlerror.ErrorPosf(def.Position, "Cannot redeclare type %s.", def.Name) + } + schema.Types[def.Name] = ast.Definitions[i] + } + + defs := append(DefinitionList{}, ast.Definitions...) + + for _, ext := range ast.Extensions { + def := schema.Types[ext.Name] + if def == nil { + schema.Types[ext.Name] = &Definition{ + Kind: ext.Kind, + Name: ext.Name, + Position: ext.Position, + } + def = schema.Types[ext.Name] + defs = append(defs, def) + } + + if def.Kind != ext.Kind { + return nil, gqlerror.ErrorPosf(ext.Position, "Cannot extend type %s because the base type is a %s, not %s.", ext.Name, def.Kind, ext.Kind) + } + + def.Directives = append(def.Directives, ext.Directives...) + def.Interfaces = append(def.Interfaces, ext.Interfaces...) + def.Fields = append(def.Fields, ext.Fields...) + def.Types = append(def.Types, ext.Types...) + def.EnumValues = append(def.EnumValues, ext.EnumValues...) + } + + for _, def := range defs { + switch def.Kind { + case Union: + for _, t := range def.Types { + schema.AddPossibleType(def.Name, schema.Types[t]) + schema.AddImplements(t, def) + } + case InputObject, Object: + for _, intf := range def.Interfaces { + schema.AddPossibleType(intf, def) + schema.AddImplements(def.Name, schema.Types[intf]) + } + schema.AddPossibleType(def.Name, def) + case Interface: + for _, intf := range def.Interfaces { + schema.AddPossibleType(intf, def) + schema.AddImplements(def.Name, schema.Types[intf]) + } + } + } + + for i, dir := range ast.Directives { + if schema.Directives[dir.Name] != nil { + // While the spec says SDL must not (§3.5) explicitly define builtin + // scalars, it may (§3.13) define builtin directives. Here we check for + // that, and reject doubly-defined directives otherwise. + switch dir.Name { + case "include", "skip", "deprecated", "specifiedBy": // the builtins + // In principle here we might want to validate that the + // directives are the same. But they might not be, if the + // server has an older spec than we do. (Plus, validating this + // is a lot of work.) So we just keep the first one we saw. + // That's an arbitrary choice, but in theory the only way it + // fails is if the server is using features newer than this + // version of gqlparser, in which case they're in trouble + // anyway. + default: + return nil, gqlerror.ErrorPosf(dir.Position, "Cannot redeclare directive %s.", dir.Name) + } + } + schema.Directives[dir.Name] = ast.Directives[i] + } + + if len(ast.Schema) > 1 { + return nil, gqlerror.ErrorPosf(ast.Schema[1].Position, "Cannot have multiple schema entry points, consider schema extensions instead.") + } + + if len(ast.Schema) == 1 { + schema.Description = ast.Schema[0].Description + for _, entrypoint := range ast.Schema[0].OperationTypes { + def := schema.Types[entrypoint.Type] + if def == nil { + return nil, gqlerror.ErrorPosf(entrypoint.Position, "Schema root %s refers to a type %s that does not exist.", entrypoint.Operation, entrypoint.Type) + } + switch entrypoint.Operation { + case Query: + schema.Query = def + case Mutation: + schema.Mutation = def + case Subscription: + schema.Subscription = def + } + } + } + + for _, ext := range ast.SchemaExtension { + for _, entrypoint := range ext.OperationTypes { + def := schema.Types[entrypoint.Type] + if def == nil { + return nil, gqlerror.ErrorPosf(entrypoint.Position, "Schema root %s refers to a type %s that does not exist.", entrypoint.Operation, entrypoint.Type) + } + switch entrypoint.Operation { + case Query: + schema.Query = def + case Mutation: + schema.Mutation = def + case Subscription: + schema.Subscription = def + } + } + } + + if err := validateTypeDefinitions(&schema); err != nil { + return nil, err + } + + if err := validateDirectiveDefinitions(&schema); err != nil { + return nil, err + } + + // Inferred root operation type names should be performed only when a `schema` directive is + // **not** provided, when it is, `Mutation` and `Subscription` becomes valid types and are not + // assigned as a root operation on the schema. + if len(ast.Schema) == 0 { + if schema.Query == nil && schema.Types["Query"] != nil { + schema.Query = schema.Types["Query"] + } + + if schema.Mutation == nil && schema.Types["Mutation"] != nil { + schema.Mutation = schema.Types["Mutation"] + } + + if schema.Subscription == nil && schema.Types["Subscription"] != nil { + schema.Subscription = schema.Types["Subscription"] + } + } + + if schema.Query != nil { + schema.Query.Fields = append( + schema.Query.Fields, + &FieldDefinition{ + Name: "__schema", + Type: NonNullNamedType("__Schema", nil), + }, + &FieldDefinition{ + Name: "__type", + Type: NamedType("__Type", nil), + Arguments: ArgumentDefinitionList{ + {Name: "name", Type: NonNullNamedType("String", nil)}, + }, + }, + ) + } + + return &schema, nil +} + +func validateTypeDefinitions(schema *Schema) *gqlerror.Error { + types := make([]string, 0, len(schema.Types)) + for typ := range schema.Types { + types = append(types, typ) + } + sort.Strings(types) + for _, typ := range types { + err := validateDefinition(schema, schema.Types[typ]) + if err != nil { + return err + } + } + return nil +} + +func validateDirectiveDefinitions(schema *Schema) *gqlerror.Error { + directives := make([]string, 0, len(schema.Directives)) + for directive := range schema.Directives { + directives = append(directives, directive) + } + sort.Strings(directives) + for _, directive := range directives { + err := validateDirective(schema, schema.Directives[directive]) + if err != nil { + return err + } + } + return nil +} + +func validateDirective(schema *Schema, def *DirectiveDefinition) *gqlerror.Error { + if err := validateName(def.Position, def.Name); err != nil { + // now, GraphQL spec doesn't have reserved directive name + return err + } + + return validateArgs(schema, def.Arguments, def) +} + +func validateDefinition(schema *Schema, def *Definition) *gqlerror.Error { + for _, field := range def.Fields { + if err := validateName(field.Position, field.Name); err != nil { + // now, GraphQL spec doesn't have reserved field name + return err + } + if err := validateTypeRef(schema, field.Type); err != nil { + return err + } + if err := validateArgs(schema, field.Arguments, nil); err != nil { + return err + } + wantDirLocation := LocationFieldDefinition + if def.Kind == InputObject { + wantDirLocation = LocationInputFieldDefinition + } + if err := validateDirectives(schema, field.Directives, wantDirLocation, nil); err != nil { + return err + } + } + + for _, typ := range def.Types { + typDef := schema.Types[typ] + if typDef == nil { + return gqlerror.ErrorPosf(def.Position, "Undefined type %s.", strconv.Quote(typ)) + } + if !isValidKind(typDef.Kind, Object) { + return gqlerror.ErrorPosf(def.Position, "%s type %s must be %s.", def.Kind, strconv.Quote(typ), kindList(Object)) + } + } + + for _, intf := range def.Interfaces { + if err := validateImplements(schema, def, intf); err != nil { + return err + } + } + + switch def.Kind { + case Object, Interface: + if len(def.Fields) == 0 { + return gqlerror.ErrorPosf(def.Position, "%s %s: must define one or more fields.", def.Kind, def.Name) + } + for _, field := range def.Fields { + if typ, ok := schema.Types[field.Type.Name()]; ok { + if !isValidKind(typ.Kind, Scalar, Object, Interface, Union, Enum) { + return gqlerror.ErrorPosf(field.Position, "%s %s: field must be one of %s.", def.Kind, def.Name, kindList(Scalar, Object, Interface, Union, Enum)) + } + } + } + case Enum: + if len(def.EnumValues) == 0 { + return gqlerror.ErrorPosf(def.Position, "%s %s: must define one or more unique enum values.", def.Kind, def.Name) + } + for _, value := range def.EnumValues { + for _, nonEnum := range [3]string{"true", "false", "null"} { + if value.Name == nonEnum { + return gqlerror.ErrorPosf(def.Position, "%s %s: non-enum value %s.", def.Kind, def.Name, value.Name) + } + } + } + case InputObject: + if len(def.Fields) == 0 { + return gqlerror.ErrorPosf(def.Position, "%s %s: must define one or more input fields.", def.Kind, def.Name) + } + for _, field := range def.Fields { + if typ, ok := schema.Types[field.Type.Name()]; ok { + if !isValidKind(typ.Kind, Scalar, Enum, InputObject) { + return gqlerror.ErrorPosf(field.Position, "%s %s: field must be one of %s.", typ.Kind, field.Name, kindList(Scalar, Enum, InputObject)) + } + } + } + } + + for idx, field1 := range def.Fields { + for _, field2 := range def.Fields[idx+1:] { + if field1.Name == field2.Name { + return gqlerror.ErrorPosf(field2.Position, "Field %s.%s can only be defined once.", def.Name, field2.Name) + } + } + } + + if !def.BuiltIn { + // GraphQL spec has reserved type names a lot! + err := validateName(def.Position, def.Name) + if err != nil { + return err + } + } + + return validateDirectives(schema, def.Directives, DirectiveLocation(def.Kind), nil) +} + +func validateTypeRef(schema *Schema, typ *Type) *gqlerror.Error { + if schema.Types[typ.Name()] == nil { + return gqlerror.ErrorPosf(typ.Position, "Undefined type %s.", typ.Name()) + } + return nil +} + +func validateArgs(schema *Schema, args ArgumentDefinitionList, currentDirective *DirectiveDefinition) *gqlerror.Error { + for _, arg := range args { + if err := validateName(arg.Position, arg.Name); err != nil { + // now, GraphQL spec doesn't have reserved argument name + return err + } + if err := validateTypeRef(schema, arg.Type); err != nil { + return err + } + def := schema.Types[arg.Type.Name()] + if !def.IsInputType() { + return gqlerror.ErrorPosf( + arg.Position, + "cannot use %s as argument %s because %s is not a valid input type", + arg.Type.String(), + arg.Name, + def.Kind, + ) + } + if err := validateDirectives(schema, arg.Directives, LocationArgumentDefinition, currentDirective); err != nil { + return err + } + } + return nil +} + +func validateDirectives(schema *Schema, dirs DirectiveList, location DirectiveLocation, currentDirective *DirectiveDefinition) *gqlerror.Error { + for _, dir := range dirs { + if err := validateName(dir.Position, dir.Name); err != nil { + // now, GraphQL spec doesn't have reserved directive name + return err + } + if currentDirective != nil && dir.Name == currentDirective.Name { + return gqlerror.ErrorPosf(dir.Position, "Directive %s cannot refer to itself.", currentDirective.Name) + } + if schema.Directives[dir.Name] == nil { + return gqlerror.ErrorPosf(dir.Position, "Undefined directive %s.", dir.Name) + } + validKind := false + for _, dirLocation := range schema.Directives[dir.Name].Locations { + if dirLocation == location { + validKind = true + break + } + } + if !validKind { + return gqlerror.ErrorPosf(dir.Position, "Directive %s is not applicable on %s.", dir.Name, location) + } + dir.Definition = schema.Directives[dir.Name] + } + return nil +} + +func validateImplements(schema *Schema, def *Definition, intfName string) *gqlerror.Error { + // see validation rules at the bottom of + // https://facebook.github.io/graphql/October2021/#sec-Objects + intf := schema.Types[intfName] + if intf == nil { + return gqlerror.ErrorPosf(def.Position, "Undefined type %s.", strconv.Quote(intfName)) + } + if intf.Kind != Interface { + return gqlerror.ErrorPosf(def.Position, "%s is a non interface type %s.", strconv.Quote(intfName), intf.Kind) + } + for _, requiredField := range intf.Fields { + foundField := def.Fields.ForName(requiredField.Name) + if foundField == nil { + return gqlerror.ErrorPosf(def.Position, + `For %s to implement %s it must have a field called %s.`, + def.Name, intf.Name, requiredField.Name, + ) + } + + if !isCovariant(schema, requiredField.Type, foundField.Type) { + return gqlerror.ErrorPosf(foundField.Position, + `For %s to implement %s the field %s must have type %s.`, + def.Name, intf.Name, requiredField.Name, requiredField.Type.String(), + ) + } + + for _, requiredArg := range requiredField.Arguments { + foundArg := foundField.Arguments.ForName(requiredArg.Name) + if foundArg == nil { + return gqlerror.ErrorPosf(foundField.Position, + `For %s to implement %s the field %s must have the same arguments but it is missing %s.`, + def.Name, intf.Name, requiredField.Name, requiredArg.Name, + ) + } + + if !requiredArg.Type.IsCompatible(foundArg.Type) { + return gqlerror.ErrorPosf(foundArg.Position, + `For %s to implement %s the field %s must have the same arguments but %s has the wrong type.`, + def.Name, intf.Name, requiredField.Name, requiredArg.Name, + ) + } + } + for _, foundArgs := range foundField.Arguments { + if requiredField.Arguments.ForName(foundArgs.Name) == nil && foundArgs.Type.NonNull && foundArgs.DefaultValue == nil { + return gqlerror.ErrorPosf(foundArgs.Position, + `For %s to implement %s any additional arguments on %s must be optional or have a default value but %s is required.`, + def.Name, intf.Name, foundField.Name, foundArgs.Name, + ) + } + } + } + return validateTypeImplementsAncestors(schema, def, intfName) +} + +// validateTypeImplementsAncestors +// https://github.com/graphql/graphql-js/blob/47bd8c8897c72d3efc17ecb1599a95cee6bac5e8/src/type/validate.ts#L428 +func validateTypeImplementsAncestors(schema *Schema, def *Definition, intfName string) *gqlerror.Error { + intf := schema.Types[intfName] + if intf == nil { + return gqlerror.ErrorPosf(def.Position, "Undefined type %s.", strconv.Quote(intfName)) + } + for _, transitive := range intf.Interfaces { + if !containsString(def.Interfaces, transitive) { + if transitive == def.Name { + return gqlerror.ErrorPosf(def.Position, + `Type %s cannot implement %s because it would create a circular reference.`, + def.Name, intfName, + ) + } + return gqlerror.ErrorPosf(def.Position, + `Type %s must implement %s because it is implemented by %s.`, + def.Name, transitive, intfName, + ) + } + } + return nil +} + +func containsString(slice []string, want string) bool { + for _, str := range slice { + if want == str { + return true + } + } + return false +} + +func isCovariant(schema *Schema, required *Type, actual *Type) bool { + if required.NonNull && !actual.NonNull { + return false + } + + if required.NamedType != "" { + if required.NamedType == actual.NamedType { + return true + } + for _, pt := range schema.PossibleTypes[required.NamedType] { + if pt.Name == actual.NamedType { + return true + } + } + return false + } + + if required.Elem != nil && actual.Elem == nil { + return false + } + + return isCovariant(schema, required.Elem, actual.Elem) +} + +func validateName(pos *Position, name string) *gqlerror.Error { + if strings.HasPrefix(name, "__") { + return gqlerror.ErrorPosf(pos, `Name "%s" must not begin with "__", which is reserved by GraphQL introspection.`, name) + } + return nil +} + +func isValidKind(kind DefinitionKind, valid ...DefinitionKind) bool { + for _, k := range valid { + if kind == k { + return true + } + } + return false +} + +func kindList(kinds ...DefinitionKind) string { + s := make([]string, len(kinds)) + for i, k := range kinds { + s[i] = string(k) + } + return strings.Join(s, ", ") +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/schema_test.yml b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/schema_test.yml new file mode 100644 index 00000000..7034a469 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/schema_test.yml @@ -0,0 +1,678 @@ +types: + - name: cannot be redeclared + input: | + type A { + name: String + } + type A { + name: String + } + error: + message: "Cannot redeclare type A." + locations: [{line: 4, column: 6}] + - name: cannot be duplicated field at same definition 1 + input: | + type A { + name: String + name: String + } + error: + message: "Field A.name can only be defined once." + locations: [{line: 3, column: 3}] + - name: cannot be duplicated field at same definition 2 + input: | + type A { + name: String + } + extend type A { + name: String + } + error: + message: "Field A.name can only be defined once." + locations: [{line: 5, column: 3}] + - name: cannot be duplicated field at same definition 3 + input: | + type A { + name: String + } + extend type A { + age: Int + age: Int + } + error: + message: "Field A.age can only be defined once." + locations: [{line: 6, column: 3}] + +object types: + - name: must define one or more fields + input: | + directive @D on OBJECT + + # This pattern rejected by parser + # type InvalidObject1 {} + + type InvalidObject2 @D + + type ValidObject { + id: ID + } + extend type ValidObject @D + extend type ValidObject { + b: Int + } + error: + message: 'OBJECT InvalidObject2: must define one or more fields.' + locations: [{line: 6, column: 6}] + - name: check reserved names on type name + input: | + type __FooBar { + id: ID + } + error: + message: 'Name "__FooBar" must not begin with "__", which is reserved by GraphQL introspection.' + locations: [{line: 1, column: 6}] + - name: check reserved names on type field + input: | + type FooBar { + __id: ID + } + error: + message: 'Name "__id" must not begin with "__", which is reserved by GraphQL introspection.' + locations: [{line: 2, column: 3}] + + - name: check reserved names on type field argument + input: | + type FooBar { + foo(__bar: ID): ID + } + error: + message: 'Name "__bar" must not begin with "__", which is reserved by GraphQL introspection.' + locations: [{line: 2, column: 7}] + + - name: must not allow input object as field type + input: | + input Input { + id: ID + } + type Query { + input: Input! + } + error: + message: 'OBJECT Query: field must be one of SCALAR, OBJECT, INTERFACE, UNION, ENUM.' + locations: [{line: 5, column: 3}] + +interfaces: + - name: must exist + input: | + type Thing implements Object { + id: ID! + } + + type Query { + Things: [Thing!]! + } + error: + message: 'Undefined type "Object".' + locations: [{line: 1, column: 6}] + + - name: must be an interface + input: | + type Thing implements Object { + id: ID! + } + + type Query { + Things: [Thing!]! + } + + type Object { + name: String + } + error: + message: '"Object" is a non interface type OBJECT.' + locations: [{line: 1, column: 6}] + + - name: must define one or more fields + input: | + directive @D on INTERFACE + + # This pattern rejected by parser + # interface InvalidInterface1 {} + + interface InvalidInterface2 @D + + interface ValidInterface { + id: ID + } + extend interface ValidInterface @D + extend interface ValidInterface { + b: Int + } + error: + message: 'INTERFACE InvalidInterface2: must define one or more fields.' + locations: [{line: 6, column: 11}] + + - name: check reserved names on type name + input: | + interface __FooBar { + id: ID + } + error: + message: 'Name "__FooBar" must not begin with "__", which is reserved by GraphQL introspection.' + locations: [{line: 1, column: 11}] + + - name: must not allow input object as field type + input: | + input Input { + id: ID + } + type Query { + foo: Foo! + } + interface Foo { + input: Input! + } + error: + message: 'INTERFACE Foo: field must be one of SCALAR, OBJECT, INTERFACE, UNION, ENUM.' + locations: [{line: 8, column: 3}] + + - name: must have all fields from interface + input: | + type Bar implements BarInterface { + someField: Int! + } + + interface BarInterface { + id: ID! + } + error: + message: 'For Bar to implement BarInterface it must have a field called id.' + locations: [{line: 1, column: 6}] + + - name: must have same type of fields + input: | + type Bar implements BarInterface { + id: Int! + } + + interface BarInterface { + id: ID! + } + error: + message: 'For Bar to implement BarInterface the field id must have type ID!.' + locations: [{line: 2, column: 5}] + + - name: must have all required arguments + input: | + type Bar implements BarInterface { + id: ID! + } + + interface BarInterface { + id(ff: Int!): ID! + } + error: + message: 'For Bar to implement BarInterface the field id must have the same arguments but it is missing ff.' + locations: [{line: 2, column: 5}] + + - name: must have same argument types + input: | + type Bar implements BarInterface { + id(ff: ID!): ID! + } + + interface BarInterface { + id(ff: Int!): ID! + } + error: + message: 'For Bar to implement BarInterface the field id must have the same arguments but ff has the wrong type.' + locations: [{line: 2, column: 8}] + + - name: may defined additional nullable arguments + input: | + type Bar implements BarInterface { + id(opt: Int): ID! + } + + interface BarInterface { + id: ID! + } + + - name: may defined additional required arguments with defaults + input: | + type Bar implements BarInterface { + id(opt: Int! = 1): ID! + } + + interface BarInterface { + id: ID! + } + + - name: must not define additional required arguments without defaults + input: | + type Bar implements BarInterface { + id(opt: Int!): ID! + } + + interface BarInterface { + id: ID! + } + error: + message: 'For Bar to implement BarInterface any additional arguments on id must be optional or have a default value but opt is required.' + locations: [{line: 2, column: 8}] + + - name: can have covariant argument types + input: | + union U = A|B + + type A { name: String } + type B { name: String } + + type Bar implements BarInterface { + f: A! + } + + interface BarInterface { + f: U! + } + + - name: may define intermediate interfaces + input: | + interface IA { + id: ID! + } + + interface IIA implements IA { + id: ID! + } + + type A implements IIA & IA { + id: ID! + } + + - name: Type Foo must implement Baz because it is implemented by Bar + input: | + interface Baz { + baz: String + } + + interface Bar implements Baz { + bar: String + baz: String + } + + type Foo implements Bar { + foo: String + bar: String + baz: String + } + error: + message: 'Type Foo must implement Baz because it is implemented by Bar.' + locations: [{line: 10, column: 6}] + + - name: circular reference error + input: | + interface Circular1 implements Circular2 { + id: ID! + } + + interface Circular2 implements Circular1 { + id: ID! + } + error: + message: 'Type Circular1 cannot implement Circular2 because it would create a circular reference.' + locations: [{line: 1, column: 11}] + +inputs: + - name: must define one or more input fields + input: | + directive @D on INPUT_OBJECT + + # This pattern rejected by parser + # input InvalidInput1 {} + + input InvalidInput2 @D + + input ValidInput { + id: ID + } + extend input ValidInput @D + extend input ValidInput { + b: Int + } + error: + message: 'INPUT_OBJECT InvalidInput2: must define one or more input fields.' + locations: [{line: 6, column: 7}] + - name: check reserved names on type name + input: | + input __FooBar { + id: ID + } + error: + message: 'Name "__FooBar" must not begin with "__", which is reserved by GraphQL introspection.' + locations: [{line: 1, column: 7}] + + - name: fields cannot be Objects + input: | + type Object { id: ID } + input Foo { a: Object! } + error: + message: 'OBJECT a: field must be one of SCALAR, ENUM, INPUT_OBJECT.' + locations: [{line: 2, column: 13}] + + - name: fields cannot be Interfaces + input: | + interface Interface { id: ID! } + input Foo { a: Interface! } + error: + message: 'INTERFACE a: field must be one of SCALAR, ENUM, INPUT_OBJECT.' + locations: [{line: 2, column: 13}] + + - name: fields cannot be Unions + input: | + type Object { id: ID } + union Union = Object + input Foo { a: Union! } + error: + message: 'UNION a: field must be one of SCALAR, ENUM, INPUT_OBJECT.' + locations: [{line: 3, column: 13}] + +args: + - name: Valid arg types + input: | + input Input { id: ID } + enum Enum { A } + scalar Scalar + + type Query { + f(a: Input, b: Scalar, c: Enum): Boolean! + } + + - name: Objects not allowed + input: | + type Object { id: ID } + type Query { f(a: Object): Boolean! } + + error: + message: 'cannot use Object as argument a because OBJECT is not a valid input type' + locations: [{line: 2, column: 16}] + + - name: Union not allowed + input: | + type Object { id: ID } + union Union = Object + type Query { f(a: Union): Boolean! } + + error: + message: 'cannot use Union as argument a because UNION is not a valid input type' + locations: [{line: 3, column: 16}] + + - name: Interface not allowed + input: | + interface Interface { id: ID } + type Query { f(a: Interface): Boolean! } + + error: + message: 'cannot use Interface as argument a because INTERFACE is not a valid input type' + locations: [{line: 2, column: 16}] + +enums: + - name: must define one or more unique enum values + input: | + directive @D on ENUM + + # This pattern rejected by parser + # enum InvalidEmum1 {} + + enum InvalidEnum2 @D + + enum ValidEnum { + FOO + } + extend enum ValidEnum @D + extend enum ValidEnum { + BAR + } + error: + message: 'ENUM InvalidEnum2: must define one or more unique enum values.' + locations: [{line: 6, column: 6}] + - name: check reserved names on type name + input: | + enum __FooBar { + A + B + } + error: + message: 'Name "__FooBar" must not begin with "__", which is reserved by GraphQL introspection.' + locations: [{line: 1, column: 6}] + +unions: + - name: union types must be defined + input: | + union Foo = Bar | Baz + type Bar { + id: ID + } + error: + message: "Undefined type \"Baz\"." + locations: [{line: 1, column: 7}] + - name: union types must be objects + input: | + union Foo = Baz + interface Baz { + id: ID + } + error: + message: "UNION type \"Baz\" must be OBJECT." + locations: [{line: 1, column: 7}] + + - name: unions of pure type extensions are valid + input: | + + type Review { + body: String! + author: User! @provides(fields: "username") + product: Product! + } + + extend type User @key(fields: "id") { + id: ID! @external + reviews: [Review] + } + + extend type Product @key(fields: "upc") { + upc: String! @external + reviews: [Review] + } + + union Foo = User | Product + scalar _Any + scalar _FieldSet + directive @external on FIELD_DEFINITION + directive @requires(fields: _FieldSet!) on FIELD_DEFINITION + directive @provides(fields: _FieldSet!) on FIELD_DEFINITION + directive @key(fields: _FieldSet!) on OBJECT | INTERFACE + directive @extends on OBJECT + + + +type extensions: + - name: can extend non existant types + input: | + extend type A { + name: String + } + + + - name: cannot extend incorret type existant types + input: | + scalar A + extend type A { + name: String + } + error: + message: "Cannot extend type A because the base type is a SCALAR, not OBJECT." + locations: [{line: 2, column: 13}] + +directives: + - name: cannot redeclare directives + input: | + directive @A on FIELD_DEFINITION + directive @A on FIELD_DEFINITION + error: + message: "Cannot redeclare directive A." + locations: [{line: 2, column: 12}] + + - name: can redeclare builtin directives + input: | + directive @skip(if: Boolean!) on FIELD | FRAGMENT_SPREAD | INLINE_FRAGMENT + directive @skip(if: Boolean!) on FIELD | FRAGMENT_SPREAD | INLINE_FRAGMENT + + - name: must be declared + input: | + type User { + name: String @foo + } + error: + message: "Undefined directive foo." + locations: [{line: 2, column: 17}] + + - name: cannot be self-referential + input: | + directive @A(foo: Int! @A) on FIELD_DEFINITION + error: + message: "Directive A cannot refer to itself." + locations: [{line: 1, column: 25}] + - name: check reserved names on type name + input: | + directive @__A on FIELD_DEFINITION + error: + message: 'Name "__A" must not begin with "__", which is reserved by GraphQL introspection.' + locations: [{line: 1, column: 12}] + + - name: Valid arg types + input: | + input Input { id: ID } + enum Enum { A } + scalar Scalar + + directive @A(a: Input, b: Scalar, c: Enum) on FIELD_DEFINITION + + - name: Objects not allowed + input: | + type Object { id: ID } + directive @A(a: Object) on FIELD_DEFINITION + + error: + message: 'cannot use Object as argument a because OBJECT is not a valid input type' + locations: [{line: 2, column: 14}] + + - name: Union not allowed + input: | + type Object { id: ID } + union Union = Object + directive @A(a: Union) on FIELD_DEFINITION + + error: + message: 'cannot use Union as argument a because UNION is not a valid input type' + locations: [{line: 3, column: 14}] + + - name: Interface not allowed + input: | + interface Interface { id: ID } + directive @A(a: Interface) on FIELD_DEFINITION + + error: + message: 'cannot use Interface as argument a because INTERFACE is not a valid input type' + locations: [{line: 2, column: 14}] + + - name: Invalid location usage not allowed + input: | + directive @test on FIELD_DEFINITION + input I1 @test { f: String } + + error: + message: 'Directive test is not applicable on INPUT_OBJECT.' + locations: [{line: 2, column: 11}] + + - name: Valid location usage + input: | + directive @testInputField on INPUT_FIELD_DEFINITION + directive @testField on FIELD_DEFINITION + directive @inp on INPUT_OBJECT + input I1 @inp { f: String @testInputField } + type P { name: String @testField } + interface I { id: ID @testField } + + +entry points: + - name: multiple schema entry points + input: | + schema { + query: Query + } + schema { + query: Query + } + scalar Query + error: + message: "Cannot have multiple schema entry points, consider schema extensions instead." + locations: [{line: 4, column: 8}] + + - name: Undefined schema entrypoint + input: | + schema { + query: Query + } + error: + message: "Schema root query refers to a type Query that does not exist." + locations: [{line: 2, column: 3}] + +entry point extensions: + - name: Undefined schema entrypoint + input: | + schema { + query: Query + } + scalar Query + extend schema { + mutation: Mutation + } + error: + message: "Schema root mutation refers to a type Mutation that does not exist." + locations: [{line: 6, column: 3}] + +type references: + - name: Field types + input: | + type User { + posts: Post + } + error: + message: "Undefined type Post." + locations: [{line: 2, column: 10}] + + - name: Arg types + input: | + type User { + posts(foo: FooBar): String + } + error: + message: "Undefined type FooBar." + locations: [{line: 2, column: 14}] + + - name: Directive arg types + input: | + directive @Foo(foo: FooBar) on FIELD_DEFINITION + + error: + message: "Undefined type FooBar." + locations: [{line: 1, column: 21}] + + - name: Invalid enum value + input: | + enum Enum { true } + + error: + message: "ENUM Enum: non-enum value true." + locations: [{line: 1, column: 6}] diff --git a/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/suggestionList.go b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/suggestionList.go new file mode 100644 index 00000000..f0bbc327 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/suggestionList.go @@ -0,0 +1,69 @@ +package validator + +import ( + "math" + "sort" + "strings" + + "github.com/agnivade/levenshtein" +) + +// Given an invalid input string and a list of valid options, returns a filtered +// list of valid options sorted based on their similarity with the input. +func SuggestionList(input string, options []string) []string { + var results []string + optionsByDistance := map[string]int{} + + for _, option := range options { + distance := lexicalDistance(input, option) + threshold := calcThreshold(input) + if distance <= threshold { + results = append(results, option) + optionsByDistance[option] = distance + } + } + + sort.Slice(results, func(i, j int) bool { + return optionsByDistance[results[i]] < optionsByDistance[results[j]] + }) + return results +} + +func calcThreshold(a string) (threshold int) { + // the logic is copied from here + // https://github.com/graphql/graphql-js/blob/47bd8c8897c72d3efc17ecb1599a95cee6bac5e8/src/jsutils/suggestionList.ts#L14 + threshold = int(math.Floor(float64(len(a))*0.4) + 1) + + if threshold < 1 { + threshold = 1 + } + return +} + +// Computes the lexical distance between strings A and B. +// +// The "distance" between two strings is given by counting the minimum number +// of edits needed to transform string A into string B. An edit can be an +// insertion, deletion, or substitution of a single character, or a swap of two +// adjacent characters. +// +// Includes a custom alteration from Damerau-Levenshtein to treat case changes +// as a single edit which helps identify mis-cased values with an edit distance +// of 1. +// +// This distance can be useful for detecting typos in input or sorting +func lexicalDistance(a, b string) int { + if a == b { + return 0 + } + + a = strings.ToLower(a) + b = strings.ToLower(b) + + // Any case change counts as a single edit + if a == b { + return 1 + } + + return levenshtein.ComputeDistance(a, b) +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/validator.go b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/validator.go new file mode 100644 index 00000000..05f5b916 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/validator.go @@ -0,0 +1,45 @@ +package validator + +import ( + //nolint:revive + . "github.com/open-policy-agent/opa/internal/gqlparser/ast" + "github.com/open-policy-agent/opa/internal/gqlparser/gqlerror" +) + +type AddErrFunc func(options ...ErrorOption) + +type ruleFunc func(observers *Events, addError AddErrFunc) + +type rule struct { + name string + rule ruleFunc +} + +var rules []rule + +// addRule to rule set. +// f is called once each time `Validate` is executed. +func AddRule(name string, f ruleFunc) { + rules = append(rules, rule{name: name, rule: f}) +} + +func Validate(schema *Schema, doc *QueryDocument) gqlerror.List { + var errs gqlerror.List + + observers := &Events{} + for i := range rules { + rule := rules[i] + rule.rule(observers, func(options ...ErrorOption) { + err := &gqlerror.Error{ + Rule: rule.name, + } + for _, o := range options { + o(err) + } + errs = append(errs, err) + }) + } + + Walk(schema, doc, observers) + return errs +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/vars.go b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/vars.go new file mode 100644 index 00000000..86be7c4d --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/vars.go @@ -0,0 +1,258 @@ +package validator + +import ( + "encoding/json" + "fmt" + "reflect" + "strconv" + "strings" + + "github.com/open-policy-agent/opa/internal/gqlparser/ast" + "github.com/open-policy-agent/opa/internal/gqlparser/gqlerror" +) + +var ErrUnexpectedType = fmt.Errorf("Unexpected Type") + +// VariableValues coerces and validates variable values +func VariableValues(schema *ast.Schema, op *ast.OperationDefinition, variables map[string]interface{}) (map[string]interface{}, error) { + coercedVars := map[string]interface{}{} + + validator := varValidator{ + path: ast.Path{ast.PathName("variable")}, + schema: schema, + } + + for _, v := range op.VariableDefinitions { + validator.path = append(validator.path, ast.PathName(v.Variable)) + + if !v.Definition.IsInputType() { + return nil, gqlerror.ErrorPathf(validator.path, "must an input type") + } + + val, hasValue := variables[v.Variable] + + if !hasValue { + if v.DefaultValue != nil { + var err error + val, err = v.DefaultValue.Value(nil) + if err != nil { + return nil, gqlerror.WrapPath(validator.path, err) + } + hasValue = true + } else if v.Type.NonNull { + return nil, gqlerror.ErrorPathf(validator.path, "must be defined") + } + } + + if hasValue { + if val == nil { + if v.Type.NonNull { + return nil, gqlerror.ErrorPathf(validator.path, "cannot be null") + } + coercedVars[v.Variable] = nil + } else { + rv := reflect.ValueOf(val) + + jsonNumber, isJSONNumber := val.(json.Number) + if isJSONNumber { + if v.Type.NamedType == "Int" { + n, err := jsonNumber.Int64() + if err != nil { + return nil, gqlerror.ErrorPathf(validator.path, "cannot use value %d as %s", n, v.Type.NamedType) + } + rv = reflect.ValueOf(n) + } else if v.Type.NamedType == "Float" { + f, err := jsonNumber.Float64() + if err != nil { + return nil, gqlerror.ErrorPathf(validator.path, "cannot use value %f as %s", f, v.Type.NamedType) + } + rv = reflect.ValueOf(f) + + } + } + if rv.Kind() == reflect.Ptr || rv.Kind() == reflect.Interface { + rv = rv.Elem() + } + + rval, err := validator.validateVarType(v.Type, rv) + if err != nil { + return nil, err + } + coercedVars[v.Variable] = rval.Interface() + } + } + + validator.path = validator.path[0 : len(validator.path)-1] + } + return coercedVars, nil +} + +type varValidator struct { + path ast.Path + schema *ast.Schema +} + +func (v *varValidator) validateVarType(typ *ast.Type, val reflect.Value) (reflect.Value, *gqlerror.Error) { + currentPath := v.path + resetPath := func() { + v.path = currentPath + } + defer resetPath() + if typ.Elem != nil { + if val.Kind() != reflect.Slice { + // GraphQL spec says that non-null values should be coerced to an array when possible. + // Hence if the value is not a slice, we create a slice and add val to it. + slc := reflect.MakeSlice(reflect.SliceOf(val.Type()), 0, 0) + slc = reflect.Append(slc, val) + val = slc + } + for i := 0; i < val.Len(); i++ { + resetPath() + v.path = append(v.path, ast.PathIndex(i)) + field := val.Index(i) + if field.Kind() == reflect.Ptr || field.Kind() == reflect.Interface { + if typ.Elem.NonNull && field.IsNil() { + return val, gqlerror.ErrorPathf(v.path, "cannot be null") + } + field = field.Elem() + } + _, err := v.validateVarType(typ.Elem, field) + if err != nil { + return val, err + } + } + return val, nil + } + def := v.schema.Types[typ.NamedType] + if def == nil { + panic(fmt.Errorf("missing def for %s", typ.NamedType)) + } + + if !typ.NonNull && !val.IsValid() { + // If the type is not null and we got a invalid value namely null/nil, then it's valid + return val, nil + } + + switch def.Kind { + case ast.Enum: + kind := val.Type().Kind() + if kind != reflect.Int && kind != reflect.Int32 && kind != reflect.Int64 && kind != reflect.String { + return val, gqlerror.ErrorPathf(v.path, "enums must be ints or strings") + } + isValidEnum := false + for _, enumVal := range def.EnumValues { + if strings.EqualFold(val.String(), enumVal.Name) { + isValidEnum = true + } + } + if !isValidEnum { + return val, gqlerror.ErrorPathf(v.path, "%s is not a valid %s", val.String(), def.Name) + } + return val, nil + case ast.Scalar: + kind := val.Type().Kind() + switch typ.NamedType { + case "Int": + if kind == reflect.Int || kind == reflect.Int32 || kind == reflect.Int64 || kind == reflect.Float32 || kind == reflect.Float64 || IsValidIntString(val, kind) { + return val, nil + } + case "Float": + if kind == reflect.Float32 || kind == reflect.Float64 || kind == reflect.Int || kind == reflect.Int32 || kind == reflect.Int64 || IsValidFloatString(val, kind) { + return val, nil + } + case "String": + if kind == reflect.String { + return val, nil + } + + case "Boolean": + if kind == reflect.Bool { + return val, nil + } + + case "ID": + if kind == reflect.Int || kind == reflect.Int32 || kind == reflect.Int64 || kind == reflect.String { + return val, nil + } + default: + // assume custom scalars are ok + return val, nil + } + return val, gqlerror.ErrorPathf(v.path, "cannot use %s as %s", kind.String(), typ.NamedType) + case ast.InputObject: + if val.Kind() != reflect.Map { + return val, gqlerror.ErrorPathf(v.path, "must be a %s", def.Name) + } + + // check for unknown fields + for _, name := range val.MapKeys() { + val.MapIndex(name) + fieldDef := def.Fields.ForName(name.String()) + resetPath() + v.path = append(v.path, ast.PathName(name.String())) + + switch { + case name.String() == "__typename": + continue + case fieldDef == nil: + return val, gqlerror.ErrorPathf(v.path, "unknown field") + } + } + + for _, fieldDef := range def.Fields { + resetPath() + v.path = append(v.path, ast.PathName(fieldDef.Name)) + + field := val.MapIndex(reflect.ValueOf(fieldDef.Name)) + if !field.IsValid() { + if fieldDef.Type.NonNull { + if fieldDef.DefaultValue != nil { + var err error + _, err = fieldDef.DefaultValue.Value(nil) + if err == nil { + continue + } + } + return val, gqlerror.ErrorPathf(v.path, "must be defined") + } + continue + } + + if field.Kind() == reflect.Ptr || field.Kind() == reflect.Interface { + if fieldDef.Type.NonNull && field.IsNil() { + return val, gqlerror.ErrorPathf(v.path, "cannot be null") + } + //allow null object field and skip it + if !fieldDef.Type.NonNull && field.IsNil() { + continue + } + field = field.Elem() + } + cval, err := v.validateVarType(fieldDef.Type, field) + if err != nil { + return val, err + } + val.SetMapIndex(reflect.ValueOf(fieldDef.Name), cval) + } + default: + panic(fmt.Errorf("unsupported type %s", def.Kind)) + } + return val, nil +} + +func IsValidIntString(val reflect.Value, kind reflect.Kind) bool { + if kind != reflect.String { + return false + } + _, e := strconv.ParseInt(fmt.Sprintf("%v", val.Interface()), 10, 64) + + return e == nil +} + +func IsValidFloatString(val reflect.Value, kind reflect.Kind) bool { + if kind != reflect.String { + return false + } + _, e := strconv.ParseFloat(fmt.Sprintf("%v", val.Interface()), 64) + return e == nil +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/walk.go b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/walk.go new file mode 100644 index 00000000..f7228718 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/gqlparser/validator/walk.go @@ -0,0 +1,292 @@ +package validator + +import ( + "context" + "fmt" + + "github.com/open-policy-agent/opa/internal/gqlparser/ast" +) + +type Events struct { + operationVisitor []func(walker *Walker, operation *ast.OperationDefinition) + field []func(walker *Walker, field *ast.Field) + fragment []func(walker *Walker, fragment *ast.FragmentDefinition) + inlineFragment []func(walker *Walker, inlineFragment *ast.InlineFragment) + fragmentSpread []func(walker *Walker, fragmentSpread *ast.FragmentSpread) + directive []func(walker *Walker, directive *ast.Directive) + directiveList []func(walker *Walker, directives []*ast.Directive) + value []func(walker *Walker, value *ast.Value) + variable []func(walker *Walker, variable *ast.VariableDefinition) +} + +func (o *Events) OnOperation(f func(walker *Walker, operation *ast.OperationDefinition)) { + o.operationVisitor = append(o.operationVisitor, f) +} +func (o *Events) OnField(f func(walker *Walker, field *ast.Field)) { + o.field = append(o.field, f) +} +func (o *Events) OnFragment(f func(walker *Walker, fragment *ast.FragmentDefinition)) { + o.fragment = append(o.fragment, f) +} +func (o *Events) OnInlineFragment(f func(walker *Walker, inlineFragment *ast.InlineFragment)) { + o.inlineFragment = append(o.inlineFragment, f) +} +func (o *Events) OnFragmentSpread(f func(walker *Walker, fragmentSpread *ast.FragmentSpread)) { + o.fragmentSpread = append(o.fragmentSpread, f) +} +func (o *Events) OnDirective(f func(walker *Walker, directive *ast.Directive)) { + o.directive = append(o.directive, f) +} +func (o *Events) OnDirectiveList(f func(walker *Walker, directives []*ast.Directive)) { + o.directiveList = append(o.directiveList, f) +} +func (o *Events) OnValue(f func(walker *Walker, value *ast.Value)) { + o.value = append(o.value, f) +} +func (o *Events) OnVariable(f func(walker *Walker, variable *ast.VariableDefinition)) { + o.variable = append(o.variable, f) +} + +func Walk(schema *ast.Schema, document *ast.QueryDocument, observers *Events) { + w := Walker{ + Observers: observers, + Schema: schema, + Document: document, + } + + w.walk() +} + +type Walker struct { + Context context.Context + Observers *Events + Schema *ast.Schema + Document *ast.QueryDocument + + validatedFragmentSpreads map[string]bool + CurrentOperation *ast.OperationDefinition +} + +func (w *Walker) walk() { + for _, child := range w.Document.Operations { + w.validatedFragmentSpreads = make(map[string]bool) + w.walkOperation(child) + } + for _, child := range w.Document.Fragments { + w.validatedFragmentSpreads = make(map[string]bool) + w.walkFragment(child) + } +} + +func (w *Walker) walkOperation(operation *ast.OperationDefinition) { + w.CurrentOperation = operation + for _, varDef := range operation.VariableDefinitions { + varDef.Definition = w.Schema.Types[varDef.Type.Name()] + for _, v := range w.Observers.variable { + v(w, varDef) + } + if varDef.DefaultValue != nil { + varDef.DefaultValue.ExpectedType = varDef.Type + varDef.DefaultValue.Definition = w.Schema.Types[varDef.Type.Name()] + } + } + + var def *ast.Definition + var loc ast.DirectiveLocation + switch operation.Operation { + case ast.Query, "": + def = w.Schema.Query + loc = ast.LocationQuery + case ast.Mutation: + def = w.Schema.Mutation + loc = ast.LocationMutation + case ast.Subscription: + def = w.Schema.Subscription + loc = ast.LocationSubscription + } + + for _, varDef := range operation.VariableDefinitions { + if varDef.DefaultValue != nil { + w.walkValue(varDef.DefaultValue) + } + w.walkDirectives(varDef.Definition, varDef.Directives, ast.LocationVariableDefinition) + } + + w.walkDirectives(def, operation.Directives, loc) + w.walkSelectionSet(def, operation.SelectionSet) + + for _, v := range w.Observers.operationVisitor { + v(w, operation) + } + w.CurrentOperation = nil +} + +func (w *Walker) walkFragment(it *ast.FragmentDefinition) { + def := w.Schema.Types[it.TypeCondition] + + it.Definition = def + + w.walkDirectives(def, it.Directives, ast.LocationFragmentDefinition) + w.walkSelectionSet(def, it.SelectionSet) + + for _, v := range w.Observers.fragment { + v(w, it) + } +} + +func (w *Walker) walkDirectives(parentDef *ast.Definition, directives []*ast.Directive, location ast.DirectiveLocation) { + for _, dir := range directives { + def := w.Schema.Directives[dir.Name] + dir.Definition = def + dir.ParentDefinition = parentDef + dir.Location = location + + for _, arg := range dir.Arguments { + var argDef *ast.ArgumentDefinition + if def != nil { + argDef = def.Arguments.ForName(arg.Name) + } + + w.walkArgument(argDef, arg) + } + + for _, v := range w.Observers.directive { + v(w, dir) + } + } + + for _, v := range w.Observers.directiveList { + v(w, directives) + } +} + +func (w *Walker) walkValue(value *ast.Value) { + if value.Kind == ast.Variable && w.CurrentOperation != nil { + value.VariableDefinition = w.CurrentOperation.VariableDefinitions.ForName(value.Raw) + if value.VariableDefinition != nil { + value.VariableDefinition.Used = true + } + } + + if value.Kind == ast.ObjectValue { + for _, child := range value.Children { + if value.Definition != nil { + fieldDef := value.Definition.Fields.ForName(child.Name) + if fieldDef != nil { + child.Value.ExpectedType = fieldDef.Type + child.Value.Definition = w.Schema.Types[fieldDef.Type.Name()] + } + } + w.walkValue(child.Value) + } + } + + if value.Kind == ast.ListValue { + for _, child := range value.Children { + if value.ExpectedType != nil && value.ExpectedType.Elem != nil { + child.Value.ExpectedType = value.ExpectedType.Elem + child.Value.Definition = value.Definition + } + + w.walkValue(child.Value) + } + } + + for _, v := range w.Observers.value { + v(w, value) + } +} + +func (w *Walker) walkArgument(argDef *ast.ArgumentDefinition, arg *ast.Argument) { + if argDef != nil { + arg.Value.ExpectedType = argDef.Type + arg.Value.Definition = w.Schema.Types[argDef.Type.Name()] + } + + w.walkValue(arg.Value) +} + +func (w *Walker) walkSelectionSet(parentDef *ast.Definition, it ast.SelectionSet) { + for _, child := range it { + w.walkSelection(parentDef, child) + } +} + +func (w *Walker) walkSelection(parentDef *ast.Definition, it ast.Selection) { + switch it := it.(type) { + case *ast.Field: + var def *ast.FieldDefinition + if it.Name == "__typename" { + def = &ast.FieldDefinition{ + Name: "__typename", + Type: ast.NamedType("String", nil), + } + } else if parentDef != nil { + def = parentDef.Fields.ForName(it.Name) + } + + it.Definition = def + it.ObjectDefinition = parentDef + + var nextParentDef *ast.Definition + if def != nil { + nextParentDef = w.Schema.Types[def.Type.Name()] + } + + for _, arg := range it.Arguments { + var argDef *ast.ArgumentDefinition + if def != nil { + argDef = def.Arguments.ForName(arg.Name) + } + + w.walkArgument(argDef, arg) + } + + w.walkDirectives(nextParentDef, it.Directives, ast.LocationField) + w.walkSelectionSet(nextParentDef, it.SelectionSet) + + for _, v := range w.Observers.field { + v(w, it) + } + + case *ast.InlineFragment: + it.ObjectDefinition = parentDef + + nextParentDef := parentDef + if it.TypeCondition != "" { + nextParentDef = w.Schema.Types[it.TypeCondition] + } + + w.walkDirectives(nextParentDef, it.Directives, ast.LocationInlineFragment) + w.walkSelectionSet(nextParentDef, it.SelectionSet) + + for _, v := range w.Observers.inlineFragment { + v(w, it) + } + + case *ast.FragmentSpread: + def := w.Document.Fragments.ForName(it.Name) + it.Definition = def + it.ObjectDefinition = parentDef + + var nextParentDef *ast.Definition + if def != nil { + nextParentDef = w.Schema.Types[def.TypeCondition] + } + + w.walkDirectives(nextParentDef, it.Directives, ast.LocationFragmentSpread) + + if def != nil && !w.validatedFragmentSpreads[def.Name] { + // prevent inifinite recursion + w.validatedFragmentSpreads[def.Name] = true + w.walkSelectionSet(nextParentDef, def.SelectionSet) + } + + for _, v := range w.Observers.fragmentSpread { + v(w, it) + } + + default: + panic(fmt.Errorf("unsupported %T", it)) + } +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/json/patch/patch.go b/vendor/github.com/open-policy-agent/opa/internal/json/patch/patch.go new file mode 100644 index 00000000..31c89869 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/json/patch/patch.go @@ -0,0 +1,45 @@ +// Copyright 2021 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package patch + +import ( + "strings" + + "github.com/open-policy-agent/opa/storage" +) + +// ParsePatchPathEscaped returns a new path for the given escaped str. +// This is based on storage.ParsePathEscaped so will do URL unescaping of +// the provided str for backwards compatibility, but also handles the +// specific escape strings defined in RFC 6901 (JSON Pointer) because +// that's what's mandated by RFC 6902 (JSON Patch). +func ParsePatchPathEscaped(str string) (path storage.Path, ok bool) { + path, ok = storage.ParsePathEscaped(str) + if !ok { + return + } + for i := range path { + // RFC 6902 section 4: "[The "path" member's] value is a string containing + // a JSON-Pointer value [RFC6901] that references a location within the + // target document (the "target location") where the operation is performed." + // + // RFC 6901 section 3: "Because the characters '~' (%x7E) and '/' (%x2F) + // have special meanings in JSON Pointer, '~' needs to be encoded as '~0' + // and '/' needs to be encoded as '~1' when these characters appear in a + // reference token." + + // RFC 6901 section 4: "Evaluation of each reference token begins by + // decoding any escaped character sequence. This is performed by first + // transforming any occurrence of the sequence '~1' to '/', and then + // transforming any occurrence of the sequence '~0' to '~'. By performing + // the substitutions in this order, an implementation avoids the error of + // turning '~01' first into '~1' and then into '/', which would be + // incorrect (the string '~01' correctly becomes '~1' after transformation)." + path[i] = strings.Replace(path[i], "~1", "/", -1) + path[i] = strings.Replace(path[i], "~0", "~", -1) + } + + return +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/jwx/buffer/buffer.go b/vendor/github.com/open-policy-agent/opa/internal/jwx/buffer/buffer.go index ca4ac419..c383ff3b 100644 --- a/vendor/github.com/open-policy-agent/opa/internal/jwx/buffer/buffer.go +++ b/vendor/github.com/open-policy-agent/opa/internal/jwx/buffer/buffer.go @@ -7,8 +7,7 @@ import ( "encoding/base64" "encoding/binary" "encoding/json" - - "github.com/pkg/errors" + "fmt" ) // Buffer wraps `[]byte` and provides functions that are often used in @@ -35,7 +34,7 @@ func FromUint(v uint64) Buffer { func FromBase64(v []byte) (Buffer, error) { b := Buffer{} if err := b.Base64Decode(v); err != nil { - return Buffer(nil), errors.Wrap(err, "failed to decode from base64") + return Buffer(nil), fmt.Errorf("failed to decode from base64: %w", err) } return b, nil @@ -85,7 +84,7 @@ func (b *Buffer) Base64Decode(v []byte) error { out := make([]byte, enc.DecodedLen(len(v))) n, err := enc.Decode(out, v) if err != nil { - return errors.Wrap(err, "failed to decode from base64") + return fmt.Errorf("failed to decode from base64: %w", err) } out = out[:n] *b = Buffer(out) @@ -97,7 +96,7 @@ func (b *Buffer) Base64Decode(v []byte) error { func (b Buffer) MarshalJSON() ([]byte, error) { v, err := b.Base64Encode() if err != nil { - return nil, errors.Wrap(err, "failed to encode to base64") + return nil, fmt.Errorf("failed to encode to base64: %w", err) } return json.Marshal(string(v)) } @@ -107,7 +106,7 @@ func (b Buffer) MarshalJSON() ([]byte, error) { func (b *Buffer) UnmarshalJSON(data []byte) error { var x string if err := json.Unmarshal(data, &x); err != nil { - return errors.Wrap(err, "failed to unmarshal JSON") + return fmt.Errorf("failed to unmarshal JSON: %w", err) } return b.Base64Decode([]byte(x)) } diff --git a/vendor/github.com/open-policy-agent/opa/internal/jwx/jwa/key_type.go b/vendor/github.com/open-policy-agent/opa/internal/jwx/jwa/key_type.go index 076bd39e..98f0cc42 100644 --- a/vendor/github.com/open-policy-agent/opa/internal/jwx/jwa/key_type.go +++ b/vendor/github.com/open-policy-agent/opa/internal/jwx/jwa/key_type.go @@ -1,9 +1,9 @@ package jwa import ( + "errors" + "fmt" "strconv" - - "github.com/pkg/errors" ) // KeyType represents the key type ("kty") that are supported @@ -29,11 +29,11 @@ func (keyType *KeyType) Accept(value interface{}) error { case KeyType: tmp = x default: - return errors.Errorf(`invalid type for jwa.KeyType: %T`, value) + return fmt.Errorf("invalid type for jwa.KeyType: %T", value) } _, ok := keyTypeAlg[tmp.String()] if !ok { - return errors.Errorf("Unknown Key Type algorithm") + return errors.New("unknown Key Type algorithm") } *keyType = tmp @@ -53,14 +53,14 @@ func (keyType *KeyType) UnmarshalJSON(data []byte) error { var err error quoted, err = strconv.Unquote(string(data)) if err != nil { - return errors.Wrap(err, "Failed to process signature algorithm") + return fmt.Errorf("failed to process signature algorithm: %w", err) } } else { quoted = string(data) } _, ok := keyTypeAlg[quoted] if !ok { - return errors.Errorf("Unknown signature algorithm") + return errors.New("unknown signature algorithm") } *keyType = KeyType(quoted) return nil diff --git a/vendor/github.com/open-policy-agent/opa/internal/jwx/jwa/signature.go b/vendor/github.com/open-policy-agent/opa/internal/jwx/jwa/signature.go index a0988eca..45e40017 100644 --- a/vendor/github.com/open-policy-agent/opa/internal/jwx/jwa/signature.go +++ b/vendor/github.com/open-policy-agent/opa/internal/jwx/jwa/signature.go @@ -1,9 +1,9 @@ package jwa import ( + "errors" + "fmt" "strconv" - - "github.com/pkg/errors" ) // SignatureAlgorithm represents the various signature algorithms as described in https://tools.ietf.org/html/rfc7518#section-3.1 @@ -27,6 +27,7 @@ const ( RS384 SignatureAlgorithm = "RS384" // RSASSA-PKCS-v1.5 using SHA-384 RS512 SignatureAlgorithm = "RS512" // RSASSA-PKCS-v1.5 using SHA-512 NoValue SignatureAlgorithm = "" // No value is different from none + Unsupported SignatureAlgorithm = "unsupported" ) // Accept is used when conversion from values given by @@ -39,11 +40,11 @@ func (signature *SignatureAlgorithm) Accept(value interface{}) error { case SignatureAlgorithm: tmp = x default: - return errors.Errorf(`invalid type for jwa.SignatureAlgorithm: %T`, value) + return fmt.Errorf("invalid type for jwa.SignatureAlgorithm: %T", value) } _, ok := signatureAlg[tmp.String()] if !ok { - return errors.Errorf("Unknown signature algorithm") + return errors.New("unknown signature algorithm") } *signature = tmp return nil @@ -62,14 +63,15 @@ func (signature *SignatureAlgorithm) UnmarshalJSON(data []byte) error { var err error quoted, err = strconv.Unquote(string(data)) if err != nil { - return errors.Wrap(err, "Failed to process signature algorithm") + return fmt.Errorf("failed to process signature algorithm: %w", err) } } else { quoted = string(data) } _, ok := signatureAlg[quoted] if !ok { - return errors.Errorf("Unknown signature algorithm") + *signature = Unsupported + return nil } *signature = SignatureAlgorithm(quoted) return nil diff --git a/vendor/github.com/open-policy-agent/opa/internal/jwx/jwk/ecdsa.go b/vendor/github.com/open-policy-agent/opa/internal/jwx/jwk/ecdsa.go index 30bee46b..b46689f0 100644 --- a/vendor/github.com/open-policy-agent/opa/internal/jwx/jwk/ecdsa.go +++ b/vendor/github.com/open-policy-agent/opa/internal/jwx/jwk/ecdsa.go @@ -3,10 +3,10 @@ package jwk import ( "crypto/ecdsa" "crypto/elliptic" + "errors" + "fmt" "math/big" - "github.com/pkg/errors" - "github.com/open-policy-agent/opa/internal/jwx/jwa" ) @@ -15,7 +15,7 @@ func newECDSAPublicKey(key *ecdsa.PublicKey) (*ECDSAPublicKey, error) { var hdr StandardHeaders err := hdr.Set(KeyTypeKey, jwa.EC) if err != nil { - return nil, errors.Wrapf(err, "Failed to set Key Type") + return nil, fmt.Errorf("failed to set Key Type: %w", err) } return &ECDSAPublicKey{ @@ -29,7 +29,7 @@ func newECDSAPrivateKey(key *ecdsa.PrivateKey) (*ECDSAPrivateKey, error) { var hdr StandardHeaders err := hdr.Set(KeyTypeKey, jwa.EC) if err != nil { - return nil, errors.Wrapf(err, "Failed to set Key Type") + return nil, fmt.Errorf("failed to set Key Type: %w", err) } return &ECDSAPrivateKey{ @@ -54,7 +54,7 @@ func (k *ECDSAPublicKey) GenerateKey(keyJSON *RawKeyJSON) error { var x, y big.Int if keyJSON.X == nil || keyJSON.Y == nil || keyJSON.Crv == "" { - return errors.Errorf("Missing mandatory key parameters X, Y or Crv") + return errors.New("missing mandatory key parameters X, Y or Crv") } x.SetBytes(keyJSON.X.Bytes()) @@ -69,7 +69,7 @@ func (k *ECDSAPublicKey) GenerateKey(keyJSON *RawKeyJSON) error { case jwa.P521: curve = elliptic.P521() default: - return errors.Errorf(`invalid curve name %s`, keyJSON.Crv) + return fmt.Errorf("invalid curve name %s", keyJSON.Crv) } *k = ECDSAPublicKey{ @@ -87,12 +87,12 @@ func (k *ECDSAPublicKey) GenerateKey(keyJSON *RawKeyJSON) error { func (k *ECDSAPrivateKey) GenerateKey(keyJSON *RawKeyJSON) error { if keyJSON.D == nil { - return errors.Errorf("Missing mandatory key parameter D") + return errors.New("missing mandatory key parameter D") } eCDSAPublicKey := &ECDSAPublicKey{} err := eCDSAPublicKey.GenerateKey(keyJSON) if err != nil { - return errors.Wrap(err, `failed to generate public key`) + return fmt.Errorf("failed to generate public key: %w", err) } dBytes := keyJSON.D.Bytes() // The length of this octet string MUST be ceiling(log-base-2(n)/8) @@ -106,7 +106,7 @@ func (k *ECDSAPrivateKey) GenerateKey(keyJSON *RawKeyJSON) error { n := eCDSAPublicKey.key.Params().N octetLength := (new(big.Int).Sub(n, big.NewInt(1)).BitLen() + 7) >> 3 if octetLength-len(dBytes) != 0 { - return errors.Errorf("Failed to generate private key. Incorrect D value") + return errors.New("failed to generate private key. Incorrect D value") } privateKey := &ecdsa.PrivateKey{ PublicKey: *eCDSAPublicKey.key, diff --git a/vendor/github.com/open-policy-agent/opa/internal/jwx/jwk/headers.go b/vendor/github.com/open-policy-agent/opa/internal/jwx/jwk/headers.go index cf700ee8..b0fd51e9 100644 --- a/vendor/github.com/open-policy-agent/opa/internal/jwx/jwk/headers.go +++ b/vendor/github.com/open-policy-agent/opa/internal/jwx/jwk/headers.go @@ -1,7 +1,7 @@ package jwk import ( - "github.com/pkg/errors" + "fmt" "github.com/open-policy-agent/opa/internal/jwx/jwa" ) @@ -122,7 +122,7 @@ func (h *StandardHeaders) Set(name string, value interface{}) error { case AlgorithmKey: var acceptor jwa.SignatureAlgorithm if err := acceptor.Accept(value); err != nil { - return errors.Wrapf(err, `invalid value for %s key`, AlgorithmKey) + return fmt.Errorf("invalid value for %s key: %w", AlgorithmKey, err) } h.Algorithm = &acceptor return nil @@ -131,15 +131,15 @@ func (h *StandardHeaders) Set(name string, value interface{}) error { h.KeyID = v return nil } - return errors.Errorf("invalid value for %s key: %T", KeyIDKey, value) + return fmt.Errorf("invalid value for %s key: %T", KeyIDKey, value) case KeyOpsKey: if err := h.KeyOps.Accept(value); err != nil { - return errors.Wrapf(err, "invalid value for %s key", KeyOpsKey) + return fmt.Errorf("invalid value for %s key: %w", KeyOpsKey, err) } return nil case KeyTypeKey: if err := h.KeyType.Accept(value); err != nil { - return errors.Wrapf(err, "invalid value for %s key", KeyTypeKey) + return fmt.Errorf("invalid value for %s key: %w", KeyTypeKey, err) } return nil case KeyUsageKey: @@ -147,15 +147,15 @@ func (h *StandardHeaders) Set(name string, value interface{}) error { h.KeyUsage = v return nil } - return errors.Errorf("invalid value for %s key: %T", KeyUsageKey, value) + return fmt.Errorf("invalid value for %s key: %T", KeyUsageKey, value) case PrivateParamsKey: if v, ok := value.(map[string]interface{}); ok { h.PrivateParams = v return nil } - return errors.Errorf("invalid value for %s key: %T", PrivateParamsKey, value) + return fmt.Errorf("invalid value for %s key: %T", PrivateParamsKey, value) default: - return errors.Errorf(`invalid key: %s`, name) + return fmt.Errorf("invalid key: %s", name) } } @@ -164,14 +164,14 @@ func (h StandardHeaders) Walk(f func(string, interface{}) error) error { for _, key := range []string{AlgorithmKey, KeyIDKey, KeyOpsKey, KeyTypeKey, KeyUsageKey, PrivateParamsKey} { if v, ok := h.Get(key); ok { if err := f(key, v); err != nil { - return errors.Wrapf(err, `walk function returned error for %s`, key) + return fmt.Errorf("walk function returned error for %s: %w", key, err) } } } for k, v := range h.PrivateParams { if err := f(k, v); err != nil { - return errors.Wrapf(err, `walk function returned error for %s`, k) + return fmt.Errorf("walk function returned error for %s: %w", k, err) } } return nil diff --git a/vendor/github.com/open-policy-agent/opa/internal/jwx/jwk/jwk.go b/vendor/github.com/open-policy-agent/opa/internal/jwx/jwk/jwk.go index 22ccf8df..aa22a383 100644 --- a/vendor/github.com/open-policy-agent/opa/internal/jwx/jwk/jwk.go +++ b/vendor/github.com/open-policy-agent/opa/internal/jwx/jwk/jwk.go @@ -5,8 +5,8 @@ import ( "crypto/ecdsa" "crypto/rsa" "encoding/json" - - "github.com/pkg/errors" + "errors" + "fmt" "github.com/open-policy-agent/opa/internal/jwx/jwa" ) @@ -17,7 +17,7 @@ import ( // public key cannot be deduced, an error is returned func GetPublicKey(key interface{}) (interface{}, error) { if key == nil { - return nil, errors.New(`jwk.New requires a non-nil key`) + return nil, errors.New("jwk.New requires a non-nil key") } switch v := key.(type) { @@ -32,7 +32,7 @@ func GetPublicKey(key interface{}) (interface{}, error) { case []byte: return v, nil default: - return nil, errors.Errorf(`invalid key type %T`, key) + return nil, fmt.Errorf("invalid key type %T", key) } } @@ -54,7 +54,7 @@ func GetKeyTypeFromKey(key interface{}) jwa.KeyType { // New creates a jwk.Key from the given key. func New(key interface{}) (Key, error) { if key == nil { - return nil, errors.New(`jwk.New requires a non-nil key`) + return nil, errors.New("jwk.New requires a non-nil key") } switch v := key.(type) { @@ -69,7 +69,7 @@ func New(key interface{}) (Key, error) { case []byte: return newSymmetricKey(v) default: - return nil, errors.Errorf(`invalid key type %T`, key) + return nil, fmt.Errorf("invalid key type %T", key) } } @@ -80,7 +80,7 @@ func parse(jwkSrc string) (*Set, error) { rawKeySetJSON := &RawKeySetJSON{} err := json.Unmarshal([]byte(jwkSrc), rawKeySetJSON) if err != nil { - return nil, errors.Wrap(err, "Failed to unmarshal JWK Set") + return nil, fmt.Errorf("failed to unmarshal JWK Set: %w", err) } if len(rawKeySetJSON.Keys) == 0 { @@ -88,20 +88,23 @@ func parse(jwkSrc string) (*Set, error) { rawKeyJSON := &RawKeyJSON{} err := json.Unmarshal([]byte(jwkSrc), rawKeyJSON) if err != nil { - return nil, errors.Wrap(err, "Failed to unmarshal JWK") + return nil, fmt.Errorf("failed to unmarshal JWK: %w", err) } jwkKey, err = rawKeyJSON.GenerateKey() if err != nil { - return nil, errors.Wrap(err, "Failed to generate key") + return nil, fmt.Errorf("failed to generate key: %w", err) } // Add to set jwkKeySet.Keys = append(jwkKeySet.Keys, jwkKey) } else { for i := range rawKeySetJSON.Keys { rawKeyJSON := rawKeySetJSON.Keys[i] + if rawKeyJSON.Algorithm != nil && *rawKeyJSON.Algorithm == jwa.Unsupported { + continue + } jwkKey, err = rawKeyJSON.GenerateKey() if err != nil { - return nil, errors.Wrap(err, "Failed to generate key: %s") + return nil, fmt.Errorf("failed to generate key: %w", err) } jwkKeySet.Keys = append(jwkKeySet.Keys, jwkKey) } @@ -140,11 +143,11 @@ func (r *RawKeyJSON) GenerateKey() (Key, error) { case jwa.OctetSeq: key = &SymmetricKey{} default: - return nil, errors.Errorf(`Unrecognized key type`) + return nil, errors.New("unrecognized key type") } err := key.GenerateKey(r) if err != nil { - return nil, errors.Wrap(err, "Failed to generate key from JWK") + return nil, fmt.Errorf("failed to generate key from JWK: %w", err) } return key, nil } diff --git a/vendor/github.com/open-policy-agent/opa/internal/jwx/jwk/rsa.go b/vendor/github.com/open-policy-agent/opa/internal/jwx/jwk/rsa.go index 1a5cba47..11b8e3b5 100644 --- a/vendor/github.com/open-policy-agent/opa/internal/jwx/jwk/rsa.go +++ b/vendor/github.com/open-policy-agent/opa/internal/jwx/jwk/rsa.go @@ -3,10 +3,10 @@ package jwk import ( "crypto/rsa" "encoding/binary" + "errors" + "fmt" "math/big" - "github.com/pkg/errors" - "github.com/open-policy-agent/opa/internal/jwx/jwa" ) @@ -15,7 +15,7 @@ func newRSAPublicKey(key *rsa.PublicKey) (*RSAPublicKey, error) { var hdr StandardHeaders err := hdr.Set(KeyTypeKey, jwa.RSA) if err != nil { - return nil, errors.Wrapf(err, "Failed to set Key Type") + return nil, fmt.Errorf("failed to set Key Type: %w", err) } return &RSAPublicKey{ StandardHeaders: &hdr, @@ -28,7 +28,7 @@ func newRSAPrivateKey(key *rsa.PrivateKey) (*RSAPrivateKey, error) { var hdr StandardHeaders err := hdr.Set(KeyTypeKey, jwa.RSA) if err != nil { - return nil, errors.Wrapf(err, "Failed to set Key Type") + return nil, fmt.Errorf("failed to set Key Type: %w", err) } var algoParams jwa.AlgorithmParameters @@ -67,7 +67,7 @@ func newRSAPrivateKey(key *rsa.PrivateKey) (*RSAPrivateKey, error) { // Materialize returns the standard RSA Public Key representation stored in the internal representation func (k *RSAPublicKey) Materialize() (interface{}, error) { if k.key == nil { - return nil, errors.New(`key has no rsa.PublicKey associated with it`) + return nil, errors.New("key has no rsa.PublicKey associated with it") } return k.key, nil } @@ -75,7 +75,7 @@ func (k *RSAPublicKey) Materialize() (interface{}, error) { // Materialize returns the standard RSA Private Key representation stored in the internal representation func (k *RSAPrivateKey) Materialize() (interface{}, error) { if k.key == nil { - return nil, errors.New(`key has no rsa.PrivateKey associated with it`) + return nil, errors.New("key has no rsa.PrivateKey associated with it") } return k.key, nil } @@ -84,7 +84,7 @@ func (k *RSAPrivateKey) Materialize() (interface{}, error) { func (k *RSAPublicKey) GenerateKey(keyJSON *RawKeyJSON) error { if keyJSON.N == nil || keyJSON.E == nil { - return errors.Errorf("Missing mandatory key parameters N or E") + return errors.New("missing mandatory key parameters N or E") } rsaPublicKey := &rsa.PublicKey{ N: (&big.Int{}).SetBytes(keyJSON.N.Bytes()), @@ -101,11 +101,11 @@ func (k *RSAPrivateKey) GenerateKey(keyJSON *RawKeyJSON) error { rsaPublicKey := &RSAPublicKey{} err := rsaPublicKey.GenerateKey(keyJSON) if err != nil { - return errors.Wrap(err, "failed to generate public key") + return fmt.Errorf("failed to generate public key: %w", err) } if keyJSON.D == nil || keyJSON.P == nil || keyJSON.Q == nil { - return errors.Errorf("Missing mandatory key parameters D, P or Q") + return errors.New("missing mandatory key parameters D, P or Q") } privateKey := &rsa.PrivateKey{ PublicKey: *rsaPublicKey.key, diff --git a/vendor/github.com/open-policy-agent/opa/internal/jwx/jwk/symmetric.go b/vendor/github.com/open-policy-agent/opa/internal/jwx/jwk/symmetric.go index 8a073615..e0cc0751 100644 --- a/vendor/github.com/open-policy-agent/opa/internal/jwx/jwk/symmetric.go +++ b/vendor/github.com/open-policy-agent/opa/internal/jwx/jwk/symmetric.go @@ -1,7 +1,7 @@ package jwk import ( - "github.com/pkg/errors" + "fmt" "github.com/open-policy-agent/opa/internal/jwx/jwa" ) @@ -11,7 +11,7 @@ func newSymmetricKey(key []byte) (*SymmetricKey, error) { err := hdr.Set(KeyTypeKey, jwa.OctetSeq) if err != nil { - return nil, errors.Wrapf(err, "Failed to set Key Type") + return nil, fmt.Errorf("failed to set Key Type: %w", err) } return &SymmetricKey{ StandardHeaders: &hdr, diff --git a/vendor/github.com/open-policy-agent/opa/internal/jwx/jws/headers.go b/vendor/github.com/open-policy-agent/opa/internal/jwx/jws/headers.go index 045e38fa..0c8b3550 100644 --- a/vendor/github.com/open-policy-agent/opa/internal/jwx/jws/headers.go +++ b/vendor/github.com/open-policy-agent/opa/internal/jwx/jws/headers.go @@ -1,7 +1,7 @@ package jws import ( - "github.com/pkg/errors" + "fmt" "github.com/open-policy-agent/opa/internal/jwx/jwa" ) @@ -103,7 +103,7 @@ func (h *StandardHeaders) Set(name string, value interface{}) error { switch name { case AlgorithmKey: if err := h.Algorithm.Accept(value); err != nil { - return errors.Wrapf(err, `invalid value for %s key`, AlgorithmKey) + return fmt.Errorf("invalid value for %s key: %w", AlgorithmKey, err) } return nil case ContentTypeKey: @@ -111,44 +111,44 @@ func (h *StandardHeaders) Set(name string, value interface{}) error { h.ContentType = v return nil } - return errors.Errorf(`invalid value for %s key: %T`, ContentTypeKey, value) + return fmt.Errorf("invalid value for %s key: %T", ContentTypeKey, value) case CriticalKey: if v, ok := value.([]string); ok { h.Critical = v return nil } - return errors.Errorf(`invalid value for %s key: %T`, CriticalKey, value) + return fmt.Errorf("invalid value for %s key: %T", CriticalKey, value) case JWKKey: if v, ok := value.(string); ok { h.JWK = v return nil } - return errors.Errorf(`invalid value for %s key: %T`, JWKKey, value) + return fmt.Errorf("invalid value for %s key: %T", JWKKey, value) case JWKSetURLKey: if v, ok := value.(string); ok { h.JWKSetURL = v return nil } - return errors.Errorf(`invalid value for %s key: %T`, JWKSetURLKey, value) + return fmt.Errorf("invalid value for %s key: %T", JWKSetURLKey, value) case KeyIDKey: if v, ok := value.(string); ok { h.KeyID = v return nil } - return errors.Errorf(`invalid value for %s key: %T`, KeyIDKey, value) + return fmt.Errorf("invalid value for %s key: %T", KeyIDKey, value) case PrivateParamsKey: if v, ok := value.(map[string]interface{}); ok { h.PrivateParams = v return nil } - return errors.Errorf(`invalid value for %s key: %T`, PrivateParamsKey, value) + return fmt.Errorf("invalid value for %s key: %T", PrivateParamsKey, value) case TypeKey: if v, ok := value.(string); ok { h.Type = v return nil } - return errors.Errorf(`invalid value for %s key: %T`, TypeKey, value) + return fmt.Errorf("invalid value for %s key: %T", TypeKey, value) default: - return errors.Errorf(`invalid key: %s`, name) + return fmt.Errorf("invalid key: %s", name) } } diff --git a/vendor/github.com/open-policy-agent/opa/internal/jwx/jws/jws.go b/vendor/github.com/open-policy-agent/opa/internal/jwx/jws/jws.go index bfa498bb..c163a3bd 100644 --- a/vendor/github.com/open-policy-agent/opa/internal/jwx/jws/jws.go +++ b/vendor/github.com/open-policy-agent/opa/internal/jwx/jws/jws.go @@ -24,6 +24,8 @@ import ( "crypto/rand" "encoding/base64" "encoding/json" + "errors" + "fmt" "io" "strings" @@ -31,8 +33,6 @@ import ( "github.com/open-policy-agent/opa/internal/jwx/jwk" "github.com/open-policy-agent/opa/internal/jwx/jws/sign" "github.com/open-policy-agent/opa/internal/jwx/jws/verify" - - "github.com/pkg/errors" ) // SignLiteral generates a Signature for the given Payload and Headers, and serializes @@ -50,7 +50,7 @@ func SignLiteral(payload []byte, alg jwa.SignatureAlgorithm, key interface{}, hd ) signer, err := sign.New(alg) if err != nil { - return nil, errors.Wrap(err, `failed to create signer`) + return nil, fmt.Errorf("failed to create signer: %w", err) } var signature []byte @@ -61,7 +61,7 @@ func SignLiteral(payload []byte, alg jwa.SignatureAlgorithm, key interface{}, hd signature, err = signer.Sign([]byte(signingInput), key) } if err != nil { - return nil, errors.Wrap(err, `failed to sign Payload`) + return nil, fmt.Errorf("failed to sign Payload: %w", err) } encodedSignature := base64.RawURLEncoding.EncodeToString(signature) compactSerialization := strings.Join( @@ -83,12 +83,12 @@ func SignWithOption(payload []byte, alg jwa.SignatureAlgorithm, key interface{}) err := headers.Set(AlgorithmKey, alg) if err != nil { - return nil, errors.Wrap(err, "Failed to set alg value") + return nil, fmt.Errorf("failed to set alg value: %w", err) } hdrBuf, err := json.Marshal(headers) if err != nil { - return nil, errors.Wrap(err, `failed to marshal Headers`) + return nil, fmt.Errorf("failed to marshal Headers: %w", err) } // NOTE(sr): we don't use SignWithOption -- if we did, this rand.Reader // should come from the BuiltinContext's Seed, too. @@ -104,7 +104,7 @@ func Verify(buf []byte, alg jwa.SignatureAlgorithm, key interface{}) (ret []byte verifier, err := verify.New(alg) if err != nil { - return nil, errors.Wrap(err, "failed to create verifier") + return nil, fmt.Errorf("failed to create verifier: %w", err) } buf = bytes.TrimSpace(buf) @@ -114,7 +114,7 @@ func Verify(buf []byte, alg jwa.SignatureAlgorithm, key interface{}) (ret []byte parts, err := SplitCompact(string(buf[:])) if err != nil { - return nil, errors.Wrap(err, `failed extract from compact serialization format`) + return nil, fmt.Errorf("failed extract from compact serialization format: %w", err) } signingInput := strings.Join( @@ -126,16 +126,16 @@ func Verify(buf []byte, alg jwa.SignatureAlgorithm, key interface{}) (ret []byte decodedSignature, err := base64.RawURLEncoding.DecodeString(parts[2]) if err != nil { - return nil, errors.Wrap(err, "Failed to decode signature") + return nil, fmt.Errorf("failed to decode signature: %w", err) } if err := verifier.Verify([]byte(signingInput), decodedSignature, key); err != nil { - return nil, errors.Wrap(err, "Failed to verify message") + return nil, fmt.Errorf("failed to verify message: %w", err) } if decodedPayload, err := base64.RawURLEncoding.DecodeString(parts[1]); err == nil { return decodedPayload, nil } - return nil, errors.Wrap(err, "Failed to decode Payload") + return nil, fmt.Errorf("failed to decode Payload: %w", err) } // VerifyWithJWK verifies the JWS message using the specified JWK @@ -143,7 +143,7 @@ func VerifyWithJWK(buf []byte, key jwk.Key) (payload []byte, err error) { keyVal, err := key.Materialize() if err != nil { - return nil, errors.Wrap(err, "Failed to materialize key") + return nil, fmt.Errorf("failed to materialize key: %w", err) } return Verify(buf, key.GetAlgorithm(), keyVal) } @@ -179,7 +179,7 @@ func SplitCompact(jwsCompact string) ([]string, error) { parts := strings.Split(jwsCompact, ".") if len(parts) < 3 { - return nil, errors.New("Failed to split compact serialization") + return nil, errors.New("failed to split compact serialization") } return parts, nil } @@ -190,24 +190,24 @@ func parseCompact(str string) (m *Message, err error) { var decodedHeader, decodedPayload, decodedSignature []byte parts, err := SplitCompact(str) if err != nil { - return nil, errors.Wrap(err, `invalid compact serialization format`) + return nil, fmt.Errorf("invalid compact serialization format: %w", err) } if decodedHeader, err = base64.RawURLEncoding.DecodeString(parts[0]); err != nil { - return nil, errors.Wrap(err, `failed to decode Headers`) + return nil, fmt.Errorf("failed to decode Headers: %w", err) } var hdr StandardHeaders if err := json.Unmarshal(decodedHeader, &hdr); err != nil { - return nil, errors.Wrap(err, `failed to parse JOSE Headers`) + return nil, fmt.Errorf("failed to parse JOSE Headers: %w", err) } if decodedPayload, err = base64.RawURLEncoding.DecodeString(parts[1]); err != nil { - return nil, errors.Wrap(err, `failed to decode Payload`) + return nil, fmt.Errorf("failed to decode Payload: %w", err) } if len(parts) > 2 { if decodedSignature, err = base64.RawURLEncoding.DecodeString(parts[2]); err != nil { - return nil, errors.Wrap(err, `failed to decode Signature`) + return nil, fmt.Errorf("failed to decode Signature: %w", err) } } diff --git a/vendor/github.com/open-policy-agent/opa/internal/jwx/jws/sign/ecdsa.go b/vendor/github.com/open-policy-agent/opa/internal/jwx/jws/sign/ecdsa.go index 62af72b6..db1aadec 100644 --- a/vendor/github.com/open-policy-agent/opa/internal/jwx/jws/sign/ecdsa.go +++ b/vendor/github.com/open-policy-agent/opa/internal/jwx/jws/sign/ecdsa.go @@ -4,11 +4,11 @@ import ( "crypto" "crypto/ecdsa" "crypto/rand" + "errors" + "fmt" "io" "github.com/open-policy-agent/opa/internal/jwx/jwa" - - "github.com/pkg/errors" ) var ecdsaSignFuncs = map[jwa.SignatureAlgorithm]ecdsaSignFunc{} @@ -37,7 +37,7 @@ func makeECDSASignFunc(hash crypto.Hash) ecdsaSignFunc { h.Write(payload) r, s, err := ecdsa.Sign(rnd, key, h.Sum(nil)) if err != nil { - return nil, errors.Wrap(err, "failed to sign payload using ecdsa") + return nil, fmt.Errorf("failed to sign payload using ecdsa: %w", err) } rBytes := r.Bytes() @@ -56,7 +56,7 @@ func makeECDSASignFunc(hash crypto.Hash) ecdsaSignFunc { func newECDSA(alg jwa.SignatureAlgorithm) (*ECDSASigner, error) { signfn, ok := ecdsaSignFuncs[alg] if !ok { - return nil, errors.Errorf(`unsupported algorithm while trying to create ECDSA signer: %s`, alg) + return nil, fmt.Errorf("unsupported algorithm while trying to create ECDSA signer: %s", alg) } return &ECDSASigner{ @@ -74,12 +74,12 @@ func (s ECDSASigner) Algorithm() jwa.SignatureAlgorithm { // source (such as `rand.Reader`). func (s ECDSASigner) SignWithRand(payload []byte, key interface{}, r io.Reader) ([]byte, error) { if key == nil { - return nil, errors.New(`missing private key while signing payload`) + return nil, errors.New("missing private key while signing payload") } privateKey, ok := key.(*ecdsa.PrivateKey) if !ok { - return nil, errors.Errorf(`invalid key type %T. *ecdsa.PrivateKey is required`, key) + return nil, fmt.Errorf("invalid key type %T. *ecdsa.PrivateKey is required", key) } return s.sign(payload, privateKey, r) } diff --git a/vendor/github.com/open-policy-agent/opa/internal/jwx/jws/verify/hmac.go b/vendor/github.com/open-policy-agent/opa/internal/jwx/jws/verify/hmac.go index 77e45887..1f8d0498 100644 --- a/vendor/github.com/open-policy-agent/opa/internal/jwx/jws/verify/hmac.go +++ b/vendor/github.com/open-policy-agent/opa/internal/jwx/jws/verify/hmac.go @@ -2,18 +2,18 @@ package verify import ( "crypto/hmac" - - "github.com/pkg/errors" + "fmt" "github.com/open-policy-agent/opa/internal/jwx/jwa" "github.com/open-policy-agent/opa/internal/jwx/jws/sign" + "github.com/pkg/errors" ) func newHMAC(alg jwa.SignatureAlgorithm) (*HMACVerifier, error) { s, err := sign.New(alg) if err != nil { - return nil, errors.Wrap(err, `failed to generate HMAC signer`) + return nil, fmt.Errorf("failed to generate HMAC signer: %w", err) } return &HMACVerifier{signer: s}, nil } @@ -23,11 +23,11 @@ func (v HMACVerifier) Verify(signingInput, signature []byte, key interface{}) (e expected, err := v.signer.Sign(signingInput, key) if err != nil { - return errors.Wrap(err, `failed to generated signature`) + return fmt.Errorf("failed to generated signature: %w", err) } if !hmac.Equal(signature, expected) { - return errors.New(`failed to match hmac signature`) + return errors.New("failed to match hmac signature") } return nil } diff --git a/vendor/github.com/open-policy-agent/opa/internal/planner/planner.go b/vendor/github.com/open-policy-agent/opa/internal/planner/planner.go index f3835f47..62e2b4c3 100644 --- a/vendor/github.com/open-policy-agent/opa/internal/planner/planner.go +++ b/vendor/github.com/open-policy-agent/opa/internal/planner/planner.go @@ -14,7 +14,7 @@ import ( "github.com/open-policy-agent/opa/ast" "github.com/open-policy-agent/opa/ast/location" "github.com/open-policy-agent/opa/internal/debug" - "github.com/open-policy-agent/opa/internal/ir" + "github.com/open-policy-agent/opa/ir" ) // QuerySet represents the input to the planner. @@ -36,11 +36,12 @@ type Planner struct { externs map[string]*ast.Builtin // built-in functions that are required in execution environment decls map[string]*ast.Builtin // built-in functions that may be provided in execution environment rules *ruletrie // rules that may be planned + mocks *functionMocksStack // replacements for built-in functions funcs *funcstack // functions that have been planned plan *ir.Plan // in-progress query plan curr *ir.Block // in-progress query block vars *varstack // in-scope variables - ltarget ir.LocalOrConst // target variable or constant of last planned statement + ltarget ir.Operand // target variable or constant of last planned statement lnext ir.Local // next variable to use loc *location.Location // location currently "being planned" debug debug.Debug // debug information produced during planning @@ -76,6 +77,7 @@ func New() *Planner { }), rules: newRuletrie(), funcs: newFuncstack(), + mocks: newFunctionMocksStack(), debug: debug.Discard(), } } @@ -342,7 +344,7 @@ func (p *Planner) planRules(rules []*ast.Rule) (string, error) { p.appendStmtToBlock( &ir.AssignVarOnceStmt{ Target: fn.Return, - Source: lresult, + Source: op(lresult), }, end) *blocks = append(*blocks, end) @@ -400,7 +402,7 @@ func (p *Planner) planFuncParams(params []ir.Local, args ast.Args, idx int, iter if idx >= len(args) { return iter() } - return p.planUnifyLocal(params[idx], args[idx], func() error { + return p.planUnifyLocal(op(params[idx]), args[idx], func() error { return p.planFuncParams(params, args, idx+1, iter) }) } @@ -460,13 +462,13 @@ func (p *Planner) planQueries() error { if !rw.IsGenerated() && !rw.IsWildcard() { p.appendStmt(&ir.ObjectInsertStmt{ Object: lr, - Key: lvarnames[rw], - Value: p.vars.GetOrEmpty(qv), + Key: op(lvarnames[rw]), + Value: p.vars.GetOpOrEmpty(qv), }) } } - p.appendStmt(&ir.ResultSetAdd{ + p.appendStmt(&ir.ResultSetAddStmt{ Value: lr, }) @@ -514,16 +516,18 @@ func (p *Planner) planQuery(q ast.Body, index int, iter planiter) error { // TODO(tsandall): improve errors to include location information. func (p *Planner) planExpr(e *ast.Expr, iter planiter) error { - if e.Negated { + switch { + case e.Negated: return p.planNot(e, iter) - } - if len(e.With) > 0 { + case len(e.With) > 0: return p.planWith(e, iter) - } - if e.IsCall() { + case e.IsCall(): return p.planExprCall(e, iter) + + case e.IsEvery(): + return p.planExprEvery(e, iter) } return p.planExprTerm(e, iter) @@ -549,23 +553,40 @@ func (p *Planner) planNot(e *ast.Expr, iter planiter) error { func (p *Planner) planWith(e *ast.Expr, iter planiter) error { - // Plan the values that will be applied by the with modifiers. All values + // Plan the values that will be applied by the `with` modifiers. All values // must be defined for the overall expression to evaluate. - values := make([]*ast.Term, len(e.With)) + values := make([]*ast.Term, 0, len(e.With)) // NOTE(sr): we could be overallocating if there are builtin replacements + targets := make([]ast.Ref, 0, len(e.With)) + + mocks := frame{} + + for _, w := range e.With { + v := w.Target.Value.(ast.Ref) + + switch { + case p.isFunction(v): // nothing to do - for i := range e.With { - values[i] = e.With[i].Value + case ast.DefaultRootDocument.Equal(v[0]) || + ast.InputRootDocument.Equal(v[0]): + + values = append(values, w.Value) + targets = append(targets, w.Target.Value.(ast.Ref)) + + continue // not a mock + } + + mocks[w.Target.String()] = w.Value } - return p.planTermSlice(values, func(locals []ir.LocalOrConst) error { + return p.planTermSlice(values, func(locals []ir.Operand) error { + + p.mocks.PushFrame(mocks) - paths := make([][]int, len(e.With)) + paths := make([][]int, len(targets)) saveVars := ast.NewVarSet() dataRefs := []ast.Ref{} - for i := range e.With { - - target := e.With[i].Target.Value.(ast.Ref) + for i, target := range targets { paths[i] = make([]int, len(target)-1) for j := 1; j < len(target); j++ { @@ -589,15 +610,16 @@ func (p *Planner) planWith(e *ast.Expr, iter planiter) error { for i, v := range saveVars.Sorted() { lorig := p.vars.GetOrEmpty(v) lsave := p.newLocal() - p.appendStmt(&ir.AssignVarStmt{Source: lorig, Target: lsave}) + p.appendStmt(&ir.AssignVarStmt{Source: op(lorig), Target: lsave}) restore[i] = [2]ir.Local{lorig, lsave} } // If any of the `with` statements targeted the data document, overwriting - // parts of the ruletrie, we shadow the existing planned functions during - // expression planning. This causes the planner to re-plan any rules that - // may be required during planning of this expression (transitively). - shadowing := p.dataRefsShadowRuletrie(dataRefs) + // parts of the ruletrie; or if a function has been mocked, we shadow the + // existing planned functions during expression planning. + // This causes the planner to re-plan any rules that may be required during + // planning of this expression (transitively). + shadowing := p.dataRefsShadowRuletrie(dataRefs) || len(mocks) > 0 if shadowing { p.funcs.Push(map[string]string{}) for _, ref := range dataRefs { @@ -606,6 +628,7 @@ func (p *Planner) planWith(e *ast.Expr, iter planiter) error { } err := p.planWithRec(e, paths, locals, 0, func() error { + p.mocks.PopFrame() if shadowing { p.funcs.Pop() for i := len(dataRefs) - 1; i >= 0; i-- { @@ -617,6 +640,7 @@ func (p *Planner) planWith(e *ast.Expr, iter planiter) error { err := iter() + p.mocks.PushFrame(mocks) if shadowing { p.funcs.Push(map[string]string{}) for _, ref := range dataRefs { @@ -629,6 +653,7 @@ func (p *Planner) planWith(e *ast.Expr, iter planiter) error { return err }) + p.mocks.PopFrame() if shadowing { p.funcs.Pop() for i := len(dataRefs) - 1; i >= 0; i-- { @@ -640,8 +665,8 @@ func (p *Planner) planWith(e *ast.Expr, iter planiter) error { }) } -func (p *Planner) planWithRec(e *ast.Expr, targets [][]int, values []ir.LocalOrConst, index int, iter planiter) error { - if index >= len(e.With) { +func (p *Planner) planWithRec(e *ast.Expr, targets [][]int, values []ir.Operand, index int, iter planiter) error { + if index >= len(targets) { return p.planExpr(e.NoWith(), iter) } @@ -688,7 +713,7 @@ func (p *Planner) planWithUndoRec(restore [][2]ir.Local, index int, iter planite p.appendStmt(&ir.WithStmt{ Local: lorig, - Value: lsave, + Value: op(lsave), Block: block, }) @@ -708,12 +733,80 @@ func (p *Planner) planExprTerm(e *ast.Expr, iter planiter) error { return p.planTerm(e.Terms.(*ast.Term), func() error { p.appendStmt(&ir.NotEqualStmt{ A: p.ltarget, - B: ir.Bool(false), + B: op(ir.Bool(false)), }) return iter() }) } +func (p *Planner) planExprEvery(e *ast.Expr, iter planiter) error { + every := e.Terms.(*ast.Every) + + cond0 := p.newLocal() // outer not + cond1 := p.newLocal() // inner not + + // We're using condition variables together with IsDefinedStmt to encode + // this: + // every x, y in xs { p(x,y) } + // ~> p(x1, y1) AND p(x2, y2) AND ... AND p(xn, yn) + // ~> NOT (NOT p(x1, y1) OR NOT p(x2, y2) OR ... OR NOT p(xn, yn)) + // + // cond1 is initialized to 0, and set to TRUE if p(xi, yi) succeeds for + // a binding of (xi, yi). We then use IsUndefined to check that this has NOT + // happened (NOT p(xi, yi)). + // cond0 is initialized to 0, and set to TRUE if cond1 happens to not + // be set: it's encoding the NOT ( ... OR ... OR ... ) part of this. + + p.appendStmt(&ir.ResetLocalStmt{ + Target: cond0, + }) + + err := p.planTerm(every.Domain, func() error { + return p.planScan(every.Key, func(ir.Local) error { + p.appendStmt(&ir.ResetLocalStmt{ + Target: cond1, + }) + nested := &ir.BlockStmt{Blocks: []*ir.Block{{}}} + + prev := p.curr + p.curr = nested.Blocks[0] + + lval := p.ltarget + err := p.planUnifyLocal(lval, every.Value, func() error { + return p.planQuery(every.Body, 0, func() error { + p.appendStmt(&ir.AssignVarStmt{ + Source: op(ir.Bool(true)), + Target: cond1, + }) + return nil + }) + }) + if err != nil { + return err + } + + p.curr = prev + p.appendStmt(nested) + p.appendStmt(&ir.IsUndefinedStmt{ + Source: cond1, + }) + p.appendStmt(&ir.AssignVarStmt{ + Source: op(ir.Bool(true)), + Target: cond0, + }) + return nil + }) + }) + if err != nil { + return err + } + + p.appendStmt(&ir.IsUndefinedStmt{ + Source: cond0, + }) + return iter() +} + func (p *Planner) planExprCall(e *ast.Expr, iter planiter) error { operator := e.Operator().String() switch operator { @@ -726,21 +819,55 @@ func (p *Planner) planExprCall(e *ast.Expr, iter planiter) error { var name string var arity int var void bool - var args []ir.LocalOrConst + var args []ir.Operand + var err error + + operands := e.Operands() + op := e.Operator() + + if replacement := p.mocks.Lookup(operator); replacement != nil { + switch r := replacement.Value.(type) { + case ast.Ref: + if !r.HasPrefix(ast.DefaultRootRef) && !r.HasPrefix(ast.InputRootRef) { + // replacement is builtin + operator = r.String() + bi := p.decls[operator] + p.externs[operator] = bi + + // void functions and relations are forbidden; arity validation happened in compiler + return p.planExprCallFunc(operator, len(bi.Decl.FuncArgs().Args), void, operands, args, iter) + } + + // replacement is a function (rule) + if node := p.rules.Lookup(r); node != nil { + p.mocks.Push() // new scope + name, err = p.planRules(node.Rules()) + if err != nil { + return err + } + p.mocks.Pop() + return p.planExprCallFunc(name, node.Arity(), void, operands, p.defaultOperands(), iter) + } + + return fmt.Errorf("illegal replacement of operator %q by %v", operator, replacement) - node := p.rules.Lookup(e.Operator()) + default: // replacement is a value + if bi, ok := p.decls[operator]; ok { + return p.planExprCallValue(replacement, len(bi.Decl.FuncArgs().Args), operands, iter) + } + if node := p.rules.Lookup(op); node != nil { + return p.planExprCallValue(replacement, node.Arity(), operands, iter) + } + } + } - if node != nil { - var err error + if node := p.rules.Lookup(op); node != nil { name, err = p.planRules(node.Rules()) if err != nil { return err } arity = node.Arity() - args = []ir.LocalOrConst{ - p.vars.GetOrEmpty(ast.InputRootDocument.Value.(ast.Var)), - p.vars.GetOrEmpty(ast.DefaultRootDocument.Value.(ast.Var)), - } + args = p.defaultOperands() } else if decl, ok := p.decls[operator]; ok { relation = decl.Relation arity = len(decl.Decl.Args()) @@ -751,8 +878,6 @@ func (p *Planner) planExprCall(e *ast.Expr, iter planiter) error { return fmt.Errorf("illegal call: unknown operator %q", operator) } - operands := e.Operands() - if len(operands) < arity || len(operands) > arity+1 { return fmt.Errorf("illegal call: wrong number of operands: got %v, want %v)", len(operands), arity) } @@ -765,12 +890,12 @@ func (p *Planner) planExprCall(e *ast.Expr, iter planiter) error { } } -func (p *Planner) planExprCallRelation(name string, arity int, operands []*ast.Term, args []ir.LocalOrConst, iter planiter) error { +func (p *Planner) planExprCallRelation(name string, arity int, operands []*ast.Term, args []ir.Operand, iter planiter) error { if len(operands) == arity { - return p.planCallArgs(operands, 0, args, func(args []ir.LocalOrConst) error { - p.ltarget = p.newLocal() - ltarget := p.ltarget.(ir.Local) + return p.planCallArgs(operands, 0, args, func(args []ir.Operand) error { + p.ltarget = p.newOperand() + ltarget := p.ltarget.Value.(ir.Local) p.appendStmt(&ir.CallStmt{ Func: name, Args: args, @@ -780,7 +905,7 @@ func (p *Planner) planExprCallRelation(name string, arity int, operands []*ast.T lsize := p.newLocal() p.appendStmt(&ir.LenStmt{ - Source: ltarget, + Source: op(ltarget), Target: lsize, }) @@ -792,22 +917,22 @@ func (p *Planner) planExprCallRelation(name string, arity int, operands []*ast.T }) p.appendStmt(&ir.NotEqualStmt{ - A: lsize, - B: lzero, + A: op(lsize), + B: op(lzero), }) return iter() }) } - return p.planCallArgs(operands[:len(operands)-1], 0, args, func(args []ir.LocalOrConst) error { + return p.planCallArgs(operands[:len(operands)-1], 0, args, func(args []ir.Operand) error { - p.ltarget = p.newLocal() + p.ltarget = p.newOperand() p.appendStmt(&ir.CallStmt{ Func: name, Args: args, - Result: p.ltarget.(ir.Local), + Result: p.ltarget.Value.(ir.Local), }) return p.planScanValues(operands[len(operands)-1], func(ir.Local) error { @@ -816,14 +941,14 @@ func (p *Planner) planExprCallRelation(name string, arity int, operands []*ast.T }) } -func (p *Planner) planExprCallFunc(name string, arity int, void bool, operands []*ast.Term, args []ir.LocalOrConst, iter planiter) error { +func (p *Planner) planExprCallFunc(name string, arity int, void bool, operands []*ast.Term, args []ir.Operand, iter planiter) error { if len(operands) == arity { // definition: f(x) = y { ... } // call: f(x) # result not captured - return p.planCallArgs(operands, 0, args, func(args []ir.LocalOrConst) error { - p.ltarget = p.newLocal() - ltarget := p.ltarget.(ir.Local) + return p.planCallArgs(operands, 0, args, func(args []ir.Operand) error { + p.ltarget = p.newOperand() + ltarget := p.ltarget.Value.(ir.Local) p.appendStmt(&ir.CallStmt{ Func: name, Args: args, @@ -832,8 +957,8 @@ func (p *Planner) planExprCallFunc(name string, arity int, void bool, operands [ if !void { p.appendStmt(&ir.NotEqualStmt{ - A: ltarget, - B: ir.Bool(false), + A: op(ltarget), + B: op(ir.Bool(false)), }) } @@ -842,19 +967,42 @@ func (p *Planner) planExprCallFunc(name string, arity int, void bool, operands [ } // definition: f(x) = y { ... } - // call: f(x, 1) # caller captures result - return p.planCallArgs(operands[:len(operands)-1], 0, args, func(args []ir.LocalOrConst) error { + // call: f(x, 1) # caller captures result + return p.planCallArgs(operands[:len(operands)-1], 0, args, func(args []ir.Operand) error { result := p.newLocal() p.appendStmt(&ir.CallStmt{ Func: name, Args: args, Result: result, }) - return p.planUnifyLocal(result, operands[len(operands)-1], iter) + return p.planUnifyLocal(op(result), operands[len(operands)-1], iter) + }) +} + +func (p *Planner) planExprCallValue(value *ast.Term, arity int, operands []*ast.Term, iter planiter) error { + if len(operands) == arity { // call: f(x) # result not captured + return p.planCallArgs(operands, 0, nil, func([]ir.Operand) error { + p.ltarget = p.newOperand() + return p.planTerm(value, func() error { + p.appendStmt(&ir.NotEqualStmt{ + A: p.ltarget, + B: op(ir.Bool(false)), + }) + return iter() + }) + }) + } + + // call: f(x, 1) # caller captures result + return p.planCallArgs(operands[:len(operands)-1], 0, nil, func([]ir.Operand) error { + p.ltarget = p.newOperand() + return p.planTerm(value, func() error { + return p.planUnifyLocal(p.ltarget, operands[len(operands)-1], iter) + }) }) } -func (p *Planner) planCallArgs(terms []*ast.Term, idx int, args []ir.LocalOrConst, iter func([]ir.LocalOrConst) error) error { +func (p *Planner) planCallArgs(terms []*ast.Term, idx int, args []ir.Operand, iter func([]ir.Operand) error) error { if idx >= len(terms) { return iter(args) } @@ -905,7 +1053,7 @@ func (p *Planner) planUnify(a, b *ast.Term, iter planiter) error { func (p *Planner) planUnifyVar(a ast.Var, b *ast.Term, iter planiter) error { - if la, ok := p.vars.Get(a); ok { + if la, ok := p.vars.GetOp(a); ok { return p.planUnifyLocal(la, b, iter) } @@ -920,7 +1068,7 @@ func (p *Planner) planUnifyVar(a ast.Var, b *ast.Term, iter planiter) error { }) } -func (p *Planner) planUnifyLocal(a ir.LocalOrConst, b *ast.Term, iter planiter) error { +func (p *Planner) planUnifyLocal(a ir.Operand, b *ast.Term, iter planiter) error { switch vb := b.Value.(type) { case ast.Null, ast.Boolean, ast.Number, ast.String, ast.Ref, ast.Set, *ast.SetComprehension, *ast.ArrayComprehension, *ast.ObjectComprehension: return p.planTerm(b, func() error { @@ -931,7 +1079,7 @@ func (p *Planner) planUnifyLocal(a ir.LocalOrConst, b *ast.Term, iter planiter) return iter() }) case ast.Var: - if lv, ok := p.vars.Get(vb); ok { + if lv, ok := p.vars.GetOp(vb); ok { p.appendStmt(&ir.EqualStmt{ A: a, B: lv, @@ -954,7 +1102,7 @@ func (p *Planner) planUnifyLocal(a ir.LocalOrConst, b *ast.Term, iter planiter) return fmt.Errorf("not implemented: unifyLocal(%v, %v)", a, b) } -func (p *Planner) planUnifyLocalArray(a ir.LocalOrConst, b *ast.Array, iter planiter) error { +func (p *Planner) planUnifyLocalArray(a ir.Operand, b *ast.Array, iter planiter) error { p.appendStmt(&ir.IsArrayStmt{ Source: a, }) @@ -973,8 +1121,8 @@ func (p *Planner) planUnifyLocalArray(a ir.LocalOrConst, b *ast.Array, iter plan }) p.appendStmt(&ir.EqualStmt{ - A: alen, - B: blen, + A: op(alen), + B: op(blen), }) lkey := p.newLocal() @@ -988,7 +1136,7 @@ func (p *Planner) planUnifyLocalArray(a ir.LocalOrConst, b *ast.Array, iter plan return p.planUnifyLocalArrayRec(a, 0, b, lkey, lval, iter) } -func (p *Planner) planUnifyLocalArrayRec(a ir.LocalOrConst, index int, b *ast.Array, lkey, lval ir.Local, iter planiter) error { +func (p *Planner) planUnifyLocalArrayRec(a ir.Operand, index int, b *ast.Array, lkey, lval ir.Local, iter planiter) error { if b.Len() == index { return iter() } @@ -999,12 +1147,12 @@ func (p *Planner) planUnifyLocalArrayRec(a ir.LocalOrConst, index int, b *ast.Ar }) p.appendStmt(&ir.DotStmt{ - Source: a.(ir.Local), - Key: lkey, + Source: a, + Key: op(lkey), Target: lval, }) - return p.planUnifyLocal(lval, b.Elem(index), func() error { + return p.planUnifyLocal(op(lval), b.Elem(index), func() error { return p.planUnifyLocalArrayRec(a, index+1, b, lkey, lval, iter) }) } @@ -1034,7 +1182,7 @@ func (p *Planner) planUnifyObjects(a, b ast.Object, iter planiter) error { }) } -func (p *Planner) planUnifyObjectLocals(a, b ir.LocalOrConst, keys []*ast.Term, index int, l0, l1 ir.Local, iter planiter) error { +func (p *Planner) planUnifyObjectLocals(a, b ir.Operand, keys []*ast.Term, index int, l0, l1 ir.Local, iter planiter) error { if index == len(keys) { return iter() } @@ -1051,15 +1199,15 @@ func (p *Planner) planUnifyObjectLocals(a, b ir.LocalOrConst, keys []*ast.Term, Target: l1, }) p.appendStmt(&ir.EqualStmt{ - A: l0, - B: l1, + A: op(l0), + B: op(l1), }) return p.planUnifyObjectLocals(a, b, keys, index+1, l0, l1, iter) }) } -func (p *Planner) planUnifyLocalObject(a ir.LocalOrConst, b ast.Object, iter planiter) error { +func (p *Planner) planUnifyLocalObject(a ir.Operand, b ast.Object, iter planiter) error { p.appendStmt(&ir.IsObjectStmt{ Source: a, }) @@ -1078,8 +1226,8 @@ func (p *Planner) planUnifyLocalObject(a ir.LocalOrConst, b ast.Object, iter pla }) p.appendStmt(&ir.EqualStmt{ - A: alen, - B: blen, + A: op(alen), + B: op(blen), }) lval := p.newLocal() @@ -1088,7 +1236,7 @@ func (p *Planner) planUnifyLocalObject(a ir.LocalOrConst, b ast.Object, iter pla return p.planUnifyLocalObjectRec(a, 0, bkeys, b, lval, iter) } -func (p *Planner) planUnifyLocalObjectRec(a ir.LocalOrConst, index int, keys []*ast.Term, b ast.Object, lval ir.Local, iter planiter) error { +func (p *Planner) planUnifyLocalObjectRec(a ir.Operand, index int, keys []*ast.Term, b ast.Object, lval ir.Local, iter planiter) error { if index == len(keys) { return iter() @@ -1100,7 +1248,7 @@ func (p *Planner) planUnifyLocalObjectRec(a ir.LocalOrConst, index int, keys []* Key: p.ltarget, Target: lval, }) - return p.planUnifyLocal(lval, b.Get(keys[index]), func() error { + return p.planUnifyLocal(op(lval), b.Get(keys[index]), func() error { return p.planUnifyLocalObjectRec(a, index+1, keys, b, lval, iter) }) }) @@ -1174,14 +1322,14 @@ func (p *Planner) planNull(null ast.Null, iter planiter) error { Target: target, }) - p.ltarget = target + p.ltarget = op(target) return iter() } func (p *Planner) planBoolean(b ast.Boolean, iter planiter) error { - p.ltarget = ir.Bool(b) + p.ltarget = op(ir.Bool(b)) return iter() } @@ -1195,21 +1343,21 @@ func (p *Planner) planNumber(num ast.Number, iter planiter) error { Target: target, }) - p.ltarget = target + p.ltarget = op(target) return iter() } func (p *Planner) planString(str ast.String, iter planiter) error { - p.ltarget = ir.StringIndex(p.getStringConst(string(str))) + p.ltarget = op(ir.StringIndex(p.getStringConst(string(str)))) return iter() } func (p *Planner) planVar(v ast.Var, iter planiter) error { - p.ltarget = p.vars.GetOrElse(v, func() ir.Local { + p.ltarget = op(p.vars.GetOrElse(v, func() ir.Local { return p.newLocal() - }) + })) return iter() } @@ -1227,7 +1375,7 @@ func (p *Planner) planArray(arr *ast.Array, iter planiter) error { func (p *Planner) planArrayRec(arr *ast.Array, index int, larr ir.Local, iter planiter) error { if index == arr.Len() { - p.ltarget = larr + p.ltarget = op(larr) return iter() } @@ -1255,7 +1403,7 @@ func (p *Planner) planObject(obj ast.Object, iter planiter) error { func (p *Planner) planObjectRec(obj ast.Object, index int, keys []*ast.Term, lobj ir.Local, iter planiter) error { if index == len(keys) { - p.ltarget = lobj + p.ltarget = op(lobj) return iter() } @@ -1287,7 +1435,7 @@ func (p *Planner) planSet(set ast.Set, iter planiter) error { func (p *Planner) planSetRec(set ast.Set, index int, elems []*ast.Term, lset ir.Local, iter planiter) error { if index == len(elems) { - p.ltarget = lset + p.ltarget = op(lset) return iter() } @@ -1387,7 +1535,7 @@ func (p *Planner) planComprehension(body ast.Body, closureIter planiter, target }, }) - p.ltarget = target + p.ltarget = op(target) return iter() } @@ -1410,7 +1558,7 @@ func (p *Planner) planRef(ref ast.Ref, iter planiter) error { }) } - p.ltarget, ok = p.vars.Get(head) + p.ltarget, ok = p.vars.GetOp(head) if !ok { return fmt.Errorf("illegal ref: unsafe head") } @@ -1473,23 +1621,26 @@ func (p *Planner) planRefData(virtual *ruletrie, base *baseptr, ref ast.Ref, ind // We're planning a structure like this: // - // block a - // block b - // block c1 - // opa_mapping_lookup || br c1 - // call_indirect || br a - // br b - // end - // block c2 - // dot i || br c2 - // dot i+1 || br c2 - // br b + // block res + // block a + // block b + // block c1 + // opa_mapping_lookup || br c1 + // call_indirect || br res + // br b + // end + // block c2 + // dot i || br c2 + // dot i+1 || br c2 + // br b + // end + // br a // end - // br a - // end - // dot i+2 || br a - // dot i+3 || br a - // end + // dot i+2 || br res + // dot i+3 || br res + // end; a + // [add_to_result_set] + // end; res // // We have to do it like this because the dot IR stmts // are compiled to `br 0`, the innermost block, if they @@ -1499,7 +1650,7 @@ func (p *Planner) planRefData(virtual *ruletrie, base *baseptr, ref ast.Ref, ind // to call_dynamic. ltarget := p.newLocal() - p.ltarget = ltarget + p.ltarget = op(ltarget) prev := p.curr callDynBlock := &ir.Block{} // "c1" in the sketch @@ -1516,22 +1667,22 @@ func (p *Planner) planRefData(virtual *ruletrie, base *baseptr, ref ast.Ref, ind dotBlock := &ir.Block{} // "c2" in the sketch above p.curr = dotBlock - p.ltarget = p.vars.GetOrEmpty(ast.DefaultRootDocument.Value.(ast.Var)) + p.ltarget = p.vars.GetOpOrEmpty(ast.DefaultRootDocument.Value.(ast.Var)) return p.planRefRec(ref[:index+1], 1, func() error { p.appendStmt(&ir.AssignVarStmt{ - Source: p.ltarget.(ir.Local), + Source: p.ltarget, Target: ltarget, }) p.appendStmt(&ir.BreakStmt{Index: 1}) - p.ltarget = ltarget + p.ltarget = op(ltarget) outerBlock := &ir.Block{Stmts: []ir.Stmt{ &ir.BlockStmt{Blocks: []*ir.Block{ { // block "b" in the sketch above Stmts: []ir.Stmt{ &ir.BlockStmt{Blocks: []*ir.Block{callDynBlock, dotBlock}}, - &ir.BreakStmt{Index: 1}}, + &ir.BreakStmt{Index: 2}}, }, }}, }} @@ -1558,7 +1709,7 @@ func (p *Planner) planRefData(virtual *ruletrie, base *baseptr, ref ast.Ref, ind rules := vchild.Rules() if len(rules) > 0 { - p.ltarget = p.newLocal() + p.ltarget = p.newOperand() funcName, err := p.planRules(rules) if err != nil { @@ -1566,12 +1717,9 @@ func (p *Planner) planRefData(virtual *ruletrie, base *baseptr, ref ast.Ref, ind } p.appendStmt(&ir.CallStmt{ - Func: funcName, - Args: []ir.LocalOrConst{ - p.vars.GetOrEmpty(ast.InputRootDocument.Value.(ast.Var)), - p.vars.GetOrEmpty(ast.DefaultRootDocument.Value.(ast.Var)), - }, - Result: p.ltarget.(ir.Local), + Func: funcName, + Args: p.defaultOperands(), + Result: p.ltarget.Value.(ir.Local), }) return p.planRefRec(ref, index+1, iter) @@ -1629,11 +1777,11 @@ func (p *Planner) planRefData(virtual *ruletrie, base *baseptr, ref ast.Ref, ind // trees that are rooted at the same key as any of the virtual sub trees. To // prevent this we build a set of keys that are to be excluded and check // below during the base scan. - var lexclude *ir.Local + var lexclude *ir.Operand if exclude.Len() > 0 { if err := p.planSet(exclude, func() error { - v := p.ltarget.(ir.Local) + v := p.ltarget lexclude = &v return nil }); err != nil { @@ -1643,7 +1791,7 @@ func (p *Planner) planRefData(virtual *ruletrie, base *baseptr, ref ast.Ref, ind // Perform a scan of the base documents starting from the location referred // to by the 'path' data pointer. Use the `lexclude` set to avoid revisiting // sub trees. - p.ltarget = base.local + p.ltarget = op(base.local) return p.planRefRec(base.path, 0, func() error { return p.planScan(ref[index], func(lkey ir.Local) error { if lexclude != nil { @@ -1651,24 +1799,24 @@ func (p *Planner) planRefData(virtual *ruletrie, base *baseptr, ref ast.Ref, ind p.appendStmt(&ir.NotStmt{ Block: p.blockWithStmt(&ir.DotStmt{ Source: *lexclude, - Key: lkey, + Key: op(lkey), Target: lignore, })}) } // Assume that virtual sub trees have been visited already so // recurse without the virtual node. - return p.planRefData(nil, &baseptr{local: p.ltarget.(ir.Local)}, ref, index+1, iter) + return p.planRefData(nil, &baseptr{local: p.ltarget.Value.(ir.Local)}, ref, index+1, iter) }) }) } // There is nothing to exclude, so we do the same thing done above, but // use planRefRec to avoid the scan if ref[index] is ground or seen. - p.ltarget = base.local + p.ltarget = op(base.local) base.path = append(base.path, ref[index]) return p.planRefRec(base.path, 0, func() error { - return p.planRefData(nil, &baseptr{local: p.ltarget.(ir.Local)}, ref, index+1, iter) + return p.planRefData(nil, &baseptr{local: p.ltarget.Value.(ir.Local)}, ref, index+1, iter) }) } @@ -1705,7 +1853,7 @@ func (p *Planner) planRefDataExtent(virtual *ruletrie, base *baseptr, iter plani err := p.planRefDataExtent(child, nil, func() error { p.appendStmt(&ir.ObjectInsertStmt{ Object: vtarget, - Key: lkey, + Key: op(lkey), Value: p.ltarget, }) return nil @@ -1727,17 +1875,14 @@ func (p *Planner) planRefDataExtent(virtual *ruletrie, base *baseptr, iter plani // Add leaf to object if defined. b := &ir.Block{} p.appendStmtToBlock(&ir.CallStmt{ - Func: funcName, - Args: []ir.LocalOrConst{ - p.vars.GetOrEmpty(ast.InputRootDocument.Value.(ast.Var)), - p.vars.GetOrEmpty(ast.DefaultRootDocument.Value.(ast.Var)), - }, + Func: funcName, + Args: p.defaultOperands(), Result: lvalue, }, b) p.appendStmtToBlock(&ir.ObjectInsertStmt{ Object: vtarget, - Key: lkey, - Value: lvalue, + Key: op(lkey), + Value: op(lvalue), }, b) p.appendStmt(&ir.BlockStmt{Blocks: []*ir.Block{b}}) } @@ -1746,7 +1891,7 @@ func (p *Planner) planRefDataExtent(virtual *ruletrie, base *baseptr, iter plani // document at ref. If the base pointer is unset, no further processing // is required. if base == nil { - p.ltarget = vtarget + p.ltarget = op(vtarget) return iter() } } @@ -1755,16 +1900,16 @@ func (p *Planner) planRefDataExtent(virtual *ruletrie, base *baseptr, iter plani // document value above if needed. prev := p.curr p.curr = &ir.Block{} - p.ltarget = base.local + p.ltarget = op(base.local) target := p.newLocal() err := p.planRefRec(base.path, 0, func() error { if virtual == nil { - target = p.ltarget.(ir.Local) + target = p.ltarget.Value.(ir.Local) } else { stmt := &ir.ObjectMergeStmt{ - A: p.ltarget.(ir.Local), + A: p.ltarget.Value.(ir.Local), B: vtarget, Target: target, } @@ -1788,7 +1933,7 @@ func (p *Planner) planRefDataExtent(virtual *ruletrie, base *baseptr, iter plani if virtual != nil { p.appendStmt(&ir.AssignVarStmt{ - Source: vtarget, + Source: op(vtarget), Target: target, }) } else { @@ -1801,7 +1946,7 @@ func (p *Planner) planRefDataExtent(virtual *ruletrie, base *baseptr, iter plani // At this point, target refers to either the full extent of the base and // virtual documents at ref or just the base document at ref. - p.ltarget = target + p.ltarget = op(target) return iter() } @@ -1820,7 +1965,7 @@ func (p *Planner) planDot(key *ast.Term, iter planiter) error { Target: target, }) - p.ltarget = target + p.ltarget = op(target) return iter() }) @@ -1831,7 +1976,7 @@ type scaniter func(ir.Local) error func (p *Planner) planScan(key *ast.Term, iter scaniter) error { scan := &ir.ScanStmt{ - Source: p.ltarget.(ir.Local), + Source: p.ltarget.Value.(ir.Local), Key: p.newLocal(), Value: p.newLocal(), Block: &ir.Block{}, @@ -1840,8 +1985,8 @@ func (p *Planner) planScan(key *ast.Term, iter scaniter) error { prev := p.curr p.curr = scan.Block - if err := p.planUnifyLocal(scan.Key, key, func() error { - p.ltarget = scan.Value + if err := p.planUnifyLocal(op(scan.Key), key, func() error { + p.ltarget = op(scan.Value) return iter(scan.Key) }); err != nil { return err @@ -1857,7 +2002,7 @@ func (p *Planner) planScan(key *ast.Term, iter scaniter) error { func (p *Planner) planScanValues(val *ast.Term, iter scaniter) error { scan := &ir.ScanStmt{ - Source: p.ltarget.(ir.Local), + Source: p.ltarget.Value.(ir.Local), Key: p.newLocal(), Value: p.newLocal(), Block: &ir.Block{}, @@ -1866,8 +2011,8 @@ func (p *Planner) planScanValues(val *ast.Term, iter scaniter) error { prev := p.curr p.curr = scan.Block - if err := p.planUnifyLocal(scan.Value, val, func() error { - p.ltarget = scan.Value + if err := p.planUnifyLocal(op(scan.Value), val, func() error { + p.ltarget = op(scan.Value) return iter(scan.Value) }); err != nil { return err @@ -1879,13 +2024,13 @@ func (p *Planner) planScanValues(val *ast.Term, iter scaniter) error { return nil } -type termsliceiter func([]ir.LocalOrConst) error +type termsliceiter func([]ir.Operand) error func (p *Planner) planTermSlice(terms []*ast.Term, iter termsliceiter) error { - return p.planTermSliceRec(terms, make([]ir.LocalOrConst, len(terms)), 0, iter) + return p.planTermSliceRec(terms, make([]ir.Operand, len(terms)), 0, iter) } -func (p *Planner) planTermSliceRec(terms []*ast.Term, locals []ir.LocalOrConst, index int, iter termsliceiter) error { +func (p *Planner) planTermSliceRec(terms []*ast.Term, locals []ir.Operand, index int, iter termsliceiter) error { if index >= len(terms) { return iter(locals) } @@ -1970,6 +2115,10 @@ func (p *Planner) newLocal() ir.Local { return x } +func (p *Planner) newOperand() ir.Operand { + return op(p.newLocal()) +} + func rewrittenVar(vars map[ast.Var]ast.Var, k ast.Var) ast.Var { rw, ok := vars[k] if !ok { @@ -1984,17 +2133,17 @@ func rewrittenVar(vars map[ast.Var]ast.Var, k ast.Var) ast.Var { // allow for optimization using CallDynamicStmt. // // It's possible if all of these conditions hold: -// - all vars in ref have been seen -// - all ground terms (strings) match some child key on their respective -// layer of the ruletrie -// - there are no child trees left (only rulesets) if we're done checking -// ref +// - all vars in ref have been seen +// - all ground terms (strings) match some child key on their respective +// layer of the ruletrie +// - there are no child trees left (only rulesets) if we're done checking +// ref // // The last condition is necessary because we don't deal with _which key a // var actually matched_ -- so we don't know which subtree to evaluate // with the results. -func (p *Planner) optimizeLookup(t *ruletrie, ref ast.Ref) ([][]*ast.Rule, []ir.LocalOrConst, int, bool) { - dont := func() ([][]*ast.Rule, []ir.LocalOrConst, int, bool) { +func (p *Planner) optimizeLookup(t *ruletrie, ref ast.Ref) ([][]*ast.Rule, []ir.Operand, int, bool) { + dont := func() ([][]*ast.Rule, []ir.Operand, int, bool) { return nil, nil, 0, false } if t == nil { @@ -2080,22 +2229,22 @@ func (p *Planner) optimizeLookup(t *ruletrie, ref ast.Ref) ([][]*ast.Rule, []ir. return nil, nil, index, true } - var path []ir.LocalOrConst + var path []ir.Operand // plan generation - path = append(path, ir.StringIndex(p.getStringConst(fmt.Sprintf("g%d", p.funcs.gen())))) + path = append(path, op(ir.StringIndex(p.getStringConst(fmt.Sprintf("g%d", p.funcs.gen()))))) for i := 1; i <= index; i++ { switch r := ref[i].Value.(type) { case ast.Var: - lv, ok := p.vars.Get(r) + lv, ok := p.vars.GetOp(r) if !ok { p.debugf("no optimization of %s: ref[%d] not a seen var: %v", ref, i, ref[i]) return dont() } path = append(path, lv) case ast.String: - path = append(path, ir.StringIndex(p.getStringConst(string(r)))) + path = append(path, op(ir.StringIndex(p.getStringConst(string(r))))) } } @@ -2115,3 +2264,21 @@ func (p *Planner) unseenVars(t *ast.Term) bool { }) return unseen } + +func (p *Planner) defaultOperands() []ir.Operand { + return []ir.Operand{ + p.vars.GetOpOrEmpty(ast.InputRootDocument.Value.(ast.Var)), + p.vars.GetOpOrEmpty(ast.DefaultRootDocument.Value.(ast.Var)), + } +} + +func (p *Planner) isFunction(r ast.Ref) bool { + if node := p.rules.Lookup(r); node != nil { + return node.Arity() > 0 + } + return false +} + +func op(v ir.Val) ir.Operand { + return ir.Operand{Value: v} +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/planner/rules.go b/vendor/github.com/open-policy-agent/opa/internal/planner/rules.go index 3abf3e39..5e94fb1c 100644 --- a/vendor/github.com/open-policy-agent/opa/internal/planner/rules.go +++ b/vendor/github.com/open-policy-agent/opa/internal/planner/rules.go @@ -166,3 +166,49 @@ func (t *ruletrie) Get(k ast.Value) *ruletrie { } return nodes[len(nodes)-1] } + +type functionMocksStack struct { + stack []*functionMocksElem +} + +type functionMocksElem []frame + +type frame map[string]*ast.Term + +func newFunctionMocksStack() *functionMocksStack { + stack := &functionMocksStack{} + stack.Push() + return stack +} + +func newFunctionMocksElem() *functionMocksElem { + return &functionMocksElem{} +} + +func (s *functionMocksStack) Push() { + s.stack = append(s.stack, newFunctionMocksElem()) +} + +func (s *functionMocksStack) Pop() { + s.stack = s.stack[:len(s.stack)-1] +} + +func (s *functionMocksStack) PushFrame(f frame) { + current := s.stack[len(s.stack)-1] + *current = append(*current, f) +} + +func (s *functionMocksStack) PopFrame() { + current := s.stack[len(s.stack)-1] + *current = (*current)[:len(*current)-1] +} + +func (s *functionMocksStack) Lookup(f string) *ast.Term { + current := *s.stack[len(s.stack)-1] + for i := len(current) - 1; i >= 0; i-- { + if t, ok := current[i][f]; ok { + return t + } + } + return nil +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/planner/varstack.go b/vendor/github.com/open-policy-agent/opa/internal/planner/varstack.go index 7dbf8f43..dccff1b5 100644 --- a/vendor/github.com/open-policy-agent/opa/internal/planner/varstack.go +++ b/vendor/github.com/open-policy-agent/opa/internal/planner/varstack.go @@ -6,7 +6,7 @@ package planner import ( "github.com/open-policy-agent/opa/ast" - "github.com/open-policy-agent/opa/internal/ir" + "github.com/open-policy-agent/opa/ir" ) type varstack []map[ast.Var]ir.Local @@ -42,6 +42,19 @@ func (vs varstack) Get(k ast.Var) (ir.Local, bool) { return 0, false } +func (vs varstack) GetOpOrEmpty(k ast.Var) ir.Operand { + l := vs.GetOrEmpty(k) + return ir.Operand{Value: l} +} + +func (vs varstack) GetOp(k ast.Var) (ir.Operand, bool) { + l, ok := vs.Get(k) + if !ok { + return ir.Operand{}, false + } + return ir.Operand{Value: l}, true +} + func (vs varstack) Put(k ast.Var, v ir.Local) { vs[len(vs)-1][k] = v } diff --git a/vendor/github.com/open-policy-agent/opa/internal/ref/ref.go b/vendor/github.com/open-policy-agent/opa/internal/ref/ref.go new file mode 100644 index 00000000..6e84df4b --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/internal/ref/ref.go @@ -0,0 +1,39 @@ +// Copyright 2020 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +// Package ref implements internal helpers for references +package ref + +import ( + "errors" + "strings" + + "github.com/open-policy-agent/opa/ast" + "github.com/open-policy-agent/opa/storage" +) + +// ParseDataPath returns a ref from the slash separated path s rooted at data. +// All path segments are treated as identifier strings. +func ParseDataPath(s string) (ast.Ref, error) { + + s = "/" + strings.TrimPrefix(s, "/") + + path, ok := storage.ParsePath(s) + if !ok { + return nil, errors.New("invalid path") + } + + return path.Ref(ast.DefaultRootDocument), nil +} + +// ArrayPath will take an ast.Array and build an ast.Ref using the ast.Terms in the Array +func ArrayPath(a *ast.Array) ast.Ref { + var ref ast.Ref + + a.Foreach(func(term *ast.Term) { + ref = append(ref, term) + }) + + return ref +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/rego/opa/options.go b/vendor/github.com/open-policy-agent/opa/internal/rego/opa/options.go index 1e0477a0..ea1e339c 100644 --- a/vendor/github.com/open-policy-agent/opa/internal/rego/opa/options.go +++ b/vendor/github.com/open-policy-agent/opa/internal/rego/opa/options.go @@ -6,6 +6,7 @@ import ( "github.com/open-policy-agent/opa/ast" "github.com/open-policy-agent/opa/metrics" + "github.com/open-policy-agent/opa/topdown/builtins" "github.com/open-policy-agent/opa/topdown/cache" "github.com/open-policy-agent/opa/topdown/print" ) @@ -23,6 +24,7 @@ type EvalOpts struct { Time time.Time Seed io.Reader InterQueryBuiltinCache cache.InterQueryCache + NDBuiltinCache builtins.NDBCache PrintHook print.Hook Capabilities *ast.Capabilities } diff --git a/vendor/github.com/open-policy-agent/opa/internal/strings/strings.go b/vendor/github.com/open-policy-agent/opa/internal/strings/strings.go index 63a71be7..76fb9ee8 100644 --- a/vendor/github.com/open-policy-agent/opa/internal/strings/strings.go +++ b/vendor/github.com/open-policy-agent/opa/internal/strings/strings.go @@ -16,7 +16,7 @@ import ( // "ideal" width and returns the shortened paths by replacing the middle parts of paths // with "...", ex: bundle1/.../a/b/policy.rego func TruncateFilePaths(maxIdealWidth, maxWidth int, path ...string) (map[string]string, int) { - var canShorten [][]byte + canShorten := make([][]byte, 0, len(path)) for _, p := range path { canShorten = append(canShorten, []byte(getPathFromFirstSeparator(p))) diff --git a/vendor/github.com/open-policy-agent/opa/internal/wasm/encoding/reader.go b/vendor/github.com/open-policy-agent/opa/internal/wasm/encoding/reader.go index 1ec26512..9c9c391a 100644 --- a/vendor/github.com/open-policy-agent/opa/internal/wasm/encoding/reader.go +++ b/vendor/github.com/open-policy-agent/opa/internal/wasm/encoding/reader.go @@ -11,8 +11,6 @@ import ( "io" "io/ioutil" - "github.com/pkg/errors" - "github.com/open-policy-agent/opa/internal/leb128" "github.com/open-policy-agent/opa/internal/wasm/constant" "github.com/open-policy-agent/opa/internal/wasm/instruction" @@ -27,7 +25,7 @@ func ReadModule(r io.Reader) (*module.Module, error) { wr := &reader{r: r, n: 0} module, err := readModule(wr) if err != nil { - return nil, errors.Wrapf(err, "offset 0x%x", wr.n) + return nil, fmt.Errorf("offset 0x%x: %w", wr.n, err) } return module, nil @@ -39,7 +37,7 @@ func ReadCodeEntry(r io.Reader) (*module.CodeEntry, error) { wr := &reader{r: r, n: 0} entry, err := readCodeEntry(wr) if err != nil { - return nil, errors.Wrapf(err, "offset 0x%x", wr.n) + return nil, fmt.Errorf("offset 0x%x: %w", wr.n, err) } return entry, nil @@ -97,7 +95,7 @@ func readCodeEntry(r io.Reader) (*module.CodeEntry, error) { var entry module.CodeEntry if err := readLocals(r, &entry.Func.Locals); err != nil { - return nil, errors.Wrapf(err, "local declarations") + return nil, fmt.Errorf("local declarations: %w", err) } return &entry, readExpr(r, &entry.Func.Expr) @@ -145,61 +143,61 @@ func readSections(r io.Reader, m *module.Module) error { switch id { case constant.StartSectionID: if err := readStartSection(bufr, &m.Start); err != nil { - return errors.Wrap(err, "start section") + return fmt.Errorf("start section: %w", err) } case constant.CustomSectionID: var name string if err := readByteVectorString(bufr, &name); err != nil { - return errors.Wrap(err, "read custom section type") + return fmt.Errorf("read custom section type: %w", err) } if name == "name" { if err := readCustomNameSections(bufr, &m.Names); err != nil { - return errors.Wrap(err, "custom 'name' section") + return fmt.Errorf("custom 'name' section: %w", err) } } else { if err := readCustomSection(bufr, name, &m.Customs); err != nil { - return errors.Wrap(err, "custom section") + return fmt.Errorf("custom section: %w", err) } } case constant.TypeSectionID: if err := readTypeSection(bufr, &m.Type); err != nil { - return errors.Wrap(err, "type section") + return fmt.Errorf("type section: %w", err) } case constant.ImportSectionID: if err := readImportSection(bufr, &m.Import); err != nil { - return errors.Wrap(err, "import section") + return fmt.Errorf("import section: %w", err) } case constant.TableSectionID: if err := readTableSection(bufr, &m.Table); err != nil { - return errors.Wrap(err, "table section") + return fmt.Errorf("table section: %w", err) } case constant.MemorySectionID: if err := readMemorySection(bufr, &m.Memory); err != nil { - return errors.Wrap(err, "memory section") + return fmt.Errorf("memory section: %w", err) } case constant.GlobalSectionID: if err := readGlobalSection(bufr, &m.Global); err != nil { - return errors.Wrap(err, "global section") + return fmt.Errorf("global section: %w", err) } case constant.FunctionSectionID: if err := readFunctionSection(bufr, &m.Function); err != nil { - return errors.Wrap(err, "function section") + return fmt.Errorf("function section: %w", err) } case constant.ExportSectionID: if err := readExportSection(bufr, &m.Export); err != nil { - return errors.Wrap(err, "export section") + return fmt.Errorf("export section: %w", err) } case constant.ElementSectionID: if err := readElementSection(bufr, &m.Element); err != nil { - return errors.Wrap(err, "element section") + return fmt.Errorf("element section: %w", err) } case constant.DataSectionID: if err := readDataSection(bufr, &m.Data); err != nil { - return errors.Wrap(err, "data section") + return fmt.Errorf("data section: %w", err) } case constant.CodeSectionID: if err := readRawCodeSection(bufr, &m.Code); err != nil { - return errors.Wrap(err, "code section") + return fmt.Errorf("code section: %w", err) } default: return fmt.Errorf("illegal section id") diff --git a/vendor/github.com/open-policy-agent/opa/internal/ir/ir.go b/vendor/github.com/open-policy-agent/opa/ir/ir.go similarity index 71% rename from vendor/github.com/open-policy-agent/opa/internal/ir/ir.go rename to vendor/github.com/open-policy-agent/opa/ir/ir.go index d907bce0..d98b96e2 100644 --- a/vendor/github.com/open-policy-agent/opa/internal/ir/ir.go +++ b/vendor/github.com/open-policy-agent/opa/ir/ir.go @@ -17,34 +17,34 @@ import ( type ( // Policy represents a planned policy query. Policy struct { - Static *Static - Plans *Plans - Funcs *Funcs + Static *Static `json:"static,omitempty"` + Plans *Plans `json:"plans,omitempty"` + Funcs *Funcs `json:"funcs,omitempty"` } // Static represents a static data segment that is indexed into by the policy. Static struct { - Strings []*StringConst - BuiltinFuncs []*BuiltinFunc - Files []*StringConst + Strings []*StringConst `json:"strings,omitempty"` + BuiltinFuncs []*BuiltinFunc `json:"builtin_funcs,omitempty"` + Files []*StringConst `json:"files,omitempty"` } // BuiltinFunc represents a built-in function that may be required by the // policy. BuiltinFunc struct { - Name string - Decl *types.Function + Name string `json:"name"` + Decl *types.Function `json:"decl"` } // Plans represents a collection of named query plans to expose in the policy. Plans struct { - Plans []*Plan + Plans []*Plan `json:"plans"` } // Funcs represents a collection of planned functions to include in the // policy. Funcs struct { - Funcs []*Func + Funcs []*Func `json:"funcs"` } // Func represents a named plan (function) that can be invoked. Functions @@ -52,18 +52,18 @@ type ( // input document and data documents are always passed as the first and // second arguments (respectively). Func struct { - Name string - Params []Local - Return Local - Blocks []*Block // TODO(tsandall): should this be a plan? - Path []string // optional: if non-nil, include in data function tree + Name string `json:"name"` + Params []Local `json:"params"` + Return Local `json:"return"` + Blocks []*Block `json:"blocks"` // TODO(tsandall): should this be a plan? + Path []string `json:"path,omitempty"` // optional: if non-nil, include in data function tree } // Plan represents an ordered series of blocks to execute. Plan execution // stops when a return statement is reached. Blocks are executed in-order. Plan struct { - Name string - Blocks []*Block + Name string `json:"name"` + Blocks []*Block `json:"blocks"` } // Block represents an ordered sequence of statements to execute. Blocks are @@ -71,7 +71,7 @@ type ( // or there are no more statements. If all statements are defined but no return // statement is encountered, the block is undefined. Block struct { - Stmts []Stmt + Stmts []Stmt `json:"stmts"` } // Stmt represents an operation (e.g., comparison, loop, dot, etc.) to execute. @@ -89,32 +89,9 @@ type ( // TODO(tsandall): should this be int32 for safety? Local int - // Const represents a constant value from the policy. - Const interface { - typeMarker() - } - - // NullConst represents a null value. - NullConst struct{} - - // BooleanConst represents a boolean value. - BooleanConst struct { - Value bool - } - // StringConst represents a string value. StringConst struct { - Value string - } - - // IntConst represents an integer constant. - IntConst struct { - Value int64 - } - - // FloatConst represents a floating-point constant. - FloatConst struct { - Value float64 + Value string `json:"value"` } ) @@ -153,46 +130,23 @@ func (a *Block) String() string { return fmt.Sprintf("Block (%d statements)", len(a.Stmts)) } -func (*BooleanConst) typeMarker() {} -func (*NullConst) typeMarker() {} -func (*IntConst) typeMarker() {} -func (*FloatConst) typeMarker() {} -func (*StringConst) typeMarker() {} - -// ReturnLocalStmt represents a return statement that yields a local value. -type ReturnLocalStmt struct { - Source Local - - Location -} - -// CallStmt represents a named function call. The result should be stored in the -// result local. -type CallStmt struct { - Func string - Args []LocalOrConst - Result Local - - Location +// Operand represents a value that a statement operates on. +type Operand struct { + Value Val `json:"value"` } -// CallDynamicStmt represents an indirect (data) function call. The result should -// be stored in the result local. -type CallDynamicStmt struct { - Args []Local - Result Local - Path []LocalOrConst - - Location -} - -// LocalOrConst is a tagged union of the two types, Local and StringIndex. -type LocalOrConst interface { +// Val represents an abstract value that statements operate on. There are currently +// 3 types of values: +// +// 1. Local - a local variable that can refer to any type. +// 2. StringIndex - a string constant that refers to a compiled string. +// 3. Bool - a boolean constant. +type Val interface { fmt.Stringer - localOrConst() + typeHint() string } -func (Local) localOrConst() {} +func (Local) typeHint() string { return "local" } func (l Local) String() string { return fmt.Sprintf("Local<%d>", int(l)) } @@ -201,7 +155,7 @@ func (l Local) String() string { // of a constant string. type StringIndex int -func (StringIndex) localOrConst() {} +func (StringIndex) typeHint() string { return "string_index" } func (s StringIndex) String() string { return fmt.Sprintf("String<%d>", int(s)) } @@ -209,15 +163,42 @@ func (s StringIndex) String() string { // Bool represents a constant boolean. type Bool bool -func (Bool) localOrConst() {} +func (Bool) typeHint() string { return "bool" } func (b Bool) String() string { return fmt.Sprintf("Bool<%v>", bool(b)) } +// ReturnLocalStmt represents a return statement that yields a local value. +type ReturnLocalStmt struct { + Source Local `json:"source"` + + Location +} + +// CallStmt represents a named function call. The result should be stored in the +// result local. +type CallStmt struct { + Func string `json:"func"` + Args []Operand `json:"args"` + Result Local `json:"result"` + + Location +} + +// CallDynamicStmt represents an indirect (data) function call. The result should +// be stored in the result local. +type CallDynamicStmt struct { + Args []Local `json:"args"` + Result Local `json:"result"` + Path []Operand `json:"path"` + + Location +} + // BlockStmt represents a nested block. Nested blocks and break statements can // be used to short-circuit execution. type BlockStmt struct { - Blocks []*Block + Blocks []*Block `json:"blocks"` Location } @@ -230,7 +211,7 @@ func (a *BlockStmt) String() string { // many blocks to jump starting from zero (the current block). Execution will // continue from the end of the block that is jumped to. type BreakStmt struct { - Index uint32 + Index uint32 `json:"index"` Location } @@ -239,9 +220,9 @@ type BreakStmt struct { // The source of a DotStmt may be a scalar value in which case the statement // will be undefined. type DotStmt struct { - Source LocalOrConst - Key LocalOrConst - Target Local + Source Operand `json:"source"` + Key Operand `json:"key"` + Target Local `json:"target"` Location } @@ -249,8 +230,8 @@ type DotStmt struct { // LenStmt represents a length() operation on a local variable. The // result is stored in the target local variable. type LenStmt struct { - Source LocalOrConst - Target Local + Source Operand `json:"source"` + Target Local `json:"target"` Location } @@ -258,17 +239,17 @@ type LenStmt struct { // ScanStmt represents a linear scan over a composite value. The // source may be a scalar in which case the block will never execute. type ScanStmt struct { - Source Local - Key Local - Value Local - Block *Block + Source Local `json:"source"` + Key Local `json:"key"` + Value Local `json:"value"` + Block *Block `json:"block"` Location } // NotStmt represents a negated statement. type NotStmt struct { - Block *Block + Block *Block `json:"block"` Location } @@ -276,16 +257,16 @@ type NotStmt struct { // AssignIntStmt represents an assignment of an integer value to a // local variable. type AssignIntStmt struct { - Value int64 - Target Local + Value int64 `json:"value"` + Target Local `json:"target"` Location } // AssignVarStmt represents an assignment of one local variable to another. type AssignVarStmt struct { - Source LocalOrConst - Target Local + Source Operand `json:"source"` + Target Local `json:"target"` Location } @@ -295,30 +276,30 @@ type AssignVarStmt struct { // // TODO(tsandall): is there a better name for this? type AssignVarOnceStmt struct { - Target Local - Source LocalOrConst + Source Operand `json:"source"` + Target Local `json:"target"` Location } // ResetLocalStmt resets a local variable to 0. type ResetLocalStmt struct { - Target Local + Target Local `json:"target"` Location } // MakeNullStmt constructs a local variable that refers to a null value. type MakeNullStmt struct { - Target Local + Target Local `json:"target"` Location } // MakeNumberIntStmt constructs a local variable that refers to an integer value. type MakeNumberIntStmt struct { - Value int64 - Target Local + Value int64 `json:"value"` + Target Local `json:"target"` Location } @@ -326,73 +307,73 @@ type MakeNumberIntStmt struct { // MakeNumberRefStmt constructs a local variable that refers to a number stored as a string. type MakeNumberRefStmt struct { Index int - Target Local + Target Local `json:"target"` Location } // MakeArrayStmt constructs a local variable that refers to an array value. type MakeArrayStmt struct { - Capacity int32 - Target Local + Capacity int32 `json:"capacity"` + Target Local `json:"target"` Location } // MakeObjectStmt constructs a local variable that refers to an object value. type MakeObjectStmt struct { - Target Local + Target Local `json:"target"` Location } // MakeSetStmt constructs a local variable that refers to a set value. type MakeSetStmt struct { - Target Local + Target Local `json:"target"` Location } // EqualStmt represents an value-equality check of two local variables. type EqualStmt struct { - A LocalOrConst - B LocalOrConst + A Operand `json:"a"` + B Operand `json:"b"` Location } // NotEqualStmt represents a != check of two local variables. type NotEqualStmt struct { - A LocalOrConst - B LocalOrConst + A Operand `json:"a"` + B Operand `json:"b"` Location } // IsArrayStmt represents a dynamic type check on a local variable. type IsArrayStmt struct { - Source LocalOrConst + Source Operand `json:"source"` Location } // IsObjectStmt represents a dynamic type check on a local variable. type IsObjectStmt struct { - Source LocalOrConst + Source Operand `json:"source"` Location } // IsDefinedStmt represents a check of whether a local variable is defined. type IsDefinedStmt struct { - Source Local + Source Local `json:"source"` Location } // IsUndefinedStmt represents a check of whether local variable is undefined. type IsUndefinedStmt struct { - Source Local + Source Local `json:"source"` Location } @@ -400,8 +381,8 @@ type IsUndefinedStmt struct { // ArrayAppendStmt represents a dynamic append operation of a value // onto an array. type ArrayAppendStmt struct { - Value LocalOrConst - Array Local + Value Operand `json:"value"` + Array Local `json:"array"` Location } @@ -409,9 +390,9 @@ type ArrayAppendStmt struct { // ObjectInsertStmt represents a dynamic insert operation of a // key/value pair into an object. type ObjectInsertStmt struct { - Key LocalOrConst - Value LocalOrConst - Object Local + Key Operand `json:"key"` + Value Operand `json:"value"` + Object Local `json:"object"` Location } @@ -420,9 +401,9 @@ type ObjectInsertStmt struct { // pair into an object. If the key already exists and the value differs, // execution aborts with a conflict error. type ObjectInsertOnceStmt struct { - Key LocalOrConst - Value LocalOrConst - Object Local + Key Operand `json:"key"` + Value Operand `json:"value"` + Object Local `json:"object"` Location } @@ -431,17 +412,17 @@ type ObjectInsertOnceStmt struct { // the locals refer to non-object values this operation will abort with a // conflict error. Overlapping object keys are merged recursively. type ObjectMergeStmt struct { - A Local - B Local - Target Local + A Local `json:"a"` + B Local `json:"b"` + Target Local `json:"target"` Location } // SetAddStmt represents a dynamic add operation of an element into a set. type SetAddStmt struct { - Value LocalOrConst - Set Local + Value Operand `json:"value"` + Set Local `json:"set"` Location } @@ -452,10 +433,10 @@ type SetAddStmt struct { // the Local referred to by the Path do not exist, they will be created. When // the WithStmt finishes the Local is reset to it's original value. type WithStmt struct { - Local Local - Path []int - Value LocalOrConst - Block *Block + Local Local `json:"local"` + Path []int `json:"path"` + Value Operand `json:"value"` + Block *Block `json:"block"` Location } @@ -465,9 +446,9 @@ type NopStmt struct { Location } -// ResultSetAdd adds a value into the result set returned by the query plan. -type ResultSetAdd struct { - Value Local +// ResultSetAddStmt adds a value into the result set returned by the query plan. +type ResultSetAddStmt struct { + Value Local `json:"value"` Location } @@ -475,19 +456,20 @@ type ResultSetAdd struct { // Location records the filen index, and the row and column inside that file // that a statement can be connected to. type Location struct { - Index int // filename string constant index - Col, Row int + File int `json:"file"` // filename string constant index + Col int `json:"col"` + Row int `json:"row"` file, text string // only used for debugging } // SetLocation sets the Location for a given Stmt. func (l *Location) SetLocation(index, row, col int, file, text string) { *l = Location{ - Index: index, - Row: row, - Col: col, - file: file, - text: text, + File: index, + Row: row, + Col: col, + file: file, + text: text, } } diff --git a/vendor/github.com/open-policy-agent/opa/ir/marshal.go b/vendor/github.com/open-policy-agent/opa/ir/marshal.go new file mode 100644 index 00000000..69f4b5ca --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/ir/marshal.go @@ -0,0 +1,137 @@ +// Copyright 2022 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package ir + +import ( + "encoding/json" + "reflect" +) + +func (a *Block) MarshalJSON() ([]byte, error) { + var result typedBlock + result.Stmts = make([]typedStmt, len(a.Stmts)) + for i := range a.Stmts { + tpe := reflect.Indirect(reflect.ValueOf(a.Stmts[i])).Type().Name() + result.Stmts[i] = typedStmt{ + Type: tpe, + Stmt: a.Stmts[i], + } + } + return json.Marshal(result) +} + +func (a *Block) UnmarshalJSON(bs []byte) error { + var typed rawTypedBlock + if err := json.Unmarshal(bs, &typed); err != nil { + return err + } + a.Stmts = make([]Stmt, len(typed.Stmts)) + for i := range typed.Stmts { + var err error + a.Stmts[i], err = typed.Stmts[i].Unmarshal() + if err != nil { + return err + } + } + return nil +} + +func (a *Operand) MarshalJSON() ([]byte, error) { + var result typedOperand + result.Value = a.Value + result.Type = a.Value.typeHint() + return json.Marshal(result) +} + +func (a *Operand) UnmarshalJSON(bs []byte) error { + var typed rawTypedOperand + if err := json.Unmarshal(bs, &typed); err != nil { + return err + } + x := valFactories[typed.Type]() + if err := json.Unmarshal(typed.Value, &x); err != nil { + return err + } + a.Value = x + return nil +} + +type typedBlock struct { + Stmts []typedStmt `json:"stmts"` +} + +type typedStmt struct { + Type string `json:"type"` + Stmt Stmt `json:"stmt"` +} + +type rawTypedBlock struct { + Stmts []rawTypedStmt `json:"stmts"` +} + +type rawTypedStmt struct { + Type string `json:"type"` + Stmt json.RawMessage `json:"stmt"` +} + +func (raw rawTypedStmt) Unmarshal() (Stmt, error) { + x := stmtFactories[raw.Type]() + if err := json.Unmarshal(raw.Stmt, &x); err != nil { + return nil, err + } + return x, nil +} + +type rawTypedOperand struct { + Type string `json:"type"` + Value json.RawMessage `json:"value"` +} + +type typedOperand struct { + Type string `json:"type"` + Value Val `json:"value"` +} + +var stmtFactories = map[string]func() Stmt{ + "ReturnLocalStmt": func() Stmt { return &ReturnLocalStmt{} }, + "CallStmt": func() Stmt { return &CallStmt{} }, + "CallDynamicStmt": func() Stmt { return &CallDynamicStmt{} }, + "BlockStmt": func() Stmt { return &BlockStmt{} }, + "BreakStmt": func() Stmt { return &BreakStmt{} }, + "DotStmt": func() Stmt { return &DotStmt{} }, + "LenStmt": func() Stmt { return &LenStmt{} }, + "ScanStmt": func() Stmt { return &ScanStmt{} }, + "NotStmt": func() Stmt { return &NotStmt{} }, + "AssignIntStmt": func() Stmt { return &AssignIntStmt{} }, + "AssignVarStmt": func() Stmt { return &AssignVarStmt{} }, + "AssignVarOnceStmt": func() Stmt { return &AssignVarOnceStmt{} }, + "ResetLocalStmt": func() Stmt { return &ResetLocalStmt{} }, + "MakeNullStmt": func() Stmt { return &MakeNullStmt{} }, + "MakeNumberIntStmt": func() Stmt { return &MakeNumberIntStmt{} }, + "MakeNumberRefStmt": func() Stmt { return &MakeNumberRefStmt{} }, + "MakeArrayStmt": func() Stmt { return &MakeArrayStmt{} }, + "MakeObjectStmt": func() Stmt { return &MakeObjectStmt{} }, + "MakeSetStmt": func() Stmt { return &MakeSetStmt{} }, + "EqualStmt": func() Stmt { return &EqualStmt{} }, + "NotEqualStmt": func() Stmt { return &NotEqualStmt{} }, + "IsArrayStmt": func() Stmt { return &IsArrayStmt{} }, + "IsObjectStmt": func() Stmt { return &IsObjectStmt{} }, + "IsDefinedStmt": func() Stmt { return &IsDefinedStmt{} }, + "IsUndefinedStmt": func() Stmt { return &IsUndefinedStmt{} }, + "ArrayAppendStmt": func() Stmt { return &ArrayAppendStmt{} }, + "ObjectInsertStmt": func() Stmt { return &ObjectInsertStmt{} }, + "ObjectInsertOnceStmt": func() Stmt { return &ObjectInsertOnceStmt{} }, + "ObjectMergeStmt": func() Stmt { return &ObjectMergeStmt{} }, + "SetAddStmt": func() Stmt { return &SetAddStmt{} }, + "WithStmt": func() Stmt { return &WithStmt{} }, + "NopStmt": func() Stmt { return &NopStmt{} }, + "ResultSetAddStmt": func() Stmt { return &ResultSetAddStmt{} }, +} + +var valFactories = map[string]func() Val{ + "bool": func() Val { var x Bool; return &x }, + "string_index": func() Val { var x StringIndex; return &x }, + "local": func() Val { var x Local; return &x }, +} diff --git a/vendor/github.com/open-policy-agent/opa/internal/ir/pretty.go b/vendor/github.com/open-policy-agent/opa/ir/pretty.go similarity index 100% rename from vendor/github.com/open-policy-agent/opa/internal/ir/pretty.go rename to vendor/github.com/open-policy-agent/opa/ir/pretty.go diff --git a/vendor/github.com/open-policy-agent/opa/internal/ir/walk.go b/vendor/github.com/open-policy-agent/opa/ir/walk.go similarity index 100% rename from vendor/github.com/open-policy-agent/opa/internal/ir/walk.go rename to vendor/github.com/open-policy-agent/opa/ir/walk.go diff --git a/vendor/github.com/open-policy-agent/opa/loader/filter/filter.go b/vendor/github.com/open-policy-agent/opa/loader/filter/filter.go new file mode 100644 index 00000000..f0554e0a --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/loader/filter/filter.go @@ -0,0 +1,5 @@ +package filter + +import "io/fs" + +type LoaderFilter func(abspath string, info fs.FileInfo, depth int) bool diff --git a/vendor/github.com/open-policy-agent/opa/loader/internal/embedtest/bar/bar.rego b/vendor/github.com/open-policy-agent/opa/loader/internal/embedtest/bar/bar.rego new file mode 100644 index 00000000..08bbfb0a --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/loader/internal/embedtest/bar/bar.rego @@ -0,0 +1,3 @@ +package bar + +p = true { true } diff --git a/vendor/github.com/open-policy-agent/opa/loader/internal/embedtest/bar/bar.yaml b/vendor/github.com/open-policy-agent/opa/loader/internal/embedtest/bar/bar.yaml new file mode 100644 index 00000000..8baef1b4 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/loader/internal/embedtest/bar/bar.yaml @@ -0,0 +1 @@ +abc diff --git a/vendor/github.com/open-policy-agent/opa/loader/internal/embedtest/baz/qux/qux.json b/vendor/github.com/open-policy-agent/opa/loader/internal/embedtest/baz/qux/qux.json new file mode 100644 index 00000000..19765bd5 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/loader/internal/embedtest/baz/qux/qux.json @@ -0,0 +1 @@ +null diff --git a/vendor/github.com/open-policy-agent/opa/loader/internal/embedtest/foo.json b/vendor/github.com/open-policy-agent/opa/loader/internal/embedtest/foo.json new file mode 100644 index 00000000..3cc0ecbe --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/loader/internal/embedtest/foo.json @@ -0,0 +1 @@ +[1,2,3] diff --git a/vendor/github.com/open-policy-agent/opa/loader/loader.go b/vendor/github.com/open-policy-agent/opa/loader/loader.go index d2f7cee6..7371e8b5 100644 --- a/vendor/github.com/open-policy-agent/opa/loader/loader.go +++ b/vendor/github.com/open-policy-agent/opa/loader/loader.go @@ -8,6 +8,7 @@ package loader import ( "bytes" "fmt" + "io/fs" "io/ioutil" "os" "path/filepath" @@ -20,6 +21,7 @@ import ( "github.com/open-policy-agent/opa/bundle" fileurl "github.com/open-policy-agent/opa/internal/file/url" "github.com/open-policy-agent/opa/internal/merge" + "github.com/open-policy-agent/opa/loader/filter" "github.com/open-policy-agent/opa/metrics" "github.com/open-policy-agent/opa/storage" "github.com/open-policy-agent/opa/storage/inmem" @@ -55,7 +57,13 @@ func (l *Result) Compiler() (*ast.Compiler, error) { // Store returns a Store object with the documents from this loader result. func (l *Result) Store() (storage.Store, error) { - return inmem.NewFromObject(l.Documents), nil + return l.StoreWithOpts() +} + +// StoreWithOpts returns a Store object with the documents from this loader result, +// instantiated with the passed options. +func (l *Result) StoreWithOpts(opts ...inmem.Opt) (storage.Store, error) { + return inmem.NewFromObjectWithOpts(l.Documents, opts...), nil } // RegoFile represents the result of loading a single Rego source file. @@ -67,12 +75,12 @@ type RegoFile struct { // Filter defines the interface for filtering files during loading. If the // filter returns true, the file should be excluded from the result. -type Filter func(abspath string, info os.FileInfo, depth int) bool +type Filter = filter.LoaderFilter // GlobExcludeName excludes files and directories whose names do not match the // shell style pattern at minDepth or greater. func GlobExcludeName(pattern string, minDepth int) Filter { - return func(abspath string, info os.FileInfo, depth int) bool { + return func(abspath string, info fs.FileInfo, depth int) bool { match, _ := filepath.Match(pattern, info.Name()) return match && depth >= minDepth } @@ -84,7 +92,9 @@ type FileLoader interface { All(paths []string) (*Result, error) Filtered(paths []string, filter Filter) (*Result, error) AsBundle(path string) (*bundle.Bundle, error) + WithFS(fsys fs.FS) FileLoader WithMetrics(m metrics.Metrics) FileLoader + WithFilter(filter Filter) FileLoader WithBundleVerificationConfig(*bundle.VerificationConfig) FileLoader WithSkipBundleVerification(skipVerify bool) FileLoader WithProcessAnnotation(processAnnotation bool) FileLoader @@ -100,10 +110,20 @@ func NewFileLoader() FileLoader { type fileLoader struct { metrics metrics.Metrics + filter Filter bvc *bundle.VerificationConfig skipVerify bool files map[string]bundle.FileInfo opts ast.ParserOptions + fsys fs.FS +} + +// WithFS provides an fs.FS to use for loading files. You can pass nil to +// use plain IO calls (e.g. os.Open, os.Stat, etc.), this is the default +// behaviour. +func (fl *fileLoader) WithFS(fsys fs.FS) FileLoader { + fl.fsys = fsys + return fl } // WithMetrics provides the metrics instance to use while loading @@ -112,6 +132,12 @@ func (fl *fileLoader) WithMetrics(m metrics.Metrics) FileLoader { return fl } +// WithFilter specifies the filter object to use to filter files while loading +func (fl *fileLoader) WithFilter(filter Filter) FileLoader { + fl.filter = filter + return fl +} + // WithBundleVerificationConfig sets the key configuration used to verify a signed bundle func (fl *fileLoader) WithBundleVerificationConfig(config *bundle.VerificationConfig) FileLoader { fl.bvc = config @@ -139,9 +165,17 @@ func (fl fileLoader) All(paths []string) (*Result, error) { // paths while applying the given filters. If any filter returns true, the // file/directory is excluded. func (fl fileLoader) Filtered(paths []string, filter Filter) (*Result, error) { - return all(paths, filter, func(curr *Result, path string, depth int) error { - - bs, err := ioutil.ReadFile(path) + return all(fl.fsys, paths, filter, func(curr *Result, path string, depth int) error { + + var ( + bs []byte + err error + ) + if fl.fsys != nil { + bs, err = fs.ReadFile(fl.fsys, path) + } else { + bs, err = ioutil.ReadFile(path) + } if err != nil { return err } @@ -172,7 +206,7 @@ func (fl fileLoader) AsBundle(path string) (*bundle.Bundle, error) { if err != nil { return nil, err } - bundleLoader, isDir, err := GetBundleDirectoryLoader(path) + bundleLoader, isDir, err := GetBundleDirectoryLoaderWithFilter(path, fl.filter) if err != nil { return nil, err } @@ -224,13 +258,46 @@ func GetBundleDirectoryLoader(path string) (bundle.DirectoryLoader, bool, error) return bundleLoader, fi.IsDir(), nil } -// FilteredPaths return a list of files from the specified +// GetBundleDirectoryLoaderWithFilter returns a bundle directory loader which can be used to load +// files in the directory after applying the given filter. +func GetBundleDirectoryLoaderWithFilter(path string, filter Filter) (bundle.DirectoryLoader, bool, error) { + path, err := fileurl.Clean(path) + if err != nil { + return nil, false, err + } + + fi, err := os.Stat(path) + if err != nil { + return nil, false, fmt.Errorf("error reading %q: %s", path, err) + } + + var bundleLoader bundle.DirectoryLoader + + if fi.IsDir() { + bundleLoader = bundle.NewDirectoryLoader(path).WithFilter(filter) + } else { + fh, err := os.Open(path) + if err != nil { + return nil, false, err + } + bundleLoader = bundle.NewTarballLoaderWithBaseURL(fh, path).WithFilter(filter) + } + return bundleLoader, fi.IsDir(), nil +} + +// FilteredPaths is the same as FilterPathsFS using the current diretory file +// system +func FilteredPaths(paths []string, filter Filter) ([]string, error) { + return FilteredPathsFS(nil, paths, filter) +} + +// FilteredPathsFS return a list of files from the specified // paths while applying the given filters. If any filter returns true, the // file/directory is excluded. -func FilteredPaths(paths []string, filter Filter) ([]string, error) { +func FilteredPathsFS(fsys fs.FS, paths []string, filter Filter) ([]string, error) { result := []string{} - _, err := all(paths, filter, func(_ *Result, path string, _ int) error { + _, err := all(fsys, paths, filter, func(_ *Result, path string, _ int) error { result = append(result, path) return nil }) @@ -434,7 +501,7 @@ func Dirs(paths []string) []string { unique[dir] = struct{}{} } - var u []string + u := make([]string, 0, len(unique)) for k := range unique { u = append(u, k) } @@ -507,7 +574,7 @@ func newResult() *Result { } } -func all(paths []string, filter Filter, f func(*Result, string, int) error) (*Result, error) { +func all(fsys fs.FS, paths []string, filter Filter, f func(*Result, string, int) error) (*Result, error) { errs := Errors{} root := newResult() @@ -524,7 +591,7 @@ func all(paths []string, filter Filter, f func(*Result, string, int) error) (*Re } } - allRec(path, filter, &errs, loaded, 0, f) + allRec(fsys, path, filter, &errs, loaded, 0, f) } if len(errs) > 0 { @@ -534,7 +601,7 @@ func all(paths []string, filter Filter, f func(*Result, string, int) error) (*Re return root, nil } -func allRec(path string, filter Filter, errors *Errors, loaded *Result, depth int, f func(*Result, string, int) error) { +func allRec(fsys fs.FS, path string, filter Filter, errors *Errors, loaded *Result, depth int, f func(*Result, string, int) error) { path, err := fileurl.Clean(path) if err != nil { @@ -542,7 +609,12 @@ func allRec(path string, filter Filter, errors *Errors, loaded *Result, depth in return } - info, err := os.Stat(path) + var info fs.FileInfo + if fsys != nil { + info, err = fs.Stat(fsys, path) + } else { + info, err = os.Stat(path) + } if err != nil { errors.add(err) return @@ -565,14 +637,19 @@ func allRec(path string, filter Filter, errors *Errors, loaded *Result, depth in loaded = loaded.withParent(info.Name()) } - files, err := ioutil.ReadDir(path) + var files []fs.DirEntry + if fsys != nil { + files, err = fs.ReadDir(fsys, path) + } else { + files, err = os.ReadDir(path) + } if err != nil { errors.add(err) return } for _, file := range files { - allRec(filepath.Join(path, file.Name()), filter, errors, loaded, depth+1, f) + allRec(fsys, filepath.Join(path, file.Name()), filter, errors, loaded, depth+1, f) } } diff --git a/vendor/github.com/open-policy-agent/opa/metrics/metrics.go b/vendor/github.com/open-policy-agent/opa/metrics/metrics.go index 4f31a445..53cd606a 100644 --- a/vendor/github.com/open-policy-agent/opa/metrics/metrics.go +++ b/vendor/github.com/open-policy-agent/opa/metrics/metrics.go @@ -76,7 +76,7 @@ type metric struct { Value interface{} } -func (m *metrics) Info() Info { +func (*metrics) Info() Info { return Info{ Name: "", } diff --git a/vendor/github.com/open-policy-agent/opa/rego/rego.go b/vendor/github.com/open-policy-agent/opa/rego/rego.go index 4d82bab3..97c052d0 100644 --- a/vendor/github.com/open-policy-agent/opa/rego/rego.go +++ b/vendor/github.com/open-policy-agent/opa/rego/rego.go @@ -19,16 +19,17 @@ import ( bundleUtils "github.com/open-policy-agent/opa/internal/bundle" "github.com/open-policy-agent/opa/internal/compiler/wasm" "github.com/open-policy-agent/opa/internal/future" - "github.com/open-policy-agent/opa/internal/ir" "github.com/open-policy-agent/opa/internal/planner" "github.com/open-policy-agent/opa/internal/rego/opa" "github.com/open-policy-agent/opa/internal/wasm/encoding" + "github.com/open-policy-agent/opa/ir" "github.com/open-policy-agent/opa/loader" "github.com/open-policy-agent/opa/metrics" "github.com/open-policy-agent/opa/resolver" "github.com/open-policy-agent/opa/storage" "github.com/open-policy-agent/opa/storage/inmem" "github.com/open-policy-agent/opa/topdown" + "github.com/open-policy-agent/opa/topdown/builtins" "github.com/open-policy-agent/opa/topdown/cache" "github.com/open-policy-agent/opa/topdown/print" "github.com/open-policy-agent/opa/tracing" @@ -115,6 +116,7 @@ type EvalContext struct { indexing bool earlyExit bool interQueryBuiltinCache cache.InterQueryCache + ndBuiltinCache builtins.NDBCache resolvers []refResolver sortSets bool printHook print.Hook @@ -253,6 +255,14 @@ func EvalInterQueryBuiltinCache(c cache.InterQueryCache) EvalOption { } } +// EvalNDBuiltinCache sets the non-deterministic builtin cache that built-in functions can +// use during evaluation. +func EvalNDBuiltinCache(c builtins.NDBCache) EvalOption { + return func(e *EvalContext) { + e.ndBuiltinCache = c + } +} + // EvalResolver sets a Resolver for a specified ref path for this evaluation. func EvalResolver(ref ast.Ref, r resolver.Resolver) EvalOption { return func(e *EvalContext) { @@ -508,6 +518,7 @@ type Rego struct { bundles map[string]*bundle.Bundle skipBundleVerification bool interQueryBuiltinCache cache.InterQueryCache + ndBuiltinCache builtins.NDBCache strictBuiltinErrors bool resolvers []refResolver schemaSet *ast.SchemaSet @@ -521,9 +532,10 @@ type Rego struct { // Function represents a built-in function that is callable in Rego. type Function struct { - Name string - Decl *types.Function - Memoize bool + Name string + Decl *types.Function + Memoize bool + Nondeterministic bool } // BuiltinContext contains additional attributes from the evaluator that @@ -550,8 +562,9 @@ type ( // RegisterBuiltin1 adds a built-in function globally inside the OPA runtime. func RegisterBuiltin1(decl *Function, impl Builtin1) { ast.RegisterBuiltin(&ast.Builtin{ - Name: decl.Name, - Decl: decl.Decl, + Name: decl.Name, + Decl: decl.Decl, + Nondeterministic: decl.Nondeterministic, }) topdown.RegisterBuiltinFunc(decl.Name, func(bctx BuiltinContext, terms []*ast.Term, iter func(*ast.Term) error) error { result, err := memoize(decl, bctx, terms, func() (*ast.Term, error) { return impl(bctx, terms[0]) }) @@ -562,8 +575,9 @@ func RegisterBuiltin1(decl *Function, impl Builtin1) { // RegisterBuiltin2 adds a built-in function globally inside the OPA runtime. func RegisterBuiltin2(decl *Function, impl Builtin2) { ast.RegisterBuiltin(&ast.Builtin{ - Name: decl.Name, - Decl: decl.Decl, + Name: decl.Name, + Decl: decl.Decl, + Nondeterministic: decl.Nondeterministic, }) topdown.RegisterBuiltinFunc(decl.Name, func(bctx BuiltinContext, terms []*ast.Term, iter func(*ast.Term) error) error { result, err := memoize(decl, bctx, terms, func() (*ast.Term, error) { return impl(bctx, terms[0], terms[1]) }) @@ -574,8 +588,9 @@ func RegisterBuiltin2(decl *Function, impl Builtin2) { // RegisterBuiltin3 adds a built-in function globally inside the OPA runtime. func RegisterBuiltin3(decl *Function, impl Builtin3) { ast.RegisterBuiltin(&ast.Builtin{ - Name: decl.Name, - Decl: decl.Decl, + Name: decl.Name, + Decl: decl.Decl, + Nondeterministic: decl.Nondeterministic, }) topdown.RegisterBuiltinFunc(decl.Name, func(bctx BuiltinContext, terms []*ast.Term, iter func(*ast.Term) error) error { result, err := memoize(decl, bctx, terms, func() (*ast.Term, error) { return impl(bctx, terms[0], terms[1], terms[2]) }) @@ -586,8 +601,9 @@ func RegisterBuiltin3(decl *Function, impl Builtin3) { // RegisterBuiltin4 adds a built-in function globally inside the OPA runtime. func RegisterBuiltin4(decl *Function, impl Builtin4) { ast.RegisterBuiltin(&ast.Builtin{ - Name: decl.Name, - Decl: decl.Decl, + Name: decl.Name, + Decl: decl.Decl, + Nondeterministic: decl.Nondeterministic, }) topdown.RegisterBuiltinFunc(decl.Name, func(bctx BuiltinContext, terms []*ast.Term, iter func(*ast.Term) error) error { result, err := memoize(decl, bctx, terms, func() (*ast.Term, error) { return impl(bctx, terms[0], terms[1], terms[2], terms[3]) }) @@ -598,8 +614,9 @@ func RegisterBuiltin4(decl *Function, impl Builtin4) { // RegisterBuiltinDyn adds a built-in function globally inside the OPA runtime. func RegisterBuiltinDyn(decl *Function, impl BuiltinDyn) { ast.RegisterBuiltin(&ast.Builtin{ - Name: decl.Name, - Decl: decl.Decl, + Name: decl.Name, + Decl: decl.Decl, + Nondeterministic: decl.Nondeterministic, }) topdown.RegisterBuiltinFunc(decl.Name, func(bctx BuiltinContext, terms []*ast.Term, iter func(*ast.Term) error) error { result, err := memoize(decl, bctx, terms, func() (*ast.Term, error) { return impl(bctx, terms) }) @@ -1015,6 +1032,13 @@ func InterQueryBuiltinCache(c cache.InterQueryCache) func(r *Rego) { } } +// NDBuiltinCache sets the non-deterministic builtins cache. +func NDBuiltinCache(c builtins.NDBCache) func(r *Rego) { + return func(r *Rego) { + r.ndBuiltinCache = c + } +} + // StrictBuiltinErrors tells the evaluator to treat all built-in function errors as fatal errors. func StrictBuiltinErrors(yes bool) func(r *Rego) { return func(r *Rego) { @@ -1165,6 +1189,10 @@ func (r *Rego) Eval(ctx context.Context) (ResultSet, error) { EvalSeed(r.seed), } + if r.ndBuiltinCache != nil { + evalArgs = append(evalArgs, EvalNDBuiltinCache(r.ndBuiltinCache)) + } + for _, qt := range r.queryTracers { evalArgs = append(evalArgs, EvalQueryTracer(qt)) } @@ -1237,6 +1265,10 @@ func (r *Rego) Partial(ctx context.Context) (*PartialQueries, error) { EvalInterQueryBuiltinCache(r.interQueryBuiltinCache), } + if r.ndBuiltinCache != nil { + evalArgs = append(evalArgs, EvalNDBuiltinCache(r.ndBuiltinCache)) + } + for _, t := range r.queryTracers { evalArgs = append(evalArgs, EvalQueryTracer(t)) } @@ -1773,6 +1805,7 @@ func (r *Rego) parseQuery(futureImports []*ast.Import, m metrics.Metrics) (ast.B if err != nil { return nil, err } + popts.SkipRules = true return ast.ParseBodyWithOpts(r.query, popts) } @@ -1915,6 +1948,10 @@ func (r *Rego) eval(ctx context.Context, ectx *EvalContext) (ResultSet, error) { q = q.WithTime(ectx.time) } + if ectx.ndBuiltinCache != nil { + q = q.WithNDBuiltinCache(ectx.ndBuiltinCache) + } + for i := range ectx.queryTracers { q = q.WithQueryTracer(ectx.queryTracers[i]) } @@ -1970,6 +2007,7 @@ func (r *Rego) evalWasm(ctx context.Context, ectx *EvalContext) (ResultSet, erro Time: ectx.time, Seed: ectx.seed, InterQueryBuiltinCache: ectx.interQueryBuiltinCache, + NDBuiltinCache: ectx.ndBuiltinCache, PrintHook: ectx.printHook, Capabilities: ectx.capabilities, }) @@ -2147,9 +2185,10 @@ func (r *Rego) partial(ctx context.Context, ectx *EvalContext) (*PartialQueries, var unknowns []*ast.Term - if ectx.parsedUnknowns != nil { + switch { + case ectx.parsedUnknowns != nil: unknowns = ectx.parsedUnknowns - } else if ectx.unknowns != nil { + case ectx.unknowns != nil: unknowns = make([]*ast.Term, len(ectx.unknowns)) for i := range ectx.unknowns { var err error @@ -2158,7 +2197,7 @@ func (r *Rego) partial(ctx context.Context, ectx *EvalContext) (*PartialQueries, return nil, err } } - } else { + default: // Use input document as unknown if caller has not specified any. unknowns = []*ast.Term{ast.NewTerm(ast.InputRootRef)} } @@ -2188,6 +2227,10 @@ func (r *Rego) partial(ctx context.Context, ectx *EvalContext) (*PartialQueries, q = q.WithTime(ectx.time) } + if ectx.ndBuiltinCache != nil { + q = q.WithNDBuiltinCache(ectx.ndBuiltinCache) + } + for i := range ectx.queryTracers { q = q.WithQueryTracer(ectx.queryTracers[i]) } @@ -2519,8 +2562,9 @@ func finishFunction(name string, bctx topdown.BuiltinContext, result *ast.Term, func newFunction(decl *Function, f topdown.BuiltinFunc) func(*Rego) { return func(r *Rego) { r.builtinDecls[decl.Name] = &ast.Builtin{ - Name: decl.Name, - Decl: decl.Decl, + Name: decl.Name, + Decl: decl.Decl, + Nondeterministic: decl.Nondeterministic, } r.builtinFuncs[decl.Name] = &topdown.Builtin{ Decl: r.builtinDecls[decl.Name], diff --git a/vendor/github.com/open-policy-agent/opa/storage/errors.go b/vendor/github.com/open-policy-agent/opa/storage/errors.go index d83b275a..2ea68fca 100644 --- a/vendor/github.com/open-policy-agent/opa/storage/errors.go +++ b/vendor/github.com/open-policy-agent/opa/storage/errors.go @@ -99,7 +99,7 @@ func IsIndexingNotSupported(error) bool { return false } func writeConflictError(path Path) *Error { return &Error{ Code: WriteConflictErr, - Message: fmt.Sprint(path), + Message: path.String(), } } diff --git a/vendor/github.com/open-policy-agent/opa/storage/inmem/inmem.go b/vendor/github.com/open-policy-agent/opa/storage/inmem/inmem.go index b6918004..b6433795 100644 --- a/vendor/github.com/open-policy-agent/opa/storage/inmem/inmem.go +++ b/vendor/github.com/open-policy-agent/opa/storage/inmem/inmem.go @@ -19,25 +19,46 @@ import ( "context" "fmt" "io" + "path/filepath" + "strings" "sync" "sync/atomic" + "github.com/open-policy-agent/opa/internal/merge" "github.com/open-policy-agent/opa/storage" "github.com/open-policy-agent/opa/util" ) // New returns an empty in-memory store. func New() storage.Store { - return &store{ - data: map[string]interface{}{}, - triggers: map[*handle]storage.TriggerConfig{}, - policies: map[string][]byte{}, + return NewWithOpts() +} + +// NewWithOpts returns an empty in-memory store, with extra options passed. +func NewWithOpts(opts ...Opt) storage.Store { + s := &store{ + data: map[string]interface{}{}, + triggers: map[*handle]storage.TriggerConfig{}, + policies: map[string][]byte{}, + roundTripOnWrite: true, + } + + for _, opt := range opts { + opt(s) } + + return s } // NewFromObject returns a new in-memory store from the supplied data object. func NewFromObject(data map[string]interface{}) storage.Store { - db := New() + return NewFromObjectWithOpts(data) +} + +// NewFromObject returns a new in-memory store from the supplied data object, with the +// options passed. +func NewFromObjectWithOpts(data map[string]interface{}, opts ...Opt) storage.Store { + db := NewWithOpts(opts...) ctx := context.Background() txn, err := db.NewTransaction(ctx, storage.WriteParams) if err != nil { @@ -55,12 +76,18 @@ func NewFromObject(data map[string]interface{}) storage.Store { // NewFromReader returns a new in-memory store from a reader that produces a // JSON serialized object. This function is for test purposes. func NewFromReader(r io.Reader) storage.Store { + return NewFromReaderWithOpts(r) +} + +// NewFromReader returns a new in-memory store from a reader that produces a +// JSON serialized object, with extra options. This function is for test purposes. +func NewFromReaderWithOpts(r io.Reader, opts ...Opt) storage.Store { d := util.NewJSONDecoder(r) var data map[string]interface{} if err := d.Decode(&data); err != nil { panic(err) } - return NewFromObject(data) + return NewFromObjectWithOpts(data, opts...) } type store struct { @@ -70,18 +97,22 @@ type store struct { data map[string]interface{} // raw data policies map[string][]byte // raw policies triggers map[*handle]storage.TriggerConfig // registered triggers + + // roundTripOnWrite, if true, means that every call to Write round trips the + // data through JSON before adding the data to the store. Defaults to true. + roundTripOnWrite bool } type handle struct { db *store } -func (db *store) NewTransaction(ctx context.Context, params ...storage.TransactionParams) (storage.Transaction, error) { +func (db *store) NewTransaction(_ context.Context, params ...storage.TransactionParams) (storage.Transaction, error) { var write bool - var context *storage.Context + var ctx *storage.Context if len(params) > 0 { write = params[0].Write - context = params[0].Context + ctx = params[0].Context } xid := atomic.AddUint64(&db.xid, uint64(1)) if write { @@ -89,7 +120,90 @@ func (db *store) NewTransaction(ctx context.Context, params ...storage.Transacti } else { db.rmu.RLock() } - return newTransaction(xid, write, context, db), nil + return newTransaction(xid, write, ctx, db), nil +} + +// Truncate implements the storage.Store interface. This method must be called within a transaction. +func (db *store) Truncate(ctx context.Context, txn storage.Transaction, params storage.TransactionParams, it storage.Iterator) error { + var update *storage.Update + var err error + mergedData := map[string]interface{}{} + + underlying, err := db.underlying(txn) + if err != nil { + return err + } + + for { + update, err = it.Next() + if err != nil { + break + } + + if update.IsPolicy { + err = underlying.UpsertPolicy(strings.TrimLeft(update.Path.String(), "/"), update.Value) + if err != nil { + return err + } + } else { + var value interface{} + err = util.Unmarshal(update.Value, &value) + if err != nil { + return err + } + + var key []string + dirpath := strings.TrimLeft(update.Path.String(), "/") + if len(dirpath) > 0 { + key = strings.Split(dirpath, "/") + } + + if value != nil { + obj, err := mktree(key, value) + if err != nil { + return err + } + + merged, ok := merge.InterfaceMaps(mergedData, obj) + if !ok { + return fmt.Errorf("failed to insert data file from path %s", filepath.Join(key...)) + } + mergedData = merged + } + } + } + + if err != nil && err != io.EOF { + return err + } + + // For backwards compatibility, check if `RootOverwrite` was configured. + if params.RootOverwrite { + newPath, ok := storage.ParsePathEscaped("/") + if !ok { + return fmt.Errorf("storage path invalid: %v", newPath) + } + return underlying.Write(storage.AddOp, newPath, mergedData) + } + + for _, root := range params.BasePaths { + newPath, ok := storage.ParsePathEscaped("/" + root) + if !ok { + return fmt.Errorf("storage path invalid: %v", newPath) + } + + if value, ok := lookup(newPath, mergedData); ok { + if len(newPath) > 0 { + if err := storage.MakeDir(ctx, db, txn, newPath[:len(newPath)-1]); err != nil { + return err + } + } + if err := underlying.Write(storage.AddOp, newPath, value); err != nil { + return err + } + } + } + return nil } func (db *store) Commit(ctx context.Context, txn storage.Transaction) error { @@ -101,7 +215,7 @@ func (db *store) Commit(ctx context.Context, txn storage.Transaction) error { db.rmu.Lock() event := underlying.Commit() db.runOnCommitTriggers(ctx, txn, event) - // Mark the transaction stale after executing triggers so they can + // Mark the transaction stale after executing triggers, so they can // perform store operations if needed. underlying.stale = true db.rmu.Unlock() @@ -112,7 +226,7 @@ func (db *store) Commit(ctx context.Context, txn storage.Transaction) error { return nil } -func (db *store) Abort(ctx context.Context, txn storage.Transaction) { +func (db *store) Abort(_ context.Context, txn storage.Transaction) { underlying, err := db.underlying(txn) if err != nil { panic(err) @@ -160,7 +274,7 @@ func (db *store) DeletePolicy(_ context.Context, txn storage.Transaction, id str return underlying.DeletePolicy(id) } -func (db *store) Register(ctx context.Context, txn storage.Transaction, config storage.TriggerConfig) (storage.TriggerHandle, error) { +func (db *store) Register(_ context.Context, txn storage.Transaction, config storage.TriggerConfig) (storage.TriggerHandle, error) { underlying, err := db.underlying(txn) if err != nil { return nil, err @@ -176,7 +290,7 @@ func (db *store) Register(ctx context.Context, txn storage.Transaction, config s return h, nil } -func (db *store) Read(ctx context.Context, txn storage.Transaction, path storage.Path) (interface{}, error) { +func (db *store) Read(_ context.Context, txn storage.Transaction, path storage.Path) (interface{}, error) { underlying, err := db.underlying(txn) if err != nil { return nil, err @@ -184,19 +298,21 @@ func (db *store) Read(ctx context.Context, txn storage.Transaction, path storage return underlying.Read(path) } -func (db *store) Write(ctx context.Context, txn storage.Transaction, op storage.PatchOp, path storage.Path, value interface{}) error { +func (db *store) Write(_ context.Context, txn storage.Transaction, op storage.PatchOp, path storage.Path, value interface{}) error { underlying, err := db.underlying(txn) if err != nil { return err } val := util.Reference(value) - if err := util.RoundTrip(val); err != nil { - return err + if db.roundTripOnWrite { + if err := util.RoundTrip(val); err != nil { + return err + } } return underlying.Write(op, path, *val) } -func (h *handle) Unregister(ctx context.Context, txn storage.Transaction) { +func (h *handle) Unregister(_ context.Context, txn storage.Transaction) { underlying, err := h.db.underlying(txn) if err != nil { panic(err) @@ -239,8 +355,8 @@ func (db *store) underlying(txn storage.Transaction) (*transaction, error) { return underlying, nil } -var rootMustBeObjectMsg = "root must be object" -var rootCannotBeRemovedMsg = "root cannot be removed" +const rootMustBeObjectMsg = "root must be object" +const rootCannotBeRemovedMsg = "root cannot be removed" func invalidPatchError(f string, a ...interface{}) *storage.Error { return &storage.Error{ @@ -248,3 +364,43 @@ func invalidPatchError(f string, a ...interface{}) *storage.Error { Message: fmt.Sprintf(f, a...), } } + +func mktree(path []string, value interface{}) (map[string]interface{}, error) { + if len(path) == 0 { + // For 0 length path the value is the full tree. + obj, ok := value.(map[string]interface{}) + if !ok { + return nil, invalidPatchError(rootMustBeObjectMsg) + } + return obj, nil + } + + dir := map[string]interface{}{} + for i := len(path) - 1; i > 0; i-- { + dir[path[i]] = value + value = dir + dir = map[string]interface{}{} + } + dir[path[0]] = value + + return dir, nil +} + +func lookup(path storage.Path, data map[string]interface{}) (interface{}, bool) { + if len(path) == 0 { + return data, true + } + for i := 0; i < len(path)-1; i++ { + value, ok := data[path[i]] + if !ok { + return nil, false + } + obj, ok := value.(map[string]interface{}) + if !ok { + return nil, false + } + data = obj + } + value, ok := data[path[len(path)-1]] + return value, ok +} diff --git a/vendor/github.com/open-policy-agent/opa/storage/inmem/opts.go b/vendor/github.com/open-policy-agent/opa/storage/inmem/opts.go new file mode 100644 index 00000000..fb8dc8e2 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/storage/inmem/opts.go @@ -0,0 +1,25 @@ +package inmem + +// An Opt modifies store at instantiation. +type Opt func(*store) + +// OptRoundTripOnWrite sets whether incoming objects written to store are +// round-tripped through JSON to ensure they are serializable to JSON. +// +// Callers should disable this if they can guarantee all objects passed to +// Write() are serializable to JSON. Failing to do so may result in undefined +// behavior, including panics. +// +// Usually, when only storing objects in the inmem store that have been read +// via encoding/json, this is safe to disable, and comes with an improvement +// in performance and memory use. +// +// If setting to false, callers should deep-copy any objects passed to Write() +// unless they can guarantee the objects will not be mutated after being written, +// and that mutations happening to the objects after they have been passed into +// Write() don't affect their logic. +func OptRoundTripOnWrite(enabled bool) Opt { + return func(s *store) { + s.roundTripOnWrite = enabled + } +} diff --git a/vendor/github.com/open-policy-agent/opa/storage/inmem/txn.go b/vendor/github.com/open-policy-agent/opa/storage/inmem/txn.go index 2f52972b..3a610182 100644 --- a/vendor/github.com/open-policy-agent/opa/storage/inmem/txn.go +++ b/vendor/github.com/open-policy-agent/opa/storage/inmem/txn.go @@ -296,7 +296,7 @@ func newUpdate(data interface{}, op storage.PatchOp, path storage.Path, idx int, func newUpdateArray(data []interface{}, op storage.PatchOp, path storage.Path, idx int, value interface{}) (*update, error) { if idx == len(path)-1 { - if path[idx] == "-" { + if path[idx] == "-" || path[idx] == strconv.Itoa(len(data)) { if op != storage.AddOp { return nil, invalidPatchError("%v: invalid patch path", path) } @@ -311,20 +311,21 @@ func newUpdateArray(data []interface{}, op storage.PatchOp, path storage.Path, i return nil, err } - if op == storage.AddOp { + switch op { + case storage.AddOp: cpy := make([]interface{}, len(data)+1) copy(cpy[:pos], data[:pos]) copy(cpy[pos+1:], data[pos:]) cpy[pos] = value return &update{path[:len(path)-1], false, cpy}, nil - } else if op == storage.RemoveOp { + case storage.RemoveOp: cpy := make([]interface{}, len(data)-1) copy(cpy[:pos], data[:pos]) copy(cpy[pos:], data[pos+1:]) return &update{path[:len(path)-1], false, cpy}, nil - } else { + default: cpy := make([]interface{}, len(data)) copy(cpy, data) cpy[pos] = value @@ -374,6 +375,9 @@ func (u *update) Apply(data interface{}) interface{} { } switch parent := parent.(type) { case map[string]interface{}: + if parent == nil { + parent = make(map[string]interface{}, 1) + } parent[key] = u.value case []interface{}: idx, err := strconv.Atoi(key) diff --git a/vendor/github.com/open-policy-agent/opa/storage/interface.go b/vendor/github.com/open-policy-agent/opa/storage/interface.go index f1507161..6baca9a5 100644 --- a/vendor/github.com/open-policy-agent/opa/storage/interface.go +++ b/vendor/github.com/open-policy-agent/opa/storage/interface.go @@ -6,6 +6,8 @@ package storage import ( "context" + + "github.com/open-policy-agent/opa/metrics" ) // Transaction defines the interface that identifies a consistent snapshot over @@ -20,25 +22,42 @@ type Store interface { Policy // NewTransaction is called create a new transaction in the store. - NewTransaction(ctx context.Context, params ...TransactionParams) (Transaction, error) + NewTransaction(context.Context, ...TransactionParams) (Transaction, error) // Read is called to fetch a document referred to by path. - Read(ctx context.Context, txn Transaction, path Path) (interface{}, error) + Read(context.Context, Transaction, Path) (interface{}, error) // Write is called to modify a document referred to by path. - Write(ctx context.Context, txn Transaction, op PatchOp, path Path, value interface{}) error + Write(context.Context, Transaction, PatchOp, Path, interface{}) error // Commit is called to finish the transaction. If Commit returns an error, the // transaction must be automatically aborted by the Store implementation. - Commit(ctx context.Context, txn Transaction) error + Commit(context.Context, Transaction) error + + // Truncate is called to make a copy of the underlying store, write documents in the new store + // by creating multiple transactions in the new store as needed and finally swapping + // over to the new storage instance. This method must be called within a transaction on the original store. + Truncate(context.Context, Transaction, TransactionParams, Iterator) error // Abort is called to cancel the transaction. - Abort(ctx context.Context, txn Transaction) + Abort(context.Context, Transaction) +} + +// MakeDirer defines the interface a Store could realize to override the +// generic MakeDir functionality in storage.MakeDir +type MakeDirer interface { + MakeDir(context.Context, Transaction, Path) error } // TransactionParams describes a new transaction. type TransactionParams struct { + // BasePaths indicates the top-level paths where write operations will be performed in this transaction. + BasePaths []string + + // RootOverwrite is deprecated. Use BasePaths instead. + RootOverwrite bool + // Write indicates if this transaction will perform any write operations. Write bool @@ -71,6 +90,27 @@ func (ctx *Context) Put(key, value interface{}) { ctx.values[key] = value } +var metricsKey = struct{}{} + +// WithMetrics allows passing metrics via the Context. +// It puts the metrics object in the ctx, and returns the same +// ctx (not a copy) for convenience. +func (ctx *Context) WithMetrics(m metrics.Metrics) *Context { + ctx.values[metricsKey] = m + return ctx +} + +// Metrics() allows using a Context's metrics. Returns nil if metrics +// were not attached to the Context. +func (ctx *Context) Metrics() metrics.Metrics { + if m, ok := ctx.values[metricsKey]; ok { + if met, ok := m.(metrics.Metrics); ok { + return met + } + } + return nil +} + // WriteParams specifies the TransactionParams for a write transaction. var WriteParams = TransactionParams{ Write: true, @@ -90,7 +130,7 @@ const ( // interface which may be used if the backend does not support writes. type WritesNotSupported struct{} -func (WritesNotSupported) Write(ctx context.Context, txn Transaction, op PatchOp, path Path, value interface{}) error { +func (WritesNotSupported) Write(context.Context, Transaction, PatchOp, Path, interface{}) error { return writesNotSupportedError() } @@ -169,13 +209,13 @@ type TriggerConfig struct { // OnCommit is invoked when a transaction is successfully committed. The // callback is invoked with a handle to the write transaction that // successfully committed before other clients see the changes. - OnCommit func(ctx context.Context, txn Transaction, event TriggerEvent) + OnCommit func(context.Context, Transaction, TriggerEvent) } // Trigger defines the interface that stores implement to register for change // notifications when the store is changed. type Trigger interface { - Register(ctx context.Context, txn Transaction, config TriggerConfig) (TriggerHandle, error) + Register(context.Context, Transaction, TriggerConfig) (TriggerHandle, error) } // TriggersNotSupported provides default implementations of the Trigger @@ -190,5 +230,18 @@ func (TriggersNotSupported) Register(context.Context, Transaction, TriggerConfig // TriggerHandle defines the interface that can be used to unregister triggers that have // been registered on a Store. type TriggerHandle interface { - Unregister(ctx context.Context, txn Transaction) + Unregister(context.Context, Transaction) +} + +// Iterator defines the interface that can be used to read files from a directory starting with +// files at the base of the directory, then sub-directories etc. +type Iterator interface { + Next() (*Update, error) +} + +// Update contains information about a file +type Update struct { + Path Path + Value []byte + IsPolicy bool } diff --git a/vendor/github.com/open-policy-agent/opa/storage/internal/errors/errors.go b/vendor/github.com/open-policy-agent/opa/storage/internal/errors/errors.go index 74792f63..0bba74b9 100644 --- a/vendor/github.com/open-policy-agent/opa/storage/internal/errors/errors.go +++ b/vendor/github.com/open-policy-agent/opa/storage/internal/errors/errors.go @@ -30,3 +30,10 @@ func NewNotFoundErrorf(f string, a ...interface{}) *storage.Error { Message: msg, } } + +func NewWriteConflictError(p storage.Path) *storage.Error { + return &storage.Error{ + Code: storage.WriteConflictErr, + Message: p.String(), + } +} diff --git a/vendor/github.com/open-policy-agent/opa/storage/internal/ptr/ptr.go b/vendor/github.com/open-policy-agent/opa/storage/internal/ptr/ptr.go index 6778805f..56772f79 100644 --- a/vendor/github.com/open-policy-agent/opa/storage/internal/ptr/ptr.go +++ b/vendor/github.com/open-policy-agent/opa/storage/internal/ptr/ptr.go @@ -37,12 +37,32 @@ func Ptr(data interface{}, path storage.Path) (interface{}, error) { } func ValidateArrayIndex(arr []interface{}, s string, path storage.Path) (int, error) { - idx, err := strconv.Atoi(s) - if err != nil { + idx, ok := isInt(s) + if !ok { return 0, errors.NewNotFoundErrorWithHint(path, errors.ArrayIndexTypeMsg) } - if idx < 0 || idx >= len(arr) { + return inRange(idx, arr, path) +} + +// ValidateArrayIndexForWrite also checks that `s` is a valid way to address an +// array element like `ValidateArrayIndex`, but returns a `resource_conflict` error +// if it is not. +func ValidateArrayIndexForWrite(arr []interface{}, s string, i int, path storage.Path) (int, error) { + idx, ok := isInt(s) + if !ok { + return 0, errors.NewWriteConflictError(path[:i-1]) + } + return inRange(idx, arr, path) +} + +func isInt(s string) (int, bool) { + idx, err := strconv.Atoi(s) + return idx, err == nil +} + +func inRange(i int, arr []interface{}, path storage.Path) (int, error) { + if i < 0 || i >= len(arr) { return 0, errors.NewNotFoundErrorWithHint(path, errors.OutOfRangeMsg) } - return idx, nil + return i, nil } diff --git a/vendor/github.com/open-policy-agent/opa/storage/storage.go b/vendor/github.com/open-policy-agent/opa/storage/storage.go index 323a0dba..1e290c50 100644 --- a/vendor/github.com/open-policy-agent/opa/storage/storage.go +++ b/vendor/github.com/open-policy-agent/opa/storage/storage.go @@ -51,14 +51,18 @@ func WriteOne(ctx context.Context, store Store, op PatchOp, path Path, value int // MakeDir inserts an empty object at path. If the parent path does not exist, // MakeDir will create it recursively. -func MakeDir(ctx context.Context, store Store, txn Transaction, path Path) (err error) { +func MakeDir(ctx context.Context, store Store, txn Transaction, path Path) error { + + // Allow the Store implementation to deal with this in its own way. + if md, ok := store.(MakeDirer); ok { + return md.MakeDir(ctx, txn, path) + } if len(path) == 0 { return nil } node, err := store.Read(ctx, txn, path) - if err != nil { if !IsNotFound(err) { return err @@ -66,19 +70,15 @@ func MakeDir(ctx context.Context, store Store, txn Transaction, path Path) (err if err := MakeDir(ctx, store, txn, path[:len(path)-1]); err != nil { return err - } else if err := store.Write(ctx, txn, AddOp, path, map[string]interface{}{}); err != nil { - return err } - return nil + return store.Write(ctx, txn, AddOp, path, map[string]interface{}{}) } if _, ok := node.(map[string]interface{}); ok { return nil } - return writeConflictError(path) - } // Txn is a convenience function that executes f inside a new transaction diff --git a/vendor/github.com/open-policy-agent/opa/topdown/aggregates.go b/vendor/github.com/open-policy-agent/opa/topdown/aggregates.go index 63d60f56..eba6db23 100644 --- a/vendor/github.com/open-policy-agent/opa/topdown/aggregates.go +++ b/vendor/github.com/open-policy-agent/opa/topdown/aggregates.go @@ -22,7 +22,7 @@ func builtinCount(a ast.Value) (ast.Value, error) { case ast.String: return ast.IntNumberTerm(len([]rune(a))).Value, nil } - return nil, builtins.NewOperandTypeErr(1, a, "array", "object", "set") + return nil, builtins.NewOperandTypeErr(1, a, "array", "object", "set", "string") } func builtinSum(a ast.Value) (ast.Value, error) { diff --git a/vendor/github.com/open-policy-agent/opa/topdown/builtins.go b/vendor/github.com/open-policy-agent/opa/topdown/builtins.go index 0df8bb84..85071235 100644 --- a/vendor/github.com/open-policy-agent/opa/topdown/builtins.go +++ b/vendor/github.com/open-policy-agent/opa/topdown/builtins.go @@ -43,6 +43,7 @@ type ( Runtime *ast.Term // runtime information on the OPA instance Cache builtins.Cache // built-in function state cache InterQueryBuiltinCache cache.InterQueryCache // cross-query built-in function state cache + NDBuiltinCache builtins.NDBCache // cache for non-deterministic built-in state Location *ast.Location // location of built-in call Tracers []Tracer // Deprecated: Use QueryTracers instead QueryTracers []QueryTracer // tracer objects for trace() built-in function diff --git a/vendor/github.com/open-policy-agent/opa/topdown/builtins/builtins.go b/vendor/github.com/open-policy-agent/opa/topdown/builtins/builtins.go index fa0e0a28..d7c3afbd 100644 --- a/vendor/github.com/open-policy-agent/opa/topdown/builtins/builtins.go +++ b/vendor/github.com/open-policy-agent/opa/topdown/builtins/builtins.go @@ -6,11 +6,13 @@ package builtins import ( + "encoding/json" "fmt" "math/big" "strings" "github.com/open-policy-agent/opa/ast" + "github.com/open-policy-agent/opa/util" ) // Cache defines the built-in cache used by the top-down evaluation. The keys @@ -28,6 +30,85 @@ func (c Cache) Get(k interface{}) (interface{}, bool) { return v, ok } +// We use an ast.Object for the cached keys/values because a naive +// map[ast.Value]ast.Value will not correctly detect value equality of +// the member keys. +type NDBCache map[string]ast.Object + +func (c NDBCache) AsValue() ast.Value { + out := ast.NewObject() + for bname, obj := range c { + out.Insert(ast.StringTerm(bname), ast.NewTerm(obj)) + } + return out +} + +// Put updates the cache for the named built-in. +// Automatically creates the 2-level hierarchy as needed. +func (c NDBCache) Put(name string, k, v ast.Value) { + if _, ok := c[name]; !ok { + c[name] = ast.NewObject() + } + c[name].Insert(ast.NewTerm(k), ast.NewTerm(v)) +} + +// Get returns the cached value for k for the named builtin. +func (c NDBCache) Get(name string, k ast.Value) (ast.Value, bool) { + if m, ok := c[name]; ok { + v := m.Get(ast.NewTerm(k)) + if v != nil { + return v.Value, true + } + return nil, false + } + return nil, false +} + +// Convenience functions for serializing the data structure. +func (c NDBCache) MarshalJSON() ([]byte, error) { + v, err := ast.JSON(c.AsValue()) + if err != nil { + return nil, err + } + return json.Marshal(v) +} + +func (c *NDBCache) UnmarshalJSON(data []byte) error { + out := map[string]ast.Object{} + var incoming interface{} + + // Note: We use util.Unmarshal instead of json.Unmarshal to get + // correct deserialization of number types. + err := util.Unmarshal(data, &incoming) + if err != nil { + return err + } + + // Convert interface types back into ast.Value types. + nestedObject, err := ast.InterfaceToValue(incoming) + if err != nil { + return err + } + + // Reconstruct NDBCache from nested ast.Object structure. + if source, ok := nestedObject.(ast.Object); ok { + err = source.Iter(func(k, v *ast.Term) error { + if obj, ok := v.Value.(ast.Object); ok { + out[string(k.Value.(ast.String))] = obj + return nil + } + return fmt.Errorf("expected Object, got other Value type in conversion") + }) + if err != nil { + return err + } + } + + *c = out + + return nil +} + // ErrOperand represents an invalid operand has been passed to a built-in // function. Built-ins should return ErrOperand to indicate a type error has // occurred. @@ -189,24 +270,30 @@ func IntToNumber(i *big.Int) ast.Number { // StringSliceOperand converts x to a []string. If the cast fails, a descriptive error is // returned. -func StringSliceOperand(x ast.Value, pos int) ([]string, error) { - a, err := ArrayOperand(x, pos) - if err != nil { - return nil, err +func StringSliceOperand(a ast.Value, pos int) ([]string, error) { + type iterable interface { + Iter(func(*ast.Term) error) error + Len() int } - var f = make([]string, a.Len()) - for k := 0; k < a.Len(); k++ { - b := a.Elem(k) - c, ok := b.Value.(ast.String) + strs, ok := a.(iterable) + if !ok { + return nil, NewOperandTypeErr(pos, a, "array", "set") + } + + var outStrs = make([]string, 0, strs.Len()) + if err := strs.Iter(func(x *ast.Term) error { + s, ok := x.Value.(ast.String) if !ok { - return nil, NewOperandElementErr(pos, x, b.Value, "[]string") + return NewOperandElementErr(pos, a, x.Value, "string") } - - f[k] = string(c) + outStrs = append(outStrs, string(s)) + return nil + }); err != nil { + return nil, err } - return f, nil + return outStrs, nil } // RuneSliceOperand converts x to a []rune. If the cast fails, a descriptive error is diff --git a/vendor/github.com/open-policy-agent/opa/topdown/cache.go b/vendor/github.com/open-policy-agent/opa/topdown/cache.go index 96c250e1..1b7c455e 100644 --- a/vendor/github.com/open-policy-agent/opa/topdown/cache.go +++ b/vendor/github.com/open-policy-agent/opa/topdown/cache.go @@ -235,3 +235,57 @@ func newComprehensionCacheHashMap() *util.HashMap { return x.(*ast.Term).Hash() }) } + +type functionMocksStack struct { + stack []*functionMocksElem +} + +type functionMocksElem []frame + +type frame map[string]*ast.Term + +func newFunctionMocksStack() *functionMocksStack { + stack := &functionMocksStack{} + stack.Push() + return stack +} + +func newFunctionMocksElem() *functionMocksElem { + return &functionMocksElem{} +} + +func (s *functionMocksStack) Push() { + s.stack = append(s.stack, newFunctionMocksElem()) +} + +func (s *functionMocksStack) Pop() { + s.stack = s.stack[:len(s.stack)-1] +} + +func (s *functionMocksStack) PopPairs() { + current := s.stack[len(s.stack)-1] + *current = (*current)[:len(*current)-1] +} + +func (s *functionMocksStack) PutPairs(mocks [][2]*ast.Term) { + el := frame{} + for i := range mocks { + el[mocks[i][0].Value.String()] = mocks[i][1] + } + s.Put(el) +} + +func (s *functionMocksStack) Put(el frame) { + current := s.stack[len(s.stack)-1] + *current = append(*current, el) +} + +func (s *functionMocksStack) Get(f ast.Ref) (*ast.Term, bool) { + current := *s.stack[len(s.stack)-1] + for i := len(current) - 1; i >= 0; i-- { + if r, ok := current[i][f.String()]; ok { + return r, true + } + } + return nil, false +} diff --git a/vendor/github.com/open-policy-agent/opa/topdown/cache/cache.go b/vendor/github.com/open-policy-agent/opa/topdown/cache/cache.go index 5e32b39c..987b00be 100644 --- a/vendor/github.com/open-policy-agent/opa/topdown/cache/cache.go +++ b/vendor/github.com/open-policy-agent/opa/topdown/cache/cache.go @@ -130,7 +130,7 @@ func (c *cache) unsafeInsert(k ast.Value, v InterQueryCacheValue) (dropped int) return dropped } - for key := c.l.Front(); key != nil && (c.usage+size > limit); key = key.Next() { + for key := c.l.Front(); key != nil && (c.usage+size > limit); key = c.l.Front() { dropKey := key.Value.(ast.Value) c.unsafeDelete(dropKey) c.l.Remove(key) diff --git a/vendor/github.com/open-policy-agent/opa/topdown/cidr.go b/vendor/github.com/open-policy-agent/opa/topdown/cidr.go index 2123698c..e2030217 100644 --- a/vendor/github.com/open-policy-agent/opa/topdown/cidr.go +++ b/vendor/github.com/open-policy-agent/opa/topdown/cidr.go @@ -73,7 +73,7 @@ func builtinNetCIDRIntersects(a, b ast.Value) (ast.Value, error) { } // If either net contains the others starting IP they are overlapping - cidrsOverlap := (cidrnetA.Contains(cidrnetB.IP) || cidrnetB.Contains(cidrnetA.IP)) + cidrsOverlap := cidrnetA.Contains(cidrnetB.IP) || cidrnetB.Contains(cidrnetA.IP) return ast.Boolean(cidrsOverlap), nil } @@ -300,7 +300,7 @@ func evalNetCIDRMerge(networks []*net.IPNet) []*net.IPNet { return nil } - var ranges cidrBlockRanges + ranges := make(cidrBlockRanges, 0, len(networks)) // For each CIDR, create an IP range. Sort them and merge when possible. for _, network := range networks { @@ -332,25 +332,25 @@ func evalNetCIDRMerge(networks []*net.IPNet) []*net.IPNet { } func generateIPNet(term *ast.Term) (*net.IPNet, error) { - switch e := term.Value.(type) { - case ast.String: - network := &net.IPNet{} - // try to parse element as an IP first, fall back to CIDR - ip := net.ParseIP(string(e)) - if ip != nil { - network.IP = ip - network.Mask = ip.DefaultMask() - } else { - var err error - _, network, err = net.ParseCIDR(string(e)) - if err != nil { - return nil, err - } - } - return network, nil - default: + e, ok := term.Value.(ast.String) + if !ok { return nil, errors.New("element must be string") } + + // try to parse element as an IP first, fall back to CIDR + ip := net.ParseIP(string(e)) + if ip == nil { + _, network, err := net.ParseCIDR(string(e)) + return network, err + } + + if ip.To4() != nil { + return &net.IPNet{ + IP: ip, + Mask: ip.DefaultMask(), + }, nil + } + return nil, errors.New("IPv6 invalid: needs prefix length") } func mergeCIDRs(ranges cidrBlockRanges) cidrBlockRanges { diff --git a/vendor/github.com/open-policy-agent/opa/topdown/copypropagation/copypropagation.go b/vendor/github.com/open-policy-agent/opa/topdown/copypropagation/copypropagation.go index 2dee16f0..daf3bac4 100644 --- a/vendor/github.com/open-policy-agent/opa/topdown/copypropagation/copypropagation.go +++ b/vendor/github.com/open-policy-agent/opa/topdown/copypropagation/copypropagation.go @@ -5,6 +5,7 @@ package copypropagation import ( + "fmt" "sort" "github.com/open-policy-agent/opa/ast" @@ -31,6 +32,18 @@ type CopyPropagator struct { sorted []ast.Var // sorted copy of vars to ensure deterministic result ensureNonEmptyBody bool compiler *ast.Compiler + localvargen *localVarGenerator +} + +type localVarGenerator struct { + next int +} + +func (l *localVarGenerator) Generate() ast.Var { + result := ast.Var(fmt.Sprintf("__localcp%d__", l.next)) + l.next++ + return result + } // New returns a new CopyPropagator that optimizes queries while preserving vars @@ -46,7 +59,7 @@ func New(livevars ast.VarSet) *CopyPropagator { return sorted[i].Compare(sorted[j]) < 0 }) - return &CopyPropagator{livevars: livevars, sorted: sorted} + return &CopyPropagator{livevars: livevars, sorted: sorted, localvargen: &localVarGenerator{}} } // WithEnsureNonEmptyBody configures p to ensure that results are always non-empty. @@ -282,12 +295,16 @@ func (t bindingPlugTransform) plugBindingsRef(pctx *plugContext, v ast.Ref) ast. // updateBindings returns false if the expression can be killed. If the // expression is killed, the binding list is updated to map a var to value. func (p *CopyPropagator) updateBindings(pctx *plugContext, expr *ast.Expr) bool { - if pctx.negated || len(expr.With) > 0 { + switch { + case pctx.negated || len(expr.With) > 0: return true - } - if expr.IsEquality() { + + case expr.IsEquality(): a, b := expr.Operand(0), expr.Operand(1) if a.Equal(b) { + if p.livevarRef(a) { + pctx.removedEqs.Put(p.localvargen.Generate(), a.Value) + } return false } k, v, keep := p.updateBindingsEq(a, b) @@ -297,7 +314,8 @@ func (p *CopyPropagator) updateBindings(pctx *plugContext, expr *ast.Expr) bool } return false } - } else if expr.IsCall() { + + case expr.IsCall(): terms := expr.Terms.([]*ast.Term) if p.compiler.GetArity(expr.Operator()) == len(terms)-2 { // with captured output output := terms[len(terms)-1] @@ -310,6 +328,21 @@ func (p *CopyPropagator) updateBindings(pctx *plugContext, expr *ast.Expr) bool return !isNoop(expr) } +func (p *CopyPropagator) livevarRef(a *ast.Term) bool { + ref, ok := a.Value.(ast.Ref) + if !ok { + return false + } + + for _, v := range p.sorted { + if ref[0].Value.Compare(v) == 0 { + return true + } + } + + return false +} + func (p *CopyPropagator) updateBindingsEq(a, b *ast.Term) (ast.Var, ast.Value, bool) { k, v, keep := p.updateBindingsEqAsymmetric(a, b) if !keep { @@ -340,8 +373,7 @@ type plugContext struct { } type binding struct { - k ast.Value - v ast.Value + k, v ast.Value } func containedIn(value ast.Value, x interface{}) bool { @@ -374,7 +406,7 @@ func sortbindings(bindings *ast.ValueMap) []*binding { return false }) sort.Slice(sorted, func(i, j int) bool { - return sorted[i].k.Compare(sorted[j].k) < 0 + return sorted[i].k.Compare(sorted[j].k) > 0 }) return sorted } @@ -397,17 +429,21 @@ func makeDisjointSets(livevars ast.VarSet, query ast.Body) (*unionFind, bool) { a, b := expr.Operand(0), expr.Operand(1) varA, ok1 := a.Value.(ast.Var) varB, ok2 := b.Value.(ast.Var) - if ok1 && ok2 { + + switch { + case ok1 && ok2: if _, ok := uf.Merge(varA, varB); !ok { return nil, false } - } else if ok1 && ast.IsConstant(b.Value) { + + case ok1 && ast.IsConstant(b.Value): root := uf.MakeSet(varA) if root.constant != nil && !root.constant.Equal(b) { return nil, false } root.constant = b - } else if ok2 && ast.IsConstant(a.Value) { + + case ok2 && ast.IsConstant(a.Value): root := uf.MakeSet(varB) if root.constant != nil && !root.constant.Equal(a) { return nil, false @@ -422,7 +458,7 @@ func makeDisjointSets(livevars ast.VarSet, query ast.Body) (*unionFind, bool) { func isNoop(expr *ast.Expr) bool { - if !expr.IsCall() { + if !expr.IsCall() && !expr.IsEvery() { term := expr.Terms.(*ast.Term) if !ast.IsConstant(term.Value) { return false diff --git a/vendor/github.com/open-policy-agent/opa/topdown/encoding.go b/vendor/github.com/open-policy-agent/opa/topdown/encoding.go index fa1cfc88..fa9b1971 100644 --- a/vendor/github.com/open-policy-agent/opa/topdown/encoding.go +++ b/vendor/github.com/open-policy-agent/opa/topdown/encoding.go @@ -55,12 +55,10 @@ func builtinJSONIsValid(a ast.Value) (ast.Value, error) { str, err := builtins.StringOperand(a, 1) if err != nil { - return nil, err + return ast.Boolean(false), nil } - var x interface{} - err = util.UnmarshalJSON([]byte(str), &x) - return ast.Boolean(err == nil), nil + return ast.Boolean(json.Valid([]byte(str))), nil } func builtinBase64Encode(a ast.Value) (ast.Value, error) { @@ -85,7 +83,7 @@ func builtinBase64Decode(a ast.Value) (ast.Value, error) { func builtinBase64IsValid(a ast.Value) (ast.Value, error) { str, err := builtins.StringOperand(a, 1) if err != nil { - return nil, err + return ast.Boolean(false), nil } _, err = base64.StdEncoding.DecodeString(string(str)) @@ -259,7 +257,7 @@ func builtinYAMLUnmarshal(a ast.Value) (ast.Value, error) { func builtinYAMLIsValid(a ast.Value) (ast.Value, error) { str, err := builtins.StringOperand(a, 1) if err != nil { - return nil, err + return ast.Boolean(false), nil } var x interface{} diff --git a/vendor/github.com/open-policy-agent/opa/topdown/eval.go b/vendor/github.com/open-policy-agent/opa/topdown/eval.go index aacff970..0ee17061 100644 --- a/vendor/github.com/open-policy-agent/opa/topdown/eval.go +++ b/vendor/github.com/open-policy-agent/opa/topdown/eval.go @@ -16,6 +16,7 @@ import ( "github.com/open-policy-agent/opa/topdown/copypropagation" "github.com/open-policy-agent/opa/topdown/print" "github.com/open-policy-agent/opa/tracing" + "github.com/open-policy-agent/opa/types" ) type evalIterator func(*eval) error @@ -77,6 +78,8 @@ type eval struct { instr *Instrumentation builtins map[string]*Builtin builtinCache builtins.Cache + ndBuiltinCache builtins.NDBCache + functionMocks *functionMocksStack virtualCache *virtualCache comprehensionCache *comprehensionCache interQueryBuiltinCache cache.InterQueryCache @@ -242,6 +245,8 @@ func (e *eval) traceEvent(op Op, x ast.Node, msg string, target *ast.Ref) { Location: x.Loc(), Message: msg, Ref: target, + input: e.input, + bindings: e.bindings, } // Skip plugging the local variables, unless any of the tracers @@ -340,17 +345,17 @@ func (e *eval) evalStep(iter evalIterator) error { var defined bool var err error - switch terms := expr.Terms.(type) { case []*ast.Term: - if expr.IsEquality() { + switch { + case expr.IsEquality(): err = e.unify(terms[1], terms[2], func() error { defined = true err := iter(e) e.traceRedo(expr) return err }) - } else { + default: err = e.evalCall(terms, func() error { defined = true err := iter(e) @@ -374,6 +379,27 @@ func (e *eval) evalStep(iter evalIterator) error { } return nil }) + case *ast.Every: + eval := evalEvery{ + e: e, + expr: expr, + generator: ast.NewBody( + ast.Equality.Expr( + ast.RefTerm(terms.Domain, terms.Key).SetLocation(terms.Domain.Location), + terms.Value, + ).SetLocation(terms.Domain.Location), + ), + body: terms.Body, + } + err = eval.eval(func() error { + defined = true + err := iter(e) + e.traceRedo(expr) + return err + }) + + default: // guard-rail for adding extra (Expr).Terms types + return fmt.Errorf("got %T terms: %[1]v", terms) } if err != nil { @@ -423,43 +449,69 @@ func (e *eval) evalNot(iter evalIterator) error { func (e *eval) evalWith(iter evalIterator) error { expr := e.query[e.index] + + // Disable inlining on all references in the expression so the result of + // partial evaluation has the same semantics w/ the with statements + // preserved. var disable []ast.Ref + disableRef := func(x ast.Ref) bool { + disable = append(disable, x.GroundPrefix()) + return false + } if e.partial() { // If the value is unknown the with statement cannot be evaluated and so // the entire expression should be saved to be safe. In the future this // could be relaxed in certain cases (e.g., if the with statement would - // have no affect.) + // have no effect.) for _, with := range expr.With { + if isFunction(e.compiler.TypeEnv, with.Target) || // non-builtin function replaced + isOtherRef(with.Target) { // built-in replaced + + ast.WalkRefs(with.Value, disableRef) + continue + } + + // with target is data or input (not built-in) if e.saveSet.ContainsRecursive(with.Value, e.bindings) { return e.saveExprMarkUnknowns(expr, e.bindings, func() error { return e.next(iter) }) } + ast.WalkRefs(with.Target, disableRef) + ast.WalkRefs(with.Value, disableRef) } - // Disable inlining on all references in the expression so the result of - // partial evaluation has the same semantics w/ the with statements - // preserved. - ast.WalkRefs(expr, func(x ast.Ref) bool { - disable = append(disable, x.GroundPrefix()) - return false - }) + ast.WalkRefs(expr.NoWith(), disableRef) } pairsInput := [][2]*ast.Term{} pairsData := [][2]*ast.Term{} + functionMocks := [][2]*ast.Term{} targets := []ast.Ref{} for i := range expr.With { + target := expr.With[i].Target plugged := e.bindings.Plug(expr.With[i].Value) - if isInputRef(expr.With[i].Target) { - pairsInput = append(pairsInput, [...]*ast.Term{expr.With[i].Target, plugged}) - } else if isDataRef(expr.With[i].Target) { - pairsData = append(pairsData, [...]*ast.Term{expr.With[i].Target, plugged}) + switch { + // NOTE(sr): ordering matters here: isFunction's ref is also covered by isDataRef + case isFunction(e.compiler.TypeEnv, target): + functionMocks = append(functionMocks, [...]*ast.Term{target, plugged}) + + case isInputRef(target): + pairsInput = append(pairsInput, [...]*ast.Term{target, plugged}) + + case isDataRef(target): + pairsData = append(pairsData, [...]*ast.Term{target, plugged}) + + default: // target must be builtin + if _, _, ok := e.builtinFunc(target.String()); ok { + functionMocks = append(functionMocks, [...]*ast.Term{target, plugged}) + continue // don't append to disabled targets below + } } - targets = append(targets, expr.With[i].Target.Value.(ast.Ref)) + targets = append(targets, target.Value.(ast.Ref)) } input, err := mergeTermWithValues(e.input, pairsInput) @@ -480,12 +532,12 @@ func (e *eval) evalWith(iter evalIterator) error { } } - oldInput, oldData := e.evalWithPush(input, data, targets, disable) + oldInput, oldData := e.evalWithPush(input, data, functionMocks, targets, disable) err = e.evalStep(func(e *eval) error { e.evalWithPop(oldInput, oldData) err := e.next(iter) - oldInput, oldData = e.evalWithPush(input, data, targets, disable) + oldInput, oldData = e.evalWithPush(input, data, functionMocks, targets, disable) return err }) @@ -494,7 +546,7 @@ func (e *eval) evalWith(iter evalIterator) error { return err } -func (e *eval) evalWithPush(input, data *ast.Term, targets, disable []ast.Ref) (*ast.Term, *ast.Term) { +func (e *eval) evalWithPush(input, data *ast.Term, functionMocks [][2]*ast.Term, targets, disable []ast.Ref) (*ast.Term, *ast.Term) { var oldInput *ast.Term if input != nil { @@ -513,6 +565,7 @@ func (e *eval) evalWithPush(input, data *ast.Term, targets, disable []ast.Ref) ( e.virtualCache.Push() e.targetStack.Push(targets) e.inliningControl.PushDisable(disable, true) + e.functionMocks.PutPairs(functionMocks) return oldInput, oldData } @@ -522,6 +575,7 @@ func (e *eval) evalWithPop(input, data *ast.Term) { e.targetStack.Pop() e.virtualCache.Pop() e.comprehensionCache.Pop() + e.functionMocks.PopPairs() e.data = data e.input = input } @@ -641,18 +695,12 @@ func (e *eval) evalNotPartialSupport(negationID uint64, expr *ast.Expr, unknowns } // Save expression that refers to support rule set. - - terms := expr.Terms - expr.Terms = nil // Prevent unnecessary copying the terms. - cpy := expr.Copy() - expr.Terms = terms + cpy := expr.CopyWithoutTerms() if len(args) > 0 { terms := make([]*ast.Term, len(args)+1) terms[0] = term - for i := 0; i < len(args); i++ { - terms[i+1] = args[i] - } + copy(terms[1:], args) cpy.Terms = terms } else { cpy.Terms = term @@ -667,7 +715,32 @@ func (e *eval) evalCall(terms []*ast.Term, iter unifyIterator) error { ref := terms[0].Value.(ast.Ref) + var mocked bool + mock, mocked := e.functionMocks.Get(ref) + if mocked { + if m, ok := mock.Value.(ast.Ref); ok { // builtin or data function + mockCall := append([]*ast.Term{ast.NewTerm(m)}, terms[1:]...) + + e.functionMocks.Push() + err := e.evalCall(mockCall, func() error { + e.functionMocks.Pop() + err := iter() + e.functionMocks.Push() + return err + }) + e.functionMocks.Pop() + return err + } + } + // 'mocked' true now indicates that the replacement is a value: if + // it was a ref to a function, we'd have called that above. + if ref[0].Equal(ast.DefaultRootDocument) { + if mocked { + f := e.compiler.TypeEnv.Get(ref).(*types.Function) + return e.evalCallValue(len(f.FuncArgs().Args), terms, mock, iter) + } + var ir *ast.IndexResult var err error if e.partial() { @@ -678,6 +751,7 @@ func (e *eval) evalCall(terms []*ast.Term, iter unifyIterator) error { if err != nil { return err } + eval := evalFunc{ e: e, ref: ref, @@ -687,11 +761,16 @@ func (e *eval) evalCall(terms []*ast.Term, iter unifyIterator) error { return eval.eval(iter) } - bi, f, ok := e.builtinFunc(ref.String()) + builtinName := ref.String() + bi, f, ok := e.builtinFunc(builtinName) if !ok { return unsupportedBuiltinErr(e.query[e.index].Location) } + if mocked { // value replacement of built-in call + return e.evalCallValue(len(bi.Decl.Args()), terms, mock, iter) + } + if e.unknown(e.query[e.index], e.bindings) { return e.saveCall(len(bi.Decl.Args()), terms, iter) } @@ -715,6 +794,7 @@ func (e *eval) evalCall(terms []*ast.Term, iter unifyIterator) error { Runtime: e.runtime, Cache: e.builtinCache, InterQueryBuiltinCache: e.interQueryBuiltinCache, + NDBuiltinCache: e.ndBuiltinCache, Location: e.query[e.index].Location, QueryTracers: e.tracers, TraceEnabled: e.traceEnabled, @@ -732,9 +812,24 @@ func (e *eval) evalCall(terms []*ast.Term, iter unifyIterator) error { f: f, terms: terms[1:], } + return eval.eval(iter) } +func (e *eval) evalCallValue(arity int, terms []*ast.Term, mock *ast.Term, iter unifyIterator) error { + switch { + case len(terms) == arity+2: // captured var + return e.unify(terms[len(terms)-1], mock, iter) + + case len(terms) == arity+1: + if mock.Value.Compare(ast.Boolean(false)) != 0 { + return iter() + } + return nil + } + panic("unreachable") +} + func (e *eval) unify(a, b *ast.Term, iter unifyIterator) error { return e.biunify(a, b, e.bindings, e.bindings, iter) } @@ -817,20 +912,20 @@ func (e *eval) biunifyObjects(a, b ast.Object, b1, b2 *bindings, iter unifyItera b = plugKeys(b, b2) } - return e.biunifyObjectsRec(a, b, b1, b2, iter, a, 0) + return e.biunifyObjectsRec(a, b, b1, b2, iter, a, a.KeysIterator()) } -func (e *eval) biunifyObjectsRec(a, b ast.Object, b1, b2 *bindings, iter unifyIterator, keys ast.Object, idx int) error { - if idx == keys.Len() { +func (e *eval) biunifyObjectsRec(a, b ast.Object, b1, b2 *bindings, iter unifyIterator, keys ast.Object, oki ast.ObjectKeysIterator) error { + key, more := oki.Next() // Get next key from iterator. + if !more { return iter() } - key, _ := keys.Elem(idx) v2 := b.Get(key) if v2 == nil { return nil } return e.biunify(a.Get(key), v2, b1, b2, func() error { - return e.biunifyObjectsRec(a, b, b1, b2, iter, keys, idx+1) + return e.biunifyObjectsRec(a, b, b1, b2, iter, keys, oki) }) } @@ -1198,8 +1293,7 @@ func (e *eval) biunifyComprehensionObject(x *ast.ObjectComprehension, b *ast.Ter } func (e *eval) saveExpr(expr *ast.Expr, b *bindings, iter unifyIterator) error { - expr.With = e.query[e.index].With - expr.Location = e.query[e.index].Location + e.updateFromQuery(expr) e.saveStack.Push(expr, b, b) e.traceSave(expr) err := iter() @@ -1208,8 +1302,7 @@ func (e *eval) saveExpr(expr *ast.Expr, b *bindings, iter unifyIterator) error { } func (e *eval) saveExprMarkUnknowns(expr *ast.Expr, b *bindings, iter unifyIterator) error { - expr.With = e.query[e.index].With - expr.Location = e.query[e.index].Location + e.updateFromQuery(expr) declArgsLen, err := e.getDeclArgsLen(expr) if err != nil { return err @@ -1234,8 +1327,7 @@ func (e *eval) saveExprMarkUnknowns(expr *ast.Expr, b *bindings, iter unifyItera func (e *eval) saveUnify(a, b *ast.Term, b1, b2 *bindings, iter unifyIterator) error { e.instr.startTimer(partialOpSaveUnify) expr := ast.Equality.Expr(a, b) - expr.With = e.query[e.index].With - expr.Location = e.query[e.index].Location + e.updateFromQuery(expr) pops := 0 if pairs := getSavePairsFromTerm(a, b1, nil); len(pairs) > 0 { pops += len(pairs) @@ -1265,8 +1357,7 @@ func (e *eval) saveUnify(a, b *ast.Term, b1, b2 *bindings, iter unifyIterator) e func (e *eval) saveCall(declArgsLen int, terms []*ast.Term, iter unifyIterator) error { expr := ast.NewExpr(terms) - expr.With = e.query[e.index].With - expr.Location = e.query[e.index].Location + e.updateFromQuery(expr) // If call-site includes output value then partial eval must add vars in output // position to the save set. @@ -1301,7 +1392,7 @@ func (e *eval) saveInlinedNegatedExprs(exprs []*ast.Expr, iter unifyIterator) er } for _, expr := range exprs { - expr.With = with + expr.With = e.updateSavedMocks(with) e.saveStack.Push(expr, nil, nil) e.traceSave(expr) } @@ -1566,6 +1657,15 @@ func (e *eval) getDeclArgsLen(x *ast.Expr) (int, error) { return len(ir.Rules[0].Head.Args), nil } +// updateFromQuery enriches the passed expression with Location and With +// fields of the currently looked-at query item (`e.query[e.index]`). +// With values are namespaced to ensure that replacement functions of +// mocked built-ins are properly referenced in the support module. +func (e *eval) updateFromQuery(expr *ast.Expr) { + expr.With = e.updateSavedMocks(e.query[e.index].With) + expr.Location = e.query[e.index].Location +} + type evalBuiltin struct { e *eval bi *ast.Builtin @@ -1574,6 +1674,11 @@ type evalBuiltin struct { terms []*ast.Term } +// Is this builtin non-deterministic, and did the caller provide an NDBCache? +func (e *evalBuiltin) canUseNDBCache(bi *ast.Builtin) bool { + return bi.Nondeterministic && e.bctx.NDBuiltinCache != nil +} + func (e evalBuiltin) eval(iter unifyIterator) error { operands := make([]*ast.Term, len(e.terms)) @@ -1585,21 +1690,70 @@ func (e evalBuiltin) eval(iter unifyIterator) error { numDeclArgs := len(e.bi.Decl.FuncArgs().Args) e.e.instr.startTimer(evalOpBuiltinCall) + var err error - err := e.f(e.bctx, operands, func(output *ast.Term) error { + // NOTE(philipc): We sometimes have to drop the very last term off + // the args list for cases where a builtin's result is used/assigned, + // because the last term will be a generated term, not an actual + // argument to the builtin. + endIndex := len(operands) + if len(operands) > numDeclArgs { + endIndex-- + } + + // We skip evaluation of the builtin entirely if the NDBCache is + // present, and we have a non-deterministic builtin already cached. + if e.canUseNDBCache(e.bi) { + e.e.instr.stopTimer(evalOpBuiltinCall) + + // Unify against the NDBCache result if present. + if v, ok := e.bctx.NDBuiltinCache.Get(e.bi.Name, ast.NewArray(operands[:endIndex]...)); ok { + switch { + case e.bi.Decl.Result() == nil: + err = iter() + case len(operands) == numDeclArgs: + if v.Compare(ast.Boolean(false)) != 0 { + err = iter() + } // else: nothing to do, don't iter() + default: + err = e.e.unify(e.terms[endIndex], ast.NewTerm(v), iter) + } + + if err != nil { + return Halt{Err: err} + } + + return nil + } + + e.e.instr.startTimer(evalOpBuiltinCall) + + // Otherwise, we'll need to go through the normal unify flow. + } + + // Normal unification flow for builtins: + err = e.f(e.bctx, operands, func(output *ast.Term) error { e.e.instr.stopTimer(evalOpBuiltinCall) var err error - if e.bi.Decl.Result() == nil { + switch { + case e.bi.Decl.Result() == nil: err = iter() - } else if len(operands) == numDeclArgs { + case len(operands) == numDeclArgs: if output.Value.Compare(ast.Boolean(false)) != 0 { err = iter() - } - } else { - err = e.e.unify(e.terms[len(e.terms)-1], output, iter) + } // else: nothing to do, don't iter() + default: + err = e.e.unify(e.terms[endIndex], output, iter) + } + + // If the NDBCache is present, we can assume this builtin + // call was not cached earlier. + if e.canUseNDBCache(e.bi) { + // Populate the NDBCache from the output term. + e.bctx.NDBuiltinCache.Put(e.bi.Name, ast.NewArray(operands[:endIndex]...), output.Value) } if err != nil { @@ -2881,15 +3035,93 @@ func (e evalTerm) save(iter unifyIterator) error { suffix := e.ref[e.pos:] ref := make(ast.Ref, len(suffix)+1) ref[0] = v - for i := 0; i < len(suffix); i++ { - ref[i+1] = suffix[i] - } + copy(ref[1:], suffix) return e.e.biunify(ast.NewTerm(ref), e.rterm, e.bindings, e.rbindings, iter) }) } +type evalEvery struct { + e *eval + expr *ast.Expr + generator ast.Body + body ast.Body +} + +func (e evalEvery) eval(iter unifyIterator) error { + // unknowns in domain or body: save the expression, PE its body + if e.e.unknown(e.generator, e.e.bindings) || e.e.unknown(e.body, e.e.bindings) { + return e.save(iter) + } + + domain := e.e.closure(e.generator) + all := true // all generator evaluations yield one successful body evaluation + + domain.traceEnter(e.expr) + + err := domain.eval(func(child *eval) error { + if !all { + // NOTE(sr): Is this good enough? We don't have a "fail EE". + // This would do extra work, like iterating needlessly if domain was a large array. + return nil + } + body := child.closure(e.body) + body.findOne = true + body.traceEnter(e.body) + done := false + err := body.eval(func(*eval) error { + body.traceExit(e.body) + done = true + body.traceRedo(e.body) + return nil + }) + if !done { + all = false + } + + child.traceRedo(e.expr) + return err + }) + if err != nil { + return err + } + if all { + err := iter() + domain.traceExit(e.expr) + return err + } + domain.traceFail(e.expr) + return nil +} + +func (e *evalEvery) save(iter unifyIterator) error { + return e.e.saveExpr(e.plug(e.expr), e.e.bindings, iter) +} + +func (e *evalEvery) plug(expr *ast.Expr) *ast.Expr { + cpy := expr.Copy() + every := cpy.Terms.(*ast.Every) + for i := range every.Body { + switch t := every.Body[i].Terms.(type) { + case *ast.Term: + every.Body[i].Terms = e.e.bindings.PlugNamespaced(t, e.e.caller.bindings) + case []*ast.Term: + for j := 1; j < len(t); j++ { // don't plug operator, t[0] + t[j] = e.e.bindings.PlugNamespaced(t[j], e.e.caller.bindings) + } + case *ast.Every: + every.Body[i] = e.plug(every.Body[i]) + } + } + + every.Key = e.e.bindings.PlugNamespaced(every.Key, e.e.caller.bindings) + every.Value = e.e.bindings.PlugNamespaced(every.Value, e.e.caller.bindings) + every.Domain = e.e.bindings.PlugNamespaced(every.Domain, e.e.caller.bindings) + cpy.Terms = every + return cpy +} + func (e *eval) comprehensionIndex(term *ast.Term) *ast.ComprehensionIndex { if e.queryCompiler != nil { return e.queryCompiler.ComprehensionIndex(term) @@ -3096,6 +3328,23 @@ func isDataRef(term *ast.Term) bool { return false } +func isOtherRef(term *ast.Term) bool { + ref, ok := term.Value.(ast.Ref) + if !ok { + panic("unreachable") + } + return !ref.HasPrefix(ast.DefaultRootRef) && !ref.HasPrefix(ast.InputRootRef) +} + +func isFunction(env *ast.TypeEnv, ref *ast.Term) bool { + r, ok := ref.Value.(ast.Ref) + if !ok { + return false + } + _, ok = env.Get(r).(*types.Function) + return ok +} + func merge(a, b ast.Value) (ast.Value, bool) { aObj, ok1 := a.(ast.Object) bObj, ok2 := b.(ast.Object) @@ -3161,3 +3410,14 @@ func suppressEarlyExit(err error) error { } return ee.prev // nil if we're done } + +func (e *eval) updateSavedMocks(withs []*ast.With) []*ast.With { + ret := make([]*ast.With, 0, len(withs)) + for _, w := range withs { + if isOtherRef(w.Target) || isFunction(e.compiler.TypeEnv, w.Target) { + continue + } + ret = append(ret, w.Copy()) + } + return ret +} diff --git a/vendor/github.com/open-policy-agent/opa/topdown/glob.go b/vendor/github.com/open-policy-agent/opa/topdown/glob.go index 98052a0c..e22cda44 100644 --- a/vendor/github.com/open-policy-agent/opa/topdown/glob.go +++ b/vendor/github.com/open-policy-agent/opa/topdown/glob.go @@ -1,7 +1,7 @@ package topdown import ( - "fmt" + "strings" "sync" "github.com/gobwas/glob" @@ -18,22 +18,34 @@ func builtinGlobMatch(a, b, c ast.Value) (ast.Value, error) { if err != nil { return nil, err } + var delimiters []rune + switch b.(type) { + case ast.Null: + delimiters = []rune{} + case *ast.Array: + delimiters, err = builtins.RuneSliceOperand(b, 2) + if err != nil { + return nil, err + } - delimiters, err := builtins.RuneSliceOperand(b, 2) - if err != nil { - return nil, err - } - - if len(delimiters) == 0 { - delimiters = []rune{'.'} + if len(delimiters) == 0 { + delimiters = []rune{'.'} + } + default: + return nil, builtins.NewOperandTypeErr(2, b, "array", "null") } - match, err := builtins.StringOperand(c, 3) if err != nil { return nil, err } - id := fmt.Sprintf("%s-%v", pattern, delimiters) + builder := strings.Builder{} + builder.WriteString(string(pattern)) + builder.WriteRune('-') + for _, v := range delimiters { + builder.WriteRune(v) + } + id := builder.String() globCacheLock.Lock() defer globCacheLock.Unlock() @@ -46,7 +58,8 @@ func builtinGlobMatch(a, b, c ast.Value) (ast.Value, error) { globCache[id] = p } - return ast.Boolean(p.Match(string(match))), nil + m := p.Match(string(match)) + return ast.Boolean(m), nil } func builtinGlobQuoteMeta(a ast.Value) (ast.Value, error) { diff --git a/vendor/github.com/open-policy-agent/opa/topdown/graphql.go b/vendor/github.com/open-policy-agent/opa/topdown/graphql.go new file mode 100644 index 00000000..e43af700 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/topdown/graphql.go @@ -0,0 +1,462 @@ +// Copyright 2022 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package topdown + +import ( + "encoding/json" + "fmt" + "strings" + + gqlast "github.com/open-policy-agent/opa/internal/gqlparser/ast" + gqlparser "github.com/open-policy-agent/opa/internal/gqlparser/parser" + gqlvalidator "github.com/open-policy-agent/opa/internal/gqlparser/validator" + + // Side-effecting import. Triggers GraphQL library's validation rule init() functions. + _ "github.com/open-policy-agent/opa/internal/gqlparser/validator/rules" + + "github.com/open-policy-agent/opa/ast" + "github.com/open-policy-agent/opa/topdown/builtins" +) + +// Parses a GraphQL schema, and returns the GraphQL AST for the schema. +func parseSchema(schema string) (*gqlast.SchemaDocument, error) { + // NOTE(philipc): We don't include the "built-in schema defs" from the + // underlying graphql parsing library here, because those definitions + // generate enormous AST blobs. In the future, if there is demand for + // a "full-spec" version of schema ASTs, we may need to provide a + // version of this function that includes the built-in schema + // definitions. + schemaAST, err := gqlparser.ParseSchema(&gqlast.Source{Input: schema}) + if err != nil { + errorParts := strings.SplitN(err.Error(), ":", 4) + msg := strings.TrimLeft(errorParts[3], " ") + return nil, fmt.Errorf("%s in GraphQL string at location %s:%s", msg, errorParts[1], errorParts[2]) + } + return schemaAST, nil +} + +// Parses a GraphQL query, and returns the GraphQL AST for the query. +func parseQuery(query string) (*gqlast.QueryDocument, error) { + queryAST, err := gqlparser.ParseQuery(&gqlast.Source{Input: query}) + if err != nil { + errorParts := strings.SplitN(err.Error(), ":", 4) + msg := strings.TrimLeft(errorParts[3], " ") + return nil, fmt.Errorf("%s in GraphQL string at location %s:%s", msg, errorParts[1], errorParts[2]) + } + return queryAST, nil +} + +// Validates a GraphQL query against a schema, and returns an error. +// In this case, we get a wrappered error list type, and pluck out +// just the first error message in the list. +func validateQuery(schema *gqlast.Schema, query *gqlast.QueryDocument) error { + // Validate the query against the schema, erroring if there's an issue. + err := gqlvalidator.Validate(schema, query) + if err != nil { + // We use strings.TrimSuffix to remove the '.' characters that the library + // authors include on most of their validation errors. This should be safe, + // since variable names in their error messages are usually quoted, and + // this affects only the last character(s) in the string. + // NOTE(philipc): We know the error location will be in the query string, + // because schema validation always happens before this function is called. + errorParts := strings.SplitN(err.Error(), ":", 4) + msg := strings.TrimSuffix(strings.TrimLeft(errorParts[3], " "), ".\n") + return fmt.Errorf("%s in GraphQL query string at location %s:%s", msg, errorParts[1], errorParts[2]) + } + return nil +} + +func getBuiltinSchema() *gqlast.SchemaDocument { + schema, err := gqlparser.ParseSchema(gqlvalidator.Prelude) + if err != nil { + panic(fmt.Errorf("Error in gqlparser Prelude (should be impossible): %w", err)) + } + return schema +} + +// NOTE(philipc): This function expects *validated* schema documents, and will break +// if it is fed arbitrary structures. +func mergeSchemaDocuments(docA *gqlast.SchemaDocument, docB *gqlast.SchemaDocument) *gqlast.SchemaDocument { + ast := &gqlast.SchemaDocument{} + ast.Merge(docA) + ast.Merge(docB) + return ast +} + +// Converts a SchemaDocument into a gqlast.Schema object that can be used for validation. +// It merges in the builtin schema typedefs exactly as gqltop.LoadSchema did internally. +func convertSchema(schemaDoc *gqlast.SchemaDocument) (*gqlast.Schema, error) { + // Merge builtin schema + schema we were provided. + builtinsSchemaDoc := getBuiltinSchema() + mergedSchemaDoc := mergeSchemaDocuments(builtinsSchemaDoc, schemaDoc) + schema, err := gqlvalidator.ValidateSchemaDocument(mergedSchemaDoc) + if err != nil { + return nil, fmt.Errorf("Error in gqlparser SchemaDocument to Schema conversion: %w", err) + } + return schema, nil +} + +// Converts an ast.Object into a gqlast.QueryDocument object. +func objectToQueryDocument(value ast.Object) (*gqlast.QueryDocument, error) { + // Convert ast.Term to interface{} for JSON encoding below. + asJSON, err := ast.JSON(value) + if err != nil { + return nil, err + } + // Marshal to JSON. + bs, err := json.Marshal(asJSON) + if err != nil { + return nil, err + } + // Unmarshal from JSON -> gqlast.QueryDocument. + var result gqlast.QueryDocument + err = json.Unmarshal(bs, &result) + if err != nil { + return nil, err + } + return &result, nil +} + +// Converts an ast.Object into a gqlast.SchemaDocument object. +func objectToSchemaDocument(value ast.Object) (*gqlast.SchemaDocument, error) { + // Convert ast.Term to interface{} for JSON encoding below. + asJSON, err := ast.JSON(value) + if err != nil { + return nil, err + } + // Marshal to JSON. + bs, err := json.Marshal(asJSON) + if err != nil { + return nil, err + } + // Unmarshal from JSON -> gqlast.SchemaDocument. + var result gqlast.SchemaDocument + err = json.Unmarshal(bs, &result) + if err != nil { + return nil, err + } + return &result, nil +} + +// Recursively traverses an AST that has been run through InterfaceToValue, +// and prunes away the fields with null or empty values, and all `Position` +// structs. +// NOTE(philipc): We currently prune away null values to reduce the level +// of clutter in the returned AST objects. In the future, if there is demand +// for ASTs that have a more regular/fixed structure, we may need to provide +// a "raw" version of the AST, where we still prune away the `Position` +// structs, but leave in the null fields. +func pruneIrrelevantGraphQLASTNodes(value ast.Value) ast.Value { + // We iterate over the Value we've been provided, and recurse down + // in the case of complex types, such as Arrays/Objects. + // We are guaranteed to only have to deal with standard JSON types, + // so this is much less ugly than what we'd need for supporting every + // extant ast type! + switch x := value.(type) { + case *ast.Array: + result := ast.NewArray() + // Iterate over the array's elements, and do the following: + // - Drop any Nulls + // - Drop any any empty object/array value (after running the pruner) + for i := 0; i < x.Len(); i++ { + vTerm := x.Elem(i) + switch v := vTerm.Value.(type) { + case ast.Null: + continue + case *ast.Array: + // Safe, because we knew the type before going to prune it. + va := pruneIrrelevantGraphQLASTNodes(v).(*ast.Array) + if va.Len() > 0 { + result = result.Append(ast.NewTerm(va)) + } + case ast.Object: + // Safe, because we knew the type before going to prune it. + vo := pruneIrrelevantGraphQLASTNodes(v).(ast.Object) + if len(vo.Keys()) > 0 { + result = result.Append(ast.NewTerm(vo)) + } + default: + result = result.Append(vTerm) + } + } + return result + case ast.Object: + result := ast.NewObject() + // Iterate over our object's keys, and do the following: + // - Drop "Position". + // - Drop any key with a Null value. + // - Drop any key with an empty object/array value (after running the pruner) + keys := x.Keys() + for _, k := range keys { + // We drop the "Position" objects because we don't need the + // source-backref/location info they provide for policy rules. + // Note that keys are ast.Strings. + if ast.String("Position").Equal(k.Value) { + continue + } + vTerm := x.Get(k) + switch v := vTerm.Value.(type) { + case ast.Null: + continue + case *ast.Array: + // Safe, because we knew the type before going to prune it. + va := pruneIrrelevantGraphQLASTNodes(v).(*ast.Array) + if va.Len() > 0 { + result.Insert(k, ast.NewTerm(va)) + } + case ast.Object: + // Safe, because we knew the type before going to prune it. + vo := pruneIrrelevantGraphQLASTNodes(v).(ast.Object) + if len(vo.Keys()) > 0 { + result.Insert(k, ast.NewTerm(vo)) + } + default: + result.Insert(k, vTerm) + } + } + return result + default: + return x + } +} + +// Reports errors from parsing/validation. +func builtinGraphQLParse(_ BuiltinContext, operands []*ast.Term, iter func(*ast.Term) error) error { + var queryDoc *gqlast.QueryDocument + var schemaDoc *gqlast.SchemaDocument + var err error + + // Parse/translate query if it's a string/object. + switch x := operands[0].Value.(type) { + case ast.String: + queryDoc, err = parseQuery(string(x)) + case ast.Object: + queryDoc, err = objectToQueryDocument(x) + default: + // Error if wrong type. + return builtins.NewOperandTypeErr(0, x, "string", "object") + } + if err != nil { + return err + } + + // Parse/translate schema if it's a string/object. + switch x := operands[1].Value.(type) { + case ast.String: + schemaDoc, err = parseSchema(string(x)) + case ast.Object: + schemaDoc, err = objectToSchemaDocument(x) + default: + // Error if wrong type. + return builtins.NewOperandTypeErr(1, x, "string", "object") + } + if err != nil { + return err + } + + // Transform the ASTs into Objects. + queryASTValue, err := ast.InterfaceToValue(queryDoc) + if err != nil { + return err + } + schemaASTValue, err := ast.InterfaceToValue(schemaDoc) + if err != nil { + return err + } + + // Validate the query against the schema, erroring if there's an issue. + schema, err := convertSchema(schemaDoc) + if err != nil { + return err + } + if err := validateQuery(schema, queryDoc); err != nil { + return err + } + + // Recursively remove irrelevant AST structures. + queryResult := pruneIrrelevantGraphQLASTNodes(queryASTValue.(ast.Object)) + querySchema := pruneIrrelevantGraphQLASTNodes(schemaASTValue.(ast.Object)) + + // Construct return value. + verified := ast.ArrayTerm( + ast.NewTerm(queryResult), + ast.NewTerm(querySchema), + ) + + return iter(verified) +} + +// Returns default value when errors occur. +func builtinGraphQLParseAndVerify(_ BuiltinContext, operands []*ast.Term, iter func(*ast.Term) error) error { + var queryDoc *gqlast.QueryDocument + var schemaDoc *gqlast.SchemaDocument + var err error + + unverified := ast.ArrayTerm( + ast.BooleanTerm(false), + ast.NewTerm(ast.NewObject()), + ast.NewTerm(ast.NewObject()), + ) + + // Parse/translate query if it's a string/object. + switch x := operands[0].Value.(type) { + case ast.String: + queryDoc, err = parseQuery(string(x)) + case ast.Object: + queryDoc, err = objectToQueryDocument(x) + default: + // Error if wrong type. + return iter(unverified) + } + if err != nil { + return iter(unverified) + } + + // Parse/translate schema if it's a string/object. + switch x := operands[1].Value.(type) { + case ast.String: + schemaDoc, err = parseSchema(string(x)) + case ast.Object: + schemaDoc, err = objectToSchemaDocument(x) + default: + // Error if wrong type. + return iter(unverified) + } + if err != nil { + return iter(unverified) + } + + // Transform the ASTs into Objects. + queryASTValue, err := ast.InterfaceToValue(queryDoc) + if err != nil { + return iter(unverified) + } + schemaASTValue, err := ast.InterfaceToValue(schemaDoc) + if err != nil { + return iter(unverified) + } + + // Validate the query against the schema, erroring if there's an issue. + schema, err := convertSchema(schemaDoc) + if err != nil { + return iter(unverified) + } + if err := validateQuery(schema, queryDoc); err != nil { + return iter(unverified) + } + + // Recursively remove irrelevant AST structures. + queryResult := pruneIrrelevantGraphQLASTNodes(queryASTValue.(ast.Object)) + querySchema := pruneIrrelevantGraphQLASTNodes(schemaASTValue.(ast.Object)) + + // Construct return value. + verified := ast.ArrayTerm( + ast.BooleanTerm(true), + ast.NewTerm(queryResult), + ast.NewTerm(querySchema), + ) + + return iter(verified) +} + +func builtinGraphQLParseQuery(_ BuiltinContext, operands []*ast.Term, iter func(*ast.Term) error) error { + raw, err := builtins.StringOperand(operands[0].Value, 1) + if err != nil { + return err + } + + // Get the highly-nested AST struct, along with any errors generated. + query, err := parseQuery(string(raw)) + if err != nil { + return err + } + + // Transform the AST into an Object. + value, err := ast.InterfaceToValue(query) + if err != nil { + return err + } + + // Recursively remove irrelevant AST structures. + result := pruneIrrelevantGraphQLASTNodes(value.(ast.Object)) + + return iter(ast.NewTerm(result)) +} + +func builtinGraphQLParseSchema(_ BuiltinContext, operands []*ast.Term, iter func(*ast.Term) error) error { + raw, err := builtins.StringOperand(operands[0].Value, 1) + if err != nil { + return err + } + + // Get the highly-nested AST struct, along with any errors generated. + schema, err := parseSchema(string(raw)) + if err != nil { + return err + } + + // Transform the AST into an Object. + value, err := ast.InterfaceToValue(schema) + if err != nil { + return err + } + + // Recursively remove irrelevant AST structures. + result := pruneIrrelevantGraphQLASTNodes(value.(ast.Object)) + + return iter(ast.NewTerm(result)) +} + +func builtinGraphQLIsValid(_ BuiltinContext, operands []*ast.Term, iter func(*ast.Term) error) error { + var queryDoc *gqlast.QueryDocument + var schemaDoc *gqlast.SchemaDocument + var err error + + switch x := operands[0].Value.(type) { + case ast.String: + queryDoc, err = parseQuery(string(x)) + case ast.Object: + queryDoc, err = objectToQueryDocument(x) + default: + // Error if wrong type. + return iter(ast.BooleanTerm(false)) + } + if err != nil { + return iter(ast.BooleanTerm(false)) + } + + switch x := operands[1].Value.(type) { + case ast.String: + schemaDoc, err = parseSchema(string(x)) + case ast.Object: + schemaDoc, err = objectToSchemaDocument(x) + default: + // Error if wrong type. + return iter(ast.BooleanTerm(false)) + } + if err != nil { + return iter(ast.BooleanTerm(false)) + } + + // Validate the query against the schema, erroring if there's an issue. + schema, err := convertSchema(schemaDoc) + if err != nil { + return iter(ast.BooleanTerm(false)) + } + if err := validateQuery(schema, queryDoc); err != nil { + return iter(ast.BooleanTerm(false)) + } + + // If we got this far, the GraphQL query passed validation. + return iter(ast.BooleanTerm(true)) +} + +func init() { + RegisterBuiltinFunc(ast.GraphQLParse.Name, builtinGraphQLParse) + RegisterBuiltinFunc(ast.GraphQLParseAndVerify.Name, builtinGraphQLParseAndVerify) + RegisterBuiltinFunc(ast.GraphQLParseQuery.Name, builtinGraphQLParseQuery) + RegisterBuiltinFunc(ast.GraphQLParseSchema.Name, builtinGraphQLParseSchema) + RegisterBuiltinFunc(ast.GraphQLIsValid.Name, builtinGraphQLIsValid) +} diff --git a/vendor/github.com/open-policy-agent/opa/topdown/http.go b/vendor/github.com/open-policy-agent/opa/topdown/http.go index c8205f18..810ac741 100644 --- a/vendor/github.com/open-policy-agent/opa/topdown/http.go +++ b/vendor/github.com/open-policy-agent/opa/topdown/http.go @@ -46,6 +46,7 @@ var allowedKeyNames = [...]string{ "body", "enable_redirect", "force_json_decode", + "force_yaml_decode", "headers", "raw_body", "tls_use_system_certs", @@ -135,7 +136,7 @@ func getHTTPResponse(bctx BuiltinContext, req ast.Object) (*ast.Term, error) { return nil, err } - // check if cache already has a response for this query + // Check if cache already has a response for this query resp, err := reqExecutor.CheckCache() if err != nil { return nil, err @@ -147,7 +148,7 @@ func getHTTPResponse(bctx BuiltinContext, req ast.Object) (*ast.Term, error) { return nil, err } defer util.Close(httpResp) - // add result to cache + // Add result to intra/inter-query cache. resp, err = reqExecutor.InsertIntoCache(httpResp) if err != nil { return nil, err @@ -431,7 +432,10 @@ func createHTTPRequest(bctx BuiltinContext, obj ast.Object) (*http.Request, *htt if err != nil { return nil, nil, err } - case "cache", "force_cache", "force_cache_duration_seconds", "force_json_decode", "raise_error", "caching_mode": // no-op + case "cache", "caching_mode", + "force_cache", "force_cache_duration_seconds", + "force_json_decode", "force_yaml_decode", + "raise_error": // no-op default: return nil, nil, fmt.Errorf("invalid parameter %q", key) } @@ -618,8 +622,13 @@ func executeHTTPRequest(req *http.Request, client *http.Client) (*http.Response, return client.Do(req) } -func isContentTypeJSON(header http.Header) bool { - return strings.Contains(header.Get("Content-Type"), "application/json") +func isContentType(header http.Header, typ ...string) bool { + for _, t := range typ { + if strings.Contains(header.Get("Content-Type"), t) { + return true + } + } + return false } // In the BuiltinContext cache we only store a single entry that points to @@ -690,7 +699,7 @@ func (c *interQueryCache) checkHTTPSendInterQueryCache() (ast.Value, error) { // check the freshness of the cached response if isCachedResponseFresh(c.bctx, headers, c.forceCacheParams) { - return cachedRespData.formatToAST(c.forceJSONDecode) + return cachedRespData.formatToAST(c.forceJSONDecode, c.forceYAMLDecode) } c.httpReq, c.httpClient, err = createHTTPRequest(c.bctx, c.key) @@ -736,16 +745,15 @@ func (c *interQueryCache) checkHTTPSendInterQueryCache() (ast.Value, error) { c.bctx.InterQueryBuiltinCache.Insert(c.key, pcv) - return cachedRespData.formatToAST(c.forceJSONDecode) + return cachedRespData.formatToAST(c.forceJSONDecode, c.forceYAMLDecode) } - newValue, respBody, err := formatHTTPResponseToAST(result, c.forceJSONDecode) + newValue, respBody, err := formatHTTPResponseToAST(result, c.forceJSONDecode, c.forceYAMLDecode) if err != nil { return nil, err } - err = insertIntoHTTPSendInterQueryCache(c.bctx, c.key, result, respBody, c.forceCacheParams != nil) - if err != nil { + if err := insertIntoHTTPSendInterQueryCache(c.bctx, c.key, result, respBody, c.forceCacheParams != nil); err != nil { return nil, err } @@ -901,8 +909,8 @@ func newInterQueryCacheData(resp *http.Response, respBody []byte) (*interQueryCa return &cv, nil } -func (c *interQueryCacheData) formatToAST(forceJSONDecode bool) (ast.Value, error) { - return prepareASTResult(c.Headers, forceJSONDecode, c.RespBody, c.Status, c.StatusCode) +func (c *interQueryCacheData) formatToAST(forceJSONDecode, forceYAMLDecode bool) (ast.Value, error) { + return prepareASTResult(c.Headers, forceJSONDecode, forceYAMLDecode, c.RespBody, c.Status, c.StatusCode) } func (c *interQueryCacheData) toCacheValue() (*interQueryCacheValue, error) { @@ -1145,14 +1153,14 @@ func parseMaxAgeCacheDirective(cc map[string]string) (deltaSeconds, error) { return deltaSeconds(val), nil } -func formatHTTPResponseToAST(resp *http.Response, forceJSONDecode bool) (ast.Value, []byte, error) { +func formatHTTPResponseToAST(resp *http.Response, forceJSONDecode, forceYAMLDecode bool) (ast.Value, []byte, error) { resultRawBody, err := ioutil.ReadAll(resp.Body) if err != nil { return nil, nil, err } - resultObj, err := prepareASTResult(resp.Header, forceJSONDecode, resultRawBody, resp.Status, resp.StatusCode) + resultObj, err := prepareASTResult(resp.Header, forceJSONDecode, forceYAMLDecode, resultRawBody, resp.Status, resp.StatusCode) if err != nil { return nil, nil, err } @@ -1160,14 +1168,17 @@ func formatHTTPResponseToAST(resp *http.Response, forceJSONDecode bool) (ast.Val return resultObj, resultRawBody, nil } -func prepareASTResult(headers http.Header, forceJSONDecode bool, body []byte, status string, statusCode int) (ast.Value, error) { +func prepareASTResult(headers http.Header, forceJSONDecode, forceYAMLDecode bool, body []byte, status string, statusCode int) (ast.Value, error) { var resultBody interface{} - // If the response body cannot be JSON decoded, + // If the response body cannot be JSON/YAML decoded, // an error will not be returned. Instead the "body" field // in the result will be null. - if isContentTypeJSON(headers) || forceJSONDecode { + switch { + case forceJSONDecode || isContentType(headers, "application/json"): _ = util.UnmarshalJSON(body, &resultBody) + case forceYAMLDecode || isContentType(headers, "application/yaml", "application/x-yaml"): + _ = util.Unmarshal(body, &resultBody) } result := make(map[string]interface{}) @@ -1224,6 +1235,7 @@ type interQueryCache struct { httpReq *http.Request httpClient *http.Client forceJSONDecode bool + forceYAMLDecode bool forceCacheParams *forceCacheParams } @@ -1239,6 +1251,10 @@ func (c *interQueryCache) CheckCache() (ast.Value, error) { if err != nil { return nil, handleHTTPSendErr(c.bctx, err) } + c.forceYAMLDecode, err = getBoolValFromReqObj(c.key, ast.StringTerm("force_yaml_decode")) + if err != nil { + return nil, handleHTTPSendErr(c.bctx, err) + } resp, err := c.checkHTTPSendInterQueryCache() @@ -1253,7 +1269,7 @@ func (c *interQueryCache) CheckCache() (ast.Value, error) { // InsertIntoCache inserts the key set on this object into the cache with the given value func (c *interQueryCache) InsertIntoCache(value *http.Response) (ast.Value, error) { - result, respBody, err := formatHTTPResponseToAST(value, c.forceJSONDecode) + result, respBody, err := formatHTTPResponseToAST(value, c.forceJSONDecode, c.forceYAMLDecode) if err != nil { return nil, handleHTTPSendErr(c.bctx, err) } @@ -1297,8 +1313,12 @@ func (c *intraQueryCache) InsertIntoCache(value *http.Response) (ast.Value, erro if err != nil { return nil, handleHTTPSendErr(c.bctx, err) } + forceYAMLDecode, err := getBoolValFromReqObj(c.key, ast.StringTerm("force_yaml_decode")) + if err != nil { + return nil, handleHTTPSendErr(c.bctx, err) + } - result, _, err := formatHTTPResponseToAST(value, forceJSONDecode) + result, _, err := formatHTTPResponseToAST(value, forceJSONDecode, forceYAMLDecode) if err != nil { return nil, handleHTTPSendErr(c.bctx, err) } diff --git a/vendor/github.com/open-policy-agent/opa/topdown/http_fixup.go b/vendor/github.com/open-policy-agent/opa/topdown/http_fixup.go new file mode 100644 index 00000000..1b9ffc23 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/topdown/http_fixup.go @@ -0,0 +1,8 @@ +//go:build !go1.18 || !darwin +// +build !go1.18 !darwin + +package topdown + +func fixupDarwinGo118(x string, _ string) string { + return x +} diff --git a/vendor/github.com/open-policy-agent/opa/topdown/http_fixup_darwin.go b/vendor/github.com/open-policy-agent/opa/topdown/http_fixup_darwin.go new file mode 100644 index 00000000..ff3058ef --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/topdown/http_fixup_darwin.go @@ -0,0 +1,13 @@ +//go:build go1.18 +// +build go1.18 + +package topdown + +func fixupDarwinGo118(x, y string) string { + switch x { + case "x509: certificate signed by unknown authority": + return y + default: + return x + } +} diff --git a/vendor/github.com/open-policy-agent/opa/topdown/json.go b/vendor/github.com/open-policy-agent/opa/topdown/json.go index 09a0ea6a..d1d8897e 100644 --- a/vendor/github.com/open-policy-agent/opa/topdown/json.go +++ b/vendor/github.com/open-policy-agent/opa/topdown/json.go @@ -203,6 +203,12 @@ func pathsToObject(paths []ast.Ref) ast.Object { node := root var done bool + // If the path is an empty JSON path, skip all further processing. + if len(path) == 0 { + done = true + } + + // Otherwise, we should have 1+ path segments to work with. for i := 0; i < len(path)-1 && !done; i++ { k := path[i] diff --git a/vendor/github.com/open-policy-agent/opa/topdown/net.go b/vendor/github.com/open-policy-agent/opa/topdown/net.go index a167cf13..53452052 100644 --- a/vendor/github.com/open-policy-agent/opa/topdown/net.go +++ b/vendor/github.com/open-policy-agent/opa/topdown/net.go @@ -6,6 +6,7 @@ package topdown import ( "net" + "strings" "github.com/open-policy-agent/opa/ast" "github.com/open-policy-agent/opa/topdown/builtins" @@ -36,7 +37,7 @@ func builtinLookupIPAddr(bctx BuiltinContext, operands []*ast.Term, iter func(*a addrs, err := resolv.LookupIPAddr(bctx.Context, name) if err != nil { // NOTE(sr): We can't do better than this right now, see https://github.com/golang/go/issues/36208 - if err.Error() == "operation was canceled" || err.Error() == "i/o timeout" { + if strings.Contains(err.Error(), "operation was canceled") || strings.Contains(err.Error(), "i/o timeout") { return Halt{ Err: &Error{ Code: CancelErr, diff --git a/vendor/github.com/open-policy-agent/opa/topdown/object.go b/vendor/github.com/open-policy-agent/opa/topdown/object.go index ff01b4f9..63917067 100644 --- a/vendor/github.com/open-policy-agent/opa/topdown/object.go +++ b/vendor/github.com/open-policy-agent/opa/topdown/object.go @@ -6,8 +6,8 @@ package topdown import ( "github.com/open-policy-agent/opa/ast" + "github.com/open-policy-agent/opa/internal/ref" "github.com/open-policy-agent/opa/topdown/builtins" - "github.com/open-policy-agent/opa/types" ) func builtinObjectUnion(_ BuiltinContext, operands []*ast.Term, iter func(*ast.Term) error) error { @@ -26,6 +26,38 @@ func builtinObjectUnion(_ BuiltinContext, operands []*ast.Term, iter func(*ast.T return iter(ast.NewTerm(r)) } +func builtinObjectUnionN(_ BuiltinContext, operands []*ast.Term, iter func(*ast.Term) error) error { + arr, err := builtins.ArrayOperand(operands[0].Value, 1) + if err != nil { + return err + } + + // Because we need merge-with-overwrite behavior, we can iterate + // back-to-front, and get a mostly correct set of key assignments that + // give us the "last assignment wins, with merges" behavior we want. + // However, if a non-object overwrites an object value anywhere in the + // chain of assignments for a key, we have to "freeze" that key to + // prevent accidentally picking up nested objects that could merge with + // it from earlier in the input array. + // Example: + // Input: [{"a": {"b": 2}}, {"a": 4}, {"a": {"c": 3}}] + // Want Output: {"a": {"c": 3}} + result := ast.NewObject() + frozenKeys := map[*ast.Term]struct{}{} + for i := arr.Len() - 1; i >= 0; i-- { + o, ok := arr.Elem(i).Value.(ast.Object) + if !ok { + return builtins.NewOperandElementErr(1, arr, arr.Elem(i).Value, "object") + } + mergewithOverwriteInPlace(result, o, frozenKeys) + if err != nil { + return err + } + } + + return iter(ast.NewTerm(result)) +} + func builtinObjectRemove(_ BuiltinContext, operands []*ast.Term, iter func(*ast.Term) error) error { // Expect an object and an array/set/object of keys obj, err := builtins.ObjectOperand(operands[0].Value, 1) @@ -81,11 +113,29 @@ func builtinObjectGet(_ BuiltinContext, operands []*ast.Term, iter func(*ast.Ter return err } - if ret := object.Get(operands[1]); ret != nil { - return iter(ret) + // if the get key is not an array, attempt to get the top level key for the operand value in the object + path, err := builtins.ArrayOperand(operands[1].Value, 2) + if err != nil { + if ret := object.Get(operands[1]); ret != nil { + return iter(ret) + } + + return iter(operands[2]) + } + + // if the path is empty, then we skip selecting nested keys and return the whole object + if path.Len() == 0 { + return iter(operands[0]) + } + + // build an ast.Ref from the array and see if it matches within the object + pathRef := ref.ArrayPath(path) + value, err := object.Find(pathRef) + if err != nil { + return iter(operands[2]) } - return iter(operands[2]) + return iter(ast.NewTerm(value)) } // getObjectKeysParam returns a set of key values @@ -110,7 +160,7 @@ func getObjectKeysParam(arrayOrSet ast.Value) (ast.Set, error) { return nil }) default: - return nil, builtins.NewOperandTypeErr(2, arrayOrSet, ast.TypeName(types.Object{}), ast.TypeName(types.S), ast.TypeName(types.Array{})) + return nil, builtins.NewOperandTypeErr(2, arrayOrSet, "object", "set", "array") } return keys, nil @@ -132,8 +182,35 @@ func mergeWithOverwrite(objA, objB ast.Object) ast.Object { return merged } +// Modifies obj with any new keys from other, and recursively +// merges any keys where the values are both objects. +func mergewithOverwriteInPlace(obj, other ast.Object, frozenKeys map[*ast.Term]struct{}) { + other.Foreach(func(k, v *ast.Term) { + v2 := obj.Get(k) + // The key didn't exist in other, keep the original value. + if v2 == nil { + obj.Insert(k, v) + return + } + // The key exists in both. Merge or reject change. + updateValueObj, ok2 := v.Value.(ast.Object) + originalValueObj, ok1 := v2.Value.(ast.Object) + // Both are objects? Merge recursively. + if ok1 && ok2 { + // Check to make sure that this key isn't frozen before merging. + if _, ok := frozenKeys[v2]; !ok { + mergewithOverwriteInPlace(originalValueObj, updateValueObj, frozenKeys) + } + } else { + // Else, original value wins. Freeze the key. + frozenKeys[v2] = struct{}{} + } + }) +} + func init() { RegisterBuiltinFunc(ast.ObjectUnion.Name, builtinObjectUnion) + RegisterBuiltinFunc(ast.ObjectUnionN.Name, builtinObjectUnionN) RegisterBuiltinFunc(ast.ObjectRemove.Name, builtinObjectRemove) RegisterBuiltinFunc(ast.ObjectFilter.Name, builtinObjectFilter) RegisterBuiltinFunc(ast.ObjectGet.Name, builtinObjectGet) diff --git a/vendor/github.com/open-policy-agent/opa/topdown/parse.go b/vendor/github.com/open-policy-agent/opa/topdown/parse.go index dc6e6fc5..df12cbc9 100644 --- a/vendor/github.com/open-policy-agent/opa/topdown/parse.go +++ b/vendor/github.com/open-policy-agent/opa/topdown/parse.go @@ -7,6 +7,7 @@ package topdown import ( "bytes" "encoding/json" + "fmt" "github.com/open-policy-agent/opa/ast" "github.com/open-policy-agent/opa/topdown/builtins" @@ -42,6 +43,17 @@ func builtinRegoParseModule(a, b ast.Value) (ast.Value, error) { return term.Value, nil } +func registerRegoMetadataBuiltinFunction(builtin *ast.Builtin) { + f := func(BuiltinContext, []*ast.Term, func(*ast.Term) error) error { + // The compiler should replace all usage of this function, so the only way to get here is within a query; + // which cannot define rules. + return fmt.Errorf("the %s function must only be called within the scope of a rule", builtin.Name) + } + RegisterBuiltinFunc(builtin.Name, f) +} + func init() { RegisterFunctionalBuiltin2(ast.RegoParseModule.Name, builtinRegoParseModule) + registerRegoMetadataBuiltinFunction(ast.RegoMetadataChain) + registerRegoMetadataBuiltinFunction(ast.RegoMetadataRule) } diff --git a/vendor/github.com/open-policy-agent/opa/topdown/parse_bytes.go b/vendor/github.com/open-policy-agent/opa/topdown/parse_bytes.go index 6e58d232..9d8fe506 100644 --- a/vendor/github.com/open-policy-agent/opa/topdown/parse_bytes.go +++ b/vendor/github.com/open-policy-agent/opa/topdown/parse_bytes.go @@ -32,17 +32,17 @@ const ( ) func parseNumBytesError(msg string) error { - return fmt.Errorf("%s error: %s", ast.UnitsParseBytes.Name, msg) + return fmt.Errorf("%s: %s", ast.UnitsParseBytes.Name, msg) } -func errUnitNotRecognized(unit string) error { +func errBytesUnitNotRecognized(unit string) error { return parseNumBytesError(fmt.Sprintf("byte unit %s not recognized", unit)) } var ( - errNoAmount = parseNumBytesError("no byte amount provided") - errNumConv = parseNumBytesError("could not parse byte amount to a number") - errIncludesSpaces = parseNumBytesError("spaces not allowed in resource strings") + errBytesValueNoAmount = parseNumBytesError("no byte amount provided") + errBytesValueNumConv = parseNumBytesError("could not parse byte amount to a number") + errBytesValueIncludesSpaces = parseNumBytesError("spaces not allowed in resource strings") ) func builtinNumBytes(bctx BuiltinContext, operands []*ast.Term, iter func(*ast.Term) error) error { @@ -56,12 +56,12 @@ func builtinNumBytes(bctx BuiltinContext, operands []*ast.Term, iter func(*ast.T s := formatString(raw) if strings.Contains(s, " ") { - return errIncludesSpaces + return errBytesValueIncludesSpaces } num, unit := extractNumAndUnit(s) if num == "" { - return errNoAmount + return errBytesValueNoAmount } switch unit { @@ -92,12 +92,12 @@ func builtinNumBytes(bctx BuiltinContext, operands []*ast.Term, iter func(*ast.T case "eib", "ei": m.SetUint64(ei) default: - return errUnitNotRecognized(unit) + return errBytesUnitNotRecognized(unit) } numFloat, ok := new(big.Float).SetString(num) if !ok { - return errNumConv + return errBytesValueNumConv } var total big.Int diff --git a/vendor/github.com/open-policy-agent/opa/topdown/parse_units.go b/vendor/github.com/open-policy-agent/opa/topdown/parse_units.go new file mode 100644 index 00000000..daf24021 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/topdown/parse_units.go @@ -0,0 +1,125 @@ +// Copyright 2022 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package topdown + +import ( + "encoding/json" + "fmt" + "math/big" + "strings" + + "github.com/open-policy-agent/opa/ast" + "github.com/open-policy-agent/opa/topdown/builtins" +) + +// Binary Si unit constants are borrowed from topdown/parse_bytes +const siMilli = 0.001 +const ( + siK uint64 = 1000 + siM = siK * 1000 + siG = siM * 1000 + siT = siG * 1000 + siP = siT * 1000 + siE = siP * 1000 +) + +func parseUnitsError(msg string) error { + return fmt.Errorf("%s: %s", ast.UnitsParse.Name, msg) +} + +func errUnitNotRecognized(unit string) error { + return parseUnitsError(fmt.Sprintf("unit %s not recognized", unit)) +} + +var ( + errNoAmount = parseUnitsError("no amount provided") + errNumConv = parseUnitsError("could not parse amount to a number") + errIncludesSpaces = parseUnitsError("spaces not allowed in resource strings") +) + +// Accepts both normal SI and binary SI units. +func builtinUnits(_ BuiltinContext, operands []*ast.Term, iter func(*ast.Term) error) error { + var x big.Rat + + raw, err := builtins.StringOperand(operands[0].Value, 1) + if err != nil { + return err + } + + // We remove escaped quotes from strings here to retain parity with units.parse_bytes. + s := string(raw) + s = strings.Replace(s, "\"", "", -1) + + if strings.Contains(s, " ") { + return errIncludesSpaces + } + + num, unit := extractNumAndUnit(s) + if num == "" { + return errNoAmount + } + + // Unlike in units.parse_bytes, we only lowercase after the first letter, + // so that we can distinguish between 'm' and 'M'. + if len(unit) > 1 { + lower := strings.ToLower(unit[1:]) + unit = unit[:1] + lower + } + + switch unit { + case "m": + x.SetFloat64(siMilli) + case "": + x.SetUint64(none) + case "k", "K": + x.SetUint64(siK) + case "ki", "Ki": + x.SetUint64(ki) + case "M": + x.SetUint64(siM) + case "mi", "Mi": + x.SetUint64(mi) + case "g", "G": + x.SetUint64(siG) + case "gi", "Gi": + x.SetUint64(gi) + case "t", "T": + x.SetUint64(siT) + case "ti", "Ti": + x.SetUint64(ti) + case "p", "P": + x.SetUint64(siP) + case "pi", "Pi": + x.SetUint64(pi) + case "e", "E": + x.SetUint64(siE) + case "ei", "Ei": + x.SetUint64(ei) + default: + return errUnitNotRecognized(unit) + } + + numRat, ok := new(big.Rat).SetString(num) + if !ok { + return errNumConv + } + + numRat.Mul(numRat, &x) + + // Cleaner printout when we have a pure integer value. + if numRat.IsInt() { + return iter(ast.NumberTerm(json.Number(numRat.Num().String()))) + } + + // When using just big.Float, we had floating-point precision + // issues because quantities like 0.001 are not exactly representable. + // Rationals (such as big.Rat) do not suffer this problem, but are + // more expensive to compute with in general. + return iter(ast.NumberTerm(json.Number(numRat.FloatString(10)))) +} + +func init() { + RegisterBuiltinFunc(ast.UnitsParse.Name, builtinUnits) +} diff --git a/vendor/github.com/open-policy-agent/opa/topdown/query.go b/vendor/github.com/open-policy-agent/opa/topdown/query.go index 9a9a3c67..a8dfda49 100644 --- a/vendor/github.com/open-policy-agent/opa/topdown/query.go +++ b/vendor/github.com/open-policy-agent/opa/topdown/query.go @@ -52,6 +52,7 @@ type Query struct { indexing bool earlyExit bool interQueryBuiltinCache cache.InterQueryCache + ndBuiltinCache builtins.NDBCache strictBuiltinErrors bool printHook print.Hook tracingOpts tracing.Options @@ -242,6 +243,12 @@ func (q *Query) WithInterQueryBuiltinCache(c cache.InterQueryCache) *Query { return q } +// WithNDBuiltinCache sets the non-deterministic builtin cache. +func (q *Query) WithNDBuiltinCache(c builtins.NDBCache) *Query { + q.ndBuiltinCache = c + return q +} + // WithStrictBuiltinErrors tells the evaluator to treat all built-in function errors as fatal errors. func (q *Query) WithStrictBuiltinErrors(yes bool) *Query { q.strictBuiltinErrors = yes @@ -311,7 +318,9 @@ func (q *Query) PartialRun(ctx context.Context) (partials []ast.Body, support [] instr: q.instr, builtins: q.builtins, builtinCache: builtins.Cache{}, + functionMocks: newFunctionMocksStack(), interQueryBuiltinCache: q.interQueryBuiltinCache, + ndBuiltinCache: q.ndBuiltinCache, virtualCache: newVirtualCache(), comprehensionCache: newComprehensionCache(), saveSet: newSaveSet(q.unknowns, b, q.instr), @@ -339,6 +348,14 @@ func (q *Query) PartialRun(ctx context.Context) (partials []ast.Body, support [] defer q.metrics.Timer(metrics.RegoPartialEval).Stop() livevars := ast.NewVarSet() + for _, t := range q.unknowns { + switch v := t.Value.(type) { + case ast.Var: + livevars.Add(v) + case ast.Ref: + livevars.Add(v[0].Value.(ast.Var)) + } + } ast.WalkVars(q.query, func(x ast.Var) bool { if !x.IsGenerated() { @@ -451,7 +468,9 @@ func (q *Query) Iter(ctx context.Context, iter func(QueryResult) error) error { instr: q.instr, builtins: q.builtins, builtinCache: builtins.Cache{}, + functionMocks: newFunctionMocksStack(), interQueryBuiltinCache: q.interQueryBuiltinCache, + ndBuiltinCache: q.ndBuiltinCache, virtualCache: newVirtualCache(), comprehensionCache: newComprehensionCache(), genvarprefix: q.genvarprefix, diff --git a/vendor/github.com/open-policy-agent/opa/topdown/reachable.go b/vendor/github.com/open-policy-agent/opa/topdown/reachable.go index 6b73fa2c..a212d221 100644 --- a/vendor/github.com/open-policy-agent/opa/topdown/reachable.go +++ b/vendor/github.com/open-policy-agent/opa/topdown/reachable.go @@ -6,6 +6,7 @@ package topdown import ( "github.com/open-policy-agent/opa/ast" + "github.com/open-policy-agent/opa/topdown/builtins" ) // Helper: sets of vertices can be represented as Arrays or Sets. @@ -18,19 +19,34 @@ func foreachVertex(collection *ast.Term, f func(*ast.Term)) { } } +// numberOfEdges returns the number of elements of an array or a set (of edges) +func numberOfEdges(collection *ast.Term) int { + switch v := collection.Value.(type) { + case ast.Set: + return v.Len() + case *ast.Array: + return v.Len() + } + + return 0 +} + func builtinReachable(bctx BuiltinContext, args []*ast.Term, iter func(*ast.Term) error) error { - // Return the empty set if the first argument is not an object. - graph, ok := args[0].Value.(ast.Object) - if !ok { - return iter(ast.NewTerm(ast.NewSet())) + // Error on wrong types for args. + graph, err := builtins.ObjectOperand(args[0].Value, 1) + if err != nil { + return err } - // This is a queue that holds all nodes we still need to visit. It is - // initialised to the initial set of nodes we start out with. - queue := []*ast.Term{} - foreachVertex(args[1], func(t *ast.Term) { - queue = append(queue, t) - }) + var queue []*ast.Term + switch initial := args[1].Value.(type) { + case *ast.Array, ast.Set: + foreachVertex(ast.NewTerm(initial), func(t *ast.Term) { + queue = append(queue, t) + }) + default: + return builtins.NewOperandTypeErr(2, initial, "{array, set}") + } // This is the set of nodes we have reached. reached := ast.NewSet() @@ -56,6 +72,71 @@ func builtinReachable(bctx BuiltinContext, args []*ast.Term, iter func(*ast.Term return iter(ast.NewTerm(reached)) } +// pathBuilder is called recursively to build an array of paths that are reachable from the root +func pathBuilder(graph ast.Object, root *ast.Term, path []*ast.Term, paths []*ast.Term, reached ast.Set) []*ast.Term { + if edges := graph.Get(root); edges != nil { + path = append(path, root) + + if numberOfEdges(edges) >= 1 { + foreachVertex(edges, func(neighbor *ast.Term) { + if reached.Contains(neighbor) { + // If we've already reached this node, return current path (avoid infinite recursion) + paths = append(paths, path...) + } else { + reached.Add(root) + paths = pathBuilder(graph, neighbor, path, paths, reached) + } + }) + } else { + paths = append(paths, path...) + } + } else { + // Node is nonexistent (not in graph). Commit the current path (without adding this root) + paths = append(paths, path...) + } + + return paths +} + +func builtinReachablePaths(bctx BuiltinContext, args []*ast.Term, iter func(*ast.Term) error) error { + // Error on wrong types for args. + graph, err := builtins.ObjectOperand(args[0].Value, 1) + if err != nil { + return err + } + + // This is a queue that holds all nodes we still need to visit. It is + // initialised to the initial set of nodes we start out with. + var queue []*ast.Term + switch initial := args[1].Value.(type) { + case *ast.Array, ast.Set: + foreachVertex(ast.NewTerm(initial), func(t *ast.Term) { + queue = append(queue, t) + }) + default: + return builtins.NewOperandTypeErr(2, initial, "{array, set}") + } + + results := ast.NewSet() + + for _, node := range queue { + // Find reachable paths from edges in root node in queue and append arrays to the results set + if edges := graph.Get(node); edges != nil { + if numberOfEdges(edges) >= 1 { + foreachVertex(edges, func(neighbor *ast.Term) { + paths := pathBuilder(graph, neighbor, []*ast.Term{node}, []*ast.Term{}, ast.NewSet(node)) + results.Add(ast.ArrayTerm(paths...)) + }) + } else { + results.Add(ast.ArrayTerm(node)) + } + } + } + + return iter(ast.NewTerm(results)) +} + func init() { RegisterBuiltinFunc(ast.ReachableBuiltin.Name, builtinReachable) + RegisterBuiltinFunc(ast.ReachablePathsBuiltin.Name, builtinReachablePaths) } diff --git a/vendor/github.com/open-policy-agent/opa/topdown/regex.go b/vendor/github.com/open-policy-agent/opa/topdown/regex.go index 7e29db79..53197626 100644 --- a/vendor/github.com/open-policy-agent/opa/topdown/regex.go +++ b/vendor/github.com/open-policy-agent/opa/topdown/regex.go @@ -205,6 +205,32 @@ func builtinRegexFindAllStringSubmatch(a, b, c ast.Value) (ast.Value, error) { return ast.NewArray(outer...), nil } +func builtinRegexReplace(_ BuiltinContext, operands []*ast.Term, iter func(*ast.Term) error) error { + base, err := builtins.StringOperand(operands[0].Value, 1) + if err != nil { + return err + } + + pattern, err := builtins.StringOperand(operands[1].Value, 2) + if err != nil { + return err + } + + value, err := builtins.StringOperand(operands[2].Value, 3) + if err != nil { + return err + } + + re, err := getRegexp(string(pattern)) + if err != nil { + return err + } + + res := re.ReplaceAllString(string(base), string(value)) + + return iter(ast.StringTerm(res)) +} + func init() { regexpCache = map[string]*regexp.Regexp{} RegisterBuiltinFunc(ast.RegexIsValid.Name, builtinRegexIsValid) @@ -215,4 +241,5 @@ func init() { RegisterFunctionalBuiltin4(ast.RegexTemplateMatch.Name, builtinRegexMatchTemplate) RegisterFunctionalBuiltin3(ast.RegexFind.Name, builtinRegexFind) RegisterFunctionalBuiltin3(ast.RegexFindAllStringSubmatch.Name, builtinRegexFindAllStringSubmatch) + RegisterBuiltinFunc(ast.RegexReplace.Name, builtinRegexReplace) } diff --git a/vendor/github.com/open-policy-agent/opa/topdown/runtime.go b/vendor/github.com/open-policy-agent/opa/topdown/runtime.go index 67e183d0..7d512f7c 100644 --- a/vendor/github.com/open-policy-agent/opa/topdown/runtime.go +++ b/vendor/github.com/open-policy-agent/opa/topdown/runtime.go @@ -4,7 +4,11 @@ package topdown -import "github.com/open-policy-agent/opa/ast" +import ( + "fmt" + + "github.com/open-policy-agent/opa/ast" +) func builtinOPARuntime(bctx BuiltinContext, _ []*ast.Term, iter func(*ast.Term) error) error { @@ -12,9 +16,112 @@ func builtinOPARuntime(bctx BuiltinContext, _ []*ast.Term, iter func(*ast.Term) return iter(ast.ObjectTerm()) } + if bctx.Runtime.Get(ast.StringTerm("config")) != nil { + iface, err := ast.ValueToInterface(bctx.Runtime.Value, illegalResolver{}) + if err != nil { + return err + } + if object, ok := iface.(map[string]interface{}); ok { + if cfgRaw, ok := object["config"]; ok { + if config, ok := cfgRaw.(map[string]interface{}); ok { + configPurged, err := activeConfig(config) + if err != nil { + return err + } + object["config"] = configPurged + value, err := ast.InterfaceToValue(object) + if err != nil { + return err + } + return iter(ast.NewTerm(value)) + } + } + } + } + return iter(bctx.Runtime) } func init() { RegisterBuiltinFunc(ast.OPARuntime.Name, builtinOPARuntime) } + +func activeConfig(config map[string]interface{}) (interface{}, error) { + + if config["services"] != nil { + err := removeServiceCredentials(config["services"]) + if err != nil { + return nil, err + } + } + + if config["keys"] != nil { + err := removeCryptoKeys(config["keys"]) + if err != nil { + return nil, err + } + } + + return config, nil +} + +func removeServiceCredentials(x interface{}) error { + + switch x := x.(type) { + case []interface{}: + for _, v := range x { + err := removeKey(v, "credentials") + if err != nil { + return err + } + } + + case map[string]interface{}: + for _, v := range x { + err := removeKey(v, "credentials") + if err != nil { + return err + } + } + default: + return fmt.Errorf("illegal service config type: %T", x) + } + + return nil +} + +func removeCryptoKeys(x interface{}) error { + + switch x := x.(type) { + case map[string]interface{}: + for _, v := range x { + err := removeKey(v, "key", "private_key") + if err != nil { + return err + } + } + default: + return fmt.Errorf("illegal keys config type: %T", x) + } + + return nil +} + +func removeKey(x interface{}, keys ...string) error { + val, ok := x.(map[string]interface{}) + if !ok { + return fmt.Errorf("type assertion error") + } + + for _, key := range keys { + delete(val, key) + } + + return nil +} + +type illegalResolver struct{} + +func (illegalResolver) Resolve(ref ast.Ref) (interface{}, error) { + return nil, fmt.Errorf("illegal value: %v", ref) +} diff --git a/vendor/github.com/open-policy-agent/opa/topdown/save.go b/vendor/github.com/open-policy-agent/opa/topdown/save.go index 4ea644f0..85b9aea5 100644 --- a/vendor/github.com/open-policy-agent/opa/topdown/save.go +++ b/vendor/github.com/open-policy-agent/opa/topdown/save.go @@ -279,7 +279,7 @@ func newSaveSupport() *saveSupport { } func (s *saveSupport) List() []*ast.Module { - result := []*ast.Module{} + result := make([]*ast.Module, 0, len(s.modules)) for _, module := range s.modules { result = append(result, module) } @@ -375,12 +375,16 @@ func ignoreExprDuringPartial(expr *ast.Expr) bool { } func ignoreDuringPartial(bi *ast.Builtin) bool { + // Note(philipc): We keep this legacy check around to avoid breaking + // existing library users. + //nolint:staticcheck // We specifically ignore our own linter warning here. for _, ignore := range ast.IgnoreDuringPartialEval { if bi == ignore { return true } } - return false + // Otherwise, ensure all non-deterministic builtins are thrown out. + return bi.Nondeterministic } type inliningControl struct { diff --git a/vendor/github.com/open-policy-agent/opa/topdown/semver.go b/vendor/github.com/open-policy-agent/opa/topdown/semver.go index b91a5a99..20c02d23 100644 --- a/vendor/github.com/open-policy-agent/opa/topdown/semver.go +++ b/vendor/github.com/open-policy-agent/opa/topdown/semver.go @@ -40,8 +40,7 @@ func builtinSemVerCompare(bctx BuiltinContext, args []*ast.Term, iter func(*ast. func builtinSemVerIsValid(bctx BuiltinContext, args []*ast.Term, iter func(*ast.Term) error) error { versionString, err := builtins.StringOperand(args[0].Value, 1) if err != nil { - result := ast.BooleanTerm(false) - return iter(result) + return iter(ast.BooleanTerm(false)) } result := true diff --git a/vendor/github.com/open-policy-agent/opa/topdown/sets.go b/vendor/github.com/open-policy-agent/opa/topdown/sets.go index a9c5ad86..cb0b1d7d 100644 --- a/vendor/github.com/open-policy-agent/opa/topdown/sets.go +++ b/vendor/github.com/open-policy-agent/opa/topdown/sets.go @@ -58,22 +58,26 @@ func builtinSetIntersection(a ast.Value) (ast.Value, error) { // builtinSetUnion returns the union of the given input sets func builtinSetUnion(a ast.Value) (ast.Value, error) { + // The set union logic here is duplicated and manually inlined on + // purpose. By lifting this logic up a level, and not doing pairwise + // set unions, we avoid a number of heap allocations. This improves + // performance dramatically over the naive approach. + result := ast.NewSet() inputSet, err := builtins.SetOperand(a, 1) if err != nil { return nil, err } - result := ast.NewSet() - err = inputSet.Iter(func(x *ast.Term) error { - n, err := builtins.SetOperand(x.Value, 1) + item, err := builtins.SetOperand(x.Value, 1) if err != nil { return err } - result = result.Union(n) + item.Foreach(result.Add) return nil }) + return result, err } diff --git a/vendor/github.com/open-policy-agent/opa/topdown/strings.go b/vendor/github.com/open-policy-agent/opa/topdown/strings.go index e5118b77..1a3d3d56 100644 --- a/vendor/github.com/open-policy-agent/opa/topdown/strings.go +++ b/vendor/github.com/open-policy-agent/opa/topdown/strings.go @@ -10,10 +10,108 @@ import ( "sort" "strings" + "github.com/tchap/go-patricia/v2/patricia" + "github.com/open-policy-agent/opa/ast" "github.com/open-policy-agent/opa/topdown/builtins" ) +func builtinAnyPrefixMatch(bctx BuiltinContext, operands []*ast.Term, iter func(*ast.Term) error) error { + a, b := operands[0].Value, operands[1].Value + + var strs []string + switch a := a.(type) { + case ast.String: + strs = []string{string(a)} + case *ast.Array, ast.Set: + var err error + strs, err = builtins.StringSliceOperand(a, 1) + if err != nil { + return err + } + default: + return builtins.NewOperandTypeErr(1, a, "string", "set", "array") + } + + var prefixes []string + switch b := b.(type) { + case ast.String: + prefixes = []string{string(b)} + case *ast.Array, ast.Set: + var err error + prefixes, err = builtins.StringSliceOperand(b, 2) + if err != nil { + return err + } + default: + return builtins.NewOperandTypeErr(2, b, "string", "set", "array") + } + + return iter(ast.BooleanTerm(anyStartsWithAny(strs, prefixes))) +} + +func builtinAnySuffixMatch(bctx BuiltinContext, operands []*ast.Term, iter func(*ast.Term) error) error { + a, b := operands[0].Value, operands[1].Value + + var strsReversed []string + switch a := a.(type) { + case ast.String: + strsReversed = []string{reverseString(string(a))} + case *ast.Array, ast.Set: + strs, err := builtins.StringSliceOperand(a, 1) + if err != nil { + return err + } + strsReversed = make([]string, len(strs)) + for i := range strs { + strsReversed[i] = reverseString(strs[i]) + } + default: + return builtins.NewOperandTypeErr(1, a, "string", "set", "array") + } + + var suffixesReversed []string + switch b := b.(type) { + case ast.String: + suffixesReversed = []string{reverseString(string(b))} + case *ast.Array, ast.Set: + suffixes, err := builtins.StringSliceOperand(b, 2) + if err != nil { + return err + } + suffixesReversed = make([]string, len(suffixes)) + for i := range suffixes { + suffixesReversed[i] = reverseString(suffixes[i]) + } + default: + return builtins.NewOperandTypeErr(2, b, "string", "set", "array") + } + + return iter(ast.BooleanTerm(anyStartsWithAny(strsReversed, suffixesReversed))) +} + +func anyStartsWithAny(strs []string, prefixes []string) bool { + if len(strs) == 0 || len(prefixes) == 0 { + return false + } + if len(strs) == 1 && len(prefixes) == 1 { + return strings.HasPrefix(strs[0], prefixes[0]) + } + + trie := patricia.NewTrie() + for i := 0; i < len(strs); i++ { + trie.Insert([]byte(strs[i]), true) + } + + for i := 0; i < len(prefixes); i++ { + if trie.MatchSubtree([]byte(prefixes[i])) { + return true + } + } + + return false +} + func builtinFormatInt(a, b ast.Value) (ast.Value, error) { input, err := builtins.NumberOperand(a, 1) @@ -87,19 +185,50 @@ func builtinConcat(a, b ast.Value) (ast.Value, error) { return ast.String(strings.Join(strs, string(join))), nil } -func builtinIndexOf(a, b ast.Value) (ast.Value, error) { - runesEqual := func(a, b []rune) bool { - if len(a) != len(b) { +func runesEqual(a, b []rune) bool { + if len(a) != len(b) { + return false + } + for i, v := range a { + if v != b[i] { return false } - for i, v := range a { - if v != b[i] { - return false + } + return true +} + +func builtinIndexOf(a, b ast.Value) (ast.Value, error) { + base, err := builtins.StringOperand(a, 1) + if err != nil { + return nil, err + } + + search, err := builtins.StringOperand(b, 2) + if err != nil { + return nil, err + } + if len(string(search)) == 0 { + return nil, fmt.Errorf("empty search character") + } + + baseRunes := []rune(string(base)) + searchRunes := []rune(string(search)) + searchLen := len(searchRunes) + + for i, r := range baseRunes { + if len(baseRunes) >= i+searchLen { + if r == searchRunes[0] && runesEqual(baseRunes[i:i+searchLen], searchRunes) { + return ast.IntNumberTerm(i).Value, nil } + } else { + break } - return true } + return ast.IntNumberTerm(-1).Value, nil +} + +func builtinIndexOfN(a, b ast.Value) (ast.Value, error) { base, err := builtins.StringOperand(a, 1) if err != nil { return nil, err @@ -117,17 +246,18 @@ func builtinIndexOf(a, b ast.Value) (ast.Value, error) { searchRunes := []rune(string(search)) searchLen := len(searchRunes) + var arr []*ast.Term for i, r := range baseRunes { - if r == searchRunes[0] { - if len(baseRunes) >= i+searchLen { - if runesEqual(baseRunes[i:i+searchLen], searchRunes) { - return ast.IntNumberTerm(i).Value, nil - } + if len(baseRunes) >= i+searchLen { + if r == searchRunes[0] && runesEqual(baseRunes[i:i+searchLen], searchRunes) { + arr = append(arr, ast.IntNumberTerm(i)) } + } else { + break } } - return ast.IntNumberTerm(-1).Value, nil + return ast.NewArray(arr...), nil } func builtinSubstring(a, b, c ast.Value) (ast.Value, error) { @@ -275,7 +405,7 @@ func builtinReplaceN(a, b ast.Value) (ast.Value, error) { return nil, err } - var oldnewArr []string + oldnewArr := make([]string, 0, len(keys)*2) for _, k := range keys { keyVal, ok := k.Value.(ast.String) if !ok { @@ -418,7 +548,11 @@ func builtinReverse(bctx BuiltinContext, operands []*ast.Term, iter func(*ast.Te return err } - sRunes := []rune(string(s)) + return iter(ast.StringTerm(reverseString(string(s)))) +} + +func reverseString(str string) string { + sRunes := []rune(string(str)) length := len(sRunes) reversedRunes := make([]rune, length) @@ -426,15 +560,14 @@ func builtinReverse(bctx BuiltinContext, operands []*ast.Term, iter func(*ast.Te reversedRunes[length-index-1] = r } - reversedString := string(reversedRunes) - - return iter(ast.StringTerm(reversedString)) + return string(reversedRunes) } func init() { RegisterFunctionalBuiltin2(ast.FormatInt.Name, builtinFormatInt) RegisterFunctionalBuiltin2(ast.Concat.Name, builtinConcat) RegisterFunctionalBuiltin2(ast.IndexOf.Name, builtinIndexOf) + RegisterFunctionalBuiltin2(ast.IndexOfN.Name, builtinIndexOfN) RegisterFunctionalBuiltin3(ast.Substring.Name, builtinSubstring) RegisterFunctionalBuiltin2(ast.Contains.Name, builtinContains) RegisterFunctionalBuiltin2(ast.StartsWith.Name, builtinStartsWith) @@ -451,5 +584,7 @@ func init() { RegisterFunctionalBuiltin2(ast.TrimSuffix.Name, builtinTrimSuffix) RegisterFunctionalBuiltin1(ast.TrimSpace.Name, builtinTrimSpace) RegisterFunctionalBuiltin2(ast.Sprintf.Name, builtinSprintf) + RegisterBuiltinFunc(ast.AnyPrefixMatch.Name, builtinAnyPrefixMatch) + RegisterBuiltinFunc(ast.AnySuffixMatch.Name, builtinAnySuffixMatch) RegisterBuiltinFunc(ast.StringReverse.Name, builtinReverse) } diff --git a/vendor/github.com/open-policy-agent/opa/topdown/subset.go b/vendor/github.com/open-policy-agent/opa/topdown/subset.go new file mode 100644 index 00000000..19c6571c --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/topdown/subset.go @@ -0,0 +1,263 @@ +// Copyright 2022 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +package topdown + +import ( + "github.com/open-policy-agent/opa/ast" + "github.com/open-policy-agent/opa/topdown/builtins" +) + +func bothObjects(t1, t2 *ast.Term) (bool, ast.Object, ast.Object) { + if (t1 == nil) || (t2 == nil) { + return false, nil, nil + } + + obj1, ok := t1.Value.(ast.Object) + if !ok { + return false, nil, nil + } + + obj2, ok := t2.Value.(ast.Object) + if !ok { + return false, nil, nil + } + + return true, obj1, obj2 +} + +func bothSets(t1, t2 *ast.Term) (bool, ast.Set, ast.Set) { + if (t1 == nil) || (t2 == nil) { + return false, nil, nil + } + + set1, ok := t1.Value.(ast.Set) + if !ok { + return false, nil, nil + } + + set2, ok := t2.Value.(ast.Set) + if !ok { + return false, nil, nil + } + + return true, set1, set2 +} + +func bothArrays(t1, t2 *ast.Term) (bool, *ast.Array, *ast.Array) { + if (t1 == nil) || (t2 == nil) { + return false, nil, nil + } + + array1, ok := t1.Value.(*ast.Array) + if !ok { + return false, nil, nil + } + + array2, ok := t2.Value.(*ast.Array) + if !ok { + return false, nil, nil + } + + return true, array1, array2 +} + +func arraySet(t1, t2 *ast.Term) (bool, *ast.Array, ast.Set) { + if (t1 == nil) || (t2 == nil) { + return false, nil, nil + } + + array, ok := t1.Value.(*ast.Array) + if !ok { + return false, nil, nil + } + + set, ok := t2.Value.(ast.Set) + if !ok { + return false, nil, nil + } + + return true, array, set +} + +// objectSubset implements the subset operation on a pair of objects. +// +// This function will try to recursively apply the subset operation where it +// can, such as if both super and sub have an object or set as the value +// associated with a key. +func objectSubset(super ast.Object, sub ast.Object) bool { + var superTerm *ast.Term + isSubset := true + + sub.Until(func(key, subTerm *ast.Term) bool { + // This really wants to be a for loop, hence the somewhat + // weird internal structure. However, using Until() in this + // was is a performance optimization, as it avoids performing + // any key hashing on the sub-object. + + superTerm = super.Get(key) + + // subTerm is can't be nil because we got it from Until(), so + // we only need to verify that super is non-nil. + if superTerm == nil { + isSubset = false + return true // break, not a subset + } + + if subTerm.Equal(superTerm) { + return false // continue + } + + // If both of the terms are objects then we want to apply + // the subset operation recursively, otherwise we just compare + // them normally. If only one term is an object, then we + // do a normal comparison which will come up false. + if ok, superObj, subObj := bothObjects(superTerm, subTerm); ok { + if !objectSubset(superObj, subObj) { + isSubset = false + return true // break, not a subset + } + + return false // continue + } + + if ok, superSet, subSet := bothSets(superTerm, subTerm); ok { + if !setSubset(superSet, subSet) { + isSubset = false + return true // break, not a subset + } + + return false // continue + } + + if ok, superArray, subArray := bothArrays(superTerm, subTerm); ok { + if !arraySubset(superArray, subArray) { + isSubset = false + return true // break, not a subset + } + + return false // continue + } + + // We have already checked for exact equality, as well as for + // all of the types of nested subsets we care about, so if we + // get here it means this isn't a subset. + isSubset = false + return true // break, not a subset + }) + + return isSubset +} + +// setSubset implements the subset operation on sets. +// +// Unlike in the object case, this is not recursive, we just compare values +// using ast.Set.Contains() because we have no well defined way to "match up" +// objects that are in different sets. +func setSubset(super ast.Set, sub ast.Set) bool { + isSubset := true + sub.Until(func(t *ast.Term) bool { + if !super.Contains(t) { + isSubset = false + return true + } + return false + }) + + return isSubset +} + +// arraySubset implements the subset operation on arrays. +// +// This is defined to mean that the entire "sub" array must appear in +// the "super" array. For the same rationale as setSubset(), we do not attempt +// to recurse into values. +func arraySubset(super, sub *ast.Array) bool { + // Notice that this is essentially string search. The naive approach + // used here is O(n^2). This should probably be rewritten later to use + // Boyer-Moore or something. + + if sub.Len() > super.Len() { + return false + } + + if sub.Equal(super) { + return true + } + + superCursor := 0 + subCursor := 0 + for { + if subCursor == sub.Len() { + return true + } + + if superCursor == super.Len() { + return false + } + + subElem := sub.Elem(subCursor) + superElem := sub.Elem(superCursor + subCursor) + if superElem == nil { + return false + } + + if superElem.Value.Compare(subElem.Value) == 0 { + subCursor++ + } else { + superCursor++ + subCursor = 0 + } + } +} + +// arraySetSubset implements the subset operation on array and set. +// +// This is defined to mean that the entire "sub" set must appear in +// the "super" array with no consideration of ordering. +// For the same rationale as setSubset(), we do not attempt +// to recurse into values. +func arraySetSubset(super *ast.Array, sub ast.Set) bool { + unmatched := sub.Len() + return super.Until(func(t *ast.Term) bool { + if sub.Contains(t) { + unmatched-- + } + if unmatched == 0 { + return true + } + return false + }) +} + +func builtinObjectSubset(_ BuiltinContext, operands []*ast.Term, iter func(*ast.Term) error) error { + superTerm := operands[0] + subTerm := operands[1] + + if ok, superObj, subObj := bothObjects(superTerm, subTerm); ok { + // Both operands are objects. + return iter(ast.BooleanTerm(objectSubset(superObj, subObj))) + } + + if ok, superSet, subSet := bothSets(superTerm, subTerm); ok { + // Both operands are sets. + return iter(ast.BooleanTerm(setSubset(superSet, subSet))) + } + + if ok, superArray, subArray := bothArrays(superTerm, subTerm); ok { + // Both operands are sets. + return iter(ast.BooleanTerm(arraySubset(superArray, subArray))) + } + + if ok, superArray, subSet := arraySet(superTerm, subTerm); ok { + // Super operand is array and sub operand is set + return iter(ast.BooleanTerm(arraySetSubset(superArray, subSet))) + } + + return builtins.ErrOperand("both arguments object.subset must be of the same type or array and set") +} + +func init() { + RegisterBuiltinFunc(ast.ObjectSubset.Name, builtinObjectSubset) +} diff --git a/vendor/github.com/open-policy-agent/opa/topdown/tokens.go b/vendor/github.com/open-policy-agent/opa/topdown/tokens.go index 64828831..5309cd0b 100644 --- a/vendor/github.com/open-policy-agent/opa/topdown/tokens.go +++ b/vendor/github.com/open-policy-agent/opa/topdown/tokens.go @@ -15,14 +15,14 @@ import ( "encoding/hex" "encoding/json" "encoding/pem" + "errors" "fmt" "hash" "math/big" "strings" - "github.com/pkg/errors" - "github.com/open-policy-agent/opa/ast" + "github.com/open-policy-agent/opa/internal/jwx/jwa" "github.com/open-policy-agent/opa/internal/jwx/jwk" "github.com/open-policy-agent/opa/internal/jwx/jws" "github.com/open-policy-agent/opa/topdown/builtins" @@ -254,7 +254,12 @@ func builtinJWTVerifyES512(bctx BuiltinContext, args []*ast.Term, iter func(*ast return err } -func verifyES(publicKey interface{}, digest []byte, signature []byte) error { +func verifyES(publicKey interface{}, digest []byte, signature []byte) (err error) { + defer func() { + if r := recover(); r != nil { + err = fmt.Errorf("ECDSA signature verification error: %v", r) + } + }() publicKeyEcdsa, ok := publicKey.(*ecdsa.PublicKey) if !ok { return fmt.Errorf("incorrect public key type") @@ -269,9 +274,16 @@ func verifyES(publicKey interface{}, digest []byte, signature []byte) error { return fmt.Errorf("ECDSA signature verification error") } -// getKeyFromCertOrJWK returns the public key found in a X.509 certificate or JWK key(s). +type verificationKey struct { + alg string + kid string + key interface{} +} + +// getKeysFromCertOrJWK returns the public key found in a X.509 certificate or JWK key(s). // A valid PEM block is never valid JSON (and vice versa), hence can try parsing both. -func getKeyFromCertOrJWK(certificate string) ([]interface{}, error) { +// When provided a JWKS, each key additionally likely contains a key ID and the key algorithm. +func getKeysFromCertOrJWK(certificate string) ([]verificationKey, error) { if block, rest := pem.Decode([]byte(certificate)); block != nil { if len(rest) > 0 { return nil, fmt.Errorf("extra data after a PEM certificate block") @@ -280,19 +292,18 @@ func getKeyFromCertOrJWK(certificate string) ([]interface{}, error) { if block.Type == blockTypeCertificate { cert, err := x509.ParseCertificate(block.Bytes) if err != nil { - return nil, errors.Wrap(err, "failed to parse a PEM certificate") + return nil, fmt.Errorf("failed to parse a PEM certificate: %w", err) } - - return []interface{}{cert.PublicKey}, nil + return []verificationKey{{key: cert.PublicKey}}, nil } if block.Type == "PUBLIC KEY" { key, err := x509.ParsePKIXPublicKey(block.Bytes) if err != nil { - return nil, errors.Wrap(err, "failed to parse a PEM public key") + return nil, fmt.Errorf("failed to parse a PEM public key: %w", err) } - return []interface{}{key}, nil + return []verificationKey{{key: key}}, nil } return nil, fmt.Errorf("failed to extract a Key from the PEM certificate") @@ -300,21 +311,34 @@ func getKeyFromCertOrJWK(certificate string) ([]interface{}, error) { jwks, err := jwk.ParseString(certificate) if err != nil { - return nil, errors.Wrap(err, "failed to parse a JWK key (set)") + return nil, fmt.Errorf("failed to parse a JWK key (set): %w", err) } - var keys []interface{} + keys := make([]verificationKey, 0, len(jwks.Keys)) for _, k := range jwks.Keys { key, err := k.Materialize() if err != nil { return nil, err } - keys = append(keys, key) + keys = append(keys, verificationKey{ + alg: k.GetAlgorithm().String(), + kid: k.GetKeyID(), + key: key, + }) } return keys, nil } +func getKeyByKid(kid string, keys []verificationKey) *verificationKey { + for _, key := range keys { + if key.kid == kid { + return &key + } + } + return nil +} + // Implements JWT signature verification. func builtinJWTVerify(a ast.Value, b ast.Value, hasher func() hash.Hash, verify func(publicKey interface{}, digest []byte, signature []byte) error) (ast.Value, error) { token, err := decodeJWT(a) @@ -327,7 +351,7 @@ func builtinJWTVerify(a ast.Value, b ast.Value, hasher func() hash.Hash, verify return nil, err } - keys, err := getKeyFromCertOrJWK(string(s)) + keys, err := getKeysFromCertOrJWK(string(s)) if err != nil { return nil, err } @@ -337,14 +361,45 @@ func builtinJWTVerify(a ast.Value, b ast.Value, hasher func() hash.Hash, verify return nil, err } + err = token.decodeHeader() + if err != nil { + return nil, err + } + header, err := parseTokenHeader(token) + if err != nil { + return nil, err + } + // Validate the JWT signature - for _, key := range keys { - err = verify(key, - getInputSHA([]byte(token.header+"."+token.payload), hasher), - []byte(signature)) - if err == nil { - return ast.Boolean(true), nil + // First, check if there's a matching key ID (`kid`) in both token header and key(s). + // If a match is found, verify using only that key. Only applicable when a JWKS was provided. + if header.kid != "" { + if key := getKeyByKid(header.kid, keys); key != nil { + err = verify(key.key, getInputSHA([]byte(token.header+"."+token.payload), hasher), []byte(signature)) + + return ast.Boolean(err == nil), nil + } + } + + // If no key ID matched, try to verify using any key in the set + // If an alg is present in both the JWT header and the key, skip verification unless they match + for _, key := range keys { + if key.alg == "" { + // No algorithm provided for the key - this is likely a certificate and not a JWKS, so + // we'll need to verify to find out + err = verify(key.key, getInputSHA([]byte(token.header+"."+token.payload), hasher), []byte(signature)) + if err == nil { + return ast.Boolean(true), nil + } + } else { + if header.alg != key.alg { + continue + } + err = verify(key.key, getInputSHA([]byte(token.header+"."+token.payload), hasher), []byte(signature)) + if err == nil { + return ast.Boolean(true), nil + } } } @@ -446,7 +501,7 @@ func builtinJWTVerifyHS512(bctx BuiltinContext, args []*ast.Term, iter func(*ast // tokenConstraints holds decoded JWT verification constraints. type tokenConstraints struct { // The set of asymmetric keys we can verify with. - keys []interface{} + keys []verificationKey // The single symmetric key we will verify with. secret string @@ -496,10 +551,11 @@ func tokenConstraintCert(value ast.Value, constraints *tokenConstraints) error { return fmt.Errorf("cert constraint: must be a string") } - keys, err := getKeyFromCertOrJWK(string(s)) + keys, err := getKeysFromCertOrJWK(string(s)) if err != nil { return err } + constraints.keys = keys return nil } @@ -596,14 +652,36 @@ func (constraints *tokenConstraints) verify(kid, alg, header, payload, signature } // If we're configured with asymmetric key(s) then only trust that if constraints.keys != nil { + if kid != "" { + if key := getKeyByKid(kid, constraints.keys); key != nil { + err := a.verify(key.key, a.hash, plaintext, []byte(signature)) + if err != nil { + return errSignatureNotVerified + } + return nil + } + } + verified := false for _, key := range constraints.keys { - err := a.verify(key, a.hash, plaintext, []byte(signature)) - if err == nil { - verified = true - break + if key.alg == "" { + err := a.verify(key.key, a.hash, plaintext, []byte(signature)) + if err == nil { + verified = true + break + } + } else { + if alg != key.alg { + continue + } + err := a.verify(key.key, a.hash, plaintext, []byte(signature)) + if err == nil { + verified = true + break + } } } + if !verified { return errSignatureNotVerified } @@ -710,7 +788,12 @@ func verifyRSAPSS(key interface{}, hash crypto.Hash, digest []byte, signature [] return nil } -func verifyECDSA(key interface{}, hash crypto.Hash, digest []byte, signature []byte) error { +func verifyECDSA(key interface{}, hash crypto.Hash, digest []byte, signature []byte) (err error) { + defer func() { + if r := recover(); r != nil { + err = fmt.Errorf("ECDSA signature verification error: %v", r) + } + }() publicKeyEcdsa, ok := key.(*ecdsa.PublicKey) if !ok { return fmt.Errorf("incorrect public key type") @@ -844,6 +927,9 @@ func commonBuiltinJWTEncodeSign(bctx BuiltinContext, inputHeaders, jwsPayload, j return err } alg := standardHeaders.GetAlgorithm() + if alg == jwa.Unsupported { + return fmt.Errorf("unknown signature algorithm") + } if (standardHeaders.Type == "" || standardHeaders.Type == headerJwt) && !json.Valid([]byte(jwsPayload)) { return fmt.Errorf("type is JWT but payload is not JSON") @@ -988,18 +1074,28 @@ func builtinJWTDecodeVerify(bctx BuiltinContext, args []*ast.Term, iter func(*as } // RFC7159 4.1.4 exp if exp := payload.Get(jwtExpKey); exp != nil { - // constraints.time is in nanoseconds but exp Value is in seconds - compareTime := ast.FloatNumberTerm(float64(constraints.time) / 1000000000) - if ast.Compare(compareTime, exp.Value.(ast.Number)) != -1 { - return iter(unverified) + switch exp.Value.(type) { + case ast.Number: + // constraints.time is in nanoseconds but exp Value is in seconds + compareTime := ast.FloatNumberTerm(float64(constraints.time) / 1000000000) + if ast.Compare(compareTime, exp.Value.(ast.Number)) != -1 { + return iter(unverified) + } + default: + return fmt.Errorf("exp value must be a number") } } // RFC7159 4.1.5 nbf if nbf := payload.Get(jwtNbfKey); nbf != nil { - // constraints.time is in nanoseconds but nbf Value is in seconds - compareTime := ast.FloatNumberTerm(float64(constraints.time) / 1000000000) - if ast.Compare(compareTime, nbf.Value.(ast.Number)) == -1 { - return iter(unverified) + switch nbf.Value.(type) { + case ast.Number: + // constraints.time is in nanoseconds but nbf Value is in seconds + compareTime := ast.FloatNumberTerm(float64(constraints.time) / 1000000000) + if ast.Compare(compareTime, nbf.Value.(ast.Number)) == -1 { + return iter(unverified) + } + default: + return fmt.Errorf("nbf value must be a number") } } diff --git a/vendor/github.com/open-policy-agent/opa/topdown/trace.go b/vendor/github.com/open-policy-agent/opa/topdown/trace.go index 49aa5aff..ef50f91f 100644 --- a/vendor/github.com/open-policy-agent/opa/topdown/trace.go +++ b/vendor/github.com/open-policy-agent/opa/topdown/trace.go @@ -78,6 +78,9 @@ type Event struct { LocalMetadata map[ast.Var]VarMetadata // Contains metadata for the local variable bindings. Nil if variables were not included in the trace event. Message string // Contains message for Note events. Ref *ast.Ref // Identifies the subject ref for the event. Only applies to Index and Wasm operations. + + input *ast.Term + bindings *bindings } // HasRule returns true if the Event contains an ast.Rule. @@ -119,6 +122,17 @@ func (evt *Event) String() string { return fmt.Sprintf("%v %v %v (qid=%v, pqid=%v)", evt.Op, evt.Node, evt.Locals, evt.QueryID, evt.ParentID) } +// Input returns the input object as it was at the event. +func (evt *Event) Input() *ast.Term { + return evt.input +} + +// Plug plugs event bindings into the provided ast.Term. Because bindings are mutable, this only makes sense to do when +// the event is emitted rather than on recorded trace events as the bindings are going to be different by then. +func (evt *Event) Plug(term *ast.Term) *ast.Term { + return evt.bindings.Plug(term) +} + func (evt *Event) equalNodes(other *Event) bool { switch a := evt.Node.(type) { case ast.Body: @@ -268,11 +282,10 @@ func formatEvent(event *Event, depth int) string { func formatEventPadding(event *Event, depth int) string { spaces := formatEventSpaces(event, depth) - padding := "" if spaces > 1 { - padding += strings.Repeat("| ", spaces-1) + return strings.Repeat("| ", spaces-1) } - return padding + return "" } func formatEventSpaces(event *Event, depth int) int { @@ -399,7 +412,18 @@ func rewrite(event *Event) *Event { switch v := event.Node.(type) { case *ast.Expr: - node = v.Copy() + expr := v.Copy() + + // Hide generated local vars in 'key' position that have not been + // rewritten. + if ev, ok := v.Terms.(*ast.Every); ok { + if kv, ok := ev.Key.Value.(ast.Var); ok { + if rw, ok := cpy.LocalMetadata[kv]; !ok || rw.Name.IsGenerated() { + expr.Terms.(*ast.Every).Key = nil + } + } + } + node = expr case ast.Body: node = v.Copy() case *ast.Rule: diff --git a/vendor/github.com/open-policy-agent/opa/topdown/type.go b/vendor/github.com/open-policy-agent/opa/topdown/type.go index 2fe7a39a..ff35f777 100644 --- a/vendor/github.com/open-policy-agent/opa/topdown/type.go +++ b/vendor/github.com/open-policy-agent/opa/topdown/type.go @@ -1,4 +1,4 @@ -// Copyright 2018 The OPA Authors. All rights reserved. +// Copyright 2022 The OPA Authors. All rights reserved. // Use of this source code is governed by an Apache2 // license that can be found in the LICENSE file. @@ -13,7 +13,7 @@ func builtinIsNumber(a ast.Value) (ast.Value, error) { case ast.Number: return ast.Boolean(true), nil default: - return nil, BuiltinEmpty{} + return ast.Boolean(false), nil } } @@ -22,7 +22,7 @@ func builtinIsString(a ast.Value) (ast.Value, error) { case ast.String: return ast.Boolean(true), nil default: - return nil, BuiltinEmpty{} + return ast.Boolean(false), nil } } @@ -31,7 +31,7 @@ func builtinIsBoolean(a ast.Value) (ast.Value, error) { case ast.Boolean: return ast.Boolean(true), nil default: - return nil, BuiltinEmpty{} + return ast.Boolean(false), nil } } @@ -40,7 +40,7 @@ func builtinIsArray(a ast.Value) (ast.Value, error) { case *ast.Array: return ast.Boolean(true), nil default: - return nil, BuiltinEmpty{} + return ast.Boolean(false), nil } } @@ -49,7 +49,7 @@ func builtinIsSet(a ast.Value) (ast.Value, error) { case ast.Set: return ast.Boolean(true), nil default: - return nil, BuiltinEmpty{} + return ast.Boolean(false), nil } } @@ -58,7 +58,7 @@ func builtinIsObject(a ast.Value) (ast.Value, error) { case ast.Object: return ast.Boolean(true), nil default: - return nil, BuiltinEmpty{} + return ast.Boolean(false), nil } } @@ -67,7 +67,7 @@ func builtinIsNull(a ast.Value) (ast.Value, error) { case ast.Null: return ast.Boolean(true), nil default: - return nil, BuiltinEmpty{} + return ast.Boolean(false), nil } } diff --git a/vendor/github.com/open-policy-agent/opa/types/types.go b/vendor/github.com/open-policy-agent/opa/types/types.go index 9b5cc66c..a4b0ad38 100644 --- a/vendor/github.com/open-policy-agent/opa/types/types.go +++ b/vendor/github.com/open-policy-agent/opa/types/types.go @@ -48,6 +48,42 @@ func NewNull() Null { return Null{} } +type NamedType struct { + Name, Descr string + Type Type +} + +func (n *NamedType) typeMarker() string { return n.Type.typeMarker() } +func (n *NamedType) String() string { return n.Name + ": " + n.Type.String() } +func (n *NamedType) MarshalJSON() ([]byte, error) { + var obj map[string]interface{} + switch x := n.Type.(type) { + case interface{ toMap() map[string]interface{} }: + obj = x.toMap() + default: + obj = map[string]interface{}{ + "type": n.Type.typeMarker(), + } + } + obj["name"] = n.Name + if n.Descr != "" { + obj["description"] = n.Descr + } + return json.Marshal(obj) +} + +func (n *NamedType) Description(d string) *NamedType { + n.Descr = d + return n +} + +func Named(name string, t Type) *NamedType { + return &NamedType{ + Type: t, + Name: name, + } +} + // MarshalJSON returns the JSON encoding of t. func (t Null) MarshalJSON() ([]byte, error) { return json.Marshal(map[string]interface{}{ @@ -55,6 +91,15 @@ func (t Null) MarshalJSON() ([]byte, error) { }) } +func unwrap(t Type) Type { + switch t := t.(type) { + case *NamedType: + return t.Type + default: + return t + } +} + func (t Null) String() string { return typeNull } @@ -100,7 +145,7 @@ func (t String) MarshalJSON() ([]byte, error) { }) } -func (t String) String() string { +func (String) String() string { return typeString } @@ -142,6 +187,10 @@ func NewArray(static []Type, dynamic Type) *Array { // MarshalJSON returns the JSON encoding of t. func (t *Array) MarshalJSON() ([]byte, error) { + return json.Marshal(t.toMap()) +} + +func (t *Array) toMap() map[string]interface{} { repr := map[string]interface{}{ "type": t.typeMarker(), } @@ -151,7 +200,7 @@ func (t *Array) MarshalJSON() ([]byte, error) { if t.dynamic != nil { repr["dynamic"] = t.dynamic } - return json.Marshal(repr) + return repr } func (t *Array) String() string { @@ -207,13 +256,17 @@ func NewSet(of Type) *Set { // MarshalJSON returns the JSON encoding of t. func (t *Set) MarshalJSON() ([]byte, error) { + return json.Marshal(t.toMap()) +} + +func (t *Set) toMap() map[string]interface{} { repr := map[string]interface{}{ "type": t.typeMarker(), } if t.of != nil { repr["of"] = t.of } - return json.Marshal(repr) + return repr } func (t *Set) String() string { @@ -332,6 +385,10 @@ func (t *Object) Keys() []interface{} { // MarshalJSON returns the JSON encoding of t. func (t *Object) MarshalJSON() ([]byte, error) { + return json.Marshal(t.toMap()) +} + +func (t *Object) toMap() map[string]interface{} { repr := map[string]interface{}{ "type": t.typeMarker(), } @@ -341,7 +398,7 @@ func (t *Object) MarshalJSON() ([]byte, error) { if t.dynamic != nil { repr["dynamic"] = t.dynamic } - return json.Marshal(repr) + return repr } // Select returns the type of the named property. @@ -395,13 +452,17 @@ func (t Any) Contains(other Type) bool { // MarshalJSON returns the JSON encoding of t. func (t Any) MarshalJSON() ([]byte, error) { - data := map[string]interface{}{ + return json.Marshal(t.toMap()) +} + +func (t Any) toMap() map[string]interface{} { + repr := map[string]interface{}{ "type": t.typeMarker(), } if len(t) != 0 { - data["of"] = []Type(t) + repr["of"] = []Type(t) } - return json.Marshal(data) + return repr } // Merge return a new Any type that is the superset of t and other. @@ -487,8 +548,7 @@ func Arity(x Type) int { return len(f.FuncArgs().Args) } -// NewFunction returns a new Function object where xs[:len(xs)-1] are arguments -// and xs[len(xs)-1] is the result type. +// NewFunction returns a new Function object of the given argument and result types. func NewFunction(args []Type, result Type) *Function { return &Function{ args: args, @@ -512,19 +572,34 @@ func NewVariadicFunction(args []Type, varargs Type, result Type) *Function { // FuncArgs returns the function's arguments. func (t *Function) FuncArgs() FuncArgs { - return FuncArgs{Args: t.Args(), Variadic: t.variadic} + return FuncArgs{Args: t.Args(), Variadic: unwrap(t.variadic)} +} + +// NamedFuncArgs returns the function's arguments, with a name and +// description if available. +func (t *Function) NamedFuncArgs() FuncArgs { + args := make([]Type, len(t.args)) + copy(args, t.args) + return FuncArgs{Args: args, Variadic: t.variadic} } // Args returns the function's arguments as a slice, ignoring variadic arguments. // Deprecated: Use FuncArgs instead. func (t *Function) Args() []Type { cpy := make([]Type, len(t.args)) - copy(cpy, t.args) + for i := range t.args { + cpy[i] = unwrap(t.args[i]) + } return cpy } // Result returns the function's result type. func (t *Function) Result() Type { + return unwrap(t.result) +} + +// Result returns the function's result type, without stripping name and description. +func (t *Function) NamedResult() Type { return t.result } @@ -566,12 +641,13 @@ func (t *Function) UnmarshalJSON(bs []byte) error { return nil } -// Union returns a new function represnting the union of t and other. Functions +// Union returns a new function representing the union of t and other. Functions // must have the same arity to be unioned. func (t *Function) Union(other *Function) *Function { if other == nil { return t - } else if t == nil { + } + if t == nil { return other } @@ -608,7 +684,7 @@ type FuncArgs struct { } func (a FuncArgs) String() string { - var buf []string + buf := make([]string, 0, len(a.Args)+1) for i := range a.Args { buf = append(buf, Sprint(a.Args[i])) } @@ -618,6 +694,7 @@ func (a FuncArgs) String() string { return "(" + strings.Join(buf, ", ") + ")" } +// Arg returns the nth argument's type. func (a FuncArgs) Arg(x int) Type { if x < len(a.Args) { return a.Args[x] @@ -627,6 +704,7 @@ func (a FuncArgs) Arg(x int) Type { // Compare returns -1, 0, 1 based on comparison between a and b. func Compare(a, b Type) int { + a, b = unwrap(a), unwrap(b) x := typeOrder(a) y := typeOrder(b) if x > y { @@ -731,7 +809,7 @@ func Compare(a, b Type) int { // Contains returns true if a is a superset or equal to b. func Contains(a, b Type) bool { - if any, ok := a.(Any); ok { + if any, ok := unwrap(a).(Any); ok { return any.Contains(b) } return Compare(a, b) == 0 @@ -740,6 +818,7 @@ func Contains(a, b Type) bool { // Or returns a type that represents the union of a and b. If one type is a // superset of the other, the superset is returned unchanged. func Or(a, b Type) Type { + a, b = unwrap(a), unwrap(b) if a == nil { return b } else if b == nil { @@ -768,7 +847,7 @@ func Or(a, b Type) Type { // Select returns a property or item of a. func Select(a Type, x interface{}) Type { - switch a := a.(type) { + switch a := unwrap(a).(type) { case *Array: n, ok := x.(json.Number) if !ok { @@ -811,7 +890,7 @@ func Select(a Type, x interface{}) Type { // keys are always number types, for objects the keys are always string types, // and for sets the keys are always the type of the set element. func Keys(a Type) Type { - switch a := a.(type) { + switch a := unwrap(a).(type) { case *Array: return N case *Object: @@ -841,7 +920,7 @@ func Keys(a Type) Type { // Values returns the type of values that can be enumerated for a. func Values(a Type) Type { - switch a := a.(type) { + switch a := unwrap(a).(type) { case *Array: var tpe Type for i := range a.static { @@ -874,7 +953,7 @@ func Values(a Type) Type { // Nil returns true if a's type is unknown. func Nil(a Type) bool { - switch a := a.(type) { + switch a := unwrap(a).(type) { case nil: return true case *Function: @@ -969,7 +1048,7 @@ func typeSliceCompare(a, b []Type) int { } func typeOrder(x Type) int { - switch x.(type) { + switch unwrap(x).(type) { case Null: return 0 case Boolean: diff --git a/vendor/github.com/open-policy-agent/opa/util/backoff.go b/vendor/github.com/open-policy-agent/opa/util/backoff.go index 1b126fbe..944b5728 100644 --- a/vendor/github.com/open-policy-agent/opa/util/backoff.go +++ b/vendor/github.com/open-policy-agent/opa/util/backoff.go @@ -9,6 +9,15 @@ import ( "time" ) +func init() { + // NOTE(sr): We don't need good random numbers here; it's used for jittering + // the backup timing a bit. But anyways, let's make it random enough; without + // a call to rand.Seed() we'd get the same stream of numbers for each program + // run. (Or not, if some other packages happens to seed the global randomness + // source.) + rand.Seed(time.Now().UnixNano()) +} + // DefaultBackoff returns a delay with an exponential backoff based on the // number of retries. func DefaultBackoff(base, max float64, retries int) time.Duration { diff --git a/vendor/github.com/open-policy-agent/opa/util/compare.go b/vendor/github.com/open-policy-agent/opa/util/compare.go index dd187590..83087e6b 100644 --- a/vendor/github.com/open-policy-agent/opa/util/compare.go +++ b/vendor/github.com/open-policy-agent/opa/util/compare.go @@ -14,7 +14,7 @@ import ( // Compare returns 0 if a equals b, -1 if a is less than b, and 1 if b is than a. // // For comparison between values of different types, the following ordering is used: -// nil < bool < float64 < string < []interface{} < map[string]interface{}. Slices and maps +// nil < bool < int, float64 < string < []interface{} < map[string]interface{}. Slices and maps // are compared recursively. If one slice or map is a subset of the other slice or map // it is considered "less than". Nil is always equal to nil. // @@ -45,6 +45,26 @@ func Compare(a, b interface{}) int { case json.Number: return compareJSONNumber(a, b) } + case int: + switch b := b.(type) { + case int: + if a == b { + return 0 + } else if a < b { + return -1 + } + return 1 + } + case float64: + switch b := b.(type) { + case float64: + if a == b { + return 0 + } else if a < b { + return -1 + } + return 1 + } case string: switch b := b.(type) { case string: @@ -150,6 +170,10 @@ func sortOrder(v interface{}) int { return boolSort case json.Number: return numberSort + case int: + return numberSort + case float64: + return numberSort case string: return stringSort case []interface{}: diff --git a/vendor/github.com/open-policy-agent/opa/util/json.go b/vendor/github.com/open-policy-agent/opa/util/json.go index cfc94e06..283c4969 100644 --- a/vendor/github.com/open-policy-agent/opa/util/json.go +++ b/vendor/github.com/open-policy-agent/opa/util/json.go @@ -105,6 +105,9 @@ func Reference(x interface{}) *interface{} { // Unmarshal decodes a YAML or JSON value into the specified type. func Unmarshal(bs []byte, v interface{}) error { + if json.Valid(bs) { + return UnmarshalJSON(bs, v) + } bs, err := yaml.YAMLToJSON(bs) if err != nil { return err diff --git a/vendor/github.com/open-policy-agent/opa/version/version.go b/vendor/github.com/open-policy-agent/opa/version/version.go index 0b9c998e..daecddbd 100644 --- a/vendor/github.com/open-policy-agent/opa/version/version.go +++ b/vendor/github.com/open-policy-agent/opa/version/version.go @@ -10,11 +10,14 @@ import ( ) // Version is the canonical version of OPA. -var Version = "0.36.1" +var Version = "0.45.0" // GoVersion is the version of Go this was built with var GoVersion = runtime.Version() +// Platform is the runtime OS and architecture of this OPA binary +var Platform = runtime.GOOS + "/" + runtime.GOARCH + // Additional version information that is displayed by the "version" command and used to // identify the version of running instances of OPA. var ( diff --git a/vendor/github.com/open-policy-agent/opa/version/version_go1.18.go b/vendor/github.com/open-policy-agent/opa/version/version_go1.18.go new file mode 100644 index 00000000..8c02ca49 --- /dev/null +++ b/vendor/github.com/open-policy-agent/opa/version/version_go1.18.go @@ -0,0 +1,33 @@ +// Copyright 2022 The OPA Authors. All rights reserved. +// Use of this source code is governed by an Apache2 +// license that can be found in the LICENSE file. + +//go:build go1.18 +// +build go1.18 + +package version + +import ( + "runtime/debug" +) + +func init() { + bi, ok := debug.ReadBuildInfo() + if !ok { + return + } + dirty := false + for _, s := range bi.Settings { + switch s.Key { + case "vcs.time": + Timestamp = s.Value + case "vcs.revision": + Vcs = s.Value + case "vcs.modified": + dirty = s.Value == "true" + } + } + if dirty { + Vcs = Vcs + "-dirty" + } +} diff --git a/vendor/github.com/prometheus/client_model/go/metrics.pb.go b/vendor/github.com/prometheus/client_model/go/metrics.pb.go index 2f4930d9..35904ea1 100644 --- a/vendor/github.com/prometheus/client_model/go/metrics.pb.go +++ b/vendor/github.com/prometheus/client_model/go/metrics.pb.go @@ -1,5 +1,5 @@ // Code generated by protoc-gen-go. DO NOT EDIT. -// source: metrics.proto +// source: io/prometheus/client/metrics.proto package io_prometheus_client @@ -24,11 +24,18 @@ const _ = proto.ProtoPackageIsVersion3 // please upgrade the proto package type MetricType int32 const ( - MetricType_COUNTER MetricType = 0 - MetricType_GAUGE MetricType = 1 - MetricType_SUMMARY MetricType = 2 - MetricType_UNTYPED MetricType = 3 + // COUNTER must use the Metric field "counter". + MetricType_COUNTER MetricType = 0 + // GAUGE must use the Metric field "gauge". + MetricType_GAUGE MetricType = 1 + // SUMMARY must use the Metric field "summary". + MetricType_SUMMARY MetricType = 2 + // UNTYPED must use the Metric field "untyped". + MetricType_UNTYPED MetricType = 3 + // HISTOGRAM must use the Metric field "histogram". MetricType_HISTOGRAM MetricType = 4 + // GAUGE_HISTOGRAM must use the Metric field "histogram". + MetricType_GAUGE_HISTOGRAM MetricType = 5 ) var MetricType_name = map[int32]string{ @@ -37,14 +44,16 @@ var MetricType_name = map[int32]string{ 2: "SUMMARY", 3: "UNTYPED", 4: "HISTOGRAM", + 5: "GAUGE_HISTOGRAM", } var MetricType_value = map[string]int32{ - "COUNTER": 0, - "GAUGE": 1, - "SUMMARY": 2, - "UNTYPED": 3, - "HISTOGRAM": 4, + "COUNTER": 0, + "GAUGE": 1, + "SUMMARY": 2, + "UNTYPED": 3, + "HISTOGRAM": 4, + "GAUGE_HISTOGRAM": 5, } func (x MetricType) Enum() *MetricType { @@ -67,7 +76,7 @@ func (x *MetricType) UnmarshalJSON(data []byte) error { } func (MetricType) EnumDescriptor() ([]byte, []int) { - return fileDescriptor_6039342a2ba47b72, []int{0} + return fileDescriptor_d1e5ddb18987a258, []int{0} } type LabelPair struct { @@ -82,7 +91,7 @@ func (m *LabelPair) Reset() { *m = LabelPair{} } func (m *LabelPair) String() string { return proto.CompactTextString(m) } func (*LabelPair) ProtoMessage() {} func (*LabelPair) Descriptor() ([]byte, []int) { - return fileDescriptor_6039342a2ba47b72, []int{0} + return fileDescriptor_d1e5ddb18987a258, []int{0} } func (m *LabelPair) XXX_Unmarshal(b []byte) error { @@ -128,7 +137,7 @@ func (m *Gauge) Reset() { *m = Gauge{} } func (m *Gauge) String() string { return proto.CompactTextString(m) } func (*Gauge) ProtoMessage() {} func (*Gauge) Descriptor() ([]byte, []int) { - return fileDescriptor_6039342a2ba47b72, []int{1} + return fileDescriptor_d1e5ddb18987a258, []int{1} } func (m *Gauge) XXX_Unmarshal(b []byte) error { @@ -168,7 +177,7 @@ func (m *Counter) Reset() { *m = Counter{} } func (m *Counter) String() string { return proto.CompactTextString(m) } func (*Counter) ProtoMessage() {} func (*Counter) Descriptor() ([]byte, []int) { - return fileDescriptor_6039342a2ba47b72, []int{2} + return fileDescriptor_d1e5ddb18987a258, []int{2} } func (m *Counter) XXX_Unmarshal(b []byte) error { @@ -215,7 +224,7 @@ func (m *Quantile) Reset() { *m = Quantile{} } func (m *Quantile) String() string { return proto.CompactTextString(m) } func (*Quantile) ProtoMessage() {} func (*Quantile) Descriptor() ([]byte, []int) { - return fileDescriptor_6039342a2ba47b72, []int{3} + return fileDescriptor_d1e5ddb18987a258, []int{3} } func (m *Quantile) XXX_Unmarshal(b []byte) error { @@ -263,7 +272,7 @@ func (m *Summary) Reset() { *m = Summary{} } func (m *Summary) String() string { return proto.CompactTextString(m) } func (*Summary) ProtoMessage() {} func (*Summary) Descriptor() ([]byte, []int) { - return fileDescriptor_6039342a2ba47b72, []int{4} + return fileDescriptor_d1e5ddb18987a258, []int{4} } func (m *Summary) XXX_Unmarshal(b []byte) error { @@ -316,7 +325,7 @@ func (m *Untyped) Reset() { *m = Untyped{} } func (m *Untyped) String() string { return proto.CompactTextString(m) } func (*Untyped) ProtoMessage() {} func (*Untyped) Descriptor() ([]byte, []int) { - return fileDescriptor_6039342a2ba47b72, []int{5} + return fileDescriptor_d1e5ddb18987a258, []int{5} } func (m *Untyped) XXX_Unmarshal(b []byte) error { @@ -345,9 +354,34 @@ func (m *Untyped) GetValue() float64 { } type Histogram struct { - SampleCount *uint64 `protobuf:"varint,1,opt,name=sample_count,json=sampleCount" json:"sample_count,omitempty"` - SampleSum *float64 `protobuf:"fixed64,2,opt,name=sample_sum,json=sampleSum" json:"sample_sum,omitempty"` - Bucket []*Bucket `protobuf:"bytes,3,rep,name=bucket" json:"bucket,omitempty"` + SampleCount *uint64 `protobuf:"varint,1,opt,name=sample_count,json=sampleCount" json:"sample_count,omitempty"` + SampleCountFloat *float64 `protobuf:"fixed64,4,opt,name=sample_count_float,json=sampleCountFloat" json:"sample_count_float,omitempty"` + SampleSum *float64 `protobuf:"fixed64,2,opt,name=sample_sum,json=sampleSum" json:"sample_sum,omitempty"` + // Buckets for the conventional histogram. + Bucket []*Bucket `protobuf:"bytes,3,rep,name=bucket" json:"bucket,omitempty"` + // schema defines the bucket schema. Currently, valid numbers are -4 <= n <= 8. + // They are all for base-2 bucket schemas, where 1 is a bucket boundary in each case, and + // then each power of two is divided into 2^n logarithmic buckets. + // Or in other words, each bucket boundary is the previous boundary times 2^(2^-n). + // In the future, more bucket schemas may be added using numbers < -4 or > 8. + Schema *int32 `protobuf:"zigzag32,5,opt,name=schema" json:"schema,omitempty"` + ZeroThreshold *float64 `protobuf:"fixed64,6,opt,name=zero_threshold,json=zeroThreshold" json:"zero_threshold,omitempty"` + ZeroCount *uint64 `protobuf:"varint,7,opt,name=zero_count,json=zeroCount" json:"zero_count,omitempty"` + ZeroCountFloat *float64 `protobuf:"fixed64,8,opt,name=zero_count_float,json=zeroCountFloat" json:"zero_count_float,omitempty"` + // Negative buckets for the native histogram. + NegativeSpan []*BucketSpan `protobuf:"bytes,9,rep,name=negative_span,json=negativeSpan" json:"negative_span,omitempty"` + // Use either "negative_delta" or "negative_count", the former for + // regular histograms with integer counts, the latter for float + // histograms. + NegativeDelta []int64 `protobuf:"zigzag64,10,rep,name=negative_delta,json=negativeDelta" json:"negative_delta,omitempty"` + NegativeCount []float64 `protobuf:"fixed64,11,rep,name=negative_count,json=negativeCount" json:"negative_count,omitempty"` + // Positive buckets for the native histogram. + PositiveSpan []*BucketSpan `protobuf:"bytes,12,rep,name=positive_span,json=positiveSpan" json:"positive_span,omitempty"` + // Use either "positive_delta" or "positive_count", the former for + // regular histograms with integer counts, the latter for float + // histograms. + PositiveDelta []int64 `protobuf:"zigzag64,13,rep,name=positive_delta,json=positiveDelta" json:"positive_delta,omitempty"` + PositiveCount []float64 `protobuf:"fixed64,14,rep,name=positive_count,json=positiveCount" json:"positive_count,omitempty"` XXX_NoUnkeyedLiteral struct{} `json:"-"` XXX_unrecognized []byte `json:"-"` XXX_sizecache int32 `json:"-"` @@ -357,7 +391,7 @@ func (m *Histogram) Reset() { *m = Histogram{} } func (m *Histogram) String() string { return proto.CompactTextString(m) } func (*Histogram) ProtoMessage() {} func (*Histogram) Descriptor() ([]byte, []int) { - return fileDescriptor_6039342a2ba47b72, []int{6} + return fileDescriptor_d1e5ddb18987a258, []int{6} } func (m *Histogram) XXX_Unmarshal(b []byte) error { @@ -385,6 +419,13 @@ func (m *Histogram) GetSampleCount() uint64 { return 0 } +func (m *Histogram) GetSampleCountFloat() float64 { + if m != nil && m.SampleCountFloat != nil { + return *m.SampleCountFloat + } + return 0 +} + func (m *Histogram) GetSampleSum() float64 { if m != nil && m.SampleSum != nil { return *m.SampleSum @@ -399,8 +440,81 @@ func (m *Histogram) GetBucket() []*Bucket { return nil } +func (m *Histogram) GetSchema() int32 { + if m != nil && m.Schema != nil { + return *m.Schema + } + return 0 +} + +func (m *Histogram) GetZeroThreshold() float64 { + if m != nil && m.ZeroThreshold != nil { + return *m.ZeroThreshold + } + return 0 +} + +func (m *Histogram) GetZeroCount() uint64 { + if m != nil && m.ZeroCount != nil { + return *m.ZeroCount + } + return 0 +} + +func (m *Histogram) GetZeroCountFloat() float64 { + if m != nil && m.ZeroCountFloat != nil { + return *m.ZeroCountFloat + } + return 0 +} + +func (m *Histogram) GetNegativeSpan() []*BucketSpan { + if m != nil { + return m.NegativeSpan + } + return nil +} + +func (m *Histogram) GetNegativeDelta() []int64 { + if m != nil { + return m.NegativeDelta + } + return nil +} + +func (m *Histogram) GetNegativeCount() []float64 { + if m != nil { + return m.NegativeCount + } + return nil +} + +func (m *Histogram) GetPositiveSpan() []*BucketSpan { + if m != nil { + return m.PositiveSpan + } + return nil +} + +func (m *Histogram) GetPositiveDelta() []int64 { + if m != nil { + return m.PositiveDelta + } + return nil +} + +func (m *Histogram) GetPositiveCount() []float64 { + if m != nil { + return m.PositiveCount + } + return nil +} + +// A Bucket of a conventional histogram, each of which is treated as +// an individual counter-like time series by Prometheus. type Bucket struct { CumulativeCount *uint64 `protobuf:"varint,1,opt,name=cumulative_count,json=cumulativeCount" json:"cumulative_count,omitempty"` + CumulativeCountFloat *float64 `protobuf:"fixed64,4,opt,name=cumulative_count_float,json=cumulativeCountFloat" json:"cumulative_count_float,omitempty"` UpperBound *float64 `protobuf:"fixed64,2,opt,name=upper_bound,json=upperBound" json:"upper_bound,omitempty"` Exemplar *Exemplar `protobuf:"bytes,3,opt,name=exemplar" json:"exemplar,omitempty"` XXX_NoUnkeyedLiteral struct{} `json:"-"` @@ -412,7 +526,7 @@ func (m *Bucket) Reset() { *m = Bucket{} } func (m *Bucket) String() string { return proto.CompactTextString(m) } func (*Bucket) ProtoMessage() {} func (*Bucket) Descriptor() ([]byte, []int) { - return fileDescriptor_6039342a2ba47b72, []int{7} + return fileDescriptor_d1e5ddb18987a258, []int{7} } func (m *Bucket) XXX_Unmarshal(b []byte) error { @@ -440,6 +554,13 @@ func (m *Bucket) GetCumulativeCount() uint64 { return 0 } +func (m *Bucket) GetCumulativeCountFloat() float64 { + if m != nil && m.CumulativeCountFloat != nil { + return *m.CumulativeCountFloat + } + return 0 +} + func (m *Bucket) GetUpperBound() float64 { if m != nil && m.UpperBound != nil { return *m.UpperBound @@ -454,6 +575,59 @@ func (m *Bucket) GetExemplar() *Exemplar { return nil } +// A BucketSpan defines a number of consecutive buckets in a native +// histogram with their offset. Logically, it would be more +// straightforward to include the bucket counts in the Span. However, +// the protobuf representation is more compact in the way the data is +// structured here (with all the buckets in a single array separate +// from the Spans). +type BucketSpan struct { + Offset *int32 `protobuf:"zigzag32,1,opt,name=offset" json:"offset,omitempty"` + Length *uint32 `protobuf:"varint,2,opt,name=length" json:"length,omitempty"` + XXX_NoUnkeyedLiteral struct{} `json:"-"` + XXX_unrecognized []byte `json:"-"` + XXX_sizecache int32 `json:"-"` +} + +func (m *BucketSpan) Reset() { *m = BucketSpan{} } +func (m *BucketSpan) String() string { return proto.CompactTextString(m) } +func (*BucketSpan) ProtoMessage() {} +func (*BucketSpan) Descriptor() ([]byte, []int) { + return fileDescriptor_d1e5ddb18987a258, []int{8} +} + +func (m *BucketSpan) XXX_Unmarshal(b []byte) error { + return xxx_messageInfo_BucketSpan.Unmarshal(m, b) +} +func (m *BucketSpan) XXX_Marshal(b []byte, deterministic bool) ([]byte, error) { + return xxx_messageInfo_BucketSpan.Marshal(b, m, deterministic) +} +func (m *BucketSpan) XXX_Merge(src proto.Message) { + xxx_messageInfo_BucketSpan.Merge(m, src) +} +func (m *BucketSpan) XXX_Size() int { + return xxx_messageInfo_BucketSpan.Size(m) +} +func (m *BucketSpan) XXX_DiscardUnknown() { + xxx_messageInfo_BucketSpan.DiscardUnknown(m) +} + +var xxx_messageInfo_BucketSpan proto.InternalMessageInfo + +func (m *BucketSpan) GetOffset() int32 { + if m != nil && m.Offset != nil { + return *m.Offset + } + return 0 +} + +func (m *BucketSpan) GetLength() uint32 { + if m != nil && m.Length != nil { + return *m.Length + } + return 0 +} + type Exemplar struct { Label []*LabelPair `protobuf:"bytes,1,rep,name=label" json:"label,omitempty"` Value *float64 `protobuf:"fixed64,2,opt,name=value" json:"value,omitempty"` @@ -467,7 +641,7 @@ func (m *Exemplar) Reset() { *m = Exemplar{} } func (m *Exemplar) String() string { return proto.CompactTextString(m) } func (*Exemplar) ProtoMessage() {} func (*Exemplar) Descriptor() ([]byte, []int) { - return fileDescriptor_6039342a2ba47b72, []int{8} + return fileDescriptor_d1e5ddb18987a258, []int{9} } func (m *Exemplar) XXX_Unmarshal(b []byte) error { @@ -526,7 +700,7 @@ func (m *Metric) Reset() { *m = Metric{} } func (m *Metric) String() string { return proto.CompactTextString(m) } func (*Metric) ProtoMessage() {} func (*Metric) Descriptor() ([]byte, []int) { - return fileDescriptor_6039342a2ba47b72, []int{9} + return fileDescriptor_d1e5ddb18987a258, []int{10} } func (m *Metric) XXX_Unmarshal(b []byte) error { @@ -610,7 +784,7 @@ func (m *MetricFamily) Reset() { *m = MetricFamily{} } func (m *MetricFamily) String() string { return proto.CompactTextString(m) } func (*MetricFamily) ProtoMessage() {} func (*MetricFamily) Descriptor() ([]byte, []int) { - return fileDescriptor_6039342a2ba47b72, []int{10} + return fileDescriptor_d1e5ddb18987a258, []int{11} } func (m *MetricFamily) XXX_Unmarshal(b []byte) error { @@ -669,55 +843,72 @@ func init() { proto.RegisterType((*Untyped)(nil), "io.prometheus.client.Untyped") proto.RegisterType((*Histogram)(nil), "io.prometheus.client.Histogram") proto.RegisterType((*Bucket)(nil), "io.prometheus.client.Bucket") + proto.RegisterType((*BucketSpan)(nil), "io.prometheus.client.BucketSpan") proto.RegisterType((*Exemplar)(nil), "io.prometheus.client.Exemplar") proto.RegisterType((*Metric)(nil), "io.prometheus.client.Metric") proto.RegisterType((*MetricFamily)(nil), "io.prometheus.client.MetricFamily") } -func init() { proto.RegisterFile("metrics.proto", fileDescriptor_6039342a2ba47b72) } - -var fileDescriptor_6039342a2ba47b72 = []byte{ - // 665 bytes of a gzipped FileDescriptorProto - 0x1f, 0x8b, 0x08, 0x00, 0x00, 0x00, 0x00, 0x00, 0x02, 0xff, 0xac, 0x54, 0xcd, 0x6e, 0xd3, 0x4c, - 0x14, 0xfd, 0xdc, 0x38, 0x3f, 0xbe, 0x69, 0x3f, 0xa2, 0x51, 0x17, 0x56, 0xa1, 0x24, 0x78, 0x55, - 0x58, 0x38, 0xa2, 0x6a, 0x05, 0x2a, 0xb0, 0x68, 0x4b, 0x48, 0x91, 0x48, 0x5b, 0x26, 0xc9, 0xa2, - 0xb0, 0x88, 0x1c, 0x77, 0x70, 0x2c, 0x3c, 0xb1, 0xb1, 0x67, 0x2a, 0xb2, 0x66, 0xc1, 0x16, 0x5e, - 0x81, 0x17, 0x05, 0xcd, 0x8f, 0x6d, 0x2a, 0xb9, 0x95, 0x40, 0xec, 0x66, 0xee, 0x3d, 0xe7, 0xfa, - 0xcc, 0xf8, 0x9c, 0x81, 0x0d, 0x4a, 0x58, 0x1a, 0xfa, 0x99, 0x9b, 0xa4, 0x31, 0x8b, 0xd1, 0x66, - 0x18, 0x8b, 0x15, 0x25, 0x6c, 0x41, 0x78, 0xe6, 0xfa, 0x51, 0x48, 0x96, 0x6c, 0xab, 0x1b, 0xc4, - 0x71, 0x10, 0x91, 0xbe, 0xc4, 0xcc, 0xf9, 0x87, 0x3e, 0x0b, 0x29, 0xc9, 0x98, 0x47, 0x13, 0x45, - 0x73, 0xf6, 0xc1, 0x7a, 0xe3, 0xcd, 0x49, 0x74, 0xee, 0x85, 0x29, 0x42, 0x60, 0x2e, 0x3d, 0x4a, - 0x6c, 0xa3, 0x67, 0xec, 0x58, 0x58, 0xae, 0xd1, 0x26, 0xd4, 0xaf, 0xbc, 0x88, 0x13, 0x7b, 0x4d, - 0x16, 0xd5, 0xc6, 0xd9, 0x86, 0xfa, 0xd0, 0xe3, 0xc1, 0x6f, 0x6d, 0xc1, 0x31, 0xf2, 0xf6, 0x7b, - 0x68, 0x1e, 0xc7, 0x7c, 0xc9, 0x48, 0x5a, 0x0d, 0x40, 0x07, 0xd0, 0x22, 0x9f, 0x09, 0x4d, 0x22, - 0x2f, 0x95, 0x83, 0xdb, 0xbb, 0xf7, 0xdd, 0xaa, 0x03, 0xb8, 0x03, 0x8d, 0xc2, 0x05, 0xde, 0x79, - 0x0e, 0xad, 0xb7, 0xdc, 0x5b, 0xb2, 0x30, 0x22, 0x68, 0x0b, 0x5a, 0x9f, 0xf4, 0x5a, 0x7f, 0xa0, - 0xd8, 0x5f, 0x57, 0x5e, 0x48, 0xfb, 0x6a, 0x40, 0x73, 0xcc, 0x29, 0xf5, 0xd2, 0x15, 0x7a, 0x00, - 0xeb, 0x99, 0x47, 0x93, 0x88, 0xcc, 0x7c, 0xa1, 0x56, 0x4e, 0x30, 0x71, 0x5b, 0xd5, 0xe4, 0x01, - 0xd0, 0x36, 0x80, 0x86, 0x64, 0x9c, 0xea, 0x49, 0x96, 0xaa, 0x8c, 0x39, 0x15, 0xe7, 0x28, 0xbe, - 0x5f, 0xeb, 0xd5, 0x6e, 0x3e, 0x47, 0xae, 0xb8, 0xd4, 0xe7, 0x74, 0xa1, 0x39, 0x5d, 0xb2, 0x55, - 0x42, 0x2e, 0x6f, 0xb8, 0xc5, 0x2f, 0x06, 0x58, 0x27, 0x61, 0xc6, 0xe2, 0x20, 0xf5, 0xe8, 0x3f, - 0x10, 0xbb, 0x07, 0x8d, 0x39, 0xf7, 0x3f, 0x12, 0xa6, 0xa5, 0xde, 0xab, 0x96, 0x7a, 0x24, 0x31, - 0x58, 0x63, 0x9d, 0x6f, 0x06, 0x34, 0x54, 0x09, 0x3d, 0x84, 0x8e, 0xcf, 0x29, 0x8f, 0x3c, 0x16, - 0x5e, 0x5d, 0x97, 0x71, 0xa7, 0xac, 0x2b, 0x29, 0x5d, 0x68, 0xf3, 0x24, 0x21, 0xe9, 0x6c, 0x1e, - 0xf3, 0xe5, 0xa5, 0xd6, 0x02, 0xb2, 0x74, 0x24, 0x2a, 0xd7, 0x1c, 0x50, 0xfb, 0x43, 0x07, 0x7c, - 0x37, 0xa0, 0x95, 0x97, 0xd1, 0x3e, 0xd4, 0x23, 0xe1, 0x60, 0xdb, 0x90, 0x87, 0xea, 0x56, 0x4f, - 0x29, 0x4c, 0x8e, 0x15, 0xba, 0xda, 0x1d, 0xe8, 0x29, 0x58, 0x45, 0x42, 0xb4, 0xac, 0x2d, 0x57, - 0x65, 0xc8, 0xcd, 0x33, 0xe4, 0x4e, 0x72, 0x04, 0x2e, 0xc1, 0xce, 0xcf, 0x35, 0x68, 0x8c, 0x64, - 0x22, 0xff, 0x56, 0xd1, 0x63, 0xa8, 0x07, 0x22, 0x53, 0x3a, 0x10, 0x77, 0xab, 0x69, 0x32, 0x76, - 0x58, 0x21, 0xd1, 0x13, 0x68, 0xfa, 0x2a, 0x67, 0x5a, 0xec, 0x76, 0x35, 0x49, 0x87, 0x11, 0xe7, - 0x68, 0x41, 0xcc, 0x54, 0x08, 0x6c, 0xf3, 0x36, 0xa2, 0x4e, 0x0a, 0xce, 0xd1, 0x82, 0xc8, 0x95, - 0x69, 0xed, 0xfa, 0x6d, 0x44, 0xed, 0x6c, 0x9c, 0xa3, 0xd1, 0x0b, 0xb0, 0x16, 0xb9, 0x97, 0xed, - 0xa6, 0xa4, 0xde, 0x70, 0x31, 0x85, 0xe5, 0x71, 0xc9, 0x10, 0xee, 0x2f, 0xee, 0x7a, 0x46, 0x33, - 0xbb, 0xd1, 0x33, 0x76, 0x6a, 0xb8, 0x5d, 0xd4, 0x46, 0x99, 0xf3, 0xc3, 0x80, 0x75, 0xf5, 0x07, - 0x5e, 0x79, 0x34, 0x8c, 0x56, 0x95, 0xcf, 0x19, 0x02, 0x73, 0x41, 0xa2, 0x44, 0xbf, 0x66, 0x72, - 0x8d, 0xf6, 0xc0, 0x14, 0x1a, 0xe5, 0x15, 0xfe, 0xbf, 0xdb, 0xab, 0x56, 0xa5, 0x26, 0x4f, 0x56, - 0x09, 0xc1, 0x12, 0x2d, 0xd2, 0xa4, 0x5e, 0x60, 0xdb, 0xbc, 0x2d, 0x4d, 0x8a, 0x87, 0x35, 0xf6, - 0xd1, 0x08, 0xa0, 0x9c, 0x84, 0xda, 0xd0, 0x3c, 0x3e, 0x9b, 0x9e, 0x4e, 0x06, 0xb8, 0xf3, 0x1f, - 0xb2, 0xa0, 0x3e, 0x3c, 0x9c, 0x0e, 0x07, 0x1d, 0x43, 0xd4, 0xc7, 0xd3, 0xd1, 0xe8, 0x10, 0x5f, - 0x74, 0xd6, 0xc4, 0x66, 0x7a, 0x3a, 0xb9, 0x38, 0x1f, 0xbc, 0xec, 0xd4, 0xd0, 0x06, 0x58, 0x27, - 0xaf, 0xc7, 0x93, 0xb3, 0x21, 0x3e, 0x1c, 0x75, 0xcc, 0x23, 0x0c, 0x95, 0xef, 0xfe, 0xbb, 0x83, - 0x20, 0x64, 0x0b, 0x3e, 0x77, 0xfd, 0x98, 0xf6, 0xcb, 0x6e, 0x5f, 0x75, 0x67, 0x34, 0xbe, 0x24, - 0x51, 0x3f, 0x88, 0x9f, 0x85, 0xf1, 0xac, 0xec, 0xce, 0x54, 0xf7, 0x57, 0x00, 0x00, 0x00, 0xff, - 0xff, 0xd0, 0x84, 0x91, 0x73, 0x59, 0x06, 0x00, 0x00, +func init() { + proto.RegisterFile("io/prometheus/client/metrics.proto", fileDescriptor_d1e5ddb18987a258) +} + +var fileDescriptor_d1e5ddb18987a258 = []byte{ + // 896 bytes of a gzipped FileDescriptorProto + 0x1f, 0x8b, 0x08, 0x00, 0x00, 0x00, 0x00, 0x00, 0x02, 0xff, 0x9c, 0x56, 0xdd, 0x8e, 0xdb, 0x44, + 0x18, 0xc5, 0x9b, 0x5f, 0x7f, 0xd9, 0x6c, 0xd3, 0x61, 0x55, 0x59, 0x0b, 0xcb, 0x06, 0x4b, 0x48, + 0x0b, 0x42, 0x8e, 0x40, 0x5b, 0x81, 0x0a, 0x5c, 0xec, 0xb6, 0xe9, 0x16, 0x89, 0xb4, 0x65, 0x92, + 0x5c, 0x14, 0x2e, 0xac, 0x49, 0x32, 0xeb, 0x58, 0x78, 0x3c, 0xc6, 0x1e, 0x57, 0x2c, 0x2f, 0xc0, + 0x35, 0xaf, 0xc0, 0xc3, 0xf0, 0x22, 0x3c, 0x08, 0x68, 0xfe, 0xec, 0xdd, 0xe2, 0x94, 0xd2, 0x3b, + 0x7f, 0x67, 0xce, 0xf7, 0xcd, 0x39, 0xe3, 0xc9, 0x71, 0xc0, 0x8f, 0xf9, 0x24, 0xcb, 0x39, 0xa3, + 0x62, 0x4b, 0xcb, 0x62, 0xb2, 0x4e, 0x62, 0x9a, 0x8a, 0x09, 0xa3, 0x22, 0x8f, 0xd7, 0x45, 0x90, + 0xe5, 0x5c, 0x70, 0x74, 0x18, 0xf3, 0xa0, 0xe6, 0x04, 0x9a, 0x73, 0x74, 0x12, 0x71, 0x1e, 0x25, + 0x74, 0xa2, 0x38, 0xab, 0xf2, 0x6a, 0x22, 0x62, 0x46, 0x0b, 0x41, 0x58, 0xa6, 0xdb, 0xfc, 0xfb, + 0xe0, 0x7e, 0x47, 0x56, 0x34, 0x79, 0x4e, 0xe2, 0x1c, 0x21, 0x68, 0xa7, 0x84, 0x51, 0xcf, 0x19, + 0x3b, 0xa7, 0x2e, 0x56, 0xcf, 0xe8, 0x10, 0x3a, 0x2f, 0x49, 0x52, 0x52, 0x6f, 0x4f, 0x81, 0xba, + 0xf0, 0x8f, 0xa1, 0x73, 0x49, 0xca, 0xe8, 0xc6, 0xb2, 0xec, 0x71, 0xec, 0xf2, 0x8f, 0xd0, 0x7b, + 0xc8, 0xcb, 0x54, 0xd0, 0xbc, 0x99, 0x80, 0x1e, 0x40, 0x9f, 0xfe, 0x42, 0x59, 0x96, 0x90, 0x5c, + 0x0d, 0x1e, 0x7c, 0xfe, 0x41, 0xd0, 0x64, 0x20, 0x98, 0x1a, 0x16, 0xae, 0xf8, 0xfe, 0xd7, 0xd0, + 0xff, 0xbe, 0x24, 0xa9, 0x88, 0x13, 0x8a, 0x8e, 0xa0, 0xff, 0xb3, 0x79, 0x36, 0x1b, 0x54, 0xf5, + 0x6d, 0xe5, 0x95, 0xb4, 0xdf, 0x1c, 0xe8, 0xcd, 0x4b, 0xc6, 0x48, 0x7e, 0x8d, 0x3e, 0x84, 0xfd, + 0x82, 0xb0, 0x2c, 0xa1, 0xe1, 0x5a, 0xaa, 0x55, 0x13, 0xda, 0x78, 0xa0, 0x31, 0x65, 0x00, 0x1d, + 0x03, 0x18, 0x4a, 0x51, 0x32, 0x33, 0xc9, 0xd5, 0xc8, 0xbc, 0x64, 0xd2, 0x47, 0xb5, 0x7f, 0x6b, + 0xdc, 0xda, 0xed, 0xc3, 0x2a, 0xae, 0xf5, 0xf9, 0x27, 0xd0, 0x5b, 0xa6, 0xe2, 0x3a, 0xa3, 0x9b, + 0x1d, 0xa7, 0xf8, 0x57, 0x1b, 0xdc, 0x27, 0x71, 0x21, 0x78, 0x94, 0x13, 0xf6, 0x26, 0x62, 0x3f, + 0x05, 0x74, 0x93, 0x12, 0x5e, 0x25, 0x9c, 0x08, 0xaf, 0xad, 0x66, 0x8e, 0x6e, 0x10, 0x1f, 0x4b, + 0xfc, 0xbf, 0xac, 0x9d, 0x41, 0x77, 0x55, 0xae, 0x7f, 0xa2, 0xc2, 0x18, 0x7b, 0xbf, 0xd9, 0xd8, + 0x85, 0xe2, 0x60, 0xc3, 0x45, 0xf7, 0xa0, 0x5b, 0xac, 0xb7, 0x94, 0x11, 0xaf, 0x33, 0x76, 0x4e, + 0xef, 0x62, 0x53, 0xa1, 0x8f, 0xe0, 0xe0, 0x57, 0x9a, 0xf3, 0x50, 0x6c, 0x73, 0x5a, 0x6c, 0x79, + 0xb2, 0xf1, 0xba, 0x6a, 0xc3, 0xa1, 0x44, 0x17, 0x16, 0x94, 0x9a, 0x14, 0x4d, 0x5b, 0xec, 0x29, + 0x8b, 0xae, 0x44, 0xb4, 0xc1, 0x53, 0x18, 0xd5, 0xcb, 0xc6, 0x5e, 0x5f, 0xcd, 0x39, 0xa8, 0x48, + 0xda, 0xdc, 0x14, 0x86, 0x29, 0x8d, 0x88, 0x88, 0x5f, 0xd2, 0xb0, 0xc8, 0x48, 0xea, 0xb9, 0xca, + 0xc4, 0xf8, 0x75, 0x26, 0xe6, 0x19, 0x49, 0xf1, 0xbe, 0x6d, 0x93, 0x95, 0x94, 0x5d, 0x8d, 0xd9, + 0xd0, 0x44, 0x10, 0x0f, 0xc6, 0xad, 0x53, 0x84, 0xab, 0xe1, 0x8f, 0x24, 0x78, 0x8b, 0xa6, 0xa5, + 0x0f, 0xc6, 0x2d, 0xe9, 0xce, 0xa2, 0x5a, 0xfe, 0x14, 0x86, 0x19, 0x2f, 0xe2, 0x5a, 0xd4, 0xfe, + 0x9b, 0x8a, 0xb2, 0x6d, 0x56, 0x54, 0x35, 0x46, 0x8b, 0x1a, 0x6a, 0x51, 0x16, 0xad, 0x44, 0x55, + 0x34, 0x2d, 0xea, 0x40, 0x8b, 0xb2, 0xa8, 0x12, 0xe5, 0xff, 0xe9, 0x40, 0x57, 0x6f, 0x85, 0x3e, + 0x86, 0xd1, 0xba, 0x64, 0x65, 0x72, 0xd3, 0x88, 0xbe, 0x66, 0x77, 0x6a, 0x5c, 0x5b, 0x39, 0x83, + 0x7b, 0xaf, 0x52, 0x6f, 0x5d, 0xb7, 0xc3, 0x57, 0x1a, 0xf4, 0x5b, 0x39, 0x81, 0x41, 0x99, 0x65, + 0x34, 0x0f, 0x57, 0xbc, 0x4c, 0x37, 0xe6, 0xce, 0x81, 0x82, 0x2e, 0x24, 0x72, 0x2b, 0x17, 0x5a, + 0xff, 0x3b, 0x17, 0xa0, 0x3e, 0x32, 0x79, 0x11, 0xf9, 0xd5, 0x55, 0x41, 0xb5, 0x83, 0xbb, 0xd8, + 0x54, 0x12, 0x4f, 0x68, 0x1a, 0x89, 0xad, 0xda, 0x7d, 0x88, 0x4d, 0xe5, 0xff, 0xee, 0x40, 0xdf, + 0x0e, 0x45, 0xf7, 0xa1, 0x93, 0xc8, 0x54, 0xf4, 0x1c, 0xf5, 0x82, 0x4e, 0x9a, 0x35, 0x54, 0xc1, + 0x89, 0x35, 0xbb, 0x39, 0x71, 0xd0, 0x97, 0xe0, 0x56, 0xa9, 0x6b, 0x4c, 0x1d, 0x05, 0x3a, 0x97, + 0x03, 0x9b, 0xcb, 0xc1, 0xc2, 0x32, 0x70, 0x4d, 0xf6, 0xff, 0xde, 0x83, 0xee, 0x4c, 0xa5, 0xfc, + 0xdb, 0x2a, 0xfa, 0x0c, 0x3a, 0x91, 0xcc, 0x69, 0x13, 0xb2, 0xef, 0x35, 0xb7, 0xa9, 0x28, 0xc7, + 0x9a, 0x89, 0xbe, 0x80, 0xde, 0x5a, 0x67, 0xb7, 0x11, 0x7b, 0xdc, 0xdc, 0x64, 0x02, 0x1e, 0x5b, + 0xb6, 0x6c, 0x2c, 0x74, 0xb0, 0xaa, 0x3b, 0xb0, 0xb3, 0xd1, 0xa4, 0x2f, 0xb6, 0x6c, 0xd9, 0x58, + 0xea, 0x20, 0x54, 0xa1, 0xb1, 0xb3, 0xd1, 0xa4, 0x25, 0xb6, 0x6c, 0xf4, 0x0d, 0xb8, 0x5b, 0x9b, + 0x8f, 0x2a, 0x2c, 0x76, 0x1e, 0x4c, 0x15, 0xa3, 0xb8, 0xee, 0x90, 0x89, 0x5a, 0x9d, 0x75, 0xc8, + 0x0a, 0x95, 0x48, 0x2d, 0x3c, 0xa8, 0xb0, 0x59, 0xe1, 0xff, 0xe1, 0xc0, 0xbe, 0x7e, 0x03, 0x8f, + 0x09, 0x8b, 0x93, 0xeb, 0xc6, 0x4f, 0x24, 0x82, 0xf6, 0x96, 0x26, 0x99, 0xf9, 0x42, 0xaa, 0x67, + 0x74, 0x06, 0x6d, 0xa9, 0x51, 0x1d, 0xe1, 0xc1, 0xae, 0x5f, 0xb8, 0x9e, 0xbc, 0xb8, 0xce, 0x28, + 0x56, 0x6c, 0x99, 0xb9, 0xfa, 0xab, 0xee, 0xb5, 0x5f, 0x97, 0xb9, 0xba, 0x0f, 0x1b, 0xee, 0x27, + 0x2b, 0x80, 0x7a, 0x12, 0x1a, 0x40, 0xef, 0xe1, 0xb3, 0xe5, 0xd3, 0xc5, 0x14, 0x8f, 0xde, 0x41, + 0x2e, 0x74, 0x2e, 0xcf, 0x97, 0x97, 0xd3, 0x91, 0x23, 0xf1, 0xf9, 0x72, 0x36, 0x3b, 0xc7, 0x2f, + 0x46, 0x7b, 0xb2, 0x58, 0x3e, 0x5d, 0xbc, 0x78, 0x3e, 0x7d, 0x34, 0x6a, 0xa1, 0x21, 0xb8, 0x4f, + 0xbe, 0x9d, 0x2f, 0x9e, 0x5d, 0xe2, 0xf3, 0xd9, 0xa8, 0x8d, 0xde, 0x85, 0x3b, 0xaa, 0x27, 0xac, + 0xc1, 0xce, 0x05, 0x86, 0xc6, 0x3f, 0x18, 0x3f, 0x3c, 0x88, 0x62, 0xb1, 0x2d, 0x57, 0xc1, 0x9a, + 0xb3, 0x7f, 0xff, 0x45, 0x09, 0x19, 0xdf, 0xd0, 0x64, 0x12, 0xf1, 0xaf, 0x62, 0x1e, 0xd6, 0xab, + 0xa1, 0x5e, 0xfd, 0x27, 0x00, 0x00, 0xff, 0xff, 0x16, 0x77, 0x81, 0x98, 0xd7, 0x08, 0x00, 0x00, } diff --git a/vendor/github.com/rcrowley/go-metrics/.travis.yml b/vendor/github.com/rcrowley/go-metrics/.travis.yml index 409a5b63..ce9afeae 100644 --- a/vendor/github.com/rcrowley/go-metrics/.travis.yml +++ b/vendor/github.com/rcrowley/go-metrics/.travis.yml @@ -13,6 +13,7 @@ go: - "1.12" - "1.13" - "1.14" + - "1.15" script: - ./validate.sh diff --git a/vendor/github.com/stretchr/testify/assert/assertion_compare.go b/vendor/github.com/stretchr/testify/assert/assertion_compare.go index 41649d26..95d8e59d 100644 --- a/vendor/github.com/stretchr/testify/assert/assertion_compare.go +++ b/vendor/github.com/stretchr/testify/assert/assertion_compare.go @@ -1,8 +1,10 @@ package assert import ( + "bytes" "fmt" "reflect" + "time" ) type CompareType int @@ -30,6 +32,9 @@ var ( float64Type = reflect.TypeOf(float64(1)) stringType = reflect.TypeOf("") + + timeType = reflect.TypeOf(time.Time{}) + bytesType = reflect.TypeOf([]byte{}) ) func compare(obj1, obj2 interface{}, kind reflect.Kind) (CompareType, bool) { @@ -299,6 +304,47 @@ func compare(obj1, obj2 interface{}, kind reflect.Kind) (CompareType, bool) { return compareLess, true } } + // Check for known struct types we can check for compare results. + case reflect.Struct: + { + // All structs enter here. We're not interested in most types. + if !canConvert(obj1Value, timeType) { + break + } + + // time.Time can compared! + timeObj1, ok := obj1.(time.Time) + if !ok { + timeObj1 = obj1Value.Convert(timeType).Interface().(time.Time) + } + + timeObj2, ok := obj2.(time.Time) + if !ok { + timeObj2 = obj2Value.Convert(timeType).Interface().(time.Time) + } + + return compare(timeObj1.UnixNano(), timeObj2.UnixNano(), reflect.Int64) + } + case reflect.Slice: + { + // We only care about the []byte type. + if !canConvert(obj1Value, bytesType) { + break + } + + // []byte can be compared! + bytesObj1, ok := obj1.([]byte) + if !ok { + bytesObj1 = obj1Value.Convert(bytesType).Interface().([]byte) + + } + bytesObj2, ok := obj2.([]byte) + if !ok { + bytesObj2 = obj2Value.Convert(bytesType).Interface().([]byte) + } + + return CompareType(bytes.Compare(bytesObj1, bytesObj2)), true + } } return compareEqual, false @@ -310,7 +356,10 @@ func compare(obj1, obj2 interface{}, kind reflect.Kind) (CompareType, bool) { // assert.Greater(t, float64(2), float64(1)) // assert.Greater(t, "b", "a") func Greater(t TestingT, e1 interface{}, e2 interface{}, msgAndArgs ...interface{}) bool { - return compareTwoValues(t, e1, e2, []CompareType{compareGreater}, "\"%v\" is not greater than \"%v\"", msgAndArgs) + if h, ok := t.(tHelper); ok { + h.Helper() + } + return compareTwoValues(t, e1, e2, []CompareType{compareGreater}, "\"%v\" is not greater than \"%v\"", msgAndArgs...) } // GreaterOrEqual asserts that the first element is greater than or equal to the second @@ -320,7 +369,10 @@ func Greater(t TestingT, e1 interface{}, e2 interface{}, msgAndArgs ...interface // assert.GreaterOrEqual(t, "b", "a") // assert.GreaterOrEqual(t, "b", "b") func GreaterOrEqual(t TestingT, e1 interface{}, e2 interface{}, msgAndArgs ...interface{}) bool { - return compareTwoValues(t, e1, e2, []CompareType{compareGreater, compareEqual}, "\"%v\" is not greater than or equal to \"%v\"", msgAndArgs) + if h, ok := t.(tHelper); ok { + h.Helper() + } + return compareTwoValues(t, e1, e2, []CompareType{compareGreater, compareEqual}, "\"%v\" is not greater than or equal to \"%v\"", msgAndArgs...) } // Less asserts that the first element is less than the second @@ -329,7 +381,10 @@ func GreaterOrEqual(t TestingT, e1 interface{}, e2 interface{}, msgAndArgs ...in // assert.Less(t, float64(1), float64(2)) // assert.Less(t, "a", "b") func Less(t TestingT, e1 interface{}, e2 interface{}, msgAndArgs ...interface{}) bool { - return compareTwoValues(t, e1, e2, []CompareType{compareLess}, "\"%v\" is not less than \"%v\"", msgAndArgs) + if h, ok := t.(tHelper); ok { + h.Helper() + } + return compareTwoValues(t, e1, e2, []CompareType{compareLess}, "\"%v\" is not less than \"%v\"", msgAndArgs...) } // LessOrEqual asserts that the first element is less than or equal to the second @@ -339,7 +394,10 @@ func Less(t TestingT, e1 interface{}, e2 interface{}, msgAndArgs ...interface{}) // assert.LessOrEqual(t, "a", "b") // assert.LessOrEqual(t, "b", "b") func LessOrEqual(t TestingT, e1 interface{}, e2 interface{}, msgAndArgs ...interface{}) bool { - return compareTwoValues(t, e1, e2, []CompareType{compareLess, compareEqual}, "\"%v\" is not less than or equal to \"%v\"", msgAndArgs) + if h, ok := t.(tHelper); ok { + h.Helper() + } + return compareTwoValues(t, e1, e2, []CompareType{compareLess, compareEqual}, "\"%v\" is not less than or equal to \"%v\"", msgAndArgs...) } // Positive asserts that the specified element is positive @@ -347,8 +405,11 @@ func LessOrEqual(t TestingT, e1 interface{}, e2 interface{}, msgAndArgs ...inter // assert.Positive(t, 1) // assert.Positive(t, 1.23) func Positive(t TestingT, e interface{}, msgAndArgs ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } zero := reflect.Zero(reflect.TypeOf(e)) - return compareTwoValues(t, e, zero.Interface(), []CompareType{compareGreater}, "\"%v\" is not positive", msgAndArgs) + return compareTwoValues(t, e, zero.Interface(), []CompareType{compareGreater}, "\"%v\" is not positive", msgAndArgs...) } // Negative asserts that the specified element is negative @@ -356,8 +417,11 @@ func Positive(t TestingT, e interface{}, msgAndArgs ...interface{}) bool { // assert.Negative(t, -1) // assert.Negative(t, -1.23) func Negative(t TestingT, e interface{}, msgAndArgs ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } zero := reflect.Zero(reflect.TypeOf(e)) - return compareTwoValues(t, e, zero.Interface(), []CompareType{compareLess}, "\"%v\" is not negative", msgAndArgs) + return compareTwoValues(t, e, zero.Interface(), []CompareType{compareLess}, "\"%v\" is not negative", msgAndArgs...) } func compareTwoValues(t TestingT, e1 interface{}, e2 interface{}, allowedComparesResults []CompareType, failMessage string, msgAndArgs ...interface{}) bool { diff --git a/vendor/github.com/stretchr/testify/assert/assertion_compare_can_convert.go b/vendor/github.com/stretchr/testify/assert/assertion_compare_can_convert.go new file mode 100644 index 00000000..da867903 --- /dev/null +++ b/vendor/github.com/stretchr/testify/assert/assertion_compare_can_convert.go @@ -0,0 +1,16 @@ +//go:build go1.17 +// +build go1.17 + +// TODO: once support for Go 1.16 is dropped, this file can be +// merged/removed with assertion_compare_go1.17_test.go and +// assertion_compare_legacy.go + +package assert + +import "reflect" + +// Wrapper around reflect.Value.CanConvert, for compatibility +// reasons. +func canConvert(value reflect.Value, to reflect.Type) bool { + return value.CanConvert(to) +} diff --git a/vendor/github.com/stretchr/testify/assert/assertion_compare_legacy.go b/vendor/github.com/stretchr/testify/assert/assertion_compare_legacy.go new file mode 100644 index 00000000..1701af2a --- /dev/null +++ b/vendor/github.com/stretchr/testify/assert/assertion_compare_legacy.go @@ -0,0 +1,16 @@ +//go:build !go1.17 +// +build !go1.17 + +// TODO: once support for Go 1.16 is dropped, this file can be +// merged/removed with assertion_compare_go1.17_test.go and +// assertion_compare_can_convert.go + +package assert + +import "reflect" + +// Older versions of Go does not have the reflect.Value.CanConvert +// method. +func canConvert(value reflect.Value, to reflect.Type) bool { + return false +} diff --git a/vendor/github.com/stretchr/testify/assert/assertion_format.go b/vendor/github.com/stretchr/testify/assert/assertion_format.go index 4dfd1229..7880b8f9 100644 --- a/vendor/github.com/stretchr/testify/assert/assertion_format.go +++ b/vendor/github.com/stretchr/testify/assert/assertion_format.go @@ -123,6 +123,18 @@ func ErrorAsf(t TestingT, err error, target interface{}, msg string, args ...int return ErrorAs(t, err, target, append([]interface{}{msg}, args...)...) } +// ErrorContainsf asserts that a function returned an error (i.e. not `nil`) +// and that the error contains the specified substring. +// +// actualObj, err := SomeFunction() +// assert.ErrorContainsf(t, err, expectedErrorSubString, "error message %s", "formatted") +func ErrorContainsf(t TestingT, theError error, contains string, msg string, args ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + return ErrorContains(t, theError, contains, append([]interface{}{msg}, args...)...) +} + // ErrorIsf asserts that at least one of the errors in err's chain matches target. // This is a wrapper for errors.Is. func ErrorIsf(t TestingT, err error, target error, msg string, args ...interface{}) bool { @@ -724,6 +736,16 @@ func WithinDurationf(t TestingT, expected time.Time, actual time.Time, delta tim return WithinDuration(t, expected, actual, delta, append([]interface{}{msg}, args...)...) } +// WithinRangef asserts that a time is within a time range (inclusive). +// +// assert.WithinRangef(t, time.Now(), time.Now().Add(-time.Second), time.Now().Add(time.Second), "error message %s", "formatted") +func WithinRangef(t TestingT, actual time.Time, start time.Time, end time.Time, msg string, args ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + return WithinRange(t, actual, start, end, append([]interface{}{msg}, args...)...) +} + // YAMLEqf asserts that two YAML strings are equivalent. func YAMLEqf(t TestingT, expected string, actual string, msg string, args ...interface{}) bool { if h, ok := t.(tHelper); ok { diff --git a/vendor/github.com/stretchr/testify/assert/assertion_forward.go b/vendor/github.com/stretchr/testify/assert/assertion_forward.go index 25337a6f..339515b8 100644 --- a/vendor/github.com/stretchr/testify/assert/assertion_forward.go +++ b/vendor/github.com/stretchr/testify/assert/assertion_forward.go @@ -222,6 +222,30 @@ func (a *Assertions) ErrorAsf(err error, target interface{}, msg string, args .. return ErrorAsf(a.t, err, target, msg, args...) } +// ErrorContains asserts that a function returned an error (i.e. not `nil`) +// and that the error contains the specified substring. +// +// actualObj, err := SomeFunction() +// a.ErrorContains(err, expectedErrorSubString) +func (a *Assertions) ErrorContains(theError error, contains string, msgAndArgs ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return ErrorContains(a.t, theError, contains, msgAndArgs...) +} + +// ErrorContainsf asserts that a function returned an error (i.e. not `nil`) +// and that the error contains the specified substring. +// +// actualObj, err := SomeFunction() +// a.ErrorContainsf(err, expectedErrorSubString, "error message %s", "formatted") +func (a *Assertions) ErrorContainsf(theError error, contains string, msg string, args ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return ErrorContainsf(a.t, theError, contains, msg, args...) +} + // ErrorIs asserts that at least one of the errors in err's chain matches target. // This is a wrapper for errors.Is. func (a *Assertions) ErrorIs(err error, target error, msgAndArgs ...interface{}) bool { @@ -1437,6 +1461,26 @@ func (a *Assertions) WithinDurationf(expected time.Time, actual time.Time, delta return WithinDurationf(a.t, expected, actual, delta, msg, args...) } +// WithinRange asserts that a time is within a time range (inclusive). +// +// a.WithinRange(time.Now(), time.Now().Add(-time.Second), time.Now().Add(time.Second)) +func (a *Assertions) WithinRange(actual time.Time, start time.Time, end time.Time, msgAndArgs ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return WithinRange(a.t, actual, start, end, msgAndArgs...) +} + +// WithinRangef asserts that a time is within a time range (inclusive). +// +// a.WithinRangef(time.Now(), time.Now().Add(-time.Second), time.Now().Add(time.Second), "error message %s", "formatted") +func (a *Assertions) WithinRangef(actual time.Time, start time.Time, end time.Time, msg string, args ...interface{}) bool { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + return WithinRangef(a.t, actual, start, end, msg, args...) +} + // YAMLEq asserts that two YAML strings are equivalent. func (a *Assertions) YAMLEq(expected string, actual string, msgAndArgs ...interface{}) bool { if h, ok := a.t.(tHelper); ok { diff --git a/vendor/github.com/stretchr/testify/assert/assertion_order.go b/vendor/github.com/stretchr/testify/assert/assertion_order.go index 1c3b4718..75944878 100644 --- a/vendor/github.com/stretchr/testify/assert/assertion_order.go +++ b/vendor/github.com/stretchr/testify/assert/assertion_order.go @@ -50,7 +50,7 @@ func isOrdered(t TestingT, object interface{}, allowedComparesResults []CompareT // assert.IsIncreasing(t, []float{1, 2}) // assert.IsIncreasing(t, []string{"a", "b"}) func IsIncreasing(t TestingT, object interface{}, msgAndArgs ...interface{}) bool { - return isOrdered(t, object, []CompareType{compareLess}, "\"%v\" is not less than \"%v\"", msgAndArgs) + return isOrdered(t, object, []CompareType{compareLess}, "\"%v\" is not less than \"%v\"", msgAndArgs...) } // IsNonIncreasing asserts that the collection is not increasing @@ -59,7 +59,7 @@ func IsIncreasing(t TestingT, object interface{}, msgAndArgs ...interface{}) boo // assert.IsNonIncreasing(t, []float{2, 1}) // assert.IsNonIncreasing(t, []string{"b", "a"}) func IsNonIncreasing(t TestingT, object interface{}, msgAndArgs ...interface{}) bool { - return isOrdered(t, object, []CompareType{compareEqual, compareGreater}, "\"%v\" is not greater than or equal to \"%v\"", msgAndArgs) + return isOrdered(t, object, []CompareType{compareEqual, compareGreater}, "\"%v\" is not greater than or equal to \"%v\"", msgAndArgs...) } // IsDecreasing asserts that the collection is decreasing @@ -68,7 +68,7 @@ func IsNonIncreasing(t TestingT, object interface{}, msgAndArgs ...interface{}) // assert.IsDecreasing(t, []float{2, 1}) // assert.IsDecreasing(t, []string{"b", "a"}) func IsDecreasing(t TestingT, object interface{}, msgAndArgs ...interface{}) bool { - return isOrdered(t, object, []CompareType{compareGreater}, "\"%v\" is not greater than \"%v\"", msgAndArgs) + return isOrdered(t, object, []CompareType{compareGreater}, "\"%v\" is not greater than \"%v\"", msgAndArgs...) } // IsNonDecreasing asserts that the collection is not decreasing @@ -77,5 +77,5 @@ func IsDecreasing(t TestingT, object interface{}, msgAndArgs ...interface{}) boo // assert.IsNonDecreasing(t, []float{1, 2}) // assert.IsNonDecreasing(t, []string{"a", "b"}) func IsNonDecreasing(t TestingT, object interface{}, msgAndArgs ...interface{}) bool { - return isOrdered(t, object, []CompareType{compareLess, compareEqual}, "\"%v\" is not less than or equal to \"%v\"", msgAndArgs) + return isOrdered(t, object, []CompareType{compareLess, compareEqual}, "\"%v\" is not less than or equal to \"%v\"", msgAndArgs...) } diff --git a/vendor/github.com/stretchr/testify/assert/assertions.go b/vendor/github.com/stretchr/testify/assert/assertions.go index bcac4401..fa1245b1 100644 --- a/vendor/github.com/stretchr/testify/assert/assertions.go +++ b/vendor/github.com/stretchr/testify/assert/assertions.go @@ -8,6 +8,7 @@ import ( "fmt" "math" "os" + "path/filepath" "reflect" "regexp" "runtime" @@ -144,7 +145,8 @@ func CallerInfo() []string { if len(parts) > 1 { dir := parts[len(parts)-2] if (dir != "assert" && dir != "mock" && dir != "require") || file == "mock_test.go" { - callers = append(callers, fmt.Sprintf("%s:%d", file, line)) + path, _ := filepath.Abs(file) + callers = append(callers, fmt.Sprintf("%s:%d", path, line)) } } @@ -563,16 +565,17 @@ func isEmpty(object interface{}) bool { switch objValue.Kind() { // collection types are empty when they have no element - case reflect.Array, reflect.Chan, reflect.Map, reflect.Slice: + case reflect.Chan, reflect.Map, reflect.Slice: return objValue.Len() == 0 - // pointers are empty if nil or if the value they point to is empty + // pointers are empty if nil or if the value they point to is empty case reflect.Ptr: if objValue.IsNil() { return true } deref := objValue.Elem().Interface() return isEmpty(deref) - // for all other types, compare against the zero value + // for all other types, compare against the zero value + // array types are empty when they match their zero-initialized state default: zero := reflect.Zero(objValue.Type()) return reflect.DeepEqual(object, zero.Interface()) @@ -718,10 +721,14 @@ func NotEqualValues(t TestingT, expected, actual interface{}, msgAndArgs ...inte // return (false, false) if impossible. // return (true, false) if element was not found. // return (true, true) if element was found. -func includeElement(list interface{}, element interface{}) (ok, found bool) { +func containsElement(list interface{}, element interface{}) (ok, found bool) { listValue := reflect.ValueOf(list) - listKind := reflect.TypeOf(list).Kind() + listType := reflect.TypeOf(list) + if listType == nil { + return false, false + } + listKind := listType.Kind() defer func() { if e := recover(); e != nil { ok = false @@ -764,7 +771,7 @@ func Contains(t TestingT, s, contains interface{}, msgAndArgs ...interface{}) bo h.Helper() } - ok, found := includeElement(s, contains) + ok, found := containsElement(s, contains) if !ok { return Fail(t, fmt.Sprintf("%#v could not be applied builtin len()", s), msgAndArgs...) } @@ -787,7 +794,7 @@ func NotContains(t TestingT, s, contains interface{}, msgAndArgs ...interface{}) h.Helper() } - ok, found := includeElement(s, contains) + ok, found := containsElement(s, contains) if !ok { return Fail(t, fmt.Sprintf("\"%s\" could not be applied builtin len()", s), msgAndArgs...) } @@ -811,7 +818,6 @@ func Subset(t TestingT, list, subset interface{}, msgAndArgs ...interface{}) (ok return true // we consider nil to be equal to the nil set } - subsetValue := reflect.ValueOf(subset) defer func() { if e := recover(); e != nil { ok = false @@ -821,17 +827,35 @@ func Subset(t TestingT, list, subset interface{}, msgAndArgs ...interface{}) (ok listKind := reflect.TypeOf(list).Kind() subsetKind := reflect.TypeOf(subset).Kind() - if listKind != reflect.Array && listKind != reflect.Slice { + if listKind != reflect.Array && listKind != reflect.Slice && listKind != reflect.Map { return Fail(t, fmt.Sprintf("%q has an unsupported type %s", list, listKind), msgAndArgs...) } - if subsetKind != reflect.Array && subsetKind != reflect.Slice { + if subsetKind != reflect.Array && subsetKind != reflect.Slice && listKind != reflect.Map { return Fail(t, fmt.Sprintf("%q has an unsupported type %s", subset, subsetKind), msgAndArgs...) } + subsetValue := reflect.ValueOf(subset) + if subsetKind == reflect.Map && listKind == reflect.Map { + listValue := reflect.ValueOf(list) + subsetKeys := subsetValue.MapKeys() + + for i := 0; i < len(subsetKeys); i++ { + subsetKey := subsetKeys[i] + subsetElement := subsetValue.MapIndex(subsetKey).Interface() + listElement := listValue.MapIndex(subsetKey).Interface() + + if !ObjectsAreEqual(subsetElement, listElement) { + return Fail(t, fmt.Sprintf("\"%s\" does not contain \"%s\"", list, subsetElement), msgAndArgs...) + } + } + + return true + } + for i := 0; i < subsetValue.Len(); i++ { element := subsetValue.Index(i).Interface() - ok, found := includeElement(list, element) + ok, found := containsElement(list, element) if !ok { return Fail(t, fmt.Sprintf("\"%s\" could not be applied builtin len()", list), msgAndArgs...) } @@ -852,10 +876,9 @@ func NotSubset(t TestingT, list, subset interface{}, msgAndArgs ...interface{}) h.Helper() } if subset == nil { - return Fail(t, fmt.Sprintf("nil is the empty set which is a subset of every set"), msgAndArgs...) + return Fail(t, "nil is the empty set which is a subset of every set", msgAndArgs...) } - subsetValue := reflect.ValueOf(subset) defer func() { if e := recover(); e != nil { ok = false @@ -865,17 +888,35 @@ func NotSubset(t TestingT, list, subset interface{}, msgAndArgs ...interface{}) listKind := reflect.TypeOf(list).Kind() subsetKind := reflect.TypeOf(subset).Kind() - if listKind != reflect.Array && listKind != reflect.Slice { + if listKind != reflect.Array && listKind != reflect.Slice && listKind != reflect.Map { return Fail(t, fmt.Sprintf("%q has an unsupported type %s", list, listKind), msgAndArgs...) } - if subsetKind != reflect.Array && subsetKind != reflect.Slice { + if subsetKind != reflect.Array && subsetKind != reflect.Slice && listKind != reflect.Map { return Fail(t, fmt.Sprintf("%q has an unsupported type %s", subset, subsetKind), msgAndArgs...) } + subsetValue := reflect.ValueOf(subset) + if subsetKind == reflect.Map && listKind == reflect.Map { + listValue := reflect.ValueOf(list) + subsetKeys := subsetValue.MapKeys() + + for i := 0; i < len(subsetKeys); i++ { + subsetKey := subsetKeys[i] + subsetElement := subsetValue.MapIndex(subsetKey).Interface() + listElement := listValue.MapIndex(subsetKey).Interface() + + if !ObjectsAreEqual(subsetElement, listElement) { + return true + } + } + + return Fail(t, fmt.Sprintf("%q is a subset of %q", subset, list), msgAndArgs...) + } + for i := 0; i < subsetValue.Len(); i++ { element := subsetValue.Index(i).Interface() - ok, found := includeElement(list, element) + ok, found := containsElement(list, element) if !ok { return Fail(t, fmt.Sprintf("\"%s\" could not be applied builtin len()", list), msgAndArgs...) } @@ -1000,27 +1041,21 @@ func Condition(t TestingT, comp Comparison, msgAndArgs ...interface{}) bool { type PanicTestFunc func() // didPanic returns true if the function passed to it panics. Otherwise, it returns false. -func didPanic(f PanicTestFunc) (bool, interface{}, string) { - - didPanic := false - var message interface{} - var stack string - func() { - - defer func() { - if message = recover(); message != nil { - didPanic = true - stack = string(debug.Stack()) - } - }() - - // call the target function - f() +func didPanic(f PanicTestFunc) (didPanic bool, message interface{}, stack string) { + didPanic = true + defer func() { + message = recover() + if didPanic { + stack = string(debug.Stack()) + } }() - return didPanic, message, stack + // call the target function + f() + didPanic = false + return } // Panics asserts that the code inside the specified PanicTestFunc panics. @@ -1111,6 +1146,27 @@ func WithinDuration(t TestingT, expected, actual time.Time, delta time.Duration, return true } +// WithinRange asserts that a time is within a time range (inclusive). +// +// assert.WithinRange(t, time.Now(), time.Now().Add(-time.Second), time.Now().Add(time.Second)) +func WithinRange(t TestingT, actual, start, end time.Time, msgAndArgs ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + + if end.Before(start) { + return Fail(t, "Start should be before end", msgAndArgs...) + } + + if actual.Before(start) { + return Fail(t, fmt.Sprintf("Time %v expected to be in time range %v to %v, but is before the range", actual, start, end), msgAndArgs...) + } else if actual.After(end) { + return Fail(t, fmt.Sprintf("Time %v expected to be in time range %v to %v, but is after the range", actual, start, end), msgAndArgs...) + } + + return true +} + func toFloat(x interface{}) (float64, bool) { var xf float64 xok := true @@ -1161,11 +1217,15 @@ func InDelta(t TestingT, expected, actual interface{}, delta float64, msgAndArgs bf, bok := toFloat(actual) if !aok || !bok { - return Fail(t, fmt.Sprintf("Parameters must be numerical"), msgAndArgs...) + return Fail(t, "Parameters must be numerical", msgAndArgs...) + } + + if math.IsNaN(af) && math.IsNaN(bf) { + return true } if math.IsNaN(af) { - return Fail(t, fmt.Sprintf("Expected must not be NaN"), msgAndArgs...) + return Fail(t, "Expected must not be NaN", msgAndArgs...) } if math.IsNaN(bf) { @@ -1188,7 +1248,7 @@ func InDeltaSlice(t TestingT, expected, actual interface{}, delta float64, msgAn if expected == nil || actual == nil || reflect.TypeOf(actual).Kind() != reflect.Slice || reflect.TypeOf(expected).Kind() != reflect.Slice { - return Fail(t, fmt.Sprintf("Parameters must be slice"), msgAndArgs...) + return Fail(t, "Parameters must be slice", msgAndArgs...) } actualSlice := reflect.ValueOf(actual) @@ -1250,8 +1310,12 @@ func InDeltaMapValues(t TestingT, expected, actual interface{}, delta float64, m func calcRelativeError(expected, actual interface{}) (float64, error) { af, aok := toFloat(expected) - if !aok { - return 0, fmt.Errorf("expected value %q cannot be converted to float", expected) + bf, bok := toFloat(actual) + if !aok || !bok { + return 0, fmt.Errorf("Parameters must be numerical") + } + if math.IsNaN(af) && math.IsNaN(bf) { + return 0, nil } if math.IsNaN(af) { return 0, errors.New("expected value must not be NaN") @@ -1259,10 +1323,6 @@ func calcRelativeError(expected, actual interface{}) (float64, error) { if af == 0 { return 0, fmt.Errorf("expected value must have a value other than zero to calculate the relative error") } - bf, bok := toFloat(actual) - if !bok { - return 0, fmt.Errorf("actual value %q cannot be converted to float", actual) - } if math.IsNaN(bf) { return 0, errors.New("actual value must not be NaN") } @@ -1298,7 +1358,7 @@ func InEpsilonSlice(t TestingT, expected, actual interface{}, epsilon float64, m if expected == nil || actual == nil || reflect.TypeOf(actual).Kind() != reflect.Slice || reflect.TypeOf(expected).Kind() != reflect.Slice { - return Fail(t, fmt.Sprintf("Parameters must be slice"), msgAndArgs...) + return Fail(t, "Parameters must be slice", msgAndArgs...) } actualSlice := reflect.ValueOf(actual) @@ -1375,6 +1435,27 @@ func EqualError(t TestingT, theError error, errString string, msgAndArgs ...inte return true } +// ErrorContains asserts that a function returned an error (i.e. not `nil`) +// and that the error contains the specified substring. +// +// actualObj, err := SomeFunction() +// assert.ErrorContains(t, err, expectedErrorSubString) +func ErrorContains(t TestingT, theError error, contains string, msgAndArgs ...interface{}) bool { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if !Error(t, theError, msgAndArgs...) { + return false + } + + actual := theError.Error() + if !strings.Contains(actual, contains) { + return Fail(t, fmt.Sprintf("Error %#v does not contain %#v", actual, contains), msgAndArgs...) + } + + return true +} + // matchRegexp return true if a specified regexp matches a string. func matchRegexp(rx interface{}, str interface{}) bool { @@ -1588,12 +1669,17 @@ func diff(expected interface{}, actual interface{}) string { } var e, a string - if et != reflect.TypeOf("") { - e = spewConfig.Sdump(expected) - a = spewConfig.Sdump(actual) - } else { + + switch et { + case reflect.TypeOf(""): e = reflect.ValueOf(expected).String() a = reflect.ValueOf(actual).String() + case reflect.TypeOf(time.Time{}): + e = spewConfigStringerEnabled.Sdump(expected) + a = spewConfigStringerEnabled.Sdump(actual) + default: + e = spewConfig.Sdump(expected) + a = spewConfig.Sdump(actual) } diff, _ := difflib.GetUnifiedDiffString(difflib.UnifiedDiff{ @@ -1625,6 +1711,14 @@ var spewConfig = spew.ConfigState{ MaxDepth: 10, } +var spewConfigStringerEnabled = spew.ConfigState{ + Indent: " ", + DisablePointerAddresses: true, + DisableCapacities: true, + SortKeys: true, + MaxDepth: 10, +} + type tHelper interface { Helper() } diff --git a/vendor/github.com/stretchr/testify/require/require.go b/vendor/github.com/stretchr/testify/require/require.go index 51820df2..880853f5 100644 --- a/vendor/github.com/stretchr/testify/require/require.go +++ b/vendor/github.com/stretchr/testify/require/require.go @@ -280,6 +280,36 @@ func ErrorAsf(t TestingT, err error, target interface{}, msg string, args ...int t.FailNow() } +// ErrorContains asserts that a function returned an error (i.e. not `nil`) +// and that the error contains the specified substring. +// +// actualObj, err := SomeFunction() +// assert.ErrorContains(t, err, expectedErrorSubString) +func ErrorContains(t TestingT, theError error, contains string, msgAndArgs ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.ErrorContains(t, theError, contains, msgAndArgs...) { + return + } + t.FailNow() +} + +// ErrorContainsf asserts that a function returned an error (i.e. not `nil`) +// and that the error contains the specified substring. +// +// actualObj, err := SomeFunction() +// assert.ErrorContainsf(t, err, expectedErrorSubString, "error message %s", "formatted") +func ErrorContainsf(t TestingT, theError error, contains string, msg string, args ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.ErrorContainsf(t, theError, contains, msg, args...) { + return + } + t.FailNow() +} + // ErrorIs asserts that at least one of the errors in err's chain matches target. // This is a wrapper for errors.Is. func ErrorIs(t TestingT, err error, target error, msgAndArgs ...interface{}) { @@ -1834,6 +1864,32 @@ func WithinDurationf(t TestingT, expected time.Time, actual time.Time, delta tim t.FailNow() } +// WithinRange asserts that a time is within a time range (inclusive). +// +// assert.WithinRange(t, time.Now(), time.Now().Add(-time.Second), time.Now().Add(time.Second)) +func WithinRange(t TestingT, actual time.Time, start time.Time, end time.Time, msgAndArgs ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.WithinRange(t, actual, start, end, msgAndArgs...) { + return + } + t.FailNow() +} + +// WithinRangef asserts that a time is within a time range (inclusive). +// +// assert.WithinRangef(t, time.Now(), time.Now().Add(-time.Second), time.Now().Add(time.Second), "error message %s", "formatted") +func WithinRangef(t TestingT, actual time.Time, start time.Time, end time.Time, msg string, args ...interface{}) { + if h, ok := t.(tHelper); ok { + h.Helper() + } + if assert.WithinRangef(t, actual, start, end, msg, args...) { + return + } + t.FailNow() +} + // YAMLEq asserts that two YAML strings are equivalent. func YAMLEq(t TestingT, expected string, actual string, msgAndArgs ...interface{}) { if h, ok := t.(tHelper); ok { diff --git a/vendor/github.com/stretchr/testify/require/require_forward.go b/vendor/github.com/stretchr/testify/require/require_forward.go index ed54a9d8..960bf6f2 100644 --- a/vendor/github.com/stretchr/testify/require/require_forward.go +++ b/vendor/github.com/stretchr/testify/require/require_forward.go @@ -223,6 +223,30 @@ func (a *Assertions) ErrorAsf(err error, target interface{}, msg string, args .. ErrorAsf(a.t, err, target, msg, args...) } +// ErrorContains asserts that a function returned an error (i.e. not `nil`) +// and that the error contains the specified substring. +// +// actualObj, err := SomeFunction() +// a.ErrorContains(err, expectedErrorSubString) +func (a *Assertions) ErrorContains(theError error, contains string, msgAndArgs ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + ErrorContains(a.t, theError, contains, msgAndArgs...) +} + +// ErrorContainsf asserts that a function returned an error (i.e. not `nil`) +// and that the error contains the specified substring. +// +// actualObj, err := SomeFunction() +// a.ErrorContainsf(err, expectedErrorSubString, "error message %s", "formatted") +func (a *Assertions) ErrorContainsf(theError error, contains string, msg string, args ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + ErrorContainsf(a.t, theError, contains, msg, args...) +} + // ErrorIs asserts that at least one of the errors in err's chain matches target. // This is a wrapper for errors.Is. func (a *Assertions) ErrorIs(err error, target error, msgAndArgs ...interface{}) { @@ -1438,6 +1462,26 @@ func (a *Assertions) WithinDurationf(expected time.Time, actual time.Time, delta WithinDurationf(a.t, expected, actual, delta, msg, args...) } +// WithinRange asserts that a time is within a time range (inclusive). +// +// a.WithinRange(time.Now(), time.Now().Add(-time.Second), time.Now().Add(time.Second)) +func (a *Assertions) WithinRange(actual time.Time, start time.Time, end time.Time, msgAndArgs ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + WithinRange(a.t, actual, start, end, msgAndArgs...) +} + +// WithinRangef asserts that a time is within a time range (inclusive). +// +// a.WithinRangef(time.Now(), time.Now().Add(-time.Second), time.Now().Add(time.Second), "error message %s", "formatted") +func (a *Assertions) WithinRangef(actual time.Time, start time.Time, end time.Time, msg string, args ...interface{}) { + if h, ok := a.t.(tHelper); ok { + h.Helper() + } + WithinRangef(a.t, actual, start, end, msg, args...) +} + // YAMLEq asserts that two YAML strings are equivalent. func (a *Assertions) YAMLEq(expected string, actual string, msgAndArgs ...interface{}) { if h, ok := a.t.(tHelper); ok { diff --git a/vendor/github.com/tchap/go-patricia/v2/AUTHORS b/vendor/github.com/tchap/go-patricia/v2/AUTHORS new file mode 100644 index 00000000..e640b0bf --- /dev/null +++ b/vendor/github.com/tchap/go-patricia/v2/AUTHORS @@ -0,0 +1,3 @@ +This is the complete list of go-patricia copyright holders: + +Ondřej Kupka diff --git a/vendor/github.com/tchap/go-patricia/v2/LICENSE b/vendor/github.com/tchap/go-patricia/v2/LICENSE new file mode 100644 index 00000000..e50d398e --- /dev/null +++ b/vendor/github.com/tchap/go-patricia/v2/LICENSE @@ -0,0 +1,20 @@ +The MIT License (MIT) + +Copyright (c) 2014 The AUTHORS + +Permission is hereby granted, free of charge, to any person obtaining a copy of +this software and associated documentation files (the "Software"), to deal in +the Software without restriction, including without limitation the rights to +use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of +the Software, and to permit persons to whom the Software is furnished to do so, +subject to the following conditions: + +The above copyright notice and this permission notice shall be included in all +copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS +FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR +COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER +IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN +CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. diff --git a/vendor/github.com/tchap/go-patricia/v2/patricia/children.go b/vendor/github.com/tchap/go-patricia/v2/patricia/children.go new file mode 100644 index 00000000..bcfd0a5d --- /dev/null +++ b/vendor/github.com/tchap/go-patricia/v2/patricia/children.go @@ -0,0 +1,363 @@ +// Copyright (c) 2014 The go-patricia AUTHORS +// +// Use of this source code is governed by The MIT License +// that can be found in the LICENSE file. + +package patricia + +import ( + "fmt" + "io" + "sort" +) + +type childList interface { + length() int + head() *Trie + add(child *Trie) childList + remove(b byte) + replace(b byte, child *Trie) + next(b byte) *Trie + walk(prefix *Prefix, visitor VisitorFunc) error + print(w io.Writer, indent int) + clone() childList + total() int +} + +type tries []*Trie + +func (t tries) Len() int { + return len(t) +} + +func (t tries) Less(i, j int) bool { + strings := sort.StringSlice{string(t[i].prefix), string(t[j].prefix)} + return strings.Less(0, 1) +} + +func (t tries) Swap(i, j int) { + t[i], t[j] = t[j], t[i] +} + +type sparseChildList struct { + children tries +} + +func newSparseChildList(maxChildrenPerSparseNode int) childList { + return &sparseChildList{ + children: make(tries, 0, maxChildrenPerSparseNode), + } +} + +func (list *sparseChildList) length() int { + return len(list.children) +} + +func (list *sparseChildList) head() *Trie { + return list.children[0] +} + +func (list *sparseChildList) add(child *Trie) childList { + // Search for an empty spot and insert the child if possible. + if len(list.children) != cap(list.children) { + list.children = append(list.children, child) + return list + } + + // Otherwise we have to transform to the dense list type. + return newDenseChildList(list, child) +} + +func (list *sparseChildList) remove(b byte) { + for i, node := range list.children { + if node.prefix[0] == b { + list.children[i] = list.children[len(list.children)-1] + list.children[len(list.children)-1] = nil + list.children = list.children[:len(list.children)-1] + return + } + } + + // This is not supposed to be reached. + panic("removing non-existent child") +} + +func (list *sparseChildList) replace(b byte, child *Trie) { + // Make a consistency check. + if p0 := child.prefix[0]; p0 != b { + panic(fmt.Errorf("child prefix mismatch: %v != %v", p0, b)) + } + + // Seek the child and replace it. + for i, node := range list.children { + if node.prefix[0] == b { + list.children[i] = child + return + } + } +} + +func (list *sparseChildList) next(b byte) *Trie { + for _, child := range list.children { + if child.prefix[0] == b { + return child + } + } + return nil +} + +func (list *sparseChildList) walk(prefix *Prefix, visitor VisitorFunc) error { + + sort.Sort(list.children) + + for _, child := range list.children { + *prefix = append(*prefix, child.prefix...) + if child.item != nil { + err := visitor(*prefix, child.item) + if err != nil { + if err == SkipSubtree { + *prefix = (*prefix)[:len(*prefix)-len(child.prefix)] + continue + } + *prefix = (*prefix)[:len(*prefix)-len(child.prefix)] + return err + } + } + + err := child.children.walk(prefix, visitor) + *prefix = (*prefix)[:len(*prefix)-len(child.prefix)] + if err != nil { + return err + } + } + + return nil +} + +func (list *sparseChildList) total() int { + tot := 0 + for _, child := range list.children { + if child != nil { + tot = tot + child.total() + } + } + return tot +} + +func (list *sparseChildList) clone() childList { + clones := make(tries, len(list.children), cap(list.children)) + for i, child := range list.children { + clones[i] = child.Clone() + } + + return &sparseChildList{ + children: clones, + } +} + +func (list *sparseChildList) print(w io.Writer, indent int) { + for _, child := range list.children { + if child != nil { + child.print(w, indent) + } + } +} + +type denseChildList struct { + min int + max int + numChildren int + headIndex int + children []*Trie +} + +func newDenseChildList(list *sparseChildList, child *Trie) childList { + var ( + min int = 255 + max int = 0 + ) + for _, child := range list.children { + b := int(child.prefix[0]) + if b < min { + min = b + } + if b > max { + max = b + } + } + + b := int(child.prefix[0]) + if b < min { + min = b + } + if b > max { + max = b + } + + children := make([]*Trie, max-min+1) + for _, child := range list.children { + children[int(child.prefix[0])-min] = child + } + children[int(child.prefix[0])-min] = child + + return &denseChildList{ + min: min, + max: max, + numChildren: list.length() + 1, + headIndex: 0, + children: children, + } +} + +func (list *denseChildList) length() int { + return list.numChildren +} + +func (list *denseChildList) head() *Trie { + return list.children[list.headIndex] +} + +func (list *denseChildList) add(child *Trie) childList { + b := int(child.prefix[0]) + var i int + + switch { + case list.min <= b && b <= list.max: + if list.children[b-list.min] != nil { + panic("dense child list collision detected") + } + i = b - list.min + list.children[i] = child + + case b < list.min: + children := make([]*Trie, list.max-b+1) + i = 0 + children[i] = child + copy(children[list.min-b:], list.children) + list.children = children + list.min = b + + default: // b > list.max + children := make([]*Trie, b-list.min+1) + i = b - list.min + children[i] = child + copy(children, list.children) + list.children = children + list.max = b + } + + list.numChildren++ + if i < list.headIndex { + list.headIndex = i + } + return list +} + +func (list *denseChildList) remove(b byte) { + i := int(b) - list.min + if list.children[i] == nil { + // This is not supposed to be reached. + panic("removing non-existent child") + } + list.numChildren-- + list.children[i] = nil + + // Update head index. + if i == list.headIndex { + for ; i < len(list.children); i++ { + if list.children[i] != nil { + list.headIndex = i + return + } + } + } +} + +func (list *denseChildList) replace(b byte, child *Trie) { + // Make a consistency check. + if p0 := child.prefix[0]; p0 != b { + panic(fmt.Errorf("child prefix mismatch: %v != %v", p0, b)) + } + + // Replace the child. + list.children[int(b)-list.min] = child +} + +func (list *denseChildList) next(b byte) *Trie { + i := int(b) + if i < list.min || list.max < i { + return nil + } + return list.children[i-list.min] +} + +func (list *denseChildList) walk(prefix *Prefix, visitor VisitorFunc) error { + for _, child := range list.children { + if child == nil { + continue + } + *prefix = append(*prefix, child.prefix...) + if child.item != nil { + if err := visitor(*prefix, child.item); err != nil { + if err == SkipSubtree { + *prefix = (*prefix)[:len(*prefix)-len(child.prefix)] + continue + } + *prefix = (*prefix)[:len(*prefix)-len(child.prefix)] + return err + } + } + + err := child.children.walk(prefix, visitor) + *prefix = (*prefix)[:len(*prefix)-len(child.prefix)] + if err != nil { + return err + } + } + + return nil +} + +func (list *denseChildList) print(w io.Writer, indent int) { + for _, child := range list.children { + if child != nil { + child.print(w, indent) + } + } +} + +func (list *denseChildList) clone() childList { + clones := make(tries, cap(list.children)) + + if list.numChildren != 0 { + clonedCount := 0 + for i := list.headIndex; i < len(list.children); i++ { + child := list.children[i] + if child != nil { + clones[i] = child.Clone() + clonedCount++ + if clonedCount == list.numChildren { + break + } + } + } + } + + return &denseChildList{ + min: list.min, + max: list.max, + numChildren: list.numChildren, + headIndex: list.headIndex, + children: clones, + } +} + +func (list *denseChildList) total() int { + tot := 0 + for _, child := range list.children { + if child != nil { + tot = tot + child.total() + } + } + return tot +} diff --git a/vendor/github.com/tchap/go-patricia/v2/patricia/patricia.go b/vendor/github.com/tchap/go-patricia/v2/patricia/patricia.go new file mode 100644 index 00000000..7b9975e3 --- /dev/null +++ b/vendor/github.com/tchap/go-patricia/v2/patricia/patricia.go @@ -0,0 +1,606 @@ +// Copyright (c) 2014 The go-patricia AUTHORS +// +// Use of this source code is governed by The MIT License +// that can be found in the LICENSE file. + +package patricia + +import ( + "bytes" + "errors" + "fmt" + "io" + "strings" +) + +//------------------------------------------------------------------------------ +// Trie +//------------------------------------------------------------------------------ + +const ( + DefaultMaxPrefixPerNode = 10 + DefaultMaxChildrenPerSparseNode = 8 +) + +type ( + Prefix []byte + Item interface{} + VisitorFunc func(prefix Prefix, item Item) error +) + +// Trie is a generic patricia trie that allows fast retrieval of items by prefix. +// and other funky stuff. +// +// Trie is not thread-safe. +type Trie struct { + prefix Prefix + item Item + + maxPrefixPerNode int + maxChildrenPerSparseNode int + + children childList +} + +// Public API ------------------------------------------------------------------ + +type Option func(*Trie) + +// Trie constructor. +func NewTrie(options ...Option) *Trie { + trie := &Trie{} + + for _, opt := range options { + opt(trie) + } + + if trie.maxPrefixPerNode <= 0 { + trie.maxPrefixPerNode = DefaultMaxPrefixPerNode + } + if trie.maxChildrenPerSparseNode <= 0 { + trie.maxChildrenPerSparseNode = DefaultMaxChildrenPerSparseNode + } + + trie.children = newSparseChildList(trie.maxChildrenPerSparseNode) + return trie +} + +func MaxPrefixPerNode(value int) Option { + return func(trie *Trie) { + trie.maxPrefixPerNode = value + } +} + +func MaxChildrenPerSparseNode(value int) Option { + return func(trie *Trie) { + trie.maxChildrenPerSparseNode = value + } +} + +// Clone makes a copy of an existing trie. +// Items stored in both tries become shared, obviously. +func (trie *Trie) Clone() *Trie { + return &Trie{ + prefix: append(Prefix(nil), trie.prefix...), + item: trie.item, + maxPrefixPerNode: trie.maxPrefixPerNode, + maxChildrenPerSparseNode: trie.maxChildrenPerSparseNode, + children: trie.children.clone(), + } +} + +// Item returns the item stored in the root of this trie. +func (trie *Trie) Item() Item { + return trie.item +} + +// Insert inserts a new item into the trie using the given prefix. Insert does +// not replace existing items. It returns false if an item was already in place. +func (trie *Trie) Insert(key Prefix, item Item) (inserted bool) { + return trie.put(key, item, false) +} + +// Set works much like Insert, but it always sets the item, possibly replacing +// the item previously inserted. +func (trie *Trie) Set(key Prefix, item Item) { + trie.put(key, item, true) +} + +// Get returns the item located at key. +// +// This method is a bit dangerous, because Get can as well end up in an internal +// node that is not really representing any user-defined value. So when nil is +// a valid value being used, it is not possible to tell if the value was inserted +// into the tree by the user or not. A possible workaround for this is not to use +// nil interface as a valid value, even using zero value of any type is enough +// to prevent this bad behaviour. +func (trie *Trie) Get(key Prefix) (item Item) { + _, node, found, leftover := trie.findSubtree(key) + if !found || len(leftover) != 0 { + return nil + } + return node.item +} + +// Match returns what Get(prefix) != nil would return. The same warning as for +// Get applies here as well. +func (trie *Trie) Match(prefix Prefix) (matchedExactly bool) { + return trie.Get(prefix) != nil +} + +// MatchSubtree returns true when there is a subtree representing extensions +// to key, that is if there are any keys in the tree which have key as prefix. +func (trie *Trie) MatchSubtree(key Prefix) (matched bool) { + _, _, matched, _ = trie.findSubtree(key) + return +} + +// Visit calls visitor on every node containing a non-nil item +// in alphabetical order. +// +// If an error is returned from visitor, the function stops visiting the tree +// and returns that error, unless it is a special error - SkipSubtree. In that +// case Visit skips the subtree represented by the current node and continues +// elsewhere. +func (trie *Trie) Visit(visitor VisitorFunc) error { + return trie.walk(nil, visitor) +} + +func (trie *Trie) size() int { + n := 0 + + trie.walk(nil, func(prefix Prefix, item Item) error { + n++ + return nil + }) + + return n +} + +func (trie *Trie) total() int { + return 1 + trie.children.total() +} + +// VisitSubtree works much like Visit, but it only visits nodes matching prefix. +func (trie *Trie) VisitSubtree(prefix Prefix, visitor VisitorFunc) error { + // Nil prefix not allowed. + if prefix == nil { + panic(ErrNilPrefix) + } + + // Empty trie must be handled explicitly. + if trie.prefix == nil { + return nil + } + + // Locate the relevant subtree. + _, root, found, leftover := trie.findSubtree(prefix) + if !found { + return nil + } + prefix = append(prefix, leftover...) + + // Visit it. + return root.walk(prefix, visitor) +} + +// VisitPrefixes visits only nodes that represent prefixes of key. +// To say the obvious, returning SkipSubtree from visitor makes no sense here. +func (trie *Trie) VisitPrefixes(key Prefix, visitor VisitorFunc) error { + // Nil key not allowed. + if key == nil { + panic(ErrNilPrefix) + } + + // Empty trie must be handled explicitly. + if trie.prefix == nil { + return nil + } + + // Walk the path matching key prefixes. + node := trie + prefix := key + offset := 0 + for { + // Compute what part of prefix matches. + common := node.longestCommonPrefixLength(key) + key = key[common:] + offset += common + + // Partial match means that there is no subtree matching prefix. + if common < len(node.prefix) { + return nil + } + + // Call the visitor. + if item := node.item; item != nil { + if err := visitor(prefix[:offset], item); err != nil { + return err + } + } + + if len(key) == 0 { + // This node represents key, we are finished. + return nil + } + + // There is some key suffix left, move to the children. + child := node.children.next(key[0]) + if child == nil { + // There is nowhere to continue, return. + return nil + } + + node = child + } +} + +// Delete deletes the item represented by the given prefix. +// +// True is returned if the matching node was found and deleted. +func (trie *Trie) Delete(key Prefix) (deleted bool) { + // Nil prefix not allowed. + if key == nil { + panic(ErrNilPrefix) + } + + // Empty trie must be handled explicitly. + if trie.prefix == nil { + return false + } + + // Find the relevant node. + path, found, _ := trie.findSubtreePath(key) + if !found { + return false + } + + node := path[len(path)-1] + var parent *Trie + if len(path) != 1 { + parent = path[len(path)-2] + } + + // If the item is already set to nil, there is nothing to do. + if node.item == nil { + return false + } + + // Delete the item. + node.item = nil + + // Initialise i before goto. + // Will be used later in a loop. + i := len(path) - 1 + + // In case there are some child nodes, we cannot drop the whole subtree. + // We can try to compact nodes, though. + if node.children.length() != 0 { + goto Compact + } + + // In case we are at the root, just reset it and we are done. + if parent == nil { + node.reset() + return true + } + + // We can drop a subtree. + // Find the first ancestor that has its value set or it has 2 or more child nodes. + // That will be the node where to drop the subtree at. + for ; i >= 0; i-- { + if current := path[i]; current.item != nil || current.children.length() >= 2 { + break + } + } + + // Handle the case when there is no such node. + // In other words, we can reset the whole tree. + if i == -1 { + path[0].reset() + return true + } + + // We can just remove the subtree here. + node = path[i] + if i == 0 { + parent = nil + } else { + parent = path[i-1] + } + // i+1 is always a valid index since i is never pointing to the last node. + // The loop above skips at least the last node since we are sure that the item + // is set to nil and it has no children, othewise we would be compacting instead. + node.children.remove(path[i+1].prefix[0]) + +Compact: + // The node is set to the first non-empty ancestor, + // so try to compact since that might be possible now. + if compacted := node.compact(); compacted != node { + if parent == nil { + *node = *compacted + } else { + parent.children.replace(node.prefix[0], compacted) + *parent = *parent.compact() + } + } + + return true +} + +// DeleteSubtree finds the subtree exactly matching prefix and deletes it. +// +// True is returned if the subtree was found and deleted. +func (trie *Trie) DeleteSubtree(prefix Prefix) (deleted bool) { + // Nil prefix not allowed. + if prefix == nil { + panic(ErrNilPrefix) + } + + // Empty trie must be handled explicitly. + if trie.prefix == nil { + return false + } + + // Locate the relevant subtree. + parent, root, found, _ := trie.findSubtree(prefix) + if !found { + return false + } + + // If we are in the root of the trie, reset the trie. + if parent == nil { + root.reset() + return true + } + + // Otherwise remove the root node from its parent. + parent.children.remove(root.prefix[0]) + return true +} + +// Internal helper methods ----------------------------------------------------- + +func (trie *Trie) empty() bool { + return trie.item == nil && trie.children.length() == 0 +} + +func (trie *Trie) reset() { + trie.prefix = nil + trie.children = newSparseChildList(trie.maxPrefixPerNode) +} + +func (trie *Trie) put(key Prefix, item Item, replace bool) (inserted bool) { + // Nil prefix not allowed. + if key == nil { + panic(ErrNilPrefix) + } + + var ( + common int + node *Trie = trie + child *Trie + ) + + if node.prefix == nil { + if len(key) <= trie.maxPrefixPerNode { + node.prefix = key + goto InsertItem + } + node.prefix = key[:trie.maxPrefixPerNode] + key = key[trie.maxPrefixPerNode:] + goto AppendChild + } + + for { + // Compute the longest common prefix length. + common = node.longestCommonPrefixLength(key) + key = key[common:] + + // Only a part matches, split. + if common < len(node.prefix) { + goto SplitPrefix + } + + // common == len(node.prefix) since never (common > len(node.prefix)) + // common == len(former key) <-> 0 == len(key) + // -> former key == node.prefix + if len(key) == 0 { + goto InsertItem + } + + // Check children for matching prefix. + child = node.children.next(key[0]) + if child == nil { + goto AppendChild + } + node = child + } + +SplitPrefix: + // Split the prefix if necessary. + child = new(Trie) + *child = *node + *node = *NewTrie() + node.prefix = child.prefix[:common] + child.prefix = child.prefix[common:] + child = child.compact() + node.children = node.children.add(child) + +AppendChild: + // Keep appending children until whole prefix is inserted. + // This loop starts with empty node.prefix that needs to be filled. + for len(key) != 0 { + child := NewTrie() + if len(key) <= trie.maxPrefixPerNode { + child.prefix = key + node.children = node.children.add(child) + node = child + goto InsertItem + } else { + child.prefix = key[:trie.maxPrefixPerNode] + key = key[trie.maxPrefixPerNode:] + node.children = node.children.add(child) + node = child + } + } + +InsertItem: + // Try to insert the item if possible. + if replace || node.item == nil { + node.item = item + return true + } + return false +} + +func (trie *Trie) compact() *Trie { + // Only a node with a single child can be compacted. + if trie.children.length() != 1 { + return trie + } + + child := trie.children.head() + + // If any item is set, we cannot compact since we want to retain + // the ability to do searching by key. This makes compaction less usable, + // but that simply cannot be avoided. + if trie.item != nil || child.item != nil { + return trie + } + + // Make sure the combined prefixes fit into a single node. + if len(trie.prefix)+len(child.prefix) > trie.maxPrefixPerNode { + return trie + } + + // Concatenate the prefixes, move the items. + child.prefix = append(trie.prefix, child.prefix...) + if trie.item != nil { + child.item = trie.item + } + + return child +} + +func (trie *Trie) findSubtree(prefix Prefix) (parent *Trie, root *Trie, found bool, leftover Prefix) { + // Find the subtree matching prefix. + root = trie + for { + // Compute what part of prefix matches. + common := root.longestCommonPrefixLength(prefix) + prefix = prefix[common:] + + // We used up the whole prefix, subtree found. + if len(prefix) == 0 { + found = true + leftover = root.prefix[common:] + return + } + + // Partial match means that there is no subtree matching prefix. + if common < len(root.prefix) { + leftover = root.prefix[common:] + return + } + + // There is some prefix left, move to the children. + child := root.children.next(prefix[0]) + if child == nil { + // There is nowhere to continue, there is no subtree matching prefix. + return + } + + parent = root + root = child + } +} + +func (trie *Trie) findSubtreePath(prefix Prefix) (path []*Trie, found bool, leftover Prefix) { + // Find the subtree matching prefix. + root := trie + var subtreePath []*Trie + for { + // Append the current root to the path. + subtreePath = append(subtreePath, root) + + // Compute what part of prefix matches. + common := root.longestCommonPrefixLength(prefix) + prefix = prefix[common:] + + // We used up the whole prefix, subtree found. + if len(prefix) == 0 { + path = subtreePath + found = true + leftover = root.prefix[common:] + return + } + + // Partial match means that there is no subtree matching prefix. + if common < len(root.prefix) { + leftover = root.prefix[common:] + return + } + + // There is some prefix left, move to the children. + child := root.children.next(prefix[0]) + if child == nil { + // There is nowhere to continue, there is no subtree matching prefix. + return + } + + root = child + } +} + +func (trie *Trie) walk(actualRootPrefix Prefix, visitor VisitorFunc) error { + var prefix Prefix + // Allocate a bit more space for prefix at the beginning. + if actualRootPrefix == nil { + prefix = make(Prefix, 32+len(trie.prefix)) + copy(prefix, trie.prefix) + prefix = prefix[:len(trie.prefix)] + } else { + prefix = make(Prefix, 32+len(actualRootPrefix)) + copy(prefix, actualRootPrefix) + prefix = prefix[:len(actualRootPrefix)] + } + + // Visit the root first. Not that this works for empty trie as well since + // in that case item == nil && len(children) == 0. + if trie.item != nil { + if err := visitor(prefix, trie.item); err != nil { + if err == SkipSubtree { + return nil + } + return err + } + } + + // Then continue to the children. + return trie.children.walk(&prefix, visitor) +} + +func (trie *Trie) longestCommonPrefixLength(prefix Prefix) (i int) { + for ; i < len(prefix) && i < len(trie.prefix) && prefix[i] == trie.prefix[i]; i++ { + } + return +} + +func (trie *Trie) dump() string { + writer := &bytes.Buffer{} + trie.print(writer, 0) + return writer.String() +} + +func (trie *Trie) print(writer io.Writer, indent int) { + fmt.Fprintf(writer, "%s%s %v\n", strings.Repeat(" ", indent), string(trie.prefix), trie.item) + trie.children.print(writer, indent+2) +} + +// Errors ---------------------------------------------------------------------- + +var ( + SkipSubtree = errors.New("Skip this subtree") + ErrNilPrefix = errors.New("Nil prefix passed into a method call") +) diff --git a/vendor/github.com/yashtewari/glob-intersection/match.go b/vendor/github.com/yashtewari/glob-intersection/match.go index 45a988a8..63c75c84 100644 --- a/vendor/github.com/yashtewari/glob-intersection/match.go +++ b/vendor/github.com/yashtewari/glob-intersection/match.go @@ -1,7 +1,7 @@ package gintersect import ( - "github.com/pkg/errors" + "errors" ) var ( diff --git a/vendor/github.com/yashtewari/glob-intersection/tokenize.go b/vendor/github.com/yashtewari/glob-intersection/tokenize.go index 0b674316..df0efe5a 100644 --- a/vendor/github.com/yashtewari/glob-intersection/tokenize.go +++ b/vendor/github.com/yashtewari/glob-intersection/tokenize.go @@ -3,7 +3,7 @@ package gintersect import ( "fmt" - "github.com/pkg/errors" + "errors" ) // Modifier is a special character that affects lexical analysis. @@ -65,7 +65,7 @@ func nextToken(index int, input []rune) (newIndex int, token Token, err error) { case TTSet: if r == ']' { - err = errors.Wrap(ErrInvalidInput, invalidInputMessage(input, newIndex, "set-close ']' with no preceding '['")) + err = invalidInputMessageErrorf(input, newIndex, "set-close ']' with no preceding '[': %w", ErrInvalidInput) return } @@ -75,15 +75,15 @@ func nextToken(index int, input []rune) (newIndex int, token Token, err error) { } default: - panic(errors.Wrapf(errBadImplementation, "encountered unhandled token type: %v", ttype)) + panic(fmt.Errorf("encountered unhandled token type %v: %w", ttype, errBadImplementation)) } } else if _, ok := flagRunes[r]; ok { - err = errors.Wrap(ErrInvalidInput, invalidInputMessage(input, newIndex, "flag '%s' must be preceded by a non-flag", string(r))) + err = invalidInputMessageErrorf(input, newIndex, "flag '%s' must be preceded by a non-flag: %w", string(r), ErrInvalidInput) return } else if m, ok := modifierRunes[r]; ok { - panic(errors.Wrapf(errBadImplementation, "encountered unhandled modifier: %v", m)) + panic(fmt.Errorf("encountered unhandled modifier %v: %w", m, errBadImplementation)) } else { // Nothing special to do. token = NewCharacter(r) @@ -133,11 +133,11 @@ func nextTokenSet(index int, input []rune) (newIndex int, t Token, err error) { switch r { case '-': if !prevExists { - err = errors.Wrap(ErrInvalidInput, invalidInputMessage(input, newIndex, "range character '-' must be preceded by a Unicode character")) + err = invalidInputMessageErrorf(input, newIndex, "range character '-' must be preceded by a Unicode character: %w", ErrInvalidInput) return } if newIndex >= len(input)-1 { - err = errors.Wrap(ErrInvalidInput, invalidInputMessage(input, newIndex, "range character '-' must be followed by a Unicode character")) + err = invalidInputMessageErrorf(input, newIndex, "range character '-' must be followed by a Unicode character: %w", ErrInvalidInput) return } @@ -146,12 +146,12 @@ func nextTokenSet(index int, input []rune) (newIndex int, t Token, err error) { if !escaped { if r == ']' || r == '-' { - err = errors.Wrap(ErrInvalidInput, invalidInputMessage(input, newIndex, "range character '-' cannot be followed by a special symbol")) + err = invalidInputMessageErrorf(input, newIndex, "range character '-' cannot be followed by a special symbol: %w", ErrInvalidInput) return } } if r < prev { - err = errors.Wrap(ErrInvalidInput, invalidInputMessage(input, newIndex, "range is out of order: '%s' comes before '%s' in Unicode", string(r), string(prev))) + err = invalidInputMessageErrorf(input, newIndex, "range is out of order: '%s' comes before '%s' in Unicode: %w", string(r), string(prev), ErrInvalidInput) return } @@ -183,7 +183,7 @@ func nextTokenSet(index int, input []rune) (newIndex int, t Token, err error) { // End of input is reached before the set completes. if !complete { - err = errors.Wrap(ErrInvalidInput, invalidInputMessage(input, newIndex, "found [ without matching ]")) + err = invalidInputMessageErrorf(input, newIndex, "found [ without matching ]: %w", ErrInvalidInput) } else { t = NewSet(runes) } @@ -233,10 +233,10 @@ func nextRune(index int, input []rune) (newIndex int, r rune, escaped bool, err if index < len(input)-1 { newIndex, r, escaped = index+2, input[index+1], true } else if index == len(input)-1 { - err = errors.Wrap(ErrInvalidInput, invalidInputMessage(input, index, "input ends with a \\ (escape) character")) + err = invalidInputMessageErrorf(input, index, "input ends with a \\ (escape) character: %w", ErrInvalidInput) } default: - panic(errors.Wrapf(errBadImplementation, "encountered unhandled modifier: %v", m)) + panic(fmt.Errorf("encountered unhandled modifier %v: %w", m, errBadImplementation)) } } else { newIndex, r, escaped = index+1, input[index], false @@ -245,7 +245,8 @@ func nextRune(index int, input []rune) (newIndex int, r rune, escaped bool, err return } -// invalidInputMessage wraps the message describing invalid input with the input itself and index at which it is invalid. -func invalidInputMessage(input []rune, index int, message string, args ...interface{}) string { - return fmt.Sprintf("input:%s, pos:%d, %s", string(input), index, fmt.Sprintf(message, args...)) +// invalidInputMessageErrorf wraps the message describing invalid input with the input itself and index at which it is invalid. +func invalidInputMessageErrorf(input []rune, index int, message string, args ...interface{}) error { + message = fmt.Sprintf("input:%s, pos:%d, %s", string(input), index, message) + return fmt.Errorf(message, args...) } diff --git a/vendor/golang.org/x/crypto/AUTHORS b/vendor/golang.org/x/crypto/AUTHORS new file mode 100644 index 00000000..2b00ddba --- /dev/null +++ b/vendor/golang.org/x/crypto/AUTHORS @@ -0,0 +1,3 @@ +# This source code refers to The Go Authors for copyright purposes. +# The master list of authors is in the main Go distribution, +# visible at https://tip.golang.org/AUTHORS. diff --git a/vendor/golang.org/x/crypto/CONTRIBUTORS b/vendor/golang.org/x/crypto/CONTRIBUTORS new file mode 100644 index 00000000..1fbd3e97 --- /dev/null +++ b/vendor/golang.org/x/crypto/CONTRIBUTORS @@ -0,0 +1,3 @@ +# This source code was written by the Go contributors. +# The master list of contributors is in the main Go distribution, +# visible at https://tip.golang.org/CONTRIBUTORS. diff --git a/vendor/golang.org/x/crypto/LICENSE b/vendor/golang.org/x/crypto/LICENSE new file mode 100644 index 00000000..6a66aea5 --- /dev/null +++ b/vendor/golang.org/x/crypto/LICENSE @@ -0,0 +1,27 @@ +Copyright (c) 2009 The Go Authors. All rights reserved. + +Redistribution and use in source and binary forms, with or without +modification, are permitted provided that the following conditions are +met: + + * Redistributions of source code must retain the above copyright +notice, this list of conditions and the following disclaimer. + * Redistributions in binary form must reproduce the above +copyright notice, this list of conditions and the following disclaimer +in the documentation and/or other materials provided with the +distribution. + * Neither the name of Google Inc. nor the names of its +contributors may be used to endorse or promote products derived from +this software without specific prior written permission. + +THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS +"AS IS" AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT +LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR +A PARTICULAR PURPOSE ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT +OWNER OR CONTRIBUTORS BE LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, +SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT +LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, +DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON ANY +THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT +(INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE +OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE. diff --git a/vendor/golang.org/x/crypto/PATENTS b/vendor/golang.org/x/crypto/PATENTS new file mode 100644 index 00000000..73309904 --- /dev/null +++ b/vendor/golang.org/x/crypto/PATENTS @@ -0,0 +1,22 @@ +Additional IP Rights Grant (Patents) + +"This implementation" means the copyrightable works distributed by +Google as part of the Go project. + +Google hereby grants to You a perpetual, worldwide, non-exclusive, +no-charge, royalty-free, irrevocable (except as stated in this section) +patent license to make, have made, use, offer to sell, sell, import, +transfer and otherwise run, modify and propagate the contents of this +implementation of Go, where such license applies only to those patent +claims, both currently owned or controlled by Google and acquired in +the future, licensable by Google that are necessarily infringed by this +implementation of Go. This grant does not include claims that would be +infringed only as a consequence of further modification of this +implementation. If you or your agent or exclusive licensee institute or +order or agree to the institution of patent litigation against any +entity (including a cross-claim or counterclaim in a lawsuit) alleging +that this implementation of Go or any code incorporated within this +implementation of Go constitutes direct or contributory patent +infringement, or inducement of patent infringement, then any patent +rights granted to you under this License for this implementation of Go +shall terminate as of the date such litigation is filed. diff --git a/vendor/golang.org/x/crypto/bcrypt/base64.go b/vendor/golang.org/x/crypto/bcrypt/base64.go new file mode 100644 index 00000000..fc311609 --- /dev/null +++ b/vendor/golang.org/x/crypto/bcrypt/base64.go @@ -0,0 +1,35 @@ +// Copyright 2011 The Go Authors. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +package bcrypt + +import "encoding/base64" + +const alphabet = "./ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789" + +var bcEncoding = base64.NewEncoding(alphabet) + +func base64Encode(src []byte) []byte { + n := bcEncoding.EncodedLen(len(src)) + dst := make([]byte, n) + bcEncoding.Encode(dst, src) + for dst[n-1] == '=' { + n-- + } + return dst[:n] +} + +func base64Decode(src []byte) ([]byte, error) { + numOfEquals := 4 - (len(src) % 4) + for i := 0; i < numOfEquals; i++ { + src = append(src, '=') + } + + dst := make([]byte, bcEncoding.DecodedLen(len(src))) + n, err := bcEncoding.Decode(dst, src) + if err != nil { + return nil, err + } + return dst[:n], nil +} diff --git a/vendor/golang.org/x/crypto/bcrypt/bcrypt.go b/vendor/golang.org/x/crypto/bcrypt/bcrypt.go new file mode 100644 index 00000000..aeb73f81 --- /dev/null +++ b/vendor/golang.org/x/crypto/bcrypt/bcrypt.go @@ -0,0 +1,295 @@ +// Copyright 2011 The Go Authors. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +// Package bcrypt implements Provos and Mazières's bcrypt adaptive hashing +// algorithm. See http://www.usenix.org/event/usenix99/provos/provos.pdf +package bcrypt // import "golang.org/x/crypto/bcrypt" + +// The code is a port of Provos and Mazières's C implementation. +import ( + "crypto/rand" + "crypto/subtle" + "errors" + "fmt" + "io" + "strconv" + + "golang.org/x/crypto/blowfish" +) + +const ( + MinCost int = 4 // the minimum allowable cost as passed in to GenerateFromPassword + MaxCost int = 31 // the maximum allowable cost as passed in to GenerateFromPassword + DefaultCost int = 10 // the cost that will actually be set if a cost below MinCost is passed into GenerateFromPassword +) + +// The error returned from CompareHashAndPassword when a password and hash do +// not match. +var ErrMismatchedHashAndPassword = errors.New("crypto/bcrypt: hashedPassword is not the hash of the given password") + +// The error returned from CompareHashAndPassword when a hash is too short to +// be a bcrypt hash. +var ErrHashTooShort = errors.New("crypto/bcrypt: hashedSecret too short to be a bcrypted password") + +// The error returned from CompareHashAndPassword when a hash was created with +// a bcrypt algorithm newer than this implementation. +type HashVersionTooNewError byte + +func (hv HashVersionTooNewError) Error() string { + return fmt.Sprintf("crypto/bcrypt: bcrypt algorithm version '%c' requested is newer than current version '%c'", byte(hv), majorVersion) +} + +// The error returned from CompareHashAndPassword when a hash starts with something other than '$' +type InvalidHashPrefixError byte + +func (ih InvalidHashPrefixError) Error() string { + return fmt.Sprintf("crypto/bcrypt: bcrypt hashes must start with '$', but hashedSecret started with '%c'", byte(ih)) +} + +type InvalidCostError int + +func (ic InvalidCostError) Error() string { + return fmt.Sprintf("crypto/bcrypt: cost %d is outside allowed range (%d,%d)", int(ic), int(MinCost), int(MaxCost)) +} + +const ( + majorVersion = '2' + minorVersion = 'a' + maxSaltSize = 16 + maxCryptedHashSize = 23 + encodedSaltSize = 22 + encodedHashSize = 31 + minHashSize = 59 +) + +// magicCipherData is an IV for the 64 Blowfish encryption calls in +// bcrypt(). It's the string "OrpheanBeholderScryDoubt" in big-endian bytes. +var magicCipherData = []byte{ + 0x4f, 0x72, 0x70, 0x68, + 0x65, 0x61, 0x6e, 0x42, + 0x65, 0x68, 0x6f, 0x6c, + 0x64, 0x65, 0x72, 0x53, + 0x63, 0x72, 0x79, 0x44, + 0x6f, 0x75, 0x62, 0x74, +} + +type hashed struct { + hash []byte + salt []byte + cost int // allowed range is MinCost to MaxCost + major byte + minor byte +} + +// GenerateFromPassword returns the bcrypt hash of the password at the given +// cost. If the cost given is less than MinCost, the cost will be set to +// DefaultCost, instead. Use CompareHashAndPassword, as defined in this package, +// to compare the returned hashed password with its cleartext version. +func GenerateFromPassword(password []byte, cost int) ([]byte, error) { + p, err := newFromPassword(password, cost) + if err != nil { + return nil, err + } + return p.Hash(), nil +} + +// CompareHashAndPassword compares a bcrypt hashed password with its possible +// plaintext equivalent. Returns nil on success, or an error on failure. +func CompareHashAndPassword(hashedPassword, password []byte) error { + p, err := newFromHash(hashedPassword) + if err != nil { + return err + } + + otherHash, err := bcrypt(password, p.cost, p.salt) + if err != nil { + return err + } + + otherP := &hashed{otherHash, p.salt, p.cost, p.major, p.minor} + if subtle.ConstantTimeCompare(p.Hash(), otherP.Hash()) == 1 { + return nil + } + + return ErrMismatchedHashAndPassword +} + +// Cost returns the hashing cost used to create the given hashed +// password. When, in the future, the hashing cost of a password system needs +// to be increased in order to adjust for greater computational power, this +// function allows one to establish which passwords need to be updated. +func Cost(hashedPassword []byte) (int, error) { + p, err := newFromHash(hashedPassword) + if err != nil { + return 0, err + } + return p.cost, nil +} + +func newFromPassword(password []byte, cost int) (*hashed, error) { + if cost < MinCost { + cost = DefaultCost + } + p := new(hashed) + p.major = majorVersion + p.minor = minorVersion + + err := checkCost(cost) + if err != nil { + return nil, err + } + p.cost = cost + + unencodedSalt := make([]byte, maxSaltSize) + _, err = io.ReadFull(rand.Reader, unencodedSalt) + if err != nil { + return nil, err + } + + p.salt = base64Encode(unencodedSalt) + hash, err := bcrypt(password, p.cost, p.salt) + if err != nil { + return nil, err + } + p.hash = hash + return p, err +} + +func newFromHash(hashedSecret []byte) (*hashed, error) { + if len(hashedSecret) < minHashSize { + return nil, ErrHashTooShort + } + p := new(hashed) + n, err := p.decodeVersion(hashedSecret) + if err != nil { + return nil, err + } + hashedSecret = hashedSecret[n:] + n, err = p.decodeCost(hashedSecret) + if err != nil { + return nil, err + } + hashedSecret = hashedSecret[n:] + + // The "+2" is here because we'll have to append at most 2 '=' to the salt + // when base64 decoding it in expensiveBlowfishSetup(). + p.salt = make([]byte, encodedSaltSize, encodedSaltSize+2) + copy(p.salt, hashedSecret[:encodedSaltSize]) + + hashedSecret = hashedSecret[encodedSaltSize:] + p.hash = make([]byte, len(hashedSecret)) + copy(p.hash, hashedSecret) + + return p, nil +} + +func bcrypt(password []byte, cost int, salt []byte) ([]byte, error) { + cipherData := make([]byte, len(magicCipherData)) + copy(cipherData, magicCipherData) + + c, err := expensiveBlowfishSetup(password, uint32(cost), salt) + if err != nil { + return nil, err + } + + for i := 0; i < 24; i += 8 { + for j := 0; j < 64; j++ { + c.Encrypt(cipherData[i:i+8], cipherData[i:i+8]) + } + } + + // Bug compatibility with C bcrypt implementations. We only encode 23 of + // the 24 bytes encrypted. + hsh := base64Encode(cipherData[:maxCryptedHashSize]) + return hsh, nil +} + +func expensiveBlowfishSetup(key []byte, cost uint32, salt []byte) (*blowfish.Cipher, error) { + csalt, err := base64Decode(salt) + if err != nil { + return nil, err + } + + // Bug compatibility with C bcrypt implementations. They use the trailing + // NULL in the key string during expansion. + // We copy the key to prevent changing the underlying array. + ckey := append(key[:len(key):len(key)], 0) + + c, err := blowfish.NewSaltedCipher(ckey, csalt) + if err != nil { + return nil, err + } + + var i, rounds uint64 + rounds = 1 << cost + for i = 0; i < rounds; i++ { + blowfish.ExpandKey(ckey, c) + blowfish.ExpandKey(csalt, c) + } + + return c, nil +} + +func (p *hashed) Hash() []byte { + arr := make([]byte, 60) + arr[0] = '$' + arr[1] = p.major + n := 2 + if p.minor != 0 { + arr[2] = p.minor + n = 3 + } + arr[n] = '$' + n++ + copy(arr[n:], []byte(fmt.Sprintf("%02d", p.cost))) + n += 2 + arr[n] = '$' + n++ + copy(arr[n:], p.salt) + n += encodedSaltSize + copy(arr[n:], p.hash) + n += encodedHashSize + return arr[:n] +} + +func (p *hashed) decodeVersion(sbytes []byte) (int, error) { + if sbytes[0] != '$' { + return -1, InvalidHashPrefixError(sbytes[0]) + } + if sbytes[1] > majorVersion { + return -1, HashVersionTooNewError(sbytes[1]) + } + p.major = sbytes[1] + n := 3 + if sbytes[2] != '$' { + p.minor = sbytes[2] + n++ + } + return n, nil +} + +// sbytes should begin where decodeVersion left off. +func (p *hashed) decodeCost(sbytes []byte) (int, error) { + cost, err := strconv.Atoi(string(sbytes[0:2])) + if err != nil { + return -1, err + } + err = checkCost(cost) + if err != nil { + return -1, err + } + p.cost = cost + return 3, nil +} + +func (p *hashed) String() string { + return fmt.Sprintf("&{hash: %#v, salt: %#v, cost: %d, major: %c, minor: %c}", string(p.hash), p.salt, p.cost, p.major, p.minor) +} + +func checkCost(cost int) error { + if cost < MinCost || cost > MaxCost { + return InvalidCostError(cost) + } + return nil +} diff --git a/vendor/golang.org/x/crypto/blowfish/block.go b/vendor/golang.org/x/crypto/blowfish/block.go new file mode 100644 index 00000000..9d80f195 --- /dev/null +++ b/vendor/golang.org/x/crypto/blowfish/block.go @@ -0,0 +1,159 @@ +// Copyright 2010 The Go Authors. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +package blowfish + +// getNextWord returns the next big-endian uint32 value from the byte slice +// at the given position in a circular manner, updating the position. +func getNextWord(b []byte, pos *int) uint32 { + var w uint32 + j := *pos + for i := 0; i < 4; i++ { + w = w<<8 | uint32(b[j]) + j++ + if j >= len(b) { + j = 0 + } + } + *pos = j + return w +} + +// ExpandKey performs a key expansion on the given *Cipher. Specifically, it +// performs the Blowfish algorithm's key schedule which sets up the *Cipher's +// pi and substitution tables for calls to Encrypt. This is used, primarily, +// by the bcrypt package to reuse the Blowfish key schedule during its +// set up. It's unlikely that you need to use this directly. +func ExpandKey(key []byte, c *Cipher) { + j := 0 + for i := 0; i < 18; i++ { + // Using inlined getNextWord for performance. + var d uint32 + for k := 0; k < 4; k++ { + d = d<<8 | uint32(key[j]) + j++ + if j >= len(key) { + j = 0 + } + } + c.p[i] ^= d + } + + var l, r uint32 + for i := 0; i < 18; i += 2 { + l, r = encryptBlock(l, r, c) + c.p[i], c.p[i+1] = l, r + } + + for i := 0; i < 256; i += 2 { + l, r = encryptBlock(l, r, c) + c.s0[i], c.s0[i+1] = l, r + } + for i := 0; i < 256; i += 2 { + l, r = encryptBlock(l, r, c) + c.s1[i], c.s1[i+1] = l, r + } + for i := 0; i < 256; i += 2 { + l, r = encryptBlock(l, r, c) + c.s2[i], c.s2[i+1] = l, r + } + for i := 0; i < 256; i += 2 { + l, r = encryptBlock(l, r, c) + c.s3[i], c.s3[i+1] = l, r + } +} + +// This is similar to ExpandKey, but folds the salt during the key +// schedule. While ExpandKey is essentially expandKeyWithSalt with an all-zero +// salt passed in, reusing ExpandKey turns out to be a place of inefficiency +// and specializing it here is useful. +func expandKeyWithSalt(key []byte, salt []byte, c *Cipher) { + j := 0 + for i := 0; i < 18; i++ { + c.p[i] ^= getNextWord(key, &j) + } + + j = 0 + var l, r uint32 + for i := 0; i < 18; i += 2 { + l ^= getNextWord(salt, &j) + r ^= getNextWord(salt, &j) + l, r = encryptBlock(l, r, c) + c.p[i], c.p[i+1] = l, r + } + + for i := 0; i < 256; i += 2 { + l ^= getNextWord(salt, &j) + r ^= getNextWord(salt, &j) + l, r = encryptBlock(l, r, c) + c.s0[i], c.s0[i+1] = l, r + } + + for i := 0; i < 256; i += 2 { + l ^= getNextWord(salt, &j) + r ^= getNextWord(salt, &j) + l, r = encryptBlock(l, r, c) + c.s1[i], c.s1[i+1] = l, r + } + + for i := 0; i < 256; i += 2 { + l ^= getNextWord(salt, &j) + r ^= getNextWord(salt, &j) + l, r = encryptBlock(l, r, c) + c.s2[i], c.s2[i+1] = l, r + } + + for i := 0; i < 256; i += 2 { + l ^= getNextWord(salt, &j) + r ^= getNextWord(salt, &j) + l, r = encryptBlock(l, r, c) + c.s3[i], c.s3[i+1] = l, r + } +} + +func encryptBlock(l, r uint32, c *Cipher) (uint32, uint32) { + xl, xr := l, r + xl ^= c.p[0] + xr ^= ((c.s0[byte(xl>>24)] + c.s1[byte(xl>>16)]) ^ c.s2[byte(xl>>8)]) + c.s3[byte(xl)] ^ c.p[1] + xl ^= ((c.s0[byte(xr>>24)] + c.s1[byte(xr>>16)]) ^ c.s2[byte(xr>>8)]) + c.s3[byte(xr)] ^ c.p[2] + xr ^= ((c.s0[byte(xl>>24)] + c.s1[byte(xl>>16)]) ^ c.s2[byte(xl>>8)]) + c.s3[byte(xl)] ^ c.p[3] + xl ^= ((c.s0[byte(xr>>24)] + c.s1[byte(xr>>16)]) ^ c.s2[byte(xr>>8)]) + c.s3[byte(xr)] ^ c.p[4] + xr ^= ((c.s0[byte(xl>>24)] + c.s1[byte(xl>>16)]) ^ c.s2[byte(xl>>8)]) + c.s3[byte(xl)] ^ c.p[5] + xl ^= ((c.s0[byte(xr>>24)] + c.s1[byte(xr>>16)]) ^ c.s2[byte(xr>>8)]) + c.s3[byte(xr)] ^ c.p[6] + xr ^= ((c.s0[byte(xl>>24)] + c.s1[byte(xl>>16)]) ^ c.s2[byte(xl>>8)]) + c.s3[byte(xl)] ^ c.p[7] + xl ^= ((c.s0[byte(xr>>24)] + c.s1[byte(xr>>16)]) ^ c.s2[byte(xr>>8)]) + c.s3[byte(xr)] ^ c.p[8] + xr ^= ((c.s0[byte(xl>>24)] + c.s1[byte(xl>>16)]) ^ c.s2[byte(xl>>8)]) + c.s3[byte(xl)] ^ c.p[9] + xl ^= ((c.s0[byte(xr>>24)] + c.s1[byte(xr>>16)]) ^ c.s2[byte(xr>>8)]) + c.s3[byte(xr)] ^ c.p[10] + xr ^= ((c.s0[byte(xl>>24)] + c.s1[byte(xl>>16)]) ^ c.s2[byte(xl>>8)]) + c.s3[byte(xl)] ^ c.p[11] + xl ^= ((c.s0[byte(xr>>24)] + c.s1[byte(xr>>16)]) ^ c.s2[byte(xr>>8)]) + c.s3[byte(xr)] ^ c.p[12] + xr ^= ((c.s0[byte(xl>>24)] + c.s1[byte(xl>>16)]) ^ c.s2[byte(xl>>8)]) + c.s3[byte(xl)] ^ c.p[13] + xl ^= ((c.s0[byte(xr>>24)] + c.s1[byte(xr>>16)]) ^ c.s2[byte(xr>>8)]) + c.s3[byte(xr)] ^ c.p[14] + xr ^= ((c.s0[byte(xl>>24)] + c.s1[byte(xl>>16)]) ^ c.s2[byte(xl>>8)]) + c.s3[byte(xl)] ^ c.p[15] + xl ^= ((c.s0[byte(xr>>24)] + c.s1[byte(xr>>16)]) ^ c.s2[byte(xr>>8)]) + c.s3[byte(xr)] ^ c.p[16] + xr ^= c.p[17] + return xr, xl +} + +func decryptBlock(l, r uint32, c *Cipher) (uint32, uint32) { + xl, xr := l, r + xl ^= c.p[17] + xr ^= ((c.s0[byte(xl>>24)] + c.s1[byte(xl>>16)]) ^ c.s2[byte(xl>>8)]) + c.s3[byte(xl)] ^ c.p[16] + xl ^= ((c.s0[byte(xr>>24)] + c.s1[byte(xr>>16)]) ^ c.s2[byte(xr>>8)]) + c.s3[byte(xr)] ^ c.p[15] + xr ^= ((c.s0[byte(xl>>24)] + c.s1[byte(xl>>16)]) ^ c.s2[byte(xl>>8)]) + c.s3[byte(xl)] ^ c.p[14] + xl ^= ((c.s0[byte(xr>>24)] + c.s1[byte(xr>>16)]) ^ c.s2[byte(xr>>8)]) + c.s3[byte(xr)] ^ c.p[13] + xr ^= ((c.s0[byte(xl>>24)] + c.s1[byte(xl>>16)]) ^ c.s2[byte(xl>>8)]) + c.s3[byte(xl)] ^ c.p[12] + xl ^= ((c.s0[byte(xr>>24)] + c.s1[byte(xr>>16)]) ^ c.s2[byte(xr>>8)]) + c.s3[byte(xr)] ^ c.p[11] + xr ^= ((c.s0[byte(xl>>24)] + c.s1[byte(xl>>16)]) ^ c.s2[byte(xl>>8)]) + c.s3[byte(xl)] ^ c.p[10] + xl ^= ((c.s0[byte(xr>>24)] + c.s1[byte(xr>>16)]) ^ c.s2[byte(xr>>8)]) + c.s3[byte(xr)] ^ c.p[9] + xr ^= ((c.s0[byte(xl>>24)] + c.s1[byte(xl>>16)]) ^ c.s2[byte(xl>>8)]) + c.s3[byte(xl)] ^ c.p[8] + xl ^= ((c.s0[byte(xr>>24)] + c.s1[byte(xr>>16)]) ^ c.s2[byte(xr>>8)]) + c.s3[byte(xr)] ^ c.p[7] + xr ^= ((c.s0[byte(xl>>24)] + c.s1[byte(xl>>16)]) ^ c.s2[byte(xl>>8)]) + c.s3[byte(xl)] ^ c.p[6] + xl ^= ((c.s0[byte(xr>>24)] + c.s1[byte(xr>>16)]) ^ c.s2[byte(xr>>8)]) + c.s3[byte(xr)] ^ c.p[5] + xr ^= ((c.s0[byte(xl>>24)] + c.s1[byte(xl>>16)]) ^ c.s2[byte(xl>>8)]) + c.s3[byte(xl)] ^ c.p[4] + xl ^= ((c.s0[byte(xr>>24)] + c.s1[byte(xr>>16)]) ^ c.s2[byte(xr>>8)]) + c.s3[byte(xr)] ^ c.p[3] + xr ^= ((c.s0[byte(xl>>24)] + c.s1[byte(xl>>16)]) ^ c.s2[byte(xl>>8)]) + c.s3[byte(xl)] ^ c.p[2] + xl ^= ((c.s0[byte(xr>>24)] + c.s1[byte(xr>>16)]) ^ c.s2[byte(xr>>8)]) + c.s3[byte(xr)] ^ c.p[1] + xr ^= c.p[0] + return xr, xl +} diff --git a/vendor/golang.org/x/crypto/blowfish/cipher.go b/vendor/golang.org/x/crypto/blowfish/cipher.go new file mode 100644 index 00000000..213bf204 --- /dev/null +++ b/vendor/golang.org/x/crypto/blowfish/cipher.go @@ -0,0 +1,99 @@ +// Copyright 2010 The Go Authors. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +// Package blowfish implements Bruce Schneier's Blowfish encryption algorithm. +// +// Blowfish is a legacy cipher and its short block size makes it vulnerable to +// birthday bound attacks (see https://sweet32.info). It should only be used +// where compatibility with legacy systems, not security, is the goal. +// +// Deprecated: any new system should use AES (from crypto/aes, if necessary in +// an AEAD mode like crypto/cipher.NewGCM) or XChaCha20-Poly1305 (from +// golang.org/x/crypto/chacha20poly1305). +package blowfish // import "golang.org/x/crypto/blowfish" + +// The code is a port of Bruce Schneier's C implementation. +// See https://www.schneier.com/blowfish.html. + +import "strconv" + +// The Blowfish block size in bytes. +const BlockSize = 8 + +// A Cipher is an instance of Blowfish encryption using a particular key. +type Cipher struct { + p [18]uint32 + s0, s1, s2, s3 [256]uint32 +} + +type KeySizeError int + +func (k KeySizeError) Error() string { + return "crypto/blowfish: invalid key size " + strconv.Itoa(int(k)) +} + +// NewCipher creates and returns a Cipher. +// The key argument should be the Blowfish key, from 1 to 56 bytes. +func NewCipher(key []byte) (*Cipher, error) { + var result Cipher + if k := len(key); k < 1 || k > 56 { + return nil, KeySizeError(k) + } + initCipher(&result) + ExpandKey(key, &result) + return &result, nil +} + +// NewSaltedCipher creates a returns a Cipher that folds a salt into its key +// schedule. For most purposes, NewCipher, instead of NewSaltedCipher, is +// sufficient and desirable. For bcrypt compatibility, the key can be over 56 +// bytes. +func NewSaltedCipher(key, salt []byte) (*Cipher, error) { + if len(salt) == 0 { + return NewCipher(key) + } + var result Cipher + if k := len(key); k < 1 { + return nil, KeySizeError(k) + } + initCipher(&result) + expandKeyWithSalt(key, salt, &result) + return &result, nil +} + +// BlockSize returns the Blowfish block size, 8 bytes. +// It is necessary to satisfy the Block interface in the +// package "crypto/cipher". +func (c *Cipher) BlockSize() int { return BlockSize } + +// Encrypt encrypts the 8-byte buffer src using the key k +// and stores the result in dst. +// Note that for amounts of data larger than a block, +// it is not safe to just call Encrypt on successive blocks; +// instead, use an encryption mode like CBC (see crypto/cipher/cbc.go). +func (c *Cipher) Encrypt(dst, src []byte) { + l := uint32(src[0])<<24 | uint32(src[1])<<16 | uint32(src[2])<<8 | uint32(src[3]) + r := uint32(src[4])<<24 | uint32(src[5])<<16 | uint32(src[6])<<8 | uint32(src[7]) + l, r = encryptBlock(l, r, c) + dst[0], dst[1], dst[2], dst[3] = byte(l>>24), byte(l>>16), byte(l>>8), byte(l) + dst[4], dst[5], dst[6], dst[7] = byte(r>>24), byte(r>>16), byte(r>>8), byte(r) +} + +// Decrypt decrypts the 8-byte buffer src using the key k +// and stores the result in dst. +func (c *Cipher) Decrypt(dst, src []byte) { + l := uint32(src[0])<<24 | uint32(src[1])<<16 | uint32(src[2])<<8 | uint32(src[3]) + r := uint32(src[4])<<24 | uint32(src[5])<<16 | uint32(src[6])<<8 | uint32(src[7]) + l, r = decryptBlock(l, r, c) + dst[0], dst[1], dst[2], dst[3] = byte(l>>24), byte(l>>16), byte(l>>8), byte(l) + dst[4], dst[5], dst[6], dst[7] = byte(r>>24), byte(r>>16), byte(r>>8), byte(r) +} + +func initCipher(c *Cipher) { + copy(c.p[0:], p[0:]) + copy(c.s0[0:], s0[0:]) + copy(c.s1[0:], s1[0:]) + copy(c.s2[0:], s2[0:]) + copy(c.s3[0:], s3[0:]) +} diff --git a/vendor/golang.org/x/crypto/blowfish/const.go b/vendor/golang.org/x/crypto/blowfish/const.go new file mode 100644 index 00000000..d0407759 --- /dev/null +++ b/vendor/golang.org/x/crypto/blowfish/const.go @@ -0,0 +1,199 @@ +// Copyright 2010 The Go Authors. All rights reserved. +// Use of this source code is governed by a BSD-style +// license that can be found in the LICENSE file. + +// The startup permutation array and substitution boxes. +// They are the hexadecimal digits of PI; see: +// https://www.schneier.com/code/constants.txt. + +package blowfish + +var s0 = [256]uint32{ + 0xd1310ba6, 0x98dfb5ac, 0x2ffd72db, 0xd01adfb7, 0xb8e1afed, 0x6a267e96, + 0xba7c9045, 0xf12c7f99, 0x24a19947, 0xb3916cf7, 0x0801f2e2, 0x858efc16, + 0x636920d8, 0x71574e69, 0xa458fea3, 0xf4933d7e, 0x0d95748f, 0x728eb658, + 0x718bcd58, 0x82154aee, 0x7b54a41d, 0xc25a59b5, 0x9c30d539, 0x2af26013, + 0xc5d1b023, 0x286085f0, 0xca417918, 0xb8db38ef, 0x8e79dcb0, 0x603a180e, + 0x6c9e0e8b, 0xb01e8a3e, 0xd71577c1, 0xbd314b27, 0x78af2fda, 0x55605c60, + 0xe65525f3, 0xaa55ab94, 0x57489862, 0x63e81440, 0x55ca396a, 0x2aab10b6, + 0xb4cc5c34, 0x1141e8ce, 0xa15486af, 0x7c72e993, 0xb3ee1411, 0x636fbc2a, + 0x2ba9c55d, 0x741831f6, 0xce5c3e16, 0x9b87931e, 0xafd6ba33, 0x6c24cf5c, + 0x7a325381, 0x28958677, 0x3b8f4898, 0x6b4bb9af, 0xc4bfe81b, 0x66282193, + 0x61d809cc, 0xfb21a991, 0x487cac60, 0x5dec8032, 0xef845d5d, 0xe98575b1, + 0xdc262302, 0xeb651b88, 0x23893e81, 0xd396acc5, 0x0f6d6ff3, 0x83f44239, + 0x2e0b4482, 0xa4842004, 0x69c8f04a, 0x9e1f9b5e, 0x21c66842, 0xf6e96c9a, + 0x670c9c61, 0xabd388f0, 0x6a51a0d2, 0xd8542f68, 0x960fa728, 0xab5133a3, + 0x6eef0b6c, 0x137a3be4, 0xba3bf050, 0x7efb2a98, 0xa1f1651d, 0x39af0176, + 0x66ca593e, 0x82430e88, 0x8cee8619, 0x456f9fb4, 0x7d84a5c3, 0x3b8b5ebe, + 0xe06f75d8, 0x85c12073, 0x401a449f, 0x56c16aa6, 0x4ed3aa62, 0x363f7706, + 0x1bfedf72, 0x429b023d, 0x37d0d724, 0xd00a1248, 0xdb0fead3, 0x49f1c09b, + 0x075372c9, 0x80991b7b, 0x25d479d8, 0xf6e8def7, 0xe3fe501a, 0xb6794c3b, + 0x976ce0bd, 0x04c006ba, 0xc1a94fb6, 0x409f60c4, 0x5e5c9ec2, 0x196a2463, + 0x68fb6faf, 0x3e6c53b5, 0x1339b2eb, 0x3b52ec6f, 0x6dfc511f, 0x9b30952c, + 0xcc814544, 0xaf5ebd09, 0xbee3d004, 0xde334afd, 0x660f2807, 0x192e4bb3, + 0xc0cba857, 0x45c8740f, 0xd20b5f39, 0xb9d3fbdb, 0x5579c0bd, 0x1a60320a, + 0xd6a100c6, 0x402c7279, 0x679f25fe, 0xfb1fa3cc, 0x8ea5e9f8, 0xdb3222f8, + 0x3c7516df, 0xfd616b15, 0x2f501ec8, 0xad0552ab, 0x323db5fa, 0xfd238760, + 0x53317b48, 0x3e00df82, 0x9e5c57bb, 0xca6f8ca0, 0x1a87562e, 0xdf1769db, + 0xd542a8f6, 0x287effc3, 0xac6732c6, 0x8c4f5573, 0x695b27b0, 0xbbca58c8, + 0xe1ffa35d, 0xb8f011a0, 0x10fa3d98, 0xfd2183b8, 0x4afcb56c, 0x2dd1d35b, + 0x9a53e479, 0xb6f84565, 0xd28e49bc, 0x4bfb9790, 0xe1ddf2da, 0xa4cb7e33, + 0x62fb1341, 0xcee4c6e8, 0xef20cada, 0x36774c01, 0xd07e9efe, 0x2bf11fb4, + 0x95dbda4d, 0xae909198, 0xeaad8e71, 0x6b93d5a0, 0xd08ed1d0, 0xafc725e0, + 0x8e3c5b2f, 0x8e7594b7, 0x8ff6e2fb, 0xf2122b64, 0x8888b812, 0x900df01c, + 0x4fad5ea0, 0x688fc31c, 0xd1cff191, 0xb3a8c1ad, 0x2f2f2218, 0xbe0e1777, + 0xea752dfe, 0x8b021fa1, 0xe5a0cc0f, 0xb56f74e8, 0x18acf3d6, 0xce89e299, + 0xb4a84fe0, 0xfd13e0b7, 0x7cc43b81, 0xd2ada8d9, 0x165fa266, 0x80957705, + 0x93cc7314, 0x211a1477, 0xe6ad2065, 0x77b5fa86, 0xc75442f5, 0xfb9d35cf, + 0xebcdaf0c, 0x7b3e89a0, 0xd6411bd3, 0xae1e7e49, 0x00250e2d, 0x2071b35e, + 0x226800bb, 0x57b8e0af, 0x2464369b, 0xf009b91e, 0x5563911d, 0x59dfa6aa, + 0x78c14389, 0xd95a537f, 0x207d5ba2, 0x02e5b9c5, 0x83260376, 0x6295cfa9, + 0x11c81968, 0x4e734a41, 0xb3472dca, 0x7b14a94a, 0x1b510052, 0x9a532915, + 0xd60f573f, 0xbc9bc6e4, 0x2b60a476, 0x81e67400, 0x08ba6fb5, 0x571be91f, + 0xf296ec6b, 0x2a0dd915, 0xb6636521, 0xe7b9f9b6, 0xff34052e, 0xc5855664, + 0x53b02d5d, 0xa99f8fa1, 0x08ba4799, 0x6e85076a, +} + +var s1 = [256]uint32{ + 0x4b7a70e9, 0xb5b32944, 0xdb75092e, 0xc4192623, 0xad6ea6b0, 0x49a7df7d, + 0x9cee60b8, 0x8fedb266, 0xecaa8c71, 0x699a17ff, 0x5664526c, 0xc2b19ee1, + 0x193602a5, 0x75094c29, 0xa0591340, 0xe4183a3e, 0x3f54989a, 0x5b429d65, + 0x6b8fe4d6, 0x99f73fd6, 0xa1d29c07, 0xefe830f5, 0x4d2d38e6, 0xf0255dc1, + 0x4cdd2086, 0x8470eb26, 0x6382e9c6, 0x021ecc5e, 0x09686b3f, 0x3ebaefc9, + 0x3c971814, 0x6b6a70a1, 0x687f3584, 0x52a0e286, 0xb79c5305, 0xaa500737, + 0x3e07841c, 0x7fdeae5c, 0x8e7d44ec, 0x5716f2b8, 0xb03ada37, 0xf0500c0d, + 0xf01c1f04, 0x0200b3ff, 0xae0cf51a, 0x3cb574b2, 0x25837a58, 0xdc0921bd, + 0xd19113f9, 0x7ca92ff6, 0x94324773, 0x22f54701, 0x3ae5e581, 0x37c2dadc, + 0xc8b57634, 0x9af3dda7, 0xa9446146, 0x0fd0030e, 0xecc8c73e, 0xa4751e41, + 0xe238cd99, 0x3bea0e2f, 0x3280bba1, 0x183eb331, 0x4e548b38, 0x4f6db908, + 0x6f420d03, 0xf60a04bf, 0x2cb81290, 0x24977c79, 0x5679b072, 0xbcaf89af, + 0xde9a771f, 0xd9930810, 0xb38bae12, 0xdccf3f2e, 0x5512721f, 0x2e6b7124, + 0x501adde6, 0x9f84cd87, 0x7a584718, 0x7408da17, 0xbc9f9abc, 0xe94b7d8c, + 0xec7aec3a, 0xdb851dfa, 0x63094366, 0xc464c3d2, 0xef1c1847, 0x3215d908, + 0xdd433b37, 0x24c2ba16, 0x12a14d43, 0x2a65c451, 0x50940002, 0x133ae4dd, + 0x71dff89e, 0x10314e55, 0x81ac77d6, 0x5f11199b, 0x043556f1, 0xd7a3c76b, + 0x3c11183b, 0x5924a509, 0xf28fe6ed, 0x97f1fbfa, 0x9ebabf2c, 0x1e153c6e, + 0x86e34570, 0xeae96fb1, 0x860e5e0a, 0x5a3e2ab3, 0x771fe71c, 0x4e3d06fa, + 0x2965dcb9, 0x99e71d0f, 0x803e89d6, 0x5266c825, 0x2e4cc978, 0x9c10b36a, + 0xc6150eba, 0x94e2ea78, 0xa5fc3c53, 0x1e0a2df4, 0xf2f74ea7, 0x361d2b3d, + 0x1939260f, 0x19c27960, 0x5223a708, 0xf71312b6, 0xebadfe6e, 0xeac31f66, + 0xe3bc4595, 0xa67bc883, 0xb17f37d1, 0x018cff28, 0xc332ddef, 0xbe6c5aa5, + 0x65582185, 0x68ab9802, 0xeecea50f, 0xdb2f953b, 0x2aef7dad, 0x5b6e2f84, + 0x1521b628, 0x29076170, 0xecdd4775, 0x619f1510, 0x13cca830, 0xeb61bd96, + 0x0334fe1e, 0xaa0363cf, 0xb5735c90, 0x4c70a239, 0xd59e9e0b, 0xcbaade14, + 0xeecc86bc, 0x60622ca7, 0x9cab5cab, 0xb2f3846e, 0x648b1eaf, 0x19bdf0ca, + 0xa02369b9, 0x655abb50, 0x40685a32, 0x3c2ab4b3, 0x319ee9d5, 0xc021b8f7, + 0x9b540b19, 0x875fa099, 0x95f7997e, 0x623d7da8, 0xf837889a, 0x97e32d77, + 0x11ed935f, 0x16681281, 0x0e358829, 0xc7e61fd6, 0x96dedfa1, 0x7858ba99, + 0x57f584a5, 0x1b227263, 0x9b83c3ff, 0x1ac24696, 0xcdb30aeb, 0x532e3054, + 0x8fd948e4, 0x6dbc3128, 0x58ebf2ef, 0x34c6ffea, 0xfe28ed61, 0xee7c3c73, + 0x5d4a14d9, 0xe864b7e3, 0x42105d14, 0x203e13e0, 0x45eee2b6, 0xa3aaabea, + 0xdb6c4f15, 0xfacb4fd0, 0xc742f442, 0xef6abbb5, 0x654f3b1d, 0x41cd2105, + 0xd81e799e, 0x86854dc7, 0xe44b476a, 0x3d816250, 0xcf62a1f2, 0x5b8d2646, + 0xfc8883a0, 0xc1c7b6a3, 0x7f1524c3, 0x69cb7492, 0x47848a0b, 0x5692b285, + 0x095bbf00, 0xad19489d, 0x1462b174, 0x23820e00, 0x58428d2a, 0x0c55f5ea, + 0x1dadf43e, 0x233f7061, 0x3372f092, 0x8d937e41, 0xd65fecf1, 0x6c223bdb, + 0x7cde3759, 0xcbee7460, 0x4085f2a7, 0xce77326e, 0xa6078084, 0x19f8509e, + 0xe8efd855, 0x61d99735, 0xa969a7aa, 0xc50c06c2, 0x5a04abfc, 0x800bcadc, + 0x9e447a2e, 0xc3453484, 0xfdd56705, 0x0e1e9ec9, 0xdb73dbd3, 0x105588cd, + 0x675fda79, 0xe3674340, 0xc5c43465, 0x713e38d8, 0x3d28f89e, 0xf16dff20, + 0x153e21e7, 0x8fb03d4a, 0xe6e39f2b, 0xdb83adf7, +} + +var s2 = [256]uint32{ + 0xe93d5a68, 0x948140f7, 0xf64c261c, 0x94692934, 0x411520f7, 0x7602d4f7, + 0xbcf46b2e, 0xd4a20068, 0xd4082471, 0x3320f46a, 0x43b7d4b7, 0x500061af, + 0x1e39f62e, 0x97244546, 0x14214f74, 0xbf8b8840, 0x4d95fc1d, 0x96b591af, + 0x70f4ddd3, 0x66a02f45, 0xbfbc09ec, 0x03bd9785, 0x7fac6dd0, 0x31cb8504, + 0x96eb27b3, 0x55fd3941, 0xda2547e6, 0xabca0a9a, 0x28507825, 0x530429f4, + 0x0a2c86da, 0xe9b66dfb, 0x68dc1462, 0xd7486900, 0x680ec0a4, 0x27a18dee, + 0x4f3ffea2, 0xe887ad8c, 0xb58ce006, 0x7af4d6b6, 0xaace1e7c, 0xd3375fec, + 0xce78a399, 0x406b2a42, 0x20fe9e35, 0xd9f385b9, 0xee39d7ab, 0x3b124e8b, + 0x1dc9faf7, 0x4b6d1856, 0x26a36631, 0xeae397b2, 0x3a6efa74, 0xdd5b4332, + 0x6841e7f7, 0xca7820fb, 0xfb0af54e, 0xd8feb397, 0x454056ac, 0xba489527, + 0x55533a3a, 0x20838d87, 0xfe6ba9b7, 0xd096954b, 0x55a867bc, 0xa1159a58, + 0xcca92963, 0x99e1db33, 0xa62a4a56, 0x3f3125f9, 0x5ef47e1c, 0x9029317c, + 0xfdf8e802, 0x04272f70, 0x80bb155c, 0x05282ce3, 0x95c11548, 0xe4c66d22, + 0x48c1133f, 0xc70f86dc, 0x07f9c9ee, 0x41041f0f, 0x404779a4, 0x5d886e17, + 0x325f51eb, 0xd59bc0d1, 0xf2bcc18f, 0x41113564, 0x257b7834, 0x602a9c60, + 0xdff8e8a3, 0x1f636c1b, 0x0e12b4c2, 0x02e1329e, 0xaf664fd1, 0xcad18115, + 0x6b2395e0, 0x333e92e1, 0x3b240b62, 0xeebeb922, 0x85b2a20e, 0xe6ba0d99, + 0xde720c8c, 0x2da2f728, 0xd0127845, 0x95b794fd, 0x647d0862, 0xe7ccf5f0, + 0x5449a36f, 0x877d48fa, 0xc39dfd27, 0xf33e8d1e, 0x0a476341, 0x992eff74, + 0x3a6f6eab, 0xf4f8fd37, 0xa812dc60, 0xa1ebddf8, 0x991be14c, 0xdb6e6b0d, + 0xc67b5510, 0x6d672c37, 0x2765d43b, 0xdcd0e804, 0xf1290dc7, 0xcc00ffa3, + 0xb5390f92, 0x690fed0b, 0x667b9ffb, 0xcedb7d9c, 0xa091cf0b, 0xd9155ea3, + 0xbb132f88, 0x515bad24, 0x7b9479bf, 0x763bd6eb, 0x37392eb3, 0xcc115979, + 0x8026e297, 0xf42e312d, 0x6842ada7, 0xc66a2b3b, 0x12754ccc, 0x782ef11c, + 0x6a124237, 0xb79251e7, 0x06a1bbe6, 0x4bfb6350, 0x1a6b1018, 0x11caedfa, + 0x3d25bdd8, 0xe2e1c3c9, 0x44421659, 0x0a121386, 0xd90cec6e, 0xd5abea2a, + 0x64af674e, 0xda86a85f, 0xbebfe988, 0x64e4c3fe, 0x9dbc8057, 0xf0f7c086, + 0x60787bf8, 0x6003604d, 0xd1fd8346, 0xf6381fb0, 0x7745ae04, 0xd736fccc, + 0x83426b33, 0xf01eab71, 0xb0804187, 0x3c005e5f, 0x77a057be, 0xbde8ae24, + 0x55464299, 0xbf582e61, 0x4e58f48f, 0xf2ddfda2, 0xf474ef38, 0x8789bdc2, + 0x5366f9c3, 0xc8b38e74, 0xb475f255, 0x46fcd9b9, 0x7aeb2661, 0x8b1ddf84, + 0x846a0e79, 0x915f95e2, 0x466e598e, 0x20b45770, 0x8cd55591, 0xc902de4c, + 0xb90bace1, 0xbb8205d0, 0x11a86248, 0x7574a99e, 0xb77f19b6, 0xe0a9dc09, + 0x662d09a1, 0xc4324633, 0xe85a1f02, 0x09f0be8c, 0x4a99a025, 0x1d6efe10, + 0x1ab93d1d, 0x0ba5a4df, 0xa186f20f, 0x2868f169, 0xdcb7da83, 0x573906fe, + 0xa1e2ce9b, 0x4fcd7f52, 0x50115e01, 0xa70683fa, 0xa002b5c4, 0x0de6d027, + 0x9af88c27, 0x773f8641, 0xc3604c06, 0x61a806b5, 0xf0177a28, 0xc0f586e0, + 0x006058aa, 0x30dc7d62, 0x11e69ed7, 0x2338ea63, 0x53c2dd94, 0xc2c21634, + 0xbbcbee56, 0x90bcb6de, 0xebfc7da1, 0xce591d76, 0x6f05e409, 0x4b7c0188, + 0x39720a3d, 0x7c927c24, 0x86e3725f, 0x724d9db9, 0x1ac15bb4, 0xd39eb8fc, + 0xed545578, 0x08fca5b5, 0xd83d7cd3, 0x4dad0fc4, 0x1e50ef5e, 0xb161e6f8, + 0xa28514d9, 0x6c51133c, 0x6fd5c7e7, 0x56e14ec4, 0x362abfce, 0xddc6c837, + 0xd79a3234, 0x92638212, 0x670efa8e, 0x406000e0, +} + +var s3 = [256]uint32{ + 0x3a39ce37, 0xd3faf5cf, 0xabc27737, 0x5ac52d1b, 0x5cb0679e, 0x4fa33742, + 0xd3822740, 0x99bc9bbe, 0xd5118e9d, 0xbf0f7315, 0xd62d1c7e, 0xc700c47b, + 0xb78c1b6b, 0x21a19045, 0xb26eb1be, 0x6a366eb4, 0x5748ab2f, 0xbc946e79, + 0xc6a376d2, 0x6549c2c8, 0x530ff8ee, 0x468dde7d, 0xd5730a1d, 0x4cd04dc6, + 0x2939bbdb, 0xa9ba4650, 0xac9526e8, 0xbe5ee304, 0xa1fad5f0, 0x6a2d519a, + 0x63ef8ce2, 0x9a86ee22, 0xc089c2b8, 0x43242ef6, 0xa51e03aa, 0x9cf2d0a4, + 0x83c061ba, 0x9be96a4d, 0x8fe51550, 0xba645bd6, 0x2826a2f9, 0xa73a3ae1, + 0x4ba99586, 0xef5562e9, 0xc72fefd3, 0xf752f7da, 0x3f046f69, 0x77fa0a59, + 0x80e4a915, 0x87b08601, 0x9b09e6ad, 0x3b3ee593, 0xe990fd5a, 0x9e34d797, + 0x2cf0b7d9, 0x022b8b51, 0x96d5ac3a, 0x017da67d, 0xd1cf3ed6, 0x7c7d2d28, + 0x1f9f25cf, 0xadf2b89b, 0x5ad6b472, 0x5a88f54c, 0xe029ac71, 0xe019a5e6, + 0x47b0acfd, 0xed93fa9b, 0xe8d3c48d, 0x283b57cc, 0xf8d56629, 0x79132e28, + 0x785f0191, 0xed756055, 0xf7960e44, 0xe3d35e8c, 0x15056dd4, 0x88f46dba, + 0x03a16125, 0x0564f0bd, 0xc3eb9e15, 0x3c9057a2, 0x97271aec, 0xa93a072a, + 0x1b3f6d9b, 0x1e6321f5, 0xf59c66fb, 0x26dcf319, 0x7533d928, 0xb155fdf5, + 0x03563482, 0x8aba3cbb, 0x28517711, 0xc20ad9f8, 0xabcc5167, 0xccad925f, + 0x4de81751, 0x3830dc8e, 0x379d5862, 0x9320f991, 0xea7a90c2, 0xfb3e7bce, + 0x5121ce64, 0x774fbe32, 0xa8b6e37e, 0xc3293d46, 0x48de5369, 0x6413e680, + 0xa2ae0810, 0xdd6db224, 0x69852dfd, 0x09072166, 0xb39a460a, 0x6445c0dd, + 0x586cdecf, 0x1c20c8ae, 0x5bbef7dd, 0x1b588d40, 0xccd2017f, 0x6bb4e3bb, + 0xdda26a7e, 0x3a59ff45, 0x3e350a44, 0xbcb4cdd5, 0x72eacea8, 0xfa6484bb, + 0x8d6612ae, 0xbf3c6f47, 0xd29be463, 0x542f5d9e, 0xaec2771b, 0xf64e6370, + 0x740e0d8d, 0xe75b1357, 0xf8721671, 0xaf537d5d, 0x4040cb08, 0x4eb4e2cc, + 0x34d2466a, 0x0115af84, 0xe1b00428, 0x95983a1d, 0x06b89fb4, 0xce6ea048, + 0x6f3f3b82, 0x3520ab82, 0x011a1d4b, 0x277227f8, 0x611560b1, 0xe7933fdc, + 0xbb3a792b, 0x344525bd, 0xa08839e1, 0x51ce794b, 0x2f32c9b7, 0xa01fbac9, + 0xe01cc87e, 0xbcc7d1f6, 0xcf0111c3, 0xa1e8aac7, 0x1a908749, 0xd44fbd9a, + 0xd0dadecb, 0xd50ada38, 0x0339c32a, 0xc6913667, 0x8df9317c, 0xe0b12b4f, + 0xf79e59b7, 0x43f5bb3a, 0xf2d519ff, 0x27d9459c, 0xbf97222c, 0x15e6fc2a, + 0x0f91fc71, 0x9b941525, 0xfae59361, 0xceb69ceb, 0xc2a86459, 0x12baa8d1, + 0xb6c1075e, 0xe3056a0c, 0x10d25065, 0xcb03a442, 0xe0ec6e0e, 0x1698db3b, + 0x4c98a0be, 0x3278e964, 0x9f1f9532, 0xe0d392df, 0xd3a0342b, 0x8971f21e, + 0x1b0a7441, 0x4ba3348c, 0xc5be7120, 0xc37632d8, 0xdf359f8d, 0x9b992f2e, + 0xe60b6f47, 0x0fe3f11d, 0xe54cda54, 0x1edad891, 0xce6279cf, 0xcd3e7e6f, + 0x1618b166, 0xfd2c1d05, 0x848fd2c5, 0xf6fb2299, 0xf523f357, 0xa6327623, + 0x93a83531, 0x56cccd02, 0xacf08162, 0x5a75ebb5, 0x6e163697, 0x88d273cc, + 0xde966292, 0x81b949d0, 0x4c50901b, 0x71c65614, 0xe6c6c7bd, 0x327a140a, + 0x45e1d006, 0xc3f27b9a, 0xc9aa53fd, 0x62a80f00, 0xbb25bfe2, 0x35bdd2f6, + 0x71126905, 0xb2040222, 0xb6cbcf7c, 0xcd769c2b, 0x53113ec0, 0x1640e3d3, + 0x38abbd60, 0x2547adf0, 0xba38209c, 0xf746ce76, 0x77afa1c5, 0x20756060, + 0x85cbfe4e, 0x8ae88dd8, 0x7aaaf9b0, 0x4cf9aa7e, 0x1948c25c, 0x02fb8a8c, + 0x01c36ae4, 0xd6ebe1f9, 0x90d4f869, 0xa65cdea0, 0x3f09252d, 0xc208e69f, + 0xb74e6132, 0xce77e25b, 0x578fdfe3, 0x3ac372e6, +} + +var p = [18]uint32{ + 0x243f6a88, 0x85a308d3, 0x13198a2e, 0x03707344, 0xa4093822, 0x299f31d0, + 0x082efa98, 0xec4e6c89, 0x452821e6, 0x38d01377, 0xbe5466cf, 0x34e90c6c, + 0xc0ac29b7, 0xc97c50dd, 0x3f84d5b5, 0xb5470917, 0x9216d5d9, 0x8979fb1b, +} diff --git a/vendor/gopkg.in/yaml.v3/decode.go b/vendor/gopkg.in/yaml.v3/decode.go index df36e3a3..0173b698 100644 --- a/vendor/gopkg.in/yaml.v3/decode.go +++ b/vendor/gopkg.in/yaml.v3/decode.go @@ -100,7 +100,10 @@ func (p *parser) peek() yaml_event_type_t { if p.event.typ != yaml_NO_EVENT { return p.event.typ } - if !yaml_parser_parse(&p.parser, &p.event) { + // It's curious choice from the underlying API to generally return a + // positive result on success, but on this case return true in an error + // scenario. This was the source of bugs in the past (issue #666). + if !yaml_parser_parse(&p.parser, &p.event) || p.parser.error != yaml_NO_ERROR { p.fail() } return p.event.typ @@ -320,6 +323,8 @@ type decoder struct { decodeCount int aliasCount int aliasDepth int + + mergedFields map[interface{}]bool } var ( @@ -808,6 +813,11 @@ func (d *decoder) mapping(n *Node, out reflect.Value) (good bool) { } } + mergedFields := d.mergedFields + d.mergedFields = nil + + var mergeNode *Node + mapIsNew := false if out.IsNil() { out.Set(reflect.MakeMap(outt)) @@ -815,11 +825,18 @@ func (d *decoder) mapping(n *Node, out reflect.Value) (good bool) { } for i := 0; i < l; i += 2 { if isMerge(n.Content[i]) { - d.merge(n.Content[i+1], out) + mergeNode = n.Content[i+1] continue } k := reflect.New(kt).Elem() if d.unmarshal(n.Content[i], k) { + if mergedFields != nil { + ki := k.Interface() + if mergedFields[ki] { + continue + } + mergedFields[ki] = true + } kkind := k.Kind() if kkind == reflect.Interface { kkind = k.Elem().Kind() @@ -833,6 +850,12 @@ func (d *decoder) mapping(n *Node, out reflect.Value) (good bool) { } } } + + d.mergedFields = mergedFields + if mergeNode != nil { + d.merge(n, mergeNode, out) + } + d.stringMapType = stringMapType d.generalMapType = generalMapType return true @@ -844,7 +867,8 @@ func isStringMap(n *Node) bool { } l := len(n.Content) for i := 0; i < l; i += 2 { - if n.Content[i].ShortTag() != strTag { + shortTag := n.Content[i].ShortTag() + if shortTag != strTag && shortTag != mergeTag { return false } } @@ -861,7 +885,6 @@ func (d *decoder) mappingStruct(n *Node, out reflect.Value) (good bool) { var elemType reflect.Type if sinfo.InlineMap != -1 { inlineMap = out.Field(sinfo.InlineMap) - inlineMap.Set(reflect.New(inlineMap.Type()).Elem()) elemType = inlineMap.Type().Elem() } @@ -870,6 +893,9 @@ func (d *decoder) mappingStruct(n *Node, out reflect.Value) (good bool) { d.prepare(n, field) } + mergedFields := d.mergedFields + d.mergedFields = nil + var mergeNode *Node var doneFields []bool if d.uniqueKeys { doneFields = make([]bool, len(sinfo.FieldsList)) @@ -879,13 +905,20 @@ func (d *decoder) mappingStruct(n *Node, out reflect.Value) (good bool) { for i := 0; i < l; i += 2 { ni := n.Content[i] if isMerge(ni) { - d.merge(n.Content[i+1], out) + mergeNode = n.Content[i+1] continue } if !d.unmarshal(ni, name) { continue } - if info, ok := sinfo.FieldsMap[name.String()]; ok { + sname := name.String() + if mergedFields != nil { + if mergedFields[sname] { + continue + } + mergedFields[sname] = true + } + if info, ok := sinfo.FieldsMap[sname]; ok { if d.uniqueKeys { if doneFields[info.Id] { d.terrors = append(d.terrors, fmt.Sprintf("line %d: field %s already set in type %s", ni.Line, name.String(), out.Type())) @@ -911,6 +944,11 @@ func (d *decoder) mappingStruct(n *Node, out reflect.Value) (good bool) { d.terrors = append(d.terrors, fmt.Sprintf("line %d: field %s not found in type %s", ni.Line, name.String(), out.Type())) } } + + d.mergedFields = mergedFields + if mergeNode != nil { + d.merge(n, mergeNode, out) + } return true } @@ -918,19 +956,29 @@ func failWantMap() { failf("map merge requires map or sequence of maps as the value") } -func (d *decoder) merge(n *Node, out reflect.Value) { - switch n.Kind { +func (d *decoder) merge(parent *Node, merge *Node, out reflect.Value) { + mergedFields := d.mergedFields + if mergedFields == nil { + d.mergedFields = make(map[interface{}]bool) + for i := 0; i < len(parent.Content); i += 2 { + k := reflect.New(ifaceType).Elem() + if d.unmarshal(parent.Content[i], k) { + d.mergedFields[k.Interface()] = true + } + } + } + + switch merge.Kind { case MappingNode: - d.unmarshal(n, out) + d.unmarshal(merge, out) case AliasNode: - if n.Alias != nil && n.Alias.Kind != MappingNode { + if merge.Alias != nil && merge.Alias.Kind != MappingNode { failWantMap() } - d.unmarshal(n, out) + d.unmarshal(merge, out) case SequenceNode: - // Step backwards as earlier nodes take precedence. - for i := len(n.Content) - 1; i >= 0; i-- { - ni := n.Content[i] + for i := 0; i < len(merge.Content); i++ { + ni := merge.Content[i] if ni.Kind == AliasNode { if ni.Alias != nil && ni.Alias.Kind != MappingNode { failWantMap() @@ -943,6 +991,8 @@ func (d *decoder) merge(n *Node, out reflect.Value) { default: failWantMap() } + + d.mergedFields = mergedFields } func isMerge(n *Node) bool { diff --git a/vendor/gopkg.in/yaml.v3/parserc.go b/vendor/gopkg.in/yaml.v3/parserc.go index ac66fccc..268558a0 100644 --- a/vendor/gopkg.in/yaml.v3/parserc.go +++ b/vendor/gopkg.in/yaml.v3/parserc.go @@ -687,6 +687,9 @@ func yaml_parser_parse_node(parser *yaml_parser_t, event *yaml_event_t, block, i func yaml_parser_parse_block_sequence_entry(parser *yaml_parser_t, event *yaml_event_t, first bool) bool { if first { token := peek_token(parser) + if token == nil { + return false + } parser.marks = append(parser.marks, token.start_mark) skip_token(parser) } @@ -786,7 +789,7 @@ func yaml_parser_split_stem_comment(parser *yaml_parser_t, stem_len int) { } token := peek_token(parser) - if token.typ != yaml_BLOCK_SEQUENCE_START_TOKEN && token.typ != yaml_BLOCK_MAPPING_START_TOKEN { + if token == nil || token.typ != yaml_BLOCK_SEQUENCE_START_TOKEN && token.typ != yaml_BLOCK_MAPPING_START_TOKEN { return } @@ -813,6 +816,9 @@ func yaml_parser_split_stem_comment(parser *yaml_parser_t, stem_len int) { func yaml_parser_parse_block_mapping_key(parser *yaml_parser_t, event *yaml_event_t, first bool) bool { if first { token := peek_token(parser) + if token == nil { + return false + } parser.marks = append(parser.marks, token.start_mark) skip_token(parser) } @@ -922,6 +928,9 @@ func yaml_parser_parse_block_mapping_value(parser *yaml_parser_t, event *yaml_ev func yaml_parser_parse_flow_sequence_entry(parser *yaml_parser_t, event *yaml_event_t, first bool) bool { if first { token := peek_token(parser) + if token == nil { + return false + } parser.marks = append(parser.marks, token.start_mark) skip_token(parser) } diff --git a/vendor/modules.txt b/vendor/modules.txt index 4dc19682..b74a0c5a 100644 --- a/vendor/modules.txt +++ b/vendor/modules.txt @@ -1,6 +1,9 @@ # github.com/OneOfOne/xxhash v1.2.8 ## explicit; go 1.11 github.com/OneOfOne/xxhash +# github.com/agnivade/levenshtein v1.1.1 +## explicit; go 1.13 +github.com/agnivade/levenshtein # github.com/beorn7/perks v1.0.1 ## explicit; go 1.11 github.com/beorn7/perks/quantile @@ -10,6 +13,8 @@ github.com/cespare/xxhash/v2 # github.com/davecgh/go-spew v1.1.1 ## explicit github.com/davecgh/go-spew/spew +# github.com/dgraph-io/badger/v3 v3.2103.3 +## explicit; go 1.12 # github.com/docker/go-units v0.5.0 ## explicit github.com/docker/go-units @@ -30,16 +35,23 @@ github.com/gobwas/glob/util/strings ## explicit; go 1.9 github.com/golang/protobuf/proto github.com/golang/protobuf/ptypes/timestamp +# github.com/google/flatbuffers v22.9.29+incompatible +## explicit +# github.com/google/go-cmp v0.5.9 +## explicit; go 1.13 +# github.com/klauspost/compress v1.15.11 +## explicit; go 1.17 # github.com/matttproud/golang_protobuf_extensions v1.0.2 ## explicit; go 1.9 github.com/matttproud/golang_protobuf_extensions/pbutil -# github.com/open-policy-agent/opa v0.36.1 -## explicit; go 1.16 +# github.com/open-policy-agent/opa v0.45.0 +## explicit; go 1.17 github.com/open-policy-agent/opa/ast github.com/open-policy-agent/opa/ast/internal/scanner github.com/open-policy-agent/opa/ast/internal/tokens github.com/open-policy-agent/opa/ast/location github.com/open-policy-agent/opa/bundle +github.com/open-policy-agent/opa/capabilities github.com/open-policy-agent/opa/format github.com/open-policy-agent/opa/internal/bundle github.com/open-policy-agent/opa/internal/cidr/merge @@ -51,7 +63,13 @@ github.com/open-policy-agent/opa/internal/file/archive github.com/open-policy-agent/opa/internal/file/url github.com/open-policy-agent/opa/internal/future github.com/open-policy-agent/opa/internal/gojsonschema -github.com/open-policy-agent/opa/internal/ir +github.com/open-policy-agent/opa/internal/gqlparser/ast +github.com/open-policy-agent/opa/internal/gqlparser/gqlerror +github.com/open-policy-agent/opa/internal/gqlparser/lexer +github.com/open-policy-agent/opa/internal/gqlparser/parser +github.com/open-policy-agent/opa/internal/gqlparser/validator +github.com/open-policy-agent/opa/internal/gqlparser/validator/rules +github.com/open-policy-agent/opa/internal/json/patch github.com/open-policy-agent/opa/internal/jwx/buffer github.com/open-policy-agent/opa/internal/jwx/jwa github.com/open-policy-agent/opa/internal/jwx/jwk @@ -62,6 +80,7 @@ github.com/open-policy-agent/opa/internal/lcss github.com/open-policy-agent/opa/internal/leb128 github.com/open-policy-agent/opa/internal/merge github.com/open-policy-agent/opa/internal/planner +github.com/open-policy-agent/opa/internal/ref github.com/open-policy-agent/opa/internal/rego/opa github.com/open-policy-agent/opa/internal/semver github.com/open-policy-agent/opa/internal/strings @@ -75,8 +94,10 @@ github.com/open-policy-agent/opa/internal/wasm/opcode github.com/open-policy-agent/opa/internal/wasm/sdk/opa/capabilities github.com/open-policy-agent/opa/internal/wasm/types github.com/open-policy-agent/opa/internal/wasm/util +github.com/open-policy-agent/opa/ir github.com/open-policy-agent/opa/keys github.com/open-policy-agent/opa/loader +github.com/open-policy-agent/opa/loader/filter github.com/open-policy-agent/opa/metrics github.com/open-policy-agent/opa/rego github.com/open-policy-agent/opa/resolver @@ -111,7 +132,7 @@ github.com/prometheus/client_golang/prometheus/promauto github.com/prometheus/client_golang/prometheus/promhttp github.com/prometheus/client_golang/prometheus/testutil github.com/prometheus/client_golang/prometheus/testutil/promlint -# github.com/prometheus/client_model v0.2.0 +# github.com/prometheus/client_model v0.3.0 ## explicit; go 1.9 github.com/prometheus/client_model/go # github.com/prometheus/common v0.37.0 @@ -124,23 +145,32 @@ github.com/prometheus/common/model github.com/prometheus/procfs github.com/prometheus/procfs/internal/fs github.com/prometheus/procfs/internal/util -# github.com/rcrowley/go-metrics v0.0.0-20200313005456-10cdbea86bc0 +# github.com/rcrowley/go-metrics v0.0.0-20201227073835-cf1acfcdf475 ## explicit github.com/rcrowley/go-metrics -# github.com/stretchr/testify v1.7.0 +# github.com/stretchr/testify v1.8.0 ## explicit; go 1.13 github.com/stretchr/testify/assert github.com/stretchr/testify/require +# github.com/tchap/go-patricia/v2 v2.3.1 +## explicit; go 1.16 +github.com/tchap/go-patricia/v2/patricia # github.com/xeipuuv/gojsonpointer v0.0.0-20190905194746-02993c407bfb ## explicit github.com/xeipuuv/gojsonpointer # github.com/xeipuuv/gojsonreference v0.0.0-20180127040603-bd5ef7bd5415 ## explicit github.com/xeipuuv/gojsonreference -# github.com/yashtewari/glob-intersection v0.0.0-20180916065949-5c77d914dd0b -## explicit +# github.com/yashtewari/glob-intersection v0.1.0 +## explicit; go 1.17 github.com/yashtewari/glob-intersection -# golang.org/x/sys v0.0.0-20221013171732-95e765b1cc43 +# golang.org/x/crypto v0.0.0-20220525230936-793ad666bf5e +## explicit; go 1.17 +golang.org/x/crypto/bcrypt +golang.org/x/crypto/blowfish +# golang.org/x/net v0.1.0 +## explicit; go 1.17 +# golang.org/x/sys v0.1.0 ## explicit; go 1.17 golang.org/x/sys/internal/unsafeheader golang.org/x/sys/unix @@ -178,6 +208,6 @@ google.golang.org/protobuf/types/known/timestamppb # gopkg.in/yaml.v2 v2.4.0 ## explicit; go 1.15 gopkg.in/yaml.v2 -# gopkg.in/yaml.v3 v3.0.0-20210107192922-496545a6307b +# gopkg.in/yaml.v3 v3.0.1 ## explicit gopkg.in/yaml.v3 From 552f2788258dfb45243bb9a94c078ff58fd27a26 Mon Sep 17 00:00:00 2001 From: Lucas Roesler Date: Sat, 3 Dec 2022 14:52:45 +0100 Subject: [PATCH 3/5] add oicd example policy Signed-off-by: Lucas Roesler --- auth/testdata/oidc.rego | 80 +++++++++++++++++++++++++++++++++++++++++ 1 file changed, 80 insertions(+) create mode 100644 auth/testdata/oidc.rego diff --git a/auth/testdata/oidc.rego b/auth/testdata/oidc.rego new file mode 100644 index 00000000..6319a5c3 --- /dev/null +++ b/auth/testdata/oidc.rego @@ -0,0 +1,80 @@ +package api.oidc + +import future.keywords.in + +# note that cache is per replica and will be lost when scaled down +metadata_discovery(issuer_url) = http.send({ + "url": concat("", [issuer_url, "/.well-known/openid-configuration"]), + "method": "GET", + "force_cache": true, + "force_cache_duration_seconds": 86400, +}).body + +# Cache response for 24 hours +jwks_request(url) = http.send({ + "url": url, + "method": "GET", + "force_cache": true, + "force_cache_duration_seconds": 3600, # Cache response for an hour +}) + +default response = false + +# Below we extract various data from the input + +# here load from a secret +# assume that you have a secret named issuers and it contains a +# json object of allow issuer urls +# { +# "https://url1": true, +# "https://url2": true, +# "https://url3": false +# } +# the function is then configured with OPA_INPUT_SECRETS=oidc_issuers +issuers := json.unmarshal(input.data.oidc_issuers) + +# the next two are simpler and loaded directly from env vars +# assume you have set OPA_INPUT_ALLOWED_DOMAINS='{*@company_name.com,*@company_name2.com}' +# should be a valid glob https://www.openpolicyagent.org/docs/latest/policy-reference/#glob +allowed_domains := split(input.allowed_domains, ",") + +# assume you have set OPA_INPUT_EMAIL_FIELD='email' +email_field := input.email_field + +# Api/user auth via oauth +# Verifies the provided JWT against a list of known providers. +# Extracts pre-configured fields from the token. All configurations +# for an issuer are checked against the available fields in the +# token. If more than one configuration matches, one is chosen +# arbitrarily. +response := res { + print("attempt client-credentials auth") + token := trim_prefix(input.auth_header, "Bearer ") + claims := io.jwt.decode(token)[1] + + print("Look up issuer in issuers list") + + # check is issuers is in the dict/object and is not false + issuers[claims.iss] + + print("fetch metadata") + metadata := metadata_discovery(claims.iss) + + print("fetch certificates to verify signature") + jwks_endpoint := metadata.jwks_uri + jwks := jwks_request(jwks_endpoint).raw_body + + print("verify signature") + opts := object.union(issuers[idx].token_data, {"cert": jwks}) + [verified, _, _] := io.jwt.decode_verify(token, opts) + + # stops here if false + verified + + # stops here if email is not in the claims + email = claims[email_field] + + # and it must match the allowed domains glob + # see https://www.openpolicyagent.org/docs/latest/policy-reference/#glob + glob.match(allowed_domains, null, email) +} From b39f804fc54d750817b5c2683b1bd2213802eff9 Mon Sep 17 00:00:00 2001 From: Lucas Roesler Date: Sat, 3 Dec 2022 18:43:17 +0100 Subject: [PATCH 4/5] add test to demonstrate jwt valdiation Signed-off-by: Lucas Roesler --- auth/local.go | 33 +++ auth/local_test.go | 66 +++++ auth/testdata/jwt.rego | 70 +++++ auth/testdata/oidc.rego | 26 +- go.mod | 1 + go.sum | 2 + .../github.com/golang-jwt/jwt/v4/.gitignore | 4 + vendor/github.com/golang-jwt/jwt/v4/LICENSE | 9 + .../golang-jwt/jwt/v4/MIGRATION_GUIDE.md | 22 ++ vendor/github.com/golang-jwt/jwt/v4/README.md | 138 +++++++++ .../github.com/golang-jwt/jwt/v4/SECURITY.md | 19 ++ .../golang-jwt/jwt/v4/VERSION_HISTORY.md | 135 +++++++++ vendor/github.com/golang-jwt/jwt/v4/claims.go | 269 ++++++++++++++++++ vendor/github.com/golang-jwt/jwt/v4/doc.go | 4 + vendor/github.com/golang-jwt/jwt/v4/ecdsa.go | 142 +++++++++ .../golang-jwt/jwt/v4/ecdsa_utils.go | 69 +++++ .../github.com/golang-jwt/jwt/v4/ed25519.go | 85 ++++++ .../golang-jwt/jwt/v4/ed25519_utils.go | 64 +++++ vendor/github.com/golang-jwt/jwt/v4/errors.go | 112 ++++++++ vendor/github.com/golang-jwt/jwt/v4/hmac.go | 95 +++++++ .../golang-jwt/jwt/v4/map_claims.go | 151 ++++++++++ vendor/github.com/golang-jwt/jwt/v4/none.go | 52 ++++ vendor/github.com/golang-jwt/jwt/v4/parser.go | 177 ++++++++++++ .../golang-jwt/jwt/v4/parser_option.go | 29 ++ vendor/github.com/golang-jwt/jwt/v4/rsa.go | 101 +++++++ .../github.com/golang-jwt/jwt/v4/rsa_pss.go | 143 ++++++++++ .../github.com/golang-jwt/jwt/v4/rsa_utils.go | 105 +++++++ .../golang-jwt/jwt/v4/signing_method.go | 46 +++ .../golang-jwt/jwt/v4/staticcheck.conf | 1 + vendor/github.com/golang-jwt/jwt/v4/token.go | 132 +++++++++ vendor/github.com/golang-jwt/jwt/v4/types.go | 145 ++++++++++ vendor/modules.txt | 3 + 32 files changed, 2442 insertions(+), 8 deletions(-) create mode 100644 auth/testdata/jwt.rego create mode 100644 vendor/github.com/golang-jwt/jwt/v4/.gitignore create mode 100644 vendor/github.com/golang-jwt/jwt/v4/LICENSE create mode 100644 vendor/github.com/golang-jwt/jwt/v4/MIGRATION_GUIDE.md create mode 100644 vendor/github.com/golang-jwt/jwt/v4/README.md create mode 100644 vendor/github.com/golang-jwt/jwt/v4/SECURITY.md create mode 100644 vendor/github.com/golang-jwt/jwt/v4/VERSION_HISTORY.md create mode 100644 vendor/github.com/golang-jwt/jwt/v4/claims.go create mode 100644 vendor/github.com/golang-jwt/jwt/v4/doc.go create mode 100644 vendor/github.com/golang-jwt/jwt/v4/ecdsa.go create mode 100644 vendor/github.com/golang-jwt/jwt/v4/ecdsa_utils.go create mode 100644 vendor/github.com/golang-jwt/jwt/v4/ed25519.go create mode 100644 vendor/github.com/golang-jwt/jwt/v4/ed25519_utils.go create mode 100644 vendor/github.com/golang-jwt/jwt/v4/errors.go create mode 100644 vendor/github.com/golang-jwt/jwt/v4/hmac.go create mode 100644 vendor/github.com/golang-jwt/jwt/v4/map_claims.go create mode 100644 vendor/github.com/golang-jwt/jwt/v4/none.go create mode 100644 vendor/github.com/golang-jwt/jwt/v4/parser.go create mode 100644 vendor/github.com/golang-jwt/jwt/v4/parser_option.go create mode 100644 vendor/github.com/golang-jwt/jwt/v4/rsa.go create mode 100644 vendor/github.com/golang-jwt/jwt/v4/rsa_pss.go create mode 100644 vendor/github.com/golang-jwt/jwt/v4/rsa_utils.go create mode 100644 vendor/github.com/golang-jwt/jwt/v4/signing_method.go create mode 100644 vendor/github.com/golang-jwt/jwt/v4/staticcheck.conf create mode 100644 vendor/github.com/golang-jwt/jwt/v4/token.go create mode 100644 vendor/github.com/golang-jwt/jwt/v4/types.go diff --git a/auth/local.go b/auth/local.go index 23275309..dd055029 100644 --- a/auth/local.go +++ b/auth/local.go @@ -58,6 +58,15 @@ func OPAConfigFromEnv() (cfg OPAConfig) { return cfg } +// NewLocalAuthorizer creates a OPA Authorizer instance for the given Policy. +// +// This method also exposes custom functions for policies to use. Currently +// it exposes: +// - bcrypt_eq +// - constant_compare +// +// Additionally, it modifies the logging so that it will use the default log writer +// when the OPA_DEBUG environment variable is set to true. func NewLocalAuthorizer(policy Policy, cfg OPAConfig) (_ Authorizer, err error) { auth := opa{ cfg: cfg, @@ -67,6 +76,8 @@ func NewLocalAuthorizer(policy Policy, cfg OPAConfig) (_ Authorizer, err error) policy, rego.Function2( ®o.Function{ + // expose bcrypt.CompareHashAndPassword to policies + // so that they can do do secure basic auth Name: "bcrypt_eq", Decl: types.NewFunction(types.Args(types.S, types.S), types.B), }, @@ -87,6 +98,8 @@ func NewLocalAuthorizer(policy Policy, cfg OPAConfig) (_ Authorizer, err error) ), rego.Function2( ®o.Function{ + // expose subtle.constant_compare to policies + // so that they can do secure string comparisons Name: "constant_compare", Decl: types.NewFunction(types.Args(types.S, types.S), types.B), }, @@ -116,17 +129,37 @@ type opa struct { cfg OPAConfig } +// Allowed implements the Authorizer interface and validates the given input against +// the configured OPA policy. func (a opa) Allowed(ctx context.Context, input Input) (_ bool, err error) { result, err := a.query.Eval(ctx, rego.EvalInput(input)) if err != nil { return false, fmt.Errorf("can not evaluate OPA query: %w", err) } + // this block allows us to inspect the result + // it will also be useful if we want to support + // complex result sets. See the TODO below. if a.cfg.Debug { data, _ := json.Marshal(result) log.Printf("OPA query result: %s", string(data)) + + if len(result) == 1 && len(result[0].Bindings) == 0 { + if exprs := result[0].Expressions; len(exprs) == 1 { + value := exprs[0].Value + log.Printf("OPA result value: %v\n OPA Result type: %T", value, value) + } + } + } + // this only allows policies that have a single boolean result + // policies that return complex objects will be treated as false + // + // TODO: allow policies to return complex objects which are then + // integrated into the request as X_AUTH headers. + // this will allow, for example, policies to pass information + // parsed from token. return result.Allowed(), nil } diff --git a/auth/local_test.go b/auth/local_test.go index cd2f376a..eec51d83 100644 --- a/auth/local_test.go +++ b/auth/local_test.go @@ -11,6 +11,7 @@ import ( "testing" "time" + "github.com/golang-jwt/jwt/v4" "github.com/stretchr/testify/require" _ "embed" @@ -20,6 +21,8 @@ func TestOPAAuthorizer(t *testing.T) { ctx, cancel := context.WithTimeout(context.Background(), 2*time.Second) defer cancel() + const jwtSecretKey = "secret" + cases := []struct { name string cfg OPAConfig @@ -172,6 +175,51 @@ func TestOPAAuthorizer(t *testing.T) { }, expected: true, }, + { + name: "can apply custom JWT auth policy", + cfg: OPAConfig{ + Debug: true, + Query: "data.api.jwt.allow", + }, + policy: "testdata/jwt.rego", + input: Input{ + Method: http.MethodPost, + Path: "/api/private", + RawBody: `{"message": "hello world"}`, + Authorization: bearerJWT(jwtSecretKey, map[string]interface{}{"sub": "alice", "email": "alice@test.example.com"}), + Data: map[string]string{ + "jwt_key": jwtSecretKey, + // allow tokens with gmail.com or test.example.com in the email field + "allowed_domains": `{**@gmail.com,**@test.example.com}`, + "email_field": "email", + }, + }, + expected: true, + }, + { + name: "JWT policy rejects non-matching domains", + cfg: OPAConfig{ + Debug: true, + Query: "data.api.jwt.allow", + }, + policy: "testdata/jwt.rego", + input: Input{ + Method: http.MethodPost, + Path: "/api/private", + RawBody: `{"message": "hello world"}`, + Authorization: bearerJWT(jwtSecretKey, map[string]interface{}{"sub": "alice", "email": "alice@test.example.com"}), + Data: map[string]string{ + "jwt_key": jwtSecretKey, + // allow tokens with gmail.com or test.example.com in the email field + "allowed_domains": `{**@gmail.com,**@company.example.com}`, + "email_field": "email", + }, + }, + expected: false, + }, + // OICD auth can be seen in the testdata/oidc.rego file + // but is not possible to include the unit tests because + // it requires a running OIDC server. } for _, tc := range cases { @@ -197,3 +245,21 @@ func generateHMAC(key, data string) string { return out } + +func generateTestJWT(key string, claims map[string]interface{}) (string, error) { + claims["exp"] = time.Now().Add(time.Hour).Unix() + claims["iat"] = time.Now().Unix() + claims["nbf"] = time.Now().Unix() + token := jwt.NewWithClaims(jwt.SigningMethodHS256, jwt.MapClaims(claims)) + + return token.SignedString([]byte(key)) +} + +func bearerJWT(key string, claim map[string]interface{}) string { + token, err := generateTestJWT(key, claim) + if err != nil { + panic(err) + } + + return "Bearer " + token +} diff --git a/auth/testdata/jwt.rego b/auth/testdata/jwt.rego new file mode 100644 index 00000000..217850bb --- /dev/null +++ b/auth/testdata/jwt.rego @@ -0,0 +1,70 @@ +# api.jwt represents any JWT based authentication +# this may or may not be a JWT from an OAuth provider. +# When possible, prefer OIDC when implementing OAuth +# flows. +package api.jwt + +default allow = false + +# Below we extract various data from the input + +# the next two are simpler and loaded directly from env vars +# assume you have set OPA_INPUT_ALLOWED_DOMAINS='{*@company_name.com,*@company_name2.com}' +# be careful with whitespace, +# {*@company_name.com,*@company_name2.com} is not equal to {*@company_name.com, *@company_name2.com} +# in the second, the space after the comma is part of the second glob pattern +# should be a valid glob https://www.openpolicyagent.org/docs/latest/policy-reference/#glob +allowed_domains := input.data.allowed_domains + +# assume you have set OPA_INPUT_EMAIL_FIELD='email' +email_field := input.data.email_field + +# alias the now value to make rego unit testing easier +now = value { + value := time.now_ns() +} + +# Api/user auth via oauth +# Verifies the provided JWT against a list of known providers. +# Extracts pre-configured fields from the token. All configurations +# for an issuer are checked against the available fields in the +# token. If more than one configuration matches, one is chosen +# arbitrarily. +allow { + print("attempt bearer token auth") + token := trim_prefix(input.authorization, "Bearer ") + + # here load from a secret + # the jwt key is loaded from a secret by setting OPA_INPUT_SECRETS=jwt_key + # you can also pass a JWKS set via the "cert". + # You can pass a PEM encoded key via the "cert" + # This is required for the RS*, PS*, and ES* algorithms. + # A full description of the options can be found + # https://www.openpolicyagent.org/docs/latest/policy-reference/#tokens + opts := { + # You can also pass a plaintext secret when using HS256, HS384 and HS512 verification. + "secret": input.data.jwt_key, + # The time in nanoseconds to verify the token at. If this is present then the exp and nbf claims are compared against this value. + "time": now, + } + + print("verify signature") + [verified, _, claims] := io.jwt.decode_verify(token, opts) + + # stops here if false + # print("verified: ", verified) + verified + + # print("claims: ", input) + # print("email_field: ", email_field) + + # stops here if email is not in the claims + email = claims[email_field] + + # print("found email: ", email) + # print("validating against glob: ", allowed_domains) + + # and it must match the allowed domains glob + # see https://www.openpolicyagent.org/docs/latest/policy-reference/#glob + glob.match(allowed_domains, null, email) +} diff --git a/auth/testdata/oidc.rego b/auth/testdata/oidc.rego index 6319a5c3..08a98236 100644 --- a/auth/testdata/oidc.rego +++ b/auth/testdata/oidc.rego @@ -1,7 +1,5 @@ package api.oidc -import future.keywords.in - # note that cache is per replica and will be lost when scaled down metadata_discovery(issuer_url) = http.send({ "url": concat("", [issuer_url, "/.well-known/openid-configuration"]), @@ -18,7 +16,7 @@ jwks_request(url) = http.send({ "force_cache_duration_seconds": 3600, # Cache response for an hour }) -default response = false +default allow = false # Below we extract various data from the input @@ -31,15 +29,21 @@ default response = false # "https://url3": false # } # the function is then configured with OPA_INPUT_SECRETS=oidc_issuers +# +# this could also be passed as a csv string of urls +# OPA_INPUT_OIDC_ISSUERS=https://url1,https://url2,https://url3 +# and then using +# issuers := split(input.data.oidc_issuers, ",") +# to get the list of issuers issuers := json.unmarshal(input.data.oidc_issuers) # the next two are simpler and loaded directly from env vars # assume you have set OPA_INPUT_ALLOWED_DOMAINS='{*@company_name.com,*@company_name2.com}' # should be a valid glob https://www.openpolicyagent.org/docs/latest/policy-reference/#glob -allowed_domains := split(input.allowed_domains, ",") +allowed_domains := input.data.allowed_domains # assume you have set OPA_INPUT_EMAIL_FIELD='email' -email_field := input.email_field +email_field := input.data.email_field # Api/user auth via oauth # Verifies the provided JWT against a list of known providers. @@ -47,14 +51,20 @@ email_field := input.email_field # for an issuer are checked against the available fields in the # token. If more than one configuration matches, one is chosen # arbitrarily. -response := res { +allow { print("attempt client-credentials auth") - token := trim_prefix(input.auth_header, "Bearer ") + token := trim_prefix(input.authorization, "Bearer ") + + # note that the token is _NOT_ verified yet, but we need + # to extract the issuer to perform the JWKS request and + # then verify the token. + # you should have a strict allow list of issuers otherwise + # this is insecure. claims := io.jwt.decode(token)[1] print("Look up issuer in issuers list") - # check is issuers is in the dict/object and is not false + # check is issuers is in the allow list dictionary and it is not false issuers[claims.iss] print("fetch metadata") diff --git a/go.mod b/go.mod index 324fd4bd..1388d2c6 100644 --- a/go.mod +++ b/go.mod @@ -4,6 +4,7 @@ go 1.18 require ( github.com/docker/go-units v0.5.0 + github.com/golang-jwt/jwt/v4 v4.4.3 github.com/open-policy-agent/opa v0.45.0 github.com/openfaas/faas-middleware v1.2.2 github.com/prometheus/client_golang v1.13.0 diff --git a/go.sum b/go.sum index 414b3829..f51a207b 100644 --- a/go.sum +++ b/go.sum @@ -106,6 +106,8 @@ github.com/gobwas/glob v0.2.3/go.mod h1:d3Ez4x06l9bZtSvzIay5+Yzi0fmZzPgnTbPcKjJA github.com/gogo/protobuf v1.1.1/go.mod h1:r8qH/GZQm5c6nD/R0oafs1akxWv10x8SbQlK7atdtwQ= github.com/gogo/protobuf v1.3.2 h1:Ov1cvc58UF3b5XjBnZv7+opcTcQFZebYjWzi34vdm4Q= github.com/gogo/protobuf v1.3.2/go.mod h1:P1XiOD3dCwIKUDQYPy72D8LYyHL2YPYrpS2s69NZV8Q= +github.com/golang-jwt/jwt/v4 v4.4.3 h1:Hxl6lhQFj4AnOX6MLrsCb/+7tCj7DxP7VA+2rDIq5AU= +github.com/golang-jwt/jwt/v4 v4.4.3/go.mod h1:m21LjoU+eqJr34lmDMbreY2eSTRJ1cv77w39/MY0Ch0= github.com/golang/glog v0.0.0-20160126235308-23def4e6c14b/go.mod h1:SBH7ygxi8pfUlaOkMMuAQtPIUF8ecWP5IEl/CR7VP2Q= github.com/golang/glog v1.0.0 h1:nfP3RFugxnNRyKgeWd4oI1nYvXpxrx8ck8ZrcizshdQ= github.com/golang/groupcache v0.0.0-20190702054246-869f871628b6/go.mod h1:cIg4eruTrX1D+g88fzRXU5OdNfaM+9IcxsU14FzY7Hc= diff --git a/vendor/github.com/golang-jwt/jwt/v4/.gitignore b/vendor/github.com/golang-jwt/jwt/v4/.gitignore new file mode 100644 index 00000000..09573e01 --- /dev/null +++ b/vendor/github.com/golang-jwt/jwt/v4/.gitignore @@ -0,0 +1,4 @@ +.DS_Store +bin +.idea/ + diff --git a/vendor/github.com/golang-jwt/jwt/v4/LICENSE b/vendor/github.com/golang-jwt/jwt/v4/LICENSE new file mode 100644 index 00000000..35dbc252 --- /dev/null +++ b/vendor/github.com/golang-jwt/jwt/v4/LICENSE @@ -0,0 +1,9 @@ +Copyright (c) 2012 Dave Grijalva +Copyright (c) 2021 golang-jwt maintainers + +Permission is hereby granted, free of charge, to any person obtaining a copy of this software and associated documentation files (the "Software"), to deal in the Software without restriction, including without limitation the rights to use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of the Software, and to permit persons to whom the Software is furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in all copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE. + diff --git a/vendor/github.com/golang-jwt/jwt/v4/MIGRATION_GUIDE.md b/vendor/github.com/golang-jwt/jwt/v4/MIGRATION_GUIDE.md new file mode 100644 index 00000000..32966f59 --- /dev/null +++ b/vendor/github.com/golang-jwt/jwt/v4/MIGRATION_GUIDE.md @@ -0,0 +1,22 @@ +## Migration Guide (v4.0.0) + +Starting from [v4.0.0](https://github.com/golang-jwt/jwt/releases/tag/v4.0.0), the import path will be: + + "github.com/golang-jwt/jwt/v4" + +The `/v4` version will be backwards compatible with existing `v3.x.y` tags in this repo, as well as +`github.com/dgrijalva/jwt-go`. For most users this should be a drop-in replacement, if you're having +troubles migrating, please open an issue. + +You can replace all occurrences of `github.com/dgrijalva/jwt-go` or `github.com/golang-jwt/jwt` with `github.com/golang-jwt/jwt/v4`, either manually or by using tools such as `sed` or `gofmt`. + +And then you'd typically run: + +``` +go get github.com/golang-jwt/jwt/v4 +go mod tidy +``` + +## Older releases (before v3.2.0) + +The original migration guide for older releases can be found at https://github.com/dgrijalva/jwt-go/blob/master/MIGRATION_GUIDE.md. diff --git a/vendor/github.com/golang-jwt/jwt/v4/README.md b/vendor/github.com/golang-jwt/jwt/v4/README.md new file mode 100644 index 00000000..30f2f2a6 --- /dev/null +++ b/vendor/github.com/golang-jwt/jwt/v4/README.md @@ -0,0 +1,138 @@ +# jwt-go + +[![build](https://github.com/golang-jwt/jwt/actions/workflows/build.yml/badge.svg)](https://github.com/golang-jwt/jwt/actions/workflows/build.yml) +[![Go Reference](https://pkg.go.dev/badge/github.com/golang-jwt/jwt/v4.svg)](https://pkg.go.dev/github.com/golang-jwt/jwt/v4) + +A [go](http://www.golang.org) (or 'golang' for search engine friendliness) implementation of [JSON Web Tokens](https://datatracker.ietf.org/doc/html/rfc7519). + +Starting with [v4.0.0](https://github.com/golang-jwt/jwt/releases/tag/v4.0.0) this project adds Go module support, but maintains backwards compatibility with older `v3.x.y` tags and upstream `github.com/dgrijalva/jwt-go`. +See the [`MIGRATION_GUIDE.md`](./MIGRATION_GUIDE.md) for more information. + +> After the original author of the library suggested migrating the maintenance of `jwt-go`, a dedicated team of open source maintainers decided to clone the existing library into this repository. See [dgrijalva/jwt-go#462](https://github.com/dgrijalva/jwt-go/issues/462) for a detailed discussion on this topic. + + +**SECURITY NOTICE:** Some older versions of Go have a security issue in the crypto/elliptic. Recommendation is to upgrade to at least 1.15 See issue [dgrijalva/jwt-go#216](https://github.com/dgrijalva/jwt-go/issues/216) for more detail. + +**SECURITY NOTICE:** It's important that you [validate the `alg` presented is what you expect](https://auth0.com/blog/critical-vulnerabilities-in-json-web-token-libraries/). This library attempts to make it easy to do the right thing by requiring key types match the expected alg, but you should take the extra step to verify it in your usage. See the examples provided. + +### Supported Go versions + +Our support of Go versions is aligned with Go's [version release policy](https://golang.org/doc/devel/release#policy). +So we will support a major version of Go until there are two newer major releases. +We no longer support building jwt-go with unsupported Go versions, as these contain security vulnerabilities +which will not be fixed. + +## What the heck is a JWT? + +JWT.io has [a great introduction](https://jwt.io/introduction) to JSON Web Tokens. + +In short, it's a signed JSON object that does something useful (for example, authentication). It's commonly used for `Bearer` tokens in Oauth 2. A token is made of three parts, separated by `.`'s. The first two parts are JSON objects, that have been [base64url](https://datatracker.ietf.org/doc/html/rfc4648) encoded. The last part is the signature, encoded the same way. + +The first part is called the header. It contains the necessary information for verifying the last part, the signature. For example, which encryption method was used for signing and what key was used. + +The part in the middle is the interesting bit. It's called the Claims and contains the actual stuff you care about. Refer to [RFC 7519](https://datatracker.ietf.org/doc/html/rfc7519) for information about reserved keys and the proper way to add your own. + +## What's in the box? + +This library supports the parsing and verification as well as the generation and signing of JWTs. Current supported signing algorithms are HMAC SHA, RSA, RSA-PSS, and ECDSA, though hooks are present for adding your own. + +## Installation Guidelines + +1. To install the jwt package, you first need to have [Go](https://go.dev/doc/install) installed, then you can use the command below to add `jwt-go` as a dependency in your Go program. + +```sh +go get -u github.com/golang-jwt/jwt/v4 +``` + +2. Import it in your code: + +```go +import "github.com/golang-jwt/jwt/v4" +``` + +## Examples + +See [the project documentation](https://pkg.go.dev/github.com/golang-jwt/jwt/v4) for examples of usage: + +* [Simple example of parsing and validating a token](https://pkg.go.dev/github.com/golang-jwt/jwt/v4#example-Parse-Hmac) +* [Simple example of building and signing a token](https://pkg.go.dev/github.com/golang-jwt/jwt/v4#example-New-Hmac) +* [Directory of Examples](https://pkg.go.dev/github.com/golang-jwt/jwt/v4#pkg-examples) + +## Extensions + +This library publishes all the necessary components for adding your own signing methods or key functions. Simply implement the `SigningMethod` interface and register a factory method using `RegisterSigningMethod` or provide a `jwt.Keyfunc`. + +A common use case would be integrating with different 3rd party signature providers, like key management services from various cloud providers or Hardware Security Modules (HSMs) or to implement additional standards. + +| Extension | Purpose | Repo | +| --------- | -------------------------------------------------------------------------------------------------------- | ------------------------------------------ | +| GCP | Integrates with multiple Google Cloud Platform signing tools (AppEngine, IAM API, Cloud KMS) | https://github.com/someone1/gcp-jwt-go | +| AWS | Integrates with AWS Key Management Service, KMS | https://github.com/matelang/jwt-go-aws-kms | +| JWKS | Provides support for JWKS ([RFC 7517](https://datatracker.ietf.org/doc/html/rfc7517)) as a `jwt.Keyfunc` | https://github.com/MicahParks/keyfunc | + +*Disclaimer*: Unless otherwise specified, these integrations are maintained by third parties and should not be considered as a primary offer by any of the mentioned cloud providers + +## Compliance + +This library was last reviewed to comply with [RFC 7519](https://datatracker.ietf.org/doc/html/rfc7519) dated May 2015 with a few notable differences: + +* In order to protect against accidental use of [Unsecured JWTs](https://datatracker.ietf.org/doc/html/rfc7519#section-6), tokens using `alg=none` will only be accepted if the constant `jwt.UnsafeAllowNoneSignatureType` is provided as the key. + +## Project Status & Versioning + +This library is considered production ready. Feedback and feature requests are appreciated. The API should be considered stable. There should be very few backwards-incompatible changes outside of major version updates (and only with good reason). + +This project uses [Semantic Versioning 2.0.0](http://semver.org). Accepted pull requests will land on `main`. Periodically, versions will be tagged from `main`. You can find all the releases on [the project releases page](https://github.com/golang-jwt/jwt/releases). + +**BREAKING CHANGES:*** +A full list of breaking changes is available in `VERSION_HISTORY.md`. See `MIGRATION_GUIDE.md` for more information on updating your code. + +## Usage Tips + +### Signing vs Encryption + +A token is simply a JSON object that is signed by its author. this tells you exactly two things about the data: + +* The author of the token was in the possession of the signing secret +* The data has not been modified since it was signed + +It's important to know that JWT does not provide encryption, which means anyone who has access to the token can read its contents. If you need to protect (encrypt) the data, there is a companion spec, `JWE`, that provides this functionality. The companion project https://github.com/golang-jwt/jwe aims at a (very) experimental implementation of the JWE standard. + +### Choosing a Signing Method + +There are several signing methods available, and you should probably take the time to learn about the various options before choosing one. The principal design decision is most likely going to be symmetric vs asymmetric. + +Symmetric signing methods, such as HSA, use only a single secret. This is probably the simplest signing method to use since any `[]byte` can be used as a valid secret. They are also slightly computationally faster to use, though this rarely is enough to matter. Symmetric signing methods work the best when both producers and consumers of tokens are trusted, or even the same system. Since the same secret is used to both sign and validate tokens, you can't easily distribute the key for validation. + +Asymmetric signing methods, such as RSA, use different keys for signing and verifying tokens. This makes it possible to produce tokens with a private key, and allow any consumer to access the public key for verification. + +### Signing Methods and Key Types + +Each signing method expects a different object type for its signing keys. See the package documentation for details. Here are the most common ones: + +* The [HMAC signing method](https://pkg.go.dev/github.com/golang-jwt/jwt/v4#SigningMethodHMAC) (`HS256`,`HS384`,`HS512`) expect `[]byte` values for signing and validation +* The [RSA signing method](https://pkg.go.dev/github.com/golang-jwt/jwt/v4#SigningMethodRSA) (`RS256`,`RS384`,`RS512`) expect `*rsa.PrivateKey` for signing and `*rsa.PublicKey` for validation +* The [ECDSA signing method](https://pkg.go.dev/github.com/golang-jwt/jwt/v4#SigningMethodECDSA) (`ES256`,`ES384`,`ES512`) expect `*ecdsa.PrivateKey` for signing and `*ecdsa.PublicKey` for validation +* The [EdDSA signing method](https://pkg.go.dev/github.com/golang-jwt/jwt/v4#SigningMethodEd25519) (`Ed25519`) expect `ed25519.PrivateKey` for signing and `ed25519.PublicKey` for validation + +### JWT and OAuth + +It's worth mentioning that OAuth and JWT are not the same thing. A JWT token is simply a signed JSON object. It can be used anywhere such a thing is useful. There is some confusion, though, as JWT is the most common type of bearer token used in OAuth2 authentication. + +Without going too far down the rabbit hole, here's a description of the interaction of these technologies: + +* OAuth is a protocol for allowing an identity provider to be separate from the service a user is logging in to. For example, whenever you use Facebook to log into a different service (Yelp, Spotify, etc), you are using OAuth. +* OAuth defines several options for passing around authentication data. One popular method is called a "bearer token". A bearer token is simply a string that _should_ only be held by an authenticated user. Thus, simply presenting this token proves your identity. You can probably derive from here why a JWT might make a good bearer token. +* Because bearer tokens are used for authentication, it's important they're kept secret. This is why transactions that use bearer tokens typically happen over SSL. + +### Troubleshooting + +This library uses descriptive error messages whenever possible. If you are not getting the expected result, have a look at the errors. The most common place people get stuck is providing the correct type of key to the parser. See the above section on signing methods and key types. + +## More + +Documentation can be found [on pkg.go.dev](https://pkg.go.dev/github.com/golang-jwt/jwt/v4). + +The command line utility included in this project (cmd/jwt) provides a straightforward example of token creation and parsing as well as a useful tool for debugging your own integration. You'll also find several implementation examples in the documentation. + +[golang-jwt](https://github.com/orgs/golang-jwt) incorporates a modified version of the JWT logo, which is distributed under the terms of the [MIT License](https://github.com/jsonwebtoken/jsonwebtoken.github.io/blob/master/LICENSE.txt). diff --git a/vendor/github.com/golang-jwt/jwt/v4/SECURITY.md b/vendor/github.com/golang-jwt/jwt/v4/SECURITY.md new file mode 100644 index 00000000..b08402c3 --- /dev/null +++ b/vendor/github.com/golang-jwt/jwt/v4/SECURITY.md @@ -0,0 +1,19 @@ +# Security Policy + +## Supported Versions + +As of February 2022 (and until this document is updated), the latest version `v4` is supported. + +## Reporting a Vulnerability + +If you think you found a vulnerability, and even if you are not sure, please report it to jwt-go-security@googlegroups.com or one of the other [golang-jwt maintainers](https://github.com/orgs/golang-jwt/people). Please try be explicit, describe steps to reproduce the security issue with code example(s). + +You will receive a response within a timely manner. If the issue is confirmed, we will do our best to release a patch as soon as possible given the complexity of the problem. + +## Public Discussions + +Please avoid publicly discussing a potential security vulnerability. + +Let's take this offline and find a solution first, this limits the potential impact as much as possible. + +We appreciate your help! diff --git a/vendor/github.com/golang-jwt/jwt/v4/VERSION_HISTORY.md b/vendor/github.com/golang-jwt/jwt/v4/VERSION_HISTORY.md new file mode 100644 index 00000000..afbfc4e4 --- /dev/null +++ b/vendor/github.com/golang-jwt/jwt/v4/VERSION_HISTORY.md @@ -0,0 +1,135 @@ +## `jwt-go` Version History + +#### 4.0.0 + +* Introduces support for Go modules. The `v4` version will be backwards compatible with `v3.x.y`. + +#### 3.2.2 + +* Starting from this release, we are adopting the policy to support the most 2 recent versions of Go currently available. By the time of this release, this is Go 1.15 and 1.16 ([#28](https://github.com/golang-jwt/jwt/pull/28)). +* Fixed a potential issue that could occur when the verification of `exp`, `iat` or `nbf` was not required and contained invalid contents, i.e. non-numeric/date. Thanks for @thaJeztah for making us aware of that and @giorgos-f3 for originally reporting it to the formtech fork ([#40](https://github.com/golang-jwt/jwt/pull/40)). +* Added support for EdDSA / ED25519 ([#36](https://github.com/golang-jwt/jwt/pull/36)). +* Optimized allocations ([#33](https://github.com/golang-jwt/jwt/pull/33)). + +#### 3.2.1 + +* **Import Path Change**: See MIGRATION_GUIDE.md for tips on updating your code + * Changed the import path from `github.com/dgrijalva/jwt-go` to `github.com/golang-jwt/jwt` +* Fixed type confusing issue between `string` and `[]string` in `VerifyAudience` ([#12](https://github.com/golang-jwt/jwt/pull/12)). This fixes CVE-2020-26160 + +#### 3.2.0 + +* Added method `ParseUnverified` to allow users to split up the tasks of parsing and validation +* HMAC signing method returns `ErrInvalidKeyType` instead of `ErrInvalidKey` where appropriate +* Added options to `request.ParseFromRequest`, which allows for an arbitrary list of modifiers to parsing behavior. Initial set include `WithClaims` and `WithParser`. Existing usage of this function will continue to work as before. +* Deprecated `ParseFromRequestWithClaims` to simplify API in the future. + +#### 3.1.0 + +* Improvements to `jwt` command line tool +* Added `SkipClaimsValidation` option to `Parser` +* Documentation updates + +#### 3.0.0 + +* **Compatibility Breaking Changes**: See MIGRATION_GUIDE.md for tips on updating your code + * Dropped support for `[]byte` keys when using RSA signing methods. This convenience feature could contribute to security vulnerabilities involving mismatched key types with signing methods. + * `ParseFromRequest` has been moved to `request` subpackage and usage has changed + * The `Claims` property on `Token` is now type `Claims` instead of `map[string]interface{}`. The default value is type `MapClaims`, which is an alias to `map[string]interface{}`. This makes it possible to use a custom type when decoding claims. +* Other Additions and Changes + * Added `Claims` interface type to allow users to decode the claims into a custom type + * Added `ParseWithClaims`, which takes a third argument of type `Claims`. Use this function instead of `Parse` if you have a custom type you'd like to decode into. + * Dramatically improved the functionality and flexibility of `ParseFromRequest`, which is now in the `request` subpackage + * Added `ParseFromRequestWithClaims` which is the `FromRequest` equivalent of `ParseWithClaims` + * Added new interface type `Extractor`, which is used for extracting JWT strings from http requests. Used with `ParseFromRequest` and `ParseFromRequestWithClaims`. + * Added several new, more specific, validation errors to error type bitmask + * Moved examples from README to executable example files + * Signing method registry is now thread safe + * Added new property to `ValidationError`, which contains the raw error returned by calls made by parse/verify (such as those returned by keyfunc or json parser) + +#### 2.7.0 + +This will likely be the last backwards compatible release before 3.0.0, excluding essential bug fixes. + +* Added new option `-show` to the `jwt` command that will just output the decoded token without verifying +* Error text for expired tokens includes how long it's been expired +* Fixed incorrect error returned from `ParseRSAPublicKeyFromPEM` +* Documentation updates + +#### 2.6.0 + +* Exposed inner error within ValidationError +* Fixed validation errors when using UseJSONNumber flag +* Added several unit tests + +#### 2.5.0 + +* Added support for signing method none. You shouldn't use this. The API tries to make this clear. +* Updated/fixed some documentation +* Added more helpful error message when trying to parse tokens that begin with `BEARER ` + +#### 2.4.0 + +* Added new type, Parser, to allow for configuration of various parsing parameters + * You can now specify a list of valid signing methods. Anything outside this set will be rejected. + * You can now opt to use the `json.Number` type instead of `float64` when parsing token JSON +* Added support for [Travis CI](https://travis-ci.org/dgrijalva/jwt-go) +* Fixed some bugs with ECDSA parsing + +#### 2.3.0 + +* Added support for ECDSA signing methods +* Added support for RSA PSS signing methods (requires go v1.4) + +#### 2.2.0 + +* Gracefully handle a `nil` `Keyfunc` being passed to `Parse`. Result will now be the parsed token and an error, instead of a panic. + +#### 2.1.0 + +Backwards compatible API change that was missed in 2.0.0. + +* The `SignedString` method on `Token` now takes `interface{}` instead of `[]byte` + +#### 2.0.0 + +There were two major reasons for breaking backwards compatibility with this update. The first was a refactor required to expand the width of the RSA and HMAC-SHA signing implementations. There will likely be no required code changes to support this change. + +The second update, while unfortunately requiring a small change in integration, is required to open up this library to other signing methods. Not all keys used for all signing methods have a single standard on-disk representation. Requiring `[]byte` as the type for all keys proved too limiting. Additionally, this implementation allows for pre-parsed tokens to be reused, which might matter in an application that parses a high volume of tokens with a small set of keys. Backwards compatibilty has been maintained for passing `[]byte` to the RSA signing methods, but they will also accept `*rsa.PublicKey` and `*rsa.PrivateKey`. + +It is likely the only integration change required here will be to change `func(t *jwt.Token) ([]byte, error)` to `func(t *jwt.Token) (interface{}, error)` when calling `Parse`. + +* **Compatibility Breaking Changes** + * `SigningMethodHS256` is now `*SigningMethodHMAC` instead of `type struct` + * `SigningMethodRS256` is now `*SigningMethodRSA` instead of `type struct` + * `KeyFunc` now returns `interface{}` instead of `[]byte` + * `SigningMethod.Sign` now takes `interface{}` instead of `[]byte` for the key + * `SigningMethod.Verify` now takes `interface{}` instead of `[]byte` for the key +* Renamed type `SigningMethodHS256` to `SigningMethodHMAC`. Specific sizes are now just instances of this type. + * Added public package global `SigningMethodHS256` + * Added public package global `SigningMethodHS384` + * Added public package global `SigningMethodHS512` +* Renamed type `SigningMethodRS256` to `SigningMethodRSA`. Specific sizes are now just instances of this type. + * Added public package global `SigningMethodRS256` + * Added public package global `SigningMethodRS384` + * Added public package global `SigningMethodRS512` +* Moved sample private key for HMAC tests from an inline value to a file on disk. Value is unchanged. +* Refactored the RSA implementation to be easier to read +* Exposed helper methods `ParseRSAPrivateKeyFromPEM` and `ParseRSAPublicKeyFromPEM` + +#### 1.0.2 + +* Fixed bug in parsing public keys from certificates +* Added more tests around the parsing of keys for RS256 +* Code refactoring in RS256 implementation. No functional changes + +#### 1.0.1 + +* Fixed panic if RS256 signing method was passed an invalid key + +#### 1.0.0 + +* First versioned release +* API stabilized +* Supports creating, signing, parsing, and validating JWT tokens +* Supports RS256 and HS256 signing methods diff --git a/vendor/github.com/golang-jwt/jwt/v4/claims.go b/vendor/github.com/golang-jwt/jwt/v4/claims.go new file mode 100644 index 00000000..364cec87 --- /dev/null +++ b/vendor/github.com/golang-jwt/jwt/v4/claims.go @@ -0,0 +1,269 @@ +package jwt + +import ( + "crypto/subtle" + "fmt" + "time" +) + +// Claims must just have a Valid method that determines +// if the token is invalid for any supported reason +type Claims interface { + Valid() error +} + +// RegisteredClaims are a structured version of the JWT Claims Set, +// restricted to Registered Claim Names, as referenced at +// https://datatracker.ietf.org/doc/html/rfc7519#section-4.1 +// +// This type can be used on its own, but then additional private and +// public claims embedded in the JWT will not be parsed. The typical usecase +// therefore is to embedded this in a user-defined claim type. +// +// See examples for how to use this with your own claim types. +type RegisteredClaims struct { + // the `iss` (Issuer) claim. See https://datatracker.ietf.org/doc/html/rfc7519#section-4.1.1 + Issuer string `json:"iss,omitempty"` + + // the `sub` (Subject) claim. See https://datatracker.ietf.org/doc/html/rfc7519#section-4.1.2 + Subject string `json:"sub,omitempty"` + + // the `aud` (Audience) claim. See https://datatracker.ietf.org/doc/html/rfc7519#section-4.1.3 + Audience ClaimStrings `json:"aud,omitempty"` + + // the `exp` (Expiration Time) claim. See https://datatracker.ietf.org/doc/html/rfc7519#section-4.1.4 + ExpiresAt *NumericDate `json:"exp,omitempty"` + + // the `nbf` (Not Before) claim. See https://datatracker.ietf.org/doc/html/rfc7519#section-4.1.5 + NotBefore *NumericDate `json:"nbf,omitempty"` + + // the `iat` (Issued At) claim. See https://datatracker.ietf.org/doc/html/rfc7519#section-4.1.6 + IssuedAt *NumericDate `json:"iat,omitempty"` + + // the `jti` (JWT ID) claim. See https://datatracker.ietf.org/doc/html/rfc7519#section-4.1.7 + ID string `json:"jti,omitempty"` +} + +// Valid validates time based claims "exp, iat, nbf". +// There is no accounting for clock skew. +// As well, if any of the above claims are not in the token, it will still +// be considered a valid claim. +func (c RegisteredClaims) Valid() error { + vErr := new(ValidationError) + now := TimeFunc() + + // The claims below are optional, by default, so if they are set to the + // default value in Go, let's not fail the verification for them. + if !c.VerifyExpiresAt(now, false) { + delta := now.Sub(c.ExpiresAt.Time) + vErr.Inner = fmt.Errorf("%s by %s", ErrTokenExpired, delta) + vErr.Errors |= ValidationErrorExpired + } + + if !c.VerifyIssuedAt(now, false) { + vErr.Inner = ErrTokenUsedBeforeIssued + vErr.Errors |= ValidationErrorIssuedAt + } + + if !c.VerifyNotBefore(now, false) { + vErr.Inner = ErrTokenNotValidYet + vErr.Errors |= ValidationErrorNotValidYet + } + + if vErr.valid() { + return nil + } + + return vErr +} + +// VerifyAudience compares the aud claim against cmp. +// If required is false, this method will return true if the value matches or is unset +func (c *RegisteredClaims) VerifyAudience(cmp string, req bool) bool { + return verifyAud(c.Audience, cmp, req) +} + +// VerifyExpiresAt compares the exp claim against cmp (cmp < exp). +// If req is false, it will return true, if exp is unset. +func (c *RegisteredClaims) VerifyExpiresAt(cmp time.Time, req bool) bool { + if c.ExpiresAt == nil { + return verifyExp(nil, cmp, req) + } + + return verifyExp(&c.ExpiresAt.Time, cmp, req) +} + +// VerifyIssuedAt compares the iat claim against cmp (cmp >= iat). +// If req is false, it will return true, if iat is unset. +func (c *RegisteredClaims) VerifyIssuedAt(cmp time.Time, req bool) bool { + if c.IssuedAt == nil { + return verifyIat(nil, cmp, req) + } + + return verifyIat(&c.IssuedAt.Time, cmp, req) +} + +// VerifyNotBefore compares the nbf claim against cmp (cmp >= nbf). +// If req is false, it will return true, if nbf is unset. +func (c *RegisteredClaims) VerifyNotBefore(cmp time.Time, req bool) bool { + if c.NotBefore == nil { + return verifyNbf(nil, cmp, req) + } + + return verifyNbf(&c.NotBefore.Time, cmp, req) +} + +// VerifyIssuer compares the iss claim against cmp. +// If required is false, this method will return true if the value matches or is unset +func (c *RegisteredClaims) VerifyIssuer(cmp string, req bool) bool { + return verifyIss(c.Issuer, cmp, req) +} + +// StandardClaims are a structured version of the JWT Claims Set, as referenced at +// https://datatracker.ietf.org/doc/html/rfc7519#section-4. They do not follow the +// specification exactly, since they were based on an earlier draft of the +// specification and not updated. The main difference is that they only +// support integer-based date fields and singular audiences. This might lead to +// incompatibilities with other JWT implementations. The use of this is discouraged, instead +// the newer RegisteredClaims struct should be used. +// +// Deprecated: Use RegisteredClaims instead for a forward-compatible way to access registered claims in a struct. +type StandardClaims struct { + Audience string `json:"aud,omitempty"` + ExpiresAt int64 `json:"exp,omitempty"` + Id string `json:"jti,omitempty"` + IssuedAt int64 `json:"iat,omitempty"` + Issuer string `json:"iss,omitempty"` + NotBefore int64 `json:"nbf,omitempty"` + Subject string `json:"sub,omitempty"` +} + +// Valid validates time based claims "exp, iat, nbf". There is no accounting for clock skew. +// As well, if any of the above claims are not in the token, it will still +// be considered a valid claim. +func (c StandardClaims) Valid() error { + vErr := new(ValidationError) + now := TimeFunc().Unix() + + // The claims below are optional, by default, so if they are set to the + // default value in Go, let's not fail the verification for them. + if !c.VerifyExpiresAt(now, false) { + delta := time.Unix(now, 0).Sub(time.Unix(c.ExpiresAt, 0)) + vErr.Inner = fmt.Errorf("%s by %s", ErrTokenExpired, delta) + vErr.Errors |= ValidationErrorExpired + } + + if !c.VerifyIssuedAt(now, false) { + vErr.Inner = ErrTokenUsedBeforeIssued + vErr.Errors |= ValidationErrorIssuedAt + } + + if !c.VerifyNotBefore(now, false) { + vErr.Inner = ErrTokenNotValidYet + vErr.Errors |= ValidationErrorNotValidYet + } + + if vErr.valid() { + return nil + } + + return vErr +} + +// VerifyAudience compares the aud claim against cmp. +// If required is false, this method will return true if the value matches or is unset +func (c *StandardClaims) VerifyAudience(cmp string, req bool) bool { + return verifyAud([]string{c.Audience}, cmp, req) +} + +// VerifyExpiresAt compares the exp claim against cmp (cmp < exp). +// If req is false, it will return true, if exp is unset. +func (c *StandardClaims) VerifyExpiresAt(cmp int64, req bool) bool { + if c.ExpiresAt == 0 { + return verifyExp(nil, time.Unix(cmp, 0), req) + } + + t := time.Unix(c.ExpiresAt, 0) + return verifyExp(&t, time.Unix(cmp, 0), req) +} + +// VerifyIssuedAt compares the iat claim against cmp (cmp >= iat). +// If req is false, it will return true, if iat is unset. +func (c *StandardClaims) VerifyIssuedAt(cmp int64, req bool) bool { + if c.IssuedAt == 0 { + return verifyIat(nil, time.Unix(cmp, 0), req) + } + + t := time.Unix(c.IssuedAt, 0) + return verifyIat(&t, time.Unix(cmp, 0), req) +} + +// VerifyNotBefore compares the nbf claim against cmp (cmp >= nbf). +// If req is false, it will return true, if nbf is unset. +func (c *StandardClaims) VerifyNotBefore(cmp int64, req bool) bool { + if c.NotBefore == 0 { + return verifyNbf(nil, time.Unix(cmp, 0), req) + } + + t := time.Unix(c.NotBefore, 0) + return verifyNbf(&t, time.Unix(cmp, 0), req) +} + +// VerifyIssuer compares the iss claim against cmp. +// If required is false, this method will return true if the value matches or is unset +func (c *StandardClaims) VerifyIssuer(cmp string, req bool) bool { + return verifyIss(c.Issuer, cmp, req) +} + +// ----- helpers + +func verifyAud(aud []string, cmp string, required bool) bool { + if len(aud) == 0 { + return !required + } + // use a var here to keep constant time compare when looping over a number of claims + result := false + + var stringClaims string + for _, a := range aud { + if subtle.ConstantTimeCompare([]byte(a), []byte(cmp)) != 0 { + result = true + } + stringClaims = stringClaims + a + } + + // case where "" is sent in one or many aud claims + if len(stringClaims) == 0 { + return !required + } + + return result +} + +func verifyExp(exp *time.Time, now time.Time, required bool) bool { + if exp == nil { + return !required + } + return now.Before(*exp) +} + +func verifyIat(iat *time.Time, now time.Time, required bool) bool { + if iat == nil { + return !required + } + return now.After(*iat) || now.Equal(*iat) +} + +func verifyNbf(nbf *time.Time, now time.Time, required bool) bool { + if nbf == nil { + return !required + } + return now.After(*nbf) || now.Equal(*nbf) +} + +func verifyIss(iss string, cmp string, required bool) bool { + if iss == "" { + return !required + } + return subtle.ConstantTimeCompare([]byte(iss), []byte(cmp)) != 0 +} diff --git a/vendor/github.com/golang-jwt/jwt/v4/doc.go b/vendor/github.com/golang-jwt/jwt/v4/doc.go new file mode 100644 index 00000000..a86dc1a3 --- /dev/null +++ b/vendor/github.com/golang-jwt/jwt/v4/doc.go @@ -0,0 +1,4 @@ +// Package jwt is a Go implementation of JSON Web Tokens: http://self-issued.info/docs/draft-jones-json-web-token.html +// +// See README.md for more info. +package jwt diff --git a/vendor/github.com/golang-jwt/jwt/v4/ecdsa.go b/vendor/github.com/golang-jwt/jwt/v4/ecdsa.go new file mode 100644 index 00000000..eac023fc --- /dev/null +++ b/vendor/github.com/golang-jwt/jwt/v4/ecdsa.go @@ -0,0 +1,142 @@ +package jwt + +import ( + "crypto" + "crypto/ecdsa" + "crypto/rand" + "errors" + "math/big" +) + +var ( + // Sadly this is missing from crypto/ecdsa compared to crypto/rsa + ErrECDSAVerification = errors.New("crypto/ecdsa: verification error") +) + +// SigningMethodECDSA implements the ECDSA family of signing methods. +// Expects *ecdsa.PrivateKey for signing and *ecdsa.PublicKey for verification +type SigningMethodECDSA struct { + Name string + Hash crypto.Hash + KeySize int + CurveBits int +} + +// Specific instances for EC256 and company +var ( + SigningMethodES256 *SigningMethodECDSA + SigningMethodES384 *SigningMethodECDSA + SigningMethodES512 *SigningMethodECDSA +) + +func init() { + // ES256 + SigningMethodES256 = &SigningMethodECDSA{"ES256", crypto.SHA256, 32, 256} + RegisterSigningMethod(SigningMethodES256.Alg(), func() SigningMethod { + return SigningMethodES256 + }) + + // ES384 + SigningMethodES384 = &SigningMethodECDSA{"ES384", crypto.SHA384, 48, 384} + RegisterSigningMethod(SigningMethodES384.Alg(), func() SigningMethod { + return SigningMethodES384 + }) + + // ES512 + SigningMethodES512 = &SigningMethodECDSA{"ES512", crypto.SHA512, 66, 521} + RegisterSigningMethod(SigningMethodES512.Alg(), func() SigningMethod { + return SigningMethodES512 + }) +} + +func (m *SigningMethodECDSA) Alg() string { + return m.Name +} + +// Verify implements token verification for the SigningMethod. +// For this verify method, key must be an ecdsa.PublicKey struct +func (m *SigningMethodECDSA) Verify(signingString, signature string, key interface{}) error { + var err error + + // Decode the signature + var sig []byte + if sig, err = DecodeSegment(signature); err != nil { + return err + } + + // Get the key + var ecdsaKey *ecdsa.PublicKey + switch k := key.(type) { + case *ecdsa.PublicKey: + ecdsaKey = k + default: + return ErrInvalidKeyType + } + + if len(sig) != 2*m.KeySize { + return ErrECDSAVerification + } + + r := big.NewInt(0).SetBytes(sig[:m.KeySize]) + s := big.NewInt(0).SetBytes(sig[m.KeySize:]) + + // Create hasher + if !m.Hash.Available() { + return ErrHashUnavailable + } + hasher := m.Hash.New() + hasher.Write([]byte(signingString)) + + // Verify the signature + if verifystatus := ecdsa.Verify(ecdsaKey, hasher.Sum(nil), r, s); verifystatus { + return nil + } + + return ErrECDSAVerification +} + +// Sign implements token signing for the SigningMethod. +// For this signing method, key must be an ecdsa.PrivateKey struct +func (m *SigningMethodECDSA) Sign(signingString string, key interface{}) (string, error) { + // Get the key + var ecdsaKey *ecdsa.PrivateKey + switch k := key.(type) { + case *ecdsa.PrivateKey: + ecdsaKey = k + default: + return "", ErrInvalidKeyType + } + + // Create the hasher + if !m.Hash.Available() { + return "", ErrHashUnavailable + } + + hasher := m.Hash.New() + hasher.Write([]byte(signingString)) + + // Sign the string and return r, s + if r, s, err := ecdsa.Sign(rand.Reader, ecdsaKey, hasher.Sum(nil)); err == nil { + curveBits := ecdsaKey.Curve.Params().BitSize + + if m.CurveBits != curveBits { + return "", ErrInvalidKey + } + + keyBytes := curveBits / 8 + if curveBits%8 > 0 { + keyBytes += 1 + } + + // We serialize the outputs (r and s) into big-endian byte arrays + // padded with zeros on the left to make sure the sizes work out. + // Output must be 2*keyBytes long. + out := make([]byte, 2*keyBytes) + r.FillBytes(out[0:keyBytes]) // r is assigned to the first half of output. + s.FillBytes(out[keyBytes:]) // s is assigned to the second half of output. + + return EncodeSegment(out), nil + } else { + return "", err + } +} diff --git a/vendor/github.com/golang-jwt/jwt/v4/ecdsa_utils.go b/vendor/github.com/golang-jwt/jwt/v4/ecdsa_utils.go new file mode 100644 index 00000000..5700636d --- /dev/null +++ b/vendor/github.com/golang-jwt/jwt/v4/ecdsa_utils.go @@ -0,0 +1,69 @@ +package jwt + +import ( + "crypto/ecdsa" + "crypto/x509" + "encoding/pem" + "errors" +) + +var ( + ErrNotECPublicKey = errors.New("key is not a valid ECDSA public key") + ErrNotECPrivateKey = errors.New("key is not a valid ECDSA private key") +) + +// ParseECPrivateKeyFromPEM parses a PEM encoded Elliptic Curve Private Key Structure +func ParseECPrivateKeyFromPEM(key []byte) (*ecdsa.PrivateKey, error) { + var err error + + // Parse PEM block + var block *pem.Block + if block, _ = pem.Decode(key); block == nil { + return nil, ErrKeyMustBePEMEncoded + } + + // Parse the key + var parsedKey interface{} + if parsedKey, err = x509.ParseECPrivateKey(block.Bytes); err != nil { + if parsedKey, err = x509.ParsePKCS8PrivateKey(block.Bytes); err != nil { + return nil, err + } + } + + var pkey *ecdsa.PrivateKey + var ok bool + if pkey, ok = parsedKey.(*ecdsa.PrivateKey); !ok { + return nil, ErrNotECPrivateKey + } + + return pkey, nil +} + +// ParseECPublicKeyFromPEM parses a PEM encoded PKCS1 or PKCS8 public key +func ParseECPublicKeyFromPEM(key []byte) (*ecdsa.PublicKey, error) { + var err error + + // Parse PEM block + var block *pem.Block + if block, _ = pem.Decode(key); block == nil { + return nil, ErrKeyMustBePEMEncoded + } + + // Parse the key + var parsedKey interface{} + if parsedKey, err = x509.ParsePKIXPublicKey(block.Bytes); err != nil { + if cert, err := x509.ParseCertificate(block.Bytes); err == nil { + parsedKey = cert.PublicKey + } else { + return nil, err + } + } + + var pkey *ecdsa.PublicKey + var ok bool + if pkey, ok = parsedKey.(*ecdsa.PublicKey); !ok { + return nil, ErrNotECPublicKey + } + + return pkey, nil +} diff --git a/vendor/github.com/golang-jwt/jwt/v4/ed25519.go b/vendor/github.com/golang-jwt/jwt/v4/ed25519.go new file mode 100644 index 00000000..07d3aacd --- /dev/null +++ b/vendor/github.com/golang-jwt/jwt/v4/ed25519.go @@ -0,0 +1,85 @@ +package jwt + +import ( + "errors" + + "crypto" + "crypto/ed25519" + "crypto/rand" +) + +var ( + ErrEd25519Verification = errors.New("ed25519: verification error") +) + +// SigningMethodEd25519 implements the EdDSA family. +// Expects ed25519.PrivateKey for signing and ed25519.PublicKey for verification +type SigningMethodEd25519 struct{} + +// Specific instance for EdDSA +var ( + SigningMethodEdDSA *SigningMethodEd25519 +) + +func init() { + SigningMethodEdDSA = &SigningMethodEd25519{} + RegisterSigningMethod(SigningMethodEdDSA.Alg(), func() SigningMethod { + return SigningMethodEdDSA + }) +} + +func (m *SigningMethodEd25519) Alg() string { + return "EdDSA" +} + +// Verify implements token verification for the SigningMethod. +// For this verify method, key must be an ed25519.PublicKey +func (m *SigningMethodEd25519) Verify(signingString, signature string, key interface{}) error { + var err error + var ed25519Key ed25519.PublicKey + var ok bool + + if ed25519Key, ok = key.(ed25519.PublicKey); !ok { + return ErrInvalidKeyType + } + + if len(ed25519Key) != ed25519.PublicKeySize { + return ErrInvalidKey + } + + // Decode the signature + var sig []byte + if sig, err = DecodeSegment(signature); err != nil { + return err + } + + // Verify the signature + if !ed25519.Verify(ed25519Key, []byte(signingString), sig) { + return ErrEd25519Verification + } + + return nil +} + +// Sign implements token signing for the SigningMethod. +// For this signing method, key must be an ed25519.PrivateKey +func (m *SigningMethodEd25519) Sign(signingString string, key interface{}) (string, error) { + var ed25519Key crypto.Signer + var ok bool + + if ed25519Key, ok = key.(crypto.Signer); !ok { + return "", ErrInvalidKeyType + } + + if _, ok := ed25519Key.Public().(ed25519.PublicKey); !ok { + return "", ErrInvalidKey + } + + // Sign the string and return the encoded result + // ed25519 performs a two-pass hash as part of its algorithm. Therefore, we need to pass a non-prehashed message into the Sign function, as indicated by crypto.Hash(0) + sig, err := ed25519Key.Sign(rand.Reader, []byte(signingString), crypto.Hash(0)) + if err != nil { + return "", err + } + return EncodeSegment(sig), nil +} diff --git a/vendor/github.com/golang-jwt/jwt/v4/ed25519_utils.go b/vendor/github.com/golang-jwt/jwt/v4/ed25519_utils.go new file mode 100644 index 00000000..cdb5e68e --- /dev/null +++ b/vendor/github.com/golang-jwt/jwt/v4/ed25519_utils.go @@ -0,0 +1,64 @@ +package jwt + +import ( + "crypto" + "crypto/ed25519" + "crypto/x509" + "encoding/pem" + "errors" +) + +var ( + ErrNotEdPrivateKey = errors.New("key is not a valid Ed25519 private key") + ErrNotEdPublicKey = errors.New("key is not a valid Ed25519 public key") +) + +// ParseEdPrivateKeyFromPEM parses a PEM-encoded Edwards curve private key +func ParseEdPrivateKeyFromPEM(key []byte) (crypto.PrivateKey, error) { + var err error + + // Parse PEM block + var block *pem.Block + if block, _ = pem.Decode(key); block == nil { + return nil, ErrKeyMustBePEMEncoded + } + + // Parse the key + var parsedKey interface{} + if parsedKey, err = x509.ParsePKCS8PrivateKey(block.Bytes); err != nil { + return nil, err + } + + var pkey ed25519.PrivateKey + var ok bool + if pkey, ok = parsedKey.(ed25519.PrivateKey); !ok { + return nil, ErrNotEdPrivateKey + } + + return pkey, nil +} + +// ParseEdPublicKeyFromPEM parses a PEM-encoded Edwards curve public key +func ParseEdPublicKeyFromPEM(key []byte) (crypto.PublicKey, error) { + var err error + + // Parse PEM block + var block *pem.Block + if block, _ = pem.Decode(key); block == nil { + return nil, ErrKeyMustBePEMEncoded + } + + // Parse the key + var parsedKey interface{} + if parsedKey, err = x509.ParsePKIXPublicKey(block.Bytes); err != nil { + return nil, err + } + + var pkey ed25519.PublicKey + var ok bool + if pkey, ok = parsedKey.(ed25519.PublicKey); !ok { + return nil, ErrNotEdPublicKey + } + + return pkey, nil +} diff --git a/vendor/github.com/golang-jwt/jwt/v4/errors.go b/vendor/github.com/golang-jwt/jwt/v4/errors.go new file mode 100644 index 00000000..10ac8835 --- /dev/null +++ b/vendor/github.com/golang-jwt/jwt/v4/errors.go @@ -0,0 +1,112 @@ +package jwt + +import ( + "errors" +) + +// Error constants +var ( + ErrInvalidKey = errors.New("key is invalid") + ErrInvalidKeyType = errors.New("key is of invalid type") + ErrHashUnavailable = errors.New("the requested hash function is unavailable") + + ErrTokenMalformed = errors.New("token is malformed") + ErrTokenUnverifiable = errors.New("token is unverifiable") + ErrTokenSignatureInvalid = errors.New("token signature is invalid") + + ErrTokenInvalidAudience = errors.New("token has invalid audience") + ErrTokenExpired = errors.New("token is expired") + ErrTokenUsedBeforeIssued = errors.New("token used before issued") + ErrTokenInvalidIssuer = errors.New("token has invalid issuer") + ErrTokenNotValidYet = errors.New("token is not valid yet") + ErrTokenInvalidId = errors.New("token has invalid id") + ErrTokenInvalidClaims = errors.New("token has invalid claims") +) + +// The errors that might occur when parsing and validating a token +const ( + ValidationErrorMalformed uint32 = 1 << iota // Token is malformed + ValidationErrorUnverifiable // Token could not be verified because of signing problems + ValidationErrorSignatureInvalid // Signature validation failed + + // Standard Claim validation errors + ValidationErrorAudience // AUD validation failed + ValidationErrorExpired // EXP validation failed + ValidationErrorIssuedAt // IAT validation failed + ValidationErrorIssuer // ISS validation failed + ValidationErrorNotValidYet // NBF validation failed + ValidationErrorId // JTI validation failed + ValidationErrorClaimsInvalid // Generic claims validation error +) + +// NewValidationError is a helper for constructing a ValidationError with a string error message +func NewValidationError(errorText string, errorFlags uint32) *ValidationError { + return &ValidationError{ + text: errorText, + Errors: errorFlags, + } +} + +// ValidationError represents an error from Parse if token is not valid +type ValidationError struct { + Inner error // stores the error returned by external dependencies, i.e.: KeyFunc + Errors uint32 // bitfield. see ValidationError... constants + text string // errors that do not have a valid error just have text +} + +// Error is the implementation of the err interface. +func (e ValidationError) Error() string { + if e.Inner != nil { + return e.Inner.Error() + } else if e.text != "" { + return e.text + } else { + return "token is invalid" + } +} + +// Unwrap gives errors.Is and errors.As access to the inner error. +func (e *ValidationError) Unwrap() error { + return e.Inner +} + +// No errors +func (e *ValidationError) valid() bool { + return e.Errors == 0 +} + +// Is checks if this ValidationError is of the supplied error. We are first checking for the exact error message +// by comparing the inner error message. If that fails, we compare using the error flags. This way we can use +// custom error messages (mainly for backwards compatability) and still leverage errors.Is using the global error variables. +func (e *ValidationError) Is(err error) bool { + // Check, if our inner error is a direct match + if errors.Is(errors.Unwrap(e), err) { + return true + } + + // Otherwise, we need to match using our error flags + switch err { + case ErrTokenMalformed: + return e.Errors&ValidationErrorMalformed != 0 + case ErrTokenUnverifiable: + return e.Errors&ValidationErrorUnverifiable != 0 + case ErrTokenSignatureInvalid: + return e.Errors&ValidationErrorSignatureInvalid != 0 + case ErrTokenInvalidAudience: + return e.Errors&ValidationErrorAudience != 0 + case ErrTokenExpired: + return e.Errors&ValidationErrorExpired != 0 + case ErrTokenUsedBeforeIssued: + return e.Errors&ValidationErrorIssuedAt != 0 + case ErrTokenInvalidIssuer: + return e.Errors&ValidationErrorIssuer != 0 + case ErrTokenNotValidYet: + return e.Errors&ValidationErrorNotValidYet != 0 + case ErrTokenInvalidId: + return e.Errors&ValidationErrorId != 0 + case ErrTokenInvalidClaims: + return e.Errors&ValidationErrorClaimsInvalid != 0 + } + + return false +} diff --git a/vendor/github.com/golang-jwt/jwt/v4/hmac.go b/vendor/github.com/golang-jwt/jwt/v4/hmac.go new file mode 100644 index 00000000..011f68a2 --- /dev/null +++ b/vendor/github.com/golang-jwt/jwt/v4/hmac.go @@ -0,0 +1,95 @@ +package jwt + +import ( + "crypto" + "crypto/hmac" + "errors" +) + +// SigningMethodHMAC implements the HMAC-SHA family of signing methods. +// Expects key type of []byte for both signing and validation +type SigningMethodHMAC struct { + Name string + Hash crypto.Hash +} + +// Specific instances for HS256 and company +var ( + SigningMethodHS256 *SigningMethodHMAC + SigningMethodHS384 *SigningMethodHMAC + SigningMethodHS512 *SigningMethodHMAC + ErrSignatureInvalid = errors.New("signature is invalid") +) + +func init() { + // HS256 + SigningMethodHS256 = &SigningMethodHMAC{"HS256", crypto.SHA256} + RegisterSigningMethod(SigningMethodHS256.Alg(), func() SigningMethod { + return SigningMethodHS256 + }) + + // HS384 + SigningMethodHS384 = &SigningMethodHMAC{"HS384", crypto.SHA384} + RegisterSigningMethod(SigningMethodHS384.Alg(), func() SigningMethod { + return SigningMethodHS384 + }) + + // HS512 + SigningMethodHS512 = &SigningMethodHMAC{"HS512", crypto.SHA512} + RegisterSigningMethod(SigningMethodHS512.Alg(), func() SigningMethod { + return SigningMethodHS512 + }) +} + +func (m *SigningMethodHMAC) Alg() string { + return m.Name +} + +// Verify implements token verification for the SigningMethod. Returns nil if the signature is valid. +func (m *SigningMethodHMAC) Verify(signingString, signature string, key interface{}) error { + // Verify the key is the right type + keyBytes, ok := key.([]byte) + if !ok { + return ErrInvalidKeyType + } + + // Decode signature, for comparison + sig, err := DecodeSegment(signature) + if err != nil { + return err + } + + // Can we use the specified hashing method? + if !m.Hash.Available() { + return ErrHashUnavailable + } + + // This signing method is symmetric, so we validate the signature + // by reproducing the signature from the signing string and key, then + // comparing that against the provided signature. + hasher := hmac.New(m.Hash.New, keyBytes) + hasher.Write([]byte(signingString)) + if !hmac.Equal(sig, hasher.Sum(nil)) { + return ErrSignatureInvalid + } + + // No validation errors. Signature is good. + return nil +} + +// Sign implements token signing for the SigningMethod. +// Key must be []byte +func (m *SigningMethodHMAC) Sign(signingString string, key interface{}) (string, error) { + if keyBytes, ok := key.([]byte); ok { + if !m.Hash.Available() { + return "", ErrHashUnavailable + } + + hasher := hmac.New(m.Hash.New, keyBytes) + hasher.Write([]byte(signingString)) + + return EncodeSegment(hasher.Sum(nil)), nil + } + + return "", ErrInvalidKeyType +} diff --git a/vendor/github.com/golang-jwt/jwt/v4/map_claims.go b/vendor/github.com/golang-jwt/jwt/v4/map_claims.go new file mode 100644 index 00000000..2700d64a --- /dev/null +++ b/vendor/github.com/golang-jwt/jwt/v4/map_claims.go @@ -0,0 +1,151 @@ +package jwt + +import ( + "encoding/json" + "errors" + "time" + // "fmt" +) + +// MapClaims is a claims type that uses the map[string]interface{} for JSON decoding. +// This is the default claims type if you don't supply one +type MapClaims map[string]interface{} + +// VerifyAudience Compares the aud claim against cmp. +// If required is false, this method will return true if the value matches or is unset +func (m MapClaims) VerifyAudience(cmp string, req bool) bool { + var aud []string + switch v := m["aud"].(type) { + case string: + aud = append(aud, v) + case []string: + aud = v + case []interface{}: + for _, a := range v { + vs, ok := a.(string) + if !ok { + return false + } + aud = append(aud, vs) + } + } + return verifyAud(aud, cmp, req) +} + +// VerifyExpiresAt compares the exp claim against cmp (cmp <= exp). +// If req is false, it will return true, if exp is unset. +func (m MapClaims) VerifyExpiresAt(cmp int64, req bool) bool { + cmpTime := time.Unix(cmp, 0) + + v, ok := m["exp"] + if !ok { + return !req + } + + switch exp := v.(type) { + case float64: + if exp == 0 { + return verifyExp(nil, cmpTime, req) + } + + return verifyExp(&newNumericDateFromSeconds(exp).Time, cmpTime, req) + case json.Number: + v, _ := exp.Float64() + + return verifyExp(&newNumericDateFromSeconds(v).Time, cmpTime, req) + } + + return false +} + +// VerifyIssuedAt compares the exp claim against cmp (cmp >= iat). +// If req is false, it will return true, if iat is unset. +func (m MapClaims) VerifyIssuedAt(cmp int64, req bool) bool { + cmpTime := time.Unix(cmp, 0) + + v, ok := m["iat"] + if !ok { + return !req + } + + switch iat := v.(type) { + case float64: + if iat == 0 { + return verifyIat(nil, cmpTime, req) + } + + return verifyIat(&newNumericDateFromSeconds(iat).Time, cmpTime, req) + case json.Number: + v, _ := iat.Float64() + + return verifyIat(&newNumericDateFromSeconds(v).Time, cmpTime, req) + } + + return false +} + +// VerifyNotBefore compares the nbf claim against cmp (cmp >= nbf). +// If req is false, it will return true, if nbf is unset. +func (m MapClaims) VerifyNotBefore(cmp int64, req bool) bool { + cmpTime := time.Unix(cmp, 0) + + v, ok := m["nbf"] + if !ok { + return !req + } + + switch nbf := v.(type) { + case float64: + if nbf == 0 { + return verifyNbf(nil, cmpTime, req) + } + + return verifyNbf(&newNumericDateFromSeconds(nbf).Time, cmpTime, req) + case json.Number: + v, _ := nbf.Float64() + + return verifyNbf(&newNumericDateFromSeconds(v).Time, cmpTime, req) + } + + return false +} + +// VerifyIssuer compares the iss claim against cmp. +// If required is false, this method will return true if the value matches or is unset +func (m MapClaims) VerifyIssuer(cmp string, req bool) bool { + iss, _ := m["iss"].(string) + return verifyIss(iss, cmp, req) +} + +// Valid validates time based claims "exp, iat, nbf". +// There is no accounting for clock skew. +// As well, if any of the above claims are not in the token, it will still +// be considered a valid claim. +func (m MapClaims) Valid() error { + vErr := new(ValidationError) + now := TimeFunc().Unix() + + if !m.VerifyExpiresAt(now, false) { + // TODO(oxisto): this should be replaced with ErrTokenExpired + vErr.Inner = errors.New("Token is expired") + vErr.Errors |= ValidationErrorExpired + } + + if !m.VerifyIssuedAt(now, false) { + // TODO(oxisto): this should be replaced with ErrTokenUsedBeforeIssued + vErr.Inner = errors.New("Token used before issued") + vErr.Errors |= ValidationErrorIssuedAt + } + + if !m.VerifyNotBefore(now, false) { + // TODO(oxisto): this should be replaced with ErrTokenNotValidYet + vErr.Inner = errors.New("Token is not valid yet") + vErr.Errors |= ValidationErrorNotValidYet + } + + if vErr.valid() { + return nil + } + + return vErr +} diff --git a/vendor/github.com/golang-jwt/jwt/v4/none.go b/vendor/github.com/golang-jwt/jwt/v4/none.go new file mode 100644 index 00000000..f19835d2 --- /dev/null +++ b/vendor/github.com/golang-jwt/jwt/v4/none.go @@ -0,0 +1,52 @@ +package jwt + +// SigningMethodNone implements the none signing method. This is required by the spec +// but you probably should never use it. +var SigningMethodNone *signingMethodNone + +const UnsafeAllowNoneSignatureType unsafeNoneMagicConstant = "none signing method allowed" + +var NoneSignatureTypeDisallowedError error + +type signingMethodNone struct{} +type unsafeNoneMagicConstant string + +func init() { + SigningMethodNone = &signingMethodNone{} + NoneSignatureTypeDisallowedError = NewValidationError("'none' signature type is not allowed", ValidationErrorSignatureInvalid) + + RegisterSigningMethod(SigningMethodNone.Alg(), func() SigningMethod { + return SigningMethodNone + }) +} + +func (m *signingMethodNone) Alg() string { + return "none" +} + +// Only allow 'none' alg type if UnsafeAllowNoneSignatureType is specified as the key +func (m *signingMethodNone) Verify(signingString, signature string, key interface{}) (err error) { + // Key must be UnsafeAllowNoneSignatureType to prevent accidentally + // accepting 'none' signing method + if _, ok := key.(unsafeNoneMagicConstant); !ok { + return NoneSignatureTypeDisallowedError + } + // If signing method is none, signature must be an empty string + if signature != "" { + return NewValidationError( + "'none' signing method with non-empty signature", + ValidationErrorSignatureInvalid, + ) + } + + // Accept 'none' signing method. + return nil +} + +// Only allow 'none' signing if UnsafeAllowNoneSignatureType is specified as the key +func (m *signingMethodNone) Sign(signingString string, key interface{}) (string, error) { + if _, ok := key.(unsafeNoneMagicConstant); ok { + return "", nil + } + return "", NoneSignatureTypeDisallowedError +} diff --git a/vendor/github.com/golang-jwt/jwt/v4/parser.go b/vendor/github.com/golang-jwt/jwt/v4/parser.go new file mode 100644 index 00000000..c0a6f692 --- /dev/null +++ b/vendor/github.com/golang-jwt/jwt/v4/parser.go @@ -0,0 +1,177 @@ +package jwt + +import ( + "bytes" + "encoding/json" + "fmt" + "strings" +) + +type Parser struct { + // If populated, only these methods will be considered valid. + // + // Deprecated: In future releases, this field will not be exported anymore and should be set with an option to NewParser instead. + ValidMethods []string + + // Use JSON Number format in JSON decoder. + // + // Deprecated: In future releases, this field will not be exported anymore and should be set with an option to NewParser instead. + UseJSONNumber bool + + // Skip claims validation during token parsing. + // + // Deprecated: In future releases, this field will not be exported anymore and should be set with an option to NewParser instead. + SkipClaimsValidation bool +} + +// NewParser creates a new Parser with the specified options +func NewParser(options ...ParserOption) *Parser { + p := &Parser{} + + // loop through our parsing options and apply them + for _, option := range options { + option(p) + } + + return p +} + +// Parse parses, validates, verifies the signature and returns the parsed token. +// keyFunc will receive the parsed token and should return the key for validating. +func (p *Parser) Parse(tokenString string, keyFunc Keyfunc) (*Token, error) { + return p.ParseWithClaims(tokenString, MapClaims{}, keyFunc) +} + +// ParseWithClaims parses, validates, and verifies like Parse, but supplies a default object implementing the Claims +// interface. This provides default values which can be overridden and allows a caller to use their own type, rather +// than the default MapClaims implementation of Claims. +// +// Note: If you provide a custom claim implementation that embeds one of the standard claims (such as RegisteredClaims), +// make sure that a) you either embed a non-pointer version of the claims or b) if you are using a pointer, allocate the +// proper memory for it before passing in the overall claims, otherwise you might run into a panic. +func (p *Parser) ParseWithClaims(tokenString string, claims Claims, keyFunc Keyfunc) (*Token, error) { + token, parts, err := p.ParseUnverified(tokenString, claims) + if err != nil { + return token, err + } + + // Verify signing method is in the required set + if p.ValidMethods != nil { + var signingMethodValid = false + var alg = token.Method.Alg() + for _, m := range p.ValidMethods { + if m == alg { + signingMethodValid = true + break + } + } + if !signingMethodValid { + // signing method is not in the listed set + return token, NewValidationError(fmt.Sprintf("signing method %v is invalid", alg), ValidationErrorSignatureInvalid) + } + } + + // Lookup key + var key interface{} + if keyFunc == nil { + // keyFunc was not provided. short circuiting validation + return token, NewValidationError("no Keyfunc was provided.", ValidationErrorUnverifiable) + } + if key, err = keyFunc(token); err != nil { + // keyFunc returned an error + if ve, ok := err.(*ValidationError); ok { + return token, ve + } + return token, &ValidationError{Inner: err, Errors: ValidationErrorUnverifiable} + } + + vErr := &ValidationError{} + + // Validate Claims + if !p.SkipClaimsValidation { + if err := token.Claims.Valid(); err != nil { + + // If the Claims Valid returned an error, check if it is a validation error, + // If it was another error type, create a ValidationError with a generic ClaimsInvalid flag set + if e, ok := err.(*ValidationError); !ok { + vErr = &ValidationError{Inner: err, Errors: ValidationErrorClaimsInvalid} + } else { + vErr = e + } + } + } + + // Perform validation + token.Signature = parts[2] + if err = token.Method.Verify(strings.Join(parts[0:2], "."), token.Signature, key); err != nil { + vErr.Inner = err + vErr.Errors |= ValidationErrorSignatureInvalid + } + + if vErr.valid() { + token.Valid = true + return token, nil + } + + return token, vErr +} + +// ParseUnverified parses the token but doesn't validate the signature. +// +// WARNING: Don't use this method unless you know what you're doing. +// +// It's only ever useful in cases where you know the signature is valid (because it has +// been checked previously in the stack) and you want to extract values from it. +func (p *Parser) ParseUnverified(tokenString string, claims Claims) (token *Token, parts []string, err error) { + parts = strings.Split(tokenString, ".") + if len(parts) != 3 { + return nil, parts, NewValidationError("token contains an invalid number of segments", ValidationErrorMalformed) + } + + token = &Token{Raw: tokenString} + + // parse Header + var headerBytes []byte + if headerBytes, err = DecodeSegment(parts[0]); err != nil { + if strings.HasPrefix(strings.ToLower(tokenString), "bearer ") { + return token, parts, NewValidationError("tokenstring should not contain 'bearer '", ValidationErrorMalformed) + } + return token, parts, &ValidationError{Inner: err, Errors: ValidationErrorMalformed} + } + if err = json.Unmarshal(headerBytes, &token.Header); err != nil { + return token, parts, &ValidationError{Inner: err, Errors: ValidationErrorMalformed} + } + + // parse Claims + var claimBytes []byte + token.Claims = claims + + if claimBytes, err = DecodeSegment(parts[1]); err != nil { + return token, parts, &ValidationError{Inner: err, Errors: ValidationErrorMalformed} + } + dec := json.NewDecoder(bytes.NewBuffer(claimBytes)) + if p.UseJSONNumber { + dec.UseNumber() + } + // JSON Decode. Special case for map type to avoid weird pointer behavior + if c, ok := token.Claims.(MapClaims); ok { + err = dec.Decode(&c) + } else { + err = dec.Decode(&claims) + } + // Handle decode error + if err != nil { + return token, parts, &ValidationError{Inner: err, Errors: ValidationErrorMalformed} + } + + // Lookup signature method + if method, ok := token.Header["alg"].(string); ok { + if token.Method = GetSigningMethod(method); token.Method == nil { + return token, parts, NewValidationError("signing method (alg) is unavailable.", ValidationErrorUnverifiable) + } + } else { + return token, parts, NewValidationError("signing method (alg) is unspecified.", ValidationErrorUnverifiable) + } + + return token, parts, nil +} diff --git a/vendor/github.com/golang-jwt/jwt/v4/parser_option.go b/vendor/github.com/golang-jwt/jwt/v4/parser_option.go new file mode 100644 index 00000000..6ea6f952 --- /dev/null +++ b/vendor/github.com/golang-jwt/jwt/v4/parser_option.go @@ -0,0 +1,29 @@ +package jwt + +// ParserOption is used to implement functional-style options that modify the behavior of the parser. To add +// new options, just create a function (ideally beginning with With or Without) that returns an anonymous function that +// takes a *Parser type as input and manipulates its configuration accordingly. +type ParserOption func(*Parser) + +// WithValidMethods is an option to supply algorithm methods that the parser will check. Only those methods will be considered valid. +// It is heavily encouraged to use this option in order to prevent attacks such as https://auth0.com/blog/critical-vulnerabilities-in-json-web-token-libraries/. +func WithValidMethods(methods []string) ParserOption { + return func(p *Parser) { + p.ValidMethods = methods + } +} + +// WithJSONNumber is an option to configure the underlying JSON parser with UseNumber +func WithJSONNumber() ParserOption { + return func(p *Parser) { + p.UseJSONNumber = true + } +} + +// WithoutClaimsValidation is an option to disable claims validation. This option should only be used if you exactly know +// what you are doing. +func WithoutClaimsValidation() ParserOption { + return func(p *Parser) { + p.SkipClaimsValidation = true + } +} diff --git a/vendor/github.com/golang-jwt/jwt/v4/rsa.go b/vendor/github.com/golang-jwt/jwt/v4/rsa.go new file mode 100644 index 00000000..b910b19c --- /dev/null +++ b/vendor/github.com/golang-jwt/jwt/v4/rsa.go @@ -0,0 +1,101 @@ +package jwt + +import ( + "crypto" + "crypto/rand" + "crypto/rsa" +) + +// SigningMethodRSA implements the RSA family of signing methods. +// Expects *rsa.PrivateKey for signing and *rsa.PublicKey for validation +type SigningMethodRSA struct { + Name string + Hash crypto.Hash +} + +// Specific instances for RS256 and company +var ( + SigningMethodRS256 *SigningMethodRSA + SigningMethodRS384 *SigningMethodRSA + SigningMethodRS512 *SigningMethodRSA +) + +func init() { + // RS256 + SigningMethodRS256 = &SigningMethodRSA{"RS256", crypto.SHA256} + RegisterSigningMethod(SigningMethodRS256.Alg(), func() SigningMethod { + return SigningMethodRS256 + }) + + // RS384 + SigningMethodRS384 = &SigningMethodRSA{"RS384", crypto.SHA384} + RegisterSigningMethod(SigningMethodRS384.Alg(), func() SigningMethod { + return SigningMethodRS384 + }) + + // RS512 + SigningMethodRS512 = &SigningMethodRSA{"RS512", crypto.SHA512} + RegisterSigningMethod(SigningMethodRS512.Alg(), func() SigningMethod { + return SigningMethodRS512 + }) +} + +func (m *SigningMethodRSA) Alg() string { + return m.Name +} + +// Verify implements token verification for the SigningMethod +// For this signing method, must be an *rsa.PublicKey structure. +func (m *SigningMethodRSA) Verify(signingString, signature string, key interface{}) error { + var err error + + // Decode the signature + var sig []byte + if sig, err = DecodeSegment(signature); err != nil { + return err + } + + var rsaKey *rsa.PublicKey + var ok bool + + if rsaKey, ok = key.(*rsa.PublicKey); !ok { + return ErrInvalidKeyType + } + + // Create hasher + if !m.Hash.Available() { + return ErrHashUnavailable + } + hasher := m.Hash.New() + hasher.Write([]byte(signingString)) + + // Verify the signature + return rsa.VerifyPKCS1v15(rsaKey, m.Hash, hasher.Sum(nil), sig) +} + +// Sign implements token signing for the SigningMethod +// For this signing method, must be an *rsa.PrivateKey structure. +func (m *SigningMethodRSA) Sign(signingString string, key interface{}) (string, error) { + var rsaKey *rsa.PrivateKey + var ok bool + + // Validate type of key + if rsaKey, ok = key.(*rsa.PrivateKey); !ok { + return "", ErrInvalidKey + } + + // Create the hasher + if !m.Hash.Available() { + return "", ErrHashUnavailable + } + + hasher := m.Hash.New() + hasher.Write([]byte(signingString)) + + // Sign the string and return the encoded bytes + if sigBytes, err := rsa.SignPKCS1v15(rand.Reader, rsaKey, m.Hash, hasher.Sum(nil)); err == nil { + return EncodeSegment(sigBytes), nil + } else { + return "", err + } +} diff --git a/vendor/github.com/golang-jwt/jwt/v4/rsa_pss.go b/vendor/github.com/golang-jwt/jwt/v4/rsa_pss.go new file mode 100644 index 00000000..4fd6f9e6 --- /dev/null +++ b/vendor/github.com/golang-jwt/jwt/v4/rsa_pss.go @@ -0,0 +1,143 @@ +//go:build go1.4 +// +build go1.4 + +package jwt + +import ( + "crypto" + "crypto/rand" + "crypto/rsa" +) + +// SigningMethodRSAPSS implements the RSAPSS family of signing methods signing methods +type SigningMethodRSAPSS struct { + *SigningMethodRSA + Options *rsa.PSSOptions + // VerifyOptions is optional. If set overrides Options for rsa.VerifyPPS. + // Used to accept tokens signed with rsa.PSSSaltLengthAuto, what doesn't follow + // https://tools.ietf.org/html/rfc7518#section-3.5 but was used previously. + // See https://github.com/dgrijalva/jwt-go/issues/285#issuecomment-437451244 for details. + VerifyOptions *rsa.PSSOptions +} + +// Specific instances for RS/PS and company. +var ( + SigningMethodPS256 *SigningMethodRSAPSS + SigningMethodPS384 *SigningMethodRSAPSS + SigningMethodPS512 *SigningMethodRSAPSS +) + +func init() { + // PS256 + SigningMethodPS256 = &SigningMethodRSAPSS{ + SigningMethodRSA: &SigningMethodRSA{ + Name: "PS256", + Hash: crypto.SHA256, + }, + Options: &rsa.PSSOptions{ + SaltLength: rsa.PSSSaltLengthEqualsHash, + }, + VerifyOptions: &rsa.PSSOptions{ + SaltLength: rsa.PSSSaltLengthAuto, + }, + } + RegisterSigningMethod(SigningMethodPS256.Alg(), func() SigningMethod { + return SigningMethodPS256 + }) + + // PS384 + SigningMethodPS384 = &SigningMethodRSAPSS{ + SigningMethodRSA: &SigningMethodRSA{ + Name: "PS384", + Hash: crypto.SHA384, + }, + Options: &rsa.PSSOptions{ + SaltLength: rsa.PSSSaltLengthEqualsHash, + }, + VerifyOptions: &rsa.PSSOptions{ + SaltLength: rsa.PSSSaltLengthAuto, + }, + } + RegisterSigningMethod(SigningMethodPS384.Alg(), func() SigningMethod { + return SigningMethodPS384 + }) + + // PS512 + SigningMethodPS512 = &SigningMethodRSAPSS{ + SigningMethodRSA: &SigningMethodRSA{ + Name: "PS512", + Hash: crypto.SHA512, + }, + Options: &rsa.PSSOptions{ + SaltLength: rsa.PSSSaltLengthEqualsHash, + }, + VerifyOptions: &rsa.PSSOptions{ + SaltLength: rsa.PSSSaltLengthAuto, + }, + } + RegisterSigningMethod(SigningMethodPS512.Alg(), func() SigningMethod { + return SigningMethodPS512 + }) +} + +// Verify implements token verification for the SigningMethod. +// For this verify method, key must be an rsa.PublicKey struct +func (m *SigningMethodRSAPSS) Verify(signingString, signature string, key interface{}) error { + var err error + + // Decode the signature + var sig []byte + if sig, err = DecodeSegment(signature); err != nil { + return err + } + + var rsaKey *rsa.PublicKey + switch k := key.(type) { + case *rsa.PublicKey: + rsaKey = k + default: + return ErrInvalidKey + } + + // Create hasher + if !m.Hash.Available() { + return ErrHashUnavailable + } + hasher := m.Hash.New() + hasher.Write([]byte(signingString)) + + opts := m.Options + if m.VerifyOptions != nil { + opts = m.VerifyOptions + } + + return rsa.VerifyPSS(rsaKey, m.Hash, hasher.Sum(nil), sig, opts) +} + +// Sign implements token signing for the SigningMethod. +// For this signing method, key must be an rsa.PrivateKey struct +func (m *SigningMethodRSAPSS) Sign(signingString string, key interface{}) (string, error) { + var rsaKey *rsa.PrivateKey + + switch k := key.(type) { + case *rsa.PrivateKey: + rsaKey = k + default: + return "", ErrInvalidKeyType + } + + // Create the hasher + if !m.Hash.Available() { + return "", ErrHashUnavailable + } + + hasher := m.Hash.New() + hasher.Write([]byte(signingString)) + + // Sign the string and return the encoded bytes + if sigBytes, err := rsa.SignPSS(rand.Reader, rsaKey, m.Hash, hasher.Sum(nil), m.Options); err == nil { + return EncodeSegment(sigBytes), nil + } else { + return "", err + } +} diff --git a/vendor/github.com/golang-jwt/jwt/v4/rsa_utils.go b/vendor/github.com/golang-jwt/jwt/v4/rsa_utils.go new file mode 100644 index 00000000..1966c450 --- /dev/null +++ b/vendor/github.com/golang-jwt/jwt/v4/rsa_utils.go @@ -0,0 +1,105 @@ +package jwt + +import ( + "crypto/rsa" + "crypto/x509" + "encoding/pem" + "errors" +) + +var ( + ErrKeyMustBePEMEncoded = errors.New("invalid key: Key must be a PEM encoded PKCS1 or PKCS8 key") + ErrNotRSAPrivateKey = errors.New("key is not a valid RSA private key") + ErrNotRSAPublicKey = errors.New("key is not a valid RSA public key") +) + +// ParseRSAPrivateKeyFromPEM parses a PEM encoded PKCS1 or PKCS8 private key +func ParseRSAPrivateKeyFromPEM(key []byte) (*rsa.PrivateKey, error) { + var err error + + // Parse PEM block + var block *pem.Block + if block, _ = pem.Decode(key); block == nil { + return nil, ErrKeyMustBePEMEncoded + } + + var parsedKey interface{} + if parsedKey, err = x509.ParsePKCS1PrivateKey(block.Bytes); err != nil { + if parsedKey, err = x509.ParsePKCS8PrivateKey(block.Bytes); err != nil { + return nil, err + } + } + + var pkey *rsa.PrivateKey + var ok bool + if pkey, ok = parsedKey.(*rsa.PrivateKey); !ok { + return nil, ErrNotRSAPrivateKey + } + + return pkey, nil +} + +// ParseRSAPrivateKeyFromPEMWithPassword parses a PEM encoded PKCS1 or PKCS8 private key protected with password +// +// Deprecated: This function is deprecated and should not be used anymore. It uses the deprecated x509.DecryptPEMBlock +// function, which was deprecated since RFC 1423 is regarded insecure by design. Unfortunately, there is no alternative +// in the Go standard library for now. See https://github.com/golang/go/issues/8860. +func ParseRSAPrivateKeyFromPEMWithPassword(key []byte, password string) (*rsa.PrivateKey, error) { + var err error + + // Parse PEM block + var block *pem.Block + if block, _ = pem.Decode(key); block == nil { + return nil, ErrKeyMustBePEMEncoded + } + + var parsedKey interface{} + + var blockDecrypted []byte + if blockDecrypted, err = x509.DecryptPEMBlock(block, []byte(password)); err != nil { + return nil, err + } + + if parsedKey, err = x509.ParsePKCS1PrivateKey(blockDecrypted); err != nil { + if parsedKey, err = x509.ParsePKCS8PrivateKey(blockDecrypted); err != nil { + return nil, err + } + } + + var pkey *rsa.PrivateKey + var ok bool + if pkey, ok = parsedKey.(*rsa.PrivateKey); !ok { + return nil, ErrNotRSAPrivateKey + } + + return pkey, nil +} + +// ParseRSAPublicKeyFromPEM parses a PEM encoded PKCS1 or PKCS8 public key +func ParseRSAPublicKeyFromPEM(key []byte) (*rsa.PublicKey, error) { + var err error + + // Parse PEM block + var block *pem.Block + if block, _ = pem.Decode(key); block == nil { + return nil, ErrKeyMustBePEMEncoded + } + + // Parse the key + var parsedKey interface{} + if parsedKey, err = x509.ParsePKIXPublicKey(block.Bytes); err != nil { + if cert, err := x509.ParseCertificate(block.Bytes); err == nil { + parsedKey = cert.PublicKey + } else { + return nil, err + } + } + + var pkey *rsa.PublicKey + var ok bool + if pkey, ok = parsedKey.(*rsa.PublicKey); !ok { + return nil, ErrNotRSAPublicKey + } + + return pkey, nil +} diff --git a/vendor/github.com/golang-jwt/jwt/v4/signing_method.go b/vendor/github.com/golang-jwt/jwt/v4/signing_method.go new file mode 100644 index 00000000..241ae9c6 --- /dev/null +++ b/vendor/github.com/golang-jwt/jwt/v4/signing_method.go @@ -0,0 +1,46 @@ +package jwt + +import ( + "sync" +) + +var signingMethods = map[string]func() SigningMethod{} +var signingMethodLock = new(sync.RWMutex) + +// SigningMethod can be used add new methods for signing or verifying tokens. +type SigningMethod interface { + Verify(signingString, signature string, key interface{}) error // Returns nil if signature is valid + Sign(signingString string, key interface{}) (string, error) // Returns encoded signature or error + Alg() string // returns the alg identifier for this method (example: 'HS256') +} + +// RegisterSigningMethod registers the "alg" name and a factory function for signing method. +// This is typically done during init() in the method's implementation +func RegisterSigningMethod(alg string, f func() SigningMethod) { + signingMethodLock.Lock() + defer signingMethodLock.Unlock() + + signingMethods[alg] = f +} + +// GetSigningMethod retrieves a signing method from an "alg" string +func GetSigningMethod(alg string) (method SigningMethod) { + signingMethodLock.RLock() + defer signingMethodLock.RUnlock() + + if methodF, ok := signingMethods[alg]; ok { + method = methodF() + } + return +} + +// GetAlgorithms returns a list of registered "alg" names +func GetAlgorithms() (algs []string) { + signingMethodLock.RLock() + defer signingMethodLock.RUnlock() + + for alg := range signingMethods { + algs = append(algs, alg) + } + return +} diff --git a/vendor/github.com/golang-jwt/jwt/v4/staticcheck.conf b/vendor/github.com/golang-jwt/jwt/v4/staticcheck.conf new file mode 100644 index 00000000..53745d51 --- /dev/null +++ b/vendor/github.com/golang-jwt/jwt/v4/staticcheck.conf @@ -0,0 +1 @@ +checks = ["all", "-ST1000", "-ST1003", "-ST1016", "-ST1023"] diff --git a/vendor/github.com/golang-jwt/jwt/v4/token.go b/vendor/github.com/golang-jwt/jwt/v4/token.go new file mode 100644 index 00000000..71e909ea --- /dev/null +++ b/vendor/github.com/golang-jwt/jwt/v4/token.go @@ -0,0 +1,132 @@ +package jwt + +import ( + "encoding/base64" + "encoding/json" + "strings" + "time" +) + +// DecodePaddingAllowed will switch the codec used for decoding JWTs respectively. Note that the JWS RFC7515 +// states that the tokens will utilize a Base64url encoding with no padding. Unfortunately, some implementations +// of JWT are producing non-standard tokens, and thus require support for decoding. Note that this is a global +// variable, and updating it will change the behavior on a package level, and is also NOT go-routine safe. +// To use the non-recommended decoding, set this boolean to `true` prior to using this package. +var DecodePaddingAllowed bool + +// TimeFunc provides the current time when parsing token to validate "exp" claim (expiration time). +// You can override it to use another time value. This is useful for testing or if your +// server uses a different time zone than your tokens. +var TimeFunc = time.Now + +// Keyfunc will be used by the Parse methods as a callback function to supply +// the key for verification. The function receives the parsed, +// but unverified Token. This allows you to use properties in the +// Header of the token (such as `kid`) to identify which key to use. +type Keyfunc func(*Token) (interface{}, error) + +// Token represents a JWT Token. Different fields will be used depending on whether you're +// creating or parsing/verifying a token. +type Token struct { + Raw string // The raw token. Populated when you Parse a token + Method SigningMethod // The signing method used or to be used + Header map[string]interface{} // The first segment of the token + Claims Claims // The second segment of the token + Signature string // The third segment of the token. Populated when you Parse a token + Valid bool // Is the token valid? Populated when you Parse/Verify a token +} + +// New creates a new Token with the specified signing method and an empty map of claims. +func New(method SigningMethod) *Token { + return NewWithClaims(method, MapClaims{}) +} + +// NewWithClaims creates a new Token with the specified signing method and claims. +func NewWithClaims(method SigningMethod, claims Claims) *Token { + return &Token{ + Header: map[string]interface{}{ + "typ": "JWT", + "alg": method.Alg(), + }, + Claims: claims, + Method: method, + } +} + +// SignedString creates and returns a complete, signed JWT. +// The token is signed using the SigningMethod specified in the token. +func (t *Token) SignedString(key interface{}) (string, error) { + var sig, sstr string + var err error + if sstr, err = t.SigningString(); err != nil { + return "", err + } + if sig, err = t.Method.Sign(sstr, key); err != nil { + return "", err + } + return strings.Join([]string{sstr, sig}, "."), nil +} + +// SigningString generates the signing string. This is the +// most expensive part of the whole deal. Unless you +// need this for something special, just go straight for +// the SignedString. +func (t *Token) SigningString() (string, error) { + var err error + var jsonValue []byte + + if jsonValue, err = json.Marshal(t.Header); err != nil { + return "", err + } + header := EncodeSegment(jsonValue) + + if jsonValue, err = json.Marshal(t.Claims); err != nil { + return "", err + } + claim := EncodeSegment(jsonValue) + + return strings.Join([]string{header, claim}, "."), nil +} + +// Parse parses, validates, verifies the signature and returns the parsed token. +// keyFunc will receive the parsed token and should return the cryptographic key +// for verifying the signature. +// The caller is strongly encouraged to set the WithValidMethods option to +// validate the 'alg' claim in the token matches the expected algorithm. +// For more details about the importance of validating the 'alg' claim, +// see https://auth0.com/blog/critical-vulnerabilities-in-json-web-token-libraries/ +func Parse(tokenString string, keyFunc Keyfunc, options ...ParserOption) (*Token, error) { + return NewParser(options...).Parse(tokenString, keyFunc) +} + +// ParseWithClaims is a shortcut for NewParser().ParseWithClaims(). +// +// Note: If you provide a custom claim implementation that embeds one of the standard claims (such as RegisteredClaims), +// make sure that a) you either embed a non-pointer version of the claims or b) if you are using a pointer, allocate the +// proper memory for it before passing in the overall claims, otherwise you might run into a panic. +func ParseWithClaims(tokenString string, claims Claims, keyFunc Keyfunc, options ...ParserOption) (*Token, error) { + return NewParser(options...).ParseWithClaims(tokenString, claims, keyFunc) +} + +// EncodeSegment encodes a JWT specific base64url encoding with padding stripped +// +// Deprecated: In a future release, we will demote this function to a non-exported function, since it +// should only be used internally +func EncodeSegment(seg []byte) string { + return base64.RawURLEncoding.EncodeToString(seg) +} + +// DecodeSegment decodes a JWT specific base64url encoding with padding stripped +// +// Deprecated: In a future release, we will demote this function to a non-exported function, since it +// should only be used internally +func DecodeSegment(seg string) ([]byte, error) { + if DecodePaddingAllowed { + if l := len(seg) % 4; l > 0 { + seg += strings.Repeat("=", 4-l) + } + return base64.URLEncoding.DecodeString(seg) + } + + return base64.RawURLEncoding.DecodeString(seg) +} diff --git a/vendor/github.com/golang-jwt/jwt/v4/types.go b/vendor/github.com/golang-jwt/jwt/v4/types.go new file mode 100644 index 00000000..ac8e140e --- /dev/null +++ b/vendor/github.com/golang-jwt/jwt/v4/types.go @@ -0,0 +1,145 @@ +package jwt + +import ( + "encoding/json" + "fmt" + "math" + "reflect" + "strconv" + "time" +) + +// TimePrecision sets the precision of times and dates within this library. +// This has an influence on the precision of times when comparing expiry or +// other related time fields. Furthermore, it is also the precision of times +// when serializing. +// +// For backwards compatibility the default precision is set to seconds, so that +// no fractional timestamps are generated. +var TimePrecision = time.Second + +// MarshalSingleStringAsArray modifies the behaviour of the ClaimStrings type, especially +// its MarshalJSON function. +// +// If it is set to true (the default), it will always serialize the type as an +// array of strings, even if it just contains one element, defaulting to the behaviour +// of the underlying []string. If it is set to false, it will serialize to a single +// string, if it contains one element. Otherwise, it will serialize to an array of strings. +var MarshalSingleStringAsArray = true + +// NumericDate represents a JSON numeric date value, as referenced at +// https://datatracker.ietf.org/doc/html/rfc7519#section-2. +type NumericDate struct { + time.Time +} + +// NewNumericDate constructs a new *NumericDate from a standard library time.Time struct. +// It will truncate the timestamp according to the precision specified in TimePrecision. +func NewNumericDate(t time.Time) *NumericDate { + return &NumericDate{t.Truncate(TimePrecision)} +} + +// newNumericDateFromSeconds creates a new *NumericDate out of a float64 representing a +// UNIX epoch with the float fraction representing non-integer seconds. +func newNumericDateFromSeconds(f float64) *NumericDate { + round, frac := math.Modf(f) + return NewNumericDate(time.Unix(int64(round), int64(frac*1e9))) +} + +// MarshalJSON is an implementation of the json.RawMessage interface and serializes the UNIX epoch +// represented in NumericDate to a byte array, using the precision specified in TimePrecision. +func (date NumericDate) MarshalJSON() (b []byte, err error) { + var prec int + if TimePrecision < time.Second { + prec = int(math.Log10(float64(time.Second) / float64(TimePrecision))) + } + truncatedDate := date.Truncate(TimePrecision) + + // For very large timestamps, UnixNano would overflow an int64, but this + // function requires nanosecond level precision, so we have to use the + // following technique to get round the issue: + // 1. Take the normal unix timestamp to form the whole number part of the + // output, + // 2. Take the result of the Nanosecond function, which retuns the offset + // within the second of the particular unix time instance, to form the + // decimal part of the output + // 3. Concatenate them to produce the final result + seconds := strconv.FormatInt(truncatedDate.Unix(), 10) + nanosecondsOffset := strconv.FormatFloat(float64(truncatedDate.Nanosecond())/float64(time.Second), 'f', prec, 64) + + output := append([]byte(seconds), []byte(nanosecondsOffset)[1:]...) + + return output, nil +} + +// UnmarshalJSON is an implementation of the json.RawMessage interface and deserializses a +// NumericDate from a JSON representation, i.e. a json.Number. This number represents an UNIX epoch +// with either integer or non-integer seconds. +func (date *NumericDate) UnmarshalJSON(b []byte) (err error) { + var ( + number json.Number + f float64 + ) + + if err = json.Unmarshal(b, &number); err != nil { + return fmt.Errorf("could not parse NumericData: %w", err) + } + + if f, err = number.Float64(); err != nil { + return fmt.Errorf("could not convert json number value to float: %w", err) + } + + n := newNumericDateFromSeconds(f) + *date = *n + + return nil +} + +// ClaimStrings is basically just a slice of strings, but it can be either serialized from a string array or just a string. +// This type is necessary, since the "aud" claim can either be a single string or an array. +type ClaimStrings []string + +func (s *ClaimStrings) UnmarshalJSON(data []byte) (err error) { + var value interface{} + + if err = json.Unmarshal(data, &value); err != nil { + return err + } + + var aud []string + + switch v := value.(type) { + case string: + aud = append(aud, v) + case []string: + aud = ClaimStrings(v) + case []interface{}: + for _, vv := range v { + vs, ok := vv.(string) + if !ok { + return &json.UnsupportedTypeError{Type: reflect.TypeOf(vv)} + } + aud = append(aud, vs) + } + case nil: + return nil + default: + return &json.UnsupportedTypeError{Type: reflect.TypeOf(v)} + } + + *s = aud + + return +} + +func (s ClaimStrings) MarshalJSON() (b []byte, err error) { + // This handles a special case in the JWT RFC. If the string array, e.g. used by the "aud" field, + // only contains one element, it MAY be serialized as a single string. This may or may not be + // desired based on the ecosystem of other JWT library used, so we make it configurable by the + // variable MarshalSingleStringAsArray. + if len(s) == 1 && !MarshalSingleStringAsArray { + return json.Marshal(s[0]) + } + + return json.Marshal([]string(s)) +} diff --git a/vendor/modules.txt b/vendor/modules.txt index b74a0c5a..1bbf4a75 100644 --- a/vendor/modules.txt +++ b/vendor/modules.txt @@ -31,6 +31,9 @@ github.com/gobwas/glob/syntax/ast github.com/gobwas/glob/syntax/lexer github.com/gobwas/glob/util/runes github.com/gobwas/glob/util/strings +# github.com/golang-jwt/jwt/v4 v4.4.3 +## explicit; go 1.16 +github.com/golang-jwt/jwt/v4 # github.com/golang/protobuf v1.5.2 ## explicit; go 1.9 github.com/golang/protobuf/proto From c8683d5acf3638e53f2a7a8809507b33e2dc002b Mon Sep 17 00:00:00 2001 From: Lucas Roesler Date: Wed, 28 Dec 2022 12:57:08 +0100 Subject: [PATCH 5/5] feat: implement auth result parsing Support parsing structured auth policy results. This allows the policy to customize the error response code. The policy can also attach additional headers to the request so that the implementation. The primary use case for the additional headers is to pass user or client identification to the implmentation to allow for auditing and/or additional authorization checks. Signed-off-by: Lucas Roesler --- README.md | 54 ++++++++++++++++++++++++---- auth/config.go | 14 ++++++++ auth/local.go | 81 ++++++++++++++++++++++++++++++------------ auth/local_test.go | 46 ++++++++++++++---------- auth/middleware.go | 46 ++++++++++++++++-------- auth/testdata/jwt.rego | 9 +++-- main.go | 2 +- 7 files changed, 188 insertions(+), 64 deletions(-) create mode 100644 auth/config.go diff --git a/README.md b/README.md index 56a19f0f..0a3ebd55 100644 --- a/README.md +++ b/README.md @@ -138,7 +138,18 @@ This allows writing authentication and authorization policies in Rego and applyi The auth policy can be loaded from a secret _or_ can be built directly into your function. -If you have created an auth policy as a secret, then you simply need to add the secret to your `secrets` and then set the `OPA_POLICY` environment variable to the secret name during deployment. You must also set the `OPA_QUERY` environment variable. +### Loading auth policy from a secret +If you have created an auth policy as a secret, then you simply need to add the secret to your `secrets` and then set the `opa_policy` environment variable to the secret name during deployment. You must also set the `opa_query` environment variable. + +For example, if you have a policy called `basic_auth_policy` then you would + +1. create a secret named `basic_auth_policy` +2. assign / require this secret in your function spec +3. and set the following environment variable: + + ```env + opa_policy=basic_auth_policy + ``` ### Example basic auth @@ -170,8 +181,8 @@ credentials := {"bob": "secretvalue"} Then your function would need the environment variables ```env -OPA_POLICY=basic_auth_policy -OPA_QUERY=data.api.auth.basic +opa_policy=basic_auth_policy +opa_query=data.api.auth.basic ``` Note: the OPA query will always be `data.` @@ -188,13 +199,32 @@ The policy input will receive the following input data: | `input.headers` | the full request headers, note this is a map from string to a list of string, ie `map[string][]string` | | `input.rawBody` | the request body as a string | | `input.body` | the JSON parsed request body | -| `input.data` | additional input data as configured via the `OPA_INPUT_*` environnement variables | +| `input.data` | additional input data as configured via the `opa_input_*` environnement variables | + +The `headers`, `rawBody`, and `body` are not included by default. Each is controlled the value of the `opa_include_headers`, `opa_include_raw_body`, and `opa_include_body` environment variables respectively. + +The `data` field contains additional data from the environment variables that have the prefix `opa_input`. For example, `opa_input_valid_issuers=http://google.com` will then be available to the policy as `input.data.valid_issuers`. + +Secret values can also be loaded into the `data` field, this is controlled via the `opa_input_secrets`. This can be a comma separated list of secrets to be loaded. For example, if you have a secret name `slack_hash_key` and you set `opa_input_secrets=slack_hash_key`, you can access the value from `input.data.slack_hash_key` + +### Policy output +The middleware supports two types of policy output: simple and structured. + +#### Simple boolean output +The simplest form of policy output is a boolean value. If the policy returns `true` then the request will be allowed to proceed, if the policy returns `false` then the request will be rejected with a `401` status code. + +Examples can be found in the [basic auth policy](/auth/testdata/basic_auth.rego) and the [HMAC policy](/auth/testdata/hmac_auth.rego) -The `headers`, `rawBody`, and `body` are not included by default. Each is controlled the value of the `OPA_INCLUDE_HEADERS`, `OPA_INCLUDE_RAW_BODY`, and `OPA_INCLUDE_BODY` environment variables respectively. +#### Structured response output +The policy can also return a structured response. This allows the policy to return additional context to the handler by specifying additional headers to add to the request. The response must be a JSON object with the following fields: -The `data` field contains additional data from the environment variables that have the prefix `OPA_INPUT`. For example, `OPA_INPUT_VALID_ISSUERS=http://google.com` will then be available to the policy as `input.data.valid_issuers`. +| Key | Description | +| ----------- | --------------------------------------------------------------------------- | +| `allow` | a boolean value indicating if the request should be allowed or not | +| `status` | (optional) the status code to return if the request is **rejected**. Defaults to `401` | +| `headers` | (optional) a map of headers to be added to the request before it is sent to your implementation. It is only used if the request **is allowed**. | -Secret values can also be loaded into the `data` field, this is controlled via the `OPA_INPUT_SECRETS`. This can be a comma separated list of secrets to be loaded. For example, if you have a secret name `slack_hash_key` and you set `OPA_INPUT_SECRETS=slack_hash_key`, you can access the value from `input.data.slack_hash_key` +The primary use case would be extracting client or user information from the request and then adding it to the request headers (for example, as `X-User-Email`) so that your implementation can use this information to perform additional actions or enable auditing. ## Metrics @@ -230,6 +260,16 @@ Environmental variables: | `buffer_http` | deprecated alias for `http_buffer_req_body`, will be removed in future version | | `static_path` | Absolute or relative path to the directory that will be served if `mode="static"` | | `ready_path` | When non-empty, requests to `/_/ready` will invoke the function handler with this path. This can be used to provide custom readiness logic. When `max_inflight` is set, the concurrency limit is checked first before proxying the request to the function. | +| `opa_policy` | The name of the secret containing the OPA policy to be used. Alternatively, can be a path to a file, e.g. a `tar.gz`, `.rego`, or `wasm` file bundled with the function at build time. | +| `opa_query` | The dot separated query used to evaluate the policy. | +| `opa_debug` | When set to `true`, the OPA policy will be evaluated in debug mode, the output is sent to the function stdout. Additional metadata is also printed by the auth middleware. | +| `opa_skip_paths` | comma separated list of paths to skip policy evaluation, these paths will allow _all_ requests. | +| `opa_include_headers` | boolean value to indicate whether to include the request headers in the OPA input. See also [Policy Input](#policy-input). Default: `false` | +| `opa_include_body` | boolean value to indicate whether to include the request body in the OPA input, the body is will be parsed JSON. See also [Policy Input](#policy-input). Default: `false` | +| `opa_include_raw_body` | boolean value to indicate whether to include the raw request body in the OPA input. See also [Policy Input](#policy-input). Default: `false` | +| `opa_input_secrets` | comma separated list of secrets to include in the OPA input. The values are accessible using `input.data[secret_name]` or `input.data.secret_name`, see also [Policy Input](#policy-input). | +| `opa_input_*` | any other environment variable starting with `opa_input_` will be included in the OPA input. The values are accessible using `input.data[env_var_name]` or `input.data.env_var_name`, see also [Policy Input](#policy-input). | +| `opa_error_content_type` | The content type to set when the policy evaluation fails. Default: `text/plain` | Unsupported options from the [Classic Watchdog](https://github.com/openfaas/classic-watchdog): diff --git a/auth/config.go b/auth/config.go new file mode 100644 index 00000000..3141335a --- /dev/null +++ b/auth/config.go @@ -0,0 +1,14 @@ +package auth + +const ( + OPAPolicyEnv = "opa_policy" + OPAQueryEnv = "opa_query" + OPADebugEnv = "opa_debug" + OPASkipPathsEnv = "opa_skip_paths" + OPAIncludeBodyEnv = "opa_include_body" + OPAIncludeRawBodyEnv = "opa_include_raw_body" + OPAIncludeHeadersEnv = "opa_include_headers" + OPAErrorContentTypeEnv = "opa_error_content_type" + OPAInputSecretsEnv = "opa_input_secrets" + OPAInputPrefixEnv = "opa_input_" +) diff --git a/auth/local.go b/auth/local.go index dd055029..a4257cb8 100644 --- a/auth/local.go +++ b/auth/local.go @@ -53,8 +53,8 @@ type OPAConfig struct { } func OPAConfigFromEnv() (cfg OPAConfig) { - cfg.Debug = truthy("OPA_DEBUG", "false") - cfg.Query = os.Getenv("OPA_QUERY") + cfg.Debug = truthy(OPADebugEnv, "false") + cfg.Query = os.Getenv(OPAQueryEnv) return cfg } @@ -66,7 +66,7 @@ func OPAConfigFromEnv() (cfg OPAConfig) { // - constant_compare // // Additionally, it modifies the logging so that it will use the default log writer -// when the OPA_DEBUG environment variable is set to true. +// when the opa_debug environment variable is set to true. func NewLocalAuthorizer(policy Policy, cfg OPAConfig) (_ Authorizer, err error) { auth := opa{ cfg: cfg, @@ -131,36 +131,38 @@ type opa struct { // Allowed implements the Authorizer interface and validates the given input against // the configured OPA policy. -func (a opa) Allowed(ctx context.Context, input Input) (_ bool, err error) { +func (a opa) Allowed(ctx context.Context, input Input) (_ AuthResult, err error) { + resp := AuthResult{} + result, err := a.query.Eval(ctx, rego.EvalInput(input)) if err != nil { - return false, fmt.Errorf("can not evaluate OPA query: %w", err) + return resp, fmt.Errorf("can not evaluate OPA query: %w", err) } - // this block allows us to inspect the result - // it will also be useful if we want to support - // complex result sets. See the TODO below. if a.cfg.Debug { data, _ := json.Marshal(result) log.Printf("OPA query result: %s", string(data)) + } - if len(result) == 1 && len(result[0].Bindings) == 0 { - if exprs := result[0].Expressions; len(exprs) == 1 { - value := exprs[0].Value - log.Printf("OPA result value: %v\n OPA Result type: %T", value, value) - } - } + allowed, ok := checkSimpleResponse(result) + // this is a simple response that only has a single boolean result + if ok { + resp.Allow = allowed + return resp, nil + } + // Parse the structured result set + expr := findExpression(result, a.cfg.Query) + if expr == nil { + return resp, fmt.Errorf("can not find query in policy result: %q", a.cfg.Query) } - // this only allows policies that have a single boolean result - // policies that return complex objects will be treated as false - // - // TODO: allow policies to return complex objects which are then - // integrated into the request as X_AUTH headers. - // this will allow, for example, policies to pass information - // parsed from token. - return result.Allowed(), nil + resp, err = parseExpression(expr) + if a.cfg.Debug { + log.Printf("OPA query result: %+v", resp) + } + + return resp, err } // truthy converts the given env variable to a boolean. @@ -175,3 +177,38 @@ func truthy(name string, fallback string) bool { } return false } + +// checkSimpleResponse is a duplicate of the ResultSet.Allowed() method, but +// it also returns a second boolean indicating if the result is a simple boolean +// or a more complex expressions. +func checkSimpleResponse(rs rego.ResultSet) (bool, bool) { + if len(rs) == 1 && len(rs[0].Bindings) == 0 { + if exprs := rs[0].Expressions; len(exprs) == 1 { + if b, ok := exprs[0].Value.(bool); ok { + return b, true + } + } + } + return false, false +} + +func findExpression(result rego.ResultSet, query string) *rego.ExpressionValue { + for _, r := range result { + for _, e := range r.Expressions { + if e.Text == query { + return e + } + } + } + return nil +} + +func parseExpression(exp *rego.ExpressionValue) (AuthResult, error) { + var result AuthResult + data, err := json.Marshal(exp.Value) + if err != nil { + return result, err + } + err = json.Unmarshal(data, &result) + return result, err +} diff --git a/auth/local_test.go b/auth/local_test.go index eec51d83..b716898b 100644 --- a/auth/local_test.go +++ b/auth/local_test.go @@ -24,11 +24,12 @@ func TestOPAAuthorizer(t *testing.T) { const jwtSecretKey = "secret" cases := []struct { - name string - cfg OPAConfig - policy string - input Input - expected bool + name string + cfg OPAConfig + policy string + input Input + allow bool + headers map[string]string }{ { name: "correctly loads and permits request when the default is true", @@ -41,7 +42,7 @@ func TestOPAAuthorizer(t *testing.T) { Method: http.MethodGet, Path: "/api/endpoint", }, - expected: true, + allow: true, }, { name: "allow GET request on the public path", @@ -54,7 +55,7 @@ func TestOPAAuthorizer(t *testing.T) { Method: http.MethodGet, Path: "/api/public", }, - expected: true, + allow: true, }, { name: "block unauthenticated POST request on the public path", @@ -67,7 +68,7 @@ func TestOPAAuthorizer(t *testing.T) { Method: http.MethodPost, Path: "/api/public", }, - expected: false, + allow: false, }, { name: "allow POST request on the public path authenticated as admin", @@ -81,7 +82,7 @@ func TestOPAAuthorizer(t *testing.T) { Path: "/api/public", Authorization: "admin", }, - expected: true, + allow: true, }, { name: "policy can inspect the rawBody value correctly and returns true", @@ -95,7 +96,7 @@ func TestOPAAuthorizer(t *testing.T) { Path: "/api/public", RawBody: "permitted", }, - expected: true, + allow: true, }, { name: "policy can inspect the json body value correctly and returns true", @@ -109,7 +110,7 @@ func TestOPAAuthorizer(t *testing.T) { Path: "/api/public", Body: json.RawMessage(`{"override": "permitted"}`), }, - expected: true, + allow: true, }, { name: "policy can implement basic auth", @@ -123,7 +124,7 @@ func TestOPAAuthorizer(t *testing.T) { Path: "/api/private", Authorization: "Basic " + base64.StdEncoding.EncodeToString([]byte("bob:secretvalue")), }, - expected: true, + allow: true, }, { name: "can load and merge multiple policies, can evaluate basic auth", @@ -137,7 +138,7 @@ func TestOPAAuthorizer(t *testing.T) { Path: "/api/private", Authorization: "Basic " + base64.StdEncoding.EncodeToString([]byte("bob:secretvalue")), }, - expected: true, + allow: true, }, { name: "can load and merge multiple policies, can evaluate token auth", @@ -151,7 +152,7 @@ func TestOPAAuthorizer(t *testing.T) { Path: "/api/private", Authorization: "admin", }, - expected: true, + allow: true, }, { name: "can apply HMAC auth policy", @@ -173,7 +174,7 @@ func TestOPAAuthorizer(t *testing.T) { "secretKey": "secretvalue", }, }, - expected: true, + allow: true, }, { name: "can apply custom JWT auth policy", @@ -194,7 +195,10 @@ func TestOPAAuthorizer(t *testing.T) { "email_field": "email", }, }, - expected: true, + allow: true, + headers: map[string]string{ + "X-User-Email": "alice@test.example.com", + }, }, { name: "JWT policy rejects non-matching domains", @@ -215,7 +219,7 @@ func TestOPAAuthorizer(t *testing.T) { "email_field": "email", }, }, - expected: false, + allow: false, }, // OICD auth can be seen in the testdata/oidc.rego file // but is not possible to include the unit tests because @@ -231,7 +235,13 @@ func TestOPAAuthorizer(t *testing.T) { result, err := opa.Allowed(ctx, tc.input) require.NoError(t, err) - require.Equal(t, tc.expected, result) + require.Equal(t, tc.allow, result.Allow) + require.Len(t, result.Headers, len(tc.headers)) + if tc.headers != nil { + for k, v := range tc.headers { + require.Equal(t, v, result.Headers[k]) + } + } }) } } diff --git a/auth/middleware.go b/auth/middleware.go index 544eb876..63fe277e 100644 --- a/auth/middleware.go +++ b/auth/middleware.go @@ -15,9 +15,19 @@ import ( const secretDir = "/var/openfaas/secrets" +// AuthResult is the parsed response from the OPA policy. +type AuthResult struct { + // Allow indicates if the request is allowed. + Allow bool `json:"allow"` + Status int `json:"status,omitempty"` + // Headers are added to the request and passed to the handler, when the request is allowed. + // If the request is not allowed, the headers are ignored. + Headers map[string]string `json:"headers,omitempty"` +} + // Authorizer is the generic request authorizer interface. type Authorizer interface { - Allowed(context.Context, Input) (bool, error) + Allowed(context.Context, Input) (AuthResult, error) } type InputConfig struct { @@ -48,17 +58,17 @@ type Input struct { } func InputConfigFromEnv() (cfg InputConfig, err error) { - cfg.ErrorContentType = os.Getenv("OPA_CONTENT_TYPE") + cfg.ErrorContentType = os.Getenv(OPAErrorContentTypeEnv) if cfg.ErrorContentType == "" { cfg.ErrorContentType = "text/plain" } - cfg.IncludeJSONBody = truthy("OPA_INCLUDE_BODY", "false") - cfg.IncludeRawBody = truthy("OPA_INCLUDE_RAW_BODY", "false") - cfg.IncludeHeaders = truthy("OPA_INCLUDE_HEADERS", "false") + cfg.IncludeJSONBody = truthy(OPAIncludeBodyEnv, "false") + cfg.IncludeRawBody = truthy(OPAIncludeRawBodyEnv, "false") + cfg.IncludeHeaders = truthy(OPAIncludeHeadersEnv, "false") cfg.SkipPaths = make(map[string]struct{}) - paths := os.Getenv("OPA_SKIP_PATHS") + paths := os.Getenv(OPASkipPathsEnv) for _, path := range strings.Split(paths, ",") { cfg.SkipPaths[path] = struct{}{} } @@ -115,18 +125,26 @@ func New(impl Authorizer, cfg InputConfig) Middleware { input.Body = json.RawMessage(body) } - allowed, err := impl.Allowed(r.Context(), input) + result, err := impl.Allowed(r.Context(), input) if err != nil { errorWriter(w, "can not process authentication", http.StatusInternalServerError) return } - if allowed { + if result.Allow { + for k, v := range result.Headers { + r.Header.Set(k, v) + } next.ServeHTTP(w, r) return } - errorWriter(w, "unauthorized", http.StatusUnauthorized) + status := http.StatusUnauthorized + if result.Status != 0 { + status = result.Status + } + + errorWriter(w, "unauthorized", status) return }) } @@ -169,17 +187,17 @@ func NewAuthorizer(path string) (Authorizer, error) { func loadAdditionalData(options map[string]string) (map[string]string, error) { out := map[string]string{} for name, value := range options { - if !strings.HasPrefix(name, "OPA_INPUT") { + if !strings.HasPrefix(name, OPAInputPrefixEnv) { continue } - if name == "OPA_INPUT_SECRETS" { + if name == OPAInputSecretsEnv { continue } - out[strings.TrimPrefix(name, "OPA_INPUT")] = value + out[strings.TrimPrefix(name, OPAInputPrefixEnv)] = value } - names := options["OPA_INPUT_SECRETS"] + names := options[OPAInputSecretsEnv] if names == "" { return out, nil } @@ -200,7 +218,7 @@ func loadAdditionalData(options map[string]string) (map[string]string, error) { func authEnviron() map[string]string { out := map[string]string{} for _, env := range os.Environ() { - if strings.HasPrefix(env, "OPA_INPUT") { + if strings.HasPrefix(env, OPAInputPrefixEnv) { name, value, ok := cut(env, "=") if !ok { continue diff --git a/auth/testdata/jwt.rego b/auth/testdata/jwt.rego index 217850bb..ea5835f9 100644 --- a/auth/testdata/jwt.rego +++ b/auth/testdata/jwt.rego @@ -4,7 +4,7 @@ # flows. package api.jwt -default allow = false +default allow = {"allow": false} # Below we extract various data from the input @@ -30,7 +30,7 @@ now = value { # for an issuer are checked against the available fields in the # token. If more than one configuration matches, one is chosen # arbitrarily. -allow { +allow = response { print("attempt bearer token auth") token := trim_prefix(input.authorization, "Bearer ") @@ -67,4 +67,9 @@ allow { # and it must match the allowed domains glob # see https://www.openpolicyagent.org/docs/latest/policy-reference/#glob glob.match(allowed_domains, null, email) + + response := { + "allow": true, + "headers": {"X-User-Email": email}, + } } diff --git a/main.go b/main.go index db1b67ea..031c36aa 100644 --- a/main.go +++ b/main.go @@ -75,7 +75,7 @@ func main() { baseFunctionHandler := buildRequestHandler(watchdogConfig, watchdogConfig.PrefixLogs) requestHandler := baseFunctionHandler // add middlewares here - policyPath := os.Getenv("OPA_POLICY") + policyPath := os.Getenv(auth.OPAPolicyEnv) if policyPath != "" { log.Printf("loading OPA policy(s): %q", policyPath) // TODO: support remote authorizer if it starts with http:// or https://